mirror of
https://github.com/ichmagmaus111/ghostgram.git
synced 2026-06-05 01:28:27 +02:00
Update Ghostgram features
This commit is contained in:
@@ -1,15 +1,25 @@
|
||||
load("@build_bazel_rules_swift//swift:swift.bzl", "swift_library")
|
||||
|
||||
sgdeps = [
|
||||
"//submodules/BuildConfig:BuildConfig",
|
||||
"//Swiftgram/SGSimpleSettings:SGSimpleSettings",
|
||||
"//Swiftgram/SGStrings:SGStrings"
|
||||
]
|
||||
|
||||
sgsrc = [
|
||||
"//Swiftgram/SGRecentSessionApiId:SGRecentSessionApiId",
|
||||
]
|
||||
|
||||
swift_library(
|
||||
name = "SettingsUI",
|
||||
module_name = "SettingsUI",
|
||||
srcs = glob([
|
||||
srcs = sgsrc + glob([
|
||||
"Sources/**/*.swift",
|
||||
]),
|
||||
copts = [
|
||||
"-warnings-as-errors",
|
||||
],
|
||||
deps = [
|
||||
deps = sgdeps + [
|
||||
"//submodules/SSignalKit/SwiftSignalKit:SwiftSignalKit",
|
||||
"//submodules/AsyncDisplayKit:AsyncDisplayKit",
|
||||
"//submodules/Display:Display",
|
||||
@@ -135,6 +145,11 @@ swift_library(
|
||||
"//submodules/TelegramUI/Components/AlertComponent",
|
||||
"//submodules/TelegramUI/Components/AlertComponent/AlertInputFieldComponent",
|
||||
"//submodules/TelegramUI/Components/EdgeEffect",
|
||||
"//submodules/TelegramUI/Components/AvatarEditorScreen",
|
||||
"//submodules/TelegramUI/Components/Settings/PeerSelectionScreen",
|
||||
"//submodules/TelegramUI/Components/ListSectionComponent",
|
||||
"//submodules/TelegramUI/Components/ListActionItemComponent",
|
||||
"//submodules/Utils/DeviceModel",
|
||||
],
|
||||
visibility = [
|
||||
"//visibility:public",
|
||||
|
||||
@@ -1,3 +1,7 @@
|
||||
// MARK: Swiftgram
|
||||
import SGSimpleSettings
|
||||
import SGStrings
|
||||
|
||||
import Foundation
|
||||
import UIKit
|
||||
import Display
|
||||
@@ -13,6 +17,7 @@ import UrlEscaping
|
||||
import ShareController
|
||||
|
||||
private final class ProxySettingsControllerArguments {
|
||||
let toggleLocalDNS: (Bool) -> Void
|
||||
let toggleEnabled: (Bool) -> Void
|
||||
let addNewServer: () -> Void
|
||||
let activateServer: (ProxyServerSettings) -> Void
|
||||
@@ -22,7 +27,8 @@ private final class ProxySettingsControllerArguments {
|
||||
let toggleUseForCalls: (Bool) -> Void
|
||||
let shareProxyList: () -> Void
|
||||
|
||||
init(toggleEnabled: @escaping (Bool) -> Void, addNewServer: @escaping () -> Void, activateServer: @escaping (ProxyServerSettings) -> Void, editServer: @escaping (ProxyServerSettings) -> Void, removeServer: @escaping (ProxyServerSettings) -> Void, setServerWithRevealedOptions: @escaping (ProxyServerSettings?, ProxyServerSettings?) -> Void, toggleUseForCalls: @escaping (Bool) -> Void, shareProxyList: @escaping () -> Void) {
|
||||
init(toggleLocalDNS: @escaping (Bool) -> Void, toggleEnabled: @escaping (Bool) -> Void, addNewServer: @escaping () -> Void, activateServer: @escaping (ProxyServerSettings) -> Void, editServer: @escaping (ProxyServerSettings) -> Void, removeServer: @escaping (ProxyServerSettings) -> Void, setServerWithRevealedOptions: @escaping (ProxyServerSettings?, ProxyServerSettings?) -> Void, toggleUseForCalls: @escaping (Bool) -> Void, shareProxyList: @escaping () -> Void) {
|
||||
self.toggleLocalDNS = toggleLocalDNS
|
||||
self.toggleEnabled = toggleEnabled
|
||||
self.addNewServer = addNewServer
|
||||
self.activateServer = activateServer
|
||||
@@ -58,8 +64,25 @@ private enum ProxySettingsControllerEntryId: Equatable, Hashable {
|
||||
case server(String, Int32, ProxyServerConnection)
|
||||
}
|
||||
|
||||
public enum ProxySettingsEntryTag: ItemListItemTag, Equatable {
|
||||
case edit
|
||||
case useProxy
|
||||
case shareList
|
||||
case useForCalls
|
||||
|
||||
public func isEqual(to other: ItemListItemTag) -> Bool {
|
||||
if let other = other as? ProxySettingsEntryTag, self == other {
|
||||
return true
|
||||
} else {
|
||||
return false
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private enum ProxySettingsControllerEntry: ItemListNodeEntry {
|
||||
case enabled(PresentationTheme, String, Bool, Bool)
|
||||
case localDNSToggle(PresentationTheme, String, Bool)
|
||||
case localDNSNotice(PresentationTheme, String)
|
||||
case serversHeader(PresentationTheme, String)
|
||||
case addServer(PresentationTheme, String, Bool)
|
||||
case server(Int, PresentationTheme, PresentationStrings, ProxyServerSettings, Bool, DisplayProxyServerStatus, ProxySettingsServerItemEditing, Bool)
|
||||
@@ -69,6 +92,8 @@ private enum ProxySettingsControllerEntry: ItemListNodeEntry {
|
||||
|
||||
var section: ItemListSectionId {
|
||||
switch self {
|
||||
case .localDNSToggle, .localDNSNotice:
|
||||
return ProxySettingsControllerSection.enabled.rawValue
|
||||
case .enabled:
|
||||
return ProxySettingsControllerSection.enabled.rawValue
|
||||
case .serversHeader, .addServer, .server:
|
||||
@@ -83,6 +108,10 @@ private enum ProxySettingsControllerEntry: ItemListNodeEntry {
|
||||
var stableId: ProxySettingsControllerEntryId {
|
||||
switch self {
|
||||
case .enabled:
|
||||
return .index(-2)
|
||||
case .localDNSToggle:
|
||||
return .index(-1)
|
||||
case .localDNSNotice:
|
||||
return .index(0)
|
||||
case .serversHeader:
|
||||
return .index(1)
|
||||
@@ -107,6 +136,18 @@ private enum ProxySettingsControllerEntry: ItemListNodeEntry {
|
||||
} else {
|
||||
return false
|
||||
}
|
||||
case let .localDNSToggle(lhsTheme, lhsText, lhsValue):
|
||||
if case let .localDNSToggle(rhsTheme, rhsText, rhsValue) = rhs, lhsTheme === rhsTheme, lhsText == rhsText, lhsValue == rhsValue {
|
||||
return true
|
||||
} else {
|
||||
return false
|
||||
}
|
||||
case let .localDNSNotice(lhsTheme, lhsText):
|
||||
if case let .localDNSNotice(rhsTheme, rhsText) = rhs, lhsTheme === rhsTheme, lhsText == rhsText {
|
||||
return true
|
||||
} else {
|
||||
return false
|
||||
}
|
||||
case let .serversHeader(lhsTheme, lhsText):
|
||||
if case let .serversHeader(rhsTheme, rhsText) = rhs, lhsTheme === rhsTheme, lhsText == rhsText {
|
||||
return true
|
||||
@@ -155,23 +196,37 @@ private enum ProxySettingsControllerEntry: ItemListNodeEntry {
|
||||
default:
|
||||
return true
|
||||
}
|
||||
case .localDNSToggle:
|
||||
switch rhs {
|
||||
case .enabled, .localDNSToggle:
|
||||
return false
|
||||
default:
|
||||
return true
|
||||
}
|
||||
case .localDNSNotice:
|
||||
switch rhs {
|
||||
case .enabled, .localDNSToggle, .localDNSNotice:
|
||||
return false
|
||||
default:
|
||||
return true
|
||||
}
|
||||
case .serversHeader:
|
||||
switch rhs {
|
||||
case .enabled, .serversHeader:
|
||||
case .enabled, .localDNSToggle, .localDNSNotice, .serversHeader:
|
||||
return false
|
||||
default:
|
||||
return true
|
||||
}
|
||||
case .addServer:
|
||||
switch rhs {
|
||||
case .enabled, .serversHeader, .addServer:
|
||||
case .enabled, .localDNSToggle, .localDNSNotice, .serversHeader, .addServer:
|
||||
return false
|
||||
default:
|
||||
return true
|
||||
}
|
||||
case let .server(lhsIndex, _, _, _, _, _, _, _):
|
||||
switch rhs {
|
||||
case .enabled, .serversHeader, .addServer:
|
||||
case .enabled, .localDNSToggle, .localDNSNotice, .serversHeader, .addServer:
|
||||
return false
|
||||
case let .server(rhsIndex, _, _, _, _, _, _, _):
|
||||
return lhsIndex < rhsIndex
|
||||
@@ -180,14 +235,14 @@ private enum ProxySettingsControllerEntry: ItemListNodeEntry {
|
||||
}
|
||||
case .shareProxyList:
|
||||
switch rhs {
|
||||
case .enabled, .serversHeader, .addServer, .server, .shareProxyList:
|
||||
case .enabled, .localDNSToggle, .localDNSNotice, .serversHeader, .addServer, .server, .shareProxyList:
|
||||
return false
|
||||
default:
|
||||
return true
|
||||
}
|
||||
case .useForCalls:
|
||||
switch rhs {
|
||||
case .enabled, .serversHeader, .addServer, .server, .shareProxyList, .useForCalls:
|
||||
case .enabled, .localDNSToggle, .localDNSNotice, .serversHeader, .addServer, .server, .shareProxyList, .useForCalls:
|
||||
return false
|
||||
default:
|
||||
return true
|
||||
@@ -207,7 +262,13 @@ private enum ProxySettingsControllerEntry: ItemListNodeEntry {
|
||||
} else {
|
||||
arguments.toggleEnabled(value)
|
||||
}
|
||||
}, tag: ProxySettingsEntryTag.useProxy)
|
||||
case let .localDNSToggle(_, text, value):
|
||||
return ItemListSwitchItem(presentationData: presentationData, title: text, value: value, enabled: true, sectionId: self.section, style: .blocks, updated: { value in
|
||||
arguments.toggleLocalDNS(value)
|
||||
})
|
||||
case let .localDNSNotice(_, text):
|
||||
return ItemListTextItem(presentationData: presentationData, text: .markdown(text), sectionId: self.section)
|
||||
case let .serversHeader(_, text):
|
||||
return ItemListSectionHeaderItem(presentationData: presentationData, text: text, sectionId: self.section)
|
||||
case let .addServer(_, text, _):
|
||||
@@ -225,13 +286,13 @@ private enum ProxySettingsControllerEntry: ItemListNodeEntry {
|
||||
arguments.removeServer(settings)
|
||||
})
|
||||
case let .shareProxyList(_, text):
|
||||
return ProxySettingsActionItem(presentationData: presentationData, systemStyle: .glass, title: text, sectionId: self.section, editing: false, action: {
|
||||
return ProxySettingsActionItem(presentationData: presentationData, systemStyle: .glass, title: text, sectionId: self.section, editing: false, tag: ProxySettingsEntryTag.shareList, action: {
|
||||
arguments.shareProxyList()
|
||||
})
|
||||
case let .useForCalls(_, text, value):
|
||||
return ItemListSwitchItem(presentationData: presentationData, systemStyle: .glass, title: text, value: value, enableInteractiveChanges: true, enabled: true, sectionId: self.section, style: .blocks, updated: { value in
|
||||
arguments.toggleUseForCalls(value)
|
||||
})
|
||||
}, tag: ProxySettingsEntryTag.useForCalls)
|
||||
case let .useForCallsInfo(_, text):
|
||||
return ItemListTextItem(presentationData: presentationData, text: .plain(text), sectionId: self.section)
|
||||
}
|
||||
@@ -242,6 +303,9 @@ private func proxySettingsControllerEntries(theme: PresentationTheme, strings: P
|
||||
var entries: [ProxySettingsControllerEntry] = []
|
||||
|
||||
entries.append(.enabled(theme, strings.ChatSettings_ConnectionType_UseProxy, proxySettings.enabled, proxySettings.servers.isEmpty))
|
||||
// MARK: Swiftgram
|
||||
entries.append(.localDNSToggle(theme, i18n("ProxySettings.UseSystemDNS", strings.baseLanguageCode), SGSimpleSettings.shared.localDNSForProxyHost))
|
||||
entries.append(.localDNSNotice(theme, i18n("ProxySettings.UseSystemDNS.Notice", strings.baseLanguageCode)))
|
||||
entries.append(.serversHeader(theme, strings.SocksProxySetup_SavedProxies))
|
||||
entries.append(.addServer(theme, strings.SocksProxySetup_AddProxy, state.editing))
|
||||
var index = 0
|
||||
@@ -308,13 +372,14 @@ public enum ProxySettingsControllerMode {
|
||||
case modal
|
||||
}
|
||||
|
||||
public func proxySettingsController(context: AccountContext, mode: ProxySettingsControllerMode = .default) -> ViewController {
|
||||
public func proxySettingsController(context: AccountContext, mode: ProxySettingsControllerMode = .default, focusOnItemTag: ProxySettingsEntryTag? = nil) -> ViewController {
|
||||
let presentationData = context.sharedContext.currentPresentationData.with { $0 }
|
||||
return proxySettingsController(accountManager: context.sharedContext.accountManager, sharedContext: context.sharedContext, context: context, postbox: context.account.postbox, network: context.account.network, mode: mode, presentationData: presentationData, updatedPresentationData: context.sharedContext.presentationData)
|
||||
return proxySettingsController(accountManager: context.sharedContext.accountManager, sharedContext: context.sharedContext, context: context, postbox: context.account.postbox, network: context.account.network, mode: mode, presentationData: presentationData, updatedPresentationData: context.sharedContext.presentationData, focusOnItemTag: focusOnItemTag)
|
||||
}
|
||||
|
||||
public func proxySettingsController(accountManager: AccountManager<TelegramAccountManagerTypes>, sharedContext: SharedAccountContext, context: AccountContext? = nil, postbox: Postbox, network: Network, mode: ProxySettingsControllerMode, presentationData: PresentationData, updatedPresentationData: Signal<PresentationData, NoError>) -> ViewController {
|
||||
public func proxySettingsController(accountManager: AccountManager<TelegramAccountManagerTypes>, sharedContext: SharedAccountContext, context: AccountContext? = nil, postbox: Postbox, network: Network, mode: ProxySettingsControllerMode, presentationData: PresentationData, updatedPresentationData: Signal<PresentationData, NoError>, focusOnItemTag: ProxySettingsEntryTag? = nil) -> ViewController {
|
||||
var pushControllerImpl: ((ViewController) -> Void)?
|
||||
var presentControllerImpl: ((ViewController, ViewControllerPresentationArguments?) -> Void)?
|
||||
var dismissImpl: (() -> Void)?
|
||||
let stateValue = Atomic(value: ProxySettingsControllerState())
|
||||
let statePromise = ValuePromise<ProxySettingsControllerState>(stateValue.with { $0 })
|
||||
@@ -332,9 +397,35 @@ public func proxySettingsController(accountManager: AccountManager<TelegramAccou
|
||||
}
|
||||
}
|
||||
|
||||
if focusOnItemTag == ProxySettingsEntryTag.edit {
|
||||
updateState { state in
|
||||
var state = state
|
||||
state.editing = true
|
||||
return state
|
||||
}
|
||||
}
|
||||
|
||||
var shareProxyListImpl: (() -> Void)?
|
||||
|
||||
let arguments = ProxySettingsControllerArguments(toggleEnabled: { value in
|
||||
let arguments = ProxySettingsControllerArguments(toggleLocalDNS: { value in
|
||||
SGSimpleSettings.shared.localDNSForProxyHost = value
|
||||
guard let context = context else {
|
||||
return
|
||||
}
|
||||
let presentationData = context.sharedContext.currentPresentationData.with { $0 }
|
||||
let actionSheet = ActionSheetController(presentationData: presentationData)
|
||||
actionSheet.setItemGroups([ActionSheetItemGroup(items: [
|
||||
ActionSheetTextItem(title: i18n("Common.RestartRequired", presentationData.strings.baseLanguageCode)),
|
||||
ActionSheetButtonItem(title: i18n("Common.RestartNow", presentationData.strings.baseLanguageCode), color: .destructive, font: .default, action: {
|
||||
exit(0)
|
||||
})
|
||||
]), ActionSheetItemGroup(items: [
|
||||
ActionSheetButtonItem(title: presentationData.strings.Common_Cancel, color: .accent, font: .bold, action: { [weak actionSheet] in
|
||||
actionSheet?.dismissAnimated()
|
||||
})
|
||||
])])
|
||||
presentControllerImpl?(actionSheet, ViewControllerPresentationArguments(presentationAnimation: .modalSheet))
|
||||
}, toggleEnabled: { value in
|
||||
let _ = updateProxySettingsInteractively(accountManager: accountManager, { current in
|
||||
var current = current
|
||||
current.enabled = value
|
||||
@@ -431,7 +522,7 @@ public func proxySettingsController(accountManager: AccountManager<TelegramAccou
|
||||
}
|
||||
|
||||
let controllerState = ItemListControllerState(presentationData: ItemListPresentationData(presentationData), title: .text(presentationData.strings.SocksProxySetup_Title), leftNavigationButton: leftNavigationButton, rightNavigationButton: rightNavigationButton, backNavigationButton: ItemListBackButton(title: presentationData.strings.Common_Back))
|
||||
let listState = ItemListNodeState(presentationData: ItemListPresentationData(presentationData), entries: proxySettingsControllerEntries(theme: presentationData.theme, strings: presentationData.strings, state: state, proxySettings: proxySettings, statuses: statuses, connectionStatus: connectionStatus), style: .blocks)
|
||||
let listState = ItemListNodeState(presentationData: ItemListPresentationData(presentationData), entries: proxySettingsControllerEntries(theme: presentationData.theme, strings: presentationData.strings, state: state, proxySettings: proxySettings, statuses: statuses, connectionStatus: connectionStatus), style: .blocks, ensureVisibleItemTag: focusOnItemTag)
|
||||
|
||||
return (controllerState, (listState, arguments))
|
||||
}
|
||||
@@ -530,5 +621,23 @@ public func proxySettingsController(accountManager: AccountManager<TelegramAccou
|
||||
})
|
||||
}
|
||||
|
||||
if let focusOnItemTag {
|
||||
var didFocusOnItem = false
|
||||
controller.afterTransactionCompleted = { [weak controller] in
|
||||
if !didFocusOnItem, let controller {
|
||||
controller.forEachItemNode { itemNode in
|
||||
if let itemNode = itemNode as? ItemListItemNode, let tag = itemNode.tag, tag.isEqual(to: focusOnItemTag) {
|
||||
didFocusOnItem = true
|
||||
itemNode.displayHighlight()
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
// MARK: Swiftgram
|
||||
presentControllerImpl = { [weak controller] c, a in
|
||||
controller?.present(c, in: .window(.root), with: a)
|
||||
}
|
||||
|
||||
return controller
|
||||
}
|
||||
|
||||
@@ -19,15 +19,17 @@ final class ProxySettingsActionItem: ListViewItem, ItemListItem {
|
||||
let icon: ProxySettingsActionIcon
|
||||
let editing: Bool
|
||||
let sectionId: ItemListSectionId
|
||||
let tag: ItemListItemTag?
|
||||
let action: () -> Void
|
||||
|
||||
init(presentationData: ItemListPresentationData, systemStyle: ItemListSystemStyle = .legacy, title: String, icon: ProxySettingsActionIcon = .none, sectionId: ItemListSectionId, editing: Bool, action: @escaping () -> Void) {
|
||||
init(presentationData: ItemListPresentationData, systemStyle: ItemListSystemStyle = .legacy, title: String, icon: ProxySettingsActionIcon = .none, sectionId: ItemListSectionId, editing: Bool, tag: ItemListItemTag? = nil, action: @escaping () -> Void) {
|
||||
self.presentationData = presentationData
|
||||
self.systemStyle = systemStyle
|
||||
self.title = title
|
||||
self.icon = icon
|
||||
self.editing = editing
|
||||
self.sectionId = sectionId
|
||||
self.tag = tag
|
||||
self.action = action
|
||||
}
|
||||
|
||||
@@ -77,7 +79,7 @@ final class ProxySettingsActionItem: ListViewItem, ItemListItem {
|
||||
}
|
||||
}
|
||||
|
||||
private final class ProxySettingsActionItemNode: ListViewItemNode {
|
||||
private final class ProxySettingsActionItemNode: ListViewItemNode, ItemListItemNode {
|
||||
private let backgroundNode: ASDisplayNode
|
||||
private let topStripeNode: ASDisplayNode
|
||||
private let bottomStripeNode: ASDisplayNode
|
||||
@@ -89,6 +91,10 @@ private final class ProxySettingsActionItemNode: ListViewItemNode {
|
||||
|
||||
private var item: ProxySettingsActionItem?
|
||||
|
||||
var tag: ItemListItemTag? {
|
||||
return self.item?.tag
|
||||
}
|
||||
|
||||
init() {
|
||||
self.backgroundNode = ASDisplayNode()
|
||||
self.backgroundNode.isLayerBacked = true
|
||||
|
||||
@@ -8,8 +8,10 @@ import Postbox
|
||||
import TelegramPresentationData
|
||||
import ItemListUI
|
||||
import AccountContext
|
||||
import PresentationDataUtils
|
||||
import ComponentFlow
|
||||
import SliderComponent
|
||||
import AlertUI
|
||||
|
||||
private let minDeletedMessageTransparencyPercent: Int32 = Int32(AntiDeleteManager.minDeletedMessageTransparency * 100.0)
|
||||
private let maxDeletedMessageTransparencyPercent: Int32 = Int32(AntiDeleteManager.maxDeletedMessageTransparency * 100.0)
|
||||
@@ -27,6 +29,7 @@ private enum DeletedMessagesSection: Int32 {
|
||||
private enum DeletedMessagesEntry: ItemListNodeEntry {
|
||||
case enableToggle(PresentationTheme, String, Bool)
|
||||
case archiveMediaToggle(PresentationTheme, String, Bool)
|
||||
case history(PresentationTheme, String, String)
|
||||
case transparencySlider(PresentationTheme, Int32, Bool)
|
||||
case settingsInfo(PresentationTheme, String)
|
||||
|
||||
@@ -40,10 +43,12 @@ private enum DeletedMessagesEntry: ItemListNodeEntry {
|
||||
return 0
|
||||
case .archiveMediaToggle:
|
||||
return 1
|
||||
case .transparencySlider:
|
||||
case .history:
|
||||
return 2
|
||||
case .settingsInfo:
|
||||
case .transparencySlider:
|
||||
return 3
|
||||
case .settingsInfo:
|
||||
return 4
|
||||
}
|
||||
}
|
||||
|
||||
@@ -61,6 +66,12 @@ private enum DeletedMessagesEntry: ItemListNodeEntry {
|
||||
return true
|
||||
}
|
||||
return false
|
||||
case let .history(lhsTheme, lhsText, lhsValue):
|
||||
if case let .history(rhsTheme, rhsText, rhsValue) = rhs,
|
||||
lhsTheme === rhsTheme, lhsText == rhsText, lhsValue == rhsValue {
|
||||
return true
|
||||
}
|
||||
return false
|
||||
case let .transparencySlider(lhsTheme, lhsValue, lhsIsEnabled):
|
||||
if case let .transparencySlider(rhsTheme, rhsValue, rhsIsEnabled) = rhs,
|
||||
lhsTheme === rhsTheme, lhsValue == rhsValue, lhsIsEnabled == rhsIsEnabled {
|
||||
@@ -104,6 +115,17 @@ private enum DeletedMessagesEntry: ItemListNodeEntry {
|
||||
arguments.toggleArchiveMedia(value)
|
||||
}
|
||||
)
|
||||
case let .history(_, text, value):
|
||||
return ItemListDisclosureItem(
|
||||
presentationData: presentationData,
|
||||
title: text,
|
||||
label: value,
|
||||
sectionId: self.section,
|
||||
style: .blocks,
|
||||
action: {
|
||||
arguments.openHistory()
|
||||
}
|
||||
)
|
||||
case let .transparencySlider(theme, value, isEnabled):
|
||||
return DeletedMessagesTransparencySliderItem(
|
||||
theme: theme,
|
||||
@@ -125,15 +147,18 @@ private enum DeletedMessagesEntry: ItemListNodeEntry {
|
||||
private final class DeletedMessagesControllerArguments {
|
||||
let toggleEnabled: (Bool) -> Void
|
||||
let toggleArchiveMedia: (Bool) -> Void
|
||||
let openHistory: () -> Void
|
||||
let updateTransparency: (Int32) -> Void
|
||||
|
||||
init(
|
||||
toggleEnabled: @escaping (Bool) -> Void,
|
||||
toggleArchiveMedia: @escaping (Bool) -> Void,
|
||||
openHistory: @escaping () -> Void,
|
||||
updateTransparency: @escaping (Int32) -> Void
|
||||
) {
|
||||
self.toggleEnabled = toggleEnabled
|
||||
self.toggleArchiveMedia = toggleArchiveMedia
|
||||
self.openHistory = openHistory
|
||||
self.updateTransparency = updateTransparency
|
||||
}
|
||||
}
|
||||
@@ -143,11 +168,13 @@ private final class DeletedMessagesControllerArguments {
|
||||
private struct DeletedMessagesControllerState: Equatable {
|
||||
var isEnabled: Bool
|
||||
var archiveMedia: Bool
|
||||
var archivedCount: Int
|
||||
var transparencyPercent: Int32
|
||||
|
||||
static func ==(lhs: DeletedMessagesControllerState, rhs: DeletedMessagesControllerState) -> Bool {
|
||||
return lhs.isEnabled == rhs.isEnabled &&
|
||||
lhs.archiveMedia == rhs.archiveMedia &&
|
||||
lhs.archivedCount == rhs.archivedCount &&
|
||||
lhs.transparencyPercent == rhs.transparencyPercent
|
||||
}
|
||||
}
|
||||
@@ -162,6 +189,7 @@ private func deletedMessagesControllerEntries(
|
||||
|
||||
entries.append(.enableToggle(presentationData.theme, "Сохранять удалённые сообщения", state.isEnabled))
|
||||
entries.append(.archiveMediaToggle(presentationData.theme, "Архивировать медиа", state.archiveMedia))
|
||||
entries.append(.history(presentationData.theme, "История удалений", state.archivedCount == 0 ? "Пусто" : "\(state.archivedCount)"))
|
||||
entries.append(.transparencySlider(presentationData.theme, state.transparencyPercent, state.isEnabled))
|
||||
entries.append(.settingsInfo(presentationData.theme, "Когда включено, сообщения, удалённые другими пользователями, будут сохраняться локально. Прозрачность влияет только на сообщения, которые уже помечены как удалённые."))
|
||||
|
||||
@@ -171,9 +199,12 @@ private func deletedMessagesControllerEntries(
|
||||
// MARK: - Controller
|
||||
|
||||
public func deletedMessagesController(context: AccountContext) -> ViewController {
|
||||
var pushControllerImpl: ((ViewController, Bool) -> Void)?
|
||||
|
||||
let initialState = DeletedMessagesControllerState(
|
||||
isEnabled: AntiDeleteManager.shared.isEnabled,
|
||||
archiveMedia: AntiDeleteManager.shared.archiveMedia,
|
||||
archivedCount: AntiDeleteManager.shared.archivedCount,
|
||||
transparencyPercent: clampDeletedMessageTransparencyPercent(Int32(round(AntiDeleteManager.shared.deletedMessageTransparency * 100.0)))
|
||||
)
|
||||
|
||||
@@ -200,6 +231,9 @@ public func deletedMessagesController(context: AccountContext) -> ViewController
|
||||
return state
|
||||
}
|
||||
},
|
||||
openHistory: {
|
||||
pushControllerImpl?(deletedMessagesHistoryController(context: context), true)
|
||||
},
|
||||
updateTransparency: { value in
|
||||
let clampedValue = clampDeletedMessageTransparencyPercent(value)
|
||||
AntiDeleteManager.shared.deletedMessageTransparency = Double(clampedValue) / 100.0
|
||||
@@ -238,6 +272,257 @@ public func deletedMessagesController(context: AccountContext) -> ViewController
|
||||
}
|
||||
|
||||
let controller = ItemListController(context: context, state: signal)
|
||||
controller.didAppear = { _ in
|
||||
updateState { state in
|
||||
var state = state
|
||||
state.isEnabled = AntiDeleteManager.shared.isEnabled
|
||||
state.archiveMedia = AntiDeleteManager.shared.archiveMedia
|
||||
state.archivedCount = AntiDeleteManager.shared.archivedCount
|
||||
state.transparencyPercent = clampDeletedMessageTransparencyPercent(Int32(round(AntiDeleteManager.shared.deletedMessageTransparency * 100.0)))
|
||||
return state
|
||||
}
|
||||
}
|
||||
pushControllerImpl = { [weak controller] c, _ in
|
||||
controller?.push(c)
|
||||
}
|
||||
return controller
|
||||
}
|
||||
|
||||
// MARK: - Deleted History
|
||||
|
||||
private enum DeletedMessagesHistorySection: Int32 {
|
||||
case actions
|
||||
case messages
|
||||
}
|
||||
|
||||
private enum DeletedMessagesHistoryEntry: ItemListNodeEntry {
|
||||
case clear(PresentationTheme, Bool)
|
||||
case message(PresentationTheme, Int32, String, String)
|
||||
case empty(PresentationTheme, String)
|
||||
|
||||
var section: ItemListSectionId {
|
||||
switch self {
|
||||
case .clear:
|
||||
return DeletedMessagesHistorySection.actions.rawValue
|
||||
case .message, .empty:
|
||||
return DeletedMessagesHistorySection.messages.rawValue
|
||||
}
|
||||
}
|
||||
|
||||
var stableId: Int32 {
|
||||
switch self {
|
||||
case .clear:
|
||||
return 0
|
||||
case let .message(_, index, _, _):
|
||||
return 1000 + index
|
||||
case .empty:
|
||||
return 1
|
||||
}
|
||||
}
|
||||
|
||||
static func ==(lhs: DeletedMessagesHistoryEntry, rhs: DeletedMessagesHistoryEntry) -> Bool {
|
||||
switch lhs {
|
||||
case let .clear(lhsTheme, lhsEnabled):
|
||||
if case let .clear(rhsTheme, rhsEnabled) = rhs, lhsTheme === rhsTheme, lhsEnabled == rhsEnabled {
|
||||
return true
|
||||
}
|
||||
return false
|
||||
case let .message(lhsTheme, lhsIndex, lhsTitle, lhsLabel):
|
||||
if case let .message(rhsTheme, rhsIndex, rhsTitle, rhsLabel) = rhs, lhsTheme === rhsTheme, lhsIndex == rhsIndex, lhsTitle == rhsTitle, lhsLabel == rhsLabel {
|
||||
return true
|
||||
}
|
||||
return false
|
||||
case let .empty(lhsTheme, lhsText):
|
||||
if case let .empty(rhsTheme, rhsText) = rhs, lhsTheme === rhsTheme, lhsText == rhsText {
|
||||
return true
|
||||
}
|
||||
return false
|
||||
}
|
||||
}
|
||||
|
||||
static func <(lhs: DeletedMessagesHistoryEntry, rhs: DeletedMessagesHistoryEntry) -> Bool {
|
||||
return lhs.stableId < rhs.stableId
|
||||
}
|
||||
|
||||
func item(presentationData: ItemListPresentationData, arguments: Any) -> ListViewItem {
|
||||
let arguments = arguments as! DeletedMessagesHistoryControllerArguments
|
||||
|
||||
switch self {
|
||||
case let .clear(_, enabled):
|
||||
return ItemListActionItem(
|
||||
presentationData: presentationData,
|
||||
systemStyle: .glass,
|
||||
title: "Очистить историю",
|
||||
kind: enabled ? .destructive : .disabled,
|
||||
alignment: .natural,
|
||||
sectionId: self.section,
|
||||
style: .blocks,
|
||||
action: {
|
||||
if enabled {
|
||||
arguments.clearHistory()
|
||||
}
|
||||
}
|
||||
)
|
||||
case let .message(_, _, title, label):
|
||||
return ItemListTextItem(
|
||||
presentationData: presentationData,
|
||||
text: .plain("\(label)\n\(title)"),
|
||||
sectionId: self.section
|
||||
)
|
||||
case let .empty(_, text):
|
||||
return ItemListTextItem(
|
||||
presentationData: presentationData,
|
||||
text: .plain(text),
|
||||
sectionId: self.section
|
||||
)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private final class DeletedMessagesHistoryControllerArguments {
|
||||
let clearHistory: () -> Void
|
||||
|
||||
init(clearHistory: @escaping () -> Void) {
|
||||
self.clearHistory = clearHistory
|
||||
}
|
||||
}
|
||||
|
||||
private struct DeletedMessagesHistoryControllerState: Equatable {
|
||||
var messages: [AntiDeleteManager.ArchivedMessage]
|
||||
|
||||
static func ==(lhs: DeletedMessagesHistoryControllerState, rhs: DeletedMessagesHistoryControllerState) -> Bool {
|
||||
if lhs.messages.count != rhs.messages.count {
|
||||
return false
|
||||
}
|
||||
for (lhsMessage, rhsMessage) in zip(lhs.messages, rhs.messages) {
|
||||
if lhsMessage.globalId != rhsMessage.globalId ||
|
||||
lhsMessage.peerId != rhsMessage.peerId ||
|
||||
lhsMessage.messageId != rhsMessage.messageId ||
|
||||
lhsMessage.timestamp != rhsMessage.timestamp ||
|
||||
lhsMessage.deletedAt != rhsMessage.deletedAt ||
|
||||
lhsMessage.authorId != rhsMessage.authorId ||
|
||||
lhsMessage.text != rhsMessage.text ||
|
||||
lhsMessage.forwardAuthorId != rhsMessage.forwardAuthorId ||
|
||||
lhsMessage.mediaDescription != rhsMessage.mediaDescription {
|
||||
return false
|
||||
}
|
||||
}
|
||||
return true
|
||||
}
|
||||
}
|
||||
|
||||
private func deletedMessagesHistoryDateString(timestamp: Int32) -> String {
|
||||
let formatter = DateFormatter()
|
||||
formatter.locale = Locale.current
|
||||
formatter.dateStyle = .short
|
||||
formatter.timeStyle = .medium
|
||||
return formatter.string(from: Date(timeIntervalSince1970: TimeInterval(timestamp)))
|
||||
}
|
||||
|
||||
private func deletedMessagesHistoryEntries(
|
||||
presentationData: PresentationData,
|
||||
state: DeletedMessagesHistoryControllerState
|
||||
) -> [DeletedMessagesHistoryEntry] {
|
||||
var entries: [DeletedMessagesHistoryEntry] = []
|
||||
entries.append(.clear(presentationData.theme, !state.messages.isEmpty))
|
||||
|
||||
if state.messages.isEmpty {
|
||||
entries.append(.empty(presentationData.theme, "История удалений пуста."))
|
||||
return entries
|
||||
}
|
||||
|
||||
for (index, message) in state.messages.enumerated() {
|
||||
var preview = message.text.trimmingCharacters(in: .whitespacesAndNewlines)
|
||||
if preview.isEmpty {
|
||||
preview = message.mediaDescription ?? "Сообщение без текста"
|
||||
}
|
||||
preview = preview.replacingOccurrences(of: "\n", with: " ")
|
||||
if preview.count > 80 {
|
||||
preview = String(preview.prefix(80)) + "..."
|
||||
}
|
||||
|
||||
let title = "Чат \(message.peerId): \(preview)"
|
||||
let label = deletedMessagesHistoryDateString(timestamp: message.deletedAt)
|
||||
|
||||
entries.append(.message(presentationData.theme, Int32(index), title, label))
|
||||
}
|
||||
|
||||
return entries
|
||||
}
|
||||
|
||||
private func deletedMessagesHistoryController(context: AccountContext) -> ViewController {
|
||||
var presentControllerImpl: ((ViewController, ViewControllerPresentationArguments?) -> Void)?
|
||||
|
||||
let initialState = DeletedMessagesHistoryControllerState(
|
||||
messages: AntiDeleteManager.shared.getAllArchivedMessages()
|
||||
)
|
||||
|
||||
let statePromise = ValuePromise(initialState, ignoreRepeated: true)
|
||||
let stateValue = Atomic(value: initialState)
|
||||
let updateState: ((DeletedMessagesHistoryControllerState) -> DeletedMessagesHistoryControllerState) -> Void = { f in
|
||||
statePromise.set(stateValue.modify { f($0) })
|
||||
}
|
||||
|
||||
let arguments = DeletedMessagesHistoryControllerArguments(
|
||||
clearHistory: {
|
||||
let presentationData = context.sharedContext.currentPresentationData.with { $0 }
|
||||
let alert = textAlertController(
|
||||
context: context,
|
||||
title: "Очистить историю?",
|
||||
text: "Это удалит локально сохранённые удалённые сообщения.",
|
||||
actions: [
|
||||
TextAlertAction(type: .genericAction, title: presentationData.strings.Common_Cancel, action: {}),
|
||||
TextAlertAction(type: .destructiveAction, title: "Очистить", action: {
|
||||
AntiDeleteManager.shared.clearArchive()
|
||||
updateState { state in
|
||||
var state = state
|
||||
state.messages = AntiDeleteManager.shared.getAllArchivedMessages()
|
||||
return state
|
||||
}
|
||||
})
|
||||
]
|
||||
)
|
||||
presentControllerImpl?(alert, nil)
|
||||
}
|
||||
)
|
||||
|
||||
let signal: Signal<(ItemListControllerState, (ItemListNodeState, DeletedMessagesHistoryControllerArguments)), NoError> = combineLatest(
|
||||
context.sharedContext.presentationData,
|
||||
statePromise.get()
|
||||
)
|
||||
|> map { presentationData, state -> (ItemListControllerState, (ItemListNodeState, DeletedMessagesHistoryControllerArguments)) in
|
||||
let entries = deletedMessagesHistoryEntries(presentationData: presentationData, state: state)
|
||||
|
||||
let controllerState = ItemListControllerState(
|
||||
presentationData: ItemListPresentationData(presentationData),
|
||||
title: .text("История удалений"),
|
||||
leftNavigationButton: nil,
|
||||
rightNavigationButton: nil,
|
||||
backNavigationButton: ItemListBackButton(title: presentationData.strings.Common_Back),
|
||||
animateChanges: false
|
||||
)
|
||||
|
||||
let listState = ItemListNodeState(
|
||||
presentationData: ItemListPresentationData(presentationData),
|
||||
entries: entries,
|
||||
style: .blocks,
|
||||
animateChanges: false
|
||||
)
|
||||
|
||||
return (controllerState, (listState, arguments))
|
||||
}
|
||||
|
||||
let controller = ItemListController(context: context, state: signal)
|
||||
controller.didAppear = { _ in
|
||||
updateState { state in
|
||||
var state = state
|
||||
state.messages = AntiDeleteManager.shared.getAllArchivedMessages()
|
||||
return state
|
||||
}
|
||||
}
|
||||
presentControllerImpl = { [weak controller] c, a in
|
||||
controller?.present(c, in: .window(.root), with: a)
|
||||
}
|
||||
return controller
|
||||
}
|
||||
|
||||
|
||||
@@ -236,24 +236,28 @@ public func deviceSpoofController(context: AccountContext) -> ViewController {
|
||||
|
||||
let arguments = DeviceSpoofControllerArguments(
|
||||
toggleEnabled: { value in
|
||||
DeviceSpoofManager.shared.hasExplicitConfiguration = true
|
||||
DeviceSpoofManager.shared.isEnabled = value
|
||||
updateState { state in
|
||||
state.isEnabled = value
|
||||
}
|
||||
},
|
||||
selectProfile: { id in
|
||||
DeviceSpoofManager.shared.hasExplicitConfiguration = true
|
||||
DeviceSpoofManager.shared.selectedProfileId = id
|
||||
updateState { state in
|
||||
state.selectedProfileId = id
|
||||
}
|
||||
},
|
||||
updateCustomDeviceModel: { text in
|
||||
DeviceSpoofManager.shared.hasExplicitConfiguration = true
|
||||
DeviceSpoofManager.shared.customDeviceModel = text
|
||||
updateState { state in
|
||||
state.customDeviceModel = text
|
||||
}
|
||||
},
|
||||
updateCustomSystemVersion: { text in
|
||||
DeviceSpoofManager.shared.hasExplicitConfiguration = true
|
||||
DeviceSpoofManager.shared.customSystemVersion = text
|
||||
updateState { state in
|
||||
state.customSystemVersion = text
|
||||
|
||||
+66
-33
@@ -89,6 +89,20 @@ private struct SortIndex: Comparable {
|
||||
}
|
||||
}
|
||||
|
||||
public enum RecentSessionsEntryTag: ItemListItemTag, Equatable {
|
||||
case edit
|
||||
case terminateOtherSessions
|
||||
case autoTerminate
|
||||
|
||||
public func isEqual(to other: ItemListItemTag) -> Bool {
|
||||
if let other = other as? RecentSessionsEntryTag, self == other {
|
||||
return true
|
||||
} else {
|
||||
return false
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private enum RecentSessionsEntry: ItemListNodeEntry {
|
||||
case header(SortIndex, String)
|
||||
case currentSessionHeader(SortIndex, String)
|
||||
@@ -339,7 +353,7 @@ private enum RecentSessionsEntry: ItemListNodeEntry {
|
||||
arguments.openSession(session)
|
||||
})
|
||||
case let .terminateOtherSessions(_, text):
|
||||
return ItemListPeerActionItem(presentationData: presentationData, systemStyle: .glass, icon: PresentationResourcesItemList.blockDestructiveIcon(presentationData.theme), title: text, sectionId: self.section, height: .generic, color: .destructive, editing: false, action: {
|
||||
return ItemListPeerActionItem(presentationData: presentationData, systemStyle: .glass, icon: PresentationResourcesItemList.blockDestructiveIcon(presentationData.theme), title: text, sectionId: self.section, height: .generic, color: .destructive, editing: false, tag: RecentSessionsEntryTag.terminateOtherSessions, action: {
|
||||
arguments.terminateOtherSessions()
|
||||
})
|
||||
case let .terminateAllWebSessions(_, text):
|
||||
@@ -403,7 +417,7 @@ private enum RecentSessionsEntry: ItemListNodeEntry {
|
||||
case let .ttlTimeout(_, text, value):
|
||||
return ItemListDisclosureItem(presentationData: presentationData, systemStyle: .glass, title: text, label: value, sectionId: self.section, style: .blocks, action: {
|
||||
arguments.setupAuthorizationTTL()
|
||||
}, tag: PrivacyAndSecurityEntryTag.accountTimeout)
|
||||
}, tag: RecentSessionsEntryTag.autoTerminate)
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -560,13 +574,19 @@ private func recentSessionsControllerEntries(presentationData: PresentationData,
|
||||
private final class RecentSessionsControllerImpl: ItemListController, RecentSessionsController {
|
||||
}
|
||||
|
||||
public func recentSessionsController(context: AccountContext, activeSessionsContext: ActiveSessionsContext, webSessionsContext: WebSessionsContext, websitesOnly: Bool) -> ViewController & RecentSessionsController {
|
||||
public func recentSessionsController(context: AccountContext, activeSessionsContext: ActiveSessionsContext, webSessionsContext: WebSessionsContext, websitesOnly: Bool, focusOnItemTag: RecentSessionsEntryTag? = nil) -> ViewController & RecentSessionsController {
|
||||
let statePromise = ValuePromise(RecentSessionsControllerState(), ignoreRepeated: true)
|
||||
let stateValue = Atomic(value: RecentSessionsControllerState())
|
||||
let updateState: ((RecentSessionsControllerState) -> RecentSessionsControllerState) -> Void = { f in
|
||||
statePromise.set(stateValue.modify { f($0) })
|
||||
}
|
||||
|
||||
if focusOnItemTag == .edit {
|
||||
updateState {
|
||||
$0.withUpdatedEditing(true)
|
||||
}
|
||||
}
|
||||
|
||||
activeSessionsContext.loadMore()
|
||||
webSessionsContext.loadMore()
|
||||
|
||||
@@ -603,15 +623,12 @@ public func recentSessionsController(context: AccountContext, activeSessionsCont
|
||||
|
||||
let removeSessionImpl: (Int64, @escaping () -> Void) -> Void = { sessionId, completion in
|
||||
let presentationData = context.sharedContext.currentPresentationData.with { $0 }
|
||||
let controller = ActionSheetController(presentationData: presentationData)
|
||||
let dismissAction: () -> Void = { [weak controller] in
|
||||
controller?.dismissAnimated()
|
||||
}
|
||||
controller.setItemGroups([
|
||||
ActionSheetItemGroup(items: [
|
||||
ActionSheetTextItem(title: presentationData.strings.AuthSessions_TerminateSessionText),
|
||||
ActionSheetButtonItem(title: presentationData.strings.AuthSessions_TerminateSession, color: .destructive, action: {
|
||||
dismissAction()
|
||||
let controller = textAlertController(
|
||||
context: context,
|
||||
title: nil,
|
||||
text: presentationData.strings.AuthSessions_TerminateSessionText,
|
||||
actions: [
|
||||
TextAlertAction(type: .defaultDestructiveAction, title: presentationData.strings.AuthSessions_TerminateSession, action: {
|
||||
completion()
|
||||
|
||||
updateState {
|
||||
@@ -629,11 +646,12 @@ public func recentSessionsController(context: AccountContext, activeSessionsCont
|
||||
}
|
||||
context.sharedContext.updateNotificationTokensRegistration()
|
||||
}))
|
||||
})
|
||||
]),
|
||||
ActionSheetItemGroup(items: [ActionSheetButtonItem(title: presentationData.strings.Common_Cancel, action: { dismissAction() })])
|
||||
])
|
||||
presentControllerImpl?(controller, ViewControllerPresentationArguments(presentationAnimation: .modalSheet))
|
||||
}),
|
||||
TextAlertAction(type: .genericAction, title: presentationData.strings.Common_Cancel, action: {})
|
||||
],
|
||||
actionLayout: .vertical
|
||||
)
|
||||
presentControllerImpl?(controller, nil)
|
||||
}
|
||||
|
||||
let removeWebSessionImpl: (Int64) -> Void = { sessionId in
|
||||
@@ -670,16 +688,12 @@ public func recentSessionsController(context: AccountContext, activeSessionsCont
|
||||
removeSessionImpl(sessionId, {})
|
||||
}, terminateOtherSessions: {
|
||||
let presentationData = context.sharedContext.currentPresentationData.with { $0 }
|
||||
let controller = ActionSheetController(presentationData: presentationData)
|
||||
let dismissAction: () -> Void = { [weak controller] in
|
||||
controller?.dismissAnimated()
|
||||
}
|
||||
controller.setItemGroups([
|
||||
ActionSheetItemGroup(items: [
|
||||
ActionSheetTextItem(title: presentationData.strings.AuthSessions_TerminateOtherSessionsText),
|
||||
ActionSheetButtonItem(title: presentationData.strings.AuthSessions_TerminateOtherSessions, color: .destructive, action: {
|
||||
dismissAction()
|
||||
|
||||
let controller = textAlertController(
|
||||
context: context,
|
||||
title: nil,
|
||||
text: presentationData.strings.AuthSessions_TerminateOtherSessionsText,
|
||||
actions: [
|
||||
TextAlertAction(type: .defaultDestructiveAction, title: presentationData.strings.AuthSessions_TerminateOtherSessions, action: {
|
||||
updateState {
|
||||
return $0.withUpdatedTerminatingOtherSessions(true)
|
||||
}
|
||||
@@ -695,11 +709,12 @@ public func recentSessionsController(context: AccountContext, activeSessionsCont
|
||||
}
|
||||
context.sharedContext.updateNotificationTokensRegistration()
|
||||
}))
|
||||
})
|
||||
]),
|
||||
ActionSheetItemGroup(items: [ActionSheetButtonItem(title: presentationData.strings.Common_Cancel, action: { dismissAction() })])
|
||||
])
|
||||
presentControllerImpl?(controller, ViewControllerPresentationArguments(presentationAnimation: .modalSheet))
|
||||
}),
|
||||
TextAlertAction(type: .genericAction, title: presentationData.strings.Common_Cancel, action: {})
|
||||
],
|
||||
actionLayout: .vertical
|
||||
)
|
||||
presentControllerImpl?(controller, nil)
|
||||
}, openSession: { session in
|
||||
let controller = RecentSessionScreen(context: context, subject: .session(session), updateAcceptSecretChats: { value in
|
||||
updateSessionDisposable.set(activeSessionsContext.updateSessionAcceptsSecretChats(session, accepts: value).start())
|
||||
@@ -801,6 +816,10 @@ public func recentSessionsController(context: AccountContext, activeSessionsCont
|
||||
guard let appConfiguration = view.values[PreferencesKeys.appConfiguration]?.get(AppConfiguration.self) else {
|
||||
return false
|
||||
}
|
||||
// MARK: Swiftgram
|
||||
if appConfiguration.sgWebSettings.global.qrLogin {
|
||||
return true
|
||||
}
|
||||
guard let data = appConfiguration.data, let enableQR = data["qr_login_camera"] as? Bool, enableQR else {
|
||||
return false
|
||||
}
|
||||
@@ -865,7 +884,7 @@ public func recentSessionsController(context: AccountContext, activeSessionsCont
|
||||
}
|
||||
|
||||
let controllerState = ItemListControllerState(presentationData: ItemListPresentationData(presentationData), title: title, leftNavigationButton: nil, rightNavigationButton: rightNavigationButton, backNavigationButton: ItemListBackButton(title: presentationData.strings.Common_Back), animateChanges: true)
|
||||
let listState = ItemListNodeState(presentationData: ItemListPresentationData(presentationData), entries: entries, style: .blocks, emptyStateItem: emptyStateItem, crossfadeState: crossfadeState, animateChanges: animateChanges, scrollEnabled: emptyStateItem == nil)
|
||||
let listState = ItemListNodeState(presentationData: ItemListPresentationData(presentationData), entries: entries, style: .blocks, ensureVisibleItemTag: focusOnItemTag, emptyStateItem: emptyStateItem, crossfadeState: crossfadeState, animateChanges: animateChanges, scrollEnabled: emptyStateItem == nil)
|
||||
|
||||
return (controllerState, (listState, arguments))
|
||||
} |> afterDisposed {
|
||||
@@ -891,5 +910,19 @@ public func recentSessionsController(context: AccountContext, activeSessionsCont
|
||||
controller?.dismiss()
|
||||
}
|
||||
|
||||
if let focusOnItemTag {
|
||||
var didFocusOnItem = false
|
||||
controller.afterTransactionCompleted = { [weak controller] in
|
||||
if !didFocusOnItem, let controller {
|
||||
controller.forEachItemNode { itemNode in
|
||||
if let itemNode = itemNode as? ItemListItemNode, let tag = itemNode.tag, tag.isEqual(to: focusOnItemTag) {
|
||||
didFocusOnItem = true
|
||||
itemNode.displayHighlight()
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return controller
|
||||
}
|
||||
|
||||
File diff suppressed because it is too large
Load Diff
@@ -23,6 +23,7 @@ import ThemeCarouselItem
|
||||
import ThemeAccentColorScreen
|
||||
import WallpaperGridScreen
|
||||
import PeerNameColorItem
|
||||
import DeviceModel
|
||||
|
||||
private final class ThemeSettingsControllerArguments {
|
||||
let context: AccountContext
|
||||
@@ -38,13 +39,34 @@ private final class ThemeSettingsControllerArguments {
|
||||
let openBubbleSettings: () -> Void
|
||||
let openPowerSavingSettings: () -> Void
|
||||
let openStickersAndEmoji: () -> Void
|
||||
let toggleSendWithCmdEnter: (Bool) -> Void
|
||||
let toggleShowNextMediaOnTap: (Bool) -> Void
|
||||
let selectAppIcon: (PresentationAppIcon) -> Void
|
||||
let editTheme: (PresentationCloudTheme) -> Void
|
||||
let themeContextAction: (Bool, PresentationThemeReference, ASDisplayNode, ContextGesture?) -> Void
|
||||
let colorContextAction: (Bool, PresentationThemeReference, ThemeSettingsColorOption?, ASDisplayNode, ContextGesture?) -> Void
|
||||
|
||||
init(context: AccountContext, selectTheme: @escaping (PresentationThemeReference) -> Void, openThemeSettings: @escaping () -> Void, openWallpaperSettings: @escaping () -> Void, openNameColorSettings: @escaping () -> Void, selectAccentColor: @escaping (PresentationThemeAccentColor?) -> Void, openAccentColorPicker: @escaping (PresentationThemeReference, Bool) -> Void, toggleNightTheme: @escaping (Bool) -> Void, openAutoNightTheme: @escaping () -> Void, openTextSize: @escaping () -> Void, openBubbleSettings: @escaping () -> Void, openPowerSavingSettings: @escaping () -> Void, openStickersAndEmoji: @escaping () -> Void, toggleShowNextMediaOnTap: @escaping (Bool) -> Void, selectAppIcon: @escaping (PresentationAppIcon) -> Void, editTheme: @escaping (PresentationCloudTheme) -> Void, themeContextAction: @escaping (Bool, PresentationThemeReference, ASDisplayNode, ContextGesture?) -> Void, colorContextAction: @escaping (Bool, PresentationThemeReference, ThemeSettingsColorOption?, ASDisplayNode, ContextGesture?) -> Void) {
|
||||
init(
|
||||
context: AccountContext,
|
||||
selectTheme: @escaping (PresentationThemeReference) -> Void,
|
||||
openThemeSettings: @escaping () -> Void,
|
||||
openWallpaperSettings: @escaping () -> Void,
|
||||
openNameColorSettings: @escaping () -> Void,
|
||||
selectAccentColor: @escaping (PresentationThemeAccentColor?) -> Void,
|
||||
openAccentColorPicker: @escaping (PresentationThemeReference, Bool) -> Void,
|
||||
toggleNightTheme: @escaping (Bool) -> Void,
|
||||
openAutoNightTheme: @escaping () -> Void,
|
||||
openTextSize: @escaping () -> Void,
|
||||
openBubbleSettings: @escaping () -> Void,
|
||||
openPowerSavingSettings: @escaping () -> Void,
|
||||
openStickersAndEmoji: @escaping () -> Void,
|
||||
toggleSendWithCmdEnter: @escaping (Bool) -> Void,
|
||||
toggleShowNextMediaOnTap: @escaping (Bool) -> Void,
|
||||
selectAppIcon: @escaping (PresentationAppIcon) -> Void,
|
||||
editTheme: @escaping (PresentationCloudTheme) -> Void,
|
||||
themeContextAction: @escaping (Bool, PresentationThemeReference, ASDisplayNode, ContextGesture?) -> Void,
|
||||
colorContextAction: @escaping (Bool, PresentationThemeReference, ThemeSettingsColorOption?, ASDisplayNode, ContextGesture?) -> Void
|
||||
) {
|
||||
self.context = context
|
||||
self.selectTheme = selectTheme
|
||||
self.openThemeSettings = openThemeSettings
|
||||
@@ -58,6 +80,7 @@ private final class ThemeSettingsControllerArguments {
|
||||
self.openBubbleSettings = openBubbleSettings
|
||||
self.openPowerSavingSettings = openPowerSavingSettings
|
||||
self.openStickersAndEmoji = openStickersAndEmoji
|
||||
self.toggleSendWithCmdEnter = toggleSendWithCmdEnter
|
||||
self.toggleShowNextMediaOnTap = toggleShowNextMediaOnTap
|
||||
self.selectAppIcon = selectAppIcon
|
||||
self.editTheme = editTheme
|
||||
@@ -84,6 +107,9 @@ public enum ThemeSettingsEntryTag: ItemListItemTag {
|
||||
case powerSaving
|
||||
case stickersAndEmoji
|
||||
case animations
|
||||
case sendWithCmdEnter
|
||||
case tapForNextMedia
|
||||
case nightMode
|
||||
|
||||
public func isEqual(to other: ItemListItemTag) -> Bool {
|
||||
if let other = other as? ThemeSettingsEntryTag, self == other {
|
||||
@@ -110,6 +136,7 @@ private enum ThemeSettingsControllerEntry: ItemListNodeEntry {
|
||||
case powerSaving
|
||||
case stickersAndEmoji
|
||||
case otherHeader(PresentationTheme, String)
|
||||
case sendWithCmdEnter(PresentationTheme, String, Bool)
|
||||
case showNextMediaOnTap(PresentationTheme, String, Bool)
|
||||
case showNextMediaOnTapInfo(PresentationTheme, String)
|
||||
|
||||
@@ -125,7 +152,7 @@ private enum ThemeSettingsControllerEntry: ItemListNodeEntry {
|
||||
return ThemeSettingsControllerSection.icon.rawValue
|
||||
case .powerSaving, .stickersAndEmoji:
|
||||
return ThemeSettingsControllerSection.message.rawValue
|
||||
case .otherHeader, .showNextMediaOnTap, .showNextMediaOnTapInfo:
|
||||
case .otherHeader, .sendWithCmdEnter, .showNextMediaOnTap, .showNextMediaOnTapInfo:
|
||||
return ThemeSettingsControllerSection.other.rawValue
|
||||
}
|
||||
}
|
||||
@@ -162,10 +189,12 @@ private enum ThemeSettingsControllerEntry: ItemListNodeEntry {
|
||||
return 13
|
||||
case .otherHeader:
|
||||
return 14
|
||||
case .showNextMediaOnTap:
|
||||
case .sendWithCmdEnter:
|
||||
return 15
|
||||
case .showNextMediaOnTapInfo:
|
||||
case .showNextMediaOnTap:
|
||||
return 16
|
||||
case .showNextMediaOnTapInfo:
|
||||
return 17
|
||||
}
|
||||
}
|
||||
|
||||
@@ -261,6 +290,12 @@ private enum ThemeSettingsControllerEntry: ItemListNodeEntry {
|
||||
} else {
|
||||
return false
|
||||
}
|
||||
case let .sendWithCmdEnter(lhsTheme, lhsTitle, lhsValue):
|
||||
if case let .sendWithCmdEnter(rhsTheme, rhsTitle, rhsValue) = rhs, lhsTheme === rhsTheme, lhsTitle == rhsTitle, lhsValue == rhsValue {
|
||||
return true
|
||||
} else {
|
||||
return false
|
||||
}
|
||||
case let .showNextMediaOnTap(lhsTheme, lhsTitle, lhsValue):
|
||||
if case let .showNextMediaOnTap(rhsTheme, rhsTitle, rhsValue) = rhs, lhsTheme === rhsTheme, lhsTitle == rhsTitle, lhsValue == rhsValue {
|
||||
return true
|
||||
@@ -318,7 +353,7 @@ private enum ThemeSettingsControllerEntry: ItemListNodeEntry {
|
||||
case let .autoNight(_, title, value, enabled):
|
||||
return ItemListSwitchItem(presentationData: presentationData, systemStyle: .glass, title: title, value: value, enabled: enabled, sectionId: self.section, style: .blocks, updated: { value in
|
||||
arguments.toggleNightTheme(value)
|
||||
}, tag: nil)
|
||||
}, tag: ThemeSettingsEntryTag.nightMode)
|
||||
case let .autoNightTheme(_, text, value):
|
||||
return ItemListDisclosureItem(presentationData: presentationData, systemStyle: .glass, icon: nil, title: text, label: value, labelStyle: .text, sectionId: self.section, style: .blocks, disclosureStyle: .arrow, action: {
|
||||
arguments.openAutoNightTheme()
|
||||
@@ -349,17 +384,35 @@ private enum ThemeSettingsControllerEntry: ItemListNodeEntry {
|
||||
})
|
||||
case let .otherHeader(_, text):
|
||||
return ItemListSectionHeaderItem(presentationData: presentationData, text: text, sectionId: self.section)
|
||||
case let .sendWithCmdEnter(_, title, value):
|
||||
return ItemListSwitchItem(presentationData: presentationData, systemStyle: .glass, title: title, value: value, sectionId: self.section, style: .blocks, updated: { value in
|
||||
arguments.toggleSendWithCmdEnter(value)
|
||||
}, tag: ThemeSettingsEntryTag.sendWithCmdEnter)
|
||||
case let .showNextMediaOnTap(_, title, value):
|
||||
return ItemListSwitchItem(presentationData: presentationData, systemStyle: .glass, title: title, value: value, sectionId: self.section, style: .blocks, updated: { value in
|
||||
arguments.toggleShowNextMediaOnTap(value)
|
||||
}, tag: ThemeSettingsEntryTag.animations)
|
||||
}, tag: ThemeSettingsEntryTag.tapForNextMedia)
|
||||
case let .showNextMediaOnTapInfo(_, text):
|
||||
return ItemListTextItem(presentationData: presentationData, text: .plain(text), sectionId: self.section)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private func themeSettingsControllerEntries(presentationData: PresentationData, presentationThemeSettings: PresentationThemeSettings, mediaSettings: MediaDisplaySettings, themeReference: PresentationThemeReference, availableThemes: [PresentationThemeReference], availableAppIcons: [PresentationAppIcon], currentAppIconName: String?, isPremium: Bool, chatThemes: [PresentationThemeReference], animatedEmojiStickers: [String: [StickerPackItem]], accountPeer: EnginePeer?, nameColors: PeerNameColors) -> [ThemeSettingsControllerEntry] {
|
||||
private func themeSettingsControllerEntries(
|
||||
presentationData: PresentationData,
|
||||
presentationThemeSettings: PresentationThemeSettings,
|
||||
chatSettings: ChatSettings,
|
||||
mediaSettings: MediaDisplaySettings,
|
||||
themeReference: PresentationThemeReference,
|
||||
availableThemes: [PresentationThemeReference],
|
||||
availableAppIcons: [PresentationAppIcon],
|
||||
currentAppIconName: String?,
|
||||
isPremium: Bool,
|
||||
chatThemes: [PresentationThemeReference],
|
||||
animatedEmojiStickers: [String: [StickerPackItem]],
|
||||
accountPeer: EnginePeer?,
|
||||
nameColors: PeerNameColors
|
||||
) -> [ThemeSettingsControllerEntry] {
|
||||
var entries: [ThemeSettingsControllerEntry] = []
|
||||
|
||||
let strings = presentationData.strings
|
||||
@@ -435,6 +488,9 @@ private func themeSettingsControllerEntries(presentationData: PresentationData,
|
||||
}
|
||||
|
||||
entries.append(.otherHeader(presentationData.theme, strings.Appearance_Other.uppercased()))
|
||||
if DeviceModel.current.isIpad {
|
||||
entries.append(.sendWithCmdEnter(presentationData.theme, "Send with Cmd+Enter", chatSettings.sendWithCmdEnter))
|
||||
}
|
||||
entries.append(.showNextMediaOnTap(presentationData.theme, strings.Appearance_ShowNextMediaOnTap, mediaSettings.showNextMediaOnTap))
|
||||
entries.append(.showNextMediaOnTapInfo(presentationData.theme, strings.Appearance_ShowNextMediaOnTapInfo))
|
||||
|
||||
@@ -513,73 +569,96 @@ public func themeSettingsController(context: AccountContext, focusOnItemTag: The
|
||||
return animatedEmojiStickers
|
||||
}
|
||||
|
||||
let arguments = ThemeSettingsControllerArguments(context: context, selectTheme: { theme in
|
||||
let selectThemeArgument: (PresentationThemeReference) -> Void = { theme in
|
||||
selectThemeImpl?(theme)
|
||||
}, openThemeSettings: {
|
||||
}
|
||||
let openThemeSettingsArgument: () -> Void = {
|
||||
pushControllerImpl?(themePickerController(context: context))
|
||||
}, openWallpaperSettings: {
|
||||
}
|
||||
let openWallpaperSettingsArgument: () -> Void = {
|
||||
pushControllerImpl?(ThemeGridController(context: context))
|
||||
}, openNameColorSettings: {
|
||||
}
|
||||
let openNameColorSettingsArgument: () -> Void = {
|
||||
pushControllerImpl?(UserAppearanceScreen(context: context))
|
||||
}, selectAccentColor: { accentColor in
|
||||
}
|
||||
let selectAccentColorArgument: (PresentationThemeAccentColor?) -> Void = { accentColor in
|
||||
selectAccentColorImpl?(accentColor)
|
||||
}, openAccentColorPicker: { themeReference, create in
|
||||
}
|
||||
let openAccentColorPickerArgument: (PresentationThemeReference, Bool) -> Void = { themeReference, create in
|
||||
openAccentColorPickerImpl?(themeReference, create)
|
||||
}, toggleNightTheme: { value in
|
||||
}
|
||||
let toggleNightThemeArgument: (Bool) -> Void = { value in
|
||||
let _ = updatePresentationThemeSettingsInteractively(accountManager: context.sharedContext.accountManager, { current in
|
||||
var current = current
|
||||
current.automaticThemeSwitchSetting.force = value
|
||||
return current
|
||||
}).start()
|
||||
presentCrossfadeControllerImpl?(true)
|
||||
}, openAutoNightTheme: {
|
||||
}
|
||||
let openAutoNightThemeArgument: () -> Void = {
|
||||
pushControllerImpl?(themeAutoNightSettingsController(context: context))
|
||||
}, openTextSize: {
|
||||
}
|
||||
let openTextSizeArgument: () -> Void = {
|
||||
let _ = (context.sharedContext.accountManager.sharedData(keys: Set([ApplicationSpecificSharedDataKeys.presentationThemeSettings]))
|
||||
|> take(1)
|
||||
|> deliverOnMainQueue).start(next: { view in
|
||||
let settings = view.entries[ApplicationSpecificSharedDataKeys.presentationThemeSettings]?.get(PresentationThemeSettings.self) ?? PresentationThemeSettings.defaultSettings
|
||||
pushControllerImpl?(TextSizeSelectionController(context: context, presentationThemeSettings: settings))
|
||||
})
|
||||
}, openBubbleSettings: {
|
||||
}
|
||||
let openBubbleSettingsArgument: () -> Void = {
|
||||
let _ = (context.sharedContext.accountManager.sharedData(keys: Set([ApplicationSpecificSharedDataKeys.presentationThemeSettings]))
|
||||
|> take(1)
|
||||
|> deliverOnMainQueue).start(next: { view in
|
||||
let settings = view.entries[ApplicationSpecificSharedDataKeys.presentationThemeSettings]?.get(PresentationThemeSettings.self) ?? PresentationThemeSettings.defaultSettings
|
||||
pushControllerImpl?(BubbleSettingsController(context: context, presentationThemeSettings: settings))
|
||||
})
|
||||
}, openPowerSavingSettings: {
|
||||
}
|
||||
let openPowerSavingSettingsArgument: () -> Void = {
|
||||
pushControllerImpl?(energySavingSettingsScreen(context: context))
|
||||
}, openStickersAndEmoji: {
|
||||
}
|
||||
let openStickersAndEmojiArgument: () -> Void = {
|
||||
let _ = (archivedPacks.get() |> take(1) |> deliverOnMainQueue).start(next: { archivedStickerPacks in
|
||||
pushControllerImpl?(installedStickerPacksController(context: context, mode: .general, archivedPacks: archivedStickerPacks, updatedPacks: { _ in
|
||||
}))
|
||||
})
|
||||
}, toggleShowNextMediaOnTap: { value in
|
||||
}
|
||||
let toggleSendWithCmdEnterArgument: (Bool) -> Void = { value in
|
||||
let _ = updateChatSettingsInteractively(accountManager: context.sharedContext.accountManager, { current in
|
||||
return current.withUpdatedSendWithCmdEnter(value)
|
||||
}).start()
|
||||
}
|
||||
let toggleShowNextMediaOnTapArgument: (Bool) -> Void = { value in
|
||||
let _ = updateMediaDisplaySettingsInteractively(accountManager: context.sharedContext.accountManager, { current in
|
||||
return current.withUpdatedShowNextMediaOnTap(value)
|
||||
}).start()
|
||||
}, selectAppIcon: { icon in
|
||||
let _ = (context.engine.data.get(TelegramEngine.EngineData.Item.Peer.Peer(id: context.account.peerId))
|
||||
|> deliverOnMainQueue).start(next: { peer in
|
||||
let isPremium = peer?.isPremium ?? false
|
||||
if icon.isPremium && !isPremium {
|
||||
var replaceImpl: ((ViewController) -> Void)?
|
||||
let controller = PremiumDemoScreen(context: context, subject: .appIcons, source: .other, action: {
|
||||
let controller = PremiumIntroScreen(context: context, source: .appIcons)
|
||||
replaceImpl?(controller)
|
||||
})
|
||||
replaceImpl = { [weak controller] c in
|
||||
controller?.replace(with: c)
|
||||
}
|
||||
pushControllerImpl?(controller)
|
||||
} else {
|
||||
currentAppIconName.set(icon.name)
|
||||
context.sharedContext.applicationBindings.requestSetAlternateIconName(icon.isDefault ? nil : icon.name, { _ in
|
||||
})
|
||||
}
|
||||
|
||||
func selectAppIconArgument(_ icon: PresentationAppIcon) {
|
||||
let isPremium = context.isPremium
|
||||
if icon.isPremium && !isPremium {
|
||||
var replaceImpl: ((ViewController) -> Void)?
|
||||
let controller = PremiumDemoScreen(context: context, subject: .appIcons, source: .other, action: {
|
||||
let controller = PremiumIntroScreen(context: context, source: .appIcons)
|
||||
replaceImpl?(controller)
|
||||
})
|
||||
replaceImpl = { [weak controller] c in
|
||||
controller?.replace(with: c)
|
||||
}
|
||||
})
|
||||
}, editTheme: { theme in
|
||||
pushControllerImpl?(controller)
|
||||
} else if icon.isSGPro && context.sharedContext.immediateSGStatus.status < 2 {
|
||||
if let payWallController = context.sharedContext.makeSGPayWallController(context: context) {
|
||||
presentControllerImpl?(payWallController, ViewControllerPresentationArguments(presentationAnimation: .modalSheet))
|
||||
} else {
|
||||
presentControllerImpl?(context.sharedContext.makeSGUpdateIOSController(), nil)
|
||||
}
|
||||
} else {
|
||||
currentAppIconName.set(icon.name)
|
||||
context.sharedContext.applicationBindings.requestSetAlternateIconName(icon.isDefault ? nil : icon.name, { _ in
|
||||
})
|
||||
}
|
||||
}
|
||||
let editThemeArgument: (PresentationCloudTheme) -> Void = { theme in
|
||||
let controller = editThemeController(context: context, mode: .edit(theme), navigateToChat: { peerId in
|
||||
let _ = (context.engine.data.get(TelegramEngine.EngineData.Item.Peer.Peer(id: peerId))
|
||||
|> deliverOnMainQueue).start(next: { peer in
|
||||
@@ -592,7 +671,8 @@ public func themeSettingsController(context: AccountContext, focusOnItemTag: The
|
||||
})
|
||||
})
|
||||
pushControllerImpl?(controller)
|
||||
}, themeContextAction: { isCurrent, reference, node, gesture in
|
||||
}
|
||||
let themeContextActionArgument: (Bool, PresentationThemeReference, ASDisplayNode, ContextGesture?) -> Void = { isCurrent, reference, node, gesture in
|
||||
let _ = (context.sharedContext.accountManager.transaction { transaction -> (PresentationThemeAccentColor?, TelegramWallpaper?) in
|
||||
let settings = transaction.getSharedData(ApplicationSpecificSharedDataKeys.presentationThemeSettings)?.get(PresentationThemeSettings.self) ?? PresentationThemeSettings.defaultSettings
|
||||
let accentColor = settings.themeSpecificAccentColors[reference.index]
|
||||
@@ -779,10 +859,11 @@ public func themeSettingsController(context: AccountContext, focusOnItemTag: The
|
||||
})))
|
||||
}
|
||||
|
||||
let contextController = ContextController(presentationData: presentationData, source: .controller(ContextControllerContentSourceImpl(controller: themeController, sourceNode: node)), items: .single(ContextController.Items(content: .list(items))), gesture: gesture)
|
||||
let contextController = makeContextController(presentationData: presentationData, source: .controller(ContextControllerContentSourceImpl(controller: themeController, sourceNode: node)), items: .single(ContextController.Items(content: .list(items))), gesture: gesture)
|
||||
presentInGlobalOverlayImpl?(contextController, nil)
|
||||
})
|
||||
}, colorContextAction: { isCurrent, reference, accentColor, node, gesture in
|
||||
}
|
||||
let colorContextActionArgument: (Bool, PresentationThemeReference, ThemeSettingsColorOption?, ASDisplayNode, ContextGesture?) -> Void = { isCurrent, reference, accentColor, node, gesture in
|
||||
let _ = (context.sharedContext.accountManager.transaction { transaction -> (ThemeSettingsColorOption?, TelegramWallpaper?) in
|
||||
let settings = transaction.getSharedData(ApplicationSpecificSharedDataKeys.presentationThemeSettings)?.get(PresentationThemeSettings.self) ?? PresentationThemeSettings.defaultSettings
|
||||
var wallpaper: TelegramWallpaper?
|
||||
@@ -1028,17 +1109,56 @@ public func themeSettingsController(context: AccountContext, focusOnItemTag: The
|
||||
}
|
||||
}
|
||||
}
|
||||
let contextController = ContextController(presentationData: presentationData, source: .controller(ContextControllerContentSourceImpl(controller: themeController, sourceNode: node)), items: .single(ContextController.Items(content: .list(items))), gesture: gesture)
|
||||
let contextController = makeContextController(presentationData: presentationData, source: .controller(ContextControllerContentSourceImpl(controller: themeController, sourceNode: node)), items: .single(ContextController.Items(content: .list(items))), gesture: gesture)
|
||||
presentInGlobalOverlayImpl?(contextController, nil)
|
||||
})
|
||||
})
|
||||
}
|
||||
|
||||
let signal = combineLatest(queue: .mainQueue(), context.sharedContext.presentationData, context.sharedContext.accountManager.sharedData(keys: [ApplicationSpecificSharedDataKeys.presentationThemeSettings, SharedDataKeys.chatThemes, ApplicationSpecificSharedDataKeys.mediaDisplaySettings]), cloudThemes.get(), availableAppIcons, currentAppIconName.get(), removedThemeIndexesPromise.get(), animatedEmojiStickers, context.account.postbox.peerView(id: context.account.peerId), context.engine.data.subscribe(TelegramEngine.EngineData.Item.Peer.Peer(id: context.account.peerId)))
|
||||
|> map { presentationData, sharedData, cloudThemes, availableAppIcons, currentAppIconName, removedThemeIndexes, animatedEmojiStickers, peerView, accountPeer -> (ItemListControllerState, (ItemListNodeState, Any)) in
|
||||
let arguments: ThemeSettingsControllerArguments = ThemeSettingsControllerArguments(
|
||||
context: context,
|
||||
selectTheme: selectThemeArgument,
|
||||
openThemeSettings: openThemeSettingsArgument,
|
||||
openWallpaperSettings: openWallpaperSettingsArgument,
|
||||
openNameColorSettings: openNameColorSettingsArgument,
|
||||
selectAccentColor: selectAccentColorArgument,
|
||||
openAccentColorPicker: openAccentColorPickerArgument,
|
||||
toggleNightTheme: toggleNightThemeArgument,
|
||||
openAutoNightTheme: openAutoNightThemeArgument,
|
||||
openTextSize: openTextSizeArgument,
|
||||
openBubbleSettings: openBubbleSettingsArgument,
|
||||
openPowerSavingSettings: openPowerSavingSettingsArgument,
|
||||
openStickersAndEmoji: openStickersAndEmojiArgument,
|
||||
toggleSendWithCmdEnter: toggleSendWithCmdEnterArgument,
|
||||
toggleShowNextMediaOnTap: toggleShowNextMediaOnTapArgument,
|
||||
selectAppIcon: selectAppIconArgument,
|
||||
editTheme: editThemeArgument,
|
||||
themeContextAction: themeContextActionArgument,
|
||||
colorContextAction: colorContextActionArgument
|
||||
)
|
||||
|
||||
let signal: Signal<(ItemListControllerState, (ItemListNodeState, ThemeSettingsControllerArguments)), NoError> = combineLatest(
|
||||
queue: .mainQueue(),
|
||||
context.sharedContext.presentationData,
|
||||
context.sharedContext.accountManager.sharedData(keys: [
|
||||
ApplicationSpecificSharedDataKeys.presentationThemeSettings,
|
||||
ApplicationSpecificSharedDataKeys.chatSettings,
|
||||
ApplicationSpecificSharedDataKeys.mediaDisplaySettings,
|
||||
SharedDataKeys.chatThemes
|
||||
]),
|
||||
cloudThemes.get(),
|
||||
availableAppIcons,
|
||||
currentAppIconName.get(),
|
||||
removedThemeIndexesPromise.get(),
|
||||
animatedEmojiStickers,
|
||||
context.account.postbox.peerView(id: context.account.peerId),
|
||||
context.engine.data.subscribe(TelegramEngine.EngineData.Item.Peer.Peer(id: context.account.peerId))
|
||||
)
|
||||
|> map { presentationData, sharedData, cloudThemes, availableAppIcons, currentAppIconName, removedThemeIndexes, animatedEmojiStickers, peerView, accountPeer -> (ItemListControllerState, (ItemListNodeState, ThemeSettingsControllerArguments)) in
|
||||
let settings = sharedData.entries[ApplicationSpecificSharedDataKeys.presentationThemeSettings]?.get(PresentationThemeSettings.self) ?? PresentationThemeSettings.defaultSettings
|
||||
let chatSettings = sharedData.entries[ApplicationSpecificSharedDataKeys.chatSettings]?.get(ChatSettings.self) ?? ChatSettings.defaultSettings
|
||||
let mediaSettings = sharedData.entries[ApplicationSpecificSharedDataKeys.mediaDisplaySettings]?.get(MediaDisplaySettings.self) ?? MediaDisplaySettings.defaultSettings
|
||||
|
||||
let isPremium = peerView.peers[peerView.peerId]?.isPremium ?? false
|
||||
let isPremium = context.sharedContext.immediateSGStatus.status > 1
|
||||
|
||||
let themeReference: PresentationThemeReference
|
||||
if presentationData.autoNightModeTriggered {
|
||||
@@ -1075,7 +1195,7 @@ public func themeSettingsController(context: AccountContext, focusOnItemTag: The
|
||||
chatThemes.insert(.builtin(.dayClassic), at: 0)
|
||||
|
||||
let controllerState = ItemListControllerState(presentationData: ItemListPresentationData(presentationData), title: .text(presentationData.strings.Appearance_Title), leftNavigationButton: nil, rightNavigationButton: nil, backNavigationButton: ItemListBackButton(title: presentationData.strings.Common_Back))
|
||||
let listState = ItemListNodeState(presentationData: ItemListPresentationData(presentationData), entries: themeSettingsControllerEntries(presentationData: presentationData, presentationThemeSettings: settings, mediaSettings: mediaSettings, themeReference: themeReference, availableThemes: availableThemes, availableAppIcons: availableAppIcons, currentAppIconName: currentAppIconName, isPremium: isPremium, chatThemes: chatThemes, animatedEmojiStickers: animatedEmojiStickers, accountPeer: accountPeer, nameColors: context.peerNameColors), style: .blocks, ensureVisibleItemTag: focusOnItemTag, animateChanges: false)
|
||||
let listState = ItemListNodeState(presentationData: ItemListPresentationData(presentationData), entries: themeSettingsControllerEntries(presentationData: presentationData, presentationThemeSettings: settings, chatSettings: chatSettings, mediaSettings: mediaSettings, themeReference: themeReference, availableThemes: availableThemes, availableAppIcons: availableAppIcons, currentAppIconName: currentAppIconName, isPremium: isPremium, chatThemes: chatThemes, animatedEmojiStickers: animatedEmojiStickers, accountPeer: accountPeer, nameColors: context.peerNameColors), style: .blocks, ensureVisibleItemTag: focusOnItemTag, animateChanges: false)
|
||||
|
||||
return (controllerState, (listState, arguments))
|
||||
}
|
||||
@@ -1309,6 +1429,21 @@ public func themeSettingsController(context: AccountContext, focusOnItemTag: The
|
||||
presentCrossfadeControllerImpl?(true)
|
||||
})
|
||||
}
|
||||
|
||||
if let focusOnItemTag {
|
||||
var didFocusOnItem = false
|
||||
controller.afterTransactionCompleted = { [weak controller] in
|
||||
if !didFocusOnItem, let controller {
|
||||
controller.forEachItemNode { itemNode in
|
||||
if let itemNode = itemNode as? ItemListItemNode, let tag = itemNode.tag, tag.isEqual(to: focusOnItemTag) {
|
||||
didFocusOnItem = true
|
||||
itemNode.displayHighlight()
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return controller
|
||||
}
|
||||
|
||||
|
||||
Reference in New Issue
Block a user