From fa125d01926a2bc6d3f68f7c6ce4382f9f94ea2a Mon Sep 17 00:00:00 2001 From: Ilya Laktyushin Date: Sun, 17 Jan 2021 10:18:10 +0300 Subject: [PATCH] Various Improvements --- Telegram/Telegram-iOS/Resources/Invite.tgs | Bin 0 -> 29706 bytes Telegram/Telegram-iOS/Resources/PlaneLogo.tgs | Bin 0 -> 2235 bytes .../be.lproj/AppIntentVocabulary.plist | 17 + .../Telegram-iOS/be.lproj/InfoPlist.strings | 12 + .../Telegram-iOS/be.lproj/Localizable.strings | 0 .../Telegram-iOS/en.lproj/Localizable.strings | 73 + .../Telegram-iOS/ru.lproj/InfoPlist.strings | 4 +- .../Sources/ChatListController.swift | 16 + .../Sources/ChatListSearchContainerNode.swift | 21 +- .../ChatListSearchPaneContainerNode.swift | 4 +- .../Sources/Node/ChatListItem.swift | 2 +- .../Sources/Node/ChatListTypingNode.swift | 7 +- ...quenceCountrySelectionControllerNode.swift | 28 +- submodules/DatePickerNode/BUILD | 17 + .../Sources/DatePickerNode.swift | 425 + .../Source/ContextMenuContainerNode.swift | 2 +- .../Display/Source/TooltipController.swift | 37 + .../Source/TooltipControllerNode.swift | 57 +- submodules/InviteLinksUI/BUILD | 56 + .../Sources/InviteLinkEditController.swift | 428 + .../Sources/InviteLinkHeaderItem.swift | 124 + .../Sources/InviteLinkInviteController.swift | 667 ++ .../Sources/InviteLinkInviteHeaderItem.swift | 124 + .../Sources/InviteLinkListController.swift | 619 ++ .../Sources/InviteLinkQRCodeController.swift | 414 + .../Sources/InviteLinkViewController.swift | 766 ++ .../Sources/InviteLinksGridNode.swift | 504 + .../Sources/ItemListDatePickerItem.swift | 241 + .../ItemListInviteLinkDateLimitItem.swift | 409 + .../Sources/ItemListInviteLinkGridItem.swift | 253 + .../ItemListInviteLinkUsageLimitItem.swift | 415 + .../ItemListPermanentInviteLinkItem.swift | 451 + .../Sources/ItemListPeerActionItem.swift | 6 +- .../Items/ItemListSingleLineInputItem.swift | 16 +- .../TGClipboardGalleryMixin.h | 7 +- .../LegacyComponents/TGClipboardMenu.h | 2 +- .../Sources/TGClipboardGalleryMixin.m | 97 +- .../Sources/TGClipboardMenu.m | 13 +- .../Sources/TGClipboardPreviewItemView.h | 9 +- .../Sources/TGClipboardPreviewItemView.m | 9 +- .../TGMediaPickerGalleryInterfaceView.m | 6 +- .../Sources/LegacyAttachmentMenu.swift | 16 +- .../Sources/LegacyMediaPickers.swift | 42 + .../Sources/LocationSectionHeaderItem.swift | 1 - .../MtProtoKit/Sources/MTApiEnvironment.m | 473 +- submodules/PeerInfoUI/BUILD | 1 + .../Sources/ChannelAdminController.swift | 16 +- .../Sources/ChannelMembersController.swift | 3 +- .../ChannelMembersSearchControllerNode.swift | 6 +- .../Sources/ChannelVisibilityController.swift | 374 +- .../Sources/PeersNearbyController.swift | 1 - .../Postbox/Sources/ChatListViewState.swift | 6 +- submodules/Postbox/Sources/Postbox.swift | 6 +- submodules/SectionHeaderItem/BUILD | 20 + .../Sources/SectionHeaderItem.swift | 119 + .../PrivacyAndSecurityController.swift | 2 +- .../Sources/Search/SettingsSearchItem.swift | 20 +- .../Search/SettingsSearchableItems.swift | 23 +- .../Sources/ShareController.swift | 2 + .../StickerPackPreviewControllerNode.swift | 4 +- .../SyncCore/Sources/ExportedInvitation.swift | 12 +- .../Sources/TelegramChatAdminRights.swift | 5 +- submodules/TelegramApi/Sources/Api0.swift | 4 +- submodules/TelegramApi/Sources/Api1.swift | 74 +- submodules/TelegramApi/Sources/Api3.swift | 122 +- .../MediaNavigationAccessoryHeaderNode.swift | 17 +- .../Sources/VoiceChatActionButton.swift | 9 +- .../Sources/VoiceChatController.swift | 46 +- .../Sources/ExportedInvitation.swift | 4 +- .../Sources/InvitationLinks.swift | 108 +- .../ManagedSecretChatOutgoingOperations.swift | 2 +- .../TelegramCore/Sources/PeersNearby.swift | 11 +- .../Sources/TelegramChannel.swift | 4 +- .../Sources/UpdateSecretChat.swift | 2 +- .../Sources/UpdatesApiUtils.swift | 2 +- .../Sources/PresentationStrings.swift | 8397 +++++++++-------- .../Resources/PresentationResourceKey.swift | 2 + .../PresentationResourcesItemList.swift | 12 + .../Images.xcassets/Chat/Contents.json | 6 +- .../Images.xcassets/Chat/Links/Contents.json | 9 + .../Links/Expired.imageset}/Contents.json | 2 +- .../Expired.imageset/ic_linkexpired.pdf} | Bin 3785 -> 5300 bytes .../Chat/Links/Flame.imageset/Contents.json | 12 + .../Links/Flame.imageset/ic_linkfire.pdf} | Bin 3784 -> 3920 bytes .../Chat/Links/Link.imageset/Contents.json | 12 + .../Links/Link.imageset/ic_link.pdf} | Bin 4496 -> 4506 bytes .../Chat/Links/QrLogo.imageset/Contents.json | 12 + .../Links/QrLogo.imageset/ic_qrlogo.pdf} | Bin 4817 -> 4854 bytes .../Images.xcassets/Settings/Contents.json | 6 +- .../Images.xcassets/Wallet/Contents.json | 6 +- .../Contents.json | 12 - .../NavigationSettingsIcon.pdf | Bin 5839 -> 0 bytes .../Wallet/QrGem.imageset/Contents.json | 22 - .../Wallet/QrGem.imageset/QrGem@2x.png | Bin 15212 -> 0 bytes .../Wallet/QrGem.imageset/QrGem@3x.png | Bin 17086 -> 0 bytes .../ReceiveButtonIcon.imageset/Contents.json | 12 - .../Wallet/RefreshIcon.imageset/Contents.json | 12 - .../SendButtonIcon.imageset/Contents.json | 12 - .../TransactionGem.imageset/Contents.json | 12 - .../TransactionGem.imageset/SmallGem.pdf | Bin 96214 -> 0 bytes .../Resources/PresentationStrings.mapping | Bin 154584 -> 156523 bytes .../TelegramUI/Sources/AppDelegate.swift | 13 - .../TelegramUI/Sources/ChatController.swift | 1474 +-- .../ChatInterfaceStateContextMenus.swift | 17 +- .../ChatMessageBubbleContentNode.swift | 4 + .../Sources/ChatMessageBubbleItemNode.swift | 12 + .../Sources/ChatMessageItemView.swift | 4 + .../ChatMessageTextBubbleContentNode.swift | 4 + .../ChatRecentActionsControllerNode.swift | 3 +- .../ChatRecentActionsFilterController.swift | 7 +- .../ChatSearchResultsContollerNode.swift | 6 +- .../ChatStatusChecksTooltipContentNode.swift | 245 + .../Sources/ChatTextInputPanelNode.swift | 4 + submodules/TelegramUI/Sources/OpenUrl.swift | 16 - .../OverlayAudioPlayerControllerNode.swift | 1 + .../Sources/PeerInfo/PeerInfoData.swift | 10 +- .../Sources/PeerInfo/PeerInfoHeaderNode.swift | 4 +- .../Sources/PeerInfo/PeerInfoScreen.swift | 71 +- .../Sources/SharedAccountContext.swift | 4 - .../Sources/GenericEmbedImplementation.swift | 2 +- .../TooltipUI/Sources/TooltipScreen.swift | 15 +- .../UrlHandling/Sources/UrlHandling.swift | 10 - submodules/WebPBinding/Sources/UIImage+WebP.m | 51 +- .../Sources/WebSearchController.swift | 2 +- 124 files changed, 12748 insertions(+), 5704 deletions(-) create mode 100644 Telegram/Telegram-iOS/Resources/Invite.tgs create mode 100644 Telegram/Telegram-iOS/Resources/PlaneLogo.tgs create mode 100644 Telegram/Telegram-iOS/be.lproj/AppIntentVocabulary.plist create mode 100644 Telegram/Telegram-iOS/be.lproj/InfoPlist.strings create mode 100644 Telegram/Telegram-iOS/be.lproj/Localizable.strings create mode 100644 submodules/DatePickerNode/BUILD create mode 100644 submodules/DatePickerNode/Sources/DatePickerNode.swift create mode 100644 submodules/InviteLinksUI/BUILD create mode 100644 submodules/InviteLinksUI/Sources/InviteLinkEditController.swift create mode 100644 submodules/InviteLinksUI/Sources/InviteLinkHeaderItem.swift create mode 100644 submodules/InviteLinksUI/Sources/InviteLinkInviteController.swift create mode 100644 submodules/InviteLinksUI/Sources/InviteLinkInviteHeaderItem.swift create mode 100644 submodules/InviteLinksUI/Sources/InviteLinkListController.swift create mode 100644 submodules/InviteLinksUI/Sources/InviteLinkQRCodeController.swift create mode 100644 submodules/InviteLinksUI/Sources/InviteLinkViewController.swift create mode 100644 submodules/InviteLinksUI/Sources/InviteLinksGridNode.swift create mode 100644 submodules/InviteLinksUI/Sources/ItemListDatePickerItem.swift create mode 100644 submodules/InviteLinksUI/Sources/ItemListInviteLinkDateLimitItem.swift create mode 100644 submodules/InviteLinksUI/Sources/ItemListInviteLinkGridItem.swift create mode 100644 submodules/InviteLinksUI/Sources/ItemListInviteLinkUsageLimitItem.swift create mode 100644 submodules/InviteLinksUI/Sources/ItemListPermanentInviteLinkItem.swift create mode 100644 submodules/SectionHeaderItem/BUILD create mode 100644 submodules/SectionHeaderItem/Sources/SectionHeaderItem.swift create mode 100644 submodules/TelegramUI/Images.xcassets/Chat/Links/Contents.json rename submodules/TelegramUI/Images.xcassets/{Settings/EditAccount.imageset => Chat/Links/Expired.imageset}/Contents.json (75%) rename submodules/TelegramUI/Images.xcassets/{Wallet/SendButtonIcon.imageset/Group.pdf => Chat/Links/Expired.imageset/ic_linkexpired.pdf} (54%) create mode 100644 submodules/TelegramUI/Images.xcassets/Chat/Links/Flame.imageset/Contents.json rename submodules/TelegramUI/Images.xcassets/{Wallet/ReceiveButtonIcon.imageset/Group2.pdf => Chat/Links/Flame.imageset/ic_linkfire.pdf} (72%) create mode 100644 submodules/TelegramUI/Images.xcassets/Chat/Links/Link.imageset/Contents.json rename submodules/TelegramUI/Images.xcassets/{Wallet/RefreshIcon.imageset/ic_walletupdate.pdf => Chat/Links/Link.imageset/ic_link.pdf} (63%) create mode 100644 submodules/TelegramUI/Images.xcassets/Chat/Links/QrLogo.imageset/Contents.json rename submodules/TelegramUI/Images.xcassets/{Settings/EditAccount.imageset/ic_editaccount.pdf => Chat/Links/QrLogo.imageset/ic_qrlogo.pdf} (60%) delete mode 100644 submodules/TelegramUI/Images.xcassets/Wallet/NavigationSettingsIcon.imageset/Contents.json delete mode 100644 submodules/TelegramUI/Images.xcassets/Wallet/NavigationSettingsIcon.imageset/NavigationSettingsIcon.pdf delete mode 100644 submodules/TelegramUI/Images.xcassets/Wallet/QrGem.imageset/Contents.json delete mode 100644 submodules/TelegramUI/Images.xcassets/Wallet/QrGem.imageset/QrGem@2x.png delete mode 100644 submodules/TelegramUI/Images.xcassets/Wallet/QrGem.imageset/QrGem@3x.png delete mode 100644 submodules/TelegramUI/Images.xcassets/Wallet/ReceiveButtonIcon.imageset/Contents.json delete mode 100644 submodules/TelegramUI/Images.xcassets/Wallet/RefreshIcon.imageset/Contents.json delete mode 100644 submodules/TelegramUI/Images.xcassets/Wallet/SendButtonIcon.imageset/Contents.json delete mode 100644 submodules/TelegramUI/Images.xcassets/Wallet/TransactionGem.imageset/Contents.json delete mode 100644 submodules/TelegramUI/Images.xcassets/Wallet/TransactionGem.imageset/SmallGem.pdf create mode 100644 submodules/TelegramUI/Sources/ChatStatusChecksTooltipContentNode.swift diff --git a/Telegram/Telegram-iOS/Resources/Invite.tgs b/Telegram/Telegram-iOS/Resources/Invite.tgs new file mode 100644 index 0000000000000000000000000000000000000000..de9333de4440c30ee700e934d5f1308545b7aa39 GIT binary patch literal 29706 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 J;C;7D007E9Sd;(& literal 0 HcmV?d00001 diff --git a/Telegram/Telegram-iOS/be.lproj/AppIntentVocabulary.plist b/Telegram/Telegram-iOS/be.lproj/AppIntentVocabulary.plist new file mode 100644 index 0000000000..504ece4483 --- /dev/null +++ b/Telegram/Telegram-iOS/be.lproj/AppIntentVocabulary.plist @@ -0,0 +1,17 @@ + + + + + IntentPhrases + + + IntentName + INSendMessageIntent + IntentExamples + + Send a Telegram message to Alex saying I'll be there in 10 minutes + + + + + diff --git a/Telegram/Telegram-iOS/be.lproj/InfoPlist.strings b/Telegram/Telegram-iOS/be.lproj/InfoPlist.strings new file mode 100644 index 0000000000..3f3e41d337 --- /dev/null +++ b/Telegram/Telegram-iOS/be.lproj/InfoPlist.strings @@ -0,0 +1,12 @@ +/* Localized versions of Info.plist keys */ + +"NSContactsUsageDescription" = "Telegram будзе запампоўваць вашы кантакты на свае моцна абароненыя і зашыфраваныя воблачныя серверы, каб вы маглі кантактаваць са сваімі сябрамі з любой вашай прылады."; +"NSLocationWhenInUseUsageDescription" = "Калі вы адпраўляе месцазнаходжанне сваім сябрам, Telegram патрэбны доступ да сэрвісаў геалакацыі, каб размясціць вас на карце."; +"NSLocationAlwaysAndWhenInUseUsageDescription" = "Калі вы трансліруеце ваша месцазнаходжанне сябрам у чаце, Telegram патрэбны доступ у фонавым рэжыме да вашага месцазнаходжання, каб абнаўляць яго падчас трансляцыі."; +"NSLocationAlwaysUsageDescription" = "Калі вы трансліруеце ваша месцазнаходжанне сябрам у чаце, Telegram патрэбны доступ у фонавым рэжыме да вашага месцазнаходжання, каб абнаўляць яго падчас трансляцыі. Гэта таксама неабходна для таго, каб адпраўляць месцазнаходжанне праз Apple Watch."; +"NSCameraUsageDescription" = "Нам неабходны гэты дазвол, каб вы маглі рабіць і адпраўляць фота і відэа, а таксама відэавыклікі."; +"NSPhotoLibraryUsageDescription" = "Нам неабходны гэты дазвол, каб вы маглі абагульваць фота і відэа са сваёй галерэі."; +"NSPhotoLibraryAddUsageDescription" = "Нам неабходны гэты дазвол, каб вы маглі захоўваць фота і відэа ў сваю галерэю."; +"NSMicrophoneUsageDescription" = "Нам неабходны гэты дазвол, каб вы маглі запісваць і абагульваць галасавыя паведамленні і відэа з гукам."; +"NSSiriUsageDescription" = "Вы можаце адпраўляць паведамленні праз Siri."; +"NSFaceIDUsageDescription" = "Вы можаце карыстацца Face ID для разблакіравання праграмы."; diff --git a/Telegram/Telegram-iOS/be.lproj/Localizable.strings b/Telegram/Telegram-iOS/be.lproj/Localizable.strings new file mode 100644 index 0000000000..e69de29bb2 diff --git a/Telegram/Telegram-iOS/en.lproj/Localizable.strings b/Telegram/Telegram-iOS/en.lproj/Localizable.strings index 372e459d11..6ed4fae02d 100644 --- a/Telegram/Telegram-iOS/en.lproj/Localizable.strings +++ b/Telegram/Telegram-iOS/en.lproj/Localizable.strings @@ -5816,3 +5816,76 @@ Sorry for the inconvenience."; "PeerInfo.ButtonVoiceChat" = "Voice Chat"; "VoiceChat.OpenChat" = "Open Chat"; + +"GroupInfo.InviteLinks" = "Invite Links"; + +"InviteLink.Title" = "Invite Links"; +"InviteLink.PermanentLink" = "Permanent Link"; +"InviteLink.Share" = "Share Link"; +"InviteLink.PeopleJoinedNone" = "no one joined yet"; +"InviteLink.PeopleJoined_1" = "%@ people joined"; +"InviteLink.PeopleJoined_2" = "%@ people joined"; +"InviteLink.PeopleJoined_3_10" = "%@ people joined"; +"InviteLink.PeopleJoined_many" = "%@ people joined"; +"InviteLink.PeopleJoined_any" = "%@ people joined"; +"InviteLink.CreatePrivateLinkHelp" = "Anyone who has Telegram installed will be able to join your group by following this link."; +"InviteLink.Manage" = "Manage Invite Links"; + +"InviteLink.PeopleJoinedShortNoneExpired" = "no one joined"; +"InviteLink.PeopleJoinedShortNone" = "no one joined yet"; +"InviteLink.PeopleJoinedShort_1" = "%@ joined"; +"InviteLink.PeopleJoinedShort_2" = "%@ joined"; +"InviteLink.PeopleJoinedShort_3_10" = "%@ joined"; +"InviteLink.PeopleJoinedShort_many" = "%@ joined"; +"InviteLink.PeopleJoinedShort_any" = "%@ joined"; + +"InviteLink.Expired" = "expired"; +"InviteLink.UsageLimitReached" = "limit reached"; +"InviteLink.Revoked" = "revoked"; + +"InviteLink.AdditionalLinks" = "Additional Links"; +"InviteLink.Create" = "Create a New Link"; +"InviteLink.CreateInfo" = "You can create additional invite links that have limited time or number of usages."; + +"InviteLink.RevokedLinks" = "Revoked Links"; +"InviteLink.DeleteAllRevokedLinks" = "Delete All Revoked Links"; + +"InviteLink.ContextCopy" = "Copy"; +"InviteLink.ContextEdit" = "Edit"; +"InviteLink.ContextGetQRCode" = "Get QR Code"; +"InviteLink.ContextShare" = "Share"; +"InviteLink.ContextRevoke" = "Revoke"; +"InviteLink.ContextDelete" = "Delete"; + +"InviteLink.Create.Title" = "New Link"; +"InviteLink.Create.EditTitle" = "Edit Link"; + +"InviteLink.Create.TimeLimit" = "Limit By Time Period"; +"InviteLink.Create.TimeLimitExpiryDate" = "Expiry Date"; +"InviteLink.Create.TimeLimitExpiryDateNever" = "Never"; +"InviteLink.Create.TimeLimitExpiryTime" = "Time"; +"InviteLink.Create.TimeLimitInfo" = "You can make the link expire after a certain time."; +"InviteLink.Create.TimeLimitNoLimit" = "No Limit"; + +"InviteLink.Create.UsersLimit" = "Limit By Number Of Users"; +"InviteLink.Create.UsersLimitNumberOfUsers" = "Number of Uses"; +"InviteLink.Create.UsersLimitNumberOfUsersUnlimited" = "Unlimited"; +"InviteLink.Create.UsersLimitInfo" = "You can make the link expire after it has been used for a certain number of times."; +"InviteLink.Create.UsersLimitNoLimit" = "No Limit"; + +"InviteLink.Create.Revoke" = "Revoke Link"; + +"InviteLink.QRCode.Title" = "Invite by QR Code"; +"InviteLink.QRCode.Info" = "Everyone on Telegram can scan this code to join your group."; +"InviteLink.QRCode.Share" = "Share QR Code"; + +"InviteLink.InviteLink" = "Invite Link"; +"InviteLink.CreatedBy" = "Link Created By"; + +"InviteLink.DeleteAllRevokedLinksAlert.Text" = "This will delete all revoked links"; +"InviteLink.DeleteAllRevokedLinksAlert.Action" = "Delete"; + +"Conversation.ChecksTooltip.Delivered" = "Delivered"; +"Conversation.ChecksTooltip.Read" = "Read"; + +"DialogList.MultipleTypingPair" = "%@ and %@ are typing"; diff --git a/Telegram/Telegram-iOS/ru.lproj/InfoPlist.strings b/Telegram/Telegram-iOS/ru.lproj/InfoPlist.strings index f4b64e4983..08c349a3c3 100644 --- a/Telegram/Telegram-iOS/ru.lproj/InfoPlist.strings +++ b/Telegram/Telegram-iOS/ru.lproj/InfoPlist.strings @@ -2,11 +2,11 @@ "NSContactsUsageDescription" = "Актуальная информация о ваших контактах будет храниться зашифрованной в облаке Telegram, чтобы вы могли связаться с друзьями с любого устройства."; "NSLocationWhenInUseUsageDescription" = "Когда вы отправляете друзьям геопозицию, Telegram нужно разрешение, чтобы показать им карту."; -"NSLocationAlwaysAndWhenInUseUsageDescription" = "Фоновый доступ к геопозиции требуется, чтобы обновлять вашу геопозицию, когда вы транслируете её в чат с друзьями. "; +"NSLocationAlwaysAndWhenInUseUsageDescription" = "Фоновый доступ к геопозиции требуется, чтобы обновлять вашу геопозицию, когда вы транслируете её в чат с друзьями."; "NSLocationAlwaysUsageDescription" = "Фоновый доступ к геопозиции требуется, чтобы обновлять вашу геопозицию, когда вы транслируете её в чат с друзьями. Он также необходим для отправки геопозиции с Apple Watch."; "NSCameraUsageDescription" = "Это необходимо, чтобы вы могли делиться снятыми фотографиями и видео."; "NSPhotoLibraryUsageDescription" = "Это необходимо, чтобы вы могли делиться фото и видео из библиотеки устройства."; "NSPhotoLibraryAddUsageDescription" = "Это необходимо, чтобы вы могли сохранять фото и видео в библиотеку устройства."; "NSMicrophoneUsageDescription" = "Это необходимо, чтобы вы могли делиться голосовыми сообщениями и видео со звуком."; -"NSSiriUsageDescription" = "Вы можете использовать Siri для отправки сообщений"; +"NSSiriUsageDescription" = "Вы можете использовать Siri для отправки сообщений."; "NSFaceIDUsageDescription" = "Вы можете разблокировать приложение с помощью Face ID."; diff --git a/submodules/ChatListUI/Sources/ChatListController.swift b/submodules/ChatListUI/Sources/ChatListController.swift index 37939de157..ef900c946e 100644 --- a/submodules/ChatListUI/Sources/ChatListController.swift +++ b/submodules/ChatListUI/Sources/ChatListController.swift @@ -1970,6 +1970,15 @@ public class ChatListControllerImpl: TelegramBaseController, ChatListController } let _ = (signal |> deliverOnMainQueue).start() + + strongSelf.chatListDisplayNode.containerNode.updateState({ state in + var state = state + for peerId in peerIds { + state.selectedPeerIds.remove(peerId) + } + return state + }) + return true } else if value == .undo { strongSelf.chatListDisplayNode.containerNode.currentItemNode.setCurrentRemovingPeerId(peerIds.first!) @@ -2524,6 +2533,13 @@ public class ChatListControllerImpl: TelegramBaseController, ChatListController deleteSendMessageIntents(peerId: peerId) }) + + strongSelf.chatListDisplayNode.containerNode.updateState({ state in + var state = state + state.selectedPeerIds.remove(peerId) + return state + }) + completion() return true } else if value == .undo { diff --git a/submodules/ChatListUI/Sources/ChatListSearchContainerNode.swift b/submodules/ChatListUI/Sources/ChatListSearchContainerNode.swift index b94b86b190..eb554c3a8a 100644 --- a/submodules/ChatListUI/Sources/ChatListSearchContainerNode.swift +++ b/submodules/ChatListUI/Sources/ChatListSearchContainerNode.swift @@ -684,7 +684,6 @@ public final class ChatListSearchContainerNode: SearchDisplayControllerContentNo let items = context.sharedContext.chatAvailableMessageActions(postbox: context.account.postbox, accountPeerId: context.account.peerId, messageIds: [message.id], messages: messages, peers: peers) |> map { actions -> [ContextMenuItem] in var items: [ContextMenuItem] = [] - if let linkForCopying = linkForCopying { items.append(.action(ContextMenuActionItem(text: self.presentationData.strings.Conversation_ContextMenuCopyLink, icon: { theme in generateTintedImage(image: UIImage(bundleImageName: "Chat/Context Menu/Copy"), color: theme.contextMenu.primaryColor) }, action: { c, _ in @@ -948,17 +947,25 @@ public final class ChatListSearchContainerNode: SearchDisplayControllerContentNo }) }) |> deliverOnMainQueue).start(completed: { if let strongSelf = self { -// strongSelf.headerNode.navigationButtonContainer.performAction?(.selectionDone) - let controller = strongSelf.context.sharedContext.makeChatController(context: strongSelf.context, chatLocation: .peer(peerId), subject: nil, botStart: nil, mode: .standard(previewing: false)) controller.purposefulAction = { [weak self] in self?.cancel?() } - strongSelf.navigationController?.pushViewController(controller, animated: false, completion: { - if let peerSelectionController = peerSelectionController { - peerSelectionController.dismiss() + + if let navigationController = strongSelf.navigationController, let peerSelectionControllerIndex = navigationController.viewControllers.firstIndex(where: { $0 is PeerSelectionController }) { + var viewControllers = navigationController.viewControllers + viewControllers.insert(controller, at: peerSelectionControllerIndex) + navigationController.setViewControllers(viewControllers, animated: false) + Queue.mainQueue().after(0.2) { + peerSelectionController?.dismiss() } - }) + } else { + strongSelf.navigationController?.pushViewController(controller, animated: false, completion: { + if let peerSelectionController = peerSelectionController { + peerSelectionController.dismiss() + } + }) + } strongSelf.updateState { state in return state.withUpdatedSelectedMessageIds(nil) diff --git a/submodules/ChatListUI/Sources/ChatListSearchPaneContainerNode.swift b/submodules/ChatListUI/Sources/ChatListSearchPaneContainerNode.swift index 99712212d2..4c2805501b 100644 --- a/submodules/ChatListUI/Sources/ChatListSearchPaneContainerNode.swift +++ b/submodules/ChatListUI/Sources/ChatListSearchPaneContainerNode.swift @@ -129,9 +129,7 @@ final class ChatListSearchPaneContainerNode: ASDisplayNode, UIGestureRecognizerD private var pendingPanes: [ChatListSearchPaneKey: ChatListSearchPendingPane] = [:] private var transitionFraction: CGFloat = 0.0 - - var openPeerContextAction: ((Peer, ASDisplayNode, ContextGesture?) -> Void)? - + var currentPaneUpdated: ((ChatListSearchPaneKey?, CGFloat, ContainedViewLayoutTransition) -> Void)? var requestExpandTabs: (() -> Bool)? diff --git a/submodules/ChatListUI/Sources/Node/ChatListItem.swift b/submodules/ChatListUI/Sources/Node/ChatListItem.swift index c2bb76eed1..3258162d7a 100644 --- a/submodules/ChatListUI/Sources/Node/ChatListItem.swift +++ b/submodules/ChatListUI/Sources/Node/ChatListItem.swift @@ -1354,7 +1354,7 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { switch item.content { case let .peer(_, renderedPeer, _, _, presence, _ ,_ ,_, _, _, displayAsMessage, _): if !displayAsMessage { - if let peer = renderedPeer.peer as? TelegramUser, let presence = presence as? TelegramUserPresence, !isServicePeer(peer) && !peer.flags.contains(.isSupport) && peer.id != item.context.account.peerId { + if let peer = renderedPeer.chatMainPeer as? TelegramUser, let presence = presence as? TelegramUserPresence, !isServicePeer(peer) && !peer.flags.contains(.isSupport) && peer.id != item.context.account.peerId { let updatedPresence = TelegramUserPresence(status: presence.status, lastActivity: 0) let relativeStatus = relativeUserPresenceStatus(updatedPresence, relativeTo: timestamp) if case .online = relativeStatus { diff --git a/submodules/ChatListUI/Sources/Node/ChatListTypingNode.swift b/submodules/ChatListUI/Sources/Node/ChatListTypingNode.swift index ca664a419d..711821cf45 100644 --- a/submodules/ChatListUI/Sources/Node/ChatListTypingNode.swift +++ b/submodules/ChatListUI/Sources/Node/ChatListTypingNode.swift @@ -127,7 +127,12 @@ final class ChatListInputActivitiesNode: ASDisplayNode { let string: NSAttributedString if activities.count > 1 { let peerTitle = activities[0].0.compactDisplayTitle - string = NSAttributedString(string: strings.DialogList_MultipleTyping(peerTitle, strings.DialogList_MultipleTypingSuffix(activities.count - 1).0).0, font: textFont, textColor: color) + if activities.count == 2 { + let secondPeerTitle = activities[1].0.compactDisplayTitle + string = NSAttributedString(string: strings.DialogList_MultipleTypingPair(peerTitle, secondPeerTitle).0, font: textFont, textColor: color) + } else { + string = NSAttributedString(string: strings.DialogList_MultipleTyping(peerTitle, strings.DialogList_MultipleTypingSuffix(activities.count - 1).0).0, font: textFont, textColor: color) + } } else { string = NSAttributedString(string: strings.DialogList_MultipleTypingSuffix(activities.count).0, font: textFont, textColor: color) } diff --git a/submodules/CountrySelectionUI/Sources/AuthorizationSequenceCountrySelectionControllerNode.swift b/submodules/CountrySelectionUI/Sources/AuthorizationSequenceCountrySelectionControllerNode.swift index 82b9b60a54..b429fc0be1 100644 --- a/submodules/CountrySelectionUI/Sources/AuthorizationSequenceCountrySelectionControllerNode.swift +++ b/submodules/CountrySelectionUI/Sources/AuthorizationSequenceCountrySelectionControllerNode.swift @@ -58,15 +58,21 @@ private func loadCountryCodes() -> [(String, Int)] { private let countryCodes: [(String, Int)] = loadCountryCodes() -func localizedCountryNamesAndCodes(strings: PresentationStrings) -> [((String, String), String, Int)] { +func localizedCountryNamesAndCodes(strings: PresentationStrings) -> [((String, String), String, [Int])] { let locale = localeWithStrings(strings) - var result: [((String, String), String, Int)] = [] + var result: [((String, String), String, [Int])] = [] for country in AuthorizationSequenceCountrySelectionController.countries() { if country.hidden { continue } - if let englishCountryName = usEnglishLocale.localizedString(forRegionCode: country.id), let countryName = locale.localizedString(forRegionCode: country.id), let codeValue = country.countryCodes.first?.code, let code = Int(codeValue) { - result.append(((englishCountryName, countryName), country.id, code)) + if let englishCountryName = usEnglishLocale.localizedString(forRegionCode: country.id), let countryName = locale.localizedString(forRegionCode: country.id) { + var codes: [Int] = [] + for codeValue in country.countryCodes { + if let code = Int(codeValue.code) { + codes.append(code) + } + } + result.append(((englishCountryName, countryName), country.id, codes)) } else { assertionFailure() } @@ -128,7 +134,7 @@ private func matchStringTokens(_ tokens: [ValueBoxKey], with other: [ValueBoxKey return false } -private func searchCountries(items: [((String, String), String, Int)], query: String) -> [((String, String), String, Int)] { +private func searchCountries(items: [((String, String), String, [Int])], query: String) -> [((String, String), String, Int)] { let queryTokens = stringTokens(query.lowercased()) var result: [((String, String), String, Int)] = [] @@ -136,7 +142,9 @@ private func searchCountries(items: [((String, String), String, Int)], query: St let string = "\(item.0) \(item.1)" let tokens = stringTokens(string) if matchStringTokens(tokens, with: queryTokens) { - result.append(item) + for code in item.2 { + result.append((item.0, item.1, code)) + } } } @@ -158,7 +166,7 @@ final class AuthorizationSequenceCountrySelectionControllerNode: ASDisplayNode, private let sectionTitles: [String] private var searchResults: [((String, String), String, Int)] = [] - private let countryNamesAndCodes: [((String, String), String, Int)] + private let countryNamesAndCodes: [((String, String), String, [Int])] init(theme: PresentationTheme, strings: PresentationStrings, displayCodes: Bool, itemSelected: @escaping (((String, String), String, Int)) -> Void) { self.theme = theme @@ -181,14 +189,16 @@ final class AuthorizationSequenceCountrySelectionControllerNode: ASDisplayNode, self.countryNamesAndCodes = countryNamesAndCodes var sections: [(String, [((String, String), String, Int)])] = [] - for (names, id, code) in countryNamesAndCodes.sorted(by: { lhs, rhs in + for (names, id, codes) in countryNamesAndCodes.sorted(by: { lhs, rhs in return lhs.0.1 < rhs.0.1 }) { let title = String(names.1[names.1.startIndex ..< names.1.index(after: names.1.startIndex)]).uppercased() if sections.isEmpty || sections[sections.count - 1].0 != title { sections.append((title, [])) } - sections[sections.count - 1].1.append((names, id, code)) + for code in codes { + sections[sections.count - 1].1.append((names, id, code)) + } } self.sections = sections self.sectionTitles = sections.map { $0.0 } diff --git a/submodules/DatePickerNode/BUILD b/submodules/DatePickerNode/BUILD new file mode 100644 index 0000000000..f95d63dc3f --- /dev/null +++ b/submodules/DatePickerNode/BUILD @@ -0,0 +1,17 @@ +load("@build_bazel_rules_swift//swift:swift.bzl", "swift_library") + +swift_library( + name = "DatePickerNode", + module_name = "DatePickerNode", + srcs = glob([ + "Sources/**/*.swift", + ]), + deps = [ + "//submodules/AsyncDisplayKit:AsyncDisplayKit", + "//submodules/Display:Display", + "//submodules/TelegramPresentationData:TelegramPresentationData", + ], + visibility = [ + "//visibility:public", + ], +) diff --git a/submodules/DatePickerNode/Sources/DatePickerNode.swift b/submodules/DatePickerNode/Sources/DatePickerNode.swift new file mode 100644 index 0000000000..0a36621dbc --- /dev/null +++ b/submodules/DatePickerNode/Sources/DatePickerNode.swift @@ -0,0 +1,425 @@ +//import Foundation +//import Display +//import UIKit +//import AsyncDisplayKit +//import TelegramPresentationData +// +//private let textFont = Font.regular(13.0) +//private let selectedTextFont = Font.bold(13.0) +// +//public final class DatePickerTheme: Equatable { +// public let backgroundColor: UIColor +// public let textColor: UIColor +// public let secondaryTextColor: UIColor +// public let accentColor: UIColor +// public let selectionColor: UIColor +// public let selectedCurrentTextColor: UIColor +// public let secondarySelectionColor: UIColor +// +// public init(backgroundColor: UIColor, textColor: UIColor, secondaryTextColor: UIColor, accentColor: UIColor, selectionColor: UIColor, selectedCurrentTextColor: UIColor, secondarySelectionColor: UIColor) { +// self.backgroundColor = backgroundColor +// self.textColor = textColor +// self.secondaryTextColor = secondaryTextColor +// self.accentColor = accentColor +// self.selectionColor = selectionColor +// self.selectedCurrentTextColor = selectedCurrentTextColor +// self.secondarySelectionColor = secondarySelectionColor +// } +// +// public static func ==(lhs: DatePickerTheme, rhs: DatePickerTheme) -> Bool { +// if lhs.backgroundColor != rhs.backgroundColor { +// return false +// } +// if lhs.textColor != rhs.textColor { +// return false +// } +// if lhs.secondaryTextColor != rhs.secondaryTextColor { +// return false +// } +// if lhs.accentColor != rhs.accentColor { +// return false +// } +// if lhs.selectionColor != rhs.selectionColor { +// return false +// } +// if lhs.selectedCurrentTextColor != rhs.selectedCurrentTextColor { +// return false +// } +// if lhs.secondarySelectionColor != rhs.secondarySelectionColor { +// return false +// } +// return true +// } +//} +// +//public extension DatePickerTheme { +// convenience init(theme: PresentationTheme) { +// self.init(backgroundColor: theme.rootController.navigationBar.segmentedBackgroundColor, foregroundColor: theme.rootController.navigationBar.segmentedForegroundColor, shadowColor: .black, textColor: theme.rootController.navigationBar.segmentedTextColor, dividerColor: theme.rootController.navigationBar.segmentedDividerColor) +// } +//} +// +//private class SegmentedControlItemNode: HighlightTrackingButtonNode { +//} +// +//private let telegramReleaseDate = Date(timeIntervalSince1970: 1376438400.0) +// +//public final class DatePickerNode: ASDisplayNode, UIGestureRecognizerDelegate { +// private var theme: DatePickerTheme +// private var _items: [SegmentedControlItem] +// private var _selectedIndex: Int = 0 +// +// private var validLayout: SegmentedControlLayout? +// +// private let selectionNode: ASImageNode +// private var itemNodes: [SegmentedControlItemNode] +// private var dividerNodes: [ASDisplayNode] +// +// private var gestureRecognizer: UIPanGestureRecognizer? +// private var gestureSelectedIndex: Int? +// +// public var maximumDate: Date? { +// didSet { +// +// } +// } +// public var minimumDate: Date = telegramReleaseDate { +// didSet { +// +// } +// } +// public var date: Date = Date() { +// didSet { +// +// } +// } +// +// +// public var items: [SegmentedControlItem] { +// get { +// return self._items +// } +// set { +// let previousItems = self._items +// self._items = newValue +// guard previousItems != newValue else { +// return +// } +// +// self.itemNodes.forEach { $0.removeFromSupernode() } +// self.itemNodes = self._items.map { item in +// let itemNode = SegmentedControlItemNode() +// itemNode.contentEdgeInsets = UIEdgeInsets(top: 0.0, left: 8.0, bottom: 0.0, right: 8.0) +// itemNode.titleNode.maximumNumberOfLines = 1 +// itemNode.titleNode.truncationMode = .byTruncatingTail +// itemNode.setTitle(item.title, with: textFont, with: self.theme.textColor, for: .normal) +// itemNode.setTitle(item.title, with: selectedTextFont, with: self.theme.textColor, for: .selected) +// itemNode.setTitle(item.title, with: selectedTextFont, with: self.theme.textColor, for: [.selected, .highlighted]) +// return itemNode +// } +// self.setupButtons() +// self.itemNodes.forEach(self.addSubnode(_:)) +// +// let dividersCount = self._items.count > 2 ? self._items.count - 1 : 0 +// if self.dividerNodes.count != dividersCount { +// self.dividerNodes.forEach { $0.removeFromSupernode() } +// self.dividerNodes = (0 ..< dividersCount).map { _ in ASDisplayNode() } +// } +// +// if let layout = self.validLayout { +// let _ = self.updateLayout(layout, transition: .immediate) +// } +// } +// } +// +// public var selectedIndex: Int { +// get { +// return self._selectedIndex +// } +// set { +// guard newValue != self._selectedIndex else { +// return +// } +// self._selectedIndex = newValue +// if let layout = self.validLayout { +// let _ = self.updateLayout(layout, transition: .immediate) +// } +// } +// } +// +// public func setSelectedIndex(_ index: Int, animated: Bool) { +// guard index != self._selectedIndex else { +// return +// } +// self._selectedIndex = index +// if let layout = self.validLayout { +// let _ = self.updateLayout(layout, transition: .animated(duration: 0.2, curve: .easeInOut)) +// } +// } +// +// public var selectedIndexChanged: (Int) -> Void = { _ in } +// public var selectedIndexShouldChange: (Int, @escaping (Bool) -> Void) -> Void = { _, f in +// f(true) +// } +// +// public init(theme: SegmentedControlTheme, items: [SegmentedControlItem], selectedIndex: Int) { +// self.theme = theme +// self._items = items +// self._selectedIndex = selectedIndex +// +// self.selectionNode = ASImageNode() +// self.selectionNode.displaysAsynchronously = false +// self.selectionNode.displayWithoutProcessing = true +// +// self.itemNodes = items.map { item in +// let itemNode = SegmentedControlItemNode() +// itemNode.contentEdgeInsets = UIEdgeInsets(top: 0.0, left: 8.0, bottom: 0.0, right: 8.0) +// itemNode.titleNode.maximumNumberOfLines = 1 +// itemNode.titleNode.truncationMode = .byTruncatingTail +// itemNode.setTitle(item.title, with: textFont, with: theme.textColor, for: .normal) +// itemNode.setTitle(item.title, with: selectedTextFont, with: theme.textColor, for: .selected) +// itemNode.setTitle(item.title, with: selectedTextFont, with: theme.textColor, for: [.selected, .highlighted]) +// return itemNode +// } +// +// let dividersCount = items.count > 2 ? items.count - 1 : 0 +// self.dividerNodes = (0 ..< dividersCount).map { _ in +// let node = ASDisplayNode() +// node.backgroundColor = theme.dividerColor +// return node +// } +// +// super.init() +// +// self.clipsToBounds = true +// self.cornerRadius = 9.0 +// +// self.addSubnode(self.selectionNode) +// self.itemNodes.forEach(self.addSubnode(_:)) +// self.setupButtons() +// self.dividerNodes.forEach(self.addSubnode(_:)) +// +// self.backgroundColor = self.theme.backgroundColor +// self.selectionNode.image = generateSelectionImage(theme: self.theme) +// } +// +// override public func didLoad() { +// super.didLoad() +// +// self.view.disablesInteractiveTransitionGestureRecognizer = true +// +// let gestureRecognizer = UIPanGestureRecognizer(target: self, action: #selector(self.panGesture(_:))) +// gestureRecognizer.delegate = self +// self.view.addGestureRecognizer(gestureRecognizer) +// self.gestureRecognizer = gestureRecognizer +// } +// +// private func setupButtons() { +// for i in 0 ..< self.itemNodes.count { +// let itemNode = self.itemNodes[i] +// itemNode.addTarget(self, action: #selector(self.buttonPressed(_:)), forControlEvents: .touchUpInside) +// itemNode.highligthedChanged = { [weak self, weak itemNode] highlighted in +// if let strongSelf = self, let itemNode = itemNode { +// let transition = ContainedViewLayoutTransition.animated(duration: 0.25, curve: .easeInOut) +// if strongSelf.selectedIndex == i { +// if let gestureRecognizer = strongSelf.gestureRecognizer, case .began = gestureRecognizer.state { +// } else { +// strongSelf.updateButtonsHighlights(highlightedIndex: highlighted ? i : nil, gestureSelectedIndex: strongSelf.gestureSelectedIndex) +// } +// } else if highlighted { +// transition.updateAlpha(node: itemNode, alpha: 0.4) +// } +// if !highlighted { +// transition.updateAlpha(node: itemNode, alpha: 1.0) +// } +// } +// } +// } +// } +// +// private func updateButtonsHighlights(highlightedIndex: Int?, gestureSelectedIndex: Int?) { +// let transition = ContainedViewLayoutTransition.animated(duration: 0.25, curve: .easeInOut) +// if highlightedIndex == nil && gestureSelectedIndex == nil { +// transition.updateTransformScale(node: self.selectionNode, scale: 1.0) +// } else { +// transition.updateTransformScale(node: self.selectionNode, scale: 0.92) +// } +// for i in 0 ..< self.itemNodes.count { +// let itemNode = self.itemNodes[i] +// if i == highlightedIndex || i == gestureSelectedIndex { +// transition.updateTransformScale(node: itemNode, scale: 0.92) +// } else { +// transition.updateTransformScale(node: itemNode, scale: 1.0) +// } +// } +// } +// +// private func updateButtonsHighlights() { +// let transition = ContainedViewLayoutTransition.animated(duration: 0.25, curve: .easeInOut) +// if let gestureSelectedIndex = self.gestureSelectedIndex { +// for i in 0 ..< self.itemNodes.count { +// let itemNode = self.itemNodes[i] +// transition.updateTransformScale(node: itemNode, scale: i == gestureSelectedIndex ? 0.92 : 1.0) +// } +// } else { +// for itemNode in self.itemNodes { +// transition.updateTransformScale(node: itemNode, scale: 1.0) +// } +// } +// } +// +// public func updateTheme(_ theme: SegmentedControlTheme) { +// guard theme != self.theme else { +// return +// } +// self.theme = theme +// +// self.backgroundColor = self.theme.backgroundColor +// self.selectionNode.image = generateSelectionImage(theme: self.theme) +// +// for itemNode in self.itemNodes { +// if let title = itemNode.attributedTitle(for: .normal)?.string { +// itemNode.setTitle(title, with: textFont, with: self.theme.textColor, for: .normal) +// itemNode.setTitle(title, with: selectedTextFont, with: self.theme.textColor, for: .selected) +// itemNode.setTitle(title, with: selectedTextFont, with: self.theme.textColor, for: [.selected, .highlighted]) +// } +// } +// +// for dividerNode in self.dividerNodes { +// dividerNode.backgroundColor = theme.dividerColor +// } +// } +// +// public func updateLayout(_ layout: SegmentedControlLayout, transition: ContainedViewLayoutTransition) -> CGSize { +// self.validLayout = layout +// +// let calculatedWidth: CGFloat = 0.0 +// +// let width: CGFloat +// switch layout { +// case let .stretchToFill(targetWidth): +// width = targetWidth +// case let .sizeToFit(maximumWidth, minimumWidth): +// width = max(minimumWidth, min(maximumWidth, calculatedWidth)) +// } +// +// let selectedIndex: Int +// if let gestureSelectedIndex = self.gestureSelectedIndex { +// selectedIndex = gestureSelectedIndex +// } else { +// selectedIndex = self.selectedIndex +// } +// +// let size = CGSize(width: width, height: 32.0) +// if !self.itemNodes.isEmpty { +// let itemSize = CGSize(width: floorToScreenPixels(size.width / CGFloat(self.itemNodes.count)), height: size.height) +// +// transition.updateBounds(node: self.selectionNode, bounds: CGRect(origin: CGPoint(), size: itemSize)) +// transition.updatePosition(node: self.selectionNode, position: CGPoint(x: itemSize.width / 2.0 + itemSize.width * CGFloat(selectedIndex), y: size.height / 2.0)) +// +// for i in 0 ..< self.itemNodes.count { +// let itemNode = self.itemNodes[i] +// let _ = itemNode.measure(itemSize) +// transition.updateFrame(node: itemNode, frame: CGRect(origin: CGPoint(x: itemSize.width * CGFloat(i), y: (size.height - itemSize.height) / 2.0), size: itemSize)) +// +// let isSelected = selectedIndex == i +// if itemNode.isSelected != isSelected { +// if case .animated = transition { +// UIView.transition(with: itemNode.view, duration: 0.2, options: .transitionCrossDissolve, animations: { +// itemNode.isSelected = isSelected +// }, completion: nil) +// } else { +// itemNode.isSelected = isSelected +// } +// if isSelected { +// itemNode.accessibilityTraits.insert(.selected) +// } else { +// itemNode.accessibilityTraits.remove(.selected) +// } +// } +// } +// } +// +// if !self.dividerNodes.isEmpty { +// let dividerSize = CGSize(width: 1.0, height: 16.0) +// let delta: CGFloat = size.width / CGFloat(self.dividerNodes.count + 1) +// for i in 0 ..< self.dividerNodes.count { +// let dividerNode = self.dividerNodes[i] +// transition.updateFrame(node: dividerNode, frame: CGRect(origin: CGPoint(x: floorToScreenPixels(delta * CGFloat(i + 1) - dividerSize.width / 2.0), y: (size.height - dividerSize.height) / 2.0), size: dividerSize)) +// +// let dividerAlpha: CGFloat +// if (selectedIndex - 1 ... selectedIndex).contains(i) { +// dividerAlpha = 0.0 +// } else { +// dividerAlpha = 1.0 +// } +// transition.updateAlpha(node: dividerNode, alpha: dividerAlpha) +// } +// } +// +// return size +// } +// +// @objc private func buttonPressed(_ button: SegmentedControlItemNode) { +// guard let index = self.itemNodes.firstIndex(of: button) else { +// return +// } +// +// self.selectedIndexShouldChange(index, { [weak self] commit in +// if let strongSelf = self, commit { +// strongSelf._selectedIndex = index +// strongSelf.selectedIndexChanged(index) +// if let layout = strongSelf.validLayout { +// let _ = strongSelf.updateLayout(layout, transition: .animated(duration: 0.2, curve: .slide)) +// } +// } +// }) +// } +// +// public override func gestureRecognizerShouldBegin(_ gestureRecognizer: UIGestureRecognizer) -> Bool { +// return self.selectionNode.frame.contains(gestureRecognizer.location(in: self.view)) +// } +// +// @objc private func panGesture(_ recognizer: UIPanGestureRecognizer) { +// let location = recognizer.location(in: self.view) +// switch recognizer.state { +// case .changed: +// if !self.selectionNode.frame.contains(location) { +// let point = CGPoint(x: max(0.0, min(self.bounds.width, location.x)), y: 1.0) +// for i in 0 ..< self.itemNodes.count { +// let itemNode = self.itemNodes[i] +// if itemNode.frame.contains(point) { +// if i != self.gestureSelectedIndex { +// self.gestureSelectedIndex = i +// self.updateButtonsHighlights(highlightedIndex: nil, gestureSelectedIndex: i) +// if let layout = self.validLayout { +// let _ = self.updateLayout(layout, transition: .animated(duration: 0.35, curve: .slide)) +// } +// } +// break +// } +// } +// } +// case .ended: +// if let gestureSelectedIndex = self.gestureSelectedIndex { +// if gestureSelectedIndex != self.selectedIndex { +// self.selectedIndexShouldChange(gestureSelectedIndex, { [weak self] commit in +// if let strongSelf = self { +// if commit { +// strongSelf._selectedIndex = gestureSelectedIndex +// strongSelf.selectedIndexChanged(gestureSelectedIndex) +// } else { +// if let layout = strongSelf.validLayout { +// let _ = strongSelf.updateLayout(layout, transition: .animated(duration: 0.2, curve: .slide)) +// } +// } +// } +// }) +// } +// self.gestureSelectedIndex = nil +// } +// self.updateButtonsHighlights(highlightedIndex: nil, gestureSelectedIndex: nil) +// default: +// break +// } +// } +//} diff --git a/submodules/Display/Source/ContextMenuContainerNode.swift b/submodules/Display/Source/ContextMenuContainerNode.swift index aea6ae2062..ecbbabef41 100644 --- a/submodules/Display/Source/ContextMenuContainerNode.swift +++ b/submodules/Display/Source/ContextMenuContainerNode.swift @@ -51,7 +51,7 @@ public final class ContextMenuContainerNode: ASDisplayNode { let maskParams = CachedMaskParams(size: self.bounds.size, relativeArrowPosition: self.relativeArrowPosition?.0 ?? self.bounds.size.width / 2.0, arrowOnBottom: self.relativeArrowPosition?.1 ?? true) if self.cachedMaskParams != maskParams { let path = UIBezierPath() - let cornerRadius: CGFloat = 6.0 + let cornerRadius: CGFloat = 10.0 let verticalInset: CGFloat = 9.0 let arrowWidth: CGFloat = 18.0 let requestedArrowPosition = maskParams.relativeArrowPosition diff --git a/submodules/Display/Source/TooltipController.swift b/submodules/Display/Source/TooltipController.swift index 34c320883d..c9ee3dcf3a 100644 --- a/submodules/Display/Source/TooltipController.swift +++ b/submodules/Display/Source/TooltipController.swift @@ -3,10 +3,16 @@ import UIKit import AsyncDisplayKit import SwiftSignalKit +public protocol TooltipControllerCustomContentNode: ASDisplayNode { + func animateIn() + func updateLayout(size: CGSize) -> CGSize +} + public enum TooltipControllerContent: Equatable { case text(String) case attributedText(NSAttributedString) case iconAndText(UIImage, String) + case custom(TooltipControllerCustomContentNode) var text: String { switch self { @@ -14,6 +20,8 @@ public enum TooltipControllerContent: Equatable { return text case let .attributedText(text): return text.string + case .custom: + return "" } } @@ -23,6 +31,35 @@ public enum TooltipControllerContent: Equatable { } return nil } + + public static func == (lhs: TooltipControllerContent, rhs: TooltipControllerContent) -> Bool { + switch lhs { + case let .text(lhsText): + if case let .text(rhsText) = rhs, lhsText == rhsText { + return true + } else { + return false + } + case let .attributedText(lhsText): + if case let .attributedText(rhsText) = rhs, lhsText == rhsText { + return true + } else { + return false + } + case let .iconAndText(_, lhsText): + if case let .iconAndText(_, rhsText) = rhs, lhsText == rhsText { + return true + } else { + return false + } + case let .custom(lhsNode): + if case let .custom(rhsNode) = rhs, lhsNode === rhsNode { + return true + } else { + return false + } + } + } } public enum SourceAndRect { diff --git a/submodules/Display/Source/TooltipControllerNode.swift b/submodules/Display/Source/TooltipControllerNode.swift index ba0650c6ee..e67cc95e79 100644 --- a/submodules/Display/Source/TooltipControllerNode.swift +++ b/submodules/Display/Source/TooltipControllerNode.swift @@ -12,6 +12,7 @@ final class TooltipControllerNode: ASDisplayNode { private let containerNode: ContextMenuContainerNode private let imageNode: ASImageNode private let textNode: ImmediateTextNode + private var contentNode: TooltipControllerCustomContentNode? private let dismissByTapOutside: Bool @@ -45,10 +46,15 @@ final class TooltipControllerNode: ASDisplayNode { self.dismiss = dismiss + if case let .custom(contentNode) = content { + self.contentNode = contentNode + } + super.init() self.containerNode.addSubnode(self.imageNode) self.containerNode.addSubnode(self.textNode) + self.contentNode.flatMap { self.containerNode.addSubnode($0) } self.addSubnode(self.containerNode) } @@ -71,20 +77,37 @@ final class TooltipControllerNode: ASDisplayNode { func containerLayoutUpdated(_ layout: ContainerViewLayout, transition: ContainedViewLayoutTransition) { self.validLayout = layout - let maxActionsWidth = layout.size.width - 20.0 + let maxWidth = layout.size.width - 20.0 - var imageSize = CGSize() - var imageSizeWithInset = CGSize() - if let image = self.imageNode.image { - imageSize = image.size - imageSizeWithInset = CGSize(width: image.size.width + 12.0, height: image.size.height) + let contentSize: CGSize + + if let contentNode = self.contentNode { + contentSize = contentNode.updateLayout(size: layout.size) + contentNode.frame = CGRect(origin: CGPoint(), size: contentSize) + } else { + var imageSize = CGSize() + var imageSizeWithInset = CGSize() + if let image = self.imageNode.image { + imageSize = image.size + imageSizeWithInset = CGSize(width: image.size.width + 12.0, height: image.size.height) + } + + var textSize = self.textNode.updateLayout(CGSize(width: maxWidth, height: CGFloat.greatestFiniteMagnitude)) + textSize.width = ceil(textSize.width / 2.0) * 2.0 + textSize.height = ceil(textSize.height / 2.0) * 2.0 + + contentSize = CGSize(width: imageSizeWithInset.width + textSize.width + 12.0, height: textSize.height + 34.0) + + let textFrame = CGRect(origin: CGPoint(x: 6.0 + imageSizeWithInset.width, y: 17.0), size: textSize) + if transition.isAnimated, textFrame.size != self.textNode.frame.size { + transition.animatePositionAdditive(node: self.textNode, offset: CGPoint(x: textFrame.minX - self.textNode.frame.minX, y: 0.0)) + } + + let imageFrame = CGRect(origin: CGPoint(x: 10.0, y: floor((contentSize.height - imageSize.height) / 2.0)), size: imageSize) + self.imageNode.frame = imageFrame + self.textNode.frame = textFrame } - - var textSize = self.textNode.updateLayout(CGSize(width: maxActionsWidth, height: CGFloat.greatestFiniteMagnitude)) - textSize.width = ceil(textSize.width / 2.0) * 2.0 - textSize.height = ceil(textSize.height / 2.0) * 2.0 - let contentSize = CGSize(width: imageSizeWithInset.width + textSize.width + 12.0, height: textSize.height + 34.0) - + let sourceRect: CGRect = self.sourceRect ?? CGRect(origin: CGPoint(x: layout.size.width / 2.0, y: layout.size.height / 2.0), size: CGSize()) let insets = layout.insets(options: [.statusBar, .input]) @@ -105,19 +128,11 @@ final class TooltipControllerNode: ASDisplayNode { self.containerNode.relativeArrowPosition = (sourceRect.midX - horizontalOrigin, arrowOnBottom) self.containerNode.updateLayout(transition: transition) - - let textFrame = CGRect(origin: CGPoint(x: 6.0 + imageSizeWithInset.width, y: 17.0), size: textSize) - if transition.isAnimated, textFrame.size != self.textNode.frame.size { - transition.animatePositionAdditive(node: self.textNode, offset: CGPoint(x: textFrame.minX - self.textNode.frame.minX, y: 0.0)) - } - - let imageFrame = CGRect(origin: CGPoint(x: 10.0, y: floor((contentSize.height - imageSize.height) / 2.0)), size: imageSize) - self.imageNode.frame = imageFrame - self.textNode.frame = textFrame } func animateIn() { self.containerNode.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.25) + self.contentNode?.animateIn() } func animateOut(completion: @escaping () -> Void) { diff --git a/submodules/InviteLinksUI/BUILD b/submodules/InviteLinksUI/BUILD new file mode 100644 index 0000000000..0ecae93c03 --- /dev/null +++ b/submodules/InviteLinksUI/BUILD @@ -0,0 +1,56 @@ +load("@build_bazel_rules_swift//swift:swift.bzl", "swift_library") + +swift_library( + name = "InviteLinksUI", + module_name = "InviteLinksUI", + srcs = glob([ + "Sources/**/*.swift", + ]), + deps = [ + "//submodules/SSignalKit/SwiftSignalKit:SwiftSignalKit", + "//submodules/AsyncDisplayKit:AsyncDisplayKit", + "//submodules/Display:Display", + "//submodules/Postbox:Postbox", + "//submodules/TelegramCore:TelegramCore", + "//submodules/SyncCore:SyncCore", + "//submodules/TelegramPresentationData:TelegramPresentationData", + "//submodules/AccountContext:AccountContext", + "//submodules/ItemListUI:ItemListUI", + "//submodules/AlertUI:AlertUI", + "//submodules/PresentationDataUtils:PresentationDataUtils", + "//submodules/UndoUI:UndoUI", + "//submodules/TelegramUIPreferences:TelegramUIPreferences", + "//submodules/TemporaryCachedPeerDataManager:TemporaryCachedPeerDataManager", + "//submodules/ItemListPeerItem:ItemListPeerItem", + "//submodules/ItemListPeerActionItem:ItemListPeerActionItem", + "//submodules/OverlayStatusController:OverlayStatusController", + "//submodules/TelegramStringFormatting:TelegramStringFormatting", + "//submodules/SearchUI:SearchUI", + "//submodules/MergeLists:MergeLists", + "//submodules/TextFormat:TextFormat", + "//submodules/LegacyUI:LegacyUI", + "//submodules/ShareController:ShareController", + "//submodules/ContactsPeerItem:ContactsPeerItem", + "//submodules/ActivityIndicator:ActivityIndicator", + "//submodules/TelegramPermissionsUI:TelegramPermissionsUI", + "//submodules/ProgressNavigationButtonNode:ProgressNavigationButtonNode", + "//submodules/TelegramNotices:TelegramNotices", + "//submodules/PhotoResources:PhotoResources", + "//submodules/MediaResources:MediaResources", + "//submodules/NotificationSoundSelectionUI:NotificationSoundSelectionUI", + "//submodules/ContextUI:ContextUI", + "//submodules/AppBundle:AppBundle", + "//submodules/Markdown:Markdown", + "//submodules/SolidRoundedButtonNode:SolidRoundedButtonNode", + "//submodules/ChatListSearchItemHeader:ChatListSearchItemHeader", + "//submodules/QrCode:QrCode", + "//submodules/AnimatedAvatarSetNode:AnimatedAvatarSetNode", + "//submodules/DatePickerNode:DatePickerNode", + "//submodules/RadialStatusNode:RadialStatusNode", + "//submodules/SectionHeaderItem:SectionHeaderItem", + "//submodules/DirectionalPanGesture:DirectionalPanGesture", + ], + visibility = [ + "//visibility:public", + ], +) diff --git a/submodules/InviteLinksUI/Sources/InviteLinkEditController.swift b/submodules/InviteLinksUI/Sources/InviteLinkEditController.swift new file mode 100644 index 0000000000..3e60003bd7 --- /dev/null +++ b/submodules/InviteLinksUI/Sources/InviteLinkEditController.swift @@ -0,0 +1,428 @@ +import Foundation +import UIKit +import AsyncDisplayKit +import Display +import SwiftSignalKit +import Postbox +import TelegramCore +import SyncCore +import TelegramPresentationData +import TelegramUIPreferences +import ItemListUI +import PresentationDataUtils +import OverlayStatusController +import AccountContext +import AlertUI +import PresentationDataUtils +import AppBundle +import ContextUI +import TelegramStringFormatting + +private final class InviteLinkEditControllerArguments { + let context: AccountContext + let updateState: ((InviteLinkEditControllerState) -> InviteLinkEditControllerState) -> Void + let dismissInput: () -> Void + let revoke: () -> Void + + init(context: AccountContext, updateState: @escaping ((InviteLinkEditControllerState) -> InviteLinkEditControllerState) -> Void, dismissInput: @escaping () -> Void, revoke: @escaping () -> Void) { + self.context = context + self.updateState = updateState + self.dismissInput = dismissInput + self.revoke = revoke + } +} + +private enum InviteLinksEditSection: Int32 { + case time + case usage + case revoke +} + +private let invalidAmountCharacters = CharacterSet(charactersIn: "01234567890.,").inverted +func isValidNumberOfUsers(_ number: String) -> Bool { + let number = normalizeArabicNumeralString(number, type: .western) + if number.rangeOfCharacter(from: invalidAmountCharacters) != nil || number == "0" { + return false + } + return true +} + +private enum InviteLinksEditEntry: ItemListNodeEntry { + case timeHeader(PresentationTheme, String) + case timePicker(PresentationTheme, InviteLinkTimeLimit) + case timeExpiryDate(PresentationTheme, Int32?) + case timeCustomPicker(PresentationTheme, Int32?) + case timeInfo(PresentationTheme, String) + + case usageHeader(PresentationTheme, String) + case usagePicker(PresentationTheme, InviteLinkUsageLimit) + case usageCustomPicker(PresentationTheme, Int32?, Bool) + case usageInfo(PresentationTheme, String) + + case revoke(PresentationTheme, String) + + var section: ItemListSectionId { + switch self { + case .timeHeader, .timePicker, .timeExpiryDate, .timeCustomPicker, .timeInfo: + return InviteLinksEditSection.time.rawValue + case .usageHeader, .usagePicker, .usageCustomPicker, .usageInfo: + return InviteLinksEditSection.usage.rawValue + case .revoke: + return InviteLinksEditSection.revoke.rawValue + } + } + + var stableId: Int32 { + switch self { + case .timeHeader: + return 0 + case .timePicker: + return 1 + case .timeExpiryDate: + return 2 + case .timeCustomPicker: + return 3 + case .timeInfo: + return 4 + case .usageHeader: + return 5 + case .usagePicker: + return 6 + case .usageCustomPicker: + return 7 + case .usageInfo: + return 8 + case .revoke: + return 9 + } + } + + static func ==(lhs: InviteLinksEditEntry, rhs: InviteLinksEditEntry) -> Bool { + switch lhs { + case let .timeHeader(lhsTheme, lhsText): + if case let .timeHeader(rhsTheme, rhsText) = rhs, lhsTheme === rhsTheme, lhsText == rhsText { + return true + } else { + return false + } + case let .timePicker(lhsTheme, lhsValue): + if case let .timePicker(rhsTheme, rhsValue) = rhs, lhsTheme === rhsTheme, lhsValue == rhsValue { + return true + } else { + return false + } + case let .timeExpiryDate(lhsTheme, lhsDate): + if case let .timeExpiryDate(rhsTheme, rhsDate) = rhs, lhsTheme === rhsTheme, lhsDate == rhsDate { + return true + } else { + return false + } + case let .timeCustomPicker(lhsTheme, lhsDate): + if case let .timeCustomPicker(rhsTheme, rhsDate) = rhs, lhsTheme === rhsTheme, lhsDate == rhsDate { + return true + } else { + return false + } + case let .timeInfo(lhsTheme, lhsText): + if case let .timeInfo(rhsTheme, rhsText) = rhs, lhsTheme === rhsTheme, lhsText == rhsText { + return true + } else { + return false + } + case let .usageHeader(lhsTheme, lhsText): + if case let .usageHeader(rhsTheme, rhsText) = rhs, lhsTheme === rhsTheme, lhsText == rhsText { + return true + } else { + return false + } + case let .usagePicker(lhsTheme, lhsValue): + if case let .usagePicker(rhsTheme, rhsValue) = rhs, lhsTheme === rhsTheme, lhsValue == rhsValue { + return true + } else { + return false + } + case let .usageCustomPicker(lhsTheme, lhsValue, lhsFocused): + if case let .usageCustomPicker(rhsTheme, rhsValue, rhsFocused) = rhs, lhsTheme === rhsTheme, lhsValue == rhsValue, lhsFocused == rhsFocused { + return true + } else { + return false + } + case let .usageInfo(lhsTheme, lhsText): + if case let .usageInfo(rhsTheme, rhsText) = rhs, lhsTheme === rhsTheme, lhsText == rhsText { + return true + } else { + return false + } + case let .revoke(lhsTheme, lhsText): + if case let .revoke(rhsTheme, rhsText) = rhs, lhsTheme === rhsTheme, lhsText == rhsText { + return true + } else { + return false + } + } + } + + static func <(lhs: InviteLinksEditEntry, rhs: InviteLinksEditEntry) -> Bool { + return lhs.stableId < rhs.stableId + } + + func item(presentationData: ItemListPresentationData, arguments: Any) -> ListViewItem { + let arguments = arguments as! InviteLinkEditControllerArguments + switch self { + case let .timeHeader(_, text): + return ItemListSectionHeaderItem(presentationData: presentationData, text: text, sectionId: self.section) + case let .timePicker(_, value): + return ItemListInviteLinkTimeLimitItem(theme: presentationData.theme, strings: presentationData.strings, value: value, enabled: true, sectionId: self.section, updated: { value in + arguments.updateState({ state in + var updatedState = state + if value != updatedState.time { + updatedState.pickingTimeLimit = false + } + updatedState.time = value + return updatedState + }) + }) + case let .timeExpiryDate(_, value): + let text: String + if let value = value { + text = stringForFullDate(timestamp: value, strings: presentationData.strings, dateTimeFormat: PresentationDateTimeFormat(timeFormat: .regular, dateFormat: .monthFirst, dateSeparator: ".", decimalSeparator: ".", groupingSeparator: ".")) + } else { + text = presentationData.strings.InviteLink_Create_TimeLimitExpiryDateNever + } + return ItemListDisclosureItem(presentationData: presentationData, title: presentationData.strings.InviteLink_Create_TimeLimitExpiryDate, label: text, sectionId: self.section, style: .blocks, disclosureStyle: .none, action: { + arguments.dismissInput() + arguments.updateState { state in + var updatedState = state + updatedState.pickingTimeLimit = !state.pickingTimeLimit + return updatedState + } + }) + case let .timeCustomPicker(_, date): + return ItemListDatePickerItem(presentationData: presentationData, date: date, sectionId: self.section, style: .blocks, updated: { date in + arguments.updateState({ state in + var updatedState = state + updatedState.time = .custom(date) + return updatedState + }) + }) + case let .timeInfo(_, text): + return ItemListTextItem(presentationData: presentationData, text: .plain(text), sectionId: self.section) + case let .usageHeader(_, text): + return ItemListSectionHeaderItem(presentationData: presentationData, text: text, sectionId: self.section) + case let .usagePicker(_, value): + return ItemListInviteLinkUsageLimitItem(theme: presentationData.theme, strings: presentationData.strings, value: value, enabled: true, sectionId: self.section, updated: { value in + arguments.dismissInput() + arguments.updateState({ state in + var updatedState = state + if value != updatedState.usage { + updatedState.pickingTimeLimit = false + } + updatedState.usage = value + return updatedState + }) + }) + case let .usageCustomPicker(theme, value, focused): + let text = value.flatMap { String($0) } ?? (focused ? "" : presentationData.strings.InviteLink_Create_UsersLimitNumberOfUsersUnlimited) + return ItemListSingleLineInputItem(presentationData: presentationData, title: NSAttributedString(string: presentationData.strings.InviteLink_Create_UsersLimitNumberOfUsers, textColor: theme.list.itemPrimaryTextColor), text: text, placeholder: "", type: .number, alignment: .right, tag: nil, sectionId: self.section, textUpdated: { updatedText in + guard !updatedText.isEmpty else { + return + } + arguments.updateState { state in + var updatedState = state + updatedState.usage = InviteLinkUsageLimit(value: Int32(updatedText)) + return updatedState + } + }, shouldUpdateText: { text in + return isValidNumberOfUsers(text) + }, updatedFocus: { focus in + if focus { + arguments.updateState { state in + var updatedState = state + updatedState.pickingTimeLimit = false + updatedState.pickingUsageLimit = true + return updatedState + } + } else { + arguments.updateState { state in + var updatedState = state + updatedState.pickingUsageLimit = false + return updatedState + } + } + }, action: { + + }) + case let .usageInfo(_, text): + return ItemListTextItem(presentationData: presentationData, text: .plain(text), sectionId: self.section) + case let .revoke(_, text): + return ItemListActionItem(presentationData: presentationData, title: text, kind: .destructive, alignment: .center, sectionId: self.section, style: .blocks, action: { + arguments.revoke() + }, tag: nil) + } + } +} + +private func inviteLinkEditControllerEntries(invite: ExportedInvitation?, state: InviteLinkEditControllerState, presentationData: PresentationData) -> [InviteLinksEditEntry] { + var entries: [InviteLinksEditEntry] = [] + + entries.append(.timeHeader(presentationData.theme, presentationData.strings.InviteLink_Create_TimeLimit.uppercased())) + + entries.append(.timePicker(presentationData.theme, state.time)) + + let currentTime = Int32(CFAbsoluteTimeGetCurrent() + kCFAbsoluteTimeIntervalSince1970) + var time: Int32? + if case let .custom(value) = state.time { + time = value + } else if let value = state.time.value { + time = currentTime + value + } + entries.append(.timeExpiryDate(presentationData.theme, time)) + if state.pickingTimeLimit { + entries.append(.timeCustomPicker(presentationData.theme, time ?? currentTime)) + } + entries.append(.timeInfo(presentationData.theme, presentationData.strings.InviteLink_Create_TimeLimitInfo)) + + entries.append(.usageHeader(presentationData.theme, presentationData.strings.InviteLink_Create_UsersLimit.uppercased())) + entries.append(.usagePicker(presentationData.theme, state.usage)) + entries.append(.usageCustomPicker(presentationData.theme, state.usage.value, state.pickingUsageLimit)) + + entries.append(.usageInfo(presentationData.theme, presentationData.strings.InviteLink_Create_UsersLimitInfo)) + + if let _ = invite { + entries.append(.revoke(presentationData.theme, presentationData.strings.InviteLink_Create_Revoke)) + } + + return entries +} + +private struct InviteLinkEditControllerState: Equatable { + var usage: InviteLinkUsageLimit + var time: InviteLinkTimeLimit + var pickingTimeLimit = false + var pickingUsageLimit = false +} + +public func inviteLinkEditController(context: AccountContext, peerId: PeerId, invite: ExportedInvitation?, completion: (() -> Void)? = nil) -> ViewController { + var presentControllerImpl: ((ViewController, ViewControllerPresentationArguments?) -> Void)? + let actionsDisposable = DisposableSet() + + let initialState: InviteLinkEditControllerState + if let invite = invite { + var usageLimit = invite.usageLimit + if let limit = usageLimit, let count = invite.count, count > 0 { + usageLimit = limit - count + } + + let timeLimit: InviteLinkTimeLimit + let currentTime = Int32(CFAbsoluteTimeGetCurrent() + kCFAbsoluteTimeIntervalSince1970) + if let expireDate = invite.expireDate { + if currentTime >= expireDate { + timeLimit = .day + } else { + timeLimit = .custom(expireDate) + } + } else { + timeLimit = .unlimited + } + + initialState = InviteLinkEditControllerState(usage: InviteLinkUsageLimit(value: usageLimit), time: timeLimit, pickingTimeLimit: false, pickingUsageLimit: false) + } else { + initialState = InviteLinkEditControllerState(usage: .medium, time: .week, pickingTimeLimit: false, pickingUsageLimit: false) + } + + let statePromise = ValuePromise(initialState, ignoreRepeated: true) + let stateValue = Atomic(value: initialState) + let updateState: ((InviteLinkEditControllerState) -> InviteLinkEditControllerState) -> Void = { f in + statePromise.set(stateValue.modify { f($0) }) + } + + var dismissImpl: (() -> Void)? + var dismissInputImpl: (() -> Void)? + + let arguments = InviteLinkEditControllerArguments(context: context, updateState: { f in + updateState(f) + }, dismissInput: { + dismissInputImpl?() + }, revoke: { + guard let invite = invite else { + return + } + 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.GroupInfo_InviteLink_RevokeAlert_Text), + ActionSheetButtonItem(title: presentationData.strings.GroupInfo_InviteLink_RevokeLink, color: .destructive, action: { + dismissAction() + dismissImpl?() + + let _ = (revokePeerExportedInvitation(account: context.account, peerId: peerId, link: invite.link) |> deliverOnMainQueue).start(completed: { + completion?() + }) + }) + ]), + ActionSheetItemGroup(items: [ActionSheetButtonItem(title: presentationData.strings.Common_Cancel, action: { dismissAction() })]) + ]) + presentControllerImpl?(controller, nil) + }) + + let signal = combineLatest(context.sharedContext.presentationData, statePromise.get()) + |> deliverOnMainQueue + |> map { presentationData, state -> (ItemListControllerState, (ItemListNodeState, Any)) in + let leftNavigationButton = ItemListNavigationButton(content: .text(presentationData.strings.Common_Cancel), style: .regular, enabled: true, action: { + dismissImpl?() + }) + + let rightNavigationButton = ItemListNavigationButton(content: .text(presentationData.strings.Common_Done), style: .bold, enabled: true, action: { + let expireDate: Int32? + if case let .custom(value) = state.time { + expireDate = value + } else if let value = state.time.value { + let currentTime = Int32(CFAbsoluteTimeGetCurrent() + kCFAbsoluteTimeIntervalSince1970) + expireDate = currentTime + value + } else { + expireDate = nil + } + + let usageLimit = state.usage.value + if invite == nil { + let _ = (createPeerExportedInvitation(account: context.account, peerId: peerId, expireDate: expireDate, usageLimit: usageLimit) + |> deliverOnMainQueue).start(next: { result in + completion?() + dismissImpl?() + }) + } else if let invite = invite { + let _ = (editPeerExportedInvitation(account: context.account, peerId: peerId, link: invite.link, expireDate: expireDate, usageLimit: usageLimit) + |> deliverOnMainQueue).start(next: { result in + completion?() + dismissImpl?() + }) + } + }) + + let controllerState = ItemListControllerState(presentationData: ItemListPresentationData(presentationData), title: .text(invite == nil ? presentationData.strings.InviteLink_Create_Title : presentationData.strings.InviteLink_Create_EditTitle), leftNavigationButton: leftNavigationButton, rightNavigationButton: rightNavigationButton, backNavigationButton: ItemListBackButton(title: presentationData.strings.Common_Back), animateChanges: true) + let listState = ItemListNodeState(presentationData: ItemListPresentationData(presentationData), entries: inviteLinkEditControllerEntries(invite: invite, state: state, presentationData: presentationData), style: .blocks, emptyStateItem: nil, crossfadeState: false, animateChanges: false) + + return (controllerState, (listState, arguments)) + } + |> afterDisposed { + actionsDisposable.dispose() + } + + let controller = ItemListController(context: context, state: signal) + presentControllerImpl = { [weak controller] c, p in + if let controller = controller { + controller.present(c, in: .window(.root), with: p) + } + } + dismissInputImpl = { [weak controller] in + controller?.view.endEditing(true) + } + dismissImpl = { [weak controller] in + controller?.dismiss() + } + return controller +} diff --git a/submodules/InviteLinksUI/Sources/InviteLinkHeaderItem.swift b/submodules/InviteLinksUI/Sources/InviteLinkHeaderItem.swift new file mode 100644 index 0000000000..64dd3c37b1 --- /dev/null +++ b/submodules/InviteLinksUI/Sources/InviteLinkHeaderItem.swift @@ -0,0 +1,124 @@ +import Foundation +import UIKit +import Display +import AsyncDisplayKit +import SwiftSignalKit +import TelegramPresentationData +import ItemListUI +import PresentationDataUtils +import AnimatedStickerNode +import AppBundle + +class InviteLinkHeaderItem: ListViewItem, ItemListItem { + let theme: PresentationTheme + let text: String + let sectionId: ItemListSectionId + + init(theme: PresentationTheme, text: String, sectionId: ItemListSectionId) { + self.theme = theme + self.text = text + self.sectionId = sectionId + } + + func nodeConfiguredForParams(async: @escaping (@escaping () -> Void) -> Void, params: ListViewItemLayoutParams, synchronousLoads: Bool, previousItem: ListViewItem?, nextItem: ListViewItem?, completion: @escaping (ListViewItemNode, @escaping () -> (Signal?, (ListViewItemApply) -> Void)) -> Void) { + async { + let node = InviteLinkHeaderItemNode() + let (layout, apply) = node.asyncLayout()(self, params, itemListNeighbors(item: self, topItem: previousItem as? ItemListItem, bottomItem: nextItem as? ItemListItem)) + + node.contentSize = layout.contentSize + node.insets = layout.insets + + Queue.mainQueue().async { + completion(node, { + return (nil, { _ in apply() }) + }) + } + } + } + + func updateNode(async: @escaping (@escaping () -> Void) -> Void, node: @escaping () -> ListViewItemNode, params: ListViewItemLayoutParams, previousItem: ListViewItem?, nextItem: ListViewItem?, animation: ListViewItemUpdateAnimation, completion: @escaping (ListViewItemNodeLayout, @escaping (ListViewItemApply) -> Void) -> Void) { + Queue.mainQueue().async { + guard let nodeValue = node() as? InviteLinkHeaderItemNode else { + assertionFailure() + return + } + + let makeLayout = nodeValue.asyncLayout() + + async { + let (layout, apply) = makeLayout(self, params, itemListNeighbors(item: self, topItem: previousItem as? ItemListItem, bottomItem: nextItem as? ItemListItem)) + Queue.mainQueue().async { + completion(layout, { _ in + apply() + }) + } + } + } + } +} + +private let titleFont = Font.regular(13.0) + +class InviteLinkHeaderItemNode: ListViewItemNode { + private let titleNode: TextNode + private var animationNode: AnimatedStickerNode + + private var item: InviteLinkHeaderItem? + + init() { + self.titleNode = TextNode() + self.titleNode.isUserInteractionEnabled = false + self.titleNode.contentMode = .left + self.titleNode.contentsScale = UIScreen.main.scale + + self.animationNode = AnimatedStickerNode() + if let path = getAppBundle().path(forResource: "Invite", ofType: "tgs") { + self.animationNode.setup(source: AnimatedStickerNodeLocalFileSource(path: path), width: 192, height: 192, playbackMode: .loop, mode: .direct(cachePathPrefix: nil)) + self.animationNode.visibility = true + } + + super.init(layerBacked: false, dynamicBounce: false) + + self.addSubnode(self.titleNode) + self.addSubnode(self.animationNode) + } + + func asyncLayout() -> (_ item: InviteLinkHeaderItem, _ params: ListViewItemLayoutParams, _ neighbors: ItemListNeighbors) -> (ListViewItemNodeLayout, () -> Void) { + let makeTitleLayout = TextNode.asyncLayout(self.titleNode) + + return { item, params, neighbors in + let leftInset: CGFloat = 32.0 + params.leftInset + let topInset: CGFloat = 92.0 + + let attributedText = NSAttributedString(string: item.text, font: titleFont, textColor: item.theme.list.freeTextColor) + let (titleLayout, titleApply) = makeTitleLayout(TextNodeLayoutArguments(attributedString: attributedText, backgroundColor: nil, maximumNumberOfLines: 0, truncationType: .end, constrainedSize: CGSize(width: params.width - params.rightInset - leftInset * 2.0, height: CGFloat.greatestFiniteMagnitude), alignment: .center, cutout: nil, insets: UIEdgeInsets())) + + let contentSize = CGSize(width: params.width, height: topInset + titleLayout.size.height) + let insets = itemListNeighborsGroupedInsets(neighbors) + + let layout = ListViewItemNodeLayout(contentSize: contentSize, insets: insets) + + return (layout, { [weak self] in + if let strongSelf = self { + strongSelf.item = item + strongSelf.accessibilityLabel = attributedText.string + + let iconSize = CGSize(width: 96.0, height: 96.0) + strongSelf.animationNode.frame = CGRect(origin: CGPoint(x: floor((layout.size.width - iconSize.width) / 2.0), y: -10.0), size: iconSize) + strongSelf.animationNode.updateLayout(size: iconSize) + + let _ = titleApply() + strongSelf.titleNode.frame = CGRect(origin: CGPoint(x: floor((layout.size.width - titleLayout.size.width) / 2.0), y: topInset + 8.0), size: titleLayout.size) + } + }) + } + } + + override func animateInsertion(_ currentTimestamp: Double, duration: Double, short: Bool) { + self.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.4) + } + + override func animateRemoved(_ currentTimestamp: Double, duration: Double) { + self.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.15, removeOnCompletion: false) + } +} diff --git a/submodules/InviteLinksUI/Sources/InviteLinkInviteController.swift b/submodules/InviteLinksUI/Sources/InviteLinkInviteController.swift new file mode 100644 index 0000000000..24624caeec --- /dev/null +++ b/submodules/InviteLinksUI/Sources/InviteLinkInviteController.swift @@ -0,0 +1,667 @@ +import Foundation +import UIKit +import SwiftSignalKit +import TelegramPresentationData +import AppBundle +import AsyncDisplayKit +import Postbox +import SyncCore +import TelegramCore +import Display +import AccountContext +import SolidRoundedButtonNode +import ItemListUI +import ItemListPeerItem +import SectionHeaderItem +import TelegramStringFormatting +import MergeLists +import ContextUI +import ShareController +import OverlayStatusController +import PresentationDataUtils +import DirectionalPanGesture + +class InviteLinkInviteInteraction { + let context: AccountContext + let mainLinkContextAction: (ExportedInvitation?, ASDisplayNode, ContextGesture?) -> Void + let copyLink: (ExportedInvitation) -> Void + let shareLink: (ExportedInvitation) -> Void + let manageLinks: () -> Void + + init(context: AccountContext, mainLinkContextAction: @escaping (ExportedInvitation?, ASDisplayNode, ContextGesture?) -> Void, copyLink: @escaping (ExportedInvitation) -> Void, shareLink: @escaping (ExportedInvitation) -> Void, manageLinks: @escaping () -> Void) { + self.context = context + self.mainLinkContextAction = mainLinkContextAction + self.copyLink = copyLink + self.shareLink = shareLink + self.manageLinks = manageLinks + } +} + +private struct InviteLinkInviteTransaction { + let deletions: [ListViewDeleteItem] + let insertions: [ListViewInsertItem] + let updates: [ListViewUpdateItem] + let isLoading: Bool +} + +private enum InviteLinkInviteEntryId: Hashable { + case mainLink + case links(Int32) +} + +private enum InviteLinkInviteEntry: Comparable, Identifiable { + case mainLink(PresentationTheme, ExportedInvitation) + case links(Int32, PresentationTheme, [ExportedInvitation]) + + var stableId: InviteLinkInviteEntryId { + switch self { + case .mainLink: + return .mainLink + case let .links(index, _, _): + return .links(index) + } + } + + static func ==(lhs: InviteLinkInviteEntry, rhs: InviteLinkInviteEntry) -> Bool { + switch lhs { + case let .mainLink(lhsTheme, lhsInvitation): + if case let .mainLink(rhsTheme, rhsInvitation) = rhs, lhsTheme === rhsTheme, lhsInvitation == rhsInvitation { + return true + } else { + return false + } + case let .links(lhsIndex, lhsTheme, lhsInvitations): + if case let .links(rhsIndex, rhsTheme, rhsInvitations) = rhs, lhsIndex == rhsIndex, lhsTheme === rhsTheme, lhsInvitations == rhsInvitations { + return true + } else { + return false + } + } + } + + static func <(lhs: InviteLinkInviteEntry, rhs: InviteLinkInviteEntry) -> Bool { + switch lhs { + case .mainLink: + switch rhs { + case .mainLink: + return false + case .links: + return true + } + case let .links(lhsIndex, _, _): + switch rhs { + case .mainLink: + return false + case let .links(rhsIndex, _, _): + return lhsIndex < rhsIndex + } + } + } + + func item(account: Account, presentationData: PresentationData, interaction: InviteLinkInviteInteraction) -> ListViewItem { + switch self { + case let .mainLink(_, invite): + return ItemListPermanentInviteLinkItem(context: interaction.context, presentationData: ItemListPresentationData(presentationData), invite: invite, peers: [], buttonColor: nil, sectionId: 0, style: .plain, shareAction: { + interaction.shareLink(invite) + }, contextAction: { node in + interaction.mainLinkContextAction(invite, node, nil) + }, viewAction: { + }) + case let .links(_, _, invites): + return ItemListInviteLinkGridItem(presentationData: ItemListPresentationData(presentationData), invites: invites, sectionId: 0, style: .plain, tapAction: { invite in + interaction.copyLink(invite) + }, contextAction: { invite, _ in + interaction.shareLink(invite) + }) + } + } +} + +private func preparedTransition(from fromEntries: [InviteLinkInviteEntry], to toEntries: [InviteLinkInviteEntry], isLoading: Bool, account: Account, presentationData: PresentationData, interaction: InviteLinkInviteInteraction) -> InviteLinkInviteTransaction { + let (deleteIndices, indicesAndItems, updateIndices) = mergeListsStableWithUpdates(leftList: fromEntries, rightList: toEntries) + + let deletions = deleteIndices.map { ListViewDeleteItem(index: $0, directionHint: nil) } + let insertions = indicesAndItems.map { ListViewInsertItem(index: $0.0, previousIndex: $0.2, item: $0.1.item(account: account, presentationData: presentationData, interaction: interaction), directionHint: nil) } + let updates = updateIndices.map { ListViewUpdateItem(index: $0.0, previousIndex: $0.2, item: $0.1.item(account: account, presentationData: presentationData, interaction: interaction), directionHint: nil) } + + return InviteLinkInviteTransaction(deletions: deletions, insertions: insertions, updates: updates, isLoading: isLoading) +} + +public final class InviteLinkInviteController: ViewController { + private var controllerNode: Node { + return self.displayNode as! Node + } + + private var animatedIn = false + + private let context: AccountContext + private let peerId: PeerId + + private var presentationDataDisposable: Disposable? + + public init(context: AccountContext, peerId: PeerId) { + fatalError() + self.context = context + self.peerId = peerId + + super.init(navigationBarPresentationData: nil) + + self.navigationPresentation = .flatModal + self.statusBar.statusBarStyle = .Ignore + + self.blocksBackgroundWhenInOverlay = true + + self.presentationDataDisposable = (context.sharedContext.presentationData + |> deliverOnMainQueue).start(next: { [weak self] presentationData in + if let strongSelf = self { + strongSelf.controllerNode.updatePresentationData(presentationData) + } + }) + + self.statusBar.statusBarStyle = .Ignore + } + + required init(coder aDecoder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + deinit { + self.presentationDataDisposable?.dispose() + } + + override public func loadDisplayNode() { + self.displayNode = Node(context: self.context, peerId: self.peerId, controller: self) + } + + override public func loadView() { + super.loadView() + } + + private var didAppearOnce: Bool = false + private var isDismissed: Bool = false + public override func viewDidAppear(_ animated: Bool) { + super.viewDidAppear(animated) + + if !self.didAppearOnce { + self.didAppearOnce = true + + self.controllerNode.animateIn() + } + } + + override public func dismiss(completion: (() -> Void)? = nil) { + if !self.isDismissed { + self.isDismissed = true + self.didAppearOnce = false + + self.controllerNode.animateOut(completion: { [weak self] in + completion?() + self?.dismiss(animated: false) + }) + } + } + + override public func containerLayoutUpdated(_ layout: ContainerViewLayout, transition: ContainedViewLayoutTransition) { + super.containerLayoutUpdated(layout, transition: transition) + + self.controllerNode.containerLayoutUpdated(layout, transition: transition) + } + + class Node: ViewControllerTracingNode, UIGestureRecognizerDelegate { + private weak var controller: InviteLinkInviteController? + + private let context: AccountContext + private let peerId: PeerId + + private var interaction: InviteLinkInviteInteraction? + + private var presentationData: PresentationData + private let presentationDataPromise: Promise + + private var disposable: Disposable? + + private let dimNode: ASDisplayNode + private let contentNode: ASDisplayNode + private let headerNode: ASDisplayNode + private let headerBackgroundNode: ASDisplayNode + private let titleNode: ImmediateTextNode + private let doneButton: HighlightableButtonNode + private let historyBackgroundNode: ASDisplayNode + private let historyBackgroundContentNode: ASDisplayNode + private var floatingHeaderOffset: CGFloat? + private let listNode: ListView + + private var enqueuedTransitions: [InviteLinkInviteTransaction] = [] + + private var validLayout: ContainerViewLayout? + + private var presentationDataDisposable: Disposable? + private var revokeDisposable = MetaDisposable() + + init(context: AccountContext, peerId: PeerId, controller: InviteLinkInviteController) { + self.context = context + self.peerId = peerId + self.presentationData = context.sharedContext.currentPresentationData.with { $0 } + self.presentationDataPromise = Promise(self.presentationData) + self.controller = controller + + self.dimNode = ASDisplayNode() + self.dimNode.backgroundColor = UIColor(white: 0.0, alpha: 0.5) + + self.contentNode = ASDisplayNode() + + self.headerNode = ASDisplayNode() + self.headerNode.clipsToBounds = true + + self.headerBackgroundNode = ASDisplayNode() + self.headerBackgroundNode.backgroundColor = self.presentationData.theme.list.plainBackgroundColor + self.headerBackgroundNode.cornerRadius = 16.0 + + self.titleNode = ImmediateTextNode() + self.titleNode.maximumNumberOfLines = 1 + self.titleNode.textAlignment = .center + self.titleNode.attributedText = NSAttributedString(string: self.presentationData.strings.InviteLink_InviteLink, font: Font.bold(17.0), textColor: self.presentationData.theme.actionSheet.primaryTextColor) + + self.doneButton = HighlightableButtonNode() + self.doneButton.setTitle(self.presentationData.strings.Common_Done, with: Font.bold(17.0), with: self.presentationData.theme.actionSheet.controlAccentColor, for: .normal) + + self.historyBackgroundNode = ASDisplayNode() + self.historyBackgroundNode.isLayerBacked = true + + self.historyBackgroundContentNode = ASDisplayNode() + self.historyBackgroundContentNode.isLayerBacked = true + self.historyBackgroundContentNode.backgroundColor = self.presentationData.theme.list.plainBackgroundColor + + self.historyBackgroundNode.addSubnode(self.historyBackgroundContentNode) + + self.listNode = ListView() + self.listNode.verticalScrollIndicatorColor = UIColor(white: 0.0, alpha: 0.3) + self.listNode.verticalScrollIndicatorFollowsOverscroll = true + + super.init() + + self.backgroundColor = nil + self.isOpaque = false + + self.interaction = InviteLinkInviteInteraction(context: context, mainLinkContextAction: { [weak self] invite, node, gesture in + guard let node = node as? ContextExtractedContentContainingNode else { + return + } + let presentationData = context.sharedContext.currentPresentationData.with { $0 } + var items: [ContextMenuItem] = [] + + items.append(.action(ContextMenuActionItem(text: presentationData.strings.InviteLink_ContextCopy, icon: { theme in + return generateTintedImage(image: UIImage(bundleImageName: "Chat/Context Menu/Copy"), color: theme.contextMenu.primaryColor) + }, action: { _, f in + f(.dismissWithoutContent) + + if let invite = invite { + UIPasteboard.general.string = invite.link + let presentationData = context.sharedContext.currentPresentationData.with { $0 } + self?.controller?.present(OverlayStatusController(theme: presentationData.theme, type: .genericSuccess(presentationData.strings.Username_LinkCopied, false)), in: .window(.root)) + } + }))) + + items.append(.action(ContextMenuActionItem(text: presentationData.strings.InviteLink_ContextGetQRCode, icon: { theme in + return generateTintedImage(image: UIImage(bundleImageName: "Wallet/QrIcon"), color: theme.contextMenu.primaryColor) + }, action: { _, f in + f(.dismissWithoutContent) + + if let invite = invite { + let controller = InviteLinkQRCodeController(context: context, invite: invite) + self?.controller?.present(controller, in: .window(.root)) + } + }))) + + items.append(.action(ContextMenuActionItem(text: presentationData.strings.InviteLink_ContextRevoke, textColor: .destructive, icon: { theme in + return generateTintedImage(image: UIImage(bundleImageName: "Chat/Context Menu/Delete"), color: theme.actionSheet.destructiveActionTextColor) + }, action: { _, f in + f(.dismissWithoutContent) + + let controller = ActionSheetController(presentationData: presentationData) + let dismissAction: () -> Void = { [weak controller] in + controller?.dismissAnimated() + } + controller.setItemGroups([ + ActionSheetItemGroup(items: [ + ActionSheetTextItem(title: presentationData.strings.GroupInfo_InviteLink_RevokeAlert_Text), + ActionSheetButtonItem(title: presentationData.strings.GroupInfo_InviteLink_RevokeLink, color: .destructive, action: { + dismissAction() + + self?.revokeDisposable.set((ensuredExistingPeerExportedInvitation(account: context.account, peerId: peerId, revokeExisted: true) |> deliverOnMainQueue).start(completed: { + + })) + }) + ]), + ActionSheetItemGroup(items: [ActionSheetButtonItem(title: presentationData.strings.Common_Cancel, action: { dismissAction() })]) + ]) + self?.controller?.present(controller, in: .window(.root)) + }))) + + let contextController = ContextController(account: context.account, presentationData: presentationData, source: .extracted(InviteLinkContextExtractedContentSource(controller: controller, sourceNode: node)), items: .single(items), reactionItems: [], gesture: gesture) + self?.controller?.presentInGlobalOverlay(contextController) + }, copyLink: { [weak self] invite in + let shareController = ShareController(context: context, subject: .url(invite.link)) + self?.controller?.present(shareController, in: .window(.root)) + }, shareLink: { [weak self] invite in + let shareController = ShareController(context: context, subject: .url(invite.link)) + self?.controller?.present(shareController, in: .window(.root)) + }, manageLinks: { [weak self] in + let controller = inviteLinkListController(context: context, peerId: peerId) + self?.controller?.push(controller) + self?.controller?.dismiss() + }) + + let previousEntries = Atomic<[InviteLinkInviteEntry]?>(value: nil) + + let peerView = context.account.postbox.peerView(id: peerId) + self.disposable = (combineLatest(self.presentationDataPromise.get(), peerView) + |> deliverOnMainQueue).start(next: { [weak self] presentationData, view in + if let strongSelf = self { + var entries: [InviteLinkInviteEntry] = [] + + if let cachedData = view.cachedData as? CachedGroupData, let invite = cachedData.exportedInvitation { + entries.append(.mainLink(presentationData.theme, invite)) + } else if let cachedData = view.cachedData as? CachedChannelData, let invite = cachedData.exportedInvitation { + entries.append(.mainLink(presentationData.theme, invite)) + } + + + let previousEntries = previousEntries.swap(entries) + + let transition = preparedTransition(from: previousEntries ?? [], to: entries, isLoading: false, account: context.account, presentationData: presentationData, interaction: strongSelf.interaction!) + strongSelf.enqueueTransition(transition) + } + }) + + self.listNode.preloadPages = true + self.listNode.stackFromBottom = true + self.listNode.updateFloatingHeaderOffset = { [weak self] offset, transition in + if let strongSelf = self { + strongSelf.updateFloatingHeaderOffset(offset: offset, transition: transition) + } + } + self.listNode.visibleBottomContentOffsetChanged = { [weak self] offset in + if case let .known(value) = offset, value < 40.0 { + + } + } + + self.addSubnode(self.dimNode) + self.addSubnode(self.contentNode) + self.contentNode.addSubnode(self.historyBackgroundNode) + self.contentNode.addSubnode(self.listNode) + self.contentNode.addSubnode(self.headerNode) + + self.headerNode.addSubnode(self.headerBackgroundNode) +// self.headerNode.addSubnode(self.titleNode) + self.headerNode.addSubnode(self.doneButton) + + self.doneButton.addTarget(self, action: #selector(self.doneButtonPressed), forControlEvents: .touchUpInside) + + self.presentationDataDisposable = context.sharedContext.presentationData.start(next: { [weak self] presentationData in + if let strongSelf = self { + if strongSelf.presentationData.theme !== presentationData.theme || strongSelf.presentationData.strings !== presentationData.strings { + strongSelf.updatePresentationData(presentationData) + } + } + }) + } + + deinit { + self.disposable?.dispose() + self.presentationDataDisposable?.dispose() + self.revokeDisposable.dispose() + } + + override func didLoad() { + super.didLoad() + + self.view.disablesInteractiveTransitionGestureRecognizer = true + self.view.disablesInteractiveModalDismiss = true + + self.dimNode.view.addGestureRecognizer(UITapGestureRecognizer(target: self, action: #selector(self.dimTapGesture(_:)))) + + let panRecognizer = DirectionalPanGestureRecognizer(target: self, action: #selector(self.panGesture(_:))) + panRecognizer.delegate = self + panRecognizer.delaysTouchesBegan = false + panRecognizer.cancelsTouchesInView = true + self.view.addGestureRecognizer(panRecognizer) + } + + @objc private func doneButtonPressed() { + self.controller?.dismiss() + } + + func updatePresentationData(_ presentationData: PresentationData) { + self.presentationData = presentationData + self.presentationDataPromise.set(.single(presentationData)) + + self.historyBackgroundContentNode.backgroundColor = self.presentationData.theme.list.plainBackgroundColor + self.headerBackgroundNode.backgroundColor = self.presentationData.theme.list.plainBackgroundColor + self.titleNode.attributedText = NSAttributedString(string: self.presentationData.strings.InviteLink_InviteLink, font: Font.bold(17.0), textColor: self.presentationData.theme.actionSheet.primaryTextColor) + self.doneButton.setTitle(self.presentationData.strings.Common_Done, with: Font.bold(17.0), with: self.presentationData.theme.actionSheet.controlAccentColor, for: .normal) + } + + private func enqueueTransition(_ transition: InviteLinkInviteTransaction) { + self.enqueuedTransitions.append(transition) + + if let _ = self.validLayout { + while !self.enqueuedTransitions.isEmpty { + self.dequeueTransition() + } + } + } + + private func dequeueTransition() { + guard let _ = self.validLayout, let transition = self.enqueuedTransitions.first else { + return + } + self.enqueuedTransitions.remove(at: 0) + + self.listNode.transaction(deleteIndices: transition.deletions, insertIndicesAndItems: transition.insertions, updateIndicesAndItems: transition.updates, options: ListViewDeleteAndInsertOptions(), updateSizeAndInsets: nil, updateOpaqueState: nil, completion: { _ in + }) + } + + func animateIn() { + guard let layout = self.validLayout else { + return + } + let transition = ContainedViewLayoutTransition.animated(duration: 0.4, curve: .spring) + + let initialBounds = self.contentNode.bounds + self.contentNode.bounds = initialBounds.offsetBy(dx: 0.0, dy: -layout.size.height) + transition.animateView({ + self.contentNode.view.bounds = initialBounds + }) + self.dimNode.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.3) + } + + func animateOut(completion: (() -> Void)?) { + guard let layout = self.validLayout else { + return + } + var offsetCompleted = false + let internalCompletion: () -> Void = { + if offsetCompleted { + completion?() + } + } + + self.contentNode.layer.animateBoundsOriginYAdditive(from: self.contentNode.bounds.origin.y, to: -layout.size.height, duration: 0.3, timingFunction: kCAMediaTimingFunctionSpring, removeOnCompletion: false, completion: { _ in + offsetCompleted = true + internalCompletion() + }) + self.dimNode.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.2, removeOnCompletion: false) + } + + func containerLayoutUpdated(_ layout: ContainerViewLayout, transition: ContainedViewLayoutTransition) { + self.validLayout = layout + + transition.updateFrame(node: self.dimNode, frame: CGRect(origin: CGPoint(), size: layout.size)) + transition.updateFrame(node: self.contentNode, frame: CGRect(origin: CGPoint(), size: layout.size)) + + var insets = UIEdgeInsets() + insets.left = layout.safeInsets.left + insets.right = layout.safeInsets.right + insets.bottom = layout.intrinsicInsets.bottom + + let headerHeight: CGFloat = 54.0 + let visibleItemsHeight: CGFloat = 147.0 + floor(52.0 * 3.5) + + let layoutTopInset: CGFloat = max(layout.statusBarHeight ?? 0.0, layout.safeInsets.top) + + let listTopInset = layoutTopInset + headerHeight + let listNodeSize = CGSize(width: layout.size.width, height: layout.size.height - listTopInset) + + insets.top = max(0.0, listNodeSize.height - visibleItemsHeight) + + let (duration, curve) = listViewAnimationDurationAndCurve(transition: transition) + let updateSizeAndInsets = ListViewUpdateSizeAndInsets(size: listNodeSize, insets: insets, duration: duration, curve: curve) + self.listNode.transaction(deleteIndices: [], insertIndicesAndItems: [], updateIndicesAndItems: [], options: [.Synchronous, .LowLatency], scrollToItem: nil, updateSizeAndInsets: updateSizeAndInsets, stationaryItemRange: nil, updateOpaqueState: nil, completion: { _ in }) + + transition.updateFrame(node: self.listNode, frame: CGRect(origin: CGPoint(x: 0.0, y: listTopInset), size: listNodeSize)) + + transition.updateFrame(node: self.headerBackgroundNode, frame: CGRect(x: 0.0, y: 0.0, width: layout.size.width, height: 68.0)) + + let titleSize = self.titleNode.updateLayout(CGSize(width: layout.size.width, height: headerHeight)) + let titleFrame = CGRect(origin: CGPoint(x: floor((layout.size.width - titleSize.width) / 2.0), y: 18.0), size: titleSize) + transition.updateFrame(node: self.titleNode, frame: titleFrame) + + let doneSize = self.doneButton.measure(CGSize(width: layout.size.width, height: headerHeight)) + let doneFrame = CGRect(origin: CGPoint(x: layout.size.width - doneSize.width - 16.0, y: 18.0), size: doneSize) + transition.updateFrame(node: self.doneButton, frame: doneFrame) + } + + override func hitTest(_ point: CGPoint, with event: UIEvent?) -> UIView? { + let result = super.hitTest(point, with: event) + + if result === self.headerNode.view { + return self.view + } + + if result === self.headerNode.view { + return self.view + } + + if !self.bounds.contains(point) { + return nil + } + if point.y < self.headerNode.frame.minY { + return self.dimNode.view + } + return result + } + + @objc func dimTapGesture(_ recognizer: UITapGestureRecognizer) { + if case .ended = recognizer.state { + self.controller?.dismiss() + } + } + + private var panGestureArguments: CGFloat? + + @objc func panGesture(_ recognizer: UIPanGestureRecognizer) { + let contentOffset = self.listNode.visibleContentOffset() + switch recognizer.state { + case .began: + self.panGestureArguments = 0.0 + case .changed: + var translation = recognizer.translation(in: self.contentNode.view).y + if let currentPanOffset = self.panGestureArguments { + + + if case let .known(value) = contentOffset, value <= 0.5 { + } else { + translation = currentPanOffset + } + + self.panGestureArguments = translation + } + + var bounds = self.contentNode.bounds + bounds.origin.y = -translation + bounds.origin.y = min(0.0, bounds.origin.y) + self.contentNode.bounds = bounds + case .ended: + let translation = recognizer.translation(in: self.contentNode.view) + var velocity = recognizer.velocity(in: self.contentNode.view) + + if case let .known(value) = contentOffset, value > 0.0 { + velocity = CGPoint() + } else if case .unknown = contentOffset { + velocity = CGPoint() + } + + var bounds = self.contentNode.bounds + bounds.origin.y = -translation.y + bounds.origin.y = min(0.0, bounds.origin.y) + + self.panGestureArguments = nil + if bounds.minY < -60 || (bounds.minY < 0.0 && velocity.y > 300.0) { + self.controller?.dismiss() + } else { + var bounds = self.contentNode.bounds + let previousBounds = bounds + bounds.origin.y = 0.0 + self.contentNode.bounds = bounds + self.contentNode.layer.animateBounds(from: previousBounds, to: self.contentNode.bounds, duration: 0.3, timingFunction: CAMediaTimingFunctionName.easeInEaseOut.rawValue) + } + case .cancelled: + self.panGestureArguments = nil + + let previousBounds = self.contentNode.bounds + var bounds = self.contentNode.bounds + bounds.origin.y = 0.0 + self.contentNode.bounds = bounds + self.contentNode.layer.animateBounds(from: previousBounds, to: self.contentNode.bounds, duration: 0.3, timingFunction: CAMediaTimingFunctionName.easeInEaseOut.rawValue) + default: + break + } + } + + private func updateFloatingHeaderOffset(offset: CGFloat, transition: ContainedViewLayoutTransition) { + guard let validLayout = self.validLayout else { + return + } + + self.floatingHeaderOffset = offset + + let layoutTopInset: CGFloat = max(validLayout.statusBarHeight ?? 0.0, validLayout.safeInsets.top) + + let controlsHeight: CGFloat = 44.0 + + let listTopInset = layoutTopInset + controlsHeight + + let rawControlsOffset = offset + listTopInset - controlsHeight + let controlsOffset = max(layoutTopInset, rawControlsOffset) + let isOverscrolling = rawControlsOffset <= layoutTopInset + let controlsFrame = CGRect(origin: CGPoint(x: 0.0, y: controlsOffset), size: CGSize(width: validLayout.size.width, height: controlsHeight)) + + let previousFrame = self.headerNode.frame + + if !controlsFrame.equalTo(previousFrame) { + self.headerNode.frame = controlsFrame + + let positionDelta = CGPoint(x: controlsFrame.minX - previousFrame.minX, y: controlsFrame.minY - previousFrame.minY) + + transition.animateOffsetAdditive(node: self.headerNode, offset: positionDelta.y) + } + +// transition.updateAlpha(node: self.headerNode.separatorNode, alpha: isOverscrolling ? 1.0 : 0.0) + + let backgroundFrame = CGRect(origin: CGPoint(x: 0.0, y: controlsFrame.maxY), size: CGSize(width: validLayout.size.width, height: validLayout.size.height)) + + let previousBackgroundFrame = self.historyBackgroundNode.frame + + if !backgroundFrame.equalTo(previousBackgroundFrame) { + self.historyBackgroundNode.frame = backgroundFrame + self.historyBackgroundContentNode.frame = CGRect(origin: CGPoint(), size: backgroundFrame.size) + + let positionDelta = CGPoint(x: backgroundFrame.minX - previousBackgroundFrame.minX, y: backgroundFrame.minY - previousBackgroundFrame.minY) + + transition.animateOffsetAdditive(node: self.historyBackgroundNode, offset: positionDelta.y) + } + } + } +} diff --git a/submodules/InviteLinksUI/Sources/InviteLinkInviteHeaderItem.swift b/submodules/InviteLinksUI/Sources/InviteLinkInviteHeaderItem.swift new file mode 100644 index 0000000000..2030a9a011 --- /dev/null +++ b/submodules/InviteLinksUI/Sources/InviteLinkInviteHeaderItem.swift @@ -0,0 +1,124 @@ +import Foundation +import UIKit +import Display +import AsyncDisplayKit +import SwiftSignalKit +import TelegramPresentationData +import ItemListUI +import PresentationDataUtils +import AnimatedStickerNode +import AppBundle + +class InviteLinkInviteHeaderItem: ListViewItem, ItemListItem { + var sectionId: ItemListSectionId = 0 + + let theme: PresentationTheme + let text: String + + init(theme: PresentationTheme, text: String) { + self.theme = theme + self.text = text + } + + func nodeConfiguredForParams(async: @escaping (@escaping () -> Void) -> Void, params: ListViewItemLayoutParams, synchronousLoads: Bool, previousItem: ListViewItem?, nextItem: ListViewItem?, completion: @escaping (ListViewItemNode, @escaping () -> (Signal?, (ListViewItemApply) -> Void)) -> Void) { + async { + let node = InviteLinkInviteHeaderItemNode() + let (layout, apply) = node.asyncLayout()(self, params, itemListNeighbors(item: self, topItem: previousItem as? ItemListItem, bottomItem: nextItem as? ItemListItem)) + + node.contentSize = layout.contentSize + node.insets = layout.insets + + Queue.mainQueue().async { + completion(node, { + return (nil, { _ in apply() }) + }) + } + } + } + + func updateNode(async: @escaping (@escaping () -> Void) -> Void, node: @escaping () -> ListViewItemNode, params: ListViewItemLayoutParams, previousItem: ListViewItem?, nextItem: ListViewItem?, animation: ListViewItemUpdateAnimation, completion: @escaping (ListViewItemNodeLayout, @escaping (ListViewItemApply) -> Void) -> Void) { + Queue.mainQueue().async { + guard let nodeValue = node() as? InviteLinkInviteHeaderItemNode else { + assertionFailure() + return + } + + let makeLayout = nodeValue.asyncLayout() + + async { + let (layout, apply) = makeLayout(self, params, itemListNeighbors(item: self, topItem: previousItem as? ItemListItem, bottomItem: nextItem as? ItemListItem)) + Queue.mainQueue().async { + completion(layout, { _ in + apply() + }) + } + } + } + } +} + +private let titleFont = Font.regular(13.0) + +class InviteLinkInviteHeaderItemNode: ListViewItemNode { + private let titleNode: TextNode + private var animationNode: AnimatedStickerNode + + private var item: InviteLinkInviteHeaderItem? + + init() { + self.titleNode = TextNode() + self.titleNode.isUserInteractionEnabled = false + self.titleNode.contentMode = .left + self.titleNode.contentsScale = UIScreen.main.scale + + self.animationNode = AnimatedStickerNode() + if let path = getAppBundle().path(forResource: "Invite", ofType: "tgs") { + self.animationNode.setup(source: AnimatedStickerNodeLocalFileSource(path: path), width: 192, height: 192, playbackMode: .loop, mode: .direct(cachePathPrefix: nil)) + self.animationNode.visibility = true + } + + super.init(layerBacked: false, dynamicBounce: false) + + self.addSubnode(self.titleNode) + self.addSubnode(self.animationNode) + } + + func asyncLayout() -> (_ item: InviteLinkInviteHeaderItem, _ params: ListViewItemLayoutParams, _ neighbors: ItemListNeighbors) -> (ListViewItemNodeLayout, () -> Void) { + let makeTitleLayout = TextNode.asyncLayout(self.titleNode) + + return { item, params, neighbors in + let leftInset: CGFloat = 32.0 + params.leftInset + let topInset: CGFloat = 92.0 + + let attributedText = NSAttributedString(string: item.text, font: titleFont, textColor: item.theme.list.freeTextColor) + let (titleLayout, titleApply) = makeTitleLayout(TextNodeLayoutArguments(attributedString: attributedText, backgroundColor: nil, maximumNumberOfLines: 0, truncationType: .end, constrainedSize: CGSize(width: params.width - params.rightInset - leftInset * 2.0, height: CGFloat.greatestFiniteMagnitude), alignment: .center, cutout: nil, insets: UIEdgeInsets())) + + let contentSize = CGSize(width: params.width, height: topInset + titleLayout.size.height) + let insets = itemListNeighborsGroupedInsets(neighbors) + + let layout = ListViewItemNodeLayout(contentSize: contentSize, insets: insets) + + return (layout, { [weak self] in + if let strongSelf = self { + strongSelf.item = item + strongSelf.accessibilityLabel = attributedText.string + + let iconSize = CGSize(width: 96.0, height: 96.0) + strongSelf.animationNode.frame = CGRect(origin: CGPoint(x: floor((layout.size.width - iconSize.width) / 2.0), y: -10.0), size: iconSize) + strongSelf.animationNode.updateLayout(size: iconSize) + + let _ = titleApply() + strongSelf.titleNode.frame = CGRect(origin: CGPoint(x: floor((layout.size.width - titleLayout.size.width) / 2.0), y: topInset + 8.0), size: titleLayout.size) + } + }) + } + } + + override func animateInsertion(_ currentTimestamp: Double, duration: Double, short: Bool) { + self.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.4) + } + + override func animateRemoved(_ currentTimestamp: Double, duration: Double) { + self.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.15, removeOnCompletion: false) + } +} diff --git a/submodules/InviteLinksUI/Sources/InviteLinkListController.swift b/submodules/InviteLinksUI/Sources/InviteLinkListController.swift new file mode 100644 index 0000000000..cfb1615392 --- /dev/null +++ b/submodules/InviteLinksUI/Sources/InviteLinkListController.swift @@ -0,0 +1,619 @@ +import Foundation +import UIKit +import AsyncDisplayKit +import Display +import SwiftSignalKit +import Postbox +import TelegramCore +import SyncCore +import TelegramPresentationData +import TelegramUIPreferences +import ItemListUI +import PresentationDataUtils +import OverlayStatusController +import AccountContext +import AlertUI +import PresentationDataUtils +import AppBundle +import ContextUI +import TelegramStringFormatting +import ItemListPeerActionItem +import ShareController + +private final class InviteLinkListControllerArguments { + let context: AccountContext + let shareMainLink: (ExportedInvitation?) -> Void + let openMainLink: (ExportedInvitation?) -> Void + let mainLinkContextAction: (ExportedInvitation?, ASDisplayNode, ContextGesture?) -> Void + let createLink: () -> Void + let openLink: (ExportedInvitation) -> Void + let linkContextAction: (ExportedInvitation?, ASDisplayNode, ContextGesture?) -> Void + let deleteAllRevokedLinks: () -> Void + + init(context: AccountContext, shareMainLink: @escaping (ExportedInvitation?) -> Void, openMainLink: @escaping (ExportedInvitation?) -> Void, mainLinkContextAction: @escaping (ExportedInvitation?, ASDisplayNode, ContextGesture?) -> Void, createLink: @escaping () -> Void, openLink: @escaping (ExportedInvitation?) -> Void, linkContextAction: @escaping (ExportedInvitation?, ASDisplayNode, ContextGesture?) -> Void, deleteAllRevokedLinks: @escaping () -> Void) { + self.context = context + self.shareMainLink = shareMainLink + self.openMainLink = openMainLink + self.mainLinkContextAction = mainLinkContextAction + self.createLink = createLink + self.openLink = openLink + self.linkContextAction = linkContextAction + self.deleteAllRevokedLinks = deleteAllRevokedLinks + } +} + +private enum InviteLinksListSection: Int32 { + case header + case mainLink + case links + case revokedLinks +} + +private enum InviteLinksListEntry: ItemListNodeEntry { + case header(PresentationTheme, String) + + case mainLinkHeader(PresentationTheme, String) + case mainLink(PresentationTheme, ExportedInvitation?, [Peer]) + + case linksHeader(PresentationTheme, String) + case linksCreate(PresentationTheme, String) + case links(Int32, PresentationTheme, [ExportedInvitation]?) + case linksInfo(PresentationTheme, String) + case revokedLinksHeader(PresentationTheme, String) + case revokedLinksDeleteAll(PresentationTheme, String) + case revokedLinks(Int32, PresentationTheme, [ExportedInvitation]?) + + var section: ItemListSectionId { + switch self { + case .header: + return InviteLinksListSection.header.rawValue + case .mainLinkHeader, .mainLink: + return InviteLinksListSection.mainLink.rawValue + case .linksHeader, .linksCreate, .links, .linksInfo: + return InviteLinksListSection.links.rawValue + case .revokedLinksHeader, .revokedLinksDeleteAll, .revokedLinks: + return InviteLinksListSection.revokedLinks.rawValue + } + } + + var stableId: Int32 { + switch self { + case .header: + return 0 + case .mainLinkHeader: + return 1 + case .mainLink: + return 2 + case .linksHeader: + return 3 + case .linksCreate: + return 4 + case let .links(index, _, _): + return 5 + index + case .linksInfo: + return 10000 + case .revokedLinksHeader: + return 10001 + case .revokedLinksDeleteAll: + return 10002 + case let .revokedLinks(index, _, _): + return 10003 + index + } + } + + static func ==(lhs: InviteLinksListEntry, rhs: InviteLinksListEntry) -> Bool { + switch lhs { + case let .header(lhsTheme, lhsText): + if case let .header(rhsTheme, rhsText) = rhs, lhsTheme === rhsTheme, lhsText == rhsText { + return true + } else { + return false + } + case let .mainLinkHeader(lhsTheme, lhsText): + if case let .mainLinkHeader(rhsTheme, rhsText) = rhs, lhsTheme === rhsTheme, lhsText == rhsText { + return true + } else { + return false + } + case let .mainLink(lhsTheme, lhsInvite, lhsPeers): + if case let .mainLink(rhsTheme, rhsInvite, rhsPeers) = rhs, lhsTheme === rhsTheme, lhsInvite == rhsInvite, arePeerArraysEqual(lhsPeers, rhsPeers) { + return true + } else { + return false + } + case let .linksHeader(lhsTheme, lhsText): + if case let .linksHeader(rhsTheme, rhsText) = rhs, lhsTheme === rhsTheme, lhsText == rhsText { + return true + } else { + return false + } + case let .linksCreate(lhsTheme, lhsText): + if case let .linksCreate(rhsTheme, rhsText) = rhs, lhsTheme === rhsTheme, lhsText == rhsText { + return true + } else { + return false + } + case let .links(lhsIndex, lhsTheme, lhsLinks): + if case let .links(rhsIndex, rhsTheme, rhsLinks) = rhs, lhsIndex == rhsIndex, lhsTheme === rhsTheme, lhsLinks == rhsLinks { + return true + } else { + return false + } + case let .linksInfo(lhsTheme, lhsText): + if case let .linksInfo(rhsTheme, rhsText) = rhs, lhsTheme === rhsTheme, lhsText == rhsText { + return true + } else { + return false + } + case let .revokedLinksHeader(lhsTheme, lhsText): + if case let .revokedLinksHeader(rhsTheme, rhsText) = rhs, lhsTheme === rhsTheme, lhsText == rhsText { + return true + } else { + return false + } + case let .revokedLinksDeleteAll(lhsTheme, lhsText): + if case let .revokedLinksDeleteAll(rhsTheme, rhsText) = rhs, lhsTheme === rhsTheme, lhsText == rhsText { + return true + } else { + return false + } + case let .revokedLinks(lhsIndex, lhsTheme, lhsLinks): + if case let .revokedLinks(rhsIndex, rhsTheme, rhsLinks) = rhs, lhsIndex == rhsIndex, lhsTheme === rhsTheme, lhsLinks == rhsLinks { + return true + } else { + return false + } + } + } + + static func <(lhs: InviteLinksListEntry, rhs: InviteLinksListEntry) -> Bool { + return lhs.stableId < rhs.stableId + } + + func item(presentationData: ItemListPresentationData, arguments: Any) -> ListViewItem { + let arguments = arguments as! InviteLinkListControllerArguments + switch self { + case let .header(theme, text): + return InviteLinkHeaderItem(theme: theme, text: text, sectionId: self.section) + case let .mainLinkHeader(_, text): + return ItemListSectionHeaderItem(presentationData: presentationData, text: text, sectionId: self.section) + case let .mainLink(_, invite, peers): + return ItemListPermanentInviteLinkItem(context: arguments.context, presentationData: presentationData, invite: invite, peers: peers, buttonColor: nil, sectionId: self.section, style: .blocks, shareAction: { + arguments.shareMainLink(invite) + }, contextAction: { node in + arguments.mainLinkContextAction(invite, node, nil) + }, viewAction: { + if let invite = invite { + arguments.openLink(invite) + } + }) + case let .linksHeader(_, text): + return ItemListSectionHeaderItem(presentationData: presentationData, text: text, sectionId: self.section) + case let .linksCreate(theme, text): + return ItemListPeerActionItem(presentationData: presentationData, icon: PresentationResourcesItemList.plusIconImage(theme), title: text, hasSeparator: false, sectionId: self.section, editing: false, action: { + arguments.createLink() + }) + case let .links(_, _, invites): + return ItemListInviteLinkGridItem(presentationData: presentationData, invites: invites, sectionId: self.section, style: .blocks, tapAction: { invite in + arguments.openLink(invite) + }, contextAction: { invite, node in + arguments.linkContextAction(invite, node, nil) + }) + case let .linksInfo(_, text): + return ItemListTextItem(presentationData: presentationData, text: .plain(text), sectionId: self.section) + case let .revokedLinksHeader(_, text): + return ItemListSectionHeaderItem(presentationData: presentationData, text: text, sectionId: self.section) + case let .revokedLinksDeleteAll(theme, text): + return ItemListPeerActionItem(presentationData: presentationData, icon: PresentationResourcesItemList.deleteIconImage(theme), title: text, hasSeparator: false, sectionId: self.section, color: .destructive, editing: false, action: { + arguments.deleteAllRevokedLinks() + }) + case let .revokedLinks(_, _, invites): + return ItemListInviteLinkGridItem(presentationData: presentationData, invites: invites, sectionId: self.section, style: .blocks, tapAction: { invite in + arguments.openLink(invite) + }, contextAction: { invite, node in + arguments.linkContextAction(invite, node, nil) + }) + } + } +} + +private func inviteLinkListControllerEntries(presentationData: PresentationData, view: PeerView, invites: [ExportedInvitation]?, revokedInvites: [ExportedInvitation]?, mainPeers: [Peer]) -> [InviteLinksListEntry] { + var entries: [InviteLinksListEntry] = [] + + entries.append(.header(presentationData.theme, presentationData.strings.InviteLink_CreatePrivateLinkHelp)) + entries.append(.mainLinkHeader(presentationData.theme, presentationData.strings.InviteLink_PermanentLink.uppercased())) + + let mainInvite: ExportedInvitation? + if let invites = invites, let invite = invites.first(where: { $0.isPermanent && !$0.isRevoked }) { + mainInvite = invite + } else if let invite = (view.cachedData as? CachedChannelData)?.exportedInvitation { + mainInvite = invite + } else if let invite = (view.cachedData as? CachedGroupData)?.exportedInvitation { + mainInvite = invite + } else { + mainInvite = nil + } + entries.append(.mainLink(presentationData.theme, mainInvite, mainPeers)) + + entries.append(.linksHeader(presentationData.theme, presentationData.strings.InviteLink_AdditionalLinks.uppercased())) + entries.append(.linksCreate(presentationData.theme, presentationData.strings.InviteLink_Create)) + + var additionalInvites: [ExportedInvitation]? + if let invites = invites { + additionalInvites = invites.filter { $0.link != mainInvite?.link } + } + if let additionalInvites = additionalInvites { + var index: Int32 = 0 + for i in stride(from: 0, to: additionalInvites.endIndex, by: 2) { + var invitesPair: [ExportedInvitation] = [] + invitesPair.append(additionalInvites[i]) + if i + 1 < additionalInvites.count { + invitesPair.append(additionalInvites[i + 1]) + } + entries.append(.links(index, presentationData.theme, invitesPair)) + index += 1 + } + } + entries.append(.linksInfo(presentationData.theme, presentationData.strings.InviteLink_CreateInfo)) + + if let revokedInvites = revokedInvites, !revokedInvites.isEmpty { + entries.append(.revokedLinksHeader(presentationData.theme, presentationData.strings.InviteLink_RevokedLinks.uppercased())) + entries.append(.revokedLinksDeleteAll(presentationData.theme, presentationData.strings.InviteLink_DeleteAllRevokedLinks)) + + var index: Int32 = 0 + for i in stride(from: 0, to: revokedInvites.endIndex, by: 2) { + var invitesPair: [ExportedInvitation] = [] + invitesPair.append(revokedInvites[i]) + if i + 1 < revokedInvites.count { + invitesPair.append(revokedInvites[i + 1]) + } + entries.append(.revokedLinks(index, presentationData.theme, invitesPair)) + index += 1 + } + } + + return entries +} + +private struct InviteLinkListControllerState: Equatable { + var revokingPrivateLink: Bool +} + + +public func inviteLinkListController(context: AccountContext, peerId: PeerId) -> ViewController { + var pushControllerImpl: ((ViewController) -> Void)? + var presentControllerImpl: ((ViewController, ViewControllerPresentationArguments?) -> Void)? + var presentInGlobalOverlayImpl: ((ViewController) -> Void)? + + let actionsDisposable = DisposableSet() + + let statePromise = ValuePromise(InviteLinkListControllerState(revokingPrivateLink: false), ignoreRepeated: true) + let stateValue = Atomic(value: InviteLinkListControllerState(revokingPrivateLink: false)) + let updateState: ((InviteLinkListControllerState) -> InviteLinkListControllerState) -> Void = { f in + statePromise.set(stateValue.modify { f($0) }) + } + + let revokeLinkDisposable = MetaDisposable() + actionsDisposable.add(revokeLinkDisposable) + + let deleteAllRevokedLinksDisposable = MetaDisposable() + actionsDisposable.add(deleteAllRevokedLinksDisposable) + + actionsDisposable.add((context.account.viewTracker.peerView(peerId) |> filter { $0.cachedData != nil } |> take(1) |> mapToSignal { view -> Signal in + return ensuredExistingPeerExportedInvitation(account: context.account, peerId: peerId) + |> mapToSignal { _ -> Signal in + return .complete() + } + }).start()) + + var getControllerImpl: (() -> ViewController?)? + + let invitesPromise = Promise() + invitesPromise.set(.single(nil) + |> then(peerExportedInvitations(account: context.account, peerId: peerId, revoked: false))) + + let revokedInvitesPromise = Promise() + revokedInvitesPromise.set(.single(nil) + |> then(peerExportedInvitations(account: context.account, peerId: peerId, revoked: true))) + + let arguments = InviteLinkListControllerArguments(context: context, shareMainLink: { invite in + if let invite = invite { + let shareController = ShareController(context: context, subject: .url(invite.link)) + presentControllerImpl?(shareController, nil) + } + }, openMainLink: { invite in + if let invite = invite { + let controller = InviteLinkViewController(context: context, peerId: peerId, invite: invite, importersContext: nil) + pushControllerImpl?(controller) + } + }, mainLinkContextAction: { invite, node, gesture in + guard let node = node as? ContextExtractedContentContainingNode, let controller = getControllerImpl?() else { + return + } + let presentationData = context.sharedContext.currentPresentationData.with { $0 } + var items: [ContextMenuItem] = [] + + items.append(.action(ContextMenuActionItem(text: presentationData.strings.InviteLink_ContextCopy, icon: { theme in + return generateTintedImage(image: UIImage(bundleImageName: "Chat/Context Menu/Copy"), color: theme.contextMenu.primaryColor) + }, action: { _, f in + f(.dismissWithoutContent) + + if let invite = invite { + UIPasteboard.general.string = invite.link + let presentationData = context.sharedContext.currentPresentationData.with { $0 } + presentControllerImpl?(OverlayStatusController(theme: presentationData.theme, type: .genericSuccess(presentationData.strings.Username_LinkCopied, false)), nil) + } + }))) + + items.append(.action(ContextMenuActionItem(text: presentationData.strings.InviteLink_ContextGetQRCode, icon: { theme in + return generateTintedImage(image: UIImage(bundleImageName: "Wallet/QrIcon"), color: theme.contextMenu.primaryColor) + }, action: { _, f in + f(.dismissWithoutContent) + + if let invite = invite { + let controller = InviteLinkQRCodeController(context: context, invite: invite) + presentControllerImpl?(controller, nil) + } + }))) + + items.append(.action(ContextMenuActionItem(text: presentationData.strings.InviteLink_ContextRevoke, textColor: .destructive, icon: { theme in + return generateTintedImage(image: UIImage(bundleImageName: "Chat/Context Menu/Delete"), color: theme.actionSheet.destructiveActionTextColor) + }, action: { _, f in + f(.dismissWithoutContent) + + let controller = ActionSheetController(presentationData: presentationData) + let dismissAction: () -> Void = { [weak controller] in + controller?.dismissAnimated() + } + controller.setItemGroups([ + ActionSheetItemGroup(items: [ + ActionSheetTextItem(title: presentationData.strings.GroupInfo_InviteLink_RevokeAlert_Text), + ActionSheetButtonItem(title: presentationData.strings.GroupInfo_InviteLink_RevokeLink, color: .destructive, action: { + dismissAction() + + var revoke = false + updateState { state in + if !state.revokingPrivateLink { + revoke = true + var updatedState = state + updatedState.revokingPrivateLink = true + return updatedState + } else { + return state + } + } + if revoke { + revokeLinkDisposable.set((ensuredExistingPeerExportedInvitation(account: context.account, peerId: peerId, revokeExisted: true) |> deliverOnMainQueue).start(completed: { + updateState { state in + var updatedState = state + updatedState.revokingPrivateLink = false + return updatedState + } + invitesPromise.set(peerExportedInvitations(account: context.account, peerId: peerId, revoked: false)) + })) + } + }) + ]), + ActionSheetItemGroup(items: [ActionSheetButtonItem(title: presentationData.strings.Common_Cancel, action: { dismissAction() })]) + ]) + presentControllerImpl?(controller, ViewControllerPresentationArguments(presentationAnimation: .modalSheet)) + }))) + + let contextController = ContextController(account: context.account, presentationData: presentationData, source: .extracted(InviteLinkContextExtractedContentSource(controller: controller, sourceNode: node)), items: .single(items), reactionItems: [], gesture: gesture) + presentInGlobalOverlayImpl?(contextController) + }, createLink: { + let controller = inviteLinkEditController(context: context, peerId: peerId, invite: nil, completion: { + invitesPromise.set(peerExportedInvitations(account: context.account, peerId: peerId, revoked: false)) + }) + controller.navigationPresentation = .modal + pushControllerImpl?(controller) + }, openLink: { invite in + if let invite = invite { + let controller = InviteLinkViewController(context: context, peerId: peerId, invite: invite, importersContext: nil) + pushControllerImpl?(controller) + } + }, linkContextAction: { invite, node, gesture in + guard let node = node as? ContextExtractedContentContainingNode, let controller = getControllerImpl?(), let invite = invite else { + return + } + + let presentationData = context.sharedContext.currentPresentationData.with { $0 } + var items: [ContextMenuItem] = [] + + items.append(.action(ContextMenuActionItem(text: presentationData.strings.InviteLink_ContextCopy, icon: { theme in + return generateTintedImage(image: UIImage(bundleImageName: "Chat/Context Menu/Copy"), color: theme.contextMenu.primaryColor) + }, action: { _, f in + f(.dismissWithoutContent) + + UIPasteboard.general.string = invite.link + let presentationData = context.sharedContext.currentPresentationData.with { $0 } + presentControllerImpl?(OverlayStatusController(theme: presentationData.theme, type: .genericSuccess(presentationData.strings.Username_LinkCopied, false)), nil) + }))) + + if !invite.isRevoked { + items.append(.action(ContextMenuActionItem(text: presentationData.strings.InviteLink_ContextGetQRCode, icon: { theme in + return generateTintedImage(image: UIImage(bundleImageName: "Wallet/QrIcon"), color: theme.contextMenu.primaryColor) + }, action: { _, f in + f(.dismissWithoutContent) + + let controller = InviteLinkQRCodeController(context: context, invite: invite) + presentControllerImpl?(controller, nil) + }))) + + items.append(.action(ContextMenuActionItem(text: presentationData.strings.InviteLink_ContextEdit, icon: { theme in + return generateTintedImage(image: UIImage(bundleImageName: "Chat/Context Menu/Edit"), color: theme.contextMenu.primaryColor) + }, action: { _, f in + f(.dismissWithoutContent) + + let controller = inviteLinkEditController(context: context, peerId: peerId, invite: invite, completion: { + invitesPromise.set(peerExportedInvitations(account: context.account, peerId: peerId, revoked: false)) + }) + controller.navigationPresentation = .modal + pushControllerImpl?(controller) + }))) + } + + if invite.isRevoked { + items.append(.action(ContextMenuActionItem(text: presentationData.strings.InviteLink_ContextDelete, textColor: .destructive, icon: { theme in + return generateTintedImage(image: UIImage(bundleImageName: "Chat/Context Menu/Delete"), color: theme.actionSheet.destructiveActionTextColor) + }, action: { _, f in + f(.dismissWithoutContent) + + let controller = ActionSheetController(presentationData: presentationData) + let dismissAction: () -> Void = { [weak controller] in + controller?.dismissAnimated() + } + controller.setItemGroups([ + ActionSheetItemGroup(items: [ + ActionSheetTextItem(title: presentationData.strings.GroupInfo_InviteLink_RevokeAlert_Text), + ActionSheetButtonItem(title: presentationData.strings.GroupInfo_InviteLink_RevokeLink, color: .destructive, action: { + dismissAction() + + revokeLinkDisposable.set((revokePeerExportedInvitation(account: context.account, peerId: peerId, link: invite.link) |> deliverOnMainQueue).start(completed: { + + })) + }) + ]), + ActionSheetItemGroup(items: [ActionSheetButtonItem(title: presentationData.strings.Common_Cancel, action: { dismissAction() })]) + ]) + presentControllerImpl?(controller, ViewControllerPresentationArguments(presentationAnimation: .modalSheet)) + }))) + } else { + items.append(.action(ContextMenuActionItem(text: presentationData.strings.InviteLink_ContextRevoke, textColor: .destructive, icon: { theme in + return generateTintedImage(image: UIImage(bundleImageName: "Chat/Context Menu/Delete"), color: theme.actionSheet.destructiveActionTextColor) + }, action: { _, f in + f(.dismissWithoutContent) + + let controller = ActionSheetController(presentationData: presentationData) + let dismissAction: () -> Void = { [weak controller] in + controller?.dismissAnimated() + } + controller.setItemGroups([ + ActionSheetItemGroup(items: [ + ActionSheetTextItem(title: presentationData.strings.GroupInfo_InviteLink_RevokeAlert_Text), + ActionSheetButtonItem(title: presentationData.strings.GroupInfo_InviteLink_RevokeLink, color: .destructive, action: { + dismissAction() + + revokeLinkDisposable.set((revokePeerExportedInvitation(account: context.account, peerId: peerId, link: invite.link) |> deliverOnMainQueue).start(completed: { + + })) + }) + ]), + ActionSheetItemGroup(items: [ActionSheetButtonItem(title: presentationData.strings.Common_Cancel, action: { dismissAction() })]) + ]) + presentControllerImpl?(controller, ViewControllerPresentationArguments(presentationAnimation: .modalSheet)) + }))) + } + + let contextController = ContextController(account: context.account, presentationData: presentationData, source: .extracted(InviteLinkContextExtractedContentSource(controller: controller, sourceNode: node)), items: .single(items), reactionItems: [], gesture: gesture) + presentInGlobalOverlayImpl?(contextController) + }, deleteAllRevokedLinks: { + 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.InviteLink_DeleteAllRevokedLinksAlert_Text), + ActionSheetButtonItem(title: presentationData.strings.InviteLink_DeleteAllRevokedLinksAlert_Action, color: .destructive, action: { + dismissAction() + + deleteAllRevokedLinksDisposable.set((deleteAllRevokedPeerExportedInvitations(account: context.account, peerId: peerId) |> deliverOnMainQueue).start(completed: { + + })) + }) + ]), + ActionSheetItemGroup(items: [ActionSheetButtonItem(title: presentationData.strings.Common_Cancel, action: { dismissAction() })]) + ]) + presentControllerImpl?(controller, ViewControllerPresentationArguments(presentationAnimation: .modalSheet)) + }) + + let peerView = context.account.viewTracker.peerView(peerId) + |> deliverOnMainQueue + + let importersState = Promise(nil) + let importersContext: Signal = peerView + |> mapToSignal { view -> Signal in + if let cachedData = view.cachedData as? CachedGroupData, let exportedInvitation = cachedData.exportedInvitation { + return .single(exportedInvitation) + } else if let cachedData = view.cachedData as? CachedChannelData, let exportedInvitation = cachedData.exportedInvitation { + return .single(exportedInvitation) + } else { + return .complete() + } + } + |> distinctUntilChanged + |> deliverOnMainQueue + |> map { invite -> PeerInvitationImportersContext in + return PeerInvitationImportersContext(account: context.account, peerId: peerId, invite: invite) + } |> afterNext { context in + importersState.set(context.state |> map(Optional.init)) + } + + let signal = combineLatest(context.sharedContext.presentationData, peerView, importersContext, importersState.get(), invitesPromise.get(), revokedInvitesPromise.get()) + |> deliverOnMainQueue + |> map { presentationData, view, importersContext, importers, invites, revokedInvites -> (ItemListControllerState, (ItemListNodeState, Any)) in + let controllerState = ItemListControllerState(presentationData: ItemListPresentationData(presentationData), title: .text(presentationData.strings.InviteLink_Title), leftNavigationButton: nil, rightNavigationButton: nil, backNavigationButton: ItemListBackButton(title: presentationData.strings.Common_Back), animateChanges: true) + let listState = ItemListNodeState(presentationData: ItemListPresentationData(presentationData), entries: inviteLinkListControllerEntries(presentationData: presentationData, view: view, invites: invites?.list, revokedInvites: revokedInvites?.list, mainPeers: importers?.importers.compactMap { $0.peer.peer } ?? []), style: .blocks, emptyStateItem: nil, crossfadeState: false, animateChanges: false) + + return (controllerState, (listState, arguments)) + } + |> afterDisposed { + actionsDisposable.dispose() + } + + let controller = ItemListController(context: context, state: signal) + controller.didDisappear = { [weak controller] _ in + controller?.clearItemNodesHighlight(animated: true) + } + controller.visibleBottomContentOffsetChanged = { offset in + if case let .known(value) = offset, value < 40.0 { + + } + } + pushControllerImpl = { [weak controller] c in + if let controller = controller { + (controller.navigationController as? NavigationController)?.pushViewController(c, animated: true) + } + } + presentControllerImpl = { [weak controller] c, p in + if let controller = controller { + controller.present(c, in: .window(.root), with: p) + } + } + presentInGlobalOverlayImpl = { [weak controller] c in + if let controller = controller { + controller.presentInGlobalOverlay(c) + } + } + getControllerImpl = { [weak controller] in + return controller + } + return controller +} + + +final class InviteLinkContextExtractedContentSource: ContextExtractedContentSource { + var keepInPlace: Bool + let ignoreContentTouches: Bool = true + let blurBackground: Bool + + private let controller: ViewController + private let sourceNode: ContextExtractedContentContainingNode + + init(controller: ViewController, sourceNode: ContextExtractedContentContainingNode) { + self.controller = controller + self.sourceNode = sourceNode + self.keepInPlace = true + self.blurBackground = false + } + + func takeView() -> ContextControllerTakeViewInfo? { + return ContextControllerTakeViewInfo(contentContainingNode: self.sourceNode, contentAreaInScreenSpace: UIScreen.main.bounds) + } + + func putBack() -> ContextControllerPutBackViewInfo? { + return ContextControllerPutBackViewInfo(contentAreaInScreenSpace: UIScreen.main.bounds) + } +} diff --git a/submodules/InviteLinksUI/Sources/InviteLinkQRCodeController.swift b/submodules/InviteLinksUI/Sources/InviteLinkQRCodeController.swift new file mode 100644 index 0000000000..7dc743a424 --- /dev/null +++ b/submodules/InviteLinksUI/Sources/InviteLinkQRCodeController.swift @@ -0,0 +1,414 @@ +import Foundation +import UIKit +import SwiftSignalKit +import TelegramPresentationData +import AppBundle +import AsyncDisplayKit +import SyncCore +import Display +import QrCode +import AccountContext +import SolidRoundedButtonNode +import AnimatedStickerNode + +private func shareQrCode(context: AccountContext, link: String) { + let _ = (qrCode(string: link, color: .black, backgroundColor: .white, icon: .custom(UIImage(bundleImageName: "Chat/Links/QrLogo"))) + |> map { _, generator -> UIImage? in + let imageSize = CGSize(width: 768.0, height: 768.0) + let context = generator(TransformImageArguments(corners: ImageCorners(), imageSize: imageSize, boundingSize: imageSize, intrinsicInsets: UIEdgeInsets(), scale: 1.0)) + return context?.generateImage() + } + |> deliverOnMainQueue).start(next: { image in + guard let image = image else { + return + } + + let activityController = UIActivityViewController(activityItems: [image], applicationActivities: nil) + context.sharedContext.applicationBindings.presentNativeController(activityController) + }) +} + +public final class InviteLinkQRCodeController: ViewController { + private var controllerNode: Node { + return self.displayNode as! Node + } + + private var animatedIn = false + + private let context: AccountContext + private let invite: ExportedInvitation + + private var presentationDataDisposable: Disposable? + + private let idleTimerExtensionDisposable = MetaDisposable() + + public init(context: AccountContext, invite: ExportedInvitation) { + self.context = context + self.invite = invite + + super.init(navigationBarPresentationData: nil) + + self.statusBar.statusBarStyle = .Ignore + + self.blocksBackgroundWhenInOverlay = true + + self.presentationDataDisposable = (context.sharedContext.presentationData + |> deliverOnMainQueue).start(next: { [weak self] presentationData in + if let strongSelf = self { + strongSelf.controllerNode.updatePresentationData(presentationData) + } + }) + + self.idleTimerExtensionDisposable.set(self.context.sharedContext.applicationBindings.pushIdleTimerExtension()) + + self.statusBar.statusBarStyle = .Ignore + } + + required init(coder aDecoder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + deinit { + self.presentationDataDisposable?.dispose() + self.idleTimerExtensionDisposable.dispose() + } + + override public func loadDisplayNode() { + self.displayNode = Node(context: self.context, invite: self.invite) + self.controllerNode.dismiss = { [weak self] in + self?.presentingViewController?.dismiss(animated: false, completion: nil) + } + self.controllerNode.cancel = { [weak self] in + self?.dismiss() + } + } + + override public func loadView() { + super.loadView() + } + + override public func viewDidAppear(_ animated: Bool) { + super.viewDidAppear(animated) + + if !self.animatedIn { + self.animatedIn = true + self.controllerNode.animateIn() + } + } + + override public func dismiss(completion: (() -> Void)? = nil) { + self.controllerNode.animateOut(completion: completion) + } + + override public func containerLayoutUpdated(_ layout: ContainerViewLayout, transition: ContainedViewLayoutTransition) { + super.containerLayoutUpdated(layout, transition: transition) + + self.controllerNode.containerLayoutUpdated(layout, navigationBarHeight: self.navigationHeight, transition: transition) + } + + class Node: ViewControllerTracingNode, UIScrollViewDelegate { + private let context: AccountContext + private let invite: ExportedInvitation + private var presentationData: PresentationData + + private let dimNode: ASDisplayNode + private let wrappingScrollNode: ASScrollNode + private let contentContainerNode: ASDisplayNode + private let backgroundNode: ASDisplayNode + private let contentBackgroundNode: ASDisplayNode + private let titleNode: ASTextNode + private let subtitleNode: ASTextNode + private let cancelButton: HighlightableButtonNode + + private let textNode: ImmediateTextNode + private let qrButtonNode: HighlightTrackingButtonNode + private let qrImageNode: TransformImageNode + private let qrIconNode: AnimatedStickerNode + private var qrCodeSize: Int? + private let buttonNode: SolidRoundedButtonNode + + private var containerLayout: (ContainerViewLayout, CGFloat)? + + var completion: ((Int32) -> Void)? + var dismiss: (() -> Void)? + var cancel: (() -> Void)? + + init(context: AccountContext, invite: ExportedInvitation) { + self.context = context + self.invite = invite + self.presentationData = context.sharedContext.currentPresentationData.with { $0 } + + self.wrappingScrollNode = ASScrollNode() + self.wrappingScrollNode.view.alwaysBounceVertical = true + self.wrappingScrollNode.view.delaysContentTouches = false + self.wrappingScrollNode.view.canCancelContentTouches = true + + self.dimNode = ASDisplayNode() + self.dimNode.backgroundColor = UIColor(white: 0.0, alpha: 0.5) + + self.contentContainerNode = ASDisplayNode() + self.contentContainerNode.isOpaque = false + + self.backgroundNode = ASDisplayNode() + self.backgroundNode.clipsToBounds = true + self.backgroundNode.cornerRadius = 16.0 + + let backgroundColor = self.presentationData.theme.actionSheet.opaqueItemBackgroundColor + let textColor = self.presentationData.theme.actionSheet.primaryTextColor + let secondaryTextColor = self.presentationData.theme.actionSheet.secondaryTextColor + let accentColor = self.presentationData.theme.actionSheet.controlAccentColor + + self.contentBackgroundNode = ASDisplayNode() + self.contentBackgroundNode.backgroundColor = backgroundColor + + self.titleNode = ASTextNode() + self.titleNode.attributedText = NSAttributedString(string: self.presentationData.strings.InviteLink_QRCode_Title, font: Font.bold(17.0), textColor: textColor) + + self.subtitleNode = ASTextNode() + self.subtitleNode.attributedText = NSAttributedString(string: self.presentationData.strings.InviteLink_QRCode_Title, font: Font.regular(13.0), textColor: secondaryTextColor) + + self.cancelButton = HighlightableButtonNode() + self.cancelButton.setTitle(self.presentationData.strings.Common_Done, with: Font.bold(17.0), with: accentColor, for: .normal) + + self.buttonNode = SolidRoundedButtonNode(theme: SolidRoundedButtonTheme(theme: self.presentationData.theme), height: 52.0, cornerRadius: 11.0, gloss: false) + + self.textNode = ImmediateTextNode() + self.textNode.maximumNumberOfLines = 3 + self.textNode.textAlignment = .center + + self.qrButtonNode = HighlightTrackingButtonNode() + self.qrImageNode = TransformImageNode() + + self.qrIconNode = AnimatedStickerNode() + if let path = getAppBundle().path(forResource: "PlaneLogo", ofType: "tgs") { + self.qrIconNode.setup(source: AnimatedStickerNodeLocalFileSource(path: path), width: 240, height: 240, mode: .direct(cachePathPrefix: nil)) + self.qrIconNode.visibility = true + } + + super.init() + + self.backgroundColor = nil + self.isOpaque = false + + self.dimNode.view.addGestureRecognizer(UITapGestureRecognizer(target: self, action: #selector(self.dimTapGesture(_:)))) + self.addSubnode(self.dimNode) + + self.wrappingScrollNode.view.delegate = self + self.addSubnode(self.wrappingScrollNode) + + self.wrappingScrollNode.addSubnode(self.backgroundNode) + self.wrappingScrollNode.addSubnode(self.contentContainerNode) + + self.backgroundNode.addSubnode(self.contentBackgroundNode) + self.contentContainerNode.addSubnode(self.titleNode) + self.contentContainerNode.addSubnode(self.cancelButton) + self.contentContainerNode.addSubnode(self.buttonNode) + + self.contentContainerNode.addSubnode(self.textNode) + self.contentContainerNode.addSubnode(self.qrImageNode) + self.contentContainerNode.addSubnode(self.qrIconNode) + self.contentContainerNode.addSubnode(self.qrButtonNode) + + let textFont = Font.regular(16.0) + + self.textNode.attributedText = NSAttributedString(string: self.presentationData.strings.InviteLink_QRCode_Info, font: textFont, textColor: secondaryTextColor) + self.buttonNode.title = self.presentationData.strings.InviteLink_QRCode_Share + + self.cancelButton.addTarget(self, action: #selector(self.cancelButtonPressed), forControlEvents: .touchUpInside) + self.buttonNode.pressed = { [weak self] in + if let strongSelf = self{ + shareQrCode(context: strongSelf.context, link: strongSelf.invite.link) + } + } + + self.qrImageNode.setSignal(qrCode(string: self.invite.link, color: .black, backgroundColor: .white, icon: .cutout) |> beforeNext { [weak self] size, _ in + guard let strongSelf = self else { + return + } + strongSelf.qrCodeSize = size + if let (layout, navigationHeight) = strongSelf.containerLayout { + strongSelf.containerLayoutUpdated(layout, navigationBarHeight: navigationHeight, transition: .immediate) + } + } |> map { $0.1 }, attemptSynchronously: true) + + self.qrButtonNode.addTarget(self, action: #selector(self.qrPressed), forControlEvents: .touchUpInside) + self.qrButtonNode.highligthedChanged = { [weak self] highlighted in + guard let strongSelf = self else { + return + } + if highlighted { + strongSelf.qrImageNode.alpha = 0.4 + strongSelf.qrIconNode.alpha = 0.4 + } else { + strongSelf.qrImageNode.layer.animateAlpha(from: strongSelf.qrImageNode.alpha, to: 1.0, duration: 0.2) + strongSelf.qrImageNode.alpha = 1.0 + strongSelf.qrIconNode.layer.animateAlpha(from: strongSelf.qrIconNode.alpha, to: 1.0, duration: 0.2) + strongSelf.qrIconNode.alpha = 1.0 + } + } + } + + @objc private func qrPressed() { + shareQrCode(context: self.context, link: self.invite.link) + } + + func updatePresentationData(_ presentationData: PresentationData) { + let previousTheme = self.presentationData.theme + self.presentationData = presentationData + + self.contentBackgroundNode.backgroundColor = self.presentationData.theme.actionSheet.opaqueItemBackgroundColor + self.titleNode.attributedText = NSAttributedString(string: self.titleNode.attributedText?.string ?? "", font: Font.bold(17.0), textColor: self.presentationData.theme.actionSheet.primaryTextColor) + + if previousTheme !== presentationData.theme, let (layout, navigationBarHeight) = self.containerLayout { + self.containerLayoutUpdated(layout, navigationBarHeight: navigationBarHeight, transition: .immediate) + } + + self.cancelButton.setTitle(self.presentationData.strings.Common_Done, with: Font.bold(17.0), with: self.presentationData.theme.actionSheet.controlAccentColor, for: .normal) + self.buttonNode.updateTheme(SolidRoundedButtonTheme(theme: self.presentationData.theme)) + } + + override func didLoad() { + super.didLoad() + + if #available(iOSApplicationExtension 11.0, iOS 11.0, *) { + self.wrappingScrollNode.view.contentInsetAdjustmentBehavior = .never + } + } + + @objc func cancelButtonPressed() { + self.cancel?() + } + + @objc func dimTapGesture(_ recognizer: UITapGestureRecognizer) { + if case .ended = recognizer.state { + self.cancelButtonPressed() + } + } + + func animateIn() { + self.dimNode.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.4) + + let offset = self.bounds.size.height - self.contentBackgroundNode.frame.minY + + let dimPosition = self.dimNode.layer.position + self.dimNode.layer.animatePosition(from: CGPoint(x: dimPosition.x, y: dimPosition.y - offset), to: dimPosition, duration: 0.3, timingFunction: kCAMediaTimingFunctionSpring) + self.layer.animateBoundsOriginYAdditive(from: -offset, to: 0.0, duration: 0.3, timingFunction: kCAMediaTimingFunctionSpring) + } + + func animateOut(completion: (() -> Void)? = nil) { + var dimCompleted = false + var offsetCompleted = false + + let internalCompletion: () -> Void = { [weak self] in + if let strongSelf = self, dimCompleted && offsetCompleted { + strongSelf.dismiss?() + } + completion?() + } + + self.dimNode.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.3, removeOnCompletion: false, completion: { _ in + dimCompleted = true + internalCompletion() + }) + + let offset = self.bounds.size.height - self.contentBackgroundNode.frame.minY + let dimPosition = self.dimNode.layer.position + self.dimNode.layer.animatePosition(from: dimPosition, to: CGPoint(x: dimPosition.x, y: dimPosition.y - offset), duration: 0.3, timingFunction: kCAMediaTimingFunctionSpring, removeOnCompletion: false) + self.layer.animateBoundsOriginYAdditive(from: 0.0, to: -offset, duration: 0.3, timingFunction: kCAMediaTimingFunctionSpring, removeOnCompletion: false, completion: { _ in + offsetCompleted = true + internalCompletion() + }) + } + + override func hitTest(_ point: CGPoint, with event: UIEvent?) -> UIView? { + if self.bounds.contains(point) { + if !self.contentBackgroundNode.bounds.contains(self.convert(point, to: self.contentBackgroundNode)) { + return self.dimNode.view + } + } + return super.hitTest(point, with: event) + } + + func scrollViewDidEndDragging(_ scrollView: UIScrollView, willDecelerate decelerate: Bool) { + let contentOffset = scrollView.contentOffset + let additionalTopHeight = max(0.0, -contentOffset.y) + + if additionalTopHeight >= 30.0 { + self.cancelButtonPressed() + } + } + + func containerLayoutUpdated(_ layout: ContainerViewLayout, navigationBarHeight: CGFloat, transition: ContainedViewLayoutTransition) { + self.containerLayout = (layout, navigationBarHeight) + + var insets = layout.insets(options: [.statusBar, .input]) + let cleanInsets = layout.insets(options: [.statusBar]) + insets.top = max(10.0, insets.top) + + let makeImageLayout = self.qrImageNode.asyncLayout() + let imageSide: CGFloat = 240.0 + let imageSize = CGSize(width: imageSide, height: imageSide) + let imageApply = makeImageLayout(TransformImageArguments(corners: ImageCorners(), imageSize: imageSize, boundingSize: imageSize, intrinsicInsets: UIEdgeInsets(), emptyColor: nil)) + + let _ = imageApply() + + let imageFrame = CGRect(origin: CGPoint(x: floor((layout.size.width - imageSize.width) / 2.0), y: insets.top + 24.0), size: imageSize) + transition.updateFrame(node: self.qrImageNode, frame: imageFrame) + transition.updateFrame(node: self.qrButtonNode, frame: imageFrame) + + if let qrCodeSize = self.qrCodeSize { + let (_, cutoutFrame, _) = qrCodeCutout(size: qrCodeSize, dimensions: imageSize, scale: nil) + self.qrIconNode.updateLayout(size: cutoutFrame.size) + transition.updateBounds(node: self.qrIconNode, bounds: CGRect(origin: CGPoint(), size: cutoutFrame.size)) + transition.updatePosition(node: self.qrIconNode, position: imageFrame.center.offsetBy(dx: 0.0, dy: -1.0)) + } + + let inset: CGFloat = 22.0 + let textSize = self.textNode.updateLayout(CGSize(width: layout.size.width - inset * 2.0, height: CGFloat.greatestFiniteMagnitude)) + let textFrame = CGRect(origin: CGPoint(x: floor((layout.size.width - textSize.width) / 2.0), y: imageFrame.maxX + 20.0), size: textSize) + transition.updateFrame(node: self.textNode, frame: textFrame) + + let buttonSideInset: CGFloat = 16.0 + let bottomInset = insets.bottom + 10.0 + let buttonWidth = layout.size.width - buttonSideInset * 2.0 + let buttonHeight: CGFloat = 50.0 + + let buttonFrame = CGRect(origin: CGPoint(x: floor((layout.size.width - buttonWidth) / 2.0), y: layout.size.height - bottomInset - buttonHeight), size: CGSize(width: buttonWidth, height: buttonHeight)) + transition.updateFrame(node: self.buttonNode, frame: buttonFrame) + let _ = self.buttonNode.updateLayout(width: buttonFrame.width, transition: transition) + + + let titleHeight: CGFloat = 54.0 + let contentHeight = titleHeight + textSize.height + imageSize.height + bottomInset + 52.0 + 77.0 + + let width = horizontalContainerFillingSizeForLayout(layout: layout, sideInset: layout.safeInsets.left) + + let sideInset = floor((layout.size.width - width) / 2.0) + let contentContainerFrame = CGRect(origin: CGPoint(x: sideInset, y: layout.size.height - contentHeight), size: CGSize(width: width, height: contentHeight)) + let contentFrame = contentContainerFrame + + var backgroundFrame = CGRect(origin: CGPoint(x: contentFrame.minX, y: contentFrame.minY), size: CGSize(width: contentFrame.width, height: contentFrame.height + 2000.0)) + if backgroundFrame.minY < contentFrame.minY { + backgroundFrame.origin.y = contentFrame.minY + } + transition.updateFrame(node: self.backgroundNode, frame: backgroundFrame) + transition.updateFrame(node: self.contentBackgroundNode, frame: CGRect(origin: CGPoint(), size: backgroundFrame.size)) + transition.updateFrame(node: self.wrappingScrollNode, frame: CGRect(origin: CGPoint(), size: layout.size)) + transition.updateFrame(node: self.dimNode, frame: CGRect(origin: CGPoint(), size: layout.size)) + + let titleSize = self.titleNode.measure(CGSize(width: width, height: titleHeight)) + let titleFrame = CGRect(origin: CGPoint(x: floor((contentFrame.width - titleSize.width) / 2.0), y: 16.0), size: titleSize) + transition.updateFrame(node: self.titleNode, frame: titleFrame) + + let cancelSize = self.cancelButton.measure(CGSize(width: width, height: titleHeight)) + let cancelFrame = CGRect(origin: CGPoint(x: width - cancelSize.width - 16.0, y: 16.0), size: cancelSize) + transition.updateFrame(node: self.cancelButton, frame: cancelFrame) + + let buttonInset: CGFloat = 16.0 + let doneButtonHeight = self.buttonNode.updateLayout(width: contentFrame.width - buttonInset * 2.0, transition: transition) + transition.updateFrame(node: self.buttonNode, frame: CGRect(x: buttonInset, y: contentHeight - doneButtonHeight - insets.bottom - 16.0, width: contentFrame.width, height: doneButtonHeight)) + + transition.updateFrame(node: self.contentContainerNode, frame: contentContainerFrame) + } + } +} diff --git a/submodules/InviteLinksUI/Sources/InviteLinkViewController.swift b/submodules/InviteLinksUI/Sources/InviteLinkViewController.swift new file mode 100644 index 0000000000..6c71c6b6ea --- /dev/null +++ b/submodules/InviteLinksUI/Sources/InviteLinkViewController.swift @@ -0,0 +1,766 @@ +import Foundation +import UIKit +import SwiftSignalKit +import TelegramPresentationData +import AppBundle +import AsyncDisplayKit +import Postbox +import SyncCore +import TelegramCore +import Display +import AccountContext +import SolidRoundedButtonNode +import ItemListUI +import ItemListPeerItem +import SectionHeaderItem +import TelegramStringFormatting +import MergeLists +import ContextUI +import ShareController +import OverlayStatusController +import PresentationDataUtils +import DirectionalPanGesture + +class InviteLinkViewInteraction { + let context: AccountContext + let openPeer: (PeerId) -> Void + let shareLink: (ExportedInvitation) -> Void + let contextAction: (ExportedInvitation, ASDisplayNode, ContextGesture?) -> Void + + init(context: AccountContext, openPeer: @escaping (PeerId) -> Void, shareLink: @escaping (ExportedInvitation) -> Void, contextAction: @escaping (ExportedInvitation, ASDisplayNode, ContextGesture?) -> Void) { + self.context = context + self.openPeer = openPeer + self.shareLink = shareLink + self.contextAction = contextAction + } +} + +private struct InviteLinkViewTransaction { + let deletions: [ListViewDeleteItem] + let insertions: [ListViewInsertItem] + let updates: [ListViewUpdateItem] + let isLoading: Bool +} + +private enum InviteLinkViewEntryId: Hashable { + case link + case creatorHeader + case creator + case importerHeader + case importer(PeerId) +} + +private func color(for invite: ExportedInvitation) -> UIColor? { + let color: UIColor? + let availability = invitationAvailability(invite) + if invite.isRevoked { + color = nil + } else if invite.expireDate == nil && invite.usageLimit == nil { + color = nil + } else if availability >= 0.5 { + color = UIColor(rgb: 0x4aca62) + } else if availability > 0.0 { + color = UIColor(rgb: 0xf8a953) + } else { + color = UIColor(rgb: 0xf2656a) + } + return color +} + +private enum InviteLinkViewEntry: Comparable, Identifiable { + case link(PresentationTheme, ExportedInvitation) + case creatorHeader(PresentationTheme, String) + case creator(PresentationTheme, PresentationDateTimeFormat, Peer, Int32) + case importerHeader(PresentationTheme, String) + case importer(Int32, PresentationTheme, PresentationDateTimeFormat, Peer, Int32) + + var stableId: InviteLinkViewEntryId { + switch self { + case .link: + return .link + case .creatorHeader: + return .creatorHeader + case .creator: + return .creator + case .importerHeader: + return .importerHeader + case let .importer(_, _, _, peer, _): + return .importer(peer.id) + } + } + + static func ==(lhs: InviteLinkViewEntry, rhs: InviteLinkViewEntry) -> Bool { + switch lhs { + case let .link(lhsTheme, lhsInvitation): + if case let .link(rhsTheme, rhsInvitation) = rhs, lhsTheme === rhsTheme, lhsInvitation == rhsInvitation { + return true + } else { + return false + } + case let .creatorHeader(lhsTheme, lhsTitle): + if case let .creatorHeader(rhsTheme, rhsTitle) = rhs, lhsTheme === rhsTheme, lhsTitle == rhsTitle { + return true + } else { + return false + } + case let .creator(lhsTheme, lhsDateTimeFormat, lhsPeer, lhsDate): + if case let .creator(rhsTheme, rhsDateTimeFormat, rhsPeer, rhsDate) = rhs, lhsTheme === rhsTheme, lhsDateTimeFormat == rhsDateTimeFormat, arePeersEqual(lhsPeer, rhsPeer), lhsDate == rhsDate { + return true + } else { + return false + } + case let .importerHeader(lhsTheme, lhsTitle): + if case let .importerHeader(rhsTheme, rhsTitle) = rhs, lhsTheme === rhsTheme, lhsTitle == rhsTitle { + return true + } else { + return false + } + case let .importer(lhsIndex, lhsTheme, lhsDateTimeFormat, lhsPeer, lhsDate): + if case let .importer(rhsIndex, rhsTheme, rhsDateTimeFormat, rhsPeer, rhsDate) = rhs, lhsIndex == rhsIndex, lhsTheme === rhsTheme, lhsDateTimeFormat == rhsDateTimeFormat, arePeersEqual(lhsPeer, rhsPeer), lhsDate == rhsDate { + return true + } else { + return false + } + } + } + + static func <(lhs: InviteLinkViewEntry, rhs: InviteLinkViewEntry) -> Bool { + switch lhs { + case .link: + switch rhs { + case .link: + return false + case .creatorHeader, .creator, .importerHeader, .importer: + return true + } + case .creatorHeader: + switch rhs { + case .link, .creatorHeader: + return false + case .creator, .importerHeader, .importer: + return true + } + case .creator: + switch rhs { + case .link, .creatorHeader, .creator: + return false + case .importerHeader, .importer: + return true + } + case .importerHeader: + switch rhs { + case .link, .creatorHeader, .importerHeader: + return false + case .creator, .importer: + return true + } + case let .importer(lhsIndex, _, _, _, _): + switch rhs { + case .link, .creatorHeader, .creator, .importerHeader: + return false + case let .importer(rhsIndex, _, _, _, _): + return lhsIndex < rhsIndex + } + } + } + + func item(account: Account, presentationData: PresentationData, interaction: InviteLinkViewInteraction) -> ListViewItem { + switch self { + case let .link(_, invite): + let buttonColor = color(for: invite) + return ItemListPermanentInviteLinkItem(context: interaction.context, presentationData: ItemListPresentationData(presentationData), invite: invite, peers: [], buttonColor: buttonColor, sectionId: 0, style: .plain, shareAction: { + interaction.shareLink(invite) + }, contextAction: { node in + interaction.contextAction(invite, node, nil) + }, viewAction: { + }) + case let .creatorHeader(_, title): + return SectionHeaderItem(presentationData: ItemListPresentationData(presentationData), title: title) + case let .creator(_, dateTimeFormat, peer, date): + let dateString = stringForFullDate(timestamp: date, strings: presentationData.strings, dateTimeFormat: dateTimeFormat) + return ItemListPeerItem(presentationData: ItemListPresentationData(presentationData), dateTimeFormat: dateTimeFormat, nameDisplayOrder: presentationData.nameDisplayOrder, context: interaction.context, peer: peer, height: .generic, nameStyle: .distinctBold, presence: nil, text: .text(dateString, .secondary), label: .none, editing: ItemListPeerItemEditing(editable: false, editing: false, revealed: false), revealOptions: nil, switchValue: nil, enabled: true, selectable: true, sectionId: 0, action: { + interaction.openPeer(peer.id) + }, setPeerIdWithRevealedOptions: { _, _ in }, removePeer: { _ in }, hasTopStripe: false, noInsets: true, tag: nil) + case let .importerHeader(_, title): + return SectionHeaderItem(presentationData: ItemListPresentationData(presentationData), title: title) + case let .importer(_, _, dateTimeFormat, peer, date): + let dateString = stringForFullDate(timestamp: date, strings: presentationData.strings, dateTimeFormat: dateTimeFormat) + return ItemListPeerItem(presentationData: ItemListPresentationData(presentationData), dateTimeFormat: dateTimeFormat, nameDisplayOrder: presentationData.nameDisplayOrder, context: interaction.context, peer: peer, height: .generic, nameStyle: .distinctBold, presence: nil, text: .text(dateString, .secondary), label: .none, editing: ItemListPeerItemEditing(editable: false, editing: false, revealed: false), revealOptions: nil, switchValue: nil, enabled: true, selectable: true, sectionId: 0, action: { + interaction.openPeer(peer.id) + }, setPeerIdWithRevealedOptions: { _, _ in }, removePeer: { _ in }, hasTopStripe: false, noInsets: true, tag: nil) + } + } +} + +private func preparedTransition(from fromEntries: [InviteLinkViewEntry], to toEntries: [InviteLinkViewEntry], isLoading: Bool, account: Account, presentationData: PresentationData, interaction: InviteLinkViewInteraction) -> InviteLinkViewTransaction { + let (deleteIndices, indicesAndItems, updateIndices) = mergeListsStableWithUpdates(leftList: fromEntries, rightList: toEntries) + + let deletions = deleteIndices.map { ListViewDeleteItem(index: $0, directionHint: nil) } + let insertions = indicesAndItems.map { ListViewInsertItem(index: $0.0, previousIndex: $0.2, item: $0.1.item(account: account, presentationData: presentationData, interaction: interaction), directionHint: nil) } + let updates = updateIndices.map { ListViewUpdateItem(index: $0.0, previousIndex: $0.2, item: $0.1.item(account: account, presentationData: presentationData, interaction: interaction), directionHint: nil) } + + return InviteLinkViewTransaction(deletions: deletions, insertions: insertions, updates: updates, isLoading: isLoading) +} + +public final class InviteLinkViewController: ViewController { + private var controllerNode: Node { + return self.displayNode as! Node + } + + private var animatedIn = false + + private let context: AccountContext + private let peerId: PeerId + private let invite: ExportedInvitation + private let importersContext: PeerInvitationImportersContext? + + private var presentationDataDisposable: Disposable? + + public init(context: AccountContext, peerId: PeerId, invite: ExportedInvitation, importersContext: PeerInvitationImportersContext?) { + self.context = context + self.peerId = peerId + self.invite = invite + self.importersContext = importersContext + + super.init(navigationBarPresentationData: nil) + + self.navigationPresentation = .flatModal + self.statusBar.statusBarStyle = .Ignore + + self.blocksBackgroundWhenInOverlay = true + + self.presentationDataDisposable = (context.sharedContext.presentationData + |> deliverOnMainQueue).start(next: { [weak self] presentationData in + if let strongSelf = self { + strongSelf.controllerNode.updatePresentationData(presentationData) + } + }) + + self.statusBar.statusBarStyle = .Ignore + } + + required init(coder aDecoder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + deinit { + self.presentationDataDisposable?.dispose() + } + + override public func loadDisplayNode() { + self.displayNode = Node(context: self.context, peerId: self.peerId, invite: self.invite, importersContext: self.importersContext, controller: self) + } + + override public func loadView() { + super.loadView() + } + + private var didAppearOnce: Bool = false + private var isDismissed: Bool = false + public override func viewDidAppear(_ animated: Bool) { + super.viewDidAppear(animated) + + if !self.didAppearOnce { + self.didAppearOnce = true + + self.controllerNode.animateIn() + } + } + + override public func dismiss(completion: (() -> Void)? = nil) { + if !self.isDismissed { + self.isDismissed = true + self.didAppearOnce = false + + self.controllerNode.animateOut(completion: { [weak self] in + completion?() + self?.dismiss(animated: false) + }) + } + } + + override public func containerLayoutUpdated(_ layout: ContainerViewLayout, transition: ContainedViewLayoutTransition) { + super.containerLayoutUpdated(layout, transition: transition) + + self.controllerNode.containerLayoutUpdated(layout, transition: transition) + } + + class Node: ViewControllerTracingNode, UIGestureRecognizerDelegate { + private weak var controller: InviteLinkViewController? + + private let context: AccountContext + private let peerId: PeerId + private let invite: ExportedInvitation + + private var interaction: InviteLinkViewInteraction? + + private var presentationData: PresentationData + private let presentationDataPromise: Promise + + private var disposable: Disposable? + + private let dimNode: ASDisplayNode + private let contentNode: ASDisplayNode + private let headerNode: ASDisplayNode + private let headerBackgroundNode: ASDisplayNode + private let titleNode: ImmediateTextNode + private let subtitleNode: ImmediateTextNode + private let editButton: HighlightableButtonNode + private let doneButton: HighlightableButtonNode + private let historyBackgroundNode: ASDisplayNode + private let historyBackgroundContentNode: ASDisplayNode + private var floatingHeaderOffset: CGFloat? + private let listNode: ListView + + private var enqueuedTransitions: [InviteLinkViewTransaction] = [] + + private var validLayout: ContainerViewLayout? + + private var presentationDataDisposable: Disposable? + + private let importersContext: PeerInvitationImportersContext + + init(context: AccountContext, peerId: PeerId, invite: ExportedInvitation, importersContext: PeerInvitationImportersContext?, controller: InviteLinkViewController) { + self.context = context + self.peerId = peerId + self.invite = invite + let presentationData = context.sharedContext.currentPresentationData.with { $0 } + self.presentationData = presentationData + self.presentationDataPromise = Promise(self.presentationData) + self.controller = controller + + self.importersContext = importersContext ?? PeerInvitationImportersContext(account: context.account, peerId: peerId, invite: invite) + + self.dimNode = ASDisplayNode() + self.dimNode.backgroundColor = UIColor(white: 0.0, alpha: 0.5) + + self.contentNode = ASDisplayNode() + + self.headerNode = ASDisplayNode() + self.headerNode.clipsToBounds = true + + self.headerBackgroundNode = ASDisplayNode() + self.headerBackgroundNode.backgroundColor = self.presentationData.theme.list.plainBackgroundColor + self.headerBackgroundNode.cornerRadius = 16.0 + + self.titleNode = ImmediateTextNode() + self.titleNode.maximumNumberOfLines = 1 + self.titleNode.textAlignment = .center + self.titleNode.attributedText = NSAttributedString(string: self.presentationData.strings.InviteLink_InviteLink, font: Font.bold(17.0), textColor: self.presentationData.theme.actionSheet.primaryTextColor) + + self.subtitleNode = ImmediateTextNode() + self.subtitleNode.maximumNumberOfLines = 1 + self.subtitleNode.textAlignment = .center + + let buttonColor = color(for: invite) ?? presentationData.theme.actionSheet.controlAccentColor + + self.editButton = HighlightableButtonNode() + self.editButton.setTitle(self.presentationData.strings.Common_Edit, with: Font.regular(17.0), with: buttonColor, for: .normal) + + self.doneButton = HighlightableButtonNode() + self.doneButton.setTitle(self.presentationData.strings.Common_Done, with: Font.bold(17.0), with: buttonColor, for: .normal) + + self.historyBackgroundNode = ASDisplayNode() + self.historyBackgroundNode.isLayerBacked = true + + self.historyBackgroundContentNode = ASDisplayNode() + self.historyBackgroundContentNode.isLayerBacked = true + self.historyBackgroundContentNode.backgroundColor = self.presentationData.theme.list.plainBackgroundColor + + self.historyBackgroundNode.addSubnode(self.historyBackgroundContentNode) + + self.listNode = ListView() + self.listNode.verticalScrollIndicatorColor = UIColor(white: 0.0, alpha: 0.3) + self.listNode.verticalScrollIndicatorFollowsOverscroll = true + + super.init() + + self.backgroundColor = nil + self.isOpaque = false + + self.interaction = InviteLinkViewInteraction(context: context, openPeer: { [weak self] peerId in + if let strongSelf = self, let navigationController = strongSelf.controller?.navigationController as? NavigationController { + context.sharedContext.navigateToChatController(NavigateToChatControllerParams(navigationController: navigationController, context: context, chatLocation: .peer(peerId), keepStack: .always)) + } + }, shareLink: { [weak self] invite in + let shareController = ShareController(context: context, subject: .url(invite.link)) + self?.controller?.present(shareController, in: .window(.root)) + }, contextAction: { [weak self] invite, node, gesture in + guard let node = node as? ContextExtractedContentContainingNode else { + return + } + + let presentationData = context.sharedContext.currentPresentationData.with { $0 } + var items: [ContextMenuItem] = [] + + items.append(.action(ContextMenuActionItem(text: presentationData.strings.InviteLink_ContextCopy, icon: { theme in + return generateTintedImage(image: UIImage(bundleImageName: "Chat/Context Menu/Copy"), color: theme.contextMenu.primaryColor) + }, action: { [weak self] _, f in + f(.dismissWithoutContent) + + UIPasteboard.general.string = invite.link + let presentationData = context.sharedContext.currentPresentationData.with { $0 } + self?.controller?.present(OverlayStatusController(theme: presentationData.theme, type: .genericSuccess(presentationData.strings.Username_LinkCopied, false)), in: .window(.root)) + }))) + + if !invite.isRevoked { + items.append(.action(ContextMenuActionItem(text: presentationData.strings.InviteLink_ContextGetQRCode, icon: { theme in + return generateTintedImage(image: UIImage(bundleImageName: "Wallet/QrIcon"), color: theme.contextMenu.primaryColor) + }, action: { [weak self] _, f in + f(.dismissWithoutContent) + + let controller = InviteLinkQRCodeController(context: context, invite: invite) + self?.controller?.present(controller, in: .window(.root)) + }))) + } + + let contextController = ContextController(account: context.account, presentationData: presentationData, source: .extracted(InviteLinkContextExtractedContentSource(controller: controller, sourceNode: node)), items: .single(items), reactionItems: [], gesture: gesture) + self?.controller?.presentInGlobalOverlay(contextController) + }) + + let previousEntries = Atomic<[InviteLinkViewEntry]?>(value: nil) + + let creatorPeer = context.account.postbox.loadedPeerWithId(invite.adminId) + self.disposable = (combineLatest(self.presentationDataPromise.get(), self.importersContext.state, creatorPeer) + |> deliverOnMainQueue).start(next: { [weak self] presentationData, state, creatorPeer in + if let strongSelf = self { + var entries: [InviteLinkViewEntry] = [] + + entries.append(.link(presentationData.theme, invite)) + entries.append(.creatorHeader(presentationData.theme, presentationData.strings.InviteLink_CreatedBy.uppercased())) + entries.append(.creator(presentationData.theme, presentationData.dateTimeFormat, creatorPeer, invite.date)) + + if !state.importers.isEmpty { + entries.append(.importerHeader(presentationData.theme, presentationData.strings.InviteLink_PeopleJoined(Int32(state.count)).uppercased())) + } + + var index: Int32 = 0 + for importer in state.importers { + if let peer = importer.peer.peer { + entries.append(.importer(index, presentationData.theme, presentationData.dateTimeFormat, peer, importer.date)) + } + index += 1 + } + + let previousEntries = previousEntries.swap(entries) + + let transition = preparedTransition(from: previousEntries ?? [], to: entries, isLoading: false, account: context.account, presentationData: presentationData, interaction: strongSelf.interaction!) + strongSelf.enqueueTransition(transition) + } + }) + + self.listNode.preloadPages = true + self.listNode.stackFromBottom = true + self.listNode.updateFloatingHeaderOffset = { [weak self] offset, transition in + if let strongSelf = self { + strongSelf.updateFloatingHeaderOffset(offset: offset, transition: transition) + } + } + self.listNode.visibleBottomContentOffsetChanged = { [weak self] offset in + if case let .known(value) = offset, value < 40.0 { + self?.importersContext.loadMore() + } + } + + self.addSubnode(self.dimNode) + self.addSubnode(self.contentNode) + self.contentNode.addSubnode(self.historyBackgroundNode) + self.contentNode.addSubnode(self.listNode) + self.contentNode.addSubnode(self.headerNode) + + self.headerNode.addSubnode(self.headerBackgroundNode) + self.headerNode.addSubnode(self.titleNode) + self.headerNode.addSubnode(self.subtitleNode) + self.headerNode.addSubnode(self.editButton) + self.headerNode.addSubnode(self.doneButton) + + self.editButton.addTarget(self, action: #selector(self.editButtonPressed), forControlEvents: .touchUpInside) + self.doneButton.addTarget(self, action: #selector(self.doneButtonPressed), forControlEvents: .touchUpInside) + + self.presentationDataDisposable = context.sharedContext.presentationData.start(next: { [weak self] presentationData in + if let strongSelf = self { + if strongSelf.presentationData.theme !== presentationData.theme || strongSelf.presentationData.strings !== presentationData.strings { + strongSelf.updatePresentationData(presentationData) + } + } + }) + + if invite.isRevoked { + self.editButton.isHidden = true + } + } + + deinit { + self.disposable?.dispose() + self.presentationDataDisposable?.dispose() + } + + override func didLoad() { + super.didLoad() + + self.view.disablesInteractiveTransitionGestureRecognizer = true + self.view.disablesInteractiveModalDismiss = true + + self.dimNode.view.addGestureRecognizer(UITapGestureRecognizer(target: self, action: #selector(self.dimTapGesture(_:)))) + + let panRecognizer = DirectionalPanGestureRecognizer(target: self, action: #selector(self.panGesture(_:))) + panRecognizer.delegate = self + panRecognizer.delaysTouchesBegan = false + panRecognizer.cancelsTouchesInView = true + self.view.addGestureRecognizer(panRecognizer) + } + + @objc private func editButtonPressed() { + let navigationController = self.controller?.navigationController as? NavigationController + self.controller?.dismiss() + + if let navigationController = navigationController { + let controller = inviteLinkEditController(context: self.context, peerId: self.peerId, invite: self.invite) + controller.navigationPresentation = .modal + navigationController.pushViewController(controller) + } + } + + @objc private func doneButtonPressed() { + self.controller?.dismiss() + } + + func updatePresentationData(_ presentationData: PresentationData) { + self.presentationData = presentationData + self.presentationDataPromise.set(.single(presentationData)) + + self.historyBackgroundContentNode.backgroundColor = self.presentationData.theme.list.plainBackgroundColor + self.headerBackgroundNode.backgroundColor = self.presentationData.theme.list.plainBackgroundColor + self.titleNode.attributedText = NSAttributedString(string: self.presentationData.strings.InviteLink_InviteLink, font: Font.bold(17.0), textColor: self.presentationData.theme.actionSheet.primaryTextColor) + + let buttonColor = color(for: invite) ?? self.presentationData.theme.actionSheet.controlAccentColor + self.editButton.setTitle(self.presentationData.strings.Common_Edit, with: Font.regular(17.0), with: buttonColor, for: .normal) + self.doneButton.setTitle(self.presentationData.strings.Common_Done, with: Font.bold(17.0), with: buttonColor, for: .normal) + } + + private func enqueueTransition(_ transition: InviteLinkViewTransaction) { + self.enqueuedTransitions.append(transition) + + if let _ = self.validLayout { + while !self.enqueuedTransitions.isEmpty { + self.dequeueTransition() + } + } + } + + private func dequeueTransition() { + guard let _ = self.validLayout, let transition = self.enqueuedTransitions.first else { + return + } + self.enqueuedTransitions.remove(at: 0) + + self.listNode.transaction(deleteIndices: transition.deletions, insertIndicesAndItems: transition.insertions, updateIndicesAndItems: transition.updates, options: ListViewDeleteAndInsertOptions(), updateSizeAndInsets: nil, updateOpaqueState: nil, completion: { _ in + }) + } + + func animateIn() { + guard let layout = self.validLayout else { + return + } + let transition = ContainedViewLayoutTransition.animated(duration: 0.4, curve: .spring) + + let initialBounds = self.contentNode.bounds + self.contentNode.bounds = initialBounds.offsetBy(dx: 0.0, dy: -layout.size.height) + transition.animateView({ + self.contentNode.view.bounds = initialBounds + }) + self.dimNode.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.3) + } + + func animateOut(completion: (() -> Void)?) { + guard let layout = self.validLayout else { + return + } + var offsetCompleted = false + let internalCompletion: () -> Void = { + if offsetCompleted { + completion?() + } + } + + self.contentNode.layer.animateBoundsOriginYAdditive(from: self.contentNode.bounds.origin.y, to: -layout.size.height, duration: 0.3, timingFunction: kCAMediaTimingFunctionSpring, removeOnCompletion: false, completion: { _ in + offsetCompleted = true + internalCompletion() + }) + self.dimNode.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.2, removeOnCompletion: false) + } + + func containerLayoutUpdated(_ layout: ContainerViewLayout, transition: ContainedViewLayoutTransition) { + self.validLayout = layout + + transition.updateFrame(node: self.dimNode, frame: CGRect(origin: CGPoint(), size: layout.size)) + transition.updateFrame(node: self.contentNode, frame: CGRect(origin: CGPoint(), size: layout.size)) + + var insets = UIEdgeInsets() + insets.left = layout.safeInsets.left + insets.right = layout.safeInsets.right + insets.bottom = layout.intrinsicInsets.bottom + + let headerHeight: CGFloat = 54.0 + let visibleItemsHeight: CGFloat = 147.0 + floor(52.0 * 3.5) + + let layoutTopInset: CGFloat = max(layout.statusBarHeight ?? 0.0, layout.safeInsets.top) + + let listTopInset = layoutTopInset + headerHeight + let listNodeSize = CGSize(width: layout.size.width, height: layout.size.height - listTopInset) + + insets.top = max(0.0, listNodeSize.height - visibleItemsHeight) + + let (duration, curve) = listViewAnimationDurationAndCurve(transition: transition) + let updateSizeAndInsets = ListViewUpdateSizeAndInsets(size: listNodeSize, insets: insets, duration: duration, curve: curve) + self.listNode.transaction(deleteIndices: [], insertIndicesAndItems: [], updateIndicesAndItems: [], options: [.Synchronous, .LowLatency], scrollToItem: nil, updateSizeAndInsets: updateSizeAndInsets, stationaryItemRange: nil, updateOpaqueState: nil, completion: { _ in }) + + transition.updateFrame(node: self.listNode, frame: CGRect(origin: CGPoint(x: 0.0, y: listTopInset), size: listNodeSize)) + + transition.updateFrame(node: self.headerBackgroundNode, frame: CGRect(x: 0.0, y: 0.0, width: layout.size.width, height: 68.0)) + + let titleSize = self.titleNode.updateLayout(CGSize(width: layout.size.width, height: headerHeight)) + let titleFrame = CGRect(origin: CGPoint(x: floor((layout.size.width - titleSize.width) / 2.0), y: 18.0), size: titleSize) + transition.updateFrame(node: self.titleNode, frame: titleFrame) + + let editSize = self.editButton.measure(CGSize(width: layout.size.width, height: headerHeight)) + let editFrame = CGRect(origin: CGPoint(x: 16.0, y: 18.0), size: editSize) + transition.updateFrame(node: self.editButton, frame: editFrame) + + let doneSize = self.doneButton.measure(CGSize(width: layout.size.width, height: headerHeight)) + let doneFrame = CGRect(origin: CGPoint(x: layout.size.width - doneSize.width - 16.0, y: 18.0), size: doneSize) + transition.updateFrame(node: self.doneButton, frame: doneFrame) + } + + override func hitTest(_ point: CGPoint, with event: UIEvent?) -> UIView? { + let result = super.hitTest(point, with: event) + + if result === self.headerNode.view { + return self.view + } + + if result === self.headerNode.view { + return self.view + } + + if !self.bounds.contains(point) { + return nil + } + if point.y < self.headerNode.frame.minY { + return self.dimNode.view + } + return result + } + + @objc func dimTapGesture(_ recognizer: UITapGestureRecognizer) { + if case .ended = recognizer.state { + self.controller?.dismiss() + } + } + + private var panGestureArguments: CGFloat? + + @objc func panGesture(_ recognizer: UIPanGestureRecognizer) { + let contentOffset = self.listNode.visibleContentOffset() + switch recognizer.state { + case .began: + self.panGestureArguments = 0.0 + case .changed: + var translation = recognizer.translation(in: self.contentNode.view).y + if let currentPanOffset = self.panGestureArguments { + + + if case let .known(value) = contentOffset, value <= 0.5 { + } else { + translation = currentPanOffset + } + + self.panGestureArguments = translation + } + + var bounds = self.contentNode.bounds + bounds.origin.y = -translation + bounds.origin.y = min(0.0, bounds.origin.y) + self.contentNode.bounds = bounds + case .ended: + let translation = recognizer.translation(in: self.contentNode.view) + var velocity = recognizer.velocity(in: self.contentNode.view) + + if case let .known(value) = contentOffset, value > 0.0 { + velocity = CGPoint() + } else if case .unknown = contentOffset { + velocity = CGPoint() + } + + var bounds = self.contentNode.bounds + bounds.origin.y = -translation.y + bounds.origin.y = min(0.0, bounds.origin.y) + + self.panGestureArguments = nil + if bounds.minY < -60 || (bounds.minY < 0.0 && velocity.y > 300.0) { + self.controller?.dismiss() + } else { + var bounds = self.contentNode.bounds + let previousBounds = bounds + bounds.origin.y = 0.0 + self.contentNode.bounds = bounds + self.contentNode.layer.animateBounds(from: previousBounds, to: self.contentNode.bounds, duration: 0.3, timingFunction: CAMediaTimingFunctionName.easeInEaseOut.rawValue) + } + case .cancelled: + self.panGestureArguments = nil + + let previousBounds = self.contentNode.bounds + var bounds = self.contentNode.bounds + bounds.origin.y = 0.0 + self.contentNode.bounds = bounds + self.contentNode.layer.animateBounds(from: previousBounds, to: self.contentNode.bounds, duration: 0.3, timingFunction: CAMediaTimingFunctionName.easeInEaseOut.rawValue) + default: + break + } + } + + private func updateFloatingHeaderOffset(offset: CGFloat, transition: ContainedViewLayoutTransition) { + guard let validLayout = self.validLayout else { + return + } + + self.floatingHeaderOffset = offset + + let layoutTopInset: CGFloat = max(validLayout.statusBarHeight ?? 0.0, validLayout.safeInsets.top) + + let controlsHeight: CGFloat = 44.0 + + let listTopInset = layoutTopInset + controlsHeight + + let rawControlsOffset = offset + listTopInset - controlsHeight + let controlsOffset = max(layoutTopInset, rawControlsOffset) + let isOverscrolling = rawControlsOffset <= layoutTopInset + let controlsFrame = CGRect(origin: CGPoint(x: 0.0, y: controlsOffset), size: CGSize(width: validLayout.size.width, height: controlsHeight)) + + let previousFrame = self.headerNode.frame + + if !controlsFrame.equalTo(previousFrame) { + self.headerNode.frame = controlsFrame + + let positionDelta = CGPoint(x: controlsFrame.minX - previousFrame.minX, y: controlsFrame.minY - previousFrame.minY) + + transition.animateOffsetAdditive(node: self.headerNode, offset: positionDelta.y) + } + +// transition.updateAlpha(node: self.headerNode.separatorNode, alpha: isOverscrolling ? 1.0 : 0.0) + + let backgroundFrame = CGRect(origin: CGPoint(x: 0.0, y: controlsFrame.maxY), size: CGSize(width: validLayout.size.width, height: validLayout.size.height)) + + let previousBackgroundFrame = self.historyBackgroundNode.frame + + if !backgroundFrame.equalTo(previousBackgroundFrame) { + self.historyBackgroundNode.frame = backgroundFrame + self.historyBackgroundContentNode.frame = CGRect(origin: CGPoint(), size: backgroundFrame.size) + + let positionDelta = CGPoint(x: backgroundFrame.minX - previousBackgroundFrame.minX, y: backgroundFrame.minY - previousBackgroundFrame.minY) + + transition.animateOffsetAdditive(node: self.historyBackgroundNode, offset: positionDelta.y) + } + } + } +} diff --git a/submodules/InviteLinksUI/Sources/InviteLinksGridNode.swift b/submodules/InviteLinksUI/Sources/InviteLinksGridNode.swift new file mode 100644 index 0000000000..71f6b915a1 --- /dev/null +++ b/submodules/InviteLinksUI/Sources/InviteLinksGridNode.swift @@ -0,0 +1,504 @@ +import Foundation +import UIKit +import Display +import AsyncDisplayKit +import SwiftSignalKit +import SyncCore +import TelegramPresentationData +import ItemListUI +import SolidRoundedButtonNode +import RadialStatusNode + +private let itemSpacing: CGFloat = 10.0 +private let titleFont = Font.semibold(17.0) +private let subtitleFont = Font.regular(12.0) + +private func generateBackgroundImage(colors: NSArray) -> UIImage? { + return generateImage(CGSize(width: 45, height: 45), contextGenerator: { size, context in + let bounds = CGRect(origin: CGPoint(), size: size) + context.clear(bounds) + + let path = UIBezierPath(roundedRect: CGRect(origin: CGPoint(), size: size), cornerRadius: 15) + context.addPath(path.cgPath) + context.clip() + + var locations: [CGFloat] = [0.0, 1.0] + let colorSpace = CGColorSpaceCreateDeviceRGB() + let gradient = CGGradient(colorsSpace: colorSpace, colors: colors, locations: &locations)! + + context.drawLinearGradient(gradient, start: CGPoint(), end: CGPoint(x: 0.0, y: bounds.size.height), options: CGGradientDrawingOptions()) + })?.stretchableImage(withLeftCapWidth: 22, topCapHeight: 22) +} + +func invitationAvailability(_ invite: ExportedInvitation) -> CGFloat { + let currentTime = Int32(CFAbsoluteTimeGetCurrent() + kCFAbsoluteTimeIntervalSince1970) + var availability: CGFloat = 1.0 + if let expireDate = invite.expireDate { + let startDate = invite.startDate ?? invite.date + let fraction = CGFloat(expireDate - currentTime) / CGFloat(expireDate - startDate) + availability = min(fraction, availability) + } + if let usageLimit = invite.usageLimit, let count = invite.count { + let fraction = 1.0 - (CGFloat(count) / CGFloat(usageLimit)) + availability = min(fraction, availability) + } + return availability +} + +private class ItemNode: ASDisplayNode { + private let backgroundNode: ASImageNode + + private let iconNode: ASImageNode + private var timerNode: TimerNode? + + private let extractedContainerNode: ContextExtractedContentContainingNode + private let containerNode: ContextControllerSourceNode + private let buttonNode: HighlightTrackingButtonNode + private let buttonIconNode: ASImageNode + + private let titleNode: ImmediateTextNode + private let subtitleNode: ImmediateTextNode + + private var params: (size: CGSize, wide: Bool, invite: ExportedInvitation, presentationData: ItemListPresentationData)? + + var action: (() -> Void)? + var contextAction: ((ASDisplayNode) -> Void)? + + override init() { + self.backgroundNode = ASImageNode() + self.backgroundNode.displaysAsynchronously = false + self.backgroundNode.displayWithoutProcessing = true + + self.iconNode = ASImageNode() + self.iconNode.displaysAsynchronously = false + self.iconNode.displayWithoutProcessing = true + + self.buttonNode = HighlightTrackingButtonNode() + self.extractedContainerNode = ContextExtractedContentContainingNode() + self.containerNode = ContextControllerSourceNode() + self.containerNode.isGestureEnabled = false + self.buttonIconNode = ASImageNode() + self.buttonIconNode.displaysAsynchronously = false + self.buttonIconNode.displayWithoutProcessing = true + self.buttonIconNode.image = generateImage(CGSize(width: 26.0, height: 26.0), contextGenerator: { size, context in + context.clear(CGRect(origin: CGPoint(), size: size)) + + context.setFillColor(UIColor.white.cgColor) + context.fillEllipse(in: CGRect(origin: CGPoint(), size: size)) + + context.setBlendMode(.clear) + context.fillEllipse(in: CGRect(origin: CGPoint(x: 4.0, y: 11.0), size: CGSize(width: 4.0, height: 4.0))) + context.fillEllipse(in: CGRect(origin: CGPoint(x: 11.0, y: 11.0), size: CGSize(width: 4.0, height: 4.0))) + context.fillEllipse(in: CGRect(origin: CGPoint(x: 18.0, y: 11.0), size: CGSize(width: 4.0, height: 4.0))) + }) + + self.titleNode = ImmediateTextNode() + self.titleNode.maximumNumberOfLines = 2 + + self.subtitleNode = ImmediateTextNode() + self.subtitleNode.maximumNumberOfLines = 1 + + super.init() + + self.addSubnode(self.backgroundNode) + self.addSubnode(self.iconNode) + + self.containerNode.addSubnode(self.extractedContainerNode) + self.extractedContainerNode.contentNode.addSubnode(self.buttonIconNode) + self.containerNode.targetNodeForActivationProgress = self.extractedContainerNode.contentNode + self.buttonNode.addSubnode(self.containerNode) + self.addSubnode(self.buttonNode) + + self.addSubnode(self.titleNode) + self.addSubnode(self.subtitleNode) + + self.buttonNode.addTarget(self, action: #selector(self.buttonPressed), forControlEvents: .touchUpInside) + self.buttonNode.highligthedChanged = { [weak self] highlighted in + if let strongSelf = self { + if highlighted { + strongSelf.buttonIconNode.layer.removeAnimation(forKey: "opacity") + strongSelf.buttonIconNode.alpha = 0.4 + } else { + strongSelf.buttonIconNode.alpha = 1.0 + strongSelf.buttonIconNode.layer.animateAlpha(from: 0.4, to: 1.0, duration: 0.2) + } + } + } + } + + override func didLoad() { + self.view.addGestureRecognizer(UITapGestureRecognizer(target: self, action: #selector(self.tapGesture(_:)))) + } + + @objc private func tapGesture(_ gestureRecognizer: UITapGestureRecognizer) { + self.action?() + } + + @objc private func buttonPressed() { + self.contextAction?(self.extractedContainerNode) + } + + func update(size: CGSize, wide: Bool, invite: ExportedInvitation, presentationData: ItemListPresentationData, transition: ContainedViewLayoutTransition) -> CGSize { + let updated = self.params?.size != size || self.params?.wide != wide || self.params?.invite != invite + self.params = (size, wide, invite, presentationData) + + let currentTime = Int32(CFAbsoluteTimeGetCurrent() + kCFAbsoluteTimeIntervalSince1970) + let availability = invitationAvailability(invite) + + var isExpired = false + let secondaryTextColor: UIColor + if invite.isRevoked { + self.backgroundNode.image = generateBackgroundImage(colors: [UIColor(rgb: 0xd4d8db).cgColor, UIColor(rgb: 0xced2d5).cgColor]) + secondaryTextColor = UIColor(rgb: 0xf8f9f9) + } else if invite.expireDate == nil && invite.usageLimit == nil { + self.backgroundNode.image = generateBackgroundImage(colors: [UIColor(rgb: 0x00b5f7).cgColor, UIColor(rgb: 0x00b2f6).cgColor]) + secondaryTextColor = UIColor(rgb: 0xa7f4ff) + } else if availability >= 0.5 { + self.backgroundNode.image = generateBackgroundImage(colors: [UIColor(rgb: 0x4aca62).cgColor, UIColor(rgb: 0x43c85c).cgColor]) + secondaryTextColor = UIColor(rgb: 0xc5ffe6) + } else if availability > 0.0 { + self.backgroundNode.image = generateBackgroundImage(colors: [UIColor(rgb: 0xf8a953).cgColor, UIColor(rgb: 0xf7a64e).cgColor]) + secondaryTextColor = UIColor(rgb: 0xfeffd7) + } else { + self.backgroundNode.image = generateBackgroundImage(colors: [UIColor(rgb: 0xf2656a).cgColor, UIColor(rgb: 0xf25f65).cgColor]) + secondaryTextColor = UIColor(rgb: 0xffd3de) + isExpired = true + } + + let itemWidth = wide ? size.width : floor((size.width - itemSpacing) / 2.0) + + var inviteLink = invite.link.replacingOccurrences(of: "https://", with: "") + if !wide { + inviteLink = inviteLink.replacingOccurrences(of: "joinchat/", with: "joinchat/\n") + inviteLink = inviteLink.replacingOccurrences(of: "join/", with: "join/\n") + } + let title: NSMutableAttributedString = NSMutableAttributedString(string: inviteLink, font: titleFont, textColor: UIColor.white) + if inviteLink.hasPrefix("t.me/joinchat/") { + title.addAttribute(NSAttributedString.Key.foregroundColor, value: secondaryTextColor, range: NSMakeRange(0, "t.me/joinchat/".count)) + } else if inviteLink.hasPrefix("t.me/join/") { + title.addAttribute(NSAttributedString.Key.foregroundColor, value: secondaryTextColor, range: NSMakeRange(0, "t.me/join/".count)) + } + self.titleNode.attributedText = title + + var subtitleText: String = "" + if let count = invite.count { + subtitleText = presentationData.strings.InviteLink_PeopleJoinedShort(count) + } else { + subtitleText = isExpired || invite.isRevoked ? presentationData.strings.InviteLink_PeopleJoinedShortNoneExpired : presentationData.strings.InviteLink_PeopleJoinedShortNone + } + if invite.isRevoked { + if !subtitleText.isEmpty { + subtitleText += " • " + } + subtitleText += presentationData.strings.InviteLink_Revoked + self.iconNode.image = generateTintedImage(image: UIImage(bundleImageName: "Chat/Links/Expired"), color: .white) + self.timerNode?.removeFromSupernode() + self.timerNode = nil + } else if let expireDate = invite.expireDate, currentTime > expireDate { + if !subtitleText.isEmpty { + subtitleText += " • " + } + subtitleText += presentationData.strings.InviteLink_Expired + self.iconNode.image = generateTintedImage(image: UIImage(bundleImageName: "Chat/Links/Expired"), color: .white) + self.timerNode?.removeFromSupernode() + self.timerNode = nil + } else if let expireDate = invite.expireDate { + self.iconNode.image = generateTintedImage(image: UIImage(bundleImageName: "Chat/Links/Flame"), color: .white) + let timerNode: TimerNode + if let current = self.timerNode { + timerNode = current + } else { + timerNode = TimerNode() + self.timerNode = timerNode + self.addSubnode(timerNode) + } + timerNode.update(color: UIColor.white, creationTimestamp: invite.date, deadlineTimestamp: expireDate) + } else { + self.iconNode.image = generateTintedImage(image: UIImage(bundleImageName: "Chat/Links/Link"), color: .white) + self.timerNode?.removeFromSupernode() + self.timerNode = nil + } + + self.iconNode.frame = CGRect(x: 10.0, y: 10.0, width: 30.0, height: 30.0) + self.timerNode?.frame = CGRect(x: 8.0, y: 8.0, width: 34.0, height: 34.0) + + let subtitle: NSMutableAttributedString = NSMutableAttributedString(string: subtitleText, font: subtitleFont, textColor: secondaryTextColor) + self.subtitleNode.attributedText = subtitle + + let titleSize = self.titleNode.updateLayout(CGSize(width: itemWidth - 24.0, height: 100.0)) + let subtitleSize = self.subtitleNode.updateLayout(CGSize(width: itemWidth - 24.0, height: 100.0)) + + self.titleNode.frame = CGRect(origin: CGPoint(x: 12.0, y: 52.0), size: titleSize) + self.subtitleNode.frame = CGRect(origin: CGPoint(x: 12.0, y: 52.0 + titleSize.height + 3.0), size: subtitleSize) + + let itemSize = CGSize(width: itemWidth, height: wide ? 102.0 : 122.0) + + let backgroundFrame = CGRect(origin: CGPoint(), size: itemSize) + transition.updateFrame(node: self.backgroundNode, frame: backgroundFrame) + + let buttonSize = CGSize(width: 26.0, height: 26.0) + let buttonFrame = CGRect(origin: CGPoint(x: itemSize.width - buttonSize.width - 12.0, y: 12.0), size: buttonSize) + transition.updateFrame(node: self.buttonNode, frame: buttonFrame) + + self.extractedContainerNode.frame = CGRect(origin: CGPoint(), size: buttonSize) + self.extractedContainerNode.contentRect = CGRect(origin: CGPoint(), size: buttonSize) + self.buttonIconNode.frame = CGRect(origin: CGPoint(), size: buttonSize) + + return itemSize + } +} + +class InviteLinksGridNode: ASDisplayNode { + private var items: [ExportedInvitation] = [] + private var itemNodes: [String: ItemNode] = [:] + + var action: ((ExportedInvitation) -> Void)? + var contextAction: ((ASDisplayNode, ExportedInvitation) -> Void)? + + override func hitTest(_ point: CGPoint, with event: UIEvent?) -> UIView? { + let result = super.hitTest(point, with: event) + return result + } + + func update(size: CGSize, safeInset: CGFloat, items: [ExportedInvitation], presentationData: ItemListPresentationData, transition: ContainedViewLayoutTransition) -> CGSize { + self.items = items + + var contentSize: CGSize = size + var contentHeight: CGFloat = 0.0 + + let sideInset: CGFloat = 16.0 + safeInset + + var validIds = Set() + + for i in 0 ..< self.items.count { + let invite = self.items[i] + validIds.insert(invite.link) + var itemNode: ItemNode? + var wasAdded = false + if let current = self.itemNodes[invite.link] { + itemNode = current + } else { + wasAdded = true + let addedItemNode = ItemNode() + itemNode = addedItemNode + self.itemNodes[invite.link] = addedItemNode + self.addSubnode(addedItemNode) + } + if let itemNode = itemNode { + let col = CGFloat(i % 2) + let row = floor(CGFloat(i) / 2.0) + let wide = (i == self.items.count - 1 && (self.items.count % 2) != 0) + let itemSize = itemNode.update(size: CGSize(width: size.width - sideInset * 2.0, height: size.height), wide: wide, invite: invite, presentationData: presentationData, transition: transition) + var itemFrame = CGRect(origin: CGPoint(x: sideInset, y: 4.0 + row * (122.0 + itemSpacing)), size: itemSize) + if !wide && col > 0 { + itemFrame.origin.x += itemSpacing + itemSize.width + } + + contentHeight = max(contentHeight, itemFrame.maxY + itemSpacing) + + if wasAdded { + itemNode.frame = itemFrame + } else { + transition.updateFrame(node: itemNode, frame: itemFrame) + } + itemNode.action = { [weak self] in + self?.action?(invite) + } + itemNode.contextAction = { [weak self] node in + self?.contextAction?(node, invite) + } + } + } + + var removeIds: [String] = [] + for (id, _) in self.itemNodes { + if !validIds.contains(id) { + removeIds.append(id) + } + } + for id in removeIds { + if let itemNode = self.itemNodes.removeValue(forKey: id) { + itemNode.removeFromSupernode() + } + } + + contentSize.height = contentHeight + return contentSize + } +} + +private struct ContentParticle { + var position: CGPoint + var direction: CGPoint + var velocity: CGFloat + var alpha: CGFloat + var lifetime: Double + var beginTime: Double + + init(position: CGPoint, direction: CGPoint, velocity: CGFloat, alpha: CGFloat, lifetime: Double, beginTime: Double) { + self.position = position + self.direction = direction + self.velocity = velocity + self.alpha = alpha + self.lifetime = lifetime + self.beginTime = beginTime + } +} + +private final class TimerNode: ASDisplayNode { + private struct Params: Equatable { + var color: UIColor + var creationTimestamp: Int32 + var deadlineTimestamp: Int32 + } + + private let hierarchyTrackingNode: HierarchyTrackingNode + private var inHierarchyValue: Bool = false + + private var animator: ConstantDisplayLinkAnimator? + private let contentNode: ASDisplayNode + private var particles: [ContentParticle] = [] + + private var currentParams: Params? + + var reachedTimeout: (() -> Void)? + + override init() { + var updateInHierarchy: ((Bool) -> Void)? + self.hierarchyTrackingNode = HierarchyTrackingNode({ value in + updateInHierarchy?(value) + }) + + self.contentNode = ASDisplayNode() + + super.init() + + self.addSubnode(self.contentNode) + + updateInHierarchy = { [weak self] value in + guard let strongSelf = self else { + return + } + strongSelf.inHierarchyValue = value + strongSelf.animator?.isPaused = value + } + } + + deinit { + self.animator?.invalidate() + } + + func update(color: UIColor, creationTimestamp: Int32, deadlineTimestamp: Int32) { + let params = Params( + color: color, + creationTimestamp: creationTimestamp, + deadlineTimestamp: deadlineTimestamp + ) + self.currentParams = params + + self.updateValues() + } + + private func updateValues() { + guard let params = self.currentParams else { + return + } + + let color = params.color + + let currentTimestamp = Int32(CFAbsoluteTimeGetCurrent() + NSTimeIntervalSince1970) + var fraction = CGFloat(params.deadlineTimestamp - currentTimestamp) / CGFloat(params.deadlineTimestamp - params.creationTimestamp) + fraction = 1.0 - max(0.0, min(0.94, fraction)) + + let image: UIImage? + + let diameter: CGFloat = 26.0 + let inset: CGFloat = 8.0 + let lineWidth: CGFloat = 2.0 + + let timestamp = CACurrentMediaTime() + + let center = CGPoint(x: (diameter + inset) / 2.0, y: (diameter + inset) / 2.0) + let radius: CGFloat = (diameter - lineWidth / 2.0) / 2.0 + + let startAngle: CGFloat = -CGFloat.pi / 2.0 + let endAngle: CGFloat = -CGFloat.pi / 2.0 + 2.0 * CGFloat.pi * fraction + + let v = CGPoint(x: sin(endAngle), y: -cos(endAngle)) + let c = CGPoint(x: -v.y * radius + center.x, y: v.x * radius + center.y) + + let dt: CGFloat = 1.0 / 60.0 + var removeIndices: [Int] = [] + for i in 0 ..< self.particles.count { + let currentTime = timestamp - self.particles[i].beginTime + if currentTime > self.particles[i].lifetime { + removeIndices.append(i) + } else { + let input: CGFloat = CGFloat(currentTime / self.particles[i].lifetime) + let decelerated: CGFloat = (1.0 - (1.0 - input) * (1.0 - input)) + self.particles[i].alpha = 1.0 - decelerated + + var p = self.particles[i].position + let d = self.particles[i].direction + let v = self.particles[i].velocity + p = CGPoint(x: p.x + d.x * v * dt, y: p.y + d.y * v * dt) + self.particles[i].position = p + } + } + + for i in removeIndices.reversed() { + self.particles.remove(at: i) + } + + let newParticleCount = 1 + for _ in 0 ..< newParticleCount { + let degrees: CGFloat = CGFloat(arc4random_uniform(140)) - 40.0 + let angle: CGFloat = degrees * CGFloat.pi / 180.0 + + let direction = CGPoint(x: v.x * cos(angle) - v.y * sin(angle), y: v.x * sin(angle) + v.y * cos(angle)) + let velocity = (20.0 + (CGFloat(arc4random()) / CGFloat(UINT32_MAX)) * 4.0) * 0.3 + + let lifetime = Double(0.4 + CGFloat(arc4random_uniform(100)) * 0.01) + + let particle = ContentParticle(position: c, direction: direction, velocity: velocity, alpha: 1.0, lifetime: lifetime, beginTime: timestamp) + self.particles.append(particle) + } + + image = generateImage(CGSize(width: diameter + inset, height: diameter + inset), rotatedContext: { size, context in + context.clear(CGRect(origin: CGPoint(), size: size)) + context.setStrokeColor(color.cgColor) + context.setFillColor(color.cgColor) + context.setLineWidth(lineWidth) + context.setLineCap(.round) + + let path = CGMutablePath() + path.addArc(center: center, radius: radius, startAngle: startAngle, endAngle: endAngle, clockwise: true) + context.addPath(path) + context.strokePath() + + for particle in self.particles { + let size: CGFloat = 2.0 + context.setAlpha(particle.alpha) + context.fillEllipse(in: CGRect(origin: CGPoint(x: particle.position.x - size / 2.0, y: particle.position.y - size / 2.0), size: CGSize(width: size, height: size))) + } + }) + + self.contentNode.contents = image?.cgImage + if let image = image { + self.contentNode.frame = CGRect(origin: CGPoint(), size: image.size) + } + + if fraction <= .ulpOfOne { + self.animator?.invalidate() + self.animator = nil + } else { + if self.animator == nil { + let animator = ConstantDisplayLinkAnimator(update: { [weak self] in + self?.updateValues() + }) + self.animator = animator + animator.isPaused = self.inHierarchyValue + } + } + } +} diff --git a/submodules/InviteLinksUI/Sources/ItemListDatePickerItem.swift b/submodules/InviteLinksUI/Sources/ItemListDatePickerItem.swift new file mode 100644 index 0000000000..5d9a244caa --- /dev/null +++ b/submodules/InviteLinksUI/Sources/ItemListDatePickerItem.swift @@ -0,0 +1,241 @@ +import Foundation +import UIKit +import Display +import AsyncDisplayKit +import SwiftSignalKit +import SyncCore +import TelegramPresentationData +import ItemListUI + +public class ItemListDatePickerItem: ListViewItem, ItemListItem { + let presentationData: ItemListPresentationData + let date: Int32? + public let sectionId: ItemListSectionId + let style: ItemListStyle + let updated: ((Int32) -> Void)? + public let tag: ItemListItemTag? + + public init( + presentationData: ItemListPresentationData, + date: Int32?, + sectionId: ItemListSectionId, + style: ItemListStyle, + updated: ((Int32) -> Void)?, + tag: ItemListItemTag? = nil + ) { + self.presentationData = presentationData + self.date = date + self.sectionId = sectionId + self.style = style + self.updated = updated + self.tag = tag + } + + public func nodeConfiguredForParams(async: @escaping (@escaping () -> Void) -> Void, params: ListViewItemLayoutParams, synchronousLoads: Bool, previousItem: ListViewItem?, nextItem: ListViewItem?, completion: @escaping (ListViewItemNode, @escaping () -> (Signal?, (ListViewItemApply) -> Void)) -> Void) { + async { + let node = ItemListDatePickerItemNode() + let (layout, apply) = node.asyncLayout()(self, params, itemListNeighbors(item: self, topItem: previousItem as? ItemListItem, bottomItem: nextItem as? ItemListItem)) + + node.contentSize = layout.contentSize + node.insets = layout.insets + + Queue.mainQueue().async { + completion(node, { + return (nil, { _ in apply() }) + }) + } + } + } + + public func updateNode(async: @escaping (@escaping () -> Void) -> Void, node: @escaping () -> ListViewItemNode, params: ListViewItemLayoutParams, previousItem: ListViewItem?, nextItem: ListViewItem?, animation: ListViewItemUpdateAnimation, completion: @escaping (ListViewItemNodeLayout, @escaping (ListViewItemApply) -> Void) -> Void) { + Queue.mainQueue().async { + if let nodeValue = node() as? ItemListDatePickerItemNode { + let makeLayout = nodeValue.asyncLayout() + + async { + let (layout, apply) = makeLayout(self, params, itemListNeighbors(item: self, topItem: previousItem as? ItemListItem, bottomItem: nextItem as? ItemListItem)) + Queue.mainQueue().async { + completion(layout, { _ in + apply() + }) + } + } + } + } + } + + public var selectable: Bool = false +} + +public class ItemListDatePickerItemNode: ListViewItemNode, ItemListItemNode { + private let backgroundNode: ASDisplayNode + private let topStripeNode: ASDisplayNode + private let bottomStripeNode: ASDisplayNode + private let maskNode: ASImageNode + + private let datePicker: UIDatePicker + + private var item: ItemListDatePickerItem? + + override public var canBeSelected: Bool { + return false + } + + public var tag: ItemListItemTag? { + return self.item?.tag + } + + public init() { + self.backgroundNode = ASDisplayNode() + self.backgroundNode.isLayerBacked = true + self.backgroundNode.backgroundColor = .white + + self.maskNode = ASImageNode() + + self.topStripeNode = ASDisplayNode() + self.topStripeNode.isLayerBacked = true + + self.bottomStripeNode = ASDisplayNode() + self.bottomStripeNode.isLayerBacked = true + + self.datePicker = UIDatePicker() + self.datePicker.minimumDate = Date() + self.datePicker.datePickerMode = .dateAndTime + if #available(iOS 14.0, *) { + self.datePicker.preferredDatePickerStyle = .inline + } + super.init(layerBacked: false, dynamicBounce: false) + + self.datePicker.addTarget(self, action: #selector(self.datePickerUpdated), for: .valueChanged) + } + + public override func didLoad() { + super.didLoad() + + self.view.addSubview(self.datePicker) + } + + @objc private func datePickerUpdated() { + self.item?.updated?(Int32(self.datePicker.date.timeIntervalSince1970)) + } + + public func asyncLayout() -> (_ item: ItemListDatePickerItem, _ params: ListViewItemLayoutParams, _ insets: ItemListNeighbors) -> (ListViewItemNodeLayout, () -> Void) { + let currentItem = self.item + + return { item, params, neighbors in + var updatedTheme: PresentationTheme? + if currentItem?.presentationData.theme !== item.presentationData.theme { + updatedTheme = item.presentationData.theme + } + + let contentSize: CGSize + let insets: UIEdgeInsets + let separatorHeight = UIScreenPixel + let itemBackgroundColor: UIColor + let itemSeparatorColor: UIColor + + let leftInset = 16.0 + params.leftInset + let rightInset = 16.0 + params.rightInset + + let height: CGFloat = 360.0 + + switch item.style { + case .plain: + itemBackgroundColor = item.presentationData.theme.list.plainBackgroundColor + itemSeparatorColor = item.presentationData.theme.list.itemPlainSeparatorColor + contentSize = CGSize(width: params.width, height: height) + insets = itemListNeighborsPlainInsets(neighbors) + case .blocks: + itemBackgroundColor = item.presentationData.theme.list.itemBlocksBackgroundColor + itemSeparatorColor = item.presentationData.theme.list.itemBlocksSeparatorColor + contentSize = CGSize(width: params.width, height: height) + insets = itemListNeighborsGroupedInsets(neighbors) + } + + return (ListViewItemNodeLayout(contentSize: contentSize, insets: insets), { [weak self] in + if let strongSelf = self { + strongSelf.item = item + + if let _ = updatedTheme { + strongSelf.topStripeNode.backgroundColor = itemSeparatorColor + strongSelf.bottomStripeNode.backgroundColor = itemSeparatorColor + strongSelf.backgroundNode.backgroundColor = itemBackgroundColor + } + + strongSelf.datePicker.date = item.date.flatMap { Date(timeIntervalSince1970: TimeInterval($0)) } ?? Date() + strongSelf.datePicker.frame = CGRect(origin: CGPoint(x: 16.0, y: 3.0), size: CGSize(width: contentSize.width - 32.0, height: contentSize.height)) + + switch item.style { + case .plain: + if strongSelf.backgroundNode.supernode != nil { + strongSelf.backgroundNode.removeFromSupernode() + } + if strongSelf.topStripeNode.supernode != nil { + strongSelf.topStripeNode.removeFromSupernode() + } + if strongSelf.bottomStripeNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.bottomStripeNode, at: 0) + } + if strongSelf.maskNode.supernode != nil { + strongSelf.maskNode.removeFromSupernode() + } + strongSelf.bottomStripeNode.frame = CGRect(origin: CGPoint(x: leftInset, y: contentSize.height - separatorHeight), size: CGSize(width: params.width - leftInset, height: separatorHeight)) + case .blocks: + if strongSelf.backgroundNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.backgroundNode, at: 0) + } + if strongSelf.topStripeNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.topStripeNode, at: 1) + } + if strongSelf.bottomStripeNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.bottomStripeNode, at: 2) + } + if strongSelf.maskNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.maskNode, at: 3) + } + + let hasCorners = itemListHasRoundedBlockLayout(params) + var hasTopCorners = false + var hasBottomCorners = false + switch neighbors.top { + case .sameSection(false): + strongSelf.topStripeNode.isHidden = true + default: + hasTopCorners = true + strongSelf.topStripeNode.isHidden = hasCorners + } + let bottomStripeInset: CGFloat + switch neighbors.bottom { + case .sameSection(false): + bottomStripeInset = leftInset + default: + bottomStripeInset = 0.0 + hasBottomCorners = true + strongSelf.bottomStripeNode.isHidden = hasCorners + } + + strongSelf.maskNode.image = hasCorners ? PresentationResourcesItemList.cornersImage(item.presentationData.theme, top: hasTopCorners, bottom: hasBottomCorners) : nil + + strongSelf.backgroundNode.frame = CGRect(origin: CGPoint(x: 0.0, y: -min(insets.top, separatorHeight)), size: CGSize(width: params.width, height: contentSize.height + min(insets.top, separatorHeight) + min(insets.bottom, separatorHeight))) + strongSelf.maskNode.frame = strongSelf.backgroundNode.frame.insetBy(dx: params.leftInset, dy: 0.0) + strongSelf.topStripeNode.frame = CGRect(origin: CGPoint(x: 0.0, y: -min(insets.top, separatorHeight)), size: CGSize(width: params.width, height: separatorHeight)) + strongSelf.bottomStripeNode.frame = CGRect(origin: CGPoint(x: bottomStripeInset, y: contentSize.height - separatorHeight), size: CGSize(width: params.width - bottomStripeInset, height: separatorHeight)) + } + } + }) + } + } + + override public func animateInsertion(_ currentTimestamp: Double, duration: Double, short: Bool) { + self.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.4) + } + + override public func animateAdded(_ currentTimestamp: Double, duration: Double) { + self.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.2) + } + + override public func animateRemoved(_ currentTimestamp: Double, duration: Double) { + self.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.15, removeOnCompletion: false) + } +} + diff --git a/submodules/InviteLinksUI/Sources/ItemListInviteLinkDateLimitItem.swift b/submodules/InviteLinksUI/Sources/ItemListInviteLinkDateLimitItem.swift new file mode 100644 index 0000000000..73f3d266ad --- /dev/null +++ b/submodules/InviteLinksUI/Sources/ItemListInviteLinkDateLimitItem.swift @@ -0,0 +1,409 @@ +import Foundation +import UIKit +import Display +import AsyncDisplayKit +import SwiftSignalKit +import TelegramCore +import SyncCore +import TelegramUIPreferences +import TelegramPresentationData +import LegacyComponents +import ItemListUI +import PresentationDataUtils + +enum InviteLinkTimeLimit: Equatable { + case hour + case day + case week + case unlimited + case custom(Int32) + + init(position: Int) { + switch position { + case 0: + self = .hour + case 1: + self = .day + case 2: + self = .week + default: + self = .unlimited + } + } + + var value: Int32? { + switch self { + case .hour: + return 3600 + case .day: + return 86400 + case .week: + return 604800 + case .unlimited: + return nil + case let .custom(value): + return value + } + } + + var position: Int { + switch self { + case .hour: + return 0 + case .day: + return 1 + case .week: + return 2 + case .unlimited: + return 3 + case let .custom(value): + let currentTime = Int32(CFAbsoluteTimeGetCurrent() + kCFAbsoluteTimeIntervalSince1970) + let relativeValue = value - currentTime + if relativeValue < 3600 { + return 0 + } else if relativeValue < 86400 { + return 1 + } else if relativeValue < 604800 { + return 2 + } else { + return 3 + } + } + } +} + +final class ItemListInviteLinkTimeLimitItem: ListViewItem, ItemListItem { + let theme: PresentationTheme + let strings: PresentationStrings + let value: InviteLinkTimeLimit + let enabled: Bool + let sectionId: ItemListSectionId + let updated: (InviteLinkTimeLimit) -> Void + + init(theme: PresentationTheme, strings: PresentationStrings, value: InviteLinkTimeLimit, enabled: Bool, sectionId: ItemListSectionId, updated: @escaping (InviteLinkTimeLimit) -> Void) { + self.theme = theme + self.strings = strings + self.value = value + self.enabled = enabled + self.sectionId = sectionId + self.updated = updated + } + + func nodeConfiguredForParams(async: @escaping (@escaping () -> Void) -> Void, params: ListViewItemLayoutParams, synchronousLoads: Bool, previousItem: ListViewItem?, nextItem: ListViewItem?, completion: @escaping (ListViewItemNode, @escaping () -> (Signal?, (ListViewItemApply) -> Void)) -> Void) { + async { + let node = ItemListInviteLinkTimeLimitItemNode() + let (layout, apply) = node.asyncLayout()(self, params, itemListNeighbors(item: self, topItem: previousItem as? ItemListItem, bottomItem: nextItem as? ItemListItem)) + + node.contentSize = layout.contentSize + node.insets = layout.insets + + Queue.mainQueue().async { + completion(node, { + return (nil, { _ in apply() }) + }) + } + } + } + + func updateNode(async: @escaping (@escaping () -> Void) -> Void, node: @escaping () -> ListViewItemNode, params: ListViewItemLayoutParams, previousItem: ListViewItem?, nextItem: ListViewItem?, animation: ListViewItemUpdateAnimation, completion: @escaping (ListViewItemNodeLayout, @escaping (ListViewItemApply) -> Void) -> Void) { + Queue.mainQueue().async { + if let nodeValue = node() as? ItemListInviteLinkTimeLimitItemNode { + let makeLayout = nodeValue.asyncLayout() + + async { + let (layout, apply) = makeLayout(self, params, itemListNeighbors(item: self, topItem: previousItem as? ItemListItem, bottomItem: nextItem as? ItemListItem)) + Queue.mainQueue().async { + completion(layout, { _ in + apply() + }) + } + } + } + } + } +} + +private func generateKnobImage() -> UIImage? { + return generateImage(CGSize(width: 40.0, height: 40.0), rotatedContext: { size, context in + context.clear(CGRect(origin: CGPoint(), size: size)) + context.setShadow(offset: CGSize(width: 0.0, height: -2.0), blur: 3.5, color: UIColor(white: 0.0, alpha: 0.35).cgColor) + context.setFillColor(UIColor.white.cgColor) + context.fillEllipse(in: CGRect(origin: CGPoint(x: 6.0, y: 6.0), size: CGSize(width: 28.0, height: 28.0))) + }) +} + +private final class ItemListInviteLinkTimeLimitItemNode: ListViewItemNode { + private let backgroundNode: ASDisplayNode + private let topStripeNode: ASDisplayNode + private let bottomStripeNode: ASDisplayNode + private let maskNode: ASImageNode + + private let lowTextNode: TextNode + private let mediumTextNode: TextNode + private let highTextNode: TextNode + private let unlimitedTextNode: TextNode + private let customTextNode: TextNode + private var sliderView: TGPhotoEditorSliderView? + + private var item: ItemListInviteLinkTimeLimitItem? + private var layoutParams: ListViewItemLayoutParams? + + init() { + self.backgroundNode = ASDisplayNode() + self.backgroundNode.isLayerBacked = true + + self.topStripeNode = ASDisplayNode() + self.topStripeNode.isLayerBacked = true + + self.bottomStripeNode = ASDisplayNode() + self.bottomStripeNode.isLayerBacked = true + + self.maskNode = ASImageNode() + + self.lowTextNode = TextNode() + self.lowTextNode.isUserInteractionEnabled = false + self.lowTextNode.displaysAsynchronously = false + + self.mediumTextNode = TextNode() + self.mediumTextNode.isUserInteractionEnabled = false + self.mediumTextNode.displaysAsynchronously = false + + self.highTextNode = TextNode() + self.highTextNode.isUserInteractionEnabled = false + self.highTextNode.displaysAsynchronously = false + + self.unlimitedTextNode = TextNode() + self.unlimitedTextNode.isUserInteractionEnabled = false + self.unlimitedTextNode.displaysAsynchronously = false + + self.customTextNode = TextNode() + self.customTextNode.isUserInteractionEnabled = false + self.customTextNode.displaysAsynchronously = false + + super.init(layerBacked: false, dynamicBounce: false) + + self.addSubnode(self.lowTextNode) + self.addSubnode(self.mediumTextNode) + self.addSubnode(self.highTextNode) + self.addSubnode(self.unlimitedTextNode) + self.addSubnode(self.customTextNode) + } + + func updateSliderView() { + if let sliderView = self.sliderView, let item = self.item { + if case .custom = item.value { + sliderView.maximumValue = 3.0 + 1 + sliderView.positionsCount = 4 + 1 + } else { + sliderView.maximumValue = 3.0 + sliderView.positionsCount = 4 + } + sliderView.value = CGFloat(item.value.position) + + sliderView.isUserInteractionEnabled = item.enabled + sliderView.alpha = item.enabled ? 1.0 : 0.4 + sliderView.layer.allowsGroupOpacity = !item.enabled + } + } + + override func didLoad() { + super.didLoad() + + let sliderView = TGPhotoEditorSliderView() + sliderView.enablePanHandling = true + sliderView.trackCornerRadius = 1.0 + sliderView.lineSize = 2.0 + sliderView.dotSize = 5.0 + sliderView.minimumValue = 0.0 + sliderView.startValue = 0.0 + sliderView.disablesInteractiveTransitionGestureRecognizer = true + if let item = self.item, case .custom = item.value { + sliderView.maximumValue = 3.0 + 1 + sliderView.positionsCount = 4 + 1 + } else { + sliderView.maximumValue = 3.0 + sliderView.positionsCount = 4 + } + sliderView.useLinesForPositions = true + if let item = self.item, let params = self.layoutParams { + sliderView.value = CGFloat(item.value.position) + sliderView.backgroundColor = item.theme.list.itemBlocksBackgroundColor + sliderView.backColor = item.theme.list.disclosureArrowColor + sliderView.startColor = item.theme.list.disclosureArrowColor + sliderView.trackColor = item.theme.list.itemAccentColor + sliderView.knobImage = generateKnobImage() + + sliderView.frame = CGRect(origin: CGPoint(x: params.leftInset + 15.0, y: 37.0), size: CGSize(width: params.width - params.leftInset - params.rightInset - 15.0 * 2.0, height: 44.0)) + sliderView.hitTestEdgeInsets = UIEdgeInsets(top: -sliderView.frame.minX, left: 0.0, bottom: 0.0, right: -sliderView.frame.minX) + } + self.view.addSubview(sliderView) + sliderView.addTarget(self, action: #selector(self.sliderValueChanged), for: .valueChanged) + self.sliderView = sliderView + + self.updateSliderView() + } + + func asyncLayout() -> (_ item: ItemListInviteLinkTimeLimitItem, _ params: ListViewItemLayoutParams, _ neighbors: ItemListNeighbors) -> (ListViewItemNodeLayout, () -> Void) { + let currentItem = self.item + let makeLowTextLayout = TextNode.asyncLayout(self.lowTextNode) + let makeMediumTextLayout = TextNode.asyncLayout(self.mediumTextNode) + let makeHighTextLayout = TextNode.asyncLayout(self.highTextNode) + let makeUnlimitedTextLayout = TextNode.asyncLayout(self.unlimitedTextNode) + let makeCustomTextLayout = TextNode.asyncLayout(self.customTextNode) + + return { item, params, neighbors in + var themeUpdated = false + if currentItem?.theme !== item.theme { + themeUpdated = true + } + + let contentSize: CGSize + let insets: UIEdgeInsets + let separatorHeight = UIScreenPixel + + let (lowTextLayout, lowTextApply) = makeLowTextLayout(TextNodeLayoutArguments(attributedString: NSAttributedString(string: timeIntervalString(strings: item.strings, value: 3600), font: Font.regular(13.0), textColor: item.theme.list.itemSecondaryTextColor), backgroundColor: nil, maximumNumberOfLines: 1, truncationType: .end, constrainedSize: CGSize(width: params.width, height: CGFloat.greatestFiniteMagnitude), alignment: .center, lineSpacing: 0.0, cutout: nil, insets: UIEdgeInsets())) + + let (mediumTextLayout, mediumTextApply) = makeMediumTextLayout(TextNodeLayoutArguments(attributedString: NSAttributedString(string: timeIntervalString(strings: item.strings, value: 86400), font: Font.regular(13.0), textColor: item.theme.list.itemSecondaryTextColor), backgroundColor: nil, maximumNumberOfLines: 1, truncationType: .end, constrainedSize: CGSize(width: params.width, height: CGFloat.greatestFiniteMagnitude), alignment: .center, lineSpacing: 0.0, cutout: nil, insets: UIEdgeInsets())) + + let (highTextLayout, highTextApply) = makeHighTextLayout(TextNodeLayoutArguments(attributedString: NSAttributedString(string: timeIntervalString(strings: item.strings, value: 604800), font: Font.regular(13.0), textColor: item.theme.list.itemSecondaryTextColor), backgroundColor: nil, maximumNumberOfLines: 1, truncationType: .end, constrainedSize: CGSize(width: params.width, height: CGFloat.greatestFiniteMagnitude), alignment: .center, lineSpacing: 0.0, cutout: nil, insets: UIEdgeInsets())) + + let (unlimitedTextLayout, unlimitedTextApply) = makeUnlimitedTextLayout(TextNodeLayoutArguments(attributedString: NSAttributedString(string: item.strings.InviteLink_Create_TimeLimitNoLimit, font: Font.regular(13.0), textColor: item.theme.list.itemSecondaryTextColor), backgroundColor: nil, maximumNumberOfLines: 1, truncationType: .end, constrainedSize: CGSize(width: params.width, height: CGFloat.greatestFiniteMagnitude), alignment: .center, lineSpacing: 0.0, cutout: nil, insets: UIEdgeInsets())) + + let customTextString: String + if case let .custom(value) = item.value { + let currentTime = Int32(CFAbsoluteTimeGetCurrent() + kCFAbsoluteTimeIntervalSince1970) + let relativeValue = value - currentTime + if relativeValue > 0 { + customTextString = timeIntervalString(strings: item.strings, value: relativeValue) + } else { + customTextString = "" + } + } else { + customTextString = "" + } + + let (customTextLayout, customTextApply) = makeCustomTextLayout(TextNodeLayoutArguments(attributedString: NSAttributedString(string: customTextString, font: Font.regular(13.0), textColor: item.theme.list.itemSecondaryTextColor), backgroundColor: nil, maximumNumberOfLines: 1, truncationType: .end, constrainedSize: CGSize(width: params.width, height: CGFloat.greatestFiniteMagnitude), alignment: .center, lineSpacing: 0.0, cutout: nil, insets: UIEdgeInsets())) + + contentSize = CGSize(width: params.width, height: 88.0) + insets = itemListNeighborsGroupedInsets(neighbors) + + let layout = ListViewItemNodeLayout(contentSize: contentSize, insets: insets) + let layoutSize = layout.size + + return (layout, { [weak self] in + if let strongSelf = self { + strongSelf.item = item + strongSelf.layoutParams = params + + strongSelf.backgroundNode.backgroundColor = item.theme.list.itemBlocksBackgroundColor + strongSelf.topStripeNode.backgroundColor = item.theme.list.itemBlocksSeparatorColor + strongSelf.bottomStripeNode.backgroundColor = item.theme.list.itemBlocksSeparatorColor + + if strongSelf.backgroundNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.backgroundNode, at: 0) + } + if strongSelf.topStripeNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.topStripeNode, at: 1) + } + if strongSelf.bottomStripeNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.bottomStripeNode, at: 2) + } + if strongSelf.maskNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.maskNode, at: 3) + } + + let hasCorners = itemListHasRoundedBlockLayout(params) + var hasTopCorners = false + var hasBottomCorners = false + switch neighbors.top { + case .sameSection(false): + strongSelf.topStripeNode.isHidden = true + default: + hasTopCorners = true + strongSelf.topStripeNode.isHidden = hasCorners + } + let bottomStripeInset: CGFloat + let bottomStripeOffset: CGFloat + switch neighbors.bottom { + case .sameSection(false): + bottomStripeInset = 0.0 //params.leftInset + 16.0 + bottomStripeOffset = -separatorHeight + default: + bottomStripeInset = 0.0 + bottomStripeOffset = 0.0 + hasBottomCorners = true + strongSelf.bottomStripeNode.isHidden = hasCorners + } + + strongSelf.maskNode.image = hasCorners ? PresentationResourcesItemList.cornersImage(item.theme, top: hasTopCorners, bottom: hasBottomCorners) : nil + + strongSelf.backgroundNode.frame = CGRect(origin: CGPoint(x: 0.0, y: -min(insets.top, separatorHeight)), size: CGSize(width: params.width, height: contentSize.height + min(insets.top, separatorHeight) + min(insets.bottom, separatorHeight))) + strongSelf.maskNode.frame = strongSelf.backgroundNode.frame.insetBy(dx: params.leftInset, dy: 0.0) + strongSelf.topStripeNode.frame = CGRect(origin: CGPoint(x: 0.0, y: -min(insets.top, separatorHeight)), size: CGSize(width: layoutSize.width, height: separatorHeight)) + strongSelf.bottomStripeNode.frame = CGRect(origin: CGPoint(x: bottomStripeInset, y: contentSize.height + bottomStripeOffset), size: CGSize(width: layoutSize.width - bottomStripeInset, height: separatorHeight)) + + let _ = lowTextApply() + let _ = mediumTextApply() + let _ = highTextApply() + let _ = unlimitedTextApply() + let _ = customTextApply() + + var textNodes: [(TextNode, CGSize)] = [(strongSelf.lowTextNode, lowTextLayout.size), + (strongSelf.mediumTextNode, mediumTextLayout.size), + (strongSelf.highTextNode, highTextLayout.size), + (strongSelf.unlimitedTextNode, unlimitedTextLayout.size)] + if case .custom = item.value { + textNodes.insert((strongSelf.customTextNode, customTextLayout.size), at: item.value.position) + } + + let delta = (params.width - params.leftInset - params.rightInset - 18.0 * 2.0) / CGFloat(textNodes.count - 1) + for i in 0 ..< textNodes.count { + let (textNode, textSize) = textNodes[i] + + var position = params.leftInset + 18.0 + delta * CGFloat(i) + if i == textNodes.count - 1 { + position -= textSize.width + } else if i > 0 { + position -= textSize.width / 2.0 + } + + textNode.frame = CGRect(origin: CGPoint(x: position, y: 15.0), size: textSize) + } + + if let sliderView = strongSelf.sliderView { + if themeUpdated { + sliderView.backgroundColor = item.theme.list.itemBlocksBackgroundColor + sliderView.backColor = item.theme.list.disclosureArrowColor + sliderView.trackColor = item.theme.list.itemAccentColor + sliderView.knobImage = generateKnobImage() + } + + sliderView.frame = CGRect(origin: CGPoint(x: params.leftInset + 15.0, y: 37.0), size: CGSize(width: params.width - params.leftInset - params.rightInset - 15.0 * 2.0, height: 44.0)) + sliderView.hitTestEdgeInsets = UIEdgeInsets(top: -sliderView.frame.minX, left: 0.0, bottom: 0.0, right: -sliderView.frame.minX) + + strongSelf.updateSliderView() + } + } + }) + } + } + + override func animateInsertion(_ currentTimestamp: Double, duration: Double, short: Bool) { + self.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.4) + } + + override func animateRemoved(_ currentTimestamp: Double, duration: Double) { + self.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.15, removeOnCompletion: false) + } + + @objc func sliderValueChanged() { + guard let sliderView = self.sliderView else { + return + } + + let position = Int(sliderView.value) + let value = InviteLinkTimeLimit(position: position) + self.item?.updated(value) + } +} + diff --git a/submodules/InviteLinksUI/Sources/ItemListInviteLinkGridItem.swift b/submodules/InviteLinksUI/Sources/ItemListInviteLinkGridItem.swift new file mode 100644 index 0000000000..0e587b8db2 --- /dev/null +++ b/submodules/InviteLinksUI/Sources/ItemListInviteLinkGridItem.swift @@ -0,0 +1,253 @@ +import Foundation +import UIKit +import Display +import AsyncDisplayKit +import SwiftSignalKit +import SyncCore +import TelegramPresentationData +import ItemListUI + +public class ItemListInviteLinkGridItem: ListViewItem, ItemListItem { + let presentationData: ItemListPresentationData + let invites: [ExportedInvitation]? + public let sectionId: ItemListSectionId + let style: ItemListStyle + let tapAction: ((ExportedInvitation) -> Void)? + let contextAction: ((ExportedInvitation, ASDisplayNode) -> Void)? + public let tag: ItemListItemTag? + + public init( + presentationData: ItemListPresentationData, + invites: [ExportedInvitation]?, + sectionId: ItemListSectionId, + style: ItemListStyle, + tapAction: ((ExportedInvitation) -> Void)?, + contextAction: ((ExportedInvitation, ASDisplayNode) -> Void)?, + tag: ItemListItemTag? = nil + ) { + self.presentationData = presentationData + self.invites = invites + self.sectionId = sectionId + self.style = style + self.tapAction = tapAction + self.contextAction = contextAction + self.tag = tag + } + + public func nodeConfiguredForParams(async: @escaping (@escaping () -> Void) -> Void, params: ListViewItemLayoutParams, synchronousLoads: Bool, previousItem: ListViewItem?, nextItem: ListViewItem?, completion: @escaping (ListViewItemNode, @escaping () -> (Signal?, (ListViewItemApply) -> Void)) -> Void) { + async { + let node = ItemListInviteLinkGridItemNode() + let (layout, apply) = node.asyncLayout()(self, params, itemListNeighbors(item: self, topItem: previousItem as? ItemListItem, bottomItem: nextItem as? ItemListItem)) + + node.contentSize = layout.contentSize + node.insets = layout.insets + + Queue.mainQueue().async { + completion(node, { + return (nil, { _ in apply() }) + }) + } + } + } + + public func updateNode(async: @escaping (@escaping () -> Void) -> Void, node: @escaping () -> ListViewItemNode, params: ListViewItemLayoutParams, previousItem: ListViewItem?, nextItem: ListViewItem?, animation: ListViewItemUpdateAnimation, completion: @escaping (ListViewItemNodeLayout, @escaping (ListViewItemApply) -> Void) -> Void) { + Queue.mainQueue().async { + if let nodeValue = node() as? ItemListInviteLinkGridItemNode { + let makeLayout = nodeValue.asyncLayout() + + async { + let (layout, apply) = makeLayout(self, params, itemListNeighbors(item: self, topItem: previousItem as? ItemListItem, bottomItem: nextItem as? ItemListItem)) + Queue.mainQueue().async { + completion(layout, { _ in + apply() + }) + } + } + } + } + } + + public var selectable: Bool = false + + public func selected(listView: ListView){ + } +} + +public class ItemListInviteLinkGridItemNode: ListViewItemNode, ItemListItemNode { + private let backgroundNode: ASDisplayNode + private let topStripeNode: ASDisplayNode + private let bottomStripeNode: ASDisplayNode + private let maskNode: ASImageNode + + private let gridNode: InviteLinksGridNode + + private var item: ItemListInviteLinkGridItem? + + override public var canBeSelected: Bool { + return false + } + + public var tag: ItemListItemTag? { + return self.item?.tag + } + + public init() { + self.backgroundNode = ASDisplayNode() + self.backgroundNode.isLayerBacked = true + self.backgroundNode.backgroundColor = .white + + self.maskNode = ASImageNode() + + self.topStripeNode = ASDisplayNode() + self.topStripeNode.isLayerBacked = true + + self.bottomStripeNode = ASDisplayNode() + self.bottomStripeNode.isLayerBacked = true + + self.gridNode = InviteLinksGridNode() + + super.init(layerBacked: false, dynamicBounce: false) + + self.addSubnode(self.gridNode) + } + + public func asyncLayout() -> (_ item: ItemListInviteLinkGridItem, _ params: ListViewItemLayoutParams, _ insets: ItemListNeighbors) -> (ListViewItemNodeLayout, () -> Void) { + let currentItem = self.item + + return { item, params, neighbors in + var updatedTheme: PresentationTheme? + if currentItem?.presentationData.theme !== item.presentationData.theme { + updatedTheme = item.presentationData.theme + } + + let contentSize: CGSize + let insets: UIEdgeInsets + let separatorHeight = UIScreenPixel + let itemBackgroundColor: UIColor + let itemSeparatorColor: UIColor + + let leftInset = 16.0 + params.leftInset + let rightInset = 16.0 + params.rightInset + + var height: CGFloat + let count = item.invites?.count ?? 0 + if count > 0 { + if count % 2 == 0 { + height = 4.0 + 122.0 + 6.0 + } else { + height = 4.0 + 102.0 + 6.0 + } + } else { + height = 0.001 + } + + switch item.style { + case .plain: + itemBackgroundColor = item.presentationData.theme.list.plainBackgroundColor + itemSeparatorColor = item.presentationData.theme.list.itemPlainSeparatorColor + insets = itemListNeighborsPlainInsets(neighbors) + case .blocks: + itemBackgroundColor = item.presentationData.theme.list.itemBlocksBackgroundColor + itemSeparatorColor = item.presentationData.theme.list.itemBlocksSeparatorColor + insets = itemListNeighborsGroupedInsets(neighbors) + } + if case .sameSection(false) = neighbors.bottom { + } else { + height += 6.0 + } + contentSize = CGSize(width: params.width, height: height) + + return (ListViewItemNodeLayout(contentSize: contentSize, insets: insets), { [weak self] in + if let strongSelf = self { + strongSelf.item = item + + if let _ = updatedTheme { + strongSelf.topStripeNode.backgroundColor = itemSeparatorColor + strongSelf.bottomStripeNode.backgroundColor = itemSeparatorColor + strongSelf.backgroundNode.backgroundColor = itemBackgroundColor + } + + let gridSize = strongSelf.gridNode.update(size: contentSize, safeInset: params.leftInset, items: item.invites ?? [], presentationData: item.presentationData, transition: .immediate) + strongSelf.gridNode.frame = CGRect(origin: CGPoint(), size: gridSize) + strongSelf.gridNode.action = { invite in + item.tapAction?(invite) + } + strongSelf.gridNode.contextAction = { node, invite in + item.contextAction?(invite, node) + } + + switch item.style { + case .plain: + if strongSelf.backgroundNode.supernode != nil { + strongSelf.backgroundNode.removeFromSupernode() + } + if strongSelf.topStripeNode.supernode != nil { + strongSelf.topStripeNode.removeFromSupernode() + } + if strongSelf.bottomStripeNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.bottomStripeNode, at: 0) + } + if strongSelf.maskNode.supernode != nil { + strongSelf.maskNode.removeFromSupernode() + } + strongSelf.bottomStripeNode.frame = CGRect(origin: CGPoint(x: leftInset, y: contentSize.height - separatorHeight), size: CGSize(width: params.width - leftInset, height: separatorHeight)) + case .blocks: + if strongSelf.backgroundNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.backgroundNode, at: 0) + } + if strongSelf.topStripeNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.topStripeNode, at: 1) + } + if strongSelf.bottomStripeNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.bottomStripeNode, at: 2) + } + if strongSelf.maskNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.maskNode, at: 3) + } + + let hasCorners = itemListHasRoundedBlockLayout(params) + var hasTopCorners = false + var hasBottomCorners = false + switch neighbors.top { + case .sameSection(false): + strongSelf.topStripeNode.isHidden = true + default: + hasTopCorners = true + strongSelf.topStripeNode.isHidden = hasCorners + } + let bottomStripeInset: CGFloat + switch neighbors.bottom { + case .sameSection(false): + bottomStripeInset = leftInset + strongSelf.bottomStripeNode.isHidden = true + default: + bottomStripeInset = 0.0 + hasBottomCorners = true + strongSelf.bottomStripeNode.isHidden = hasCorners + } + + strongSelf.maskNode.image = hasCorners ? PresentationResourcesItemList.cornersImage(item.presentationData.theme, top: hasTopCorners, bottom: hasBottomCorners) : nil + + strongSelf.backgroundNode.frame = CGRect(origin: CGPoint(x: 0.0, y: -min(insets.top, separatorHeight)), size: CGSize(width: params.width, height: contentSize.height + min(insets.top, separatorHeight) + min(insets.bottom, separatorHeight))) + strongSelf.maskNode.frame = strongSelf.backgroundNode.frame.insetBy(dx: params.leftInset, dy: 0.0) + strongSelf.topStripeNode.frame = CGRect(origin: CGPoint(x: 0.0, y: -min(insets.top, separatorHeight)), size: CGSize(width: params.width, height: separatorHeight)) + strongSelf.bottomStripeNode.frame = CGRect(origin: CGPoint(x: bottomStripeInset, y: contentSize.height - separatorHeight), size: CGSize(width: params.width - bottomStripeInset, height: separatorHeight)) + } + } + }) + } + } + + override public func animateInsertion(_ currentTimestamp: Double, duration: Double, short: Bool) { + self.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.4) + } + + override public func animateAdded(_ currentTimestamp: Double, duration: Double) { + self.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.2) + } + + override public func animateRemoved(_ currentTimestamp: Double, duration: Double) { + self.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.15, removeOnCompletion: false) + } +} + diff --git a/submodules/InviteLinksUI/Sources/ItemListInviteLinkUsageLimitItem.swift b/submodules/InviteLinksUI/Sources/ItemListInviteLinkUsageLimitItem.swift new file mode 100644 index 0000000000..2d6a5cfe68 --- /dev/null +++ b/submodules/InviteLinksUI/Sources/ItemListInviteLinkUsageLimitItem.swift @@ -0,0 +1,415 @@ +import Foundation +import UIKit +import Display +import AsyncDisplayKit +import SwiftSignalKit +import TelegramCore +import SyncCore +import TelegramUIPreferences +import TelegramPresentationData +import LegacyComponents +import ItemListUI +import PresentationDataUtils + +enum InviteLinkUsageLimit: Equatable { + case low + case medium + case high + case unlimited + case custom(Int32) + + init(position: Int) { + switch position { + case 0: + self = .low + case 1: + self = .medium + case 2: + self = .high + default: + self = .unlimited + } + } + + init(value: Int32?) { + if let value = value { + if value == 1 { + self = .low + } else if value == 10 { + self = .medium + } else if value == 100 { + self = .high + } else { + self = .custom(value) + } + } else { + self = .unlimited + } + } + + var value: Int32? { + switch self { + case .low: + return 1 + case .medium: + return 10 + case .high: + return 100 + case .unlimited: + return nil + case let .custom(value): + return value + } + } + + var position: Int { + switch self { + case .low: + return 0 + case .medium: + return 1 + case .high: + return 2 + case .unlimited: + return 3 + case let .custom(value): + if value < 10 { + return 1 + } else if value < 100 { + return 2 + } else { + return 3 + } + } + } +} + +final class ItemListInviteLinkUsageLimitItem: ListViewItem, ItemListItem { + let theme: PresentationTheme + let strings: PresentationStrings + let value: InviteLinkUsageLimit + let enabled: Bool + let sectionId: ItemListSectionId + let updated: (InviteLinkUsageLimit) -> Void + + init(theme: PresentationTheme, strings: PresentationStrings, value: InviteLinkUsageLimit, enabled: Bool, sectionId: ItemListSectionId, updated: @escaping (InviteLinkUsageLimit) -> Void) { + self.theme = theme + self.strings = strings + self.value = value + self.enabled = enabled + self.sectionId = sectionId + self.updated = updated + } + + func nodeConfiguredForParams(async: @escaping (@escaping () -> Void) -> Void, params: ListViewItemLayoutParams, synchronousLoads: Bool, previousItem: ListViewItem?, nextItem: ListViewItem?, completion: @escaping (ListViewItemNode, @escaping () -> (Signal?, (ListViewItemApply) -> Void)) -> Void) { + async { + let node = ItemListInviteLinkUsageLimitItemNode() + let (layout, apply) = node.asyncLayout()(self, params, itemListNeighbors(item: self, topItem: previousItem as? ItemListItem, bottomItem: nextItem as? ItemListItem)) + + node.contentSize = layout.contentSize + node.insets = layout.insets + + Queue.mainQueue().async { + completion(node, { + return (nil, { _ in apply() }) + }) + } + } + } + + func updateNode(async: @escaping (@escaping () -> Void) -> Void, node: @escaping () -> ListViewItemNode, params: ListViewItemLayoutParams, previousItem: ListViewItem?, nextItem: ListViewItem?, animation: ListViewItemUpdateAnimation, completion: @escaping (ListViewItemNodeLayout, @escaping (ListViewItemApply) -> Void) -> Void) { + Queue.mainQueue().async { + if let nodeValue = node() as? ItemListInviteLinkUsageLimitItemNode { + let makeLayout = nodeValue.asyncLayout() + + async { + let (layout, apply) = makeLayout(self, params, itemListNeighbors(item: self, topItem: previousItem as? ItemListItem, bottomItem: nextItem as? ItemListItem)) + Queue.mainQueue().async { + completion(layout, { _ in + apply() + }) + } + } + } + } + } +} + +private func generateKnobImage() -> UIImage? { + return generateImage(CGSize(width: 40.0, height: 40.0), rotatedContext: { size, context in + context.clear(CGRect(origin: CGPoint(), size: size)) + context.setShadow(offset: CGSize(width: 0.0, height: -2.0), blur: 3.5, color: UIColor(white: 0.0, alpha: 0.35).cgColor) + context.setFillColor(UIColor.white.cgColor) + context.fillEllipse(in: CGRect(origin: CGPoint(x: 6.0, y: 6.0), size: CGSize(width: 28.0, height: 28.0))) + }) +} + +private final class ItemListInviteLinkUsageLimitItemNode: ListViewItemNode { + private let backgroundNode: ASDisplayNode + private let topStripeNode: ASDisplayNode + private let bottomStripeNode: ASDisplayNode + private let maskNode: ASImageNode + + private let lowTextNode: TextNode + private let mediumTextNode: TextNode + private let highTextNode: TextNode + private let unlimitedTextNode: TextNode + private let customTextNode: TextNode + private var sliderView: TGPhotoEditorSliderView? + + private var item: ItemListInviteLinkUsageLimitItem? + private var layoutParams: ListViewItemLayoutParams? + + init() { + self.backgroundNode = ASDisplayNode() + self.backgroundNode.isLayerBacked = true + + self.topStripeNode = ASDisplayNode() + self.topStripeNode.isLayerBacked = true + + self.bottomStripeNode = ASDisplayNode() + self.bottomStripeNode.isLayerBacked = true + + self.maskNode = ASImageNode() + + self.lowTextNode = TextNode() + self.lowTextNode.isUserInteractionEnabled = false + self.lowTextNode.displaysAsynchronously = false + + self.mediumTextNode = TextNode() + self.mediumTextNode.isUserInteractionEnabled = false + self.mediumTextNode.displaysAsynchronously = false + + self.highTextNode = TextNode() + self.highTextNode.isUserInteractionEnabled = false + self.highTextNode.displaysAsynchronously = false + + self.unlimitedTextNode = TextNode() + self.unlimitedTextNode.isUserInteractionEnabled = false + self.unlimitedTextNode.displaysAsynchronously = false + + self.customTextNode = TextNode() + self.customTextNode.isUserInteractionEnabled = false + self.customTextNode.displaysAsynchronously = false + + super.init(layerBacked: false, dynamicBounce: false) + + self.addSubnode(self.lowTextNode) + self.addSubnode(self.mediumTextNode) + self.addSubnode(self.highTextNode) + self.addSubnode(self.unlimitedTextNode) + self.addSubnode(self.customTextNode) + } + + func updateSliderView() { + if let sliderView = self.sliderView, let item = self.item { + if case .custom = item.value { + sliderView.maximumValue = 3.0 + 1 + sliderView.positionsCount = 4 + 1 + } else { + sliderView.maximumValue = 3.0 + sliderView.positionsCount = 4 + } + sliderView.value = CGFloat(item.value.position) + + sliderView.isUserInteractionEnabled = item.enabled + sliderView.alpha = item.enabled ? 1.0 : 0.4 + sliderView.layer.allowsGroupOpacity = !item.enabled + } + } + + override func didLoad() { + super.didLoad() + + let sliderView = TGPhotoEditorSliderView() + sliderView.enablePanHandling = true + sliderView.trackCornerRadius = 1.0 + sliderView.lineSize = 2.0 + sliderView.dotSize = 5.0 + sliderView.minimumValue = 0.0 + sliderView.startValue = 0.0 + sliderView.disablesInteractiveTransitionGestureRecognizer = true + if let item = self.item, case .custom = item.value { + sliderView.maximumValue = 3.0 + 1 + sliderView.positionsCount = 4 + 1 + } else { + sliderView.maximumValue = 3.0 + sliderView.positionsCount = 4 + } + sliderView.useLinesForPositions = true + if let item = self.item, let params = self.layoutParams { + sliderView.value = CGFloat(item.value.position) + sliderView.backgroundColor = item.theme.list.itemBlocksBackgroundColor + sliderView.backColor = item.theme.list.disclosureArrowColor + sliderView.startColor = item.theme.list.disclosureArrowColor + sliderView.trackColor = item.theme.list.itemAccentColor + sliderView.knobImage = generateKnobImage() + + sliderView.frame = CGRect(origin: CGPoint(x: params.leftInset + 15.0, y: 37.0), size: CGSize(width: params.width - params.leftInset - params.rightInset - 15.0 * 2.0, height: 44.0)) + sliderView.hitTestEdgeInsets = UIEdgeInsets(top: -sliderView.frame.minX, left: 0.0, bottom: 0.0, right: -sliderView.frame.minX) + } + self.view.addSubview(sliderView) + sliderView.addTarget(self, action: #selector(self.sliderValueChanged), for: .valueChanged) + self.sliderView = sliderView + + self.updateSliderView() + } + + func asyncLayout() -> (_ item: ItemListInviteLinkUsageLimitItem, _ params: ListViewItemLayoutParams, _ neighbors: ItemListNeighbors) -> (ListViewItemNodeLayout, () -> Void) { + let currentItem = self.item + let makeLowTextLayout = TextNode.asyncLayout(self.lowTextNode) + let makeMediumTextLayout = TextNode.asyncLayout(self.mediumTextNode) + let makeHighTextLayout = TextNode.asyncLayout(self.highTextNode) + let makeUnlimitedTextLayout = TextNode.asyncLayout(self.unlimitedTextNode) + let makeCustomTextLayout = TextNode.asyncLayout(self.customTextNode) + + return { item, params, neighbors in + var themeUpdated = false + if currentItem?.theme !== item.theme { + themeUpdated = true + } + + let contentSize: CGSize + let insets: UIEdgeInsets + let separatorHeight = UIScreenPixel + + let (lowTextLayout, lowTextApply) = makeLowTextLayout(TextNodeLayoutArguments(attributedString: NSAttributedString(string: "1", font: Font.regular(13.0), textColor: item.theme.list.itemSecondaryTextColor), backgroundColor: nil, maximumNumberOfLines: 1, truncationType: .end, constrainedSize: CGSize(width: params.width, height: CGFloat.greatestFiniteMagnitude), alignment: .center, lineSpacing: 0.0, cutout: nil, insets: UIEdgeInsets())) + + let (mediumTextLayout, mediumTextApply) = makeMediumTextLayout(TextNodeLayoutArguments(attributedString: NSAttributedString(string: "10", font: Font.regular(13.0), textColor: item.theme.list.itemSecondaryTextColor), backgroundColor: nil, maximumNumberOfLines: 1, truncationType: .end, constrainedSize: CGSize(width: params.width, height: CGFloat.greatestFiniteMagnitude), alignment: .center, lineSpacing: 0.0, cutout: nil, insets: UIEdgeInsets())) + + let (highTextLayout, highTextApply) = makeHighTextLayout(TextNodeLayoutArguments(attributedString: NSAttributedString(string: "100", font: Font.regular(13.0), textColor: item.theme.list.itemSecondaryTextColor), backgroundColor: nil, maximumNumberOfLines: 1, truncationType: .end, constrainedSize: CGSize(width: params.width, height: CGFloat.greatestFiniteMagnitude), alignment: .center, lineSpacing: 0.0, cutout: nil, insets: UIEdgeInsets())) + + let (unlimitedTextLayout, unlimitedTextApply) = makeUnlimitedTextLayout(TextNodeLayoutArguments(attributedString: NSAttributedString(string: item.strings.InviteLink_Create_UsersLimitNoLimit, font: Font.regular(13.0), textColor: item.theme.list.itemSecondaryTextColor), backgroundColor: nil, maximumNumberOfLines: 1, truncationType: .end, constrainedSize: CGSize(width: params.width, height: CGFloat.greatestFiniteMagnitude), alignment: .center, lineSpacing: 0.0, cutout: nil, insets: UIEdgeInsets())) + + let customTextString: String + if case let .custom(value) = item.value { + customTextString = "\(value)" + } else { + customTextString = "" + } + + let (customTextLayout, customTextApply) = makeCustomTextLayout(TextNodeLayoutArguments(attributedString: NSAttributedString(string: customTextString, font: Font.regular(13.0), textColor: item.theme.list.itemSecondaryTextColor), backgroundColor: nil, maximumNumberOfLines: 1, truncationType: .end, constrainedSize: CGSize(width: params.width, height: CGFloat.greatestFiniteMagnitude), alignment: .center, lineSpacing: 0.0, cutout: nil, insets: UIEdgeInsets())) + + contentSize = CGSize(width: params.width, height: 88.0) + insets = itemListNeighborsGroupedInsets(neighbors) + + let layout = ListViewItemNodeLayout(contentSize: contentSize, insets: insets) + let layoutSize = layout.size + + return (layout, { [weak self] in + if let strongSelf = self { + strongSelf.item = item + strongSelf.layoutParams = params + + strongSelf.backgroundNode.backgroundColor = item.theme.list.itemBlocksBackgroundColor + strongSelf.topStripeNode.backgroundColor = item.theme.list.itemBlocksSeparatorColor + strongSelf.bottomStripeNode.backgroundColor = item.theme.list.itemBlocksSeparatorColor + + if strongSelf.backgroundNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.backgroundNode, at: 0) + } + if strongSelf.topStripeNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.topStripeNode, at: 1) + } + if strongSelf.bottomStripeNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.bottomStripeNode, at: 2) + } + if strongSelf.maskNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.maskNode, at: 3) + } + + let hasCorners = itemListHasRoundedBlockLayout(params) + var hasTopCorners = false + var hasBottomCorners = false + switch neighbors.top { + case .sameSection(false): + strongSelf.topStripeNode.isHidden = true + default: + hasTopCorners = true + strongSelf.topStripeNode.isHidden = hasCorners + } + let bottomStripeInset: CGFloat + let bottomStripeOffset: CGFloat + switch neighbors.bottom { + case .sameSection(false): + bottomStripeInset = 0.0 //params.leftInset + 16.0 + bottomStripeOffset = -separatorHeight + default: + bottomStripeInset = 0.0 + bottomStripeOffset = 0.0 + hasBottomCorners = true + strongSelf.bottomStripeNode.isHidden = hasCorners + } + + strongSelf.maskNode.image = hasCorners ? PresentationResourcesItemList.cornersImage(item.theme, top: hasTopCorners, bottom: hasBottomCorners) : nil + + strongSelf.backgroundNode.frame = CGRect(origin: CGPoint(x: 0.0, y: -min(insets.top, separatorHeight)), size: CGSize(width: params.width, height: contentSize.height + min(insets.top, separatorHeight) + min(insets.bottom, separatorHeight))) + strongSelf.maskNode.frame = strongSelf.backgroundNode.frame.insetBy(dx: params.leftInset, dy: 0.0) + strongSelf.topStripeNode.frame = CGRect(origin: CGPoint(x: 0.0, y: -min(insets.top, separatorHeight)), size: CGSize(width: layoutSize.width, height: separatorHeight)) + strongSelf.bottomStripeNode.frame = CGRect(origin: CGPoint(x: bottomStripeInset, y: contentSize.height + bottomStripeOffset), size: CGSize(width: layoutSize.width - bottomStripeInset, height: separatorHeight)) + + let _ = lowTextApply() + let _ = mediumTextApply() + let _ = highTextApply() + let _ = unlimitedTextApply() + let _ = customTextApply() + + var textNodes: [(TextNode, CGSize)] = [(strongSelf.lowTextNode, lowTextLayout.size), + (strongSelf.mediumTextNode, mediumTextLayout.size), + (strongSelf.highTextNode, highTextLayout.size), + (strongSelf.unlimitedTextNode, unlimitedTextLayout.size)] + if case .custom = item.value { + textNodes.insert((strongSelf.customTextNode, customTextLayout.size), at: item.value.position) + } + + let delta = (params.width - params.leftInset - params.rightInset - 18.0 * 2.0) / CGFloat(textNodes.count - 1) + for i in 0 ..< textNodes.count { + let (textNode, textSize) = textNodes[i] + + var position = params.leftInset + 18.0 + delta * CGFloat(i) + if i == textNodes.count - 1 { + position -= textSize.width + } else if i > 0 { + position -= textSize.width / 2.0 + } + + textNode.frame = CGRect(origin: CGPoint(x: position, y: 15.0), size: textSize) + } + + if let sliderView = strongSelf.sliderView { + if themeUpdated { + sliderView.backgroundColor = item.theme.list.itemBlocksBackgroundColor + sliderView.backColor = item.theme.list.disclosureArrowColor + sliderView.trackColor = item.theme.list.itemAccentColor + sliderView.knobImage = generateKnobImage() + } + + sliderView.frame = CGRect(origin: CGPoint(x: params.leftInset + 15.0, y: 37.0), size: CGSize(width: params.width - params.leftInset - params.rightInset - 15.0 * 2.0, height: 44.0)) + sliderView.hitTestEdgeInsets = UIEdgeInsets(top: -sliderView.frame.minX, left: 0.0, bottom: 0.0, right: -sliderView.frame.minX) + + strongSelf.updateSliderView() + } + } + }) + } + } + + override func animateInsertion(_ currentTimestamp: Double, duration: Double, short: Bool) { + self.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.4) + } + + override func animateRemoved(_ currentTimestamp: Double, duration: Double) { + self.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.15, removeOnCompletion: false) + } + + @objc func sliderValueChanged() { + guard let sliderView = self.sliderView else { + return + } + + let position = Int(sliderView.value) + let value = InviteLinkUsageLimit(position: position) + self.item?.updated(value) + } +} + diff --git a/submodules/InviteLinksUI/Sources/ItemListPermanentInviteLinkItem.swift b/submodules/InviteLinksUI/Sources/ItemListPermanentInviteLinkItem.swift new file mode 100644 index 0000000000..a96a635119 --- /dev/null +++ b/submodules/InviteLinksUI/Sources/ItemListPermanentInviteLinkItem.swift @@ -0,0 +1,451 @@ +import Foundation +import UIKit +import Display +import AsyncDisplayKit +import SwiftSignalKit +import Postbox +import SyncCore +import AccountContext +import TelegramPresentationData +import ItemListUI +import SolidRoundedButtonNode +import AnimatedAvatarSetNode + +private func actionButtonImage(color: UIColor) -> UIImage? { + return generateImage(CGSize(width: 24.0, height: 24.0), contextGenerator: { size, context in + context.clear(CGRect(origin: CGPoint(), size: size)) + + context.setFillColor(color.cgColor) + context.fillEllipse(in: CGRect(origin: CGPoint(), size: size)) + + context.setBlendMode(.clear) + context.fillEllipse(in: CGRect(origin: CGPoint(x: 4.0, y: 10.0), size: CGSize(width: 4.0, height: 4.0))) + context.fillEllipse(in: CGRect(origin: CGPoint(x: 10.0, y: 10.0), size: CGSize(width: 4.0, height: 4.0))) + context.fillEllipse(in: CGRect(origin: CGPoint(x: 16.0, y: 10.0), size: CGSize(width: 4.0, height: 4.0))) + }) +} + +public class ItemListPermanentInviteLinkItem: ListViewItem, ItemListItem { + let context: AccountContext + let presentationData: ItemListPresentationData + let invite: ExportedInvitation? + let peers: [Peer] + let buttonColor: UIColor? + public let sectionId: ItemListSectionId + let style: ItemListStyle + let shareAction: (() -> Void)? + let contextAction: ((ASDisplayNode) -> Void)? + let viewAction: (() -> Void)? + public let tag: ItemListItemTag? + + public init( + context: AccountContext, + presentationData: ItemListPresentationData, + invite: ExportedInvitation?, + peers: [Peer], + buttonColor: UIColor?, + sectionId: ItemListSectionId, + style: ItemListStyle, + shareAction: (() -> Void)?, + contextAction: ((ASDisplayNode) -> Void)?, + viewAction: (() -> Void)?, + tag: ItemListItemTag? = nil + ) { + self.context = context + self.presentationData = presentationData + self.invite = invite + self.peers = peers + self.buttonColor = buttonColor + self.sectionId = sectionId + self.style = style + self.shareAction = shareAction + self.contextAction = contextAction + self.viewAction = viewAction + self.tag = tag + } + + public func nodeConfiguredForParams(async: @escaping (@escaping () -> Void) -> Void, params: ListViewItemLayoutParams, synchronousLoads: Bool, previousItem: ListViewItem?, nextItem: ListViewItem?, completion: @escaping (ListViewItemNode, @escaping () -> (Signal?, (ListViewItemApply) -> Void)) -> Void) { + async { + let node = ItemListPermanentInviteLinkItemNode() + let (layout, apply) = node.asyncLayout()(self, params, itemListNeighbors(item: self, topItem: previousItem as? ItemListItem, bottomItem: nextItem as? ItemListItem)) + + node.contentSize = layout.contentSize + node.insets = layout.insets + + Queue.mainQueue().async { + completion(node, { + return (nil, { _ in apply() }) + }) + } + } + } + + public func updateNode(async: @escaping (@escaping () -> Void) -> Void, node: @escaping () -> ListViewItemNode, params: ListViewItemLayoutParams, previousItem: ListViewItem?, nextItem: ListViewItem?, animation: ListViewItemUpdateAnimation, completion: @escaping (ListViewItemNodeLayout, @escaping (ListViewItemApply) -> Void) -> Void) { + Queue.mainQueue().async { + if let nodeValue = node() as? ItemListPermanentInviteLinkItemNode { + let makeLayout = nodeValue.asyncLayout() + + async { + let (layout, apply) = makeLayout(self, params, itemListNeighbors(item: self, topItem: previousItem as? ItemListItem, bottomItem: nextItem as? ItemListItem)) + Queue.mainQueue().async { + completion(layout, { _ in + apply() + }) + } + } + } + } + } + + public var selectable: Bool = false +} + +public class ItemListPermanentInviteLinkItemNode: ListViewItemNode, ItemListItemNode { + private let backgroundNode: ASDisplayNode + private let topStripeNode: ASDisplayNode + private let bottomStripeNode: ASDisplayNode + private let maskNode: ASImageNode + + private let fieldNode: ASImageNode + private let addressNode: TextNode + private let extractedContainerNode: ContextExtractedContentContainingNode + private let containerNode: ContextControllerSourceNode + private let addressButtonNode: HighlightTrackingButtonNode + private let addressButtonIconNode: ASImageNode + private var shareButtonNode: SolidRoundedButtonNode? + + private let avatarsButtonNode: HighlightTrackingButtonNode + private let avatarsContext: AnimatedAvatarSetContext + private var avatarsContent: AnimatedAvatarSetContext.Content? + private let avatarsNode: AnimatedAvatarSetNode + private let invitedPeersNode: TextNode + + private let activateArea: AccessibilityAreaNode + + private var item: ItemListPermanentInviteLinkItem? + + override public var canBeSelected: Bool { + return false + } + + public var tag: ItemListItemTag? { + return self.item?.tag + } + + public init() { + self.backgroundNode = ASDisplayNode() + self.backgroundNode.isLayerBacked = true + self.backgroundNode.backgroundColor = .white + + self.maskNode = ASImageNode() + + self.topStripeNode = ASDisplayNode() + self.topStripeNode.isLayerBacked = true + + self.bottomStripeNode = ASDisplayNode() + self.bottomStripeNode.isLayerBacked = true + + self.fieldNode = ASImageNode() + self.fieldNode.displaysAsynchronously = false + self.fieldNode.displayWithoutProcessing = true + + self.addressNode = TextNode() + self.addressNode.isUserInteractionEnabled = false + + self.addressButtonNode = HighlightTrackingButtonNode() + self.extractedContainerNode = ContextExtractedContentContainingNode() + self.containerNode = ContextControllerSourceNode() + self.containerNode.isGestureEnabled = false + self.addressButtonIconNode = ASImageNode() + self.addressButtonIconNode.displaysAsynchronously = false + self.addressButtonIconNode.displayWithoutProcessing = true + + self.avatarsButtonNode = HighlightTrackingButtonNode() + self.avatarsContext = AnimatedAvatarSetContext() + self.avatarsNode = AnimatedAvatarSetNode() + self.invitedPeersNode = TextNode() + + self.activateArea = AccessibilityAreaNode() + + super.init(layerBacked: false, dynamicBounce: false) + + self.addSubnode(self.fieldNode) + self.addSubnode(self.addressNode) + self.addSubnode(self.avatarsNode) + self.addSubnode(self.invitedPeersNode) + self.addSubnode(self.avatarsButtonNode) + + self.containerNode.addSubnode(self.extractedContainerNode) + self.extractedContainerNode.contentNode.addSubnode(self.addressButtonIconNode) + self.containerNode.targetNodeForActivationProgress = self.extractedContainerNode.contentNode + self.addressButtonNode.addSubnode(self.containerNode) + self.addSubnode(self.addressButtonNode) + + self.addSubnode(self.activateArea) + + self.addressButtonNode.addTarget(self, action: #selector(self.addressButtonPressed), forControlEvents: .touchUpInside) + self.addressButtonNode.highligthedChanged = { [weak self] highlighted in + if let strongSelf = self { + if highlighted { + strongSelf.addressButtonIconNode.layer.removeAnimation(forKey: "opacity") + strongSelf.addressButtonIconNode.alpha = 0.4 + } else { + strongSelf.addressButtonIconNode.alpha = 1.0 + strongSelf.addressButtonIconNode.layer.animateAlpha(from: 0.4, to: 1.0, duration: 0.2) + } + } + } + self.shareButtonNode?.pressed = { [weak self] in + if let strongSelf = self, let item = strongSelf.item { + item.shareAction?() + } + } + self.avatarsButtonNode.highligthedChanged = { [weak self] highlighted in + if let strongSelf = self { + if highlighted { + strongSelf.avatarsNode.layer.removeAnimation(forKey: "opacity") + strongSelf.invitedPeersNode.layer.removeAnimation(forKey: "opacity") + strongSelf.avatarsNode.alpha = 0.4 + strongSelf.invitedPeersNode.alpha = 0.4 + } else { + strongSelf.avatarsNode.alpha = 1.0 + strongSelf.invitedPeersNode.alpha = 1.0 + strongSelf.avatarsNode.layer.animateAlpha(from: 0.4, to: 1.0, duration: 0.2) + strongSelf.invitedPeersNode.layer.animateAlpha(from: 0.4, to: 1.0, duration: 0.2) + } + } + } + self.avatarsButtonNode.addTarget(self, action: #selector(self.avatarsButtonPressed), forControlEvents: .touchUpInside) + } + + @objc private func addressButtonPressed() { + if let item = self.item { + item.contextAction?(self.extractedContainerNode) + } + } + + @objc private func avatarsButtonPressed() { + if let item = self.item { + item.viewAction?() + } + } + + public func asyncLayout() -> (_ item: ItemListPermanentInviteLinkItem, _ params: ListViewItemLayoutParams, _ insets: ItemListNeighbors) -> (ListViewItemNodeLayout, () -> Void) { + let makeAddressLayout = TextNode.asyncLayout(self.addressNode) + let makeInvitedPeersLayout = TextNode.asyncLayout(self.invitedPeersNode) + + let currentItem = self.item + let avatarsContext = self.avatarsContext + + return { item, params, neighbors in + var updatedTheme: PresentationTheme? + if currentItem?.presentationData.theme !== item.presentationData.theme { + updatedTheme = item.presentationData.theme + } + + let contentSize: CGSize + let insets: UIEdgeInsets + let separatorHeight = UIScreenPixel + let itemBackgroundColor: UIColor + let itemSeparatorColor: UIColor + + let leftInset = 16.0 + params.leftInset + let rightInset = 16.0 + params.rightInset + + let titleColor: UIColor + titleColor = item.presentationData.theme.list.itemPrimaryTextColor + + let titleFont = Font.regular(item.presentationData.fontSize.itemListBaseFontSize) + + let (addressLayout, addressApply) = makeAddressLayout(TextNodeLayoutArguments(attributedString: NSAttributedString(string: item.invite.flatMap({ $0.link.replacingOccurrences(of: "https://", with: "") }) ?? "", font: titleFont, textColor: titleColor), backgroundColor: nil, maximumNumberOfLines: 1, truncationType: .end, constrainedSize: CGSize(width: params.width - params.rightInset - 20.0 - leftInset - rightInset, height: CGFloat.greatestFiniteMagnitude), alignment: .natural, cutout: nil, insets: UIEdgeInsets())) + + let subtitle: String + let subtitleColor: UIColor + if let count = item.invite?.count { + if count > 0 { + subtitle = item.presentationData.strings.InviteLink_PeopleJoined(count) + subtitleColor = item.presentationData.theme.list.itemAccentColor + } else { + subtitle = item.presentationData.strings.InviteLink_PeopleJoinedNone + subtitleColor = item.presentationData.theme.list.itemSecondaryTextColor + } + } else { + subtitle = item.presentationData.strings.InviteLink_PeopleJoinedNone + subtitleColor = item.presentationData.theme.list.itemSecondaryTextColor + } + + let (invitedPeersLayout, invitedPeersApply) = makeInvitedPeersLayout(TextNodeLayoutArguments(attributedString: NSAttributedString(string: subtitle, font: titleFont, textColor: subtitleColor), backgroundColor: nil, maximumNumberOfLines: 1, truncationType: .end, constrainedSize: CGSize(width: params.width - params.rightInset - 20.0 - leftInset - rightInset, height: CGFloat.greatestFiniteMagnitude), alignment: .natural, cutout: nil, insets: UIEdgeInsets())) + + let avatarsContent = avatarsContext.update(peers: item.peers, animated: false) + + let verticalInset: CGFloat = 16.0 + let fieldHeight: CGFloat = 52.0 + let fieldSpacing: CGFloat = 16.0 + let buttonHeight: CGFloat = 50.0 + + var height = verticalInset * 2.0 + fieldHeight + fieldSpacing + buttonHeight + 54.0 + + switch item.style { + case .plain: + height -= 57.0 + itemBackgroundColor = item.presentationData.theme.list.plainBackgroundColor + itemSeparatorColor = .clear + insets = UIEdgeInsets() + case .blocks: + itemBackgroundColor = item.presentationData.theme.list.itemBlocksBackgroundColor + itemSeparatorColor = item.presentationData.theme.list.itemBlocksSeparatorColor + insets = itemListNeighborsGroupedInsets(neighbors) + } + contentSize = CGSize(width: params.width, height: height) + + let layout = ListViewItemNodeLayout(contentSize: contentSize, insets: insets) + + return (ListViewItemNodeLayout(contentSize: contentSize, insets: insets), { [weak self] in + if let strongSelf = self { + strongSelf.item = item + strongSelf.avatarsContent = avatarsContent + + strongSelf.activateArea.frame = CGRect(origin: CGPoint(x: params.leftInset, y: 0.0), size: CGSize(width: params.width - params.leftInset - params.rightInset, height: layout.contentSize.height)) +// strongSelf.activateArea.accessibilityLabel = item.title +// strongSelf.activateArea.accessibilityValue = item.label + strongSelf.activateArea.accessibilityTraits = [] + + if let _ = updatedTheme { + strongSelf.topStripeNode.backgroundColor = itemSeparatorColor + strongSelf.bottomStripeNode.backgroundColor = itemSeparatorColor + strongSelf.backgroundNode.backgroundColor = itemBackgroundColor + strongSelf.fieldNode.image = generateStretchableFilledCircleImage(diameter: 18.0, color: item.presentationData.theme.list.freePlainInputField.backgroundColor) + strongSelf.addressButtonIconNode.image = actionButtonImage(color: item.presentationData.theme.list.freePlainInputField.controlColor) + } + + let _ = addressApply() + let _ = invitedPeersApply() + + switch item.style { + case .plain: + if strongSelf.backgroundNode.supernode != nil { + strongSelf.backgroundNode.removeFromSupernode() + } + if strongSelf.topStripeNode.supernode != nil { + strongSelf.topStripeNode.removeFromSupernode() + } + if strongSelf.bottomStripeNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.bottomStripeNode, at: 0) + } + if strongSelf.maskNode.supernode != nil { + strongSelf.maskNode.removeFromSupernode() + } + strongSelf.bottomStripeNode.frame = CGRect(origin: CGPoint(x: leftInset, y: contentSize.height - separatorHeight), size: CGSize(width: params.width - leftInset, height: separatorHeight)) + case .blocks: + if strongSelf.backgroundNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.backgroundNode, at: 0) + } + if strongSelf.topStripeNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.topStripeNode, at: 1) + } + if strongSelf.bottomStripeNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.bottomStripeNode, at: 2) + } + if strongSelf.maskNode.supernode == nil { + strongSelf.insertSubnode(strongSelf.maskNode, at: 3) + } + + let hasCorners = itemListHasRoundedBlockLayout(params) + var hasTopCorners = false + var hasBottomCorners = false + switch neighbors.top { + case .sameSection(false): + strongSelf.topStripeNode.isHidden = true + default: + hasTopCorners = true + strongSelf.topStripeNode.isHidden = hasCorners + } + let bottomStripeInset: CGFloat + switch neighbors.bottom { + case .sameSection(false): + bottomStripeInset = leftInset + default: + bottomStripeInset = 0.0 + hasBottomCorners = true + strongSelf.bottomStripeNode.isHidden = hasCorners + } + + strongSelf.maskNode.image = hasCorners ? PresentationResourcesItemList.cornersImage(item.presentationData.theme, top: hasTopCorners, bottom: hasBottomCorners) : nil + + strongSelf.backgroundNode.frame = CGRect(origin: CGPoint(x: 0.0, y: -min(insets.top, separatorHeight)), size: CGSize(width: params.width, height: contentSize.height + min(insets.top, separatorHeight) + min(insets.bottom, separatorHeight))) + strongSelf.maskNode.frame = strongSelf.backgroundNode.frame.insetBy(dx: params.leftInset, dy: 0.0) + strongSelf.topStripeNode.frame = CGRect(origin: CGPoint(x: 0.0, y: -min(insets.top, separatorHeight)), size: CGSize(width: params.width, height: separatorHeight)) + strongSelf.bottomStripeNode.frame = CGRect(origin: CGPoint(x: bottomStripeInset, y: contentSize.height - separatorHeight), size: CGSize(width: params.width - bottomStripeInset, height: separatorHeight)) + } + + let fieldFrame = CGRect(origin: CGPoint(x: leftInset, y: verticalInset), size: CGSize(width: params.width - leftInset - rightInset, height: fieldHeight)) + strongSelf.fieldNode.frame = fieldFrame + + strongSelf.addressNode.frame = CGRect(origin: CGPoint(x: fieldFrame.minX + floorToScreenPixels((fieldFrame.width - addressLayout.size.width) / 2.0), y: fieldFrame.minY + floorToScreenPixels((fieldFrame.height - addressLayout.size.height) / 2.0) + 1.0), size: addressLayout.size) + + strongSelf.addressButtonNode.frame = CGRect(origin: CGPoint(x: params.width - rightInset - 38.0, y: verticalInset + 14.0), size: CGSize(width: 24.0, height: 24.0)) + strongSelf.extractedContainerNode.frame = strongSelf.addressButtonNode.bounds + strongSelf.extractedContainerNode.contentRect = strongSelf.addressButtonNode.bounds + strongSelf.addressButtonIconNode.frame = strongSelf.addressButtonNode.bounds + + let shareButtonNode: SolidRoundedButtonNode + if let currentShareButtonNode = strongSelf.shareButtonNode { + shareButtonNode = currentShareButtonNode + } else { + let buttonTheme: SolidRoundedButtonTheme + if let buttonColor = item.buttonColor { + buttonTheme = SolidRoundedButtonTheme(backgroundColor: buttonColor, foregroundColor: item.presentationData.theme.list.itemCheckColors.foregroundColor) + } else { + buttonTheme = SolidRoundedButtonTheme(theme: item.presentationData.theme) + } + shareButtonNode = SolidRoundedButtonNode(theme: buttonTheme, height: 50.0, cornerRadius: 10.0) + shareButtonNode.title = item.presentationData.strings.InviteLink_Share + shareButtonNode.pressed = { + item.shareAction?() + } + strongSelf.addSubnode(shareButtonNode) + strongSelf.shareButtonNode = shareButtonNode + } + + let buttonWidth = contentSize.width - leftInset - rightInset + let _ = shareButtonNode.updateLayout(width: buttonWidth, transition: .immediate) + shareButtonNode.frame = CGRect(x: leftInset, y: verticalInset + fieldHeight + fieldSpacing, width: buttonWidth, height: buttonHeight) + + var totalWidth = invitedPeersLayout.size.width + var leftOrigin: CGFloat = floorToScreenPixels((params.width - invitedPeersLayout.size.width) / 2.0) + let avatarSpacing: CGFloat = 21.0 + if let avatarsContent = strongSelf.avatarsContent { + let avatarsSize = strongSelf.avatarsNode.update(context: item.context, content: avatarsContent, itemSize: CGSize(width: 32.0, height: 32.0), animated: true, synchronousLoad: true) + + if !avatarsSize.width.isZero { + totalWidth += avatarsSize.width + avatarSpacing + } + + let avatarsNodeFrame = CGRect(origin: CGPoint(x: floorToScreenPixels((params.width - totalWidth) / 2.0), y: fieldFrame.maxY + 87.0), size: avatarsSize) + strongSelf.avatarsNode.frame = avatarsNodeFrame + if !avatarsSize.width.isZero { + leftOrigin = avatarsNodeFrame.maxX + avatarSpacing + } + } + + strongSelf.invitedPeersNode.frame = CGRect(origin: CGPoint(x: leftOrigin, y: fieldFrame.maxY + 92.0), size: invitedPeersLayout.size) + + strongSelf.avatarsButtonNode.frame = CGRect(x: floorToScreenPixels((params.width - totalWidth) / 2.0), y: fieldFrame.maxY + 87.0, width: totalWidth, height: 32.0) + strongSelf.avatarsButtonNode.isUserInteractionEnabled = !item.peers.isEmpty + } + }) + } + } + + override public func animateInsertion(_ currentTimestamp: Double, duration: Double, short: Bool) { + self.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.4) + } + + override public func animateAdded(_ currentTimestamp: Double, duration: Double) { + self.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.2) + } + + override public func animateRemoved(_ currentTimestamp: Double, duration: Double) { + self.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.15, removeOnCompletion: false) + } +} diff --git a/submodules/ItemListPeerActionItem/Sources/ItemListPeerActionItem.swift b/submodules/ItemListPeerActionItem/Sources/ItemListPeerActionItem.swift index 429045ced4..bed1fb9b22 100644 --- a/submodules/ItemListPeerActionItem/Sources/ItemListPeerActionItem.swift +++ b/submodules/ItemListPeerActionItem/Sources/ItemListPeerActionItem.swift @@ -22,17 +22,19 @@ public class ItemListPeerActionItem: ListViewItem, ItemListItem { let icon: UIImage? let title: String public let alwaysPlain: Bool + let hasSeparator: Bool let editing: Bool let height: ItemListPeerActionItemHeight let color: ItemListPeerActionItemColor public let sectionId: ItemListSectionId let action: (() -> Void)? - public init(presentationData: ItemListPresentationData, icon: UIImage?, title: String, alwaysPlain: Bool = false, sectionId: ItemListSectionId, height: ItemListPeerActionItemHeight = .peerList, color: ItemListPeerActionItemColor = .accent, editing: Bool, action: (() -> Void)?) { + public init(presentationData: ItemListPresentationData, icon: UIImage?, title: String, alwaysPlain: Bool = false, hasSeparator: Bool = true, sectionId: ItemListSectionId, height: ItemListPeerActionItemHeight = .peerList, color: ItemListPeerActionItemColor = .accent, editing: Bool, action: (() -> Void)?) { self.presentationData = presentationData self.icon = icon self.title = title self.alwaysPlain = alwaysPlain + self.hasSeparator = hasSeparator self.editing = editing self.height = height self.color = color @@ -257,6 +259,8 @@ class ItemListPeerActionItemNode: ListViewItemNode { strongSelf.bottomStripeNode.isHidden = hasCorners } + strongSelf.bottomStripeNode.isHidden = strongSelf.bottomStripeNode.isHidden || !item.hasSeparator + strongSelf.maskNode.image = hasCorners ? PresentationResourcesItemList.cornersImage(item.presentationData.theme, top: hasTopCorners, bottom: hasBottomCorners) : nil strongSelf.backgroundNode.frame = CGRect(origin: CGPoint(x: 0.0, y: -min(insets.top, separatorHeight)), size: CGSize(width: params.width, height: contentSize.height + min(insets.top, separatorHeight) + min(insets.bottom, separatorHeight))) diff --git a/submodules/ItemListUI/Sources/Items/ItemListSingleLineInputItem.swift b/submodules/ItemListUI/Sources/Items/ItemListSingleLineInputItem.swift index c45b9769f7..52e80d7863 100644 --- a/submodules/ItemListUI/Sources/Items/ItemListSingleLineInputItem.swift +++ b/submodules/ItemListUI/Sources/Items/ItemListSingleLineInputItem.swift @@ -29,6 +29,11 @@ public enum ItemListSingleLineInputClearType: Equatable { } } +public enum ItemListSingleLineInputAlignment { + case `default` + case right +} + public class ItemListSingleLineInputItem: ListViewItem, ItemListItem { let presentationData: ItemListPresentationData let title: NSAttributedString @@ -36,6 +41,7 @@ public class ItemListSingleLineInputItem: ListViewItem, ItemListItem { let placeholder: String let type: ItemListSingleLineInputItemType let returnKeyType: UIReturnKeyType + let alignment: ItemListSingleLineInputAlignment let spacing: CGFloat let clearType: ItemListSingleLineInputClearType let maxLength: Int @@ -49,13 +55,14 @@ public class ItemListSingleLineInputItem: ListViewItem, ItemListItem { let cleared: (() -> Void)? public let tag: ItemListItemTag? - public init(presentationData: ItemListPresentationData, title: NSAttributedString, text: String, placeholder: String, type: ItemListSingleLineInputItemType = .regular(capitalization: true, autocorrection: true), returnKeyType: UIReturnKeyType = .`default`, spacing: CGFloat = 0.0, clearType: ItemListSingleLineInputClearType = .none, maxLength: Int = 0, enabled: Bool = true, tag: ItemListItemTag? = nil, sectionId: ItemListSectionId, textUpdated: @escaping (String) -> Void, shouldUpdateText: @escaping (String) -> Bool = { _ in return true }, processPaste: ((String) -> String)? = nil, updatedFocus: ((Bool) -> Void)? = nil, action: @escaping () -> Void, cleared: (() -> Void)? = nil) { + public init(presentationData: ItemListPresentationData, title: NSAttributedString, text: String, placeholder: String, type: ItemListSingleLineInputItemType = .regular(capitalization: true, autocorrection: true), returnKeyType: UIReturnKeyType = .`default`, alignment: ItemListSingleLineInputAlignment = .default, spacing: CGFloat = 0.0, clearType: ItemListSingleLineInputClearType = .none, maxLength: Int = 0, enabled: Bool = true, tag: ItemListItemTag? = nil, sectionId: ItemListSectionId, textUpdated: @escaping (String) -> Void, shouldUpdateText: @escaping (String) -> Bool = { _ in return true }, processPaste: ((String) -> String)? = nil, updatedFocus: ((Bool) -> Void)? = nil, action: @escaping () -> Void, cleared: (() -> Void)? = nil) { self.presentationData = presentationData self.title = title self.text = text self.placeholder = placeholder self.type = type self.returnKeyType = returnKeyType + self.alignment = alignment self.spacing = spacing self.clearType = clearType self.maxLength = maxLength @@ -325,6 +332,13 @@ public class ItemListSingleLineInputItemNode: ListViewItemNode, UITextFieldDeleg strongSelf.textNode.frame = CGRect(origin: CGPoint(x: leftInset + titleLayout.size.width + item.spacing, y: 1.0), size: CGSize(width: max(1.0, params.width - (leftInset + rightInset + titleLayout.size.width + item.spacing)), height: layout.contentSize.height - 2.0)) + switch item.alignment { + case .default: + strongSelf.textNode.textField.textAlignment = .natural + case .right: + strongSelf.textNode.textField.textAlignment = .right + } + if let image = updatedClearIcon { strongSelf.clearIconNode.image = image } diff --git a/submodules/LegacyComponents/PublicHeaders/LegacyComponents/TGClipboardGalleryMixin.h b/submodules/LegacyComponents/PublicHeaders/LegacyComponents/TGClipboardGalleryMixin.h index 04c7c032e0..56f8622698 100644 --- a/submodules/LegacyComponents/PublicHeaders/LegacyComponents/TGClipboardGalleryMixin.h +++ b/submodules/LegacyComponents/PublicHeaders/LegacyComponents/TGClipboardGalleryMixin.h @@ -16,12 +16,15 @@ @property (nonatomic, copy) void (^didTransitionOut)(); @property (nonatomic, copy) UIView *(^referenceViewForItem)(TGClipboardGalleryPhotoItem *); -@property (nonatomic, copy) void (^completeWithItem)(TGClipboardGalleryPhotoItem *item); +@property (nonatomic, copy) void (^completeWithItem)(TGClipboardGalleryPhotoItem *item, bool silentPosting, int32_t scheduleTime); @property (nonatomic, copy) void (^editorOpened)(void); @property (nonatomic, copy) void (^editorClosed)(void); -- (instancetype)initWithContext:(id)context image:(UIImage *)image images:(NSArray *)images parentController:(TGViewController *)parentController thumbnailImage:(UIImage *)thumbnailImage selectionContext:(TGMediaSelectionContext *)selectionContext editingContext:(TGMediaEditingContext *)editingContext suggestionContext:(TGSuggestionContext *)suggestionContext stickersContext:(id)stickersContext hasCaptions:(bool)hasCaptions hasTimer:(bool)hasTimer recipientName:(NSString *)recipientName; +@property (nonatomic, copy) void (^presentScheduleController)(void (^)(int32_t)); +@property (nonatomic, copy) void (^presentTimerController)(void (^)(int32_t)); + +- (instancetype)initWithContext:(id)context image:(UIImage *)image images:(NSArray *)images parentController:(TGViewController *)parentController thumbnailImage:(UIImage *)thumbnailImage selectionContext:(TGMediaSelectionContext *)selectionContext editingContext:(TGMediaEditingContext *)editingContext suggestionContext:(TGSuggestionContext *)suggestionContext stickersContext:(id)stickersContext hasCaptions:(bool)hasCaptions hasTimer:(bool)hasTimer hasSilentPosting:(bool)hasSilentPosting hasSchedule:(bool)hasSchedule reminder:(bool)reminder recipientName:(NSString *)recipientName; - (void)present; diff --git a/submodules/LegacyComponents/PublicHeaders/LegacyComponents/TGClipboardMenu.h b/submodules/LegacyComponents/PublicHeaders/LegacyComponents/TGClipboardMenu.h index efe8a725ea..847cf4193b 100644 --- a/submodules/LegacyComponents/PublicHeaders/LegacyComponents/TGClipboardMenu.h +++ b/submodules/LegacyComponents/PublicHeaders/LegacyComponents/TGClipboardMenu.h @@ -14,7 +14,7 @@ @interface TGClipboardMenu : NSObject -+ (TGMenuSheetController *)presentInParentController:(TGViewController *)parentController context:(id)context images:(NSArray *)images hasCaption:(bool)hasCaption hasTimer:(bool)hasTimer recipientName:(NSString *)recipientName suggestionContext:(TGSuggestionContext *)suggestionContext stickersContext:(id)stickersContext completed:(void (^)(TGMediaSelectionContext *selectionContext, TGMediaEditingContext *editingContext, id currentItem))completed dismissed:(void (^)(void))dismissed sourceView:(UIView *)sourceView sourceRect:(CGRect (^)(void))sourceRect; ++ (TGMenuSheetController *)presentInParentController:(TGViewController *)parentController context:(id)context images:(NSArray *)images hasCaption:(bool)hasCaption hasTimer:(bool)hasTimer hasSilentPosting:(bool)hasSilentPosting hasSchedule:(bool)hasSchedule reminder:(bool)reminder recipientName:(NSString *)recipientName suggestionContext:(TGSuggestionContext *)suggestionContext stickersContext:(id)stickersContext presentScheduleController:(void (^)(void(^)(int32_t)))presentScheduleController presentTimerController:(void (^)(void(^)(int32_t)))presentTimerController completed:(void (^)(TGMediaSelectionContext *selectionContext, TGMediaEditingContext *editingContext, id currentItem, bool silentPosting, int32_t scheduleTime))completed dismissed:(void (^)(void))dismissed sourceView:(UIView *)sourceView sourceRect:(CGRect (^)(void))sourceRect; + (NSArray *)resultSignalsForSelectionContext:(TGMediaSelectionContext *)selectionContext editingContext:(TGMediaEditingContext *)editingContext currentItem:(id)currentItem descriptionGenerator:(id (^)(id, NSString *, NSArray *, NSString *))descriptionGenerator; diff --git a/submodules/LegacyComponents/Sources/TGClipboardGalleryMixin.m b/submodules/LegacyComponents/Sources/TGClipboardGalleryMixin.m index 2e8a48022f..f0881f4d1d 100644 --- a/submodules/LegacyComponents/Sources/TGClipboardGalleryMixin.m +++ b/submodules/LegacyComponents/Sources/TGClipboardGalleryMixin.m @@ -2,6 +2,8 @@ #import +#import "LegacyComponentsInternal.h" + #import #import "TGClipboardGalleryPhotoItem.h" #import "TGClipboardGalleryModel.h" @@ -15,6 +17,8 @@ #import #import +#import "TGMediaPickerSendActionSheetController.h" + @interface TGClipboardGalleryMixin () { TGMediaEditingContext *_editingContext; @@ -35,7 +39,7 @@ @implementation TGClipboardGalleryMixin -- (instancetype)initWithContext:(id)context image:(UIImage *)image images:(NSArray *)images parentController:(TGViewController *)parentController thumbnailImage:(UIImage *)thumbnailImage selectionContext:(TGMediaSelectionContext *)selectionContext editingContext:(TGMediaEditingContext *)editingContext suggestionContext:(TGSuggestionContext *)suggestionContext stickersContext:(id)stickersContext hasCaptions:(bool)hasCaptions hasTimer:(bool)hasTimer recipientName:(NSString *)recipientName +- (instancetype)initWithContext:(id)context image:(UIImage *)image images:(NSArray *)images parentController:(TGViewController *)parentController thumbnailImage:(UIImage *)thumbnailImage selectionContext:(TGMediaSelectionContext *)selectionContext editingContext:(TGMediaEditingContext *)editingContext suggestionContext:(TGSuggestionContext *)suggestionContext stickersContext:(id)stickersContext hasCaptions:(bool)hasCaptions hasTimer:(bool)hasTimer hasSilentPosting:(bool)hasSilentPosting hasSchedule:(bool)hasSchedule reminder:(bool)reminder recipientName:(NSString *)recipientName { self = [super init]; if (self != nil) @@ -109,7 +113,96 @@ strongSelf->_galleryModel.dismiss(true, false); if (strongSelf.completeWithItem != nil) - strongSelf.completeWithItem((TGClipboardGalleryPhotoItem *)item); + strongSelf.completeWithItem((TGClipboardGalleryPhotoItem *)item, false, 0); + }; + + model.interfaceView.doneLongPressed = ^(id item) { + __strong TGClipboardGalleryMixin *strongSelf = weakSelf; + if (strongSelf == nil || !(hasSilentPosting || hasSchedule)) + return; + + if (iosMajorVersion() >= 10) { + UIImpactFeedbackGenerator *generator = [[UIImpactFeedbackGenerator alloc] initWithStyle:UIImpactFeedbackStyleMedium]; + [generator impactOccurred]; + } + + bool effectiveHasSchedule = hasSchedule; + for (id item in strongSelf->_galleryModel.selectionContext.selectedItems) + { + if ([item isKindOfClass:[TGMediaAsset class]]) + { + if ([[strongSelf->_editingContext timerForItem:item] integerValue] > 0) + { + effectiveHasSchedule = false; + break; + } + } + } + + TGMediaPickerSendActionSheetController *controller = [[TGMediaPickerSendActionSheetController alloc] initWithContext:strongSelf->_context isDark:true sendButtonFrame:strongSelf.galleryModel.interfaceView.doneButtonFrame canSendSilently:hasSilentPosting canSchedule:effectiveHasSchedule reminder:reminder hasTimer:hasTimer]; + controller.send = ^{ + __strong TGClipboardGalleryMixin *strongSelf = weakSelf; + if (strongSelf == nil) + return; + + strongSelf->_galleryModel.dismiss(true, false); + + if (strongSelf.completeWithItem != nil) + strongSelf.completeWithItem((TGClipboardGalleryPhotoItem *)item, false, 0); + }; + controller.sendSilently = ^{ + __strong TGClipboardGalleryMixin *strongSelf = weakSelf; + if (strongSelf == nil) + return; + + strongSelf->_galleryModel.dismiss(true, false); + + if (strongSelf.completeWithItem != nil) + strongSelf.completeWithItem((TGClipboardGalleryPhotoItem *)item, true, 0); + }; + controller.schedule = ^{ + __strong TGClipboardGalleryMixin *strongSelf = weakSelf; + if (strongSelf == nil) + return; + + strongSelf.presentScheduleController(^(int32_t time) { + __strong TGClipboardGalleryMixin *strongSelf = weakSelf; + if (strongSelf == nil) + return; + + strongSelf->_galleryModel.dismiss(true, false); + + if (strongSelf.completeWithItem != nil) + strongSelf.completeWithItem((TGClipboardGalleryPhotoItem *)item, false, time); + }); + }; + controller.sendWithTimer = ^{ + __strong TGClipboardGalleryMixin *strongSelf = weakSelf; + if (strongSelf == nil) + return; + + strongSelf.presentTimerController(^(int32_t time) { + __strong TGClipboardGalleryMixin *strongSelf = weakSelf; + if (strongSelf == nil) + return; + + strongSelf->_galleryModel.dismiss(true, false); + + TGMediaEditingContext *editingContext = strongSelf->_editingContext; + NSMutableArray *items = [strongSelf->_galleryModel.selectionContext.selectedItems mutableCopy]; + [items addObject:((TGClipboardGalleryPhotoItem *)item).image]; + + for (id editableItem in items) { + [editingContext setTimer:@(time) forItem:editableItem]; + } + + if (strongSelf.completeWithItem != nil) + strongSelf.completeWithItem((TGClipboardGalleryPhotoItem *)item, false, 0); + }); + }; + + TGOverlayControllerWindow *controllerWindow = [[TGOverlayControllerWindow alloc] initWithManager:[strongSelf->_context makeOverlayWindowManager] parentController:strongSelf->_parentController contentController:controller]; + controllerWindow.hidden = false; }; modernGallery.model = model; diff --git a/submodules/LegacyComponents/Sources/TGClipboardMenu.m b/submodules/LegacyComponents/Sources/TGClipboardMenu.m index 219383885d..f9be4fdcfc 100644 --- a/submodules/LegacyComponents/Sources/TGClipboardMenu.m +++ b/submodules/LegacyComponents/Sources/TGClipboardMenu.m @@ -9,7 +9,7 @@ @implementation TGClipboardMenu -+ (TGMenuSheetController *)presentInParentController:(TGViewController *)parentController context:(id)context images:(NSArray *)images hasCaption:(bool)hasCaption hasTimer:(bool)hasTimer recipientName:(NSString *)recipientName suggestionContext:(TGSuggestionContext *)suggestionContext stickersContext:(id)stickersContext completed:(void (^)(TGMediaSelectionContext *selectionContext, TGMediaEditingContext *editingContext, id currentItem))completed dismissed:(void (^)(void))dismissed sourceView:(UIView *)sourceView sourceRect:(CGRect (^)(void))sourceRect ++ (TGMenuSheetController *)presentInParentController:(TGViewController *)parentController context:(id)context images:(NSArray *)images hasCaption:(bool)hasCaption hasTimer:(bool)hasTimer hasSilentPosting:(bool)hasSilentPosting hasSchedule:(bool)hasSchedule reminder:(bool)reminder recipientName:(NSString *)recipientName suggestionContext:(TGSuggestionContext *)suggestionContext stickersContext:(id)stickersContext presentScheduleController:(void (^)(void(^)(int32_t)))presentScheduleController presentTimerController:(void (^)(void(^)(int32_t)))presentTimerController completed:(void (^)(TGMediaSelectionContext *selectionContext, TGMediaEditingContext *editingContext, id currentItem, bool silentPosting, int32_t scheduleTime))completed dismissed:(void (^)(void))dismissed sourceView:(UIView *)sourceView sourceRect:(CGRect (^)(void))sourceRect { bool centered = false; if (sourceRect == nil) @@ -46,11 +46,16 @@ previewItem.parentController = parentController; previewItem.allowCaptions = hasCaption; previewItem.hasTimer = hasTimer; + previewItem.hasSilentPosting = hasSilentPosting; + previewItem.hasSchedule = hasSchedule; + previewItem.reminder = reminder; previewItem.recipientName = recipientName; - previewItem.sendPressed = ^(UIImage *currentItem) + previewItem.presentScheduleController = presentScheduleController; + previewItem.presentTimerController = presentTimerController; + previewItem.sendPressed = ^(UIImage *currentItem, bool silentPosting, int32_t scheduleTime) { __strong TGClipboardPreviewItemView *strongPreviewItem = weakPreviewItem; - completed(strongPreviewItem.selectionContext, strongPreviewItem.editingContext, currentItem); + completed(strongPreviewItem.selectionContext, strongPreviewItem.editingContext, currentItem, silentPosting, scheduleTime); __strong TGMenuSheetController *strongController = weakController; [strongController dismissAnimated:true]; @@ -68,7 +73,7 @@ TGMenuSheetButtonItemView *sendItem = [[TGMenuSheetButtonItemView alloc] initWithTitle:sendTitle type:TGMenuSheetButtonTypeSend fontSize:20.0 action:^ { __strong TGClipboardPreviewItemView *strongPreviewItem = weakPreviewItem; - completed(strongPreviewItem.selectionContext, strongPreviewItem.editingContext, nil); + completed(strongPreviewItem.selectionContext, strongPreviewItem.editingContext, nil, false, 0); __strong TGMenuSheetController *strongController = weakController; [strongController dismissAnimated:true]; diff --git a/submodules/LegacyComponents/Sources/TGClipboardPreviewItemView.h b/submodules/LegacyComponents/Sources/TGClipboardPreviewItemView.h index 076e999245..c3b5f6ef71 100644 --- a/submodules/LegacyComponents/Sources/TGClipboardPreviewItemView.h +++ b/submodules/LegacyComponents/Sources/TGClipboardPreviewItemView.h @@ -14,13 +14,20 @@ @property (nonatomic, assign) bool hasTimer; @property (nonatomic, strong) NSString *recipientName; +@property (nonatomic, assign) bool hasSchedule; +@property (nonatomic, assign) bool hasSilentPosting; +@property (nonatomic, assign) bool reminder; + @property (nonatomic, readonly) TGMediaSelectionContext *selectionContext; @property (nonatomic, readonly) TGMediaEditingContext *editingContext; @property (nonatomic, strong) TGSuggestionContext *suggestionContext; @property (nonatomic, strong) id stickersContext; @property (nonatomic, copy) void (^selectionChanged)(NSUInteger); -@property (nonatomic, copy) void (^sendPressed)(UIImage *currentItem); +@property (nonatomic, copy) void (^sendPressed)(UIImage *currentItem, bool silentPosting, int32_t scheduleTime); + +@property (nonatomic, copy) void (^presentScheduleController)(void (^)(int32_t)); +@property (nonatomic, copy) void (^presentTimerController)(void (^)(int32_t)); - (instancetype)initWithContext:(id)context images:(NSArray *)images; diff --git a/submodules/LegacyComponents/Sources/TGClipboardPreviewItemView.m b/submodules/LegacyComponents/Sources/TGClipboardPreviewItemView.m index eaf9060867..b36c74226d 100644 --- a/submodules/LegacyComponents/Sources/TGClipboardPreviewItemView.m +++ b/submodules/LegacyComponents/Sources/TGClipboardPreviewItemView.m @@ -248,12 +248,13 @@ const CGFloat TGClipboardPreviewEdgeInset = 8.0f; strongSelf->_galleryMixin = nil; }; + - mixin.completeWithItem = ^(TGClipboardGalleryPhotoItem *item) + mixin.completeWithItem = ^(TGClipboardGalleryPhotoItem *item, bool silentPosting, int32_t scheduleTime) { __strong TGClipboardPreviewItemView *strongSelf = weakSelf; if (strongSelf != nil && strongSelf.sendPressed != nil) - strongSelf.sendPressed(item.image); + strongSelf.sendPressed(item.image, silentPosting, scheduleTime); }; } @@ -266,7 +267,9 @@ const CGFloat TGClipboardPreviewEdgeInset = 8.0f; if ([cell isKindOfClass:[TGClipboardPreviewCell class]]) thumbnailImage = cell.imageView.image; - TGClipboardGalleryMixin *mixin = [[TGClipboardGalleryMixin alloc] initWithContext:_context image:image images:_images parentController:self.parentController thumbnailImage:thumbnailImage selectionContext:_selectionContext editingContext:_editingContext suggestionContext:self.suggestionContext stickersContext:self.stickersContext hasCaptions:self.allowCaptions hasTimer:self.hasTimer recipientName:self.recipientName]; + TGClipboardGalleryMixin *mixin = [[TGClipboardGalleryMixin alloc] initWithContext:_context image:image images:_images parentController:self.parentController thumbnailImage:thumbnailImage selectionContext:_selectionContext editingContext:_editingContext suggestionContext:self.suggestionContext stickersContext:self.stickersContext hasCaptions:self.allowCaptions hasTimer:self.hasTimer hasSilentPosting:self.hasSilentPosting hasSchedule:self.hasSchedule reminder:self.reminder recipientName:self.recipientName]; + mixin.presentScheduleController = self.presentScheduleController; + mixin.presentTimerController = self.presentTimerController; [self _setupGalleryMixin:mixin]; diff --git a/submodules/LegacyComponents/Sources/TGMediaPickerGalleryInterfaceView.m b/submodules/LegacyComponents/Sources/TGMediaPickerGalleryInterfaceView.m index da2484ce0f..a9b1e07199 100644 --- a/submodules/LegacyComponents/Sources/TGMediaPickerGalleryInterfaceView.m +++ b/submodules/LegacyComponents/Sources/TGMediaPickerGalleryInterfaceView.m @@ -138,7 +138,9 @@ if (strongSelf == nil) return; - [strongSelf.window endEditing:true]; + [strongSelf.window endEditing:true]; + strongSelf->_portraitToolbarView.doneButton.userInteractionEnabled = false; + strongSelf->_landscapeToolbarView.doneButton.userInteractionEnabled = false; strongSelf->_donePressed(strongSelf->_currentItem); }; void(^toolbarDoneLongPressed)(id) = ^(id sender) @@ -591,6 +593,8 @@ [_muteButton setImage:muteIcon forState:UIControlStateNormal]; [_muteButton setImage:muteActiveIcon forState:UIControlStateSelected]; [_muteButton setImage:muteActiveIcon forState:UIControlStateSelected | UIControlStateHighlighted]; + + [self setNeedsLayout]; } - (TGPhotoEditorTab)currentTabs diff --git a/submodules/LegacyMediaPickerUI/Sources/LegacyAttachmentMenu.swift b/submodules/LegacyMediaPickerUI/Sources/LegacyAttachmentMenu.swift index 422ce445e1..775c7fc6ca 100644 --- a/submodules/LegacyMediaPickerUI/Sources/LegacyAttachmentMenu.swift +++ b/submodules/LegacyMediaPickerUI/Sources/LegacyAttachmentMenu.swift @@ -383,7 +383,7 @@ public func legacyMenuPaletteFromTheme(_ theme: PresentationTheme) -> TGMenuShee return TGMenuSheetPallete(dark: theme.overallDarkAppearance, backgroundColor: sheetTheme.opaqueItemBackgroundColor, selectionColor: sheetTheme.opaqueItemHighlightedBackgroundColor, separatorColor: sheetTheme.opaqueItemSeparatorColor, accentColor: sheetTheme.controlAccentColor, destructiveColor: sheetTheme.destructiveActionTextColor, textColor: sheetTheme.primaryTextColor, secondaryTextColor: sheetTheme.secondaryTextColor, spinnerColor: sheetTheme.secondaryTextColor, badgeTextColor: sheetTheme.controlAccentColor, badgeImage: nil, cornersImage: generateStretchableFilledCircleImage(diameter: 11.0, color: nil, strokeColor: nil, strokeWidth: nil, backgroundColor: sheetTheme.opaqueItemBackgroundColor)) } -public func presentLegacyPasteMenu(context: AccountContext, peer: Peer, chatLocation: ChatLocation, saveEditedPhotos: Bool, allowGrouping: Bool, presentationData: PresentationData, images: [UIImage], sendMessagesWithSignals: @escaping ([Any]?) -> Void, presentStickers: @escaping (@escaping (TelegramMediaFile, Bool, UIView, CGRect) -> Void) -> TGPhotoPaintStickersScreen?, present: (ViewController, Any?) -> Void, initialLayout: ContainerViewLayout? = nil) -> ViewController { +public func presentLegacyPasteMenu(context: AccountContext, peer: Peer, chatLocation: ChatLocation, saveEditedPhotos: Bool, allowGrouping: Bool, hasSchedule: Bool, presentationData: PresentationData, images: [UIImage], presentSchedulePicker: @escaping (@escaping (Int32) -> Void) -> Void, presentTimerPicker: @escaping (@escaping (Int32) -> Void) -> Void, sendMessagesWithSignals: @escaping ([Any]?, Bool, Int32) -> Void, presentStickers: @escaping (@escaping (TelegramMediaFile, Bool, UIView, CGRect) -> Void) -> TGPhotoPaintStickersScreen?, present: (ViewController, Any?) -> Void, initialLayout: ContainerViewLayout? = nil) -> ViewController { let defaultVideoPreset = defaultVideoPresetForContext(context) UserDefaults.standard.set(defaultVideoPreset.rawValue as NSNumber, forKey: "TG_preferredVideoPreset_v0") @@ -420,10 +420,18 @@ public func presentLegacyPasteMenu(context: AccountContext, peer: Peer, chatLoca completion?(coder.makeData(), animated, view, rect) }) } - - let controller = TGClipboardMenu.present(inParentController: emptyController, context: legacyController.context, images: images, hasCaption: true, hasTimer: hasTimer, recipientName: recipientName, suggestionContext: suggestionContext, stickersContext: paintStickersContext, completed: { selectionContext, editingContext, currentItem in + + let controller = TGClipboardMenu.present(inParentController: emptyController, context: legacyController.context, images: images, hasCaption: true, hasTimer: hasTimer, hasSilentPosting: hasSilentPosting, hasSchedule: hasSchedule, reminder: peer.id == context.account.peerId, recipientName: recipientName, suggestionContext: suggestionContext, stickersContext: paintStickersContext, presentScheduleController: { done in + presentSchedulePicker { time in + done?(time) + } + }, presentTimerController: { done in + presentTimerPicker { time in + done?(time) + } + }, completed: { selectionContext, editingContext, currentItem, silentPosting, scheduleTime in let signals = TGClipboardMenu.resultSignals(for: selectionContext, editingContext: editingContext, currentItem: currentItem, descriptionGenerator: legacyAssetPickerItemGenerator()) - sendMessagesWithSignals(signals) + sendMessagesWithSignals(signals, silentPosting, scheduleTime) }, dismissed: { [weak legacyController] in legacyController?.dismiss() }, sourceView: emptyController.view, sourceRect: nil)! diff --git a/submodules/LegacyMediaPickerUI/Sources/LegacyMediaPickers.swift b/submodules/LegacyMediaPickerUI/Sources/LegacyMediaPickers.swift index ee8ea08f11..cb9a18d331 100644 --- a/submodules/LegacyMediaPickerUI/Sources/LegacyMediaPickers.swift +++ b/submodules/LegacyMediaPickerUI/Sources/LegacyMediaPickers.swift @@ -296,6 +296,48 @@ public func legacyEnqueueGifMessage(account: Account, data: Data) -> Signal runOn(Queue.concurrentDefaultQueue()) } +public func legacyEnqueueVideoMessage(account: Account, data: Data) -> Signal { + return Signal { subscriber in + if let previewImage = UIImage(data: data) { + let dimensions = previewImage.size + var previewRepresentations: [TelegramMediaImageRepresentation] = [] + + let thumbnailSize = dimensions.aspectFitted(CGSize(width: 320.0, height: 320.0)) + let thumbnailImage = TGScaleImageToPixelSize(previewImage, thumbnailSize)! + if let thumbnailData = thumbnailImage.jpegData(compressionQuality: 0.4) { + let resource = LocalFileMediaResource(fileId: arc4random64()) + account.postbox.mediaBox.storeResourceData(resource.id, data: thumbnailData) + previewRepresentations.append(TelegramMediaImageRepresentation(dimensions: PixelDimensions(thumbnailSize), resource: resource, progressiveSizes: [])) + } + + var randomId: Int64 = 0 + arc4random_buf(&randomId, 8) + let tempFilePath = NSTemporaryDirectory() + "\(randomId).mp4" + + let _ = try? FileManager.default.removeItem(atPath: tempFilePath) + let _ = try? data.write(to: URL(fileURLWithPath: tempFilePath), options: [.atomic]) + + let resource = LocalFileGifMediaResource(randomId: arc4random64(), path: tempFilePath) + let fileName: String = "video.mp4" + + let finalDimensions = TGMediaVideoConverter.dimensions(for: dimensions, adjustments: nil, preset: TGMediaVideoConversionPresetAnimation) + + var fileAttributes: [TelegramMediaFileAttribute] = [] + fileAttributes.append(.Video(duration: Int(0), size: PixelDimensions(finalDimensions), flags: [.supportsStreaming])) + fileAttributes.append(.FileName(fileName: fileName)) + fileAttributes.append(.Animated) + + let media = TelegramMediaFile(fileId: MediaId(namespace: Namespaces.Media.LocalFile, id: arc4random64()), partialReference: nil, resource: resource, previewRepresentations: previewRepresentations, videoThumbnails: [], immediateThumbnailData: nil, mimeType: "video/mp4", size: nil, attributes: fileAttributes) + subscriber.putNext(.message(text: "", attributes: [], mediaReference: .standalone(media: media), replyToMessageId: nil, localGroupingKey: nil)) + subscriber.putCompletion() + } else { + subscriber.putError(Void()) + } + + return EmptyDisposable + } |> runOn(Queue.concurrentDefaultQueue()) +} + public func legacyAssetPickerEnqueueMessages(account: Account, signals: [Any]) -> Signal<[EnqueueMessage], Void> { return Signal { subscriber in let disposable = SSignal.combineSignals(signals).start(next: { anyValues in diff --git a/submodules/LocationUI/Sources/LocationSectionHeaderItem.swift b/submodules/LocationUI/Sources/LocationSectionHeaderItem.swift index 8674611913..0d138c093f 100644 --- a/submodules/LocationUI/Sources/LocationSectionHeaderItem.swift +++ b/submodules/LocationUI/Sources/LocationSectionHeaderItem.swift @@ -1,7 +1,6 @@ import Foundation import UIKit import AsyncDisplayKit -import Postbox import Display import SwiftSignalKit import TelegramPresentationData diff --git a/submodules/MtProtoKit/Sources/MTApiEnvironment.m b/submodules/MtProtoKit/Sources/MTApiEnvironment.m index e6e67a701e..f8dd947b67 100644 --- a/submodules/MtProtoKit/Sources/MTApiEnvironment.m +++ b/submodules/MtProtoKit/Sources/MTApiEnvironment.m @@ -10,88 +10,10 @@ #import -#define IFPGA_NAMESTRING @"iFPGA" - -#define IPHONE_1G_NAMESTRING @"iPhone 1G" -#define IPHONE_3G_NAMESTRING @"iPhone 3G" -#define IPHONE_3GS_NAMESTRING @"iPhone 3GS" -#define IPHONE_4_NAMESTRING @"iPhone 4" -#define IPHONE_4S_NAMESTRING @"iPhone 4S" -#define IPHONE_5_NAMESTRING @"iPhone 5" -#define IPHONE_5S_NAMESTRING @"iPhone 5S" -#define IPHONE_6_NAMESTRING @"iPhone 6" -#define IPHONE_6Plus_NAMESTRING @"iPhone 6 Plus" -#define IPHONE_6S_NAMESTRING @"iPhone 6S" -#define IPHONE_6SPlus_NAMESTRING @"iPhone 6S Plus" -#define IPHONE_7_NAMESTRING @"iPhone 7" -#define IPHONE_7Plus_NAMESTRING @"iPhone 7 Plus" -#define IPHONE_8_NAMESTRING @"iPhone 8" -#define IPHONE_8Plus_NAMESTRING @"iPhone 8 Plus" -#define IPHONE_X_NAMESTRING @"iPhone X" -#define IPHONE_SE_NAMESTRING @"iPhone SE" -#define IPHONE_SE2_NAMESTRING @"iPhone SE (2nd gen)" -#define IPHONE_XS_NAMESTRING @"iPhone XS" -#define IPHONE_XSMAX_NAMESTRING @"iPhone XS Max" -#define IPHONE_XR_NAMESTRING @"iPhone XR" -#define IPHONE_11_NAMESTRING @"iPhone 11" -#define IPHONE_11PRO_NAMESTRING @"iPhone 11 Pro" -#define IPHONE_11PROMAX_NAMESTRING @"iPhone 11 Pro Max" -#define IPHONE_12MINI_NAMESTRING @"iPhone 12 mini" -#define IPHONE_12_NAMESTRING @"iPhone 12" -#define IPHONE_12PRO_NAMESTRING @"iPhone 12 Pro" -#define IPHONE_12PROMAX_NAMESTRING @"iPhone 12 Pro Max" - -#define IPHONE_UNKNOWN_NAMESTRING @"Unknown iPhone" - -#define IPOD_1G_NAMESTRING @"iPod touch 1G" -#define IPOD_2G_NAMESTRING @"iPod touch 2G" -#define IPOD_3G_NAMESTRING @"iPod touch 3G" -#define IPOD_4G_NAMESTRING @"iPod touch 4G" -#define IPOD_5G_NAMESTRING @"iPod touch 5G" -#define IPOD_6G_NAMESTRING @"iPod touch 6G" -#define IPOD_7G_NAMESTRING @"iPod touch 7G" -#define IPOD_UNKNOWN_NAMESTRING @"Unknown iPod" - -#define IPAD_1G_NAMESTRING @"iPad 1G" -#define IPAD_2G_NAMESTRING @"iPad 2G" -#define IPAD_3G_NAMESTRING @"iPad 3G" -#define IPAD_4G_NAMESTRING @"iPad 4G" -#define IPAD_5G_NAMESTRING @"iPad Air 2" -#define IPAD_6G_NAMESTRING @"iPad Pro" -#define IPAD_PRO_3G_NAMESTRING @"iPad Pro 12.9 inch (3rd gen)" -#define IPAD_PRO_11_NAMESTRING @"iPad Pro 11 inch" -#define IPAD_PRO_6G_NAMESTRING @"iPad (6th gen)" -#define IPAD_PRO_10_5_NAMESTRING @"iPad Pro 10.5 inch" -#define IPAD_PRO_12_9_NAMESTRING @"iPad Pro 12.9 inch" -#define IPAD_UNKNOWN_NAMESTRING @"Unknown iPad" - -#define APPLETV_2G_NAMESTRING @"Apple TV 2G" -#define APPLETV_3G_NAMESTRING @"Apple TV 3G" -#define APPLETV_4G_NAMESTRING @"Apple TV 4G" -#define APPLETV_UNKNOWN_NAMESTRING @"Unknown Apple TV" - -#define IOS_FAMILY_UNKNOWN_DEVICE @"Unknown iOS device" - -#define SIMULATOR_NAMESTRING @"iPhone Simulator" -#define SIMULATOR_IPHONE_NAMESTRING @"iPhone Simulator" -#define SIMULATOR_IPAD_NAMESTRING @"iPad Simulator" -#define SIMULATOR_APPLETV_NAMESTRING @"Apple TV Simulator" - -/* - iPad8,5, iPad8,6, iPad8,7, iPad8,8 - iPad Pro 12.9" (3rd gen) - iPad8,1, iPad8,2, iPad8,3, iPad8,4 - iPad Pro 11" - iPad7,5, iPad7,6 - iPad 6th gen - iPad7,3, iPad7,4 - iPad Pro 10.5" - iPad7,1, iPad7,2 - iPad Pro 12.9" (2ng gen) - */ - typedef enum { UIDeviceUnknown, UIDeviceSimulator, - UIDeviceSimulatoriPhone, - UIDeviceSimulatoriPad, - UIDeviceSimulatorAppleTV, UIDevice1GiPhone, UIDevice3GiPhone, @@ -157,15 +79,6 @@ typedef enum { } UIDevicePlatform; -typedef enum { - UIDeviceFamilyiPhone, - UIDeviceFamilyiPod, - UIDeviceFamilyiPad, - UIDeviceFamilyAppleTV, - UIDeviceFamilyUnknown, - -} UIDeviceFamily; - static NSData * _Nullable parseHexString(NSString * _Nonnull hex) { if ([hex length] % 2 != 0) { return nil; @@ -587,79 +500,213 @@ NSString *suffix = @""; - (NSString *)platformString { - switch ([self platformType]) - { - case UIDevice1GiPhone: return IPHONE_1G_NAMESTRING; - case UIDevice3GiPhone: return IPHONE_3G_NAMESTRING; - case UIDevice3GSiPhone: return IPHONE_3GS_NAMESTRING; - case UIDevice4iPhone: return IPHONE_4_NAMESTRING; - case UIDevice4SiPhone: return IPHONE_4S_NAMESTRING; - case UIDevice5iPhone: return IPHONE_5_NAMESTRING; - case UIDevice5SiPhone: return IPHONE_5S_NAMESTRING; - case UIDevice6iPhone: return IPHONE_6_NAMESTRING; - case UIDevice6PlusiPhone: return IPHONE_6Plus_NAMESTRING; - case UIDevice6SiPhone: return IPHONE_6S_NAMESTRING; - case UIDevice6SPlusiPhone: return IPHONE_6SPlus_NAMESTRING; - case UIDevice7iPhone: return IPHONE_7_NAMESTRING; - case UIDevice7PlusiPhone: return IPHONE_7Plus_NAMESTRING; - case UIDevice8iPhone: return IPHONE_8_NAMESTRING; - case UIDevice8PlusiPhone: return IPHONE_8Plus_NAMESTRING; - case UIDeviceXiPhone: return IPHONE_X_NAMESTRING; - case UIDeviceSEPhone: return IPHONE_SE_NAMESTRING; - case UIDeviceSE2Phone: return IPHONE_SE2_NAMESTRING; - case UIDeviceXSiPhone: return IPHONE_XS_NAMESTRING; - case UIDeviceXSMaxiPhone: return IPHONE_XSMAX_NAMESTRING; - case UIDeviceXRiPhone: return IPHONE_XR_NAMESTRING; - case UIDevice11iPhone: return IPHONE_11_NAMESTRING; - case UIDevice11ProiPhone: return IPHONE_11PRO_NAMESTRING; - case UIDevice11ProMaxiPhone: return IPHONE_11PROMAX_NAMESTRING; - case UIDevice12MiniiPhone: return IPHONE_12MINI_NAMESTRING; - case UIDevice12iPhone: return IPHONE_12_NAMESTRING; - case UIDevice12ProiPhone: return IPHONE_12PRO_NAMESTRING; - case UIDevice12ProMaxiPhone: return IPHONE_12PROMAX_NAMESTRING; - case UIDeviceUnknowniPhone: return IPHONE_UNKNOWN_NAMESTRING; - - case UIDevice1GiPod: return IPOD_1G_NAMESTRING; - case UIDevice2GiPod: return IPOD_2G_NAMESTRING; - case UIDevice3GiPod: return IPOD_3G_NAMESTRING; - case UIDevice4GiPod: return IPOD_4G_NAMESTRING; - case UIDevice5GiPod: return IPOD_5G_NAMESTRING; - case UIDevice6GiPod: return IPOD_6G_NAMESTRING; - case UIDevice7GiPod: return IPOD_7G_NAMESTRING; - case UIDeviceUnknowniPod: return IPOD_UNKNOWN_NAMESTRING; - - case UIDevice1GiPad : return IPAD_1G_NAMESTRING; - case UIDevice2GiPad : return IPAD_2G_NAMESTRING; - case UIDevice3GiPad : return IPAD_3G_NAMESTRING; - case UIDevice4GiPad : return IPAD_4G_NAMESTRING; - case UIDevice5GiPad : return IPAD_5G_NAMESTRING; - case UIDevice6GiPad : return IPAD_6G_NAMESTRING; - case UIDeviceiPadPro12_93g : return IPAD_PRO_12_9_NAMESTRING; - case UIDeviceiPadPro11 : return IPAD_PRO_11_NAMESTRING; - case UIDeviceiPadPro6g : return IPAD_PRO_6G_NAMESTRING; - case UIDeviceiPadPro10_5 : return IPAD_PRO_10_5_NAMESTRING; - case UIDeviceiPadPro12_9 : return IPAD_PRO_12_9_NAMESTRING; - case UIDeviceUnknowniPad : return IPAD_UNKNOWN_NAMESTRING; - - case UIDeviceAppleTV2 : return APPLETV_2G_NAMESTRING; - case UIDeviceAppleTV3 : return APPLETV_3G_NAMESTRING; - case UIDeviceAppleTV4 : return APPLETV_4G_NAMESTRING; - case UIDeviceUnknownAppleTV: return APPLETV_UNKNOWN_NAMESTRING; - - case UIDeviceSimulator: return SIMULATOR_NAMESTRING; - case UIDeviceSimulatoriPhone: return SIMULATOR_IPHONE_NAMESTRING; - case UIDeviceSimulatoriPad: return SIMULATOR_IPAD_NAMESTRING; - case UIDeviceSimulatorAppleTV: return SIMULATOR_APPLETV_NAMESTRING; - - case UIDeviceIFPGA: return IFPGA_NAMESTRING; - - case UIDeviceOSX: return [self macHWName]; +#if TARGET_OS_IPHONE + NSString *platform = [self platform]; + + if ([platform isEqualToString:@"iPhone1,1"]) + return @"iPhone"; + if ([platform isEqualToString:@"iPhone1,2"]) + return @"iPhone 3G"; + if ([platform isEqualToString:@"iPhone2,1"]) + return @"iPhone 3GS"; + if ([platform hasPrefix:@"iPhone3"]) + return @"iPhone 4"; + if ([platform hasPrefix:@"iPhone4"]) + return @"iPhone 4S"; + if ([platform isEqualToString:@"iPhone5,1"] || + [platform isEqualToString:@"iPhone5,2"]) + return @"iPhone 5"; + if ([platform isEqualToString:@"iPhone5,3"] || + [platform isEqualToString:@"iPhone5,4"]) + return @"iPhone 5C"; + if ([platform hasPrefix:@"iPhone6"]) + return @"iPhone 5S"; + if ([platform isEqualToString:@"iPhone7,1"]) + return @"iPhone 6 Plus"; + if ([platform isEqualToString:@"iPhone7,2"]) + return @"iPhone 6"; + if ([platform isEqualToString:@"iPhone8,1"]) + return @"iPhone 6S"; + if ([platform isEqualToString:@"iPhone8,2"]) + return @"iPhone 6S Plus"; + if ([platform isEqualToString:@"iPhone8,4"]) + return @"iPhone SE"; + if ([platform isEqualToString:@"iPhone9,1"] || + [platform isEqualToString:@"iPhone9,3"]) + return @"iPhone 7"; + if ([platform isEqualToString:@"iPhone9,2"] || + [platform isEqualToString:@"iPhone9,4"]) + return @"iPhone 7 Plus"; + if ([platform isEqualToString:@"iPhone10,1"] || + [platform isEqualToString:@"iPhone10,4"]) + return @"iPhone 8"; + if ([platform isEqualToString:@"iPhone10,2"] || + [platform isEqualToString:@"iPhone10,5"]) + return @"iPhone 8 Plus"; + if ([platform isEqualToString:@"iPhone10,3"] || + [platform isEqualToString:@"iPhone10,6"]) + return @"iPhone X"; + if ([platform isEqualToString:@"iPhone11,2"]) + return @"iPhone XS"; + if ([platform isEqualToString:@"iPhone11,4"] || + [platform isEqualToString:@"iPhone11,6"]) + return @"iPhone XS Max"; + if ([platform isEqualToString:@"iPhone11,8"]) + return @"iPhone XR"; + if ([platform isEqualToString:@"iPhone12,1"]) + return @"iPhone 11"; + if ([platform isEqualToString:@"iPhone12,3"]) + return @"iPhone 11 Pro"; + if ([platform isEqualToString:@"iPhone12,5"]) + return @"iPhone 11 Pro Max"; + if ([platform isEqualToString:@"iPhone12,8"]) + return @"iPhone SE (2nd gen)"; + if ([platform isEqualToString:@"iPhone13,1"]) + return @"iPhone 12 mini"; + if ([platform isEqualToString:@"iPhone13,2"]) + return @"iPhone 12"; + if ([platform isEqualToString:@"iPhone13,3"]) + return @"iPhone 12 Pro"; + if ([platform isEqualToString:@"iPhone13,4"]) + return @"iPhone 12 Pro Max"; + + if ([platform hasPrefix:@"iPod1"]) + return @"iPod touch 1G"; + if ([platform hasPrefix:@"iPod2"]) + return @"iPod touch 2G"; + if ([platform hasPrefix:@"iPod3"]) + return @"iPod touch 3G"; + if ([platform hasPrefix:@"iPod4"]) + return @"iPod touch 4G"; + if ([platform hasPrefix:@"iPod5"]) + return @"iPod touch 5G"; + if ([platform hasPrefix:@"iPod7"]) + return @"iPod touch 6G"; + if ([platform hasPrefix:@"iPod9"]) + return @"iPod touch 7G"; + + if ([platform isEqualToString:@"iPad2,5"] || + [platform isEqualToString:@"iPad2,6"] || + [platform isEqualToString:@"iPad2,7"]) + return @"iPad mini"; + + if ([platform hasPrefix:@"iPad2"]) + return @"iPad 2G"; + + if ([platform isEqualToString:@"iPad3,1"] || + [platform isEqualToString:@"iPad3,2"] || + [platform isEqualToString:@"iPad3,3"]) + return @"iPad 3G"; + + if ([platform isEqualToString:@"iPad3,4"] || + [platform isEqualToString:@"iPad3,5"] || + [platform isEqualToString:@"iPad3,6"]) + return @"iPad 3G"; + + if ([platform isEqualToString:@"iPad4,1"] || + [platform isEqualToString:@"iPad4,2"]) + return @"iPad Air"; - default: return IOS_FAMILY_UNKNOWN_DEVICE; + if ([platform isEqualToString:@"iPad4,4"] || + [platform isEqualToString:@"iPad4,5"] || + [platform isEqualToString:@"iPad4,6"]) + return @"iPad mini Retina"; + + if ([platform isEqualToString:@"iPad4,7"] || + [platform isEqualToString:@"iPad4,8"] || + [platform isEqualToString:@"iPad4,9"]) + return @"iPad mini 3"; + + if ([platform isEqualToString:@"iPad5,1"] || + [platform isEqualToString:@"iPad5,2"]) + return @"iPad mini 4"; + + if ([platform isEqualToString:@"iPad5,3"] || + [platform isEqualToString:@"iPad5,4"]) + return @"iPad Air 2"; + + if ([platform isEqualToString:@"iPad6,3"] || + [platform isEqualToString:@"iPad6,4"]) + return @"iPad Pro 9.7 inch"; + + if ([platform isEqualToString:@"iPad6,7"] || + [platform isEqualToString:@"iPad6,8"]) + return @"iPad Pro 12.9 inch"; + + if ([platform isEqualToString:@"iPad6,11"] || + [platform isEqualToString:@"iPad6,12"]) + return @"iPad (2017)"; + + if ([platform isEqualToString:@"iPad7,1"] || + [platform isEqualToString:@"iPad7,2"]) + return @"iPad Pro (2nd gen)"; + + if ([platform isEqualToString:@"iPad7,3"] || + [platform isEqualToString:@"iPad7,4"]) + return @"iPad Pro 10.5 inch"; + + if ([platform isEqualToString:@"iPad7,5"] || + [platform isEqualToString:@"iPad7,6"]) + return @"iPad (6th gen)"; + + if ([platform isEqualToString:@"iPad7,11"] || + [platform isEqualToString:@"iPad7,12"]) + return @"iPad 10.2 inch (7th gen)"; + + if ([platform isEqualToString:@"iPad8,1"] || + [platform isEqualToString:@"iPad8,2"] || + [platform isEqualToString:@"iPad8,3"] || + [platform isEqualToString:@"iPad8,4"]) + return @"iPad Pro 11 inch (3rd gen)"; + + if ([platform isEqualToString:@"iPad8,5"] || + [platform isEqualToString:@"iPad8,6"] || + [platform isEqualToString:@"iPad8,7"] || + [platform isEqualToString:@"iPad8,8"]) + return @"iPad Pro 12.9 inch (3rd gen)"; + + if ([platform isEqualToString:@"iPad8,9"] || + [platform isEqualToString:@"iPad8,10"]) + return @"iPad Pro 11 inch (4th gen)"; + + if ([platform isEqualToString:@"iPad8,11"] || + [platform isEqualToString:@"iPad8,12"]) + return @"iPad Pro 12.9 inch (4th gen)"; + + if ([platform isEqualToString:@"iPad11,1"] || + [platform isEqualToString:@"iPad11,2"]) + return @"iPad mini (5th gen)"; + + if ([platform isEqualToString:@"iPad11,3"] || + [platform isEqualToString:@"iPad11,4"]) + return @"iPad Air (3rd gen)"; + + if ([platform isEqualToString:@"iPad11,6"] || + [platform isEqualToString:@"iPad11,7"]) + return @"iPad (8th gen)"; + + if ([platform isEqualToString:@"iPad13,1"] || + [platform isEqualToString:@"iPad13,2"]) + return @"iPad Air (4th gen)"; + + if ([platform hasPrefix:@"iPhone"]) + return @"Unknown iPhone"; + if ([platform hasPrefix:@"iPod"]) + return @"Unknown iPod"; + if ([platform hasPrefix:@"iPad"]) + return @"Unknown iPad"; + + if ([platform hasSuffix:@"86"] || [platform isEqual:@"x86_64"] || [platform isEqual:@"arm64"]) { + return @"iPhone Simulator"; } +#else + return [self macHWName]; +#endif + + return @"Unknown iOS device"; } --(NSString *)macHWName { +- (NSString *)macHWName { size_t len = 0; sysctlbyname("hw.model", NULL, &len, NULL, 0); if (len) { @@ -672,120 +719,6 @@ NSString *suffix = @""; return @"macOS"; } -- (NSUInteger)platformType -{ -#if TARGET_OS_IPHONE - NSString *platform = [self platform]; - - // The ever mysterious iFPGA - if ([platform isEqualToString:@"iFPGA"]) return UIDeviceIFPGA; - - // iPhone - if ([platform isEqualToString:@"iPhone1,1"]) return UIDevice1GiPhone; - if ([platform isEqualToString:@"iPhone1,2"]) return UIDevice3GiPhone; - if ([platform hasPrefix:@"iPhone2"]) return UIDevice3GSiPhone; - if ([platform hasPrefix:@"iPhone3"]) return UIDevice4iPhone; - if ([platform hasPrefix:@"iPhone4"]) return UIDevice4SiPhone; - if ([platform hasPrefix:@"iPhone5"]) return UIDevice5iPhone; - if ([platform hasPrefix:@"iPhone6"]) return UIDevice5SiPhone; - - if ([platform isEqualToString:@"iPhone7,1"]) return UIDevice6PlusiPhone; - if ([platform isEqualToString:@"iPhone7,2"]) return UIDevice6iPhone; - if ([platform isEqualToString:@"iPhone8,1"]) return UIDevice6SiPhone; - if ([platform isEqualToString:@"iPhone8,2"]) return UIDevice6SPlusiPhone; - if ([platform isEqualToString:@"iPhone9,1"]) return UIDevice7iPhone; - if ([platform isEqualToString:@"iPhone9,3"]) return UIDevice7iPhone; - if ([platform isEqualToString:@"iPhone9,2"]) return UIDevice7PlusiPhone; - if ([platform isEqualToString:@"iPhone9,4"]) return UIDevice7PlusiPhone; - - if ([platform isEqualToString:@"iPhone10,1"]) return UIDevice8iPhone; - if ([platform isEqualToString:@"iPhone10,4"]) return UIDevice8iPhone; - if ([platform isEqualToString:@"iPhone10,2"]) return UIDevice8PlusiPhone; - if ([platform isEqualToString:@"iPhone10,5"]) return UIDevice8PlusiPhone; - if ([platform isEqualToString:@"iPhone10,3"]) return UIDeviceXiPhone; - if ([platform isEqualToString:@"iPhone10,6"]) return UIDeviceXiPhone; - if ([platform isEqualToString:@"iPhone11,2"]) return UIDeviceXSiPhone; - if ([platform isEqualToString:@"iPhone11,6"]) return UIDeviceXSMaxiPhone; - if ([platform isEqualToString:@"iPhone11,4"]) return UIDeviceXSMaxiPhone; - if ([platform isEqualToString:@"iPhone11,8"]) return UIDeviceXRiPhone; - if ([platform isEqualToString:@"iPhone12,1"]) return UIDevice11iPhone; - if ([platform isEqualToString:@"iPhone12,3"]) return UIDevice11ProiPhone; - if ([platform isEqualToString:@"iPhone12,5"]) return UIDevice11ProMaxiPhone; - if ([platform isEqualToString:@"iPhone12,8"]) return UIDeviceSE2Phone; - if ([platform isEqualToString:@"iPhone13,1"]) return UIDevice12MiniiPhone; - if ([platform isEqualToString:@"iPhone13,2"]) return UIDevice12iPhone; - if ([platform isEqualToString:@"iPhone13,3"]) return UIDevice12ProiPhone; - if ([platform isEqualToString:@"iPhone13,4"]) return UIDevice12ProMaxiPhone; - - if ([platform isEqualToString:@"iPhone8,4"]) return UIDeviceSEPhone; - - // iPod - if ([platform hasPrefix:@"iPod1"]) return UIDevice1GiPod; - if ([platform hasPrefix:@"iPod2"]) return UIDevice2GiPod; - if ([platform hasPrefix:@"iPod3"]) return UIDevice3GiPod; - if ([platform hasPrefix:@"iPod4"]) return UIDevice4GiPod; - if ([platform hasPrefix:@"iPod5"]) return UIDevice5GiPod; - if ([platform hasPrefix:@"iPod7"]) return UIDevice6GiPod; - if ([platform hasPrefix:@"iPod9"]) return UIDevice7GiPod; - - // iPad - if ([platform hasPrefix:@"iPad1"]) return UIDevice1GiPad; - if ([platform hasPrefix:@"iPad2"]) return UIDevice2GiPad; - if ([platform hasPrefix:@"iPad3"]) return UIDevice3GiPad; - if ([platform hasPrefix:@"iPad4"]) return UIDevice4GiPad; - if ([platform hasPrefix:@"iPad5"]) return UIDevice5GiPad; - if ([platform hasPrefix:@"iPad6"]) return UIDevice6GiPad; - - if ([platform isEqualToString:@"iPad8,5"] || - [platform isEqualToString:@"iPad8,6"] || - [platform isEqualToString:@"iPad8,7"] || - [platform isEqualToString:@"iPad8,8"]) { - return UIDeviceiPadPro12_93g; - } - - if ([platform isEqualToString:@"iPad8,1"] || - [platform isEqualToString:@"iPad8,2"] || - [platform isEqualToString:@"iPad8,3"] || - [platform isEqualToString:@"iPad8,4"]) { - return UIDeviceiPadPro11; - } - - if ([platform isEqualToString:@"iPad7,5"] || - [platform isEqualToString:@"iPad7,6"]) { - return UIDeviceiPadPro6g; - } - - if ([platform isEqualToString:@"iPad7,3"] || - [platform isEqualToString:@"iPad7,4"]) { - return UIDeviceiPadPro10_5; - } - - if ([platform isEqualToString:@"iPad7,1"] || - [platform isEqualToString:@"iPad7,2"]) { - return UIDeviceiPadPro12_9; - } - - // Apple TV - if ([platform hasPrefix:@"AppleTV2"]) return UIDeviceAppleTV2; - if ([platform hasPrefix:@"AppleTV3"]) return UIDeviceAppleTV3; - - if ([platform hasPrefix:@"iPhone"]) return UIDeviceUnknowniPhone; - if ([platform hasPrefix:@"iPod"]) return UIDeviceUnknowniPod; - if ([platform hasPrefix:@"iPad"]) return UIDeviceUnknowniPad; - if ([platform hasPrefix:@"AppleTV"]) return UIDeviceUnknownAppleTV; - - // Simulator thanks Jordan Breeding - if ([platform hasSuffix:@"86"] || [platform isEqual:@"x86_64"]) - { - return UIDeviceSimulatoriPhone; - } -#else - return UIDeviceOSX; -#endif - - return UIDeviceUnknown; -} - - (NSString *)getSysInfoByName:(char *)typeSpecifier { size_t size; diff --git a/submodules/PeerInfoUI/BUILD b/submodules/PeerInfoUI/BUILD index b8bdb1454d..c89dcb5bc4 100644 --- a/submodules/PeerInfoUI/BUILD +++ b/submodules/PeerInfoUI/BUILD @@ -67,6 +67,7 @@ swift_library( "//submodules/ChatListSearchItemHeader:ChatListSearchItemHeader", "//submodules/StatisticsUI:StatisticsUI", "//submodules/ChatListFilterSettingsHeaderItem:ChatListFilterSettingsHeaderItem", + "//submodules/InviteLinksUI:InviteLinksUI", ], visibility = [ "//visibility:public", diff --git a/submodules/PeerInfoUI/Sources/ChannelAdminController.swift b/submodules/PeerInfoUI/Sources/ChannelAdminController.swift index e2a10577aa..66a67e3b3e 100644 --- a/submodules/PeerInfoUI/Sources/ChannelAdminController.swift +++ b/submodules/PeerInfoUI/Sources/ChannelAdminController.swift @@ -691,7 +691,7 @@ private func channelAdminControllerEntries(presentationData: PresentationData, s canTransfer = true } - if let initialParticipant = initialParticipant, case let .member(participant) = initialParticipant, let adminInfo = participant.adminInfo, !adminInfo.rights.flags.isEmpty && admin.id != accountPeerId { + if let initialParticipant = initialParticipant, case let .member(_, _, adminInfoValue, _, _) = initialParticipant, let adminInfo = adminInfoValue, !adminInfo.rights.flags.isEmpty && admin.id != accountPeerId { if channel.flags.contains(.isCreator) { canDismiss = true } else { @@ -772,6 +772,7 @@ private func channelAdminControllerEntries(presentationData: PresentationData, s .canBanUsers, .canInviteUsers, .canPinMessages, + .canManageCalls, .canBeAnonymous, .canAddAdmins ] @@ -782,9 +783,9 @@ private func channelAdminControllerEntries(presentationData: PresentationData, s if let updatedFlags = state.updatedFlags { currentRightsFlags = updatedFlags } else if let initialParticipant = initialParticipant, case let .member(_, _, maybeAdminRights, _, _) = initialParticipant, let adminRights = maybeAdminRights { - currentRightsFlags = adminRights.rights.flags.subtracting(.canAddAdmins) + currentRightsFlags = adminRights.rights.flags.subtracting(.canAddAdmins).subtracting(.canBeAnonymous) } else { - currentRightsFlags = accountUserRightsFlags.subtracting(.canAddAdmins) + currentRightsFlags = accountUserRightsFlags.subtracting(.canAddAdmins).subtracting(.canBeAnonymous) } var index = 0 @@ -1126,7 +1127,7 @@ public func channelAdminController(context: AccountContext, peerId: PeerId, admi })) } } - } else if let _ = channelView.peers[channelView.peerId] as? TelegramGroup { + } else if let group = channelView.peers[channelView.peerId] as? TelegramGroup { var updateFlags: TelegramChatAdminRightsFlags? var updateRank: String? updateState { current in @@ -1143,7 +1144,12 @@ public func channelAdminController(context: AccountContext, peerId: PeerId, admi } let maskRightsFlags: TelegramChatAdminRightsFlags = .groupSpecific - let defaultFlags = maskRightsFlags.subtracting(.canAddAdmins) + let defaultFlags: TelegramChatAdminRightsFlags + if case .creator = group.role { + defaultFlags = maskRightsFlags.subtracting(.canBeAnonymous) + } else { + defaultFlags = maskRightsFlags.subtracting(.canAddAdmins).subtracting(.canBeAnonymous) + } if updateFlags == nil { updateFlags = defaultFlags diff --git a/submodules/PeerInfoUI/Sources/ChannelMembersController.swift b/submodules/PeerInfoUI/Sources/ChannelMembersController.swift index 5540394f63..d1ed834b2a 100644 --- a/submodules/PeerInfoUI/Sources/ChannelMembersController.swift +++ b/submodules/PeerInfoUI/Sources/ChannelMembersController.swift @@ -13,6 +13,7 @@ import AccountContext import AlertUI import PresentationDataUtils import ItemListPeerItem +import InviteLinksUI private final class ChannelMembersControllerArguments { let context: AccountContext @@ -461,7 +462,7 @@ public func channelMembersController(context: AccountContext, peerId: PeerId) -> pushControllerImpl?(controller) } }, inviteViaLink: { - presentControllerImpl?(channelVisibilityController(context: context, peerId: peerId, mode: .privateLink, upgradedToSupergroup: { _, f in f() }), ViewControllerPresentationArguments(presentationAnimation: .modalSheet)) + pushControllerImpl?(InviteLinkInviteController(context: context, peerId: peerId)) }) let peerView = context.account.viewTracker.peerView(peerId) diff --git a/submodules/PeerInfoUI/Sources/ChannelMembersSearchControllerNode.swift b/submodules/PeerInfoUI/Sources/ChannelMembersSearchControllerNode.swift index 6722f996b2..6ffc81a030 100644 --- a/submodules/PeerInfoUI/Sources/ChannelMembersSearchControllerNode.swift +++ b/submodules/PeerInfoUI/Sources/ChannelMembersSearchControllerNode.swift @@ -361,16 +361,16 @@ class ChannelMembersSearchControllerNode: ASDisplayNode { let renderedParticipant: RenderedChannelParticipant switch participant { case .creator: - renderedParticipant = RenderedChannelParticipant(participant: .creator(id: peer.id, adminInfo: nil, rank: nil), peer: peer) + renderedParticipant = RenderedChannelParticipant(participant: .creator(id: peer.id, adminInfo: nil, rank: nil), peer: peer, presences: peerView.peerPresences) case .admin: var peers: [PeerId: Peer] = [:] peers[creator.id] = creator peers[peer.id] = peer - renderedParticipant = RenderedChannelParticipant(participant: .member(id: peer.id, invitedAt: 0, adminInfo: ChannelParticipantAdminInfo(rights: TelegramChatAdminRights(flags: .groupSpecific), promotedBy: creator.id, canBeEditedByAccountPeer: creator.id == context.account.peerId), banInfo: nil, rank: nil), peer: peer, peers: peers) + renderedParticipant = RenderedChannelParticipant(participant: .member(id: peer.id, invitedAt: 0, adminInfo: ChannelParticipantAdminInfo(rights: TelegramChatAdminRights(flags: .groupSpecific), promotedBy: creator.id, canBeEditedByAccountPeer: creator.id == context.account.peerId), banInfo: nil, rank: nil), peer: peer, peers: peers, presences: peerView.peerPresences) case .member: var peers: [PeerId: Peer] = [:] peers[peer.id] = peer - renderedParticipant = RenderedChannelParticipant(participant: .member(id: peer.id, invitedAt: 0, adminInfo: nil, banInfo: nil, rank: nil), peer: peer, peers: peers) + renderedParticipant = RenderedChannelParticipant(participant: .member(id: peer.id, invitedAt: 0, adminInfo: nil, banInfo: nil, rank: nil), peer: peer, peers: peers, presences: peerView.peerPresences) } entries.append(.peer(index, renderedParticipant, ContactsPeerItemEditing(editable: false, editing: false, revealed: false), label, enabled)) diff --git a/submodules/PeerInfoUI/Sources/ChannelVisibilityController.swift b/submodules/PeerInfoUI/Sources/ChannelVisibilityController.swift index 45c79c36f3..d4a59cc038 100644 --- a/submodules/PeerInfoUI/Sources/ChannelVisibilityController.swift +++ b/submodules/PeerInfoUI/Sources/ChannelVisibilityController.swift @@ -1,6 +1,7 @@ import Foundation import UIKit import Display +import AsyncDisplayKit import SwiftSignalKit import Postbox import TelegramCore @@ -16,22 +17,24 @@ import AlertUI import PresentationDataUtils import TelegramNotices import ItemListPeerItem +import ItemListPeerActionItem import AccountContext +import InviteLinksUI +import ContextUI private final class ChannelVisibilityControllerArguments { let context: AccountContext - let updateCurrentType: (CurrentChannelType) -> Void let updatePublicLinkText: (String?, String) -> Void let scrollToPublicLinkText: () -> Void let displayPrivateLinkMenu: (String) -> Void let setPeerIdWithRevealedOptions: (PeerId?, PeerId?) -> Void let revokePeerId: (PeerId) -> Void - let copyPrivateLink: () -> Void - let revokePrivateLink: () -> Void - let sharePrivateLink: () -> Void + let shareLink: () -> Void + let linkContextAction: (ASDisplayNode) -> Void + let manageInviteLinks: () -> Void - init(context: AccountContext, updateCurrentType: @escaping (CurrentChannelType) -> Void, updatePublicLinkText: @escaping (String?, String) -> Void, scrollToPublicLinkText: @escaping () -> Void, displayPrivateLinkMenu: @escaping (String) -> Void, setPeerIdWithRevealedOptions: @escaping (PeerId?, PeerId?) -> Void, revokePeerId: @escaping (PeerId) -> Void, copyPrivateLink: @escaping () -> Void, revokePrivateLink: @escaping () -> Void, sharePrivateLink: @escaping () -> Void) { + init(context: AccountContext, updateCurrentType: @escaping (CurrentChannelType) -> Void, updatePublicLinkText: @escaping (String?, String) -> Void, scrollToPublicLinkText: @escaping () -> Void, displayPrivateLinkMenu: @escaping (String) -> Void, setPeerIdWithRevealedOptions: @escaping (PeerId?, PeerId?) -> Void, revokePeerId: @escaping (PeerId) -> Void, shareLink: @escaping () -> Void, linkContextAction: @escaping (ASDisplayNode) -> Void, manageInviteLinks: @escaping () -> Void) { self.context = context self.updateCurrentType = updateCurrentType self.updatePublicLinkText = updatePublicLinkText @@ -39,9 +42,9 @@ private final class ChannelVisibilityControllerArguments { self.displayPrivateLinkMenu = displayPrivateLinkMenu self.setPeerIdWithRevealedOptions = setPeerIdWithRevealedOptions self.revokePeerId = revokePeerId - self.copyPrivateLink = copyPrivateLink - self.revokePrivateLink = revokePrivateLink - self.sharePrivateLink = sharePrivateLink + self.shareLink = shareLink + self.linkContextAction = linkContextAction + self.manageInviteLinks = manageInviteLinks } } @@ -49,7 +52,6 @@ private enum ChannelVisibilitySection: Int32 { case type case link case linkActions - case location } private enum ChannelVisibilityEntryTag: ItemListItemTag { @@ -73,12 +75,13 @@ private enum ChannelVisibilityEntry: ItemListNodeEntry { case publicLinkHeader(PresentationTheme, String) case publicLinkAvailability(PresentationTheme, String, Bool) - case privateLink(PresentationTheme, String, String?) case editablePublicLink(PresentationTheme, PresentationStrings, String, String) + case privateLinkHeader(PresentationTheme, String) + case privateLink(PresentationTheme, ExportedInvitation?) case privateLinkInfo(PresentationTheme, String) - case privateLinkCopy(PresentationTheme, String) - case privateLinkRevoke(PresentationTheme, String) - case privateLinkShare(PresentationTheme, String) + case privateLinkManage(PresentationTheme, String) + case privateLinkManageInfo(PresentationTheme, String) + case publicLinkInfo(PresentationTheme, String) case publicLinkStatus(PresentationTheme, String, AddressNameValidationStatus) @@ -89,9 +92,9 @@ private enum ChannelVisibilityEntry: ItemListNodeEntry { switch self { case .typeHeader, .typePublic, .typePrivate, .typeInfo: return ChannelVisibilitySection.type.rawValue - case .publicLinkHeader, .publicLinkAvailability, .privateLink, .editablePublicLink, .privateLinkInfo, .publicLinkInfo, .publicLinkStatus: + case .publicLinkHeader, .publicLinkAvailability, .privateLinkHeader, .privateLink, .editablePublicLink, .privateLinkInfo, .publicLinkInfo, .publicLinkStatus: return ChannelVisibilitySection.link.rawValue - case .privateLinkCopy, .privateLinkRevoke, .privateLinkShare: + case .privateLinkManage, .privateLinkManageInfo: return ChannelVisibilitySection.linkActions.rawValue case .existingLinksInfo, .existingLinkPeerItem: return ChannelVisibilitySection.link.rawValue @@ -112,17 +115,17 @@ private enum ChannelVisibilityEntry: ItemListNodeEntry { return 4 case .publicLinkAvailability: return 5 - case .privateLink: + case .privateLinkHeader: return 6 - case .editablePublicLink: + case .privateLink: return 7 - case .privateLinkInfo: + case .editablePublicLink: return 8 - case .privateLinkCopy: + case .privateLinkInfo: return 9 - case .privateLinkRevoke: + case .privateLinkManage: return 10 - case .privateLinkShare: + case .privateLinkManageInfo: return 11 case .publicLinkStatus: return 12 @@ -173,8 +176,14 @@ private enum ChannelVisibilityEntry: ItemListNodeEntry { } else { return false } - case let .privateLink(lhsTheme, lhsText, lhsLink): - if case let .privateLink(rhsTheme, rhsText, rhsLink) = rhs, lhsTheme === rhsTheme, lhsText == rhsText, lhsLink == rhsLink { + case let .privateLinkHeader(lhsTheme, lhsTitle): + if case let .privateLinkHeader(rhsTheme, rhsTitle) = rhs, lhsTheme === rhsTheme, lhsTitle == rhsTitle { + return true + } else { + return false + } + case let .privateLink(lhsTheme, lhsInvite): + if case let .privateLink(rhsTheme, rhsInvite) = rhs, lhsTheme === rhsTheme, lhsInvite == rhsInvite { return true } else { return false @@ -191,20 +200,14 @@ private enum ChannelVisibilityEntry: ItemListNodeEntry { } else { return false } - case let .privateLinkCopy(lhsTheme, lhsText): - if case let .privateLinkCopy(rhsTheme, rhsText) = rhs, lhsTheme === rhsTheme, lhsText == rhsText { + case let .privateLinkManage(lhsTheme, lhsText): + if case let .privateLinkManage(rhsTheme, rhsText) = rhs, lhsTheme === rhsTheme, lhsText == rhsText { return true } else { return false } - case let .privateLinkRevoke(lhsTheme, lhsText): - if case let .privateLinkRevoke(rhsTheme, rhsText) = rhs, lhsTheme === rhsTheme, lhsText == rhsText { - return true - } else { - return false - } - case let .privateLinkShare(lhsTheme, lhsText): - if case let .privateLinkShare(rhsTheme, rhsText) = rhs, lhsTheme === rhsTheme, lhsText == rhsText { + case let .privateLinkManageInfo(lhsTheme, lhsText): + if case let .privateLinkManageInfo(rhsTheme, rhsText) = rhs, lhsTheme === rhsTheme, lhsText == rhsText { return true } else { return false @@ -267,31 +270,35 @@ private enum ChannelVisibilityEntry: ItemListNodeEntry { func item(presentationData: ItemListPresentationData, arguments: Any) -> ListViewItem { let arguments = arguments as! ChannelVisibilityControllerArguments switch self { - case let .typeHeader(theme, title): + case let .typeHeader(_, title): return ItemListSectionHeaderItem(presentationData: presentationData, text: title, sectionId: self.section) - case let .typePublic(theme, text, selected): + case let .typePublic(_, text, selected): return ItemListCheckboxItem(presentationData: presentationData, title: text, style: .left, checked: selected, zeroSeparatorInsets: false, sectionId: self.section, action: { arguments.updateCurrentType(.publicChannel) }) - case let .typePrivate(theme, text, selected): + case let .typePrivate(_, text, selected): return ItemListCheckboxItem(presentationData: presentationData, title: text, style: .left, checked: selected, zeroSeparatorInsets: false, sectionId: self.section, action: { arguments.updateCurrentType(.privateChannel) }) - case let .typeInfo(theme, text): + case let .typeInfo(_, text): return ItemListTextItem(presentationData: presentationData, text: .plain(text), sectionId: self.section) - case let .publicLinkHeader(theme, title): + case let .publicLinkHeader(_, title): return ItemListSectionHeaderItem(presentationData: presentationData, text: title, sectionId: self.section) case let .publicLinkAvailability(theme, text, value): let attr = NSMutableAttributedString(string: text, textColor: value ? theme.list.freeTextColor : theme.list.freeTextErrorColor) attr.addAttribute(.font, value: Font.regular(13), range: NSMakeRange(0, attr.length)) return ItemListActivityTextItem(displayActivity: value, presentationData: presentationData, text: attr, sectionId: self.section) - case let .privateLink(theme, text, value): - return ItemListActionItem(presentationData: presentationData, title: text, kind: value != nil ? .neutral : .disabled, alignment: .natural, sectionId: self.section, style: .blocks, action: { - if let value = value { - arguments.displayPrivateLinkMenu(value) - } - }, tag: ChannelVisibilityEntryTag.privateLink) - case let .editablePublicLink(theme, strings, placeholder, currentText): + case let .privateLinkHeader(_, title): + return ItemListSectionHeaderItem(presentationData: presentationData, text: title, sectionId: self.section) + case let .privateLink(_, invite): + return ItemListPermanentInviteLinkItem(context: arguments.context, presentationData: presentationData, invite: invite, peers: [], buttonColor: nil, sectionId: self.section, style: .blocks, shareAction: { + arguments.shareLink() + }, contextAction: { node in + arguments.linkContextAction(node) + }, viewAction: { + + }) + case let .editablePublicLink(theme, _, placeholder, currentText): return ItemListSingleLineInputItem(presentationData: presentationData, title: NSAttributedString(string: "t.me/", textColor: theme.list.itemPrimaryTextColor), text: currentText, placeholder: placeholder, type: .regular(capitalization: false, autocorrection: false), clearType: .always, tag: ChannelVisibilityEntryTag.publicLink, sectionId: self.section, textUpdated: { updatedText in arguments.updatePublicLinkText(currentText, updatedText) }, updatedFocus: { focus in @@ -300,21 +307,15 @@ private enum ChannelVisibilityEntry: ItemListNodeEntry { } }, action: { }) - case let .privateLinkInfo(theme, text): + case let .privateLinkInfo(_, text): return ItemListTextItem(presentationData: presentationData, text: .plain(text), sectionId: self.section) - case let .privateLinkCopy(theme, text): - return ItemListActionItem(presentationData: presentationData, title: text, kind: .generic, alignment: .natural, sectionId: self.section, style: .blocks, action: { - arguments.copyPrivateLink() + case let .privateLinkManage(theme, text): + return ItemListPeerActionItem(presentationData: presentationData, icon: PresentationResourcesItemList.linkIcon(theme), title: text, sectionId: self.section, editing: false, action: { + arguments.manageInviteLinks() }) - case let .privateLinkRevoke(theme, text): - return ItemListActionItem(presentationData: presentationData, title: text, kind: .generic, alignment: .natural, sectionId: self.section, style: .blocks, action: { - arguments.revokePrivateLink() - }) - case let .privateLinkShare(theme, text): - return ItemListActionItem(presentationData: presentationData, title: text, kind: .generic, alignment: .natural, sectionId: self.section, style: .blocks, action: { - arguments.sharePrivateLink() - }) - case let .publicLinkInfo(theme, text): + case let .privateLinkManageInfo(_, text): + return ItemListTextItem(presentationData: presentationData, text: .markdown(text), sectionId: self.section) + case let .publicLinkInfo(_, text): return ItemListTextItem(presentationData: presentationData, text: .markdown(text), sectionId: self.section) case let .publicLinkStatus(theme, text, status): var displayActivity = false @@ -336,9 +337,9 @@ private enum ChannelVisibilityEntry: ItemListNodeEntry { displayActivity = true } return ItemListActivityTextItem(displayActivity: displayActivity, presentationData: presentationData, text: NSAttributedString(string: text, textColor: color), sectionId: self.section) - case let .existingLinksInfo(theme, text): + case let .existingLinksInfo(_, text): return ItemListTextItem(presentationData: presentationData, text: .plain(text), sectionId: self.section) - case let .existingLinkPeerItem(_, theme, strings, dateTimeFormat, nameDisplayOrder, peer, editing, enabled): + case let .existingLinkPeerItem(_, _, _, dateTimeFormat, nameDisplayOrder, peer, editing, enabled): var label = "" if let addressName = peer.addressName { label = "t.me/" + addressName @@ -591,14 +592,9 @@ private func channelVisibilityControllerEntries(presentationData: PresentationDa } } case .privateChannel: - let link = (view.cachedData as? CachedChannelData)?.exportedInvitation?.link - let text: String - if let link = link { - text = link - } else { - text = presentationData.strings.Channel_NotificationLoading - } - entries.append(.privateLink(presentationData.theme, text, link)) + let invite = (view.cachedData as? CachedChannelData)?.exportedInvitation + entries.append(.privateLinkHeader(presentationData.theme, presentationData.strings.InviteLink_PermanentLink.uppercased())) + entries.append(.privateLink(presentationData.theme, invite)) if isGroup { entries.append(.privateLinkInfo(presentationData.theme, presentationData.strings.Group_Username_CreatePrivateLinkHelp)) } else { @@ -608,34 +604,25 @@ private func channelVisibilityControllerEntries(presentationData: PresentationDa case .initialSetup: break case .generic, .privateLink: - entries.append(.privateLinkCopy(presentationData.theme, presentationData.strings.GroupInfo_InviteLink_CopyLink)) - entries.append(.privateLinkRevoke(presentationData.theme, presentationData.strings.GroupInfo_InviteLink_RevokeLink)) - entries.append(.privateLinkShare(presentationData.theme, presentationData.strings.GroupInfo_InviteLink_ShareLink)) + entries.append(.privateLinkManage(presentationData.theme, presentationData.strings.InviteLink_Manage)) + entries.append(.privateLinkManageInfo(presentationData.theme, presentationData.strings.InviteLink_CreateInfo)) } } } else if let _ = view.peers[view.peerId] as? TelegramGroup { switch mode { - case .privateLink: - let link = (view.cachedData as? CachedGroupData)?.exportedInvitation?.link - let text: String - if let link = link { - text = link - } else { - text = presentationData.strings.Channel_NotificationLoading - } - entries.append(.privateLink(presentationData.theme, text, link)) + case .privateLink: + let invite = (view.cachedData as? CachedGroupData)?.exportedInvitation + entries.append(.privateLinkHeader(presentationData.theme, presentationData.strings.InviteLink_PermanentLink.uppercased())) + entries.append(.privateLink(presentationData.theme, invite)) entries.append(.privateLinkInfo(presentationData.theme, presentationData.strings.GroupInfo_InviteLink_Help)) switch mode { case .initialSetup: break case .generic, .privateLink: - entries.append(.privateLinkCopy(presentationData.theme, presentationData.strings.GroupInfo_InviteLink_CopyLink)) - entries.append(.privateLinkRevoke(presentationData.theme, presentationData.strings.GroupInfo_InviteLink_RevokeLink)) - entries.append(.privateLinkShare(presentationData.theme, presentationData.strings.GroupInfo_InviteLink_ShareLink)) + entries.append(.privateLinkManage(presentationData.theme, presentationData.strings.InviteLink_Manage)) + entries.append(.privateLinkManageInfo(presentationData.theme, presentationData.strings.InviteLink_CreateInfo)) } - case .generic, .initialSetup: - let isGroup = true - + case .generic, .initialSetup: let selectedType: CurrentChannelType if let current = state.selectedType { selectedType = current @@ -723,22 +710,16 @@ private func channelVisibilityControllerEntries(presentationData: PresentationDa entries.append(.publicLinkInfo(presentationData.theme, presentationData.strings.Group_Username_CreatePublicLinkHelp)) } case .privateChannel: - let link = (view.cachedData as? CachedGroupData)?.exportedInvitation?.link - let text: String - if let link = link { - text = link - } else { - text = presentationData.strings.Channel_NotificationLoading - } - entries.append(.privateLink(presentationData.theme, text, link)) + let invite = (view.cachedData as? CachedGroupData)?.exportedInvitation + entries.append(.privateLinkHeader(presentationData.theme, presentationData.strings.InviteLink_PermanentLink.uppercased())) + entries.append(.privateLink(presentationData.theme, invite)) entries.append(.privateLinkInfo(presentationData.theme, presentationData.strings.Group_Username_CreatePrivateLinkHelp)) switch mode { case .initialSetup: break case .generic, .privateLink: - entries.append(.privateLinkCopy(presentationData.theme, presentationData.strings.GroupInfo_InviteLink_CopyLink)) - entries.append(.privateLinkRevoke(presentationData.theme, presentationData.strings.GroupInfo_InviteLink_RevokeLink)) - entries.append(.privateLinkShare(presentationData.theme, presentationData.strings.GroupInfo_InviteLink_ShareLink)) + entries.append(.privateLinkManage(presentationData.theme, presentationData.strings.InviteLink_Manage)) + entries.append(.privateLinkManageInfo(presentationData.theme, presentationData.strings.InviteLink_CreateInfo)) } } } @@ -747,7 +728,7 @@ private func channelVisibilityControllerEntries(presentationData: PresentationDa return entries } -private func effectiveChannelType(state: ChannelVisibilityControllerState, peer: TelegramChannel, cachedData: CachedPeerData?) -> CurrentChannelType { +private func effectiveChannelType(mode: ChannelVisibilityControllerMode, state: ChannelVisibilityControllerState, peer: TelegramChannel, cachedData: CachedPeerData?) -> CurrentChannelType { let selectedType: CurrentChannelType if let current = state.selectedType { selectedType = current @@ -756,6 +737,8 @@ private func effectiveChannelType(state: ChannelVisibilityControllerState, peer: selectedType = .publicChannel } else if let cachedChannelData = cachedData as? CachedChannelData, cachedChannelData.peerGeoLocation != nil { selectedType = .publicChannel + } else if case .initialSetup = mode { + selectedType = .publicChannel } else { selectedType = .privateChannel } @@ -763,9 +746,9 @@ private func effectiveChannelType(state: ChannelVisibilityControllerState, peer: return selectedType } -private func updatedAddressName(state: ChannelVisibilityControllerState, peer: Peer, cachedData: CachedPeerData?) -> String? { +private func updatedAddressName(mode: ChannelVisibilityControllerMode, state: ChannelVisibilityControllerState, peer: Peer, cachedData: CachedPeerData?) -> String? { if let peer = peer as? TelegramChannel { - let selectedType = effectiveChannelType(state: state, peer: peer, cachedData: cachedData) + let selectedType = effectiveChannelType(mode: mode, state: state, peer: peer, cachedData: cachedData) let currentAddressName: String @@ -831,13 +814,13 @@ public func channelVisibilityController(context: AccountContext, peerId: PeerId, })) var dismissImpl: (() -> Void)? - var dismissInputImpl: (() -> Void)? var nextImpl: (() -> Void)? var displayPrivateLinkMenuImpl: ((String) -> Void)? var scrollToPublicLinkTextImpl: (() -> Void)? var presentControllerImpl: ((ViewController, Any?) -> Void)? var pushControllerImpl: ((ViewController) -> Void)? - var clearHighlightImpl: (() -> Void)? + var presentInGlobalOverlayImpl: ((ViewController) -> Void)? + var getControllerImpl: (() -> ViewController?)? let actionsDisposable = DisposableSet() @@ -915,57 +898,7 @@ public func channelVisibilityController(context: AccountContext, peerId: PeerId, } }) })) - }, copyPrivateLink: { - let _ = (context.account.postbox.transaction { transaction -> String? in - if let cachedData = transaction.getPeerCachedData(peerId: peerId) { - if let cachedData = cachedData as? CachedChannelData { - return cachedData.exportedInvitation?.link - } else if let cachedData = cachedData as? CachedGroupData { - return cachedData.exportedInvitation?.link - } - } - return nil - } |> deliverOnMainQueue).start(next: { link in - if let link = link { - UIPasteboard.general.string = link - let presentationData = context.sharedContext.currentPresentationData.with { $0 } - presentControllerImpl?(OverlayStatusController(theme: presentationData.theme, type: .genericSuccess(presentationData.strings.Username_LinkCopied, false)), nil) - } - }) - }, revokePrivateLink: { - 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.GroupInfo_InviteLink_RevokeAlert_Text), - ActionSheetButtonItem(title: presentationData.strings.GroupInfo_InviteLink_RevokeLink, color: .destructive, action: { - dismissAction() - - var revoke = false - updateState { state in - if !state.revokingPrivateLink { - revoke = true - return state.withUpdatedRevokingPrivateLink(true) - } else { - return state - } - } - if revoke { - revokeLinkDisposable.set((ensuredExistingPeerExportedInvitation(account: context.account, peerId: peerId, revokeExisted: true) |> deliverOnMainQueue).start(completed: { - updateState { - $0.withUpdatedRevokingPrivateLink(false) - } - })) - } - }) - ]), - ActionSheetItemGroup(items: [ActionSheetButtonItem(title: presentationData.strings.Common_Cancel, action: { dismissAction() })]) - ]) - presentControllerImpl?(controller, ViewControllerPresentationArguments(presentationAnimation: .modalSheet)) - }, sharePrivateLink: { + }, shareLink: { let _ = (context.account.postbox.transaction { transaction -> String? in if let cachedData = transaction.getPeerCachedData(peerId: peerId) { if let cachedData = cachedData as? CachedChannelData { @@ -981,6 +914,101 @@ public func channelVisibilityController(context: AccountContext, peerId: PeerId, presentControllerImpl?(shareController, nil) } }) + }, linkContextAction: { node in + guard let node = node as? ContextExtractedContentContainingNode, let controller = getControllerImpl?() else { + return + } + let presentationData = context.sharedContext.currentPresentationData.with { $0 } + var items: [ContextMenuItem] = [] + + items.append(.action(ContextMenuActionItem(text: presentationData.strings.InviteLink_ContextCopy, icon: { theme in + return generateTintedImage(image: UIImage(bundleImageName: "Chat/Context Menu/Copy"), color: theme.contextMenu.primaryColor) + }, action: { _, f in + f(.dismissWithoutContent) + + let _ = (context.account.postbox.transaction { transaction -> String? in + if let cachedData = transaction.getPeerCachedData(peerId: peerId) { + if let cachedData = cachedData as? CachedChannelData { + return cachedData.exportedInvitation?.link + } else if let cachedData = cachedData as? CachedGroupData { + return cachedData.exportedInvitation?.link + } + } + return nil + } |> deliverOnMainQueue).start(next: { link in + if let link = link { + UIPasteboard.general.string = link + let presentationData = context.sharedContext.currentPresentationData.with { $0 } + presentControllerImpl?(OverlayStatusController(theme: presentationData.theme, type: .genericSuccess(presentationData.strings.Username_LinkCopied, false)), nil) + } + }) + }))) + + items.append(.action(ContextMenuActionItem(text: presentationData.strings.InviteLink_ContextGetQRCode, icon: { theme in + return generateTintedImage(image: UIImage(bundleImageName: "Wallet/QrIcon"), color: theme.contextMenu.primaryColor) + }, action: { _, f in + f(.dismissWithoutContent) + + let _ = (context.account.postbox.transaction { transaction -> ExportedInvitation? in + if let cachedData = transaction.getPeerCachedData(peerId: peerId) { + if let cachedData = cachedData as? CachedChannelData { + return cachedData.exportedInvitation + } else if let cachedData = cachedData as? CachedGroupData { + return cachedData.exportedInvitation + } + } + return nil + } |> deliverOnMainQueue).start(next: { invite in + if let invite = invite { + let controller = InviteLinkQRCodeController(context: context, invite: invite) + presentControllerImpl?(controller, nil) + } + }) + }))) + + items.append(.action(ContextMenuActionItem(text: presentationData.strings.InviteLink_ContextRevoke, textColor: .destructive, icon: { theme in + return generateTintedImage(image: UIImage(bundleImageName: "Chat/Context Menu/Delete"), color: theme.actionSheet.destructiveActionTextColor) + }, action: { _, f in + f(.dismissWithoutContent) + + let controller = ActionSheetController(presentationData: presentationData) + let dismissAction: () -> Void = { [weak controller] in + controller?.dismissAnimated() + } + controller.setItemGroups([ + ActionSheetItemGroup(items: [ + ActionSheetTextItem(title: presentationData.strings.GroupInfo_InviteLink_RevokeAlert_Text), + ActionSheetButtonItem(title: presentationData.strings.GroupInfo_InviteLink_RevokeLink, color: .destructive, action: { + dismissAction() + + var revoke = false + updateState { state in + if !state.revokingPrivateLink { + revoke = true + return state.withUpdatedRevokingPrivateLink(true) + } else { + return state + } + } + if revoke { + revokeLinkDisposable.set((ensuredExistingPeerExportedInvitation(account: context.account, peerId: peerId, revokeExisted: true) |> deliverOnMainQueue).start(completed: { + updateState { + $0.withUpdatedRevokingPrivateLink(false) + } + })) + } + }) + ]), + ActionSheetItemGroup(items: [ActionSheetButtonItem(title: presentationData.strings.Common_Cancel, action: { dismissAction() })]) + ]) + presentControllerImpl?(controller, ViewControllerPresentationArguments(presentationAnimation: .modalSheet)) + }))) + + let contextController = ContextController(account: context.account, presentationData: presentationData, source: .extracted(InviteLinkContextExtractedContentSource(controller: controller, sourceNode: node)), items: .single(items), reactionItems: [], gesture: nil) + presentInGlobalOverlayImpl?(contextController) + }, manageInviteLinks: { + let controller = inviteLinkListController(context: context, peerId: peerId) + pushControllerImpl?(controller) }) let peerView = context.account.viewTracker.peerView(peerId) @@ -1022,7 +1050,7 @@ public func channelVisibilityController(context: AccountContext, peerId: PeerId, rightNavigationButton = ItemListNavigationButton(content: .text(mode == .initialSetup ? presentationData.strings.Common_Next : presentationData.strings.Common_Done), style: state.updatingAddressName ? .activity : .bold, enabled: doneEnabled, action: { var updatedAddressNameValue: String? updateState { state in - updatedAddressNameValue = updatedAddressName(state: state, peer: peer, cachedData: view.cachedData) + updatedAddressNameValue = updatedAddressName(mode: mode, state: state, peer: peer, cachedData: view.cachedData) return state } @@ -1093,7 +1121,7 @@ public func channelVisibilityController(context: AccountContext, peerId: PeerId, rightNavigationButton = ItemListNavigationButton(content: .text(presentationData.strings.Common_Done), style: state.updatingAddressName ? .activity : .bold, enabled: doneEnabled, action: { var updatedAddressNameValue: String? updateState { state in - updatedAddressNameValue = updatedAddressName(state: state, peer: peer, cachedData: nil) + updatedAddressNameValue = updatedAddressName(mode: mode, state: state, peer: peer, cachedData: nil) return state } @@ -1251,9 +1279,6 @@ public func channelVisibilityController(context: AccountContext, peerId: PeerId, controller.dismiss() } } - dismissInputImpl = { [weak controller] in - controller?.view.endEditing(true) - } nextImpl = { [weak controller] in if let controller = controller { if case .initialSetup = mode { @@ -1360,8 +1385,37 @@ public func channelVisibilityController(context: AccountContext, peerId: PeerId, pushControllerImpl = { [weak controller] c in controller?.push(c) } - clearHighlightImpl = { [weak controller] in - controller?.clearItemNodesHighlight(animated: true) + presentInGlobalOverlayImpl = { [weak controller] c in + if let controller = controller { + controller.presentInGlobalOverlay(c) + } + } + getControllerImpl = { [weak controller] in + return controller } return controller } + +private final class InviteLinkContextExtractedContentSource: ContextExtractedContentSource { + var keepInPlace: Bool + let ignoreContentTouches: Bool = true + let blurBackground: Bool + + private let controller: ViewController + private let sourceNode: ContextExtractedContentContainingNode + + init(controller: ViewController, sourceNode: ContextExtractedContentContainingNode) { + self.controller = controller + self.sourceNode = sourceNode + self.keepInPlace = false + self.blurBackground = false + } + + func takeView() -> ContextControllerTakeViewInfo? { + return ContextControllerTakeViewInfo(contentContainingNode: self.sourceNode, contentAreaInScreenSpace: UIScreen.main.bounds) + } + + func putBack() -> ContextControllerPutBackViewInfo? { + return ContextControllerPutBackViewInfo(contentAreaInScreenSpace: UIScreen.main.bounds) + } +} diff --git a/submodules/PeersNearbyUI/Sources/PeersNearbyController.swift b/submodules/PeersNearbyUI/Sources/PeersNearbyController.swift index cd39d720cc..ee5ec1a372 100644 --- a/submodules/PeersNearbyUI/Sources/PeersNearbyController.swift +++ b/submodules/PeersNearbyUI/Sources/PeersNearbyController.swift @@ -449,7 +449,6 @@ private class PeersNearbyControllerImpl: ItemListController { } } - public func peersNearbyController(context: AccountContext) -> ViewController { var pushControllerImpl: ((ViewController) -> Void)? var replaceTopControllerImpl: ((ViewController) -> Void)? diff --git a/submodules/Postbox/Sources/ChatListViewState.swift b/submodules/Postbox/Sources/ChatListViewState.swift index a3fc67730f..765dfc264a 100644 --- a/submodules/Postbox/Sources/ChatListViewState.swift +++ b/submodules/Postbox/Sources/ChatListViewState.swift @@ -1341,6 +1341,7 @@ struct ChatListViewState { case let .IntermediateMessageEntry(index, messageIndex): var peers = SimpleDictionary() var notificationsPeerId = index.messageIndex.id.peerId + var presence: PeerPresence? if let peer = postbox.peerTable.get(index.messageIndex.id.peerId) { peers[peer.id] = peer if let notificationSettingsPeerId = peer.notificationSettingsPeerId { @@ -1350,6 +1351,9 @@ struct ChatListViewState { if let associatedPeer = postbox.peerTable.get(associatedPeerId) { peers[associatedPeer.id] = associatedPeer } + presence = postbox.peerPresenceTable.get(associatedPeerId) + } else { + presence = postbox.peerPresenceTable.get(index.messageIndex.id.peerId) } } let renderedPeer = RenderedPeer(peerId: index.messageIndex.id.peerId, peers: peers) @@ -1387,7 +1391,7 @@ struct ChatListViewState { } } - let updatedEntry: MutableChatListEntry = .MessageEntry(index: index, messages: renderedMessages, readState: postbox.readStateTable.getCombinedState(index.messageIndex.id.peerId), notificationSettings: notificationSettings, isRemovedFromTotalUnreadCount: isRemovedFromTotalUnreadCount, embeddedInterfaceState: postbox.peerChatInterfaceStateTable.get(index.messageIndex.id.peerId)?.chatListEmbeddedState, renderedPeer: renderedPeer, presence: postbox.peerPresenceTable.get(index.messageIndex.id.peerId), tagSummaryInfo: tagSummaryInfo, hasFailedMessages: false, isContact: postbox.contactsTable.isContact(peerId: index.messageIndex.id.peerId)) + let updatedEntry: MutableChatListEntry = .MessageEntry(index: index, messages: renderedMessages, readState: postbox.readStateTable.getCombinedState(index.messageIndex.id.peerId), notificationSettings: notificationSettings, isRemovedFromTotalUnreadCount: isRemovedFromTotalUnreadCount, embeddedInterfaceState: postbox.peerChatInterfaceStateTable.get(index.messageIndex.id.peerId)?.chatListEmbeddedState, renderedPeer: renderedPeer, presence: presence, tagSummaryInfo: tagSummaryInfo, hasFailedMessages: false, isContact: postbox.contactsTable.isContact(peerId: index.messageIndex.id.peerId)) if directionIndex == 0 { self.stateBySpace[space]!.orderedEntries.setLowerOrAtAnchorAtArrayIndex(listIndex, to: updatedEntry) } else { diff --git a/submodules/Postbox/Sources/Postbox.swift b/submodules/Postbox/Sources/Postbox.swift index 2e7fb9c916..078c837cdc 100644 --- a/submodules/Postbox/Sources/Postbox.swift +++ b/submodules/Postbox/Sources/Postbox.swift @@ -2787,7 +2787,11 @@ public final class Postbox { let mutableView = MutableChatListView(postbox: self, groupId: groupId, filterPredicate: filterPredicate, aroundIndex: index, count: count, summaryComponents: summaryComponents) mutableView.render(postbox: self, renderMessage: self.renderIntermediateMessage, getPeer: { id in return self.peerTable.get(id) - }, getPeerNotificationSettings: { self.peerNotificationSettingsTable.getEffective($0) }, getPeerPresence: { self.peerPresenceTable.get($0) }) + }, getPeerNotificationSettings: { + self.peerNotificationSettingsTable.getEffective($0) + }, getPeerPresence: { + self.peerPresenceTable.get($0) + }) let (index, signal) = self.viewTracker.addChatListView(mutableView) diff --git a/submodules/SectionHeaderItem/BUILD b/submodules/SectionHeaderItem/BUILD new file mode 100644 index 0000000000..e8089be45a --- /dev/null +++ b/submodules/SectionHeaderItem/BUILD @@ -0,0 +1,20 @@ +load("@build_bazel_rules_swift//swift:swift.bzl", "swift_library") + +swift_library( + name = "SectionHeaderItem", + module_name = "SectionHeaderItem", + srcs = glob([ + "Sources/**/*.swift", + ]), + deps = [ + "//submodules/SSignalKit/SwiftSignalKit:SwiftSignalKit", + "//submodules/AsyncDisplayKit:AsyncDisplayKit", + "//submodules/Display:Display", + "//submodules/TelegramPresentationData:TelegramPresentationData", + "//submodules/ItemListUI:ItemListUI", + "//submodules/ListSectionHeaderNode:ListSectionHeaderNode", + ], + visibility = [ + "//visibility:public", + ], +) diff --git a/submodules/SectionHeaderItem/Sources/SectionHeaderItem.swift b/submodules/SectionHeaderItem/Sources/SectionHeaderItem.swift new file mode 100644 index 0000000000..726cdac204 --- /dev/null +++ b/submodules/SectionHeaderItem/Sources/SectionHeaderItem.swift @@ -0,0 +1,119 @@ +import Foundation +import UIKit +import AsyncDisplayKit +import Display +import SwiftSignalKit +import TelegramPresentationData +import ListSectionHeaderNode +import ItemListUI + +public class SectionHeaderItem: ListViewItem { + let presentationData: ItemListPresentationData + let title: String + + public init(presentationData: ItemListPresentationData, title: String) { + self.presentationData = presentationData + self.title = title + } + + public func nodeConfiguredForParams(async: @escaping (@escaping () -> Void) -> Void, params: ListViewItemLayoutParams, synchronousLoads: Bool, previousItem: ListViewItem?, nextItem: ListViewItem?, completion: @escaping (ListViewItemNode, @escaping () -> (Signal?, (ListViewItemApply) -> Void)) -> Void) { + async { + let node = SectionHeaderItemNode() + let makeLayout = node.asyncLayout() + let (nodeLayout, nodeApply) = makeLayout(self, params) + node.contentSize = nodeLayout.contentSize + node.insets = nodeLayout.insets + + completion(node, nodeApply) + } + } + + public func updateNode(async: @escaping (@escaping () -> Void) -> Void, node: @escaping () -> ListViewItemNode, params: ListViewItemLayoutParams, previousItem: ListViewItem?, nextItem: ListViewItem?, animation: ListViewItemUpdateAnimation, completion: @escaping (ListViewItemNodeLayout, @escaping (ListViewItemApply) -> Void) -> Void) { + Queue.mainQueue().async { + if let nodeValue = node() as? SectionHeaderItemNode { + let layout = nodeValue.asyncLayout() + async { + let (nodeLayout, apply) = layout(self, params) + Queue.mainQueue().async { + completion(nodeLayout, { info in + apply().1(info) + }) + } + } + } + } + } + + public var selectable: Bool { + return false + } +} + +private class SectionHeaderItemNode: ListViewItemNode { + private var headerNode: ListSectionHeaderNode? + + private var item: SectionHeaderItem? + private var layoutParams: ListViewItemLayoutParams? + + required init() { + super.init(layerBacked: false, dynamicBounce: false, rotated: false, seeThrough: false) + } + + override func layoutForParams(_ params: ListViewItemLayoutParams, item: ListViewItem, previousItem: ListViewItem?, nextItem: ListViewItem?) { + if let item = self.item { + let makeLayout = self.asyncLayout() + let (nodeLayout, nodeApply) = makeLayout(item, params) + self.contentSize = nodeLayout.contentSize + self.insets = nodeLayout.insets + let _ = nodeApply() + } + } + + func asyncLayout() -> (_ item: SectionHeaderItem, _ params: ListViewItemLayoutParams) -> (ListViewItemNodeLayout, () -> (Signal?, (ListViewItemApply) -> Void)) { + let currentItem = self.item + + return { [weak self] item, params in + let contentSize = CGSize(width: params.width, height: 28.0) + let nodeLayout = ListViewItemNodeLayout(contentSize: contentSize, insets: UIEdgeInsets()) + + return (nodeLayout, { [weak self] in + var updatedTheme: PresentationTheme? + if currentItem?.presentationData.theme !== item.presentationData.theme { + updatedTheme = item.presentationData.theme + } + + return (nil, { _ in + if let strongSelf = self { + strongSelf.item = item + strongSelf.layoutParams = params + + let headerNode: ListSectionHeaderNode + if let currentHeaderNode = strongSelf.headerNode { + headerNode = currentHeaderNode + + if let _ = updatedTheme { + headerNode.updateTheme(theme: item.presentationData.theme) + } + } else { + headerNode = ListSectionHeaderNode(theme: item.presentationData.theme) + headerNode.title = item.title + strongSelf.addSubnode(headerNode) + strongSelf.headerNode = headerNode + } + + headerNode.frame = CGRect(origin: CGPoint(), size: contentSize) + headerNode.updateLayout(size: contentSize, leftInset: params.leftInset, rightInset: params.rightInset) + } + }) + }) + } + } + + override func animateInsertion(_ currentTimestamp: Double, duration: Double, short: Bool) { + self.layer.animateAlpha(from: 0.0, to: 1.0, duration: duration * 0.5) + } + + override func animateRemoved(_ currentTimestamp: Double, duration: Double) { + self.layer.animateAlpha(from: 1.0, to: 0.0, duration: duration * 0.5, removeOnCompletion: false) + } +} diff --git a/submodules/SettingsUI/Sources/Privacy and Security/PrivacyAndSecurityController.swift b/submodules/SettingsUI/Sources/Privacy and Security/PrivacyAndSecurityController.swift index 0f737c2d71..5e332cc700 100644 --- a/submodules/SettingsUI/Sources/Privacy and Security/PrivacyAndSecurityController.swift +++ b/submodules/SettingsUI/Sources/Privacy and Security/PrivacyAndSecurityController.swift @@ -530,7 +530,7 @@ public func privacyAndSecurityController(context: AccountContext, initialSetting return .single(false) } } else { - return .complete() + return .single(hasTwoStepAuth) } } diff --git a/submodules/SettingsUI/Sources/Search/SettingsSearchItem.swift b/submodules/SettingsUI/Sources/Search/SettingsSearchItem.swift index 7fb91e38c4..9684fc2db0 100644 --- a/submodules/SettingsUI/Sources/Search/SettingsSearchItem.swift +++ b/submodules/SettingsUI/Sources/Search/SettingsSearchItem.swift @@ -78,7 +78,7 @@ final class SettingsSearchItem: ItemListControllerSearch { let exceptionsList: Signal let archivedStickerPacks: Signal<[ArchivedStickerPackItem]?, NoError> let privacySettings: Signal - let hasWallet: Signal + let hasTwoStepAuth: Signal let activeSessionsContext: Signal let webSessionsContext: Signal @@ -86,7 +86,7 @@ final class SettingsSearchItem: ItemListControllerSearch { private var activity: ValuePromise = ValuePromise(ignoreRepeated: false) private let activityDisposable = MetaDisposable() - init(context: AccountContext, theme: PresentationTheme, placeholder: String, activated: Bool, updateActivated: @escaping (Bool) -> Void, presentController: @escaping (ViewController, Any?) -> Void, pushController: @escaping (ViewController) -> Void, getNavigationController: (() -> NavigationController?)?, resolvedFaqUrl: Signal, exceptionsList: Signal, archivedStickerPacks: Signal<[ArchivedStickerPackItem]?, NoError>, privacySettings: Signal, hasWallet: Signal, activeSessionsContext: Signal, webSessionsContext: Signal) { + init(context: AccountContext, theme: PresentationTheme, placeholder: String, activated: Bool, updateActivated: @escaping (Bool) -> Void, presentController: @escaping (ViewController, Any?) -> Void, pushController: @escaping (ViewController) -> Void, getNavigationController: (() -> NavigationController?)?, resolvedFaqUrl: Signal, exceptionsList: Signal, archivedStickerPacks: Signal<[ArchivedStickerPackItem]?, NoError>, privacySettings: Signal, hasTwoStepAuth: Signal, activeSessionsContext: Signal, webSessionsContext: Signal) { self.context = context self.theme = theme self.placeholder = placeholder @@ -99,7 +99,7 @@ final class SettingsSearchItem: ItemListControllerSearch { self.exceptionsList = exceptionsList self.archivedStickerPacks = archivedStickerPacks self.privacySettings = privacySettings - self.hasWallet = hasWallet + self.hasTwoStepAuth = hasTwoStepAuth self.activeSessionsContext = activeSessionsContext self.webSessionsContext = webSessionsContext self.activityDisposable.set((activity.get() |> mapToSignal { value -> Signal in @@ -165,7 +165,7 @@ final class SettingsSearchItem: ItemListControllerSearch { pushController(c) }, presentController: { c, a in presentController(c, a) - }, getNavigationController: self.getNavigationController, resolvedFaqUrl: self.resolvedFaqUrl, exceptionsList: self.exceptionsList, archivedStickerPacks: self.archivedStickerPacks, privacySettings: self.privacySettings, hasWallet: self.hasWallet, activeSessionsContext: self.activeSessionsContext, webSessionsContext: self.webSessionsContext) + }, getNavigationController: self.getNavigationController, resolvedFaqUrl: self.resolvedFaqUrl, exceptionsList: self.exceptionsList, archivedStickerPacks: self.archivedStickerPacks, privacySettings: self.privacySettings, hasTwoStepAuth: self.hasTwoStepAuth, activeSessionsContext: self.activeSessionsContext, webSessionsContext: self.webSessionsContext) } } } @@ -362,7 +362,7 @@ public final class SettingsSearchContainerNode: SearchDisplayControllerContentNo private var presentationDataDisposable: Disposable? private let presentationDataPromise: Promise - public init(context: AccountContext, openResult: @escaping (SettingsSearchableItem) -> Void, resolvedFaqUrl: Signal, exceptionsList: Signal, archivedStickerPacks: Signal<[ArchivedStickerPackItem]?, NoError>, privacySettings: Signal, hasWallet: Signal, activeSessionsContext: Signal, webSessionsContext: Signal) { + public init(context: AccountContext, openResult: @escaping (SettingsSearchableItem) -> Void, resolvedFaqUrl: Signal, exceptionsList: Signal, archivedStickerPacks: Signal<[ArchivedStickerPackItem]?, NoError>, privacySettings: Signal, hasTwoStepAuth: Signal, activeSessionsContext: Signal, webSessionsContext: Signal) { self.presentationData = context.sharedContext.currentPresentationData.with { $0 } self.presentationDataPromise = Promise(self.presentationData) @@ -389,7 +389,7 @@ public final class SettingsSearchContainerNode: SearchDisplayControllerContentNo }) let searchableItems = Promise<[SettingsSearchableItem]>() - searchableItems.set(settingsSearchableItems(context: context, notificationExceptionsList: exceptionsList, archivedStickerPacks: archivedStickerPacks, privacySettings: privacySettings, hasWallet: hasWallet, activeSessionsContext: activeSessionsContext, webSessionsContext: webSessionsContext)) + searchableItems.set(settingsSearchableItems(context: context, notificationExceptionsList: exceptionsList, archivedStickerPacks: archivedStickerPacks, privacySettings: privacySettings, hasTwoStepAuth: hasTwoStepAuth, activeSessionsContext: activeSessionsContext, webSessionsContext: webSessionsContext)) let faqItems = Promise<[SettingsSearchableItem]>() faqItems.set(faqSearchableItems(context: context, resolvedUrl: resolvedFaqUrl, suggestAccountDeletion: false)) @@ -662,13 +662,13 @@ private final class SettingsSearchItemNode: ItemListControllerSearchNode { let exceptionsList: Signal let archivedStickerPacks: Signal<[ArchivedStickerPackItem]?, NoError> let privacySettings: Signal - let hasWallet: Signal + let hasTwoStepAuth: Signal let activeSessionsContext: Signal let webSessionsContext: Signal var cancel: () -> Void - init(context: AccountContext, cancel: @escaping () -> Void, updateActivity: @escaping(Bool) -> Void, pushController: @escaping (ViewController) -> Void, presentController: @escaping (ViewController, Any?) -> Void, getNavigationController: (() -> NavigationController?)?, resolvedFaqUrl: Signal, exceptionsList: Signal, archivedStickerPacks: Signal<[ArchivedStickerPackItem]?, NoError>, privacySettings: Signal, hasWallet: Signal, activeSessionsContext: Signal, webSessionsContext: Signal) { + init(context: AccountContext, cancel: @escaping () -> Void, updateActivity: @escaping(Bool) -> Void, pushController: @escaping (ViewController) -> Void, presentController: @escaping (ViewController, Any?) -> Void, getNavigationController: (() -> NavigationController?)?, resolvedFaqUrl: Signal, exceptionsList: Signal, archivedStickerPacks: Signal<[ArchivedStickerPackItem]?, NoError>, privacySettings: Signal, hasTwoStepAuth: Signal, activeSessionsContext: Signal, webSessionsContext: Signal) { self.context = context self.presentationData = context.sharedContext.currentPresentationData.with { $0 } self.cancel = cancel @@ -679,7 +679,7 @@ private final class SettingsSearchItemNode: ItemListControllerSearchNode { self.exceptionsList = exceptionsList self.archivedStickerPacks = archivedStickerPacks self.privacySettings = privacySettings - self.hasWallet = hasWallet + self.hasTwoStepAuth = hasTwoStepAuth self.activeSessionsContext = activeSessionsContext self.webSessionsContext = webSessionsContext @@ -721,7 +721,7 @@ private final class SettingsSearchItemNode: ItemListControllerSearchNode { } }) } - }, resolvedFaqUrl: self.resolvedFaqUrl, exceptionsList: self.exceptionsList, archivedStickerPacks: self.archivedStickerPacks, privacySettings: self.privacySettings, hasWallet: self.hasWallet, activeSessionsContext: self.activeSessionsContext, webSessionsContext: self.webSessionsContext), cancel: { [weak self] in + }, resolvedFaqUrl: self.resolvedFaqUrl, exceptionsList: self.exceptionsList, archivedStickerPacks: self.archivedStickerPacks, privacySettings: self.privacySettings, hasTwoStepAuth: self.hasTwoStepAuth, activeSessionsContext: self.activeSessionsContext, webSessionsContext: self.webSessionsContext), cancel: { [weak self] in self?.cancel() }) diff --git a/submodules/SettingsUI/Sources/Search/SettingsSearchableItems.swift b/submodules/SettingsUI/Sources/Search/SettingsSearchableItems.swift index c0fea2181e..da8eb5fbe4 100644 --- a/submodules/SettingsUI/Sources/Search/SettingsSearchableItems.swift +++ b/submodules/SettingsUI/Sources/Search/SettingsSearchableItems.swift @@ -729,7 +729,7 @@ private func languageSearchableItems(context: AccountContext, localizations: [Lo return items } -func settingsSearchableItems(context: AccountContext, notificationExceptionsList: Signal, archivedStickerPacks: Signal<[ArchivedStickerPackItem]?, NoError>, privacySettings: Signal, hasWallet: Signal, activeSessionsContext: Signal, webSessionsContext: Signal) -> Signal<[SettingsSearchableItem], NoError> { +func settingsSearchableItems(context: AccountContext, notificationExceptionsList: Signal, archivedStickerPacks: Signal<[ArchivedStickerPackItem]?, NoError>, privacySettings: Signal, hasTwoStepAuth: Signal, activeSessionsContext: Signal, webSessionsContext: Signal) -> Signal<[SettingsSearchableItem], NoError> { let watchAppInstalled = (context.watchManager?.watchAppInstalled ?? .single(false)) |> take(1) @@ -825,8 +825,8 @@ func settingsSearchableItems(context: AccountContext, notificationExceptionsList } } - return combineLatest(watchAppInstalled, canAddAccount, localizations, notificationSettings, notificationExceptionsList, archivedStickerPacks, proxyServers, privacySettings, hasWallet, activeSessionsContext, activeWebSessionsContext) - |> map { watchAppInstalled, canAddAccount, localizations, notificationSettings, notificationExceptionsList, archivedStickerPacks, proxyServers, privacySettings, hasWallet, activeSessionsContext, activeWebSessionsContext in + return combineLatest(watchAppInstalled, canAddAccount, localizations, notificationSettings, notificationExceptionsList, archivedStickerPacks, proxyServers, privacySettings, hasTwoStepAuth, activeSessionsContext, activeWebSessionsContext) + |> map { watchAppInstalled, canAddAccount, localizations, notificationSettings, notificationExceptionsList, archivedStickerPacks, proxyServers, privacySettings, hasTwoStepAuth, activeSessionsContext, activeWebSessionsContext in let strings = context.sharedContext.currentPresentationData.with { $0 }.strings var allItems: [SettingsSearchableItem] = [] @@ -875,21 +875,12 @@ func settingsSearchableItems(context: AccountContext, notificationExceptionsList allItems.append(watch) } - let passport = SettingsSearchableItem(id: .passport(0), title: strings.Settings_Passport, alternate: synonyms(strings.SettingsSearch_Synonyms_Passport), icon: .passport, breadcrumbs: [], present: { context, _, present in - present(.modal, SecureIdAuthController(context: context, mode: .list)) - }) - allItems.append(passport) - - #if ENABLE_WALLET - if hasWallet { - let wallet = SettingsSearchableItem(id: .wallet(0), title: strings.Settings_Wallet, alternate: synonyms(strings.SettingsSearch_Synonyms_Wallet), icon: .wallet, breadcrumbs: [], present: { context, _, present in - context.sharedContext.openWallet(context: context, walletContext: .generic, present: { c in - present(.push, c) - }) + if let hasTwoStepAuth = hasTwoStepAuth, hasTwoStepAuth { + let passport = SettingsSearchableItem(id: .passport(0), title: strings.Settings_Passport, alternate: synonyms(strings.SettingsSearch_Synonyms_Passport), icon: .passport, breadcrumbs: [], present: { context, _, present in + present(.modal, SecureIdAuthController(context: context, mode: .list)) }) - allItems.append(wallet) + allItems.append(passport) } - #endif let support = SettingsSearchableItem(id: .support(0), title: strings.Settings_Support, alternate: synonyms(strings.SettingsSearch_Synonyms_Support), icon: .support, breadcrumbs: [], present: { context, _, present in let _ = (supportPeerId(account: context.account) diff --git a/submodules/ShareController/Sources/ShareController.swift b/submodules/ShareController/Sources/ShareController.swift index 0250e1680e..2d56f4729b 100644 --- a/submodules/ShareController/Sources/ShareController.swift +++ b/submodules/ShareController/Sources/ShareController.swift @@ -135,6 +135,8 @@ private func collectExternalShareItems(strings: PresentationStrings, dateTimeFor fileName = value } else if file.isVideo { fileName = "telegram_video.mp4" + } else if file.isVoice { + fileName = "telegram_audio.ogg" } else { fileName = "file" } diff --git a/submodules/StickerPackPreviewUI/Sources/StickerPackPreviewControllerNode.swift b/submodules/StickerPackPreviewUI/Sources/StickerPackPreviewControllerNode.swift index a7b3bfebb2..115d019f49 100644 --- a/submodules/StickerPackPreviewUI/Sources/StickerPackPreviewControllerNode.swift +++ b/submodules/StickerPackPreviewUI/Sources/StickerPackPreviewControllerNode.swift @@ -406,9 +406,9 @@ final class StickerPackPreviewControllerNode: ViewControllerTracingNode, UIScrol let minimallyRevealedRowCount: CGFloat = 3.5 let initiallyRevealedRowCount = min(minimallyRevealedRowCount, CGFloat(rowCount)) - let topInset = max(0.0, contentFrame.size.height - initiallyRevealedRowCount * itemWidth - titleAreaHeight - buttonHeight) let bottomGridInset = hasShareButton ? buttonHeight * 2.0 : buttonHeight - + let topInset = max(0.0, contentFrame.size.height - initiallyRevealedRowCount * itemWidth - titleAreaHeight - bottomGridInset) + transition.updateFrame(node: self.contentContainerNode, frame: contentContainerFrame) if let activityIndicator = self.activityIndicator { diff --git a/submodules/SyncCore/Sources/ExportedInvitation.swift b/submodules/SyncCore/Sources/ExportedInvitation.swift index 41585680b2..2eab693250 100644 --- a/submodules/SyncCore/Sources/ExportedInvitation.swift +++ b/submodules/SyncCore/Sources/ExportedInvitation.swift @@ -6,16 +6,18 @@ public struct ExportedInvitation: PostboxCoding, Equatable { public let isRevoked: Bool public let adminId: PeerId public let date: Int32 + public let startDate: Int32? public let expireDate: Int32? public let usageLimit: Int32? public let count: Int32? - public init(link: String, isPermanent: Bool, isRevoked: Bool, adminId: PeerId, date: Int32, expireDate: Int32?, usageLimit: Int32?, count: Int32?) { + public init(link: String, isPermanent: Bool, isRevoked: Bool, adminId: PeerId, date: Int32, startDate: Int32?, expireDate: Int32?, usageLimit: Int32?, count: Int32?) { self.link = link self.isPermanent = isPermanent self.isRevoked = isRevoked self.adminId = adminId self.date = date + self.startDate = startDate self.expireDate = expireDate self.usageLimit = usageLimit self.count = count @@ -27,6 +29,7 @@ public struct ExportedInvitation: PostboxCoding, Equatable { self.isRevoked = decoder.decodeBoolForKey("revoked", orElse: false) self.adminId = PeerId(decoder.decodeInt64ForKey("adminId", orElse: 0)) self.date = decoder.decodeInt32ForKey("date", orElse: 0) + self.startDate = decoder.decodeOptionalInt32ForKey("startDate") self.expireDate = decoder.decodeOptionalInt32ForKey("expireDate") self.usageLimit = decoder.decodeOptionalInt32ForKey("usageLimit") self.count = decoder.decodeOptionalInt32ForKey("count") @@ -38,6 +41,11 @@ public struct ExportedInvitation: PostboxCoding, Equatable { encoder.encodeBool(self.isRevoked, forKey: "revoked") encoder.encodeInt64(self.adminId.toInt64(), forKey: "adminId") encoder.encodeInt32(self.date, forKey: "date") + if let startDate = self.startDate { + encoder.encodeInt32(startDate, forKey: "startDate") + } else { + encoder.encodeNil(forKey: "startDate") + } if let expireDate = self.expireDate { encoder.encodeInt32(expireDate, forKey: "expireDate") } else { @@ -56,6 +64,6 @@ public struct ExportedInvitation: PostboxCoding, Equatable { } public static func ==(lhs: ExportedInvitation, rhs: ExportedInvitation) -> Bool { - return lhs.link == rhs.link && lhs.isPermanent == rhs.isPermanent && lhs.isRevoked == rhs.isRevoked && lhs.adminId == rhs.adminId && lhs.date == rhs.date && lhs.expireDate == rhs.expireDate && lhs.usageLimit == rhs.usageLimit && lhs.count == rhs.count + return lhs.link == rhs.link && lhs.isPermanent == rhs.isPermanent && lhs.isRevoked == rhs.isRevoked && lhs.adminId == rhs.adminId && lhs.date == rhs.date && lhs.startDate == rhs.startDate && lhs.expireDate == rhs.expireDate && lhs.usageLimit == rhs.usageLimit && lhs.count == rhs.count } } diff --git a/submodules/SyncCore/Sources/TelegramChatAdminRights.swift b/submodules/SyncCore/Sources/TelegramChatAdminRights.swift index d0d18e3648..4d5050cb3e 100644 --- a/submodules/SyncCore/Sources/TelegramChatAdminRights.swift +++ b/submodules/SyncCore/Sources/TelegramChatAdminRights.swift @@ -22,7 +22,7 @@ public struct TelegramChatAdminRightsFlags: OptionSet { public static let canBeAnonymous = TelegramChatAdminRightsFlags(rawValue: 1 << 10) public static let canManageCalls = TelegramChatAdminRightsFlags(rawValue: 1 << 11) - public static var all:TelegramChatAdminRightsFlags { + public static var all: TelegramChatAdminRightsFlags { return [.canChangeInfo, .canPostMessages, .canEditMessages, .canDeleteMessages, .canBanUsers, .canInviteUsers, .canPinMessages, .canAddAdmins, .canBeAnonymous, .canManageCalls] } @@ -34,8 +34,7 @@ public struct TelegramChatAdminRightsFlags: OptionSet { .canPinMessages, .canManageCalls, .canBeAnonymous, - .canAddAdmins, - .canManageCalls + .canAddAdmins ] public static var broadcastSpecific: TelegramChatAdminRightsFlags = [ diff --git a/submodules/TelegramApi/Sources/Api0.swift b/submodules/TelegramApi/Sources/Api0.swift index 9bb8a4b1a7..8a960cff3d 100644 --- a/submodules/TelegramApi/Sources/Api0.swift +++ b/submodules/TelegramApi/Sources/Api0.swift @@ -737,7 +737,7 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[-1634752813] = { return Api.messages.FavedStickers.parse_favedStickersNotModified($0) } dict[-209768682] = { return Api.messages.FavedStickers.parse_favedStickers($0) } dict[1776236393] = { return Api.ExportedChatInvite.parse_chatInviteEmpty($0) } - dict[-1448589334] = { return Api.ExportedChatInvite.parse_chatInviteExported($0) } + dict[1847917725] = { return Api.ExportedChatInvite.parse_chatInviteExported($0) } dict[-1389486888] = { return Api.account.AuthorizationForm.parse_authorizationForm($0) } dict[-1392388579] = { return Api.Authorization.parse_authorization($0) } dict[-1361650766] = { return Api.MaskCoords.parse_maskCoords($0) } @@ -869,8 +869,8 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[-1417756512] = { return Api.EncryptedChat.parse_encryptedChatEmpty($0) } dict[1006044124] = { return Api.EncryptedChat.parse_encryptedChatWaiting($0) } dict[-94974410] = { return Api.EncryptedChat.parse_encryptedChat($0) } - dict[332848423] = { return Api.EncryptedChat.parse_encryptedChatDiscarded($0) } dict[1651608194] = { return Api.EncryptedChat.parse_encryptedChatRequested($0) } + dict[505183301] = { return Api.EncryptedChat.parse_encryptedChatDiscarded($0) } dict[-901375139] = { return Api.PeerLocated.parse_peerLocated($0) } dict[-118740917] = { return Api.PeerLocated.parse_peerSelfLocated($0) } dict[922273905] = { return Api.Document.parse_documentEmpty($0) } diff --git a/submodules/TelegramApi/Sources/Api1.swift b/submodules/TelegramApi/Sources/Api1.swift index f224f02325..144a3781bb 100644 --- a/submodules/TelegramApi/Sources/Api1.swift +++ b/submodules/TelegramApi/Sources/Api1.swift @@ -20979,7 +20979,7 @@ public extension Api { } public enum ExportedChatInvite: TypeConstructorDescription { case chatInviteEmpty - case chatInviteExported(flags: Int32, link: String, adminId: Int32, date: Int32, expireDate: Int32?, usageLimit: Int32?, usage: Int32?) + case chatInviteExported(flags: Int32, link: String, adminId: Int32, date: Int32, startDate: Int32?, expireDate: Int32?, usageLimit: Int32?, usage: Int32?) public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { switch self { @@ -20989,14 +20989,15 @@ public extension Api { } break - case .chatInviteExported(let flags, let link, let adminId, let date, let expireDate, let usageLimit, let usage): + case .chatInviteExported(let flags, let link, let adminId, let date, let startDate, let expireDate, let usageLimit, let usage): if boxed { - buffer.appendInt32(-1448589334) + buffer.appendInt32(1847917725) } serializeInt32(flags, buffer: buffer, boxed: false) serializeString(link, buffer: buffer, boxed: false) serializeInt32(adminId, buffer: buffer, boxed: false) serializeInt32(date, buffer: buffer, boxed: false) + if Int(flags) & Int(1 << 4) != 0 {serializeInt32(startDate!, buffer: buffer, boxed: false)} if Int(flags) & Int(1 << 1) != 0 {serializeInt32(expireDate!, buffer: buffer, boxed: false)} if Int(flags) & Int(1 << 2) != 0 {serializeInt32(usageLimit!, buffer: buffer, boxed: false)} if Int(flags) & Int(1 << 3) != 0 {serializeInt32(usage!, buffer: buffer, boxed: false)} @@ -21008,8 +21009,8 @@ public extension Api { switch self { case .chatInviteEmpty: return ("chatInviteEmpty", []) - case .chatInviteExported(let flags, let link, let adminId, let date, let expireDate, let usageLimit, let usage): - return ("chatInviteExported", [("flags", flags), ("link", link), ("adminId", adminId), ("date", date), ("expireDate", expireDate), ("usageLimit", usageLimit), ("usage", usage)]) + case .chatInviteExported(let flags, let link, let adminId, let date, let startDate, let expireDate, let usageLimit, let usage): + return ("chatInviteExported", [("flags", flags), ("link", link), ("adminId", adminId), ("date", date), ("startDate", startDate), ("expireDate", expireDate), ("usageLimit", usageLimit), ("usage", usage)]) } } @@ -21026,20 +21027,23 @@ public extension Api { var _4: Int32? _4 = reader.readInt32() var _5: Int32? - if Int(_1!) & Int(1 << 1) != 0 {_5 = reader.readInt32() } + if Int(_1!) & Int(1 << 4) != 0 {_5 = reader.readInt32() } var _6: Int32? - if Int(_1!) & Int(1 << 2) != 0 {_6 = reader.readInt32() } + if Int(_1!) & Int(1 << 1) != 0 {_6 = reader.readInt32() } var _7: Int32? - if Int(_1!) & Int(1 << 3) != 0 {_7 = reader.readInt32() } + if Int(_1!) & Int(1 << 2) != 0 {_7 = reader.readInt32() } + var _8: Int32? + if Int(_1!) & Int(1 << 3) != 0 {_8 = reader.readInt32() } let _c1 = _1 != nil let _c2 = _2 != nil let _c3 = _3 != nil let _c4 = _4 != nil - let _c5 = (Int(_1!) & Int(1 << 1) == 0) || _5 != nil - let _c6 = (Int(_1!) & Int(1 << 2) == 0) || _6 != nil - let _c7 = (Int(_1!) & Int(1 << 3) == 0) || _7 != nil - if _c1 && _c2 && _c3 && _c4 && _c5 && _c6 && _c7 { - return Api.ExportedChatInvite.chatInviteExported(flags: _1!, link: _2!, adminId: _3!, date: _4!, expireDate: _5, usageLimit: _6, usage: _7) + let _c5 = (Int(_1!) & Int(1 << 4) == 0) || _5 != nil + let _c6 = (Int(_1!) & Int(1 << 1) == 0) || _6 != nil + let _c7 = (Int(_1!) & Int(1 << 2) == 0) || _7 != nil + let _c8 = (Int(_1!) & Int(1 << 3) == 0) || _8 != nil + if _c1 && _c2 && _c3 && _c4 && _c5 && _c6 && _c7 && _c8 { + return Api.ExportedChatInvite.chatInviteExported(flags: _1!, link: _2!, adminId: _3!, date: _4!, startDate: _5, expireDate: _6, usageLimit: _7, usage: _8) } else { return nil @@ -24065,8 +24069,8 @@ public extension Api { case encryptedChatEmpty(id: Int32) case encryptedChatWaiting(id: Int32, accessHash: Int64, date: Int32, adminId: Int32, participantId: Int32) case encryptedChat(id: Int32, accessHash: Int64, date: Int32, adminId: Int32, participantId: Int32, gAOrB: Buffer, keyFingerprint: Int64) - case encryptedChatDiscarded(id: Int32) case encryptedChatRequested(flags: Int32, folderId: Int32?, id: Int32, accessHash: Int64, date: Int32, adminId: Int32, participantId: Int32, gA: Buffer) + case encryptedChatDiscarded(flags: Int32, id: Int32) public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { switch self { @@ -24098,12 +24102,6 @@ public extension Api { serializeBytes(gAOrB, buffer: buffer, boxed: false) serializeInt64(keyFingerprint, buffer: buffer, boxed: false) break - case .encryptedChatDiscarded(let id): - if boxed { - buffer.appendInt32(332848423) - } - serializeInt32(id, buffer: buffer, boxed: false) - break case .encryptedChatRequested(let flags, let folderId, let id, let accessHash, let date, let adminId, let participantId, let gA): if boxed { buffer.appendInt32(1651608194) @@ -24117,6 +24115,13 @@ public extension Api { serializeInt32(participantId, buffer: buffer, boxed: false) serializeBytes(gA, buffer: buffer, boxed: false) break + case .encryptedChatDiscarded(let flags, let id): + if boxed { + buffer.appendInt32(505183301) + } + serializeInt32(flags, buffer: buffer, boxed: false) + serializeInt32(id, buffer: buffer, boxed: false) + break } } @@ -24128,10 +24133,10 @@ public extension Api { return ("encryptedChatWaiting", [("id", id), ("accessHash", accessHash), ("date", date), ("adminId", adminId), ("participantId", participantId)]) case .encryptedChat(let id, let accessHash, let date, let adminId, let participantId, let gAOrB, let keyFingerprint): return ("encryptedChat", [("id", id), ("accessHash", accessHash), ("date", date), ("adminId", adminId), ("participantId", participantId), ("gAOrB", gAOrB), ("keyFingerprint", keyFingerprint)]) - case .encryptedChatDiscarded(let id): - return ("encryptedChatDiscarded", [("id", id)]) case .encryptedChatRequested(let flags, let folderId, let id, let accessHash, let date, let adminId, let participantId, let gA): return ("encryptedChatRequested", [("flags", flags), ("folderId", folderId), ("id", id), ("accessHash", accessHash), ("date", date), ("adminId", adminId), ("participantId", participantId), ("gA", gA)]) + case .encryptedChatDiscarded(let flags, let id): + return ("encryptedChatDiscarded", [("flags", flags), ("id", id)]) } } @@ -24198,17 +24203,6 @@ public extension Api { return nil } } - public static func parse_encryptedChatDiscarded(_ reader: BufferReader) -> EncryptedChat? { - var _1: Int32? - _1 = reader.readInt32() - let _c1 = _1 != nil - if _c1 { - return Api.EncryptedChat.encryptedChatDiscarded(id: _1!) - } - else { - return nil - } - } public static func parse_encryptedChatRequested(_ reader: BufferReader) -> EncryptedChat? { var _1: Int32? _1 = reader.readInt32() @@ -24241,6 +24235,20 @@ public extension Api { return nil } } + public static func parse_encryptedChatDiscarded(_ reader: BufferReader) -> EncryptedChat? { + var _1: Int32? + _1 = reader.readInt32() + var _2: Int32? + _2 = reader.readInt32() + let _c1 = _1 != nil + let _c2 = _2 != nil + if _c1 && _c2 { + return Api.EncryptedChat.encryptedChatDiscarded(flags: _1!, id: _2!) + } + else { + return nil + } + } } public enum PeerLocated: TypeConstructorDescription { diff --git a/submodules/TelegramApi/Sources/Api3.swift b/submodules/TelegramApi/Sources/Api3.swift index 26b2b9f5a1..84cb1abb7e 100644 --- a/submodules/TelegramApi/Sources/Api3.swift +++ b/submodules/TelegramApi/Sources/Api3.swift @@ -2109,20 +2109,6 @@ public extension Api { }) } - public static func discardEncryption(chatId: Int32) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { - let buffer = Buffer() - buffer.appendInt32(-304536635) - serializeInt32(chatId, buffer: buffer, boxed: false) - return (FunctionDescription(name: "messages.discardEncryption", parameters: [("chatId", chatId)]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.Bool? in - let reader = BufferReader(buffer) - var result: Api.Bool? - if let signature = reader.readInt32() { - result = Api.parse(reader, signature: signature) as? Api.Bool - } - return result - }) - } - public static func setEncryptedTyping(peer: Api.InputEncryptedChat, typing: Api.Bool) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { let buffer = Buffer() buffer.appendInt32(2031374829) @@ -3903,24 +3889,6 @@ public extension Api { }) } - public static func getExportedChatInvites(flags: Int32, peer: Api.InputPeer, adminId: Api.InputUser?, offsetLink: String?, limit: Int32) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { - let buffer = Buffer() - buffer.appendInt32(1838984707) - serializeInt32(flags, buffer: buffer, boxed: false) - peer.serialize(buffer, true) - if Int(flags) & Int(1 << 0) != 0 {adminId!.serialize(buffer, true)} - if Int(flags) & Int(1 << 2) != 0 {serializeString(offsetLink!, buffer: buffer, boxed: false)} - serializeInt32(limit, buffer: buffer, boxed: false) - return (FunctionDescription(name: "messages.getExportedChatInvites", parameters: [("flags", flags), ("peer", peer), ("adminId", adminId), ("offsetLink", offsetLink), ("limit", limit)]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.messages.ExportedChatInvites? in - let reader = BufferReader(buffer) - var result: Api.messages.ExportedChatInvites? - if let signature = reader.readInt32() { - result = Api.parse(reader, signature: signature) as? Api.messages.ExportedChatInvites - } - return result - }) - } - public static func exportChatInvite(flags: Int32, peer: Api.InputPeer, expireDate: Int32?, usageLimit: Int32?) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { let buffer = Buffer() buffer.appendInt32(347716823) @@ -3973,6 +3941,96 @@ public extension Api { return result }) } + + public static func getExportedChatInvites(flags: Int32, peer: Api.InputPeer, adminId: Api.InputUser?, offsetLink: String?, limit: Int32) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(1838984707) + serializeInt32(flags, buffer: buffer, boxed: false) + peer.serialize(buffer, true) + if Int(flags) & Int(1 << 0) != 0 {adminId!.serialize(buffer, true)} + if Int(flags) & Int(1 << 2) != 0 {serializeString(offsetLink!, buffer: buffer, boxed: false)} + serializeInt32(limit, buffer: buffer, boxed: false) + return (FunctionDescription(name: "messages.getExportedChatInvites", parameters: [("flags", flags), ("peer", peer), ("adminId", adminId), ("offsetLink", offsetLink), ("limit", limit)]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.messages.ExportedChatInvites? in + let reader = BufferReader(buffer) + var result: Api.messages.ExportedChatInvites? + if let signature = reader.readInt32() { + result = Api.parse(reader, signature: signature) as? Api.messages.ExportedChatInvites + } + return result + }) + } + + public static func deleteRevokedExportedChatInvites(peer: Api.InputPeer) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(1375999075) + peer.serialize(buffer, true) + return (FunctionDescription(name: "messages.deleteRevokedExportedChatInvites", parameters: [("peer", peer)]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.Bool? in + let reader = BufferReader(buffer) + var result: Api.Bool? + if let signature = reader.readInt32() { + result = Api.parse(reader, signature: signature) as? Api.Bool + } + return result + }) + } + + public static func deleteExportedChatInvite(peer: Api.InputPeer, link: String) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(-731601877) + peer.serialize(buffer, true) + serializeString(link, buffer: buffer, boxed: false) + return (FunctionDescription(name: "messages.deleteExportedChatInvite", parameters: [("peer", peer), ("link", link)]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.Bool? in + let reader = BufferReader(buffer) + var result: Api.Bool? + if let signature = reader.readInt32() { + result = Api.parse(reader, signature: signature) as? Api.Bool + } + return result + }) + } + + public static func discardEncryption(flags: Int32, chatId: Int32) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(-208425312) + serializeInt32(flags, buffer: buffer, boxed: false) + serializeInt32(chatId, buffer: buffer, boxed: false) + return (FunctionDescription(name: "messages.discardEncryption", parameters: [("flags", flags), ("chatId", chatId)]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.Bool? in + let reader = BufferReader(buffer) + var result: Api.Bool? + if let signature = reader.readInt32() { + result = Api.parse(reader, signature: signature) as? Api.Bool + } + return result + }) + } + + public static func deleteChat(chatId: Int32) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(-2094760687) + serializeInt32(chatId, buffer: buffer, boxed: false) + return (FunctionDescription(name: "messages.deleteChat", parameters: [("chatId", chatId)]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.Bool? in + let reader = BufferReader(buffer) + var result: Api.Bool? + if let signature = reader.readInt32() { + result = Api.parse(reader, signature: signature) as? Api.Bool + } + return result + }) + } + + public static func deletePhoneCallHistory(flags: Int32) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(1828657989) + serializeInt32(flags, buffer: buffer, boxed: false) + return (FunctionDescription(name: "messages.deletePhoneCallHistory", parameters: [("flags", flags)]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.messages.AffectedHistory? in + let reader = BufferReader(buffer) + var result: Api.messages.AffectedHistory? + if let signature = reader.readInt32() { + result = Api.parse(reader, signature: signature) as? Api.messages.AffectedHistory + } + return result + }) + } } public struct channels { public static func readHistory(channel: Api.InputChannel, maxId: Int32) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { diff --git a/submodules/TelegramBaseController/Sources/MediaNavigationAccessoryHeaderNode.swift b/submodules/TelegramBaseController/Sources/MediaNavigationAccessoryHeaderNode.swift index 3be5e77363..31aab50b23 100644 --- a/submodules/TelegramBaseController/Sources/MediaNavigationAccessoryHeaderNode.swift +++ b/submodules/TelegramBaseController/Sources/MediaNavigationAccessoryHeaderNode.swift @@ -398,10 +398,16 @@ public final class MediaNavigationAccessoryHeaderNode: ASDisplayNode, UIScrollVi if scrollView.isDecelerating { self.changeTrack() } + + self.rateButton.alpha = 0.0 + self.rateButton.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.2) } public func scrollViewDidEndDecelerating(_ scrollView: UIScrollView) { self.changeTrack() + + self.rateButton.alpha = 1.0 + self.rateButton.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.2) } public func scrollViewDidEndDragging(_ scrollView: UIScrollView, willDecelerate decelerate: Bool) { @@ -409,6 +415,9 @@ public final class MediaNavigationAccessoryHeaderNode: ASDisplayNode, UIScrollVi return } self.changeTrack() + + self.rateButton.alpha = 1.0 + self.rateButton.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.2) } private func changeTrack() { @@ -457,9 +466,11 @@ public final class MediaNavigationAccessoryHeaderNode: ASDisplayNode, UIScrollVi self.rightMaskNode.transform = CATransform3DMakeScale(-1.0, 1.0, 1.0) self.rightMaskNode.frame = CGRect(x: size.width - inset - 12.0, y: 0.0, width: 12.0, height: minHeight) - self.scrollNode.view.contentSize = contentSize - self.scrollNode.view.contentOffset = CGPoint(x: contentOffset, y: 0.0) - self.initialContentOffset = contentOffset + if !self.scrollNode.view.isTracking && !self.scrollNode.view.isTracking { + self.scrollNode.view.contentSize = contentSize + self.scrollNode.view.contentOffset = CGPoint(x: contentOffset, y: 0.0) + self.initialContentOffset = contentOffset + } let bounds = CGRect(origin: CGPoint(), size: size) let closeButtonSize = self.closeButton.measure(CGSize(width: 100.0, height: 100.0)) diff --git a/submodules/TelegramCallsUI/Sources/VoiceChatActionButton.swift b/submodules/TelegramCallsUI/Sources/VoiceChatActionButton.swift index edb9183ef6..98300a09a4 100644 --- a/submodules/TelegramCallsUI/Sources/VoiceChatActionButton.swift +++ b/submodules/TelegramCallsUI/Sources/VoiceChatActionButton.swift @@ -483,7 +483,8 @@ private final class VoiceChatActionButtonBackgroundNode: ASDisplayNode { self.growingForegroundCircleLayer.isHidden = true self.foregroundGradientLayer.type = .radial - self.foregroundGradientLayer.colors = [lightBlue.cgColor, blue.cgColor] + self.foregroundGradientLayer.colors = [lightBlue.cgColor, blue.cgColor, blue.cgColor] + self.foregroundGradientLayer.locations = [0.0, 0.85, 1.0] self.foregroundGradientLayer.startPoint = CGPoint(x: 1.0, y: 0.0) self.foregroundGradientLayer.endPoint = CGPoint(x: 0.0, y: 1.0) @@ -655,16 +656,16 @@ private final class VoiceChatActionButtonBackgroundNode: ASDisplayNode { let targetScale: CGFloat if let active = active { if active { - targetColors = [activeBlue.cgColor, green.cgColor] + targetColors = [activeBlue.cgColor, green.cgColor, green.cgColor] targetScale = 0.89 outerColor = UIColor(rgb: 0x21674f) } else { - targetColors = [lightBlue.cgColor, blue.cgColor] + targetColors = [lightBlue.cgColor, blue.cgColor, blue.cgColor] targetScale = 0.85 outerColor = UIColor(rgb: 0x1d588d) } } else { - targetColors = [lightBlue.cgColor, blue.cgColor] + targetColors = [lightBlue.cgColor, blue.cgColor, blue.cgColor] targetScale = 0.3 outerColor = nil } diff --git a/submodules/TelegramCallsUI/Sources/VoiceChatController.swift b/submodules/TelegramCallsUI/Sources/VoiceChatController.swift index 06527112c8..548c25ce15 100644 --- a/submodules/TelegramCallsUI/Sources/VoiceChatController.swift +++ b/submodules/TelegramCallsUI/Sources/VoiceChatController.swift @@ -1035,6 +1035,14 @@ public final class VoiceChatController: ViewController { let subtitle = strongSelf.presentationData.strings.VoiceChat_Panel_Members(Int32(max(1, callMembers?.totalCount ?? 0))) strongSelf.currentSubtitle = subtitle + if let callState = strongSelf.callState, callState.canManageCall { + strongSelf.optionsButton.isUserInteractionEnabled = true + strongSelf.optionsButton.alpha = 1.0 + } else { + strongSelf.optionsButton.isUserInteractionEnabled = false + strongSelf.optionsButton.alpha = 0.0 + } + if let (layout, navigationHeight) = strongSelf.validLayout { strongSelf.containerLayoutUpdated(layout, navigationHeight: navigationHeight, transition: .immediate) } @@ -1074,27 +1082,6 @@ public final class VoiceChatController: ViewController { strongSelf.didSetDataReady = true strongSelf.controller?.dataReady.set(true) } - - if let peer = peerViewMainPeer(view) { - if let channel = peer as? TelegramChannel { - if channel.hasPermission(.manageCalls) { - strongSelf.optionsButton.isUserInteractionEnabled = true - strongSelf.optionsButton.alpha = 1.0 - } else { - strongSelf.optionsButton.isUserInteractionEnabled = false - strongSelf.optionsButton.alpha = 0.0 - } - } else if let group = peer as? TelegramGroup { - switch group.role { - case .creator, .admin: - strongSelf.optionsButton.isUserInteractionEnabled = true - strongSelf.optionsButton.alpha = 1.0 - default: - strongSelf.optionsButton.isUserInteractionEnabled = false - strongSelf.optionsButton.alpha = 0.0 - } - } - } }) self.audioOutputStateDisposable = (call.audioOutputState @@ -1142,7 +1129,7 @@ public final class VoiceChatController: ViewController { self.audioOutputNode.addTarget(self, action: #selector(self.audioOutputPressed), forControlEvents: .touchUpInside) - self.optionsButton.contextAction = { [weak self, weak optionsButton] sourceNode, gesture in + self.optionsButton.contextAction = { [weak self] sourceNode, gesture in guard let strongSelf = self, let controller = strongSelf.controller else { return } @@ -1181,16 +1168,7 @@ public final class VoiceChatController: ViewController { strongSelf.call.updateDefaultParticipantsAreMuted(isMuted: true) }))) } - - if !items.isEmpty { - items.append(.separator) - } - - items.append(.custom(VoiceChatRecordingContextItem(timestamp: CFAbsoluteTimeGetCurrent(), action: { (_, f) in - f(.dismissWithoutContent) - - }), false)) - + if !items.isEmpty { items.append(.separator) } @@ -1225,6 +1203,10 @@ public final class VoiceChatController: ViewController { }))) } + if items.isEmpty { + return + } + let optionsButton: VoiceChatHeaderButton if !strongSelf.recButton.isHidden { optionsButton = strongSelf.recButton diff --git a/submodules/TelegramCore/Sources/ExportedInvitation.swift b/submodules/TelegramCore/Sources/ExportedInvitation.swift index fcc085bafd..013b2692e1 100644 --- a/submodules/TelegramCore/Sources/ExportedInvitation.swift +++ b/submodules/TelegramCore/Sources/ExportedInvitation.swift @@ -9,8 +9,8 @@ extension ExportedInvitation { switch apiExportedInvite { case .chatInviteEmpty: return nil - case let .chatInviteExported(flags, link, adminId, date, expireDate, usageLimit, usage): - self = ExportedInvitation(link: link, isPermanent: (flags & (1 << 5)) != 0, isRevoked: (flags & (1 << 0)) != 0, adminId: PeerId(namespace: Namespaces.Peer.CloudUser, id: adminId), date: date, expireDate: expireDate, usageLimit: usageLimit, count: usage) + case let .chatInviteExported(flags, link, adminId, date, startDate, expireDate, usageLimit, usage): + self = ExportedInvitation(link: link, isPermanent: (flags & (1 << 5)) != 0, isRevoked: (flags & (1 << 0)) != 0, adminId: PeerId(namespace: Namespaces.Peer.CloudUser, id: adminId), date: date, startDate: startDate, expireDate: expireDate, usageLimit: usageLimit, count: usage) } } } diff --git a/submodules/TelegramCore/Sources/InvitationLinks.swift b/submodules/TelegramCore/Sources/InvitationLinks.swift index 760780660e..04a9c3d1d9 100644 --- a/submodules/TelegramCore/Sources/InvitationLinks.swift +++ b/submodules/TelegramCore/Sources/InvitationLinks.swift @@ -9,7 +9,7 @@ import SyncCore public func ensuredExistingPeerExportedInvitation(account: Account, peerId: PeerId, revokeExisted: Bool = false) -> Signal { return account.postbox.transaction { transaction -> Signal in if let peer = transaction.getPeer(peerId), let inputPeer = apiInputPeer(peer) { - var flags: Int32 = (1 << 2) + let flags: Int32 = (1 << 2) if let _ = peer as? TelegramChannel { if let cachedData = transaction.getPeerCachedData(peerId: peerId) as? CachedChannelData, cachedData.exportedInvitation != nil && !revokeExisted { return .complete() @@ -90,17 +90,26 @@ public func createPeerExportedInvitation(account: Account, peerId: PeerId, expir } |> switchToLatest } -public func peerExportedInvitations(account: Account, peerId: PeerId) -> Signal<[ExportedInvitation]?, NoError> { - return account.postbox.transaction { transaction -> Signal<[ExportedInvitation]?, NoError> in +public struct ExportedInvitations : Equatable { + public let list: [ExportedInvitation]? + public let totalCount: Int32 +} + +public func peerExportedInvitations(account: Account, peerId: PeerId, revoked: Bool, offsetLink: String? = nil) -> Signal { + return account.postbox.transaction { transaction -> Signal in if let peer = transaction.getPeer(peerId), let inputPeer = apiInputPeer(peer) { - return account.network.request(Api.functions.messages.getExportedChatInvites(flags: 0, peer: inputPeer, adminId: nil, offsetLink: nil, limit: 100)) + var flags: Int32 = 0 + if revoked { + flags |= (1 << 3) + } + return account.network.request(Api.functions.messages.getExportedChatInvites(flags: flags, peer: inputPeer, adminId: nil, offsetLink: nil, limit: 50)) |> map(Optional.init) |> `catch` { _ -> Signal in return .single(nil) } - |> mapToSignal { result -> Signal<[ExportedInvitation]?, NoError> in - return account.postbox.transaction { transaction -> [ExportedInvitation]? in - if let result = result, case let .exportedChatInvites(_, apiInvites, users) = result { + |> mapToSignal { result -> Signal in + return account.postbox.transaction { transaction -> ExportedInvitations? in + if let result = result, case let .exportedChatInvites(count, apiInvites, users) = result { var peers: [Peer] = [] var peersMap: [PeerId: Peer] = [:] for user in users { @@ -118,7 +127,7 @@ public func peerExportedInvitations(account: Account, peerId: PeerId) -> Signal< invites.append(invite) } } - return invites + return ExportedInvitations(list: invites, totalCount: count) } else { return nil } @@ -146,20 +155,22 @@ public func editPeerExportedInvitation(account: Account, peerId: PeerId, link: S } return account.network.request(Api.functions.messages.editExportedChatInvite(flags: flags, peer: inputPeer, link: link, expireDate: expireDate, usageLimit: usageLimit)) |> mapError { _ in return EditPeerExportedInvitationError.generic } - |> map { result -> ExportedInvitation? in - if case let .exportedChatInvite(invite, users) = result { - var peers: [Peer] = [] - for user in users { - let telegramUser = TelegramUser(user: user) - peers.append(telegramUser) + |> mapToSignal { result -> Signal in + return account.postbox.transaction { transaction in + if case let .exportedChatInvite(invite, users) = result { + var peers: [Peer] = [] + for user in users { + let telegramUser = TelegramUser(user: user) + peers.append(telegramUser) + } + updatePeers(transaction: transaction, peers: peers, update: { _, updated -> Peer in + return updated + }) + return ExportedInvitation(apiExportedInvite: invite) + } else { + return nil } - updatePeers(transaction: transaction, peers: peers, update: { _, updated -> Peer in - return updated - }) - return ExportedInvitation(apiExportedInvite: invite) - } else { - return nil - } + } |> mapError { _ in .generic } } } else { return .complete() @@ -173,13 +184,29 @@ public enum RevokePeerExportedInvitationError { case generic } -public func revokePeerExportedInvitation(account: Account, peerId: PeerId, link: String) -> Signal { - return account.postbox.transaction { transaction -> Signal in +public func revokePeerExportedInvitation(account: Account, peerId: PeerId, link: String) -> Signal { + return account.postbox.transaction { transaction -> Signal in if let peer = transaction.getPeer(peerId), let inputPeer = apiInputPeer(peer) { let flags: Int32 = (1 << 2) return account.network.request(Api.functions.messages.editExportedChatInvite(flags: flags, peer: inputPeer, link: link, expireDate: nil, usageLimit: nil)) |> mapError { _ in return RevokePeerExportedInvitationError.generic } - |> ignoreValues + |> mapToSignal { result -> Signal in + return account.postbox.transaction { transaction in + if case let .exportedChatInvite(invite, users) = result { + var peers: [Peer] = [] + for user in users { + let telegramUser = TelegramUser(user: user) + peers.append(telegramUser) + } + updatePeers(transaction: transaction, peers: peers, update: { _, updated -> Peer in + return updated + }) + return ExportedInvitation(apiExportedInvite: invite) + } else { + return nil + } + } |> mapError { _ in .generic } + } } else { return .complete() } @@ -188,6 +215,39 @@ public func revokePeerExportedInvitation(account: Account, peerId: PeerId, link: |> switchToLatest } +public enum DeletePeerExportedInvitationError { + case generic +} + +public func deletePeerExportedInvitation(account: Account, peerId: PeerId, link: String) -> Signal { + return account.postbox.transaction { transaction -> Signal in + if let peer = transaction.getPeer(peerId), let inputPeer = apiInputPeer(peer) { + return account.network.request(Api.functions.messages.deleteExportedChatInvite(peer: inputPeer, link: link)) + |> mapError { _ in return DeletePeerExportedInvitationError.generic } + |> ignoreValues + } else { + return .complete() + } + } + |> castError(DeletePeerExportedInvitationError.self) + |> switchToLatest +} + +public func deleteAllRevokedPeerExportedInvitations(account: Account, peerId: PeerId) -> Signal { + return account.postbox.transaction { transaction -> Signal in + if let peer = transaction.getPeer(peerId), let inputPeer = apiInputPeer(peer) { + return account.network.request(Api.functions.messages.deleteRevokedExportedChatInvites(peer: inputPeer)) + |> `catch` { _ -> Signal in + return .single(.boolFalse) + } + |> ignoreValues + } else { + return .complete() + } + } + |> switchToLatest +} + private let cachedPeerInvitationImportersCollectionSpec = ItemCacheCollectionSpec(lowWaterItemCount: 10, highWaterItemCount: 20) public struct PeerInvitationImportersState: Equatable { diff --git a/submodules/TelegramCore/Sources/ManagedSecretChatOutgoingOperations.swift b/submodules/TelegramCore/Sources/ManagedSecretChatOutgoingOperations.swift index 76685ab98e..dfd2f79cb4 100644 --- a/submodules/TelegramCore/Sources/ManagedSecretChatOutgoingOperations.swift +++ b/submodules/TelegramCore/Sources/ManagedSecretChatOutgoingOperations.swift @@ -1674,7 +1674,7 @@ private func sendBoxedDecryptedMessage(postbox: Postbox, network: Network, peer: } private func requestTerminateSecretChat(postbox: Postbox, network: Network, peerId: PeerId, tagLocalIndex: Int32, reportSpam: Bool) -> Signal { - return network.request(Api.functions.messages.discardEncryption(chatId: peerId.id)) + return network.request(Api.functions.messages.discardEncryption(flags: 0, chatId: peerId.id)) |> map(Optional.init) |> `catch` { _ in return .single(nil) diff --git a/submodules/TelegramCore/Sources/PeersNearby.swift b/submodules/TelegramCore/Sources/PeersNearby.swift index 5d9954c7d7..6823782443 100644 --- a/submodules/TelegramCore/Sources/PeersNearby.swift +++ b/submodules/TelegramCore/Sources/PeersNearby.swift @@ -34,7 +34,7 @@ public func updatePeersNearbyVisibility(account: Account, update: PeerNearbyVisi case let .visible(latitude, longitude): flags |= (1 << 0) geoPoint = .inputGeoPoint(flags: 0, lat: latitude, long: longitude, accuracyRadius: nil) - selfExpires = 0x7fffffff + selfExpires = 10800 case let .location(latitude, longitude): geoPoint = .inputGeoPoint(flags: 0, lat: latitude, long: longitude, accuracyRadius: nil) case .invisible: @@ -63,6 +63,15 @@ public func updatePeersNearbyVisibility(account: Account, update: PeerNearbyVisi |> map(Optional.init) |> `catch` { error -> Signal in if error.errorCode == 406 { + if error.errorDescription == "USERPIC_PRIVACY_REQUIRED" { + let _ = (account.postbox.transaction { transaction in + transaction.updatePreferencesEntry(key: PreferencesKeys.peersNearby, { entry in + var settings = entry as? PeersNearbyState ?? PeersNearbyState.default + settings.visibilityExpires = nil + return settings + }) + }).start() + } return .single(nil) } else { return .single(nil) diff --git a/submodules/TelegramCore/Sources/TelegramChannel.swift b/submodules/TelegramCore/Sources/TelegramChannel.swift index 37830eb04d..92bce42a85 100644 --- a/submodules/TelegramCore/Sources/TelegramChannel.swift +++ b/submodules/TelegramCore/Sources/TelegramChannel.swift @@ -138,7 +138,7 @@ public extension TelegramChannel { } } - public func hasBannedPermission(_ rights: TelegramChatBannedRightsFlags) -> (Int32, Bool)? { + func hasBannedPermission(_ rights: TelegramChatBannedRightsFlags) -> (Int32, Bool)? { if self.flags.contains(.isCreator) { return nil } @@ -154,7 +154,7 @@ public extension TelegramChannel { return nil } - public var isRestrictedBySlowmode: Bool { + var isRestrictedBySlowmode: Bool { if self.flags.contains(.isCreator) { return false } diff --git a/submodules/TelegramCore/Sources/UpdateSecretChat.swift b/submodules/TelegramCore/Sources/UpdateSecretChat.swift index f5bbd39ddd..adbcc5d84e 100644 --- a/submodules/TelegramCore/Sources/UpdateSecretChat.swift +++ b/submodules/TelegramCore/Sources/UpdateSecretChat.swift @@ -66,7 +66,7 @@ func updateSecretChat(encryptionProvider: EncryptionProvider, accountPeerId: Pee } else { Logger.shared.log("State", "got encryptedChat, but peer or state don't exist or account is not creator") } - case .encryptedChatDiscarded(_): + case let .encryptedChatDiscarded(flags, _): if let currentPeer = currentPeer, let currentState = currentState { let state = currentState.withUpdatedEmbeddedState(.terminated) let peer = currentPeer.withUpdatedEmbeddedState(state.embeddedState.peerState) diff --git a/submodules/TelegramCore/Sources/UpdatesApiUtils.swift b/submodules/TelegramCore/Sources/UpdatesApiUtils.swift index b08bde6309..f4d2f3a289 100644 --- a/submodules/TelegramCore/Sources/UpdatesApiUtils.swift +++ b/submodules/TelegramCore/Sources/UpdatesApiUtils.swift @@ -556,7 +556,7 @@ extension Api.EncryptedChat { switch self { case let .encryptedChat(id, _, _, _, _, _, _): return PeerId(namespace: Namespaces.Peer.SecretChat, id: id) - case let .encryptedChatDiscarded(id): + case let .encryptedChatDiscarded(_, id): return PeerId(namespace: Namespaces.Peer.SecretChat, id: id) case let .encryptedChatEmpty(id): return PeerId(namespace: Namespaces.Peer.SecretChat, id: id) diff --git a/submodules/TelegramPresentationData/Sources/PresentationStrings.swift b/submodules/TelegramPresentationData/Sources/PresentationStrings.swift index 8bccc7a17b..7eeb5ce924 100644 --- a/submodules/TelegramPresentationData/Sources/PresentationStrings.swift +++ b/submodules/TelegramPresentationData/Sources/PresentationStrings.swift @@ -233,4880 +233,4929 @@ public final class PresentationStrings: Equatable { public var Appearance_TintAllColors: String { return self._s[39]! } public var Group_Setup_TypePublicHelp: String { return self._s[40]! } public var ChatSettings_Cache: String { return self._s[41]! } - public var Login_InvalidLastNameError: String { return self._s[42]! } - public var PeerInfo_PaneMedia: String { return self._s[43]! } - public var GroupPermission_PermissionGloballyDisabled: String { return self._s[44]! } - public var LiveLocationUpdated_JustNow: String { return self._s[45]! } + public var InviteLink_RevokedLinks: String { return self._s[42]! } + public var Login_InvalidLastNameError: String { return self._s[43]! } + public var PeerInfo_PaneMedia: String { return self._s[44]! } + public var InviteLink_Revoked: String { return self._s[45]! } + public var GroupPermission_PermissionGloballyDisabled: String { return self._s[46]! } + public var LiveLocationUpdated_JustNow: String { return self._s[47]! } public func Map_LiveLocationPrivateDescription(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[46]!, self._r[46]!, [_0]) - } - public var Channel_Info_Members: String { return self._s[47]! } - public func Channel_CommentsGroup_HeaderSet(_ _0: String) -> (String, [(Int, NSRange)]) { return formatWithArgumentRanges(self._s[48]!, self._r[48]!, [_0]) } - public var Common_edit: String { return self._s[49]! } - public var ChatList_DeleteSavedMessagesConfirmationText: String { return self._s[51]! } - public var OldChannels_GroupEmptyFormat: String { return self._s[52]! } + public var Channel_Info_Members: String { return self._s[49]! } + public func Channel_CommentsGroup_HeaderSet(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[50]!, self._r[50]!, [_0]) + } + public var Common_edit: String { return self._s[51]! } + public var ChatList_DeleteSavedMessagesConfirmationText: String { return self._s[53]! } + public var OldChannels_GroupEmptyFormat: String { return self._s[54]! } public func PUSH_PINNED_AUDIO(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[53]!, self._r[53]!, [_1]) + return formatWithArgumentRanges(self._s[55]!, self._r[55]!, [_1]) } - public var Passport_DiscardMessageAction: String { return self._s[54]! } - public var Passport_FieldOneOf_FinalDelimeter: String { return self._s[55]! } - public var Stickers_SuggestNone: String { return self._s[56]! } - public var Channel_AdminLog_CanPinMessages: String { return self._s[57]! } - public var Stickers_Search: String { return self._s[59]! } - public var Passport_Identity_EditPersonalDetails: String { return self._s[60]! } - public var NotificationSettings_ShowNotificationsAllAccounts: String { return self._s[61]! } - public var Login_ContinueWithLocalization: String { return self._s[62]! } - public var Privacy_ProfilePhoto_NeverShareWith_Title: String { return self._s[63]! } - public var TextFormat_Italic: String { return self._s[65]! } - public var ChatList_Search_NoResultsFitlerLinks: String { return self._s[67]! } - public var Stickers_GroupChooseStickerPack: String { return self._s[68]! } - public var Notification_MessageLifetime1w: String { return self._s[69]! } - public var Channel_Management_AddModerator: String { return self._s[70]! } - public var Conversation_UnsupportedMediaPlaceholder: String { return self._s[71]! } - public var Gif_Search: String { return self._s[72]! } - public var Checkout_ErrorGeneric: String { return self._s[73]! } - public var Conversation_ContextMenuSendMessage: String { return self._s[74]! } - public var Map_SetThisLocation: String { return self._s[75]! } - public var Notifications_ExceptionsDefaultSound: String { return self._s[76]! } - public var PrivacySettings_AutoArchiveInfo: String { return self._s[77]! } - public var Stats_NotificationsTitle: String { return self._s[78]! } - public var Conversation_ClearSecretHistory: String { return self._s[80]! } + public var Passport_DiscardMessageAction: String { return self._s[56]! } + public var Passport_FieldOneOf_FinalDelimeter: String { return self._s[57]! } + public var Stickers_SuggestNone: String { return self._s[58]! } + public var Channel_AdminLog_CanPinMessages: String { return self._s[59]! } + public var Stickers_Search: String { return self._s[61]! } + public var Passport_Identity_EditPersonalDetails: String { return self._s[62]! } + public var NotificationSettings_ShowNotificationsAllAccounts: String { return self._s[63]! } + public var Login_ContinueWithLocalization: String { return self._s[64]! } + public var Privacy_ProfilePhoto_NeverShareWith_Title: String { return self._s[65]! } + public var TextFormat_Italic: String { return self._s[67]! } + public var ChatList_Search_NoResultsFitlerLinks: String { return self._s[69]! } + public var Stickers_GroupChooseStickerPack: String { return self._s[70]! } + public var Notification_MessageLifetime1w: String { return self._s[71]! } + public var Channel_Management_AddModerator: String { return self._s[72]! } + public var Conversation_UnsupportedMediaPlaceholder: String { return self._s[73]! } + public var Gif_Search: String { return self._s[74]! } + public var Checkout_ErrorGeneric: String { return self._s[75]! } + public var Conversation_ContextMenuSendMessage: String { return self._s[76]! } + public var Map_SetThisLocation: String { return self._s[77]! } + public var Notifications_ExceptionsDefaultSound: String { return self._s[78]! } + public var PrivacySettings_AutoArchiveInfo: String { return self._s[79]! } + public var Stats_NotificationsTitle: String { return self._s[80]! } + public var Conversation_ClearSecretHistory: String { return self._s[82]! } public func Notification_CallFormat(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[81]!, self._r[81]!, [_1, _2]) + return formatWithArgumentRanges(self._s[83]!, self._r[83]!, [_1, _2]) } - public var ChatListFolder_DiscardDiscard: String { return self._s[82]! } - public var PrivacyLastSeenSettings_AlwaysShareWith: String { return self._s[83]! } - public var Contacts_InviteFriends: String { return self._s[84]! } - public var Group_LinkedChannel: String { return self._s[85]! } - public var Notification_PassportValuePhone: String { return self._s[87]! } + public var ChatListFolder_DiscardDiscard: String { return self._s[84]! } + public var PrivacyLastSeenSettings_AlwaysShareWith: String { return self._s[85]! } + public var Contacts_InviteFriends: String { return self._s[86]! } + public var Group_LinkedChannel: String { return self._s[87]! } + public var Notification_PassportValuePhone: String { return self._s[89]! } public func InviteText_SingleContact(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[88]!, self._r[88]!, [_0]) + return formatWithArgumentRanges(self._s[90]!, self._r[90]!, [_0]) } - public var UserInfo_BotHelp: String { return self._s[90]! } - public var Passport_Identity_MainPage: String { return self._s[92]! } - public var LogoutOptions_ContactSupportText: String { return self._s[93]! } + public var UserInfo_BotHelp: String { return self._s[92]! } + public var Passport_Identity_MainPage: String { return self._s[94]! } + public var LogoutOptions_ContactSupportText: String { return self._s[95]! } public func VoiceOver_Chat_Title(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[94]!, self._r[94]!, [_0]) + return formatWithArgumentRanges(self._s[96]!, self._r[96]!, [_0]) } - public var StickerPack_ShowStickers: String { return self._s[96]! } - public var AttachmentMenu_PhotoOrVideo: String { return self._s[97]! } - public var Map_Satellite: String { return self._s[98]! } - public var Passport_Identity_MainPageHelp: String { return self._s[99]! } - public var Profile_About: String { return self._s[101]! } - public var Group_Setup_TypePrivate: String { return self._s[102]! } - public var Notifications_ChannelNotifications: String { return self._s[103]! } - public var Call_VoiceOver_VoiceCallIncoming: String { return self._s[104]! } + public var StickerPack_ShowStickers: String { return self._s[98]! } + public var AttachmentMenu_PhotoOrVideo: String { return self._s[99]! } + public var Map_Satellite: String { return self._s[100]! } + public var Passport_Identity_MainPageHelp: String { return self._s[101]! } + public var Profile_About: String { return self._s[103]! } + public var Group_Setup_TypePrivate: String { return self._s[104]! } + public var Notifications_ChannelNotifications: String { return self._s[105]! } + public var Call_VoiceOver_VoiceCallIncoming: String { return self._s[106]! } public func Login_WillCallYou(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[105]!, self._r[105]!, [_0]) + return formatWithArgumentRanges(self._s[107]!, self._r[107]!, [_0]) } - public var WallpaperPreview_Motion: String { return self._s[106]! } - public var Message_VideoMessage: String { return self._s[107]! } - public var SharedMedia_CategoryOther: String { return self._s[108]! } - public var Passport_FieldIdentityUploadHelp: String { return self._s[109]! } - public var PUSH_REMINDER_TITLE: String { return self._s[110]! } - public var Appearance_ThemePreview_Chat_3_Text: String { return self._s[112]! } - public var Login_ResetAccountProtected_Reset: String { return self._s[114]! } - public var Passport_Identity_TypeInternalPassportUploadScan: String { return self._s[115]! } + public var WallpaperPreview_Motion: String { return self._s[108]! } + public var Message_VideoMessage: String { return self._s[109]! } + public var SharedMedia_CategoryOther: String { return self._s[110]! } + public var Passport_FieldIdentityUploadHelp: String { return self._s[111]! } + public var PUSH_REMINDER_TITLE: String { return self._s[112]! } + public var Appearance_ThemePreview_Chat_3_Text: String { return self._s[114]! } + public var Login_ResetAccountProtected_Reset: String { return self._s[116]! } + public var Passport_Identity_TypeInternalPassportUploadScan: String { return self._s[117]! } public func Location_ProximityNotification_Notify(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[116]!, self._r[116]!, [_0]) + return formatWithArgumentRanges(self._s[118]!, self._r[118]!, [_0]) } - public var ChatList_PeerTypeContact: String { return self._s[117]! } - public var Stickers_SuggestAll: String { return self._s[119]! } - public var EmptyGroupInfo_Line3: String { return self._s[120]! } - public var Login_InvalidPhoneError: String { return self._s[121]! } - public var MediaPicker_GroupDescription: String { return self._s[122]! } - public var NetworkUsageSettings_MediaDocumentDataSection: String { return self._s[123]! } - public var Conversation_PrivateChannelTimeLimitedAlertText: String { return self._s[124]! } - public var PrivateDataSettings_Title: String { return self._s[125]! } - public var SecretChat_Title: String { return self._s[126]! } - public var Privacy_ChatsTitle: String { return self._s[127]! } - public var EditProfile_NameAndPhotoHelp: String { return self._s[128]! } - public var Watch_MessageView_Forward: String { return self._s[130]! } - public var ChannelMembers_WhoCanAddMembers_AllMembers: String { return self._s[131]! } + public var ChatList_PeerTypeContact: String { return self._s[119]! } + public var Stickers_SuggestAll: String { return self._s[121]! } + public var EmptyGroupInfo_Line3: String { return self._s[122]! } + public var Login_InvalidPhoneError: String { return self._s[123]! } + public var MediaPicker_GroupDescription: String { return self._s[124]! } + public var NetworkUsageSettings_MediaDocumentDataSection: String { return self._s[125]! } + public var Conversation_PrivateChannelTimeLimitedAlertText: String { return self._s[126]! } + public var PrivateDataSettings_Title: String { return self._s[127]! } + public var SecretChat_Title: String { return self._s[128]! } + public var Privacy_ChatsTitle: String { return self._s[129]! } + public var EditProfile_NameAndPhotoHelp: String { return self._s[130]! } + public var Watch_MessageView_Forward: String { return self._s[132]! } + public var ChannelMembers_WhoCanAddMembers_AllMembers: String { return self._s[133]! } public func PUSH_PINNED_QUIZ(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[132]!, self._r[132]!, [_1, _2]) + return formatWithArgumentRanges(self._s[134]!, self._r[134]!, [_1, _2]) } public func Channel_AdminLog_EndedVoiceChat(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[133]!, self._r[133]!, [_1]) + return formatWithArgumentRanges(self._s[135]!, self._r[135]!, [_1]) } - public var PhotoEditor_DiscardChanges: String { return self._s[134]! } - public var SocksProxySetup_AdNoticeHelp: String { return self._s[135]! } - public var Date_DialogDateFormat: String { return self._s[136]! } - public var SettingsSearch_Synonyms_Proxy_Title: String { return self._s[137]! } - public var Notifications_AlertTones: String { return self._s[138]! } - public var Permissions_SiriAllow_v0: String { return self._s[139]! } - public var Tour_StartButton: String { return self._s[140]! } - public var Stats_InstantViewInteractionsTitle: String { return self._s[141]! } - public var UserInfo_ScamUserWarning: String { return self._s[143]! } - public var NotificationsSound_Chime: String { return self._s[144]! } - public var Update_Skip: String { return self._s[145]! } + public var PhotoEditor_DiscardChanges: String { return self._s[136]! } + public var SocksProxySetup_AdNoticeHelp: String { return self._s[137]! } + public var Date_DialogDateFormat: String { return self._s[138]! } + public var SettingsSearch_Synonyms_Proxy_Title: String { return self._s[139]! } + public var Notifications_AlertTones: String { return self._s[140]! } + public var Permissions_SiriAllow_v0: String { return self._s[141]! } + public var Tour_StartButton: String { return self._s[142]! } + public var Stats_InstantViewInteractionsTitle: String { return self._s[143]! } + public var UserInfo_ScamUserWarning: String { return self._s[146]! } + public var NotificationsSound_Chime: String { return self._s[147]! } + public var Update_Skip: String { return self._s[148]! } public func ChannelInfo_ChannelForbidden(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[146]!, self._r[146]!, [_0]) + return formatWithArgumentRanges(self._s[149]!, self._r[149]!, [_0]) } - public var SettingsSearch_Synonyms_EditProfile_PhoneNumber: String { return self._s[147]! } - public var Notifications_PermissionsTitle: String { return self._s[148]! } - public var Channel_AdminLog_BanSendMedia: String { return self._s[149]! } - public var Notifications_Badge_CountUnreadMessages: String { return self._s[150]! } - public var Appearance_AppIcon: String { return self._s[151]! } - public var Passport_Identity_FilesUploadNew: String { return self._s[152]! } + public var SettingsSearch_Synonyms_EditProfile_PhoneNumber: String { return self._s[150]! } + public var Notifications_PermissionsTitle: String { return self._s[151]! } + public var Channel_AdminLog_BanSendMedia: String { return self._s[152]! } + public var Notifications_Badge_CountUnreadMessages: String { return self._s[153]! } + public var Appearance_AppIcon: String { return self._s[154]! } + public var Passport_Identity_FilesUploadNew: String { return self._s[155]! } public func Passport_Email_UseTelegramEmail(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[153]!, self._r[153]!, [_0]) + return formatWithArgumentRanges(self._s[156]!, self._r[156]!, [_0]) } - public var CreatePoll_QuizTitle: String { return self._s[154]! } - public var DialogList_DeleteConversationConfirmation: String { return self._s[155]! } - public var NotificationsSound_Calypso: String { return self._s[156]! } - public var ChannelMembers_GroupAdminsTitle: String { return self._s[157]! } - public var Checkout_NewCard_PaymentCard: String { return self._s[158]! } - public var Wallpaper_SetCustomBackground: String { return self._s[160]! } - public var Conversation_ContextMenuOpenProfile: String { return self._s[161]! } + public var CreatePoll_QuizTitle: String { return self._s[157]! } + public var DialogList_DeleteConversationConfirmation: String { return self._s[158]! } + public var NotificationsSound_Calypso: String { return self._s[159]! } + public var ChannelMembers_GroupAdminsTitle: String { return self._s[160]! } + public var Checkout_NewCard_PaymentCard: String { return self._s[161]! } + public var Wallpaper_SetCustomBackground: String { return self._s[163]! } + public var Conversation_ContextMenuOpenProfile: String { return self._s[164]! } public func PUSH_MESSAGE_VIDEO_SECRET(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[163]!, self._r[163]!, [_1]) + return formatWithArgumentRanges(self._s[166]!, self._r[166]!, [_1]) } - public var AuthSessions_Terminate: String { return self._s[164]! } - public var ShareFileTip_CloseTip: String { return self._s[165]! } - public var ChatSettings_DownloadInBackgroundInfo: String { return self._s[166]! } - public var Channel_Moderator_AccessLevelRevoke: String { return self._s[167]! } - public var Channel_AdminLogFilter_EventsDeletedMessages: String { return self._s[168]! } - public var Passport_Language_fr: String { return self._s[169]! } + public var AuthSessions_Terminate: String { return self._s[167]! } + public var ShareFileTip_CloseTip: String { return self._s[168]! } + public var ChatSettings_DownloadInBackgroundInfo: String { return self._s[169]! } + public var Channel_Moderator_AccessLevelRevoke: String { return self._s[170]! } + public var Channel_AdminLogFilter_EventsDeletedMessages: String { return self._s[171]! } + public var Passport_Language_fr: String { return self._s[172]! } public func Watch_Time_ShortTodayAt(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[171]!, self._r[171]!, [_0]) - } - public var Passport_Identity_TypeIdentityCard: String { return self._s[172]! } - public var VoiceChat_MuteForMe: String { return self._s[173]! } - public func Conversation_OpenBotLinkAllowMessages(_ _0: String) -> (String, [(Int, NSRange)]) { return formatWithArgumentRanges(self._s[174]!, self._r[174]!, [_0]) } - public var ReportPeer_ReasonCopyright: String { return self._s[175]! } - public var Permissions_PeopleNearbyText_v0: String { return self._s[177]! } - public var Channel_Stickers_NotFoundHelp: String { return self._s[178]! } - public var Passport_Identity_AddDriversLicense: String { return self._s[179]! } - public var AutoDownloadSettings_AutodownloadFiles: String { return self._s[180]! } - public var Permissions_SiriAllowInSettings_v0: String { return self._s[181]! } - public var ApplyLanguage_ChangeLanguageTitle: String { return self._s[182]! } - public var Map_LocatingError: String { return self._s[184]! } - public var ChatSettings_AutoDownloadSettings_TypePhoto: String { return self._s[185]! } + public var Passport_Identity_TypeIdentityCard: String { return self._s[175]! } + public var VoiceChat_MuteForMe: String { return self._s[176]! } + public func Conversation_OpenBotLinkAllowMessages(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[177]!, self._r[177]!, [_0]) + } + public var ReportPeer_ReasonCopyright: String { return self._s[178]! } + public var Permissions_PeopleNearbyText_v0: String { return self._s[180]! } + public var Channel_Stickers_NotFoundHelp: String { return self._s[181]! } + public var Passport_Identity_AddDriversLicense: String { return self._s[182]! } + public var AutoDownloadSettings_AutodownloadFiles: String { return self._s[183]! } + public var Permissions_SiriAllowInSettings_v0: String { return self._s[184]! } + public var ApplyLanguage_ChangeLanguageTitle: String { return self._s[185]! } + public var Map_LocatingError: String { return self._s[187]! } + public var ChatSettings_AutoDownloadSettings_TypePhoto: String { return self._s[188]! } public func VoiceOver_Chat_MusicFrom(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[187]!, self._r[187]!, [_0]) + return formatWithArgumentRanges(self._s[190]!, self._r[190]!, [_0]) } public func Contacts_AccessDeniedHelpLandscape(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[188]!, self._r[188]!, [_0]) + return formatWithArgumentRanges(self._s[191]!, self._r[191]!, [_0]) } - public var Channel_AdminLog_EmptyFilterText: String { return self._s[189]! } - public var Login_SmsRequestState2: String { return self._s[190]! } - public var Conversation_Unmute: String { return self._s[192]! } - public var TwoFactorSetup_Intro_Text: String { return self._s[193]! } - public var Channel_AdminLog_BanSendMessages: String { return self._s[194]! } + public var Channel_AdminLog_EmptyFilterText: String { return self._s[192]! } + public var Login_SmsRequestState2: String { return self._s[193]! } + public var Conversation_Unmute: String { return self._s[195]! } + public var TwoFactorSetup_Intro_Text: String { return self._s[196]! } + public var Channel_AdminLog_BanSendMessages: String { return self._s[197]! } public func Channel_Management_RemovedBy(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[195]!, self._r[195]!, [_0]) + return formatWithArgumentRanges(self._s[198]!, self._r[198]!, [_0]) } - public var AccessDenied_LocationDenied: String { return self._s[196]! } - public var Share_AuthTitle: String { return self._s[197]! } - public var Month_ShortAugust: String { return self._s[198]! } + public var AccessDenied_LocationDenied: String { return self._s[199]! } + public var Share_AuthTitle: String { return self._s[200]! } + public var Month_ShortAugust: String { return self._s[201]! } public func Notification_PinnedDeletedMessage(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[199]!, self._r[199]!, [_0]) + return formatWithArgumentRanges(self._s[202]!, self._r[202]!, [_0]) } - public var Channel_BanUser_PermissionSendMedia: String { return self._s[200]! } - public var SettingsSearch_Synonyms_Data_DownloadInBackground: String { return self._s[201]! } + public var Channel_BanUser_PermissionSendMedia: String { return self._s[203]! } + public var SettingsSearch_Synonyms_Data_DownloadInBackground: String { return self._s[204]! } public func PUSH_CONTACT_JOINED(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[202]!, self._r[202]!, [_1]) + return formatWithArgumentRanges(self._s[205]!, self._r[205]!, [_1]) } - public var WallpaperSearch_ColorTitle: String { return self._s[204]! } - public var Wallpaper_Search: String { return self._s[205]! } - public var ClearCache_StorageUsage: String { return self._s[206]! } - public var CreatePoll_TextPlaceholder: String { return self._s[207]! } - public var Conversation_EditingMessagePanelTitle: String { return self._s[208]! } - public var Channel_EditAdmin_PermissionBanUsers: String { return self._s[209]! } - public var OldChannels_NoticeCreateText: String { return self._s[210]! } - public var ProfilePhoto_MainVideo: String { return self._s[211]! } - public var VoiceChat_StatusListening: String { return self._s[212]! } - public var UserInfo_NotificationsDisabled: String { return self._s[213]! } - public var Map_Unknown: String { return self._s[214]! } - public var Notifications_MessageNotificationsAlert: String { return self._s[215]! } - public var Conversation_StopQuiz: String { return self._s[216]! } - public var Checkout_LiabilityAlertTitle: String { return self._s[217]! } + public var WallpaperSearch_ColorTitle: String { return self._s[207]! } + public var Wallpaper_Search: String { return self._s[208]! } + public var ClearCache_StorageUsage: String { return self._s[209]! } + public var CreatePoll_TextPlaceholder: String { return self._s[210]! } + public var Conversation_EditingMessagePanelTitle: String { return self._s[211]! } + public var Channel_EditAdmin_PermissionBanUsers: String { return self._s[212]! } + public var OldChannels_NoticeCreateText: String { return self._s[213]! } + public var ProfilePhoto_MainVideo: String { return self._s[214]! } + public var VoiceChat_StatusListening: String { return self._s[215]! } + public var UserInfo_NotificationsDisabled: String { return self._s[216]! } + public var Map_Unknown: String { return self._s[217]! } + public var Notifications_MessageNotificationsAlert: String { return self._s[218]! } + public var Conversation_StopQuiz: String { return self._s[219]! } + public var Checkout_LiabilityAlertTitle: String { return self._s[220]! } public func Username_UsernameIsAvailable(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[218]!, self._r[218]!, [_0]) + return formatWithArgumentRanges(self._s[221]!, self._r[221]!, [_0]) } - public var CreatePoll_OptionPlaceholder: String { return self._s[219]! } - public var Conversation_RestrictedStickers: String { return self._s[220]! } - public var MemberSearch_BotSection: String { return self._s[222]! } - public var Channel_Management_AddModeratorHelp: String { return self._s[224]! } - public var MaskStickerSettings_Title: String { return self._s[225]! } - public var ShareMenu_Comment: String { return self._s[226]! } - public var GroupInfo_Notifications: String { return self._s[227]! } - public var CheckoutInfo_ReceiverInfoTitle: String { return self._s[228]! } + public var CreatePoll_OptionPlaceholder: String { return self._s[222]! } + public var Conversation_RestrictedStickers: String { return self._s[223]! } + public var MemberSearch_BotSection: String { return self._s[225]! } + public var Channel_Management_AddModeratorHelp: String { return self._s[227]! } + public var MaskStickerSettings_Title: String { return self._s[228]! } + public var ShareMenu_Comment: String { return self._s[229]! } + public var GroupInfo_Notifications: String { return self._s[230]! } + public var CheckoutInfo_ReceiverInfoTitle: String { return self._s[231]! } public func DialogList_EncryptedChatStartedOutgoing(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[229]!, self._r[229]!, [_0]) + return formatWithArgumentRanges(self._s[232]!, self._r[232]!, [_0]) } - public var Conversation_ContextMenuCopyLink: String { return self._s[230]! } - public var VoiceChat_MutedHelp: String { return self._s[233]! } - public var ChatListFolder_CategoryMuted: String { return self._s[234]! } - public var TwoStepAuth_AddHintDescription: String { return self._s[235]! } + public var Conversation_ContextMenuCopyLink: String { return self._s[233]! } + public var VoiceChat_MutedHelp: String { return self._s[236]! } + public var ChatListFolder_CategoryMuted: String { return self._s[237]! } + public var TwoStepAuth_AddHintDescription: String { return self._s[238]! } public func VoiceOver_Chat_Duration(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[236]!, self._r[236]!, [_0]) + return formatWithArgumentRanges(self._s[239]!, self._r[239]!, [_0]) } - public var Conversation_ClousStorageInfo_Description3: String { return self._s[237]! } - public var Contacts_SortByPresence: String { return self._s[238]! } - public var Watch_Location_Access: String { return self._s[239]! } - public var WallpaperPreview_CustomColorTopText: String { return self._s[240]! } - public var Passport_Address_TypeBankStatement: String { return self._s[241]! } - public var Group_Username_RevokeExistingUsernamesInfo: String { return self._s[242]! } - public var Conversation_ClearPrivateHistory: String { return self._s[243]! } - public var ChatList_Mute: String { return self._s[246]! } - public var Channel_AdminLog_CanDeleteMessagesOfOthers: String { return self._s[247]! } - public var Stats_PostsTitle: String { return self._s[248]! } - public var Paint_Masks: String { return self._s[250]! } - public var PasscodeSettings_TryAgainIn1Minute: String { return self._s[252]! } - public var Chat_AttachmentLimitReached: String { return self._s[253]! } - public var StickerPackActionInfo_ArchivedTitle: String { return self._s[254]! } - public var Watch_Stickers_StickerPacks: String { return self._s[256]! } - public var Channel_Setup_Title: String { return self._s[257]! } - public var GroupInfo_Administrators: String { return self._s[258]! } - public var NotificationSettings_ShowNotificationsAllAccountsInfoOff: String { return self._s[260]! } - public var Conversation_ContextMenuDiscuss: String { return self._s[261]! } - public var StickerPack_BuiltinPackName: String { return self._s[262]! } - public var TwoStepAuth_RecoveryEmailChangeDescription: String { return self._s[264]! } - public var Checkout_ShippingMethod: String { return self._s[266]! } - public var ClearCache_FreeSpace: String { return self._s[267]! } - public var EditTheme_Expand_Preview_IncomingReplyText: String { return self._s[268]! } - public var SettingsSearch_Synonyms_Notifications_ChannelNotificationsSound: String { return self._s[271]! } + public var Conversation_ClousStorageInfo_Description3: String { return self._s[240]! } + public var Contacts_SortByPresence: String { return self._s[241]! } + public var Watch_Location_Access: String { return self._s[242]! } + public var WallpaperPreview_CustomColorTopText: String { return self._s[243]! } + public var Passport_Address_TypeBankStatement: String { return self._s[244]! } + public var Group_Username_RevokeExistingUsernamesInfo: String { return self._s[245]! } + public var Conversation_ClearPrivateHistory: String { return self._s[246]! } + public var ChatList_Mute: String { return self._s[249]! } + public var Channel_AdminLog_CanDeleteMessagesOfOthers: String { return self._s[250]! } + public var Stats_PostsTitle: String { return self._s[251]! } + public var Paint_Masks: String { return self._s[253]! } + public var PasscodeSettings_TryAgainIn1Minute: String { return self._s[255]! } + public var Chat_AttachmentLimitReached: String { return self._s[256]! } + public var StickerPackActionInfo_ArchivedTitle: String { return self._s[257]! } + public var Watch_Stickers_StickerPacks: String { return self._s[259]! } + public var Channel_Setup_Title: String { return self._s[260]! } + public var GroupInfo_Administrators: String { return self._s[261]! } + public var NotificationSettings_ShowNotificationsAllAccountsInfoOff: String { return self._s[263]! } + public var Conversation_ContextMenuDiscuss: String { return self._s[264]! } + public var StickerPack_BuiltinPackName: String { return self._s[265]! } + public var TwoStepAuth_RecoveryEmailChangeDescription: String { return self._s[267]! } + public var Checkout_ShippingMethod: String { return self._s[269]! } + public var ClearCache_FreeSpace: String { return self._s[270]! } + public var EditTheme_Expand_Preview_IncomingReplyText: String { return self._s[271]! } + public var SettingsSearch_Synonyms_Notifications_ChannelNotificationsSound: String { return self._s[274]! } public func TwoStepAuth_ConfirmEmailDescription(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[272]!, self._r[272]!, [_1]) + return formatWithArgumentRanges(self._s[275]!, self._r[275]!, [_1]) } - public var Conversation_typing: String { return self._s[273]! } + public var Conversation_typing: String { return self._s[276]! } public func PrivacySettings_LastSeenContactsMinus(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[275]!, self._r[275]!, [_0]) + return formatWithArgumentRanges(self._s[278]!, self._r[278]!, [_0]) } - public var WebSearch_RecentSectionTitle: String { return self._s[276]! } - public var VoiceChat_EndConfirmationTitle: String { return self._s[277]! } - public var ChatList_UnhideAction: String { return self._s[278]! } - public var PasscodeSettings_6DigitCode: String { return self._s[279]! } - public var CallFeedback_AddComment: String { return self._s[280]! } - public var LoginPassword_PasswordHelp: String { return self._s[281]! } - public var Call_Flip: String { return self._s[282]! } - public var Weekday_ShortWednesday: String { return self._s[284]! } - public var VoiceOver_Chat_PollFinalResults: String { return self._s[285]! } - public var PeerInfo_ButtonAddMember: String { return self._s[286]! } - public var Call_Decline: String { return self._s[288]! } - public var VoiceChat_InviteMemberToGroupFirstAdd: String { return self._s[289]! } - public var Join_ChannelsTooMuch: String { return self._s[291]! } + public var WebSearch_RecentSectionTitle: String { return self._s[279]! } + public var VoiceChat_EndConfirmationTitle: String { return self._s[280]! } + public var ChatList_UnhideAction: String { return self._s[282]! } + public var PasscodeSettings_6DigitCode: String { return self._s[283]! } + public var CallFeedback_AddComment: String { return self._s[284]! } + public var LoginPassword_PasswordHelp: String { return self._s[285]! } + public var Call_Flip: String { return self._s[286]! } + public var Weekday_ShortWednesday: String { return self._s[288]! } + public var VoiceOver_Chat_PollFinalResults: String { return self._s[289]! } + public var PeerInfo_ButtonAddMember: String { return self._s[290]! } + public var Call_Decline: String { return self._s[292]! } + public var VoiceChat_InviteMemberToGroupFirstAdd: String { return self._s[293]! } + public var Join_ChannelsTooMuch: String { return self._s[295]! } public func PUSH_CHANNEL_MESSAGE_NOTEXT(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[292]!, self._r[292]!, [_1]) + return formatWithArgumentRanges(self._s[296]!, self._r[296]!, [_1]) } - public var Passport_Identity_Selfie: String { return self._s[293]! } - public var Privacy_ContactsTitle: String { return self._s[294]! } - public var GroupInfo_InviteLink_Title: String { return self._s[296]! } - public var TwoFactorSetup_Password_PlaceholderPassword: String { return self._s[297]! } - public var Conversation_OpenFile: String { return self._s[298]! } - public var Map_SetThisPlace: String { return self._s[299]! } - public var Channel_Info_Management: String { return self._s[300]! } - public var Passport_Language_hr: String { return self._s[301]! } - public var VoiceChat_Title: String { return self._s[302]! } - public var EditTheme_Edit_Preview_IncomingText: String { return self._s[305]! } - public var OpenFile_Proceed: String { return self._s[306]! } - public var Conversation_SecretChatContextBotAlert: String { return self._s[308]! } - public var GroupInfo_Permissions_SlowmodeValue_Off: String { return self._s[309]! } - public var Privacy_Calls_P2PContacts: String { return self._s[310]! } - public var Appearance_PickAccentColor: String { return self._s[311]! } - public var MediaPicker_TapToUngroupDescription: String { return self._s[312]! } - public var Localization_EnglishLanguageName: String { return self._s[313]! } - public var Stickers_SuggestStickers: String { return self._s[314]! } - public var Passport_Language_ko: String { return self._s[315]! } - public var Settings_ProxyDisabled: String { return self._s[316]! } - public var PrivacySettings_PasscodeOff: String { return self._s[317]! } - public var Undo_LeftChannel: String { return self._s[318]! } - public var Appearance_AutoNightThemeDisabled: String { return self._s[319]! } - public var TextFormat_Bold: String { return self._s[320]! } - public var Login_InfoTitle: String { return self._s[321]! } - public var Channel_BanUser_PermissionSendPolls: String { return self._s[322]! } - public var Settings_AddAnotherAccount: String { return self._s[323]! } - public var GroupPermission_NewTitle: String { return self._s[324]! } - public var Login_SelectCountry_Title: String { return self._s[325]! } - public var Cache_ServiceFiles: String { return self._s[326]! } - public var Passport_Language_nl: String { return self._s[327]! } - public var Contacts_TopSection: String { return self._s[328]! } - public var Passport_Identity_DateOfBirthPlaceholder: String { return self._s[329]! } - public var VoiceChat_StatusInvited: String { return self._s[331]! } - public var Conversation_ContextMenuReport: String { return self._s[332]! } + public var Passport_Identity_Selfie: String { return self._s[297]! } + public var Privacy_ContactsTitle: String { return self._s[298]! } + public var GroupInfo_InviteLink_Title: String { return self._s[300]! } + public var TwoFactorSetup_Password_PlaceholderPassword: String { return self._s[301]! } + public var Conversation_OpenFile: String { return self._s[302]! } + public var Map_SetThisPlace: String { return self._s[303]! } + public var Channel_Info_Management: String { return self._s[304]! } + public var Passport_Language_hr: String { return self._s[305]! } + public var VoiceChat_Title: String { return self._s[306]! } + public var EditTheme_Edit_Preview_IncomingText: String { return self._s[309]! } + public var OpenFile_Proceed: String { return self._s[310]! } + public var Conversation_SecretChatContextBotAlert: String { return self._s[312]! } + public var GroupInfo_Permissions_SlowmodeValue_Off: String { return self._s[313]! } + public var Privacy_Calls_P2PContacts: String { return self._s[314]! } + public var Appearance_PickAccentColor: String { return self._s[315]! } + public var MediaPicker_TapToUngroupDescription: String { return self._s[316]! } + public var Localization_EnglishLanguageName: String { return self._s[317]! } + public var Stickers_SuggestStickers: String { return self._s[318]! } + public var Passport_Language_ko: String { return self._s[319]! } + public var Settings_ProxyDisabled: String { return self._s[320]! } + public var PrivacySettings_PasscodeOff: String { return self._s[321]! } + public var Undo_LeftChannel: String { return self._s[322]! } + public var Appearance_AutoNightThemeDisabled: String { return self._s[323]! } + public var TextFormat_Bold: String { return self._s[324]! } + public var Login_InfoTitle: String { return self._s[325]! } + public var Channel_BanUser_PermissionSendPolls: String { return self._s[326]! } + public var Settings_AddAnotherAccount: String { return self._s[327]! } + public var GroupPermission_NewTitle: String { return self._s[328]! } + public var Login_SelectCountry_Title: String { return self._s[329]! } + public var Cache_ServiceFiles: String { return self._s[330]! } + public var Passport_Language_nl: String { return self._s[331]! } + public var Contacts_TopSection: String { return self._s[332]! } + public var Passport_Identity_DateOfBirthPlaceholder: String { return self._s[333]! } + public var VoiceChat_StatusInvited: String { return self._s[335]! } + public var Conversation_ContextMenuReport: String { return self._s[336]! } public func Login_BannedPhoneBody(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[333]!, self._r[333]!, [_0]) + return formatWithArgumentRanges(self._s[337]!, self._r[337]!, [_0]) } - public var Conversation_Search: String { return self._s[334]! } - public var Group_Setup_HistoryVisibleHelp: String { return self._s[336]! } - public var ReportPeer_AlertSuccess: String { return self._s[338]! } - public var AutoNightTheme_Title: String { return self._s[340]! } + public var Conversation_Search: String { return self._s[338]! } + public var Group_Setup_HistoryVisibleHelp: String { return self._s[340]! } + public var ReportPeer_AlertSuccess: String { return self._s[342]! } + public var AutoNightTheme_Title: String { return self._s[344]! } public func Notification_PinnedTextMessage(_ _0: String, _ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[342]!, self._r[342]!, [_0, _1]) + return formatWithArgumentRanges(self._s[346]!, self._r[346]!, [_0, _1]) } public func Conversation_OpenBotLinkText(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[343]!, self._r[343]!, [_0]) + return formatWithArgumentRanges(self._s[347]!, self._r[347]!, [_0]) } - public var Conversation_ShareBotContactConfirmation: String { return self._s[344]! } - public var TwoStepAuth_RecoveryCode: String { return self._s[345]! } - public var SocksProxySetup_ConnectAndSave: String { return self._s[346]! } + public var Conversation_ShareBotContactConfirmation: String { return self._s[348]! } + public var TwoStepAuth_RecoveryCode: String { return self._s[349]! } + public var SocksProxySetup_ConnectAndSave: String { return self._s[350]! } public func MESSAGE_INVOICE(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[347]!, self._r[347]!, [_1, _2]) + return formatWithArgumentRanges(self._s[351]!, self._r[351]!, [_1, _2]) } public func Channel_AdminLog_MessageChangedGroupUsername(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[348]!, self._r[348]!, [_0]) + return formatWithArgumentRanges(self._s[352]!, self._r[352]!, [_0]) } - public var Replies_BlockAndDeleteRepliesActionTitle: String { return self._s[349]! } + public var Replies_BlockAndDeleteRepliesActionTitle: String { return self._s[353]! } public func Notification_GroupInviter(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[350]!, self._r[350]!, [_0]) - } - public var VoiceChat_CopyInviteLink: String { return self._s[351]! } - public var Conversation_InfoGroup: String { return self._s[352]! } - public func Map_AccurateTo(_ _0: String) -> (String, [(Int, NSRange)]) { return formatWithArgumentRanges(self._s[354]!, self._r[354]!, [_0]) } - public var Conversation_ChatBackground: String { return self._s[355]! } - public var PhotoEditor_Set: String { return self._s[356]! } - public func Channel_Management_PromotedBy(_ _0: String) -> (String, [(Int, NSRange)]) { + public var VoiceChat_CopyInviteLink: String { return self._s[355]! } + public var Conversation_InfoGroup: String { return self._s[356]! } + public func Map_AccurateTo(_ _0: String) -> (String, [(Int, NSRange)]) { return formatWithArgumentRanges(self._s[358]!, self._r[358]!, [_0]) } - public var IntentsSettings_SuggestedChatsContacts: String { return self._s[359]! } - public var Passport_Phone_Title: String { return self._s[361]! } - public var Conversation_EditingMessageMediaChange: String { return self._s[362]! } - public var Channel_LinkItem: String { return self._s[363]! } - public var VoiceChat_EndConfirmationText: String { return self._s[364]! } + public var Conversation_ChatBackground: String { return self._s[359]! } + public var PhotoEditor_Set: String { return self._s[360]! } + public func Channel_Management_PromotedBy(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[362]!, self._r[362]!, [_0]) + } + public var IntentsSettings_SuggestedChatsContacts: String { return self._s[363]! } + public var Passport_Phone_Title: String { return self._s[365]! } + public var Conversation_EditingMessageMediaChange: String { return self._s[366]! } + public var Channel_LinkItem: String { return self._s[367]! } + public var VoiceChat_EndConfirmationText: String { return self._s[368]! } public func PUSH_CHAT_DELETE_MEMBER(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[365]!, self._r[365]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[369]!, self._r[369]!, [_1, _2, _3]) } - public var Conversation_DeleteManyMessages: String { return self._s[366]! } - public var Notifications_Badge_IncludeMutedChats: String { return self._s[367]! } - public var AuthSessions_AddedDeviceTitle: String { return self._s[370]! } - public var Privacy_Calls_NeverAllow_Placeholder: String { return self._s[371]! } - public var Settings_ProxyConnecting: String { return self._s[372]! } - public var Theme_Colors_Accent: String { return self._s[373]! } - public var Theme_Colors_ColorWallpaperWarning: String { return self._s[374]! } + public var Conversation_DeleteManyMessages: String { return self._s[370]! } + public var Notifications_Badge_IncludeMutedChats: String { return self._s[371]! } + public var AuthSessions_AddedDeviceTitle: String { return self._s[374]! } + public var Privacy_Calls_NeverAllow_Placeholder: String { return self._s[375]! } + public var Settings_ProxyConnecting: String { return self._s[376]! } + public var Theme_Colors_Accent: String { return self._s[377]! } + public var Theme_Colors_ColorWallpaperWarning: String { return self._s[378]! } public func PUSH_PHONE_CALL_MISSED(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[376]!, self._r[376]!, [_1]) + return formatWithArgumentRanges(self._s[380]!, self._r[380]!, [_1]) } - public var Passport_Language_lo: String { return self._s[377]! } + public var Passport_Language_lo: String { return self._s[381]! } public func Watch_Time_ShortWeekdayAt(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[379]!, self._r[379]!, [_1, _2]) + return formatWithArgumentRanges(self._s[383]!, self._r[383]!, [_1, _2]) } - public var Permissions_NotificationsText_v0: String { return self._s[380]! } - public var ChatList_Context_RemoveFromRecents: String { return self._s[381]! } - public var Watch_GroupInfo_Title: String { return self._s[382]! } - public var Settings_AddDevice: String { return self._s[384]! } - public var WallpaperPreview_SwipeColorsTopText: String { return self._s[385]! } + public var Permissions_NotificationsText_v0: String { return self._s[384]! } + public var ChatList_Context_RemoveFromRecents: String { return self._s[385]! } + public var Watch_GroupInfo_Title: String { return self._s[386]! } + public var Settings_AddDevice: String { return self._s[388]! } + public var WallpaperPreview_SwipeColorsTopText: String { return self._s[389]! } public func PUSH_CHANNEL_ALBUM(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[386]!, self._r[386]!, [_1]) + return formatWithArgumentRanges(self._s[390]!, self._r[390]!, [_1]) } - public var TwoStepAuth_Disable: String { return self._s[388]! } + public var TwoStepAuth_Disable: String { return self._s[392]! } public func Conversation_AddNameToContacts(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[389]!, self._r[389]!, [_0]) + return formatWithArgumentRanges(self._s[393]!, self._r[393]!, [_0]) } public func Time_PreciseDate_m10(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[390]!, self._r[390]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[394]!, self._r[394]!, [_1, _2, _3]) } public func Login_WillSendSms(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[391]!, self._r[391]!, [_0]) + return formatWithArgumentRanges(self._s[395]!, self._r[395]!, [_0]) } - public var Channel_AdminLog_BanReadMessages: String { return self._s[392]! } - public var Undo_ChatDeleted: String { return self._s[393]! } - public var ContactInfo_URLLabelHomepage: String { return self._s[394]! } + public var Channel_AdminLog_BanReadMessages: String { return self._s[396]! } + public var Undo_ChatDeleted: String { return self._s[397]! } + public var ContactInfo_URLLabelHomepage: String { return self._s[398]! } public func PUSH_CHAT_MESSAGE_STICKER(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[395]!, self._r[395]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[399]!, self._r[399]!, [_1, _2, _3]) } - public var FastTwoStepSetup_EmailHelp: String { return self._s[396]! } - public var Contacts_SelectAll: String { return self._s[397]! } - public var Privacy_ContactsReset: String { return self._s[398]! } - public var AttachmentMenu_File: String { return self._s[400]! } - public var PasscodeSettings_EncryptData: String { return self._s[401]! } - public var EditTheme_ThemeTemplateAlertText: String { return self._s[402]! } + public var FastTwoStepSetup_EmailHelp: String { return self._s[400]! } + public var Contacts_SelectAll: String { return self._s[401]! } + public var Privacy_ContactsReset: String { return self._s[402]! } + public var AttachmentMenu_File: String { return self._s[404]! } + public var PasscodeSettings_EncryptData: String { return self._s[405]! } + public var EditTheme_ThemeTemplateAlertText: String { return self._s[406]! } public func Privacy_GroupsAndChannels_InviteToChannelError(_ _0: String, _ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[404]!, self._r[404]!, [_0, _1]) + return formatWithArgumentRanges(self._s[408]!, self._r[408]!, [_0, _1]) } public func Profile_CreateEncryptedChatOutdatedError(_ _0: String, _ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[405]!, self._r[405]!, [_0, _1]) + return formatWithArgumentRanges(self._s[409]!, self._r[409]!, [_0, _1]) } - public var PhotoEditor_ShadowsTint: String { return self._s[407]! } - public var GroupInfo_ChatAdmins: String { return self._s[408]! } - public var ArchivedChats_IntroTitle2: String { return self._s[409]! } - public var Cache_LowDiskSpaceText: String { return self._s[410]! } - public var CreatePoll_Anonymous: String { return self._s[411]! } - public var Checkout_PaymentMethod_New: String { return self._s[412]! } - public var Invitation_JoinGroup: String { return self._s[413]! } + public var PhotoEditor_ShadowsTint: String { return self._s[411]! } + public var GroupInfo_ChatAdmins: String { return self._s[412]! } + public var ArchivedChats_IntroTitle2: String { return self._s[413]! } + public var Cache_LowDiskSpaceText: String { return self._s[414]! } + public var CreatePoll_Anonymous: String { return self._s[415]! } + public var Checkout_PaymentMethod_New: String { return self._s[416]! } + public var Invitation_JoinGroup: String { return self._s[417]! } public func Time_MonthOfYear_m4(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[416]!, self._r[416]!, [_0]) - } - public var CheckoutInfo_SaveInfoHelp: String { return self._s[417]! } - public var Notification_Reply: String { return self._s[419]! } - public func Login_PhoneBannedEmailSubject(_ _0: String) -> (String, [(Int, NSRange)]) { return formatWithArgumentRanges(self._s[420]!, self._r[420]!, [_0]) } - public var Login_PhoneTitle: String { return self._s[421]! } - public var VoiceChat_UnmuteHelp: String { return self._s[422]! } - public var VoiceOver_Media_PlaybackRateNormal: String { return self._s[423]! } - public func PUSH_CHAT_MESSAGE_INVOICE(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[424]!, self._r[424]!, [_1, _2, _3]) + public var CheckoutInfo_SaveInfoHelp: String { return self._s[421]! } + public var Notification_Reply: String { return self._s[423]! } + public func Login_PhoneBannedEmailSubject(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[424]!, self._r[424]!, [_0]) } - public var Appearance_TextSize_Title: String { return self._s[425]! } - public var NetworkUsageSettings_MediaImageDataSection: String { return self._s[427]! } - public var VoiceOver_Navigation_Compose: String { return self._s[428]! } - public var Passport_InfoText: String { return self._s[429]! } - public var ApplyLanguage_ApplyLanguageAction: String { return self._s[430]! } - public var MessagePoll_LabelClosed: String { return self._s[432]! } - public var AttachmentMenu_SendAsFiles: String { return self._s[433]! } - public var KeyCommand_FocusOnInputField: String { return self._s[434]! } - public var Conversation_ContextViewThread: String { return self._s[435]! } - public var Privacy_SecretChatsLinkPreviews: String { return self._s[437]! } - public var Permissions_PeopleNearbyAllow_v0: String { return self._s[438]! } - public var Conversation_ContextMenuMention: String { return self._s[440]! } - public var CreatePoll_QuizInfo: String { return self._s[441]! } - public var Appearance_ThemePreview_ChatList_2_Name: String { return self._s[442]! } - public var Username_LinkCopied: String { return self._s[443]! } - public var IntentsSettings_SuggestedAndSpotlightChatsInfo: String { return self._s[444]! } - public var TwoStepAuth_ChangePassword: String { return self._s[445]! } - public var Watch_Suggestion_Thanks: String { return self._s[446]! } - public var Channel_TitleInfo: String { return self._s[447]! } - public var ChatList_ChatTypesSection: String { return self._s[448]! } + public var Login_PhoneTitle: String { return self._s[425]! } + public var VoiceChat_UnmuteHelp: String { return self._s[426]! } + public var VoiceOver_Media_PlaybackRateNormal: String { return self._s[427]! } + public func PUSH_CHAT_MESSAGE_INVOICE(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[428]!, self._r[428]!, [_1, _2, _3]) + } + public var Appearance_TextSize_Title: String { return self._s[429]! } + public var NetworkUsageSettings_MediaImageDataSection: String { return self._s[431]! } + public var VoiceOver_Navigation_Compose: String { return self._s[432]! } + public var Passport_InfoText: String { return self._s[433]! } + public var ApplyLanguage_ApplyLanguageAction: String { return self._s[434]! } + public var MessagePoll_LabelClosed: String { return self._s[436]! } + public var AttachmentMenu_SendAsFiles: String { return self._s[437]! } + public var KeyCommand_FocusOnInputField: String { return self._s[438]! } + public var Conversation_ContextViewThread: String { return self._s[439]! } + public var Privacy_SecretChatsLinkPreviews: String { return self._s[441]! } + public var Permissions_PeopleNearbyAllow_v0: String { return self._s[442]! } + public var Conversation_ContextMenuMention: String { return self._s[444]! } + public var CreatePoll_QuizInfo: String { return self._s[445]! } + public var Appearance_ThemePreview_ChatList_2_Name: String { return self._s[446]! } + public var Username_LinkCopied: String { return self._s[447]! } + public var IntentsSettings_SuggestedAndSpotlightChatsInfo: String { return self._s[448]! } + public var TwoStepAuth_ChangePassword: String { return self._s[449]! } + public var Watch_Suggestion_Thanks: String { return self._s[450]! } + public var Channel_TitleInfo: String { return self._s[451]! } + public var ChatList_ChatTypesSection: String { return self._s[452]! } public func Watch_LastSeen_AtDate(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[449]!, self._r[449]!, [_0]) + return formatWithArgumentRanges(self._s[453]!, self._r[453]!, [_0]) } public func Channel_AdminLog_PollStopped(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[450]!, self._r[450]!, [_0]) + return formatWithArgumentRanges(self._s[454]!, self._r[454]!, [_0]) } - public var AuthSessions_AddDevice_InvalidQRCode: String { return self._s[451]! } + public var AuthSessions_AddDevice_InvalidQRCode: String { return self._s[455]! } public func Call_MicrophoneOff(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[452]!, self._r[452]!, [_0]) + return formatWithArgumentRanges(self._s[456]!, self._r[456]!, [_0]) } - public var Channel_AdminLogFilter_ChannelEventsInfo: String { return self._s[453]! } - public var Profile_MessageLifetimeForever: String { return self._s[454]! } - public var ArchivedChats_IntroText1: String { return self._s[455]! } - public var Notifications_ChannelNotificationsPreview: String { return self._s[456]! } - public var Map_PullUpForPlaces: String { return self._s[458]! } - public var UserInfo_TelegramCall: String { return self._s[459]! } - public var Conversation_ShareMyContactInfo: String { return self._s[460]! } - public var ChatList_Tabs_All: String { return self._s[461]! } - public var Notification_PassportValueEmail: String { return self._s[462]! } - public var Notification_VideoCallIncoming: String { return self._s[463]! } - public var SettingsSearch_Synonyms_Appearance_AutoNightTheme: String { return self._s[464]! } - public var Channel_Username_InvalidTaken: String { return self._s[465]! } - public var GroupPermission_EditingDisabled: String { return self._s[466]! } - public var ChatContextMenu_TextSelectionTip: String { return self._s[467]! } - public var Passport_Language_pl: String { return self._s[469]! } - public var Call_Accept: String { return self._s[470]! } - public var ChatListFolder_NameSectionHeader: String { return self._s[471]! } + public var Channel_AdminLogFilter_ChannelEventsInfo: String { return self._s[457]! } + public var Profile_MessageLifetimeForever: String { return self._s[458]! } + public var ArchivedChats_IntroText1: String { return self._s[459]! } + public var Notifications_ChannelNotificationsPreview: String { return self._s[460]! } + public var Map_PullUpForPlaces: String { return self._s[462]! } + public var UserInfo_TelegramCall: String { return self._s[463]! } + public var Conversation_ShareMyContactInfo: String { return self._s[464]! } + public var ChatList_Tabs_All: String { return self._s[465]! } + public var Notification_PassportValueEmail: String { return self._s[466]! } + public var Notification_VideoCallIncoming: String { return self._s[467]! } + public var SettingsSearch_Synonyms_Appearance_AutoNightTheme: String { return self._s[468]! } + public var Channel_Username_InvalidTaken: String { return self._s[469]! } + public var GroupPermission_EditingDisabled: String { return self._s[470]! } + public var InviteLink_PeopleJoinedShortNone: String { return self._s[471]! } + public var ChatContextMenu_TextSelectionTip: String { return self._s[472]! } + public var Passport_Language_pl: String { return self._s[474]! } + public var Call_Accept: String { return self._s[475]! } + public var ChatListFolder_NameSectionHeader: String { return self._s[476]! } public func Passport_Identity_NativeNameTitle(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[472]!, self._r[472]!, [_0]) + return formatWithArgumentRanges(self._s[477]!, self._r[477]!, [_0]) } - public var ClearCache_Forever: String { return self._s[473]! } + public var ClearCache_Forever: String { return self._s[478]! } public func ChannelInfo_AddParticipantConfirmation(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[475]!, self._r[475]!, [_0]) + return formatWithArgumentRanges(self._s[480]!, self._r[480]!, [_0]) } - public var Group_EditAdmin_RankAdminPlaceholder: String { return self._s[476]! } - public var Calls_SubmitRating: String { return self._s[477]! } - public var Location_LiveLocationRequired_ShareLocation: String { return self._s[478]! } + public var Group_EditAdmin_RankAdminPlaceholder: String { return self._s[481]! } + public var Calls_SubmitRating: String { return self._s[482]! } + public var Location_LiveLocationRequired_ShareLocation: String { return self._s[483]! } public func ChatList_AddedToFolderTooltip(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[479]!, self._r[479]!, [_1, _2]) + return formatWithArgumentRanges(self._s[484]!, self._r[484]!, [_1, _2]) } - public var IntentsSettings_MainAccountInfo: String { return self._s[480]! } - public var Map_Hybrid: String { return self._s[482]! } - public var ChatList_Context_Archive: String { return self._s[483]! } - public var Message_PinnedDocumentMessage: String { return self._s[484]! } - public var State_ConnectingToProxyInfo: String { return self._s[485]! } - public var Passport_Identity_NativeNameGenericTitle: String { return self._s[487]! } - public var Settings_AppLanguage: String { return self._s[488]! } + public var IntentsSettings_MainAccountInfo: String { return self._s[485]! } + public var Map_Hybrid: String { return self._s[487]! } + public var ChatList_Context_Archive: String { return self._s[488]! } + public var Message_PinnedDocumentMessage: String { return self._s[489]! } + public var State_ConnectingToProxyInfo: String { return self._s[490]! } + public var Passport_Identity_NativeNameGenericTitle: String { return self._s[492]! } + public var Settings_AppLanguage: String { return self._s[493]! } public func Checkout_SavePasswordTimeoutAndFaceId(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[489]!, self._r[489]!, [_0]) - } - public var Notifications_PermissionsEnable: String { return self._s[491]! } - public var CheckoutInfo_ShippingInfoAddress1Placeholder: String { return self._s[492]! } - public func UserInfo_BlockActionTitle(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[493]!, self._r[493]!, [_0]) - } - public func AuthSessions_Message(_ _0: String) -> (String, [(Int, NSRange)]) { return formatWithArgumentRanges(self._s[494]!, self._r[494]!, [_0]) } - public var NotificationsSound_Aurora: String { return self._s[497]! } - public var ScheduledMessages_ClearAll: String { return self._s[500]! } + public var Notifications_PermissionsEnable: String { return self._s[496]! } + public var CheckoutInfo_ShippingInfoAddress1Placeholder: String { return self._s[497]! } + public func UserInfo_BlockActionTitle(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[498]!, self._r[498]!, [_0]) + } + public func AuthSessions_Message(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[499]!, self._r[499]!, [_0]) + } + public var NotificationsSound_Aurora: String { return self._s[502]! } + public var ScheduledMessages_ClearAll: String { return self._s[505]! } public func CancelResetAccount_TextSMS(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[501]!, self._r[501]!, [_0]) + return formatWithArgumentRanges(self._s[506]!, self._r[506]!, [_0]) } - public var Settings_BlockedUsers: String { return self._s[503]! } + public var Settings_BlockedUsers: String { return self._s[508]! } public func UserInfo_StartSecretChatConfirmation(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[505]!, self._r[505]!, [_0]) + return formatWithArgumentRanges(self._s[510]!, self._r[510]!, [_0]) } - public var Passport_Language_hu: String { return self._s[506]! } + public var Passport_Language_hu: String { return self._s[511]! } public func Conversation_ScheduleMessage_SendTomorrow(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[507]!, self._r[507]!, [_0]) + return formatWithArgumentRanges(self._s[512]!, self._r[512]!, [_0]) } - public var StickerPack_Share: String { return self._s[508]! } - public var Checkout_NewCard_SaveInfoEnableHelp: String { return self._s[509]! } + public var StickerPack_Share: String { return self._s[513]! } + public var Checkout_NewCard_SaveInfoEnableHelp: String { return self._s[514]! } public func ForwardedAuthors2(_ _0: String, _ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[510]!, self._r[510]!, [_0, _1]) + return formatWithArgumentRanges(self._s[515]!, self._r[515]!, [_0, _1]) } - public var Privacy_ContactsResetConfirmation: String { return self._s[511]! } - public var AppleWatch_ReplyPresets: String { return self._s[512]! } - public var Bot_GenericBotStatus: String { return self._s[513]! } - public var Appearance_ShareThemeColor: String { return self._s[514]! } - public var AuthSessions_AddDevice_UrlLoginHint: String { return self._s[515]! } - public var ReportGroupLocation_Title: String { return self._s[516]! } + public var Privacy_ContactsResetConfirmation: String { return self._s[516]! } + public var AppleWatch_ReplyPresets: String { return self._s[517]! } + public var Bot_GenericBotStatus: String { return self._s[518]! } + public var Appearance_ShareThemeColor: String { return self._s[519]! } + public var AuthSessions_AddDevice_UrlLoginHint: String { return self._s[520]! } + public var ReportGroupLocation_Title: String { return self._s[521]! } public func Activity_RemindAboutUser(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[517]!, self._r[517]!, [_0]) + return formatWithArgumentRanges(self._s[522]!, self._r[522]!, [_0]) } - public var Profile_CreateEncryptedChatError: String { return self._s[518]! } - public var Channel_EditAdmin_TransferOwnership: String { return self._s[519]! } - public var Wallpaper_ErrorNotFound: String { return self._s[520]! } - public var Bot_GenericSupportStatus: String { return self._s[521]! } - public var Activity_UploadingPhoto: String { return self._s[523]! } - public var Watch_UserInfo_Title: String { return self._s[525]! } - public var SocksProxySetup_ProxyTelegram: String { return self._s[526]! } - public var Appearance_ThemeDay: String { return self._s[527]! } + public var Profile_CreateEncryptedChatError: String { return self._s[523]! } + public var Channel_EditAdmin_TransferOwnership: String { return self._s[524]! } + public var Wallpaper_ErrorNotFound: String { return self._s[525]! } + public var Bot_GenericSupportStatus: String { return self._s[526]! } + public var Activity_UploadingPhoto: String { return self._s[528]! } + public var Watch_UserInfo_Title: String { return self._s[530]! } + public var SocksProxySetup_ProxyTelegram: String { return self._s[531]! } + public var Appearance_ThemeDay: String { return self._s[532]! } public func ApplyLanguage_ChangeLanguageOfficialText(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[528]!, self._r[528]!, [_1]) + return formatWithArgumentRanges(self._s[533]!, self._r[533]!, [_1]) } public func FileSize_B(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[529]!, self._r[529]!, [_0]) + return formatWithArgumentRanges(self._s[534]!, self._r[534]!, [_0]) } - public var Passport_Title: String { return self._s[532]! } + public var InviteLink_AdditionalLinks: String { return self._s[535]! } + public var Passport_Title: String { return self._s[538]! } public func Time_PreciseDate_m3(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[534]!, self._r[534]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[540]!, self._r[540]!, [_1, _2, _3]) } - public var CheckoutInfo_ShippingInfoCountryPlaceholder: String { return self._s[535]! } - public var SocksProxySetup_ShareLink: String { return self._s[538]! } - public var AuthSessions_OtherDevices: String { return self._s[539]! } - public var IntentsSettings_SuggestedChatsGroups: String { return self._s[540]! } - public var Watch_MessageView_Reply: String { return self._s[541]! } - public var Camera_FlashOn: String { return self._s[543]! } + public var CheckoutInfo_ShippingInfoCountryPlaceholder: String { return self._s[541]! } + public var SocksProxySetup_ShareLink: String { return self._s[544]! } + public var AuthSessions_OtherDevices: String { return self._s[545]! } + public var IntentsSettings_SuggestedChatsGroups: String { return self._s[546]! } + public var Watch_MessageView_Reply: String { return self._s[547]! } + public var Camera_FlashOn: String { return self._s[549]! } public func PUSH_MESSAGE_STICKER(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[544]!, self._r[544]!, [_1, _2]) + return formatWithArgumentRanges(self._s[550]!, self._r[550]!, [_1, _2]) } - public var Conversation_ContextMenuBlock: String { return self._s[545]! } - public var Channel_EditAdmin_PermissionEditMessages: String { return self._s[547]! } - public var Privacy_Calls_NeverAllow: String { return self._s[548]! } - public var SharedMedia_CategoryLinks: String { return self._s[549]! } - public var Conversation_PinMessageAlertGroup: String { return self._s[552]! } - public var Passport_Identity_ScansHelp: String { return self._s[553]! } - public var ShareMenu_CopyShareLink: String { return self._s[554]! } - public var StickerSettings_MaskContextInfo: String { return self._s[555]! } - public var SocksProxySetup_ProxyStatusChecking: String { return self._s[556]! } - public var AutoDownloadSettings_AutodownloadPhotos: String { return self._s[558]! } - public var Checkout_ErrorPrecheckoutFailed: String { return self._s[560]! } - public var NotificationsSound_Popcorn: String { return self._s[561]! } - public var FeatureDisabled_Oops: String { return self._s[562]! } + public var Conversation_ContextMenuBlock: String { return self._s[551]! } + public var Channel_EditAdmin_PermissionEditMessages: String { return self._s[553]! } + public var Privacy_Calls_NeverAllow: String { return self._s[554]! } + public var SharedMedia_CategoryLinks: String { return self._s[555]! } + public var Conversation_PinMessageAlertGroup: String { return self._s[558]! } + public var Passport_Identity_ScansHelp: String { return self._s[559]! } + public var ShareMenu_CopyShareLink: String { return self._s[560]! } + public var StickerSettings_MaskContextInfo: String { return self._s[561]! } + public var InviteLink_Create_EditTitle: String { return self._s[562]! } + public var SocksProxySetup_ProxyStatusChecking: String { return self._s[563]! } + public var AutoDownloadSettings_AutodownloadPhotos: String { return self._s[565]! } + public var Checkout_ErrorPrecheckoutFailed: String { return self._s[567]! } + public var NotificationsSound_Popcorn: String { return self._s[568]! } + public var FeatureDisabled_Oops: String { return self._s[569]! } public func Channel_AdminLog_MessageChangedChannelAbout(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[563]!, self._r[563]!, [_0]) + return formatWithArgumentRanges(self._s[570]!, self._r[570]!, [_0]) } - public var Notification_PinnedMessage: String { return self._s[564]! } - public var Tour_Title4: String { return self._s[565]! } + public var Notification_PinnedMessage: String { return self._s[571]! } + public var Tour_Title4: String { return self._s[572]! } public func Notification_VoiceChatInvitationForYou(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[566]!, self._r[566]!, [_1]) + return formatWithArgumentRanges(self._s[573]!, self._r[573]!, [_1]) } - public var Watch_Suggestion_OK: String { return self._s[567]! } - public var Compose_TokenListPlaceholder: String { return self._s[568]! } - public var EditTheme_Edit_TopInfo: String { return self._s[569]! } - public var Gif_NoGifsFound: String { return self._s[570]! } - public var Login_InvalidCountryCode: String { return self._s[571]! } - public var SettingsSearch_Synonyms_Notifications_ChannelNotificationsExceptions: String { return self._s[572]! } - public var Call_VoiceOver_VideoCallMissed: String { return self._s[573]! } + public var Watch_Suggestion_OK: String { return self._s[574]! } + public var Compose_TokenListPlaceholder: String { return self._s[575]! } + public var InviteLink_PermanentLink: String { return self._s[576]! } + public var EditTheme_Edit_TopInfo: String { return self._s[577]! } + public var Gif_NoGifsFound: String { return self._s[578]! } + public var Login_InvalidCountryCode: String { return self._s[579]! } + public var SettingsSearch_Synonyms_Notifications_ChannelNotificationsExceptions: String { return self._s[580]! } + public var Call_VoiceOver_VideoCallMissed: String { return self._s[581]! } public func PUSH_LOCKED_MESSAGE(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[575]!, self._r[575]!, [_1]) + return formatWithArgumentRanges(self._s[583]!, self._r[583]!, [_1]) } - public var Profile_CreateNewContact: String { return self._s[576]! } - public var AutoDownloadSettings_DataUsageLow: String { return self._s[577]! } - public var SettingsSearch_Synonyms_Notifications_InAppNotificationsPreview: String { return self._s[578]! } - public var Group_Setup_TypePublic: String { return self._s[579]! } - public var Weekday_ShortSaturday: String { return self._s[580]! } + public var Profile_CreateNewContact: String { return self._s[584]! } + public var AutoDownloadSettings_DataUsageLow: String { return self._s[585]! } + public var SettingsSearch_Synonyms_Notifications_InAppNotificationsPreview: String { return self._s[586]! } + public var Group_Setup_TypePublic: String { return self._s[587]! } + public var Weekday_ShortSaturday: String { return self._s[588]! } public func Time_MonthOfYear_m12(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[581]!, self._r[581]!, [_0]) + return formatWithArgumentRanges(self._s[589]!, self._r[589]!, [_0]) } - public var LiveLocation_MenuStopAll: String { return self._s[582]! } + public var LiveLocation_MenuStopAll: String { return self._s[590]! } public func DialogList_EncryptedChatStartedIncoming(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[583]!, self._r[583]!, [_0]) - } - public var ChatListFolder_NamePlaceholder: String { return self._s[584]! } - public var Channel_OwnershipTransfer_ErrorPublicChannelsTooMuch: String { return self._s[585]! } - public func PUSH_CHAT_MESSAGE_GAME(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[586]!, self._r[586]!, [_1, _2, _3]) - } - public var VoiceChat_ChatFullAlertText: String { return self._s[587]! } - public var Chat_GenericPsaTooltip: String { return self._s[589]! } - public var ChannelInfo_CreateVoiceChat: String { return self._s[590]! } - public func Message_ForwardedMessageShort(_ _0: String) -> (String, [(Int, NSRange)]) { return formatWithArgumentRanges(self._s[591]!, self._r[591]!, [_0]) } - public var PrivacyLastSeenSettings_AlwaysShareWith_Placeholder: String { return self._s[592]! } - public var Login_PhoneAndCountryHelp: String { return self._s[593]! } - public var SaveIncomingPhotosSettings_From: String { return self._s[595]! } - public var Conversation_JumpToDate: String { return self._s[596]! } - public var AuthSessions_AddDevice: String { return self._s[597]! } - public var Settings_FAQ: String { return self._s[599]! } - public var Username_Title: String { return self._s[600]! } - public var DialogList_Read: String { return self._s[601]! } - public var Conversation_InstantPagePreview: String { return self._s[602]! } - public var Login_ResetAccountProtected_Title: String { return self._s[604]! } - public var CallFeedback_ReasonDistortedSpeech: String { return self._s[605]! } - public var Channel_EditAdmin_PermissionChangeInfo: String { return self._s[606]! } + public var ChatListFolder_NamePlaceholder: String { return self._s[592]! } + public var Channel_OwnershipTransfer_ErrorPublicChannelsTooMuch: String { return self._s[593]! } + public func PUSH_CHAT_MESSAGE_GAME(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[594]!, self._r[594]!, [_1, _2, _3]) + } + public var VoiceChat_ChatFullAlertText: String { return self._s[595]! } + public var Chat_GenericPsaTooltip: String { return self._s[597]! } + public var ChannelInfo_CreateVoiceChat: String { return self._s[598]! } + public func Message_ForwardedMessageShort(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[599]!, self._r[599]!, [_0]) + } + public var PrivacyLastSeenSettings_AlwaysShareWith_Placeholder: String { return self._s[600]! } + public var Login_PhoneAndCountryHelp: String { return self._s[601]! } + public var SaveIncomingPhotosSettings_From: String { return self._s[603]! } + public var Conversation_JumpToDate: String { return self._s[604]! } + public var AuthSessions_AddDevice: String { return self._s[605]! } + public var Settings_FAQ: String { return self._s[607]! } + public var Username_Title: String { return self._s[608]! } + public var DialogList_Read: String { return self._s[609]! } + public var Conversation_InstantPagePreview: String { return self._s[610]! } + public var Login_ResetAccountProtected_Title: String { return self._s[612]! } + public var CallFeedback_ReasonDistortedSpeech: String { return self._s[613]! } + public var Channel_EditAdmin_PermissionChangeInfo: String { return self._s[614]! } public func Channel_AdminLog_MessageRankUsername(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[607]!, self._r[607]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[615]!, self._r[615]!, [_1, _2, _3]) } - public var WallpaperPreview_PreviewBottomText: String { return self._s[609]! } - public var Privacy_SecretChatsTitle: String { return self._s[612]! } + public var WallpaperPreview_PreviewBottomText: String { return self._s[617]! } + public var Privacy_SecretChatsTitle: String { return self._s[620]! } public func Notification_PassportValuesSentMessage(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[613]!, self._r[613]!, [_1, _2]) + return formatWithArgumentRanges(self._s[621]!, self._r[621]!, [_1, _2]) } - public var Checkout_NewCard_SaveInfoHelp: String { return self._s[614]! } - public var Conversation_ClousStorageInfo_Description4: String { return self._s[615]! } - public var PasscodeSettings_TurnPasscodeOn: String { return self._s[616]! } - public var Message_ReplyActionButtonShowReceipt: String { return self._s[617]! } + public var Checkout_NewCard_SaveInfoHelp: String { return self._s[622]! } + public var Conversation_ClousStorageInfo_Description4: String { return self._s[623]! } + public var PasscodeSettings_TurnPasscodeOn: String { return self._s[624]! } + public var Message_ReplyActionButtonShowReceipt: String { return self._s[625]! } public func PrivacyPolicy_AgeVerificationMessage(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[618]!, self._r[618]!, [_0]) + return formatWithArgumentRanges(self._s[626]!, self._r[626]!, [_0]) } - public var GroupInfo_DeleteAndExitConfirmation: String { return self._s[620]! } - public var TwoStepAuth_ConfirmationAbort: String { return self._s[621]! } - public var PrivacySettings_LastSeenEverybody: String { return self._s[622]! } - public var CallFeedback_ReasonDropped: String { return self._s[623]! } + public var GroupInfo_DeleteAndExitConfirmation: String { return self._s[628]! } + public var TwoStepAuth_ConfirmationAbort: String { return self._s[629]! } + public var PrivacySettings_LastSeenEverybody: String { return self._s[630]! } + public var CallFeedback_ReasonDropped: String { return self._s[631]! } public func ScheduledMessages_ScheduledDate(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[624]!, self._r[624]!, [_0]) + return formatWithArgumentRanges(self._s[632]!, self._r[632]!, [_0]) } - public var WebSearch_Images: String { return self._s[625]! } - public var Passport_Identity_Surname: String { return self._s[626]! } - public var Channel_Stickers_CreateYourOwn: String { return self._s[627]! } - public var TwoFactorSetup_Email_Title: String { return self._s[628]! } - public var Cache_ClearEmpty: String { return self._s[629]! } - public var AuthSessions_AddDeviceIntro_Action: String { return self._s[630]! } - public var Theme_Context_Apply: String { return self._s[631]! } - public var GroupInfo_Permissions_SearchPlaceholder: String { return self._s[632]! } - public var AutoDownloadSettings_DocumentsTitle: String { return self._s[633]! } + public var WebSearch_Images: String { return self._s[633]! } + public var Passport_Identity_Surname: String { return self._s[634]! } + public var Channel_Stickers_CreateYourOwn: String { return self._s[635]! } + public var TwoFactorSetup_Email_Title: String { return self._s[636]! } + public var Cache_ClearEmpty: String { return self._s[637]! } + public var AuthSessions_AddDeviceIntro_Action: String { return self._s[638]! } + public var Theme_Context_Apply: String { return self._s[639]! } + public var GroupInfo_Permissions_SearchPlaceholder: String { return self._s[640]! } + public var AutoDownloadSettings_DocumentsTitle: String { return self._s[641]! } public func NetworkUsageSettings_CellularUsageSince(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[634]!, self._r[634]!, [_0]) + return formatWithArgumentRanges(self._s[642]!, self._r[642]!, [_0]) } - public var Call_StatusRinging: String { return self._s[635]! } + public var Call_StatusRinging: String { return self._s[643]! } public func Map_DistanceAway(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[636]!, self._r[636]!, [_0]) + return formatWithArgumentRanges(self._s[644]!, self._r[644]!, [_0]) } public func DialogList_SingleTypingSuffix(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[637]!, self._r[637]!, [_0]) + return formatWithArgumentRanges(self._s[645]!, self._r[645]!, [_0]) } - public var Cache_ClearNone: String { return self._s[638]! } - public var PrivacyPolicy_Accept: String { return self._s[639]! } - public var Contacts_PhoneNumber: String { return self._s[640]! } - public var Passport_Identity_OneOfTypePassport: String { return self._s[641]! } - public var PhotoEditor_HighlightsTint: String { return self._s[643]! } - public var AutoDownloadSettings_AutodownloadVideos: String { return self._s[644]! } - public var Checkout_PaymentMethod_Title: String { return self._s[647]! } - public var Month_GenAugust: String { return self._s[649]! } - public var DialogList_Draft: String { return self._s[650]! } - public var ChatList_EmptyChatListFilterText: String { return self._s[651]! } - public var PeopleNearby_Description: String { return self._s[652]! } - public var WallpaperPreview_SwipeColorsBottomText: String { return self._s[653]! } - public var SettingsSearch_Synonyms_Privacy_Data_TopPeers: String { return self._s[655]! } - public var Watch_Message_ForwardedFrom: String { return self._s[656]! } - public var Notification_Mute1h: String { return self._s[657]! } - public var Appearance_ThemePreview_Chat_3_TextWithLink: String { return self._s[658]! } - public var SettingsSearch_Synonyms_Privacy_AuthSessions: String { return self._s[660]! } - public var Channel_Edit_LinkItem: String { return self._s[661]! } - public var Presence_online: String { return self._s[662]! } - public var AutoDownloadSettings_Title: String { return self._s[663]! } - public var Conversation_MessageDialogRetry: String { return self._s[664]! } - public var SettingsSearch_Synonyms_ChatSettings_OpenLinksIn: String { return self._s[666]! } - public var Channel_About_Placeholder: String { return self._s[668]! } - public var Passport_Language_sl: String { return self._s[669]! } - public var AppleWatch_Title: String { return self._s[671]! } - public var RepliesChat_DescriptionText: String { return self._s[673]! } - public var Stats_Message_PrivateShares: String { return self._s[674]! } - public var Settings_ViewPhoto: String { return self._s[675]! } - public var ChatList_DeleteSavedMessagesConfirmation: String { return self._s[676]! } - public var Cache_ClearProgress: String { return self._s[677]! } - public var Cache_Music: String { return self._s[678]! } - public var Conversation_ContextMenuShare: String { return self._s[680]! } - public var AutoDownloadSettings_Unlimited: String { return self._s[681]! } - public var Channel_OwnershipTransfer_ErrorPrivacyRestricted: String { return self._s[682]! } - public var Contacts_PermissionsAllow: String { return self._s[683]! } - public var Passport_Language_vi: String { return self._s[685]! } + public var Cache_ClearNone: String { return self._s[646]! } + public var PrivacyPolicy_Accept: String { return self._s[647]! } + public var Contacts_PhoneNumber: String { return self._s[648]! } + public var Passport_Identity_OneOfTypePassport: String { return self._s[649]! } + public var PhotoEditor_HighlightsTint: String { return self._s[651]! } + public var AutoDownloadSettings_AutodownloadVideos: String { return self._s[652]! } + public var Checkout_PaymentMethod_Title: String { return self._s[655]! } + public var Month_GenAugust: String { return self._s[657]! } + public var DialogList_Draft: String { return self._s[658]! } + public var ChatList_EmptyChatListFilterText: String { return self._s[659]! } + public var PeopleNearby_Description: String { return self._s[660]! } + public var WallpaperPreview_SwipeColorsBottomText: String { return self._s[661]! } + public var SettingsSearch_Synonyms_Privacy_Data_TopPeers: String { return self._s[663]! } + public var Watch_Message_ForwardedFrom: String { return self._s[664]! } + public var Notification_Mute1h: String { return self._s[665]! } + public var Appearance_ThemePreview_Chat_3_TextWithLink: String { return self._s[666]! } + public var SettingsSearch_Synonyms_Privacy_AuthSessions: String { return self._s[668]! } + public var Channel_Edit_LinkItem: String { return self._s[669]! } + public var Presence_online: String { return self._s[670]! } + public var AutoDownloadSettings_Title: String { return self._s[671]! } + public var Conversation_MessageDialogRetry: String { return self._s[672]! } + public var SettingsSearch_Synonyms_ChatSettings_OpenLinksIn: String { return self._s[674]! } + public var Channel_About_Placeholder: String { return self._s[676]! } + public var Passport_Language_sl: String { return self._s[677]! } + public var AppleWatch_Title: String { return self._s[679]! } + public var RepliesChat_DescriptionText: String { return self._s[681]! } + public var Stats_Message_PrivateShares: String { return self._s[682]! } + public var Settings_ViewPhoto: String { return self._s[683]! } + public var ChatList_DeleteSavedMessagesConfirmation: String { return self._s[684]! } + public var Cache_ClearProgress: String { return self._s[685]! } + public var Cache_Music: String { return self._s[686]! } + public var Conversation_ContextMenuShare: String { return self._s[688]! } + public var AutoDownloadSettings_Unlimited: String { return self._s[689]! } + public var Channel_OwnershipTransfer_ErrorPrivacyRestricted: String { return self._s[690]! } + public var Contacts_PermissionsAllow: String { return self._s[691]! } + public var Passport_Language_vi: String { return self._s[693]! } public func PUSH_MESSAGE_TEXT(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[688]!, self._r[688]!, [_1, _2]) + return formatWithArgumentRanges(self._s[696]!, self._r[696]!, [_1, _2]) } - public var Passport_Language_de: String { return self._s[689]! } - public var Notifications_PermissionsText: String { return self._s[691]! } - public var GroupRemoved_AddToGroup: String { return self._s[692]! } - public var Appearance_ThemePreview_ChatList_4_Text: String { return self._s[693]! } - public var ChangePhoneNumberCode_RequestingACall: String { return self._s[694]! } - public var Login_TermsOfServiceAgree: String { return self._s[695]! } - public var VoiceOver_Navigation_ProxySettings: String { return self._s[696]! } + public var Passport_Language_de: String { return self._s[697]! } + public var Notifications_PermissionsText: String { return self._s[699]! } + public var GroupRemoved_AddToGroup: String { return self._s[700]! } + public var Appearance_ThemePreview_ChatList_4_Text: String { return self._s[701]! } + public var ChangePhoneNumberCode_RequestingACall: String { return self._s[702]! } + public var Login_TermsOfServiceAgree: String { return self._s[703]! } + public var VoiceOver_Navigation_ProxySettings: String { return self._s[704]! } public func PUSH_CHAT_JOINED(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[697]!, self._r[697]!, [_1, _2]) + return formatWithArgumentRanges(self._s[705]!, self._r[705]!, [_1, _2]) } - public var SettingsSearch_Synonyms_Data_CallsUseLessData: String { return self._s[699]! } + public var SettingsSearch_Synonyms_Data_CallsUseLessData: String { return self._s[707]! } public func PUSH_CHAT_VOICECHAT_START(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[700]!, self._r[700]!, [_1, _2]) + return formatWithArgumentRanges(self._s[708]!, self._r[708]!, [_1, _2]) } - public var ChatListFolder_NameGroups: String { return self._s[701]! } - public var SocksProxySetup_ProxyDetailsTitle: String { return self._s[702]! } + public var ChatListFolder_NameGroups: String { return self._s[709]! } + public var SocksProxySetup_ProxyDetailsTitle: String { return self._s[710]! } public func Channel_AdminLog_MessageChangedLinkedGroup(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[703]!, self._r[703]!, [_1, _2]) + return formatWithArgumentRanges(self._s[711]!, self._r[711]!, [_1, _2]) } - public var Watch_Suggestion_TalkLater: String { return self._s[704]! } - public var Checkout_ShippingOption_Title: String { return self._s[705]! } - public var Conversation_TitleRepliesEmpty: String { return self._s[706]! } - public var CreatePoll_TextHeader: String { return self._s[707]! } - public var VoiceOver_Chat_Message: String { return self._s[709]! } - public var InfoPlist_NSLocationWhenInUseUsageDescription: String { return self._s[710]! } - public var ContactInfo_Note: String { return self._s[712]! } - public var Channel_AdminLog_InfoPanelAlertText: String { return self._s[713]! } - public var Checkout_NewCard_CardholderNameTitle: String { return self._s[714]! } - public var AutoDownloadSettings_Photos: String { return self._s[715]! } - public var UserInfo_NotificationsDefaultDisabled: String { return self._s[716]! } - public var Channel_Info_Subscribers: String { return self._s[717]! } - public var ChatList_DeleteForCurrentUser: String { return self._s[718]! } - public var ChatListFolderSettings_FoldersSection: String { return self._s[719]! } + public var Watch_Suggestion_TalkLater: String { return self._s[712]! } + public var Checkout_ShippingOption_Title: String { return self._s[713]! } + public var Conversation_TitleRepliesEmpty: String { return self._s[714]! } + public var CreatePoll_TextHeader: String { return self._s[715]! } + public var VoiceOver_Chat_Message: String { return self._s[717]! } + public var InfoPlist_NSLocationWhenInUseUsageDescription: String { return self._s[718]! } + public var ContactInfo_Note: String { return self._s[720]! } + public var Channel_AdminLog_InfoPanelAlertText: String { return self._s[721]! } + public var Checkout_NewCard_CardholderNameTitle: String { return self._s[722]! } + public var AutoDownloadSettings_Photos: String { return self._s[723]! } + public var UserInfo_NotificationsDefaultDisabled: String { return self._s[724]! } + public var Channel_Info_Subscribers: String { return self._s[725]! } + public var ChatList_DeleteForCurrentUser: String { return self._s[726]! } + public var ChatListFolderSettings_FoldersSection: String { return self._s[727]! } public func Time_PreciseDate_m9(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[723]!, self._r[723]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[731]!, self._r[731]!, [_1, _2, _3]) } - public var AutoNightTheme_System: String { return self._s[724]! } - public var Call_StatusWaiting: String { return self._s[725]! } - public var GroupInfo_GroupHistoryHidden: String { return self._s[726]! } + public var AutoNightTheme_System: String { return self._s[732]! } + public var Call_StatusWaiting: String { return self._s[733]! } + public var GroupInfo_GroupHistoryHidden: String { return self._s[734]! } public func CHAT_MESSAGE_INVOICE(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[727]!, self._r[727]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[735]!, self._r[735]!, [_1, _2, _3]) } - public var Conversation_ContextMenuCopy: String { return self._s[729]! } - public var Notifications_MessageNotificationsPreview: String { return self._s[730]! } - public var Notifications_InAppNotificationsVibrate: String { return self._s[731]! } + public var Conversation_ContextMenuCopy: String { return self._s[737]! } + public var Notifications_MessageNotificationsPreview: String { return self._s[738]! } + public var Notifications_InAppNotificationsVibrate: String { return self._s[739]! } public func Conversation_RestrictedTextTimed(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[732]!, self._r[732]!, [_0]) + return formatWithArgumentRanges(self._s[740]!, self._r[740]!, [_0]) } - public var Group_Status: String { return self._s[734]! } - public var Group_Setup_HistoryVisible: String { return self._s[735]! } - public var Conversation_DiscardVoiceMessageAction: String { return self._s[736]! } - public var Paint_Edit: String { return self._s[737]! } - public var Channel_EditAdmin_CannotEdit: String { return self._s[739]! } - public var Username_InvalidTooShort: String { return self._s[740]! } - public var ClearCache_StorageOtherApps: String { return self._s[741]! } - public var Conversation_ViewMessage: String { return self._s[742]! } - public var GroupInfo_PublicLinkAdd: String { return self._s[744]! } + public var Group_Status: String { return self._s[742]! } + public var Group_Setup_HistoryVisible: String { return self._s[743]! } + public var Conversation_DiscardVoiceMessageAction: String { return self._s[744]! } + public var Paint_Edit: String { return self._s[745]! } + public var Channel_EditAdmin_CannotEdit: String { return self._s[747]! } + public var Username_InvalidTooShort: String { return self._s[748]! } + public var ClearCache_StorageOtherApps: String { return self._s[749]! } + public var Conversation_ViewMessage: String { return self._s[750]! } + public var GroupInfo_PublicLinkAdd: String { return self._s[752]! } public func Notification_RemovedGroupPhoto(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[745]!, self._r[745]!, [_0]) + return formatWithArgumentRanges(self._s[753]!, self._r[753]!, [_0]) } - public var CallSettings_Title: String { return self._s[746]! } + public var CallSettings_Title: String { return self._s[754]! } public func Conversation_BotInteractiveUrlAlert(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[747]!, self._r[747]!, [_0]) + return formatWithArgumentRanges(self._s[755]!, self._r[755]!, [_0]) } public func VoiceOver_Chat_ContactFrom(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[750]!, self._r[750]!, [_0]) + return formatWithArgumentRanges(self._s[758]!, self._r[758]!, [_0]) } - public var PUSH_SENDER_YOU: String { return self._s[753]! } - public var Profile_ShareContactButton: String { return self._s[754]! } - public var GroupInfo_Permissions_SectionTitle: String { return self._s[755]! } - public var Map_ShareLiveLocation: String { return self._s[756]! } - public var ChatListFolder_TitleEdit: String { return self._s[757]! } - public var Passport_Address_Address: String { return self._s[759]! } - public var LastSeen_JustNow: String { return self._s[761]! } + public var PUSH_SENDER_YOU: String { return self._s[761]! } + public var Profile_ShareContactButton: String { return self._s[762]! } + public var GroupInfo_Permissions_SectionTitle: String { return self._s[763]! } + public var Map_ShareLiveLocation: String { return self._s[764]! } + public var ChatListFolder_TitleEdit: String { return self._s[765]! } + public var Passport_Address_Address: String { return self._s[767]! } + public var LastSeen_JustNow: String { return self._s[769]! } public func SecretImage_NotViewedYet(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[762]!, self._r[762]!, [_0]) + return formatWithArgumentRanges(self._s[770]!, self._r[770]!, [_0]) } - public var ContactInfo_PhoneLabelOther: String { return self._s[763]! } - public var PasscodeSettings_DoNotMatch: String { return self._s[764]! } - public var Weekday_Today: String { return self._s[767]! } - public var DialogList_Title: String { return self._s[768]! } - public var SettingsSearch_Synonyms_Notifications_MessageNotificationsPreview: String { return self._s[769]! } - public var Cache_ClearCache: String { return self._s[770]! } - public var CreatePoll_ExplanationInfo: String { return self._s[771]! } - public var Notifications_ResetAllNotificationsHelp: String { return self._s[773]! } - public var Stats_MessageTitle: String { return self._s[774]! } - public var Passport_Address_Street: String { return self._s[776]! } + public var ContactInfo_PhoneLabelOther: String { return self._s[771]! } + public var PasscodeSettings_DoNotMatch: String { return self._s[772]! } + public var Weekday_Today: String { return self._s[775]! } + public var DialogList_Title: String { return self._s[776]! } + public var SettingsSearch_Synonyms_Notifications_MessageNotificationsPreview: String { return self._s[777]! } + public var Cache_ClearCache: String { return self._s[778]! } + public var CreatePoll_ExplanationInfo: String { return self._s[779]! } + public var Notifications_ResetAllNotificationsHelp: String { return self._s[781]! } + public var Stats_MessageTitle: String { return self._s[782]! } + public var Passport_Address_Street: String { return self._s[784]! } public func Channel_AdminLog_MessageRemovedGroupUsername(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[777]!, self._r[777]!, [_0]) + return formatWithArgumentRanges(self._s[785]!, self._r[785]!, [_0]) } - public var Channel_AdminLog_ChannelEmptyText: String { return self._s[778]! } + public var Channel_AdminLog_ChannelEmptyText: String { return self._s[786]! } public func Login_PhoneGenericEmailSubject(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[779]!, self._r[779]!, [_0]) + return formatWithArgumentRanges(self._s[787]!, self._r[787]!, [_0]) } - public var TwoStepAuth_Email: String { return self._s[781]! } - public var Conversation_SecretLinkPreviewAlert: String { return self._s[782]! } - public var PrivacySettings_PasscodeOn: String { return self._s[783]! } - public var Camera_SquareMode: String { return self._s[785]! } - public var SocksProxySetup_Port: String { return self._s[786]! } - public var Watch_LastSeen_JustNow: String { return self._s[788]! } + public var TwoStepAuth_Email: String { return self._s[789]! } + public var Conversation_SecretLinkPreviewAlert: String { return self._s[790]! } + public var PrivacySettings_PasscodeOn: String { return self._s[791]! } + public var Camera_SquareMode: String { return self._s[793]! } + public var SocksProxySetup_Port: String { return self._s[794]! } + public var Watch_LastSeen_JustNow: String { return self._s[796]! } public func Location_ProximityAlertSetText(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[789]!, self._r[789]!, [_1, _2]) + return formatWithArgumentRanges(self._s[797]!, self._r[797]!, [_1, _2]) } public func PUSH_MESSAGE_GAME(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[790]!, self._r[790]!, [_1, _2]) + return formatWithArgumentRanges(self._s[798]!, self._r[798]!, [_1, _2]) } public func Watch_LastSeen_YesterdayAt(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[791]!, self._r[791]!, [_0]) + return formatWithArgumentRanges(self._s[799]!, self._r[799]!, [_0]) } - public var EditTheme_Expand_Preview_OutgoingText: String { return self._s[792]! } - public var Channel_AdminLogFilter_EventsTitle: String { return self._s[793]! } - public var Watch_Suggestion_HoldOn: String { return self._s[796]! } + public var EditTheme_Expand_Preview_OutgoingText: String { return self._s[800]! } + public var Channel_AdminLogFilter_EventsTitle: String { return self._s[801]! } + public var Watch_Suggestion_HoldOn: String { return self._s[804]! } public func PUSH_CHANNEL_MESSAGE_GEOLIVE(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[797]!, self._r[797]!, [_1]) + return formatWithArgumentRanges(self._s[805]!, self._r[805]!, [_1]) } - public var CallSettings_TabIcon: String { return self._s[798]! } - public var ScheduledMessages_SendNow: String { return self._s[799]! } - public var Stats_GroupTopWeekdaysTitle: String { return self._s[800]! } - public var UserInfo_PhoneCall: String { return self._s[801]! } - public var Month_GenMarch: String { return self._s[802]! } - public var Camera_Discard: String { return self._s[803]! } - public var InfoPlist_NSFaceIDUsageDescription: String { return self._s[804]! } - public var Passport_RequestedInformation: String { return self._s[805]! } + public var CallSettings_TabIcon: String { return self._s[806]! } + public var ScheduledMessages_SendNow: String { return self._s[807]! } + public var Stats_GroupTopWeekdaysTitle: String { return self._s[808]! } + public var UserInfo_PhoneCall: String { return self._s[809]! } + public var Month_GenMarch: String { return self._s[810]! } + public var Camera_Discard: String { return self._s[811]! } + public var InfoPlist_NSFaceIDUsageDescription: String { return self._s[812]! } + public var Passport_RequestedInformation: String { return self._s[813]! } public func Notification_ProximityYouReached(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[807]!, self._r[807]!, [_1, _2]) + return formatWithArgumentRanges(self._s[815]!, self._r[815]!, [_1, _2]) } - public var Passport_Language_ro: String { return self._s[808]! } + public var Passport_Language_ro: String { return self._s[816]! } public func PUSH_CHAT_MESSAGE_DOC(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[809]!, self._r[809]!, [_1, _2]) + return formatWithArgumentRanges(self._s[817]!, self._r[817]!, [_1, _2]) } - public var AutoDownloadSettings_ResetHelp: String { return self._s[810]! } - public var Passport_Identity_DocumentDetails: String { return self._s[812]! } - public var Passport_Address_ScansHelp: String { return self._s[813]! } - public var Location_LiveLocationRequired_Title: String { return self._s[814]! } - public var ClearCache_StorageCache: String { return self._s[815]! } - public var Theme_Colors_ColorWallpaperWarningProceed: String { return self._s[816]! } - public var Conversation_RestrictedText: String { return self._s[817]! } - public var Notifications_MessageNotifications: String { return self._s[819]! } - public var Passport_Scans: String { return self._s[820]! } - public var TwoStepAuth_SetupHintTitle: String { return self._s[822]! } - public var LogoutOptions_ContactSupportTitle: String { return self._s[823]! } - public var Passport_Identity_SelfieHelp: String { return self._s[824]! } - public var Permissions_NotificationsUnreachableText_v0: String { return self._s[825]! } - public var Privacy_PaymentsClear_PaymentInfo: String { return self._s[826]! } - public var ShareMenu_CopyShareLinkGame: String { return self._s[827]! } - public var PeerInfo_ButtonSearch: String { return self._s[828]! } + public var AutoDownloadSettings_ResetHelp: String { return self._s[818]! } + public var Passport_Identity_DocumentDetails: String { return self._s[820]! } + public var Passport_Address_ScansHelp: String { return self._s[821]! } + public var Location_LiveLocationRequired_Title: String { return self._s[822]! } + public var ClearCache_StorageCache: String { return self._s[823]! } + public var Theme_Colors_ColorWallpaperWarningProceed: String { return self._s[824]! } + public var Conversation_RestrictedText: String { return self._s[825]! } + public var Notifications_MessageNotifications: String { return self._s[827]! } + public var Passport_Scans: String { return self._s[828]! } + public var TwoStepAuth_SetupHintTitle: String { return self._s[830]! } + public var LogoutOptions_ContactSupportTitle: String { return self._s[831]! } + public var Passport_Identity_SelfieHelp: String { return self._s[832]! } + public var Permissions_NotificationsUnreachableText_v0: String { return self._s[833]! } + public var Privacy_PaymentsClear_PaymentInfo: String { return self._s[834]! } + public var ShareMenu_CopyShareLinkGame: String { return self._s[835]! } + public var PeerInfo_ButtonSearch: String { return self._s[836]! } public func Notification_ProximityReachedYou(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[831]!, self._r[831]!, [_1, _2]) + return formatWithArgumentRanges(self._s[839]!, self._r[839]!, [_1, _2]) } - public var SettingsSearch_Synonyms_Privacy_Data_ClearPaymentsInfo: String { return self._s[832]! } - public var Passport_FieldIdentityTranslationHelp: String { return self._s[834]! } - public var Conversation_InputTextSilentBroadcastPlaceholder: String { return self._s[835]! } - public var Month_GenSeptember: String { return self._s[836]! } + public var SettingsSearch_Synonyms_Privacy_Data_ClearPaymentsInfo: String { return self._s[840]! } + public var Passport_FieldIdentityTranslationHelp: String { return self._s[842]! } + public var Conversation_InputTextSilentBroadcastPlaceholder: String { return self._s[843]! } + public var Month_GenSeptember: String { return self._s[844]! } public func Call_GroupFormat(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[838]!, self._r[838]!, [_1, _2]) + return formatWithArgumentRanges(self._s[846]!, self._r[846]!, [_1, _2]) } - public var StickerPacksSettings_ArchivedPacks: String { return self._s[839]! } + public var StickerPacksSettings_ArchivedPacks: String { return self._s[847]! } public func Notification_VoiceChatInvitation(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[841]!, self._r[841]!, [_1, _2]) + return formatWithArgumentRanges(self._s[849]!, self._r[849]!, [_1, _2]) } public func Channel_Username_LinkHint(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[842]!, self._r[842]!, [_0]) + return formatWithArgumentRanges(self._s[850]!, self._r[850]!, [_0]) } public func PUSH_PINNED_CONTACT(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[844]!, self._r[844]!, [_1, _2]) + return formatWithArgumentRanges(self._s[852]!, self._r[852]!, [_1, _2]) } public func PUSH_MESSAGE_VIDEOS(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[845]!, self._r[845]!, [_1, _2]) + return formatWithArgumentRanges(self._s[853]!, self._r[853]!, [_1, _2]) } - public var Calls_NotNow: String { return self._s[847]! } - public var Settings_ChatFolders: String { return self._s[851]! } - public var Login_PadPhoneHelpTitle: String { return self._s[852]! } - public var TwoStepAuth_EnterPasswordInvalid: String { return self._s[853]! } - public var Settings_ChatBackground: String { return self._s[854]! } + public var Calls_NotNow: String { return self._s[855]! } + public var Settings_ChatFolders: String { return self._s[859]! } + public var Login_PadPhoneHelpTitle: String { return self._s[860]! } + public var TwoStepAuth_EnterPasswordInvalid: String { return self._s[861]! } + public var Settings_ChatBackground: String { return self._s[862]! } public func PUSH_CHAT_MESSAGE_CONTACT(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[856]!, self._r[856]!, [_1, _2]) + return formatWithArgumentRanges(self._s[864]!, self._r[864]!, [_1, _2]) } - public var ProxyServer_VoiceOver_Active: String { return self._s[857]! } - public var Call_StatusBusy: String { return self._s[858]! } - public var Conversation_MessageDeliveryFailed: String { return self._s[859]! } - public var Login_NetworkError: String { return self._s[861]! } - public var TwoStepAuth_SetupPasswordDescription: String { return self._s[862]! } - public var Privacy_Calls_Integration: String { return self._s[863]! } - public var DialogList_SearchSectionMessages: String { return self._s[864]! } - public var AutoDownloadSettings_VideosTitle: String { return self._s[865]! } - public var Preview_DeletePhoto: String { return self._s[866]! } - public var PrivacySettings_PhoneNumber: String { return self._s[868]! } - public var Forward_ErrorDisabledForChat: String { return self._s[869]! } - public var Watch_Compose_CurrentLocation: String { return self._s[870]! } - public var Settings_CallSettings: String { return self._s[871]! } - public var AutoDownloadSettings_TypePrivateChats: String { return self._s[872]! } - public var ChatList_Context_MarkAllAsRead: String { return self._s[873]! } - public var ChatSettings_AutoPlayAnimations: String { return self._s[874]! } - public var SaveIncomingPhotosSettings_Title: String { return self._s[875]! } - public var OwnershipTransfer_SecurityRequirements: String { return self._s[876]! } - public var Map_LiveLocationFor1Hour: String { return self._s[877]! } + public var ProxyServer_VoiceOver_Active: String { return self._s[865]! } + public var Call_StatusBusy: String { return self._s[866]! } + public var Conversation_MessageDeliveryFailed: String { return self._s[867]! } + public var Login_NetworkError: String { return self._s[869]! } + public var TwoStepAuth_SetupPasswordDescription: String { return self._s[870]! } + public var Privacy_Calls_Integration: String { return self._s[871]! } + public var DialogList_SearchSectionMessages: String { return self._s[872]! } + public var AutoDownloadSettings_VideosTitle: String { return self._s[873]! } + public var Preview_DeletePhoto: String { return self._s[874]! } + public var PrivacySettings_PhoneNumber: String { return self._s[876]! } + public var Forward_ErrorDisabledForChat: String { return self._s[877]! } + public var Watch_Compose_CurrentLocation: String { return self._s[878]! } + public var Settings_CallSettings: String { return self._s[879]! } + public var AutoDownloadSettings_TypePrivateChats: String { return self._s[880]! } + public var ChatList_Context_MarkAllAsRead: String { return self._s[881]! } + public var ChatSettings_AutoPlayAnimations: String { return self._s[882]! } + public var SaveIncomingPhotosSettings_Title: String { return self._s[883]! } + public var OwnershipTransfer_SecurityRequirements: String { return self._s[884]! } + public var Map_LiveLocationFor1Hour: String { return self._s[885]! } public func Privacy_GroupsAndChannels_InviteToGroupError(_ _0: String, _ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[878]!, self._r[878]!, [_0, _1]) + return formatWithArgumentRanges(self._s[886]!, self._r[886]!, [_0, _1]) } public func Notification_PinnedLiveLocationMessage(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[879]!, self._r[879]!, [_0]) + return formatWithArgumentRanges(self._s[887]!, self._r[887]!, [_0]) } - public var Conversation_UnvotePoll: String { return self._s[880]! } - public var TwoStepAuth_EnterEmailCode: String { return self._s[881]! } + public var Conversation_UnvotePoll: String { return self._s[888]! } + public var TwoStepAuth_EnterEmailCode: String { return self._s[889]! } public func LOCAL_MESSAGE_FWDS(_ _1: String, _ _2: Int) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[882]!, self._r[882]!, [_1, "\(_2)"]) + return formatWithArgumentRanges(self._s[890]!, self._r[890]!, [_1, "\(_2)"]) } - public var Passport_InfoTitle: String { return self._s[883]! } + public var Passport_InfoTitle: String { return self._s[891]! } public func Conversation_Bytes(_ _0: Int) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[884]!, self._r[884]!, ["\(_0)"]) + return formatWithArgumentRanges(self._s[892]!, self._r[892]!, ["\(_0)"]) } - public var AccentColor_Title: String { return self._s[885]! } + public var AccentColor_Title: String { return self._s[893]! } public func PUSH_MESSAGE_INVOICE(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[886]!, self._r[886]!, [_1, _2]) + return formatWithArgumentRanges(self._s[894]!, self._r[894]!, [_1, _2]) } public func Notification_JoinedChannel(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[889]!, self._r[889]!, [_0]) + return formatWithArgumentRanges(self._s[897]!, self._r[897]!, [_0]) } - public var AutoDownloadSettings_DataUsageCustom: String { return self._s[890]! } - public var Conversation_ShareBotLocationConfirmation: String { return self._s[891]! } - public var PrivacyPhoneNumberSettings_WhoCanSeeMyPhoneNumber: String { return self._s[892]! } - public var VoiceOver_Editing_ClearText: String { return self._s[893]! } - public var Conversation_Unarchive: String { return self._s[894]! } - public var Notification_CallOutgoing: String { return self._s[895]! } - public var Channel_Setup_PublicNoLink: String { return self._s[896]! } - public var Passport_Identity_GenderPlaceholder: String { return self._s[897]! } - public var Message_Animation: String { return self._s[898]! } - public var SettingsSearch_Synonyms_Appearance_Animations: String { return self._s[899]! } - public var ChatSettings_ConnectionType_Title: String { return self._s[900]! } + public var AutoDownloadSettings_DataUsageCustom: String { return self._s[898]! } + public var Conversation_ShareBotLocationConfirmation: String { return self._s[899]! } + public var PrivacyPhoneNumberSettings_WhoCanSeeMyPhoneNumber: String { return self._s[900]! } + public var VoiceOver_Editing_ClearText: String { return self._s[901]! } + public var Conversation_Unarchive: String { return self._s[902]! } + public var Notification_CallOutgoing: String { return self._s[903]! } + public var Channel_Setup_PublicNoLink: String { return self._s[904]! } + public var Passport_Identity_GenderPlaceholder: String { return self._s[905]! } + public var Message_Animation: String { return self._s[906]! } + public var SettingsSearch_Synonyms_Appearance_Animations: String { return self._s[907]! } + public var ChatSettings_ConnectionType_Title: String { return self._s[908]! } public func Watch_Time_ShortFullAt(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[901]!, self._r[901]!, [_1, _2]) + return formatWithArgumentRanges(self._s[909]!, self._r[909]!, [_1, _2]) } - public var Notification_CallBack: String { return self._s[903]! } - public var Appearance_Title: String { return self._s[905]! } - public var NotificationsSound_Glass: String { return self._s[907]! } - public var AutoDownloadSettings_CellularTitle: String { return self._s[909]! } - public var Notifications_PermissionsSuppressWarningTitle: String { return self._s[911]! } - public var ChatSearch_SearchPlaceholder: String { return self._s[912]! } - public var Passport_Identity_AddPassport: String { return self._s[913]! } - public var GroupPermission_NoAddMembers: String { return self._s[915]! } - public var ContactList_Context_SendMessage: String { return self._s[916]! } - public var PhotoEditor_GrainTool: String { return self._s[917]! } - public var Settings_CopyPhoneNumber: String { return self._s[918]! } - public var Passport_Address_City: String { return self._s[919]! } - public var ChannelRemoved_RemoveInfo: String { return self._s[920]! } - public var SocksProxySetup_Password: String { return self._s[922]! } - public var Settings_Passport: String { return self._s[923]! } - public var Channel_MessagePhotoUpdated: String { return self._s[925]! } - public var Stats_LanguagesTitle: String { return self._s[926]! } - public var ChatList_PeerTypeGroup: String { return self._s[927]! } - public var Privacy_Calls_P2PHelp: String { return self._s[928]! } - public var VoiceOver_Chat_PollNoVotes: String { return self._s[929]! } - public var Embed_PlayingInPIP: String { return self._s[930]! } - public var BlockedUsers_BlockUser: String { return self._s[933]! } - public var Login_CancelPhoneVerificationContinue: String { return self._s[934]! } + public var Notification_CallBack: String { return self._s[911]! } + public var Appearance_Title: String { return self._s[913]! } + public var NotificationsSound_Glass: String { return self._s[915]! } + public var AutoDownloadSettings_CellularTitle: String { return self._s[917]! } + public var Notifications_PermissionsSuppressWarningTitle: String { return self._s[919]! } + public var ChatSearch_SearchPlaceholder: String { return self._s[920]! } + public var Passport_Identity_AddPassport: String { return self._s[921]! } + public var GroupPermission_NoAddMembers: String { return self._s[923]! } + public var ContactList_Context_SendMessage: String { return self._s[924]! } + public var PhotoEditor_GrainTool: String { return self._s[925]! } + public var Settings_CopyPhoneNumber: String { return self._s[926]! } + public var Passport_Address_City: String { return self._s[927]! } + public var ChannelRemoved_RemoveInfo: String { return self._s[928]! } + public var SocksProxySetup_Password: String { return self._s[930]! } + public var Settings_Passport: String { return self._s[931]! } + public var Channel_MessagePhotoUpdated: String { return self._s[933]! } + public var Stats_LanguagesTitle: String { return self._s[934]! } + public var ChatList_PeerTypeGroup: String { return self._s[935]! } + public var Privacy_Calls_P2PHelp: String { return self._s[936]! } + public var VoiceOver_Chat_PollNoVotes: String { return self._s[937]! } + public var Embed_PlayingInPIP: String { return self._s[938]! } + public var BlockedUsers_BlockUser: String { return self._s[941]! } + public var Login_CancelPhoneVerificationContinue: String { return self._s[942]! } public func PUSH_CHANNEL_MESSAGE(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[935]!, self._r[935]!, [_1]) + return formatWithArgumentRanges(self._s[943]!, self._r[943]!, [_1]) } - public var AuthSessions_LoggedIn: String { return self._s[936]! } - public var Channel_AdminLog_MessagePreviousCaption: String { return self._s[937]! } - public var Activity_UploadingDocument: String { return self._s[938]! } - public var PeopleNearby_NoMembers: String { return self._s[939]! } - public var SettingsSearch_Synonyms_Stickers_Masks: String { return self._s[942]! } - public var ChatSettings_AutoPlayVideos: String { return self._s[943]! } - public var VoiceOver_Chat_OpenLinkHint: String { return self._s[944]! } - public var Settings_ViewVideo: String { return self._s[945]! } - public var Map_ShowPlaces: String { return self._s[947]! } - public var Passport_Phone_UseTelegramNumberHelp: String { return self._s[948]! } - public var SettingsSearch_Synonyms_Appearance_ChatBackground_Custom: String { return self._s[949]! } + public var AuthSessions_LoggedIn: String { return self._s[944]! } + public var Channel_AdminLog_MessagePreviousCaption: String { return self._s[945]! } + public var Activity_UploadingDocument: String { return self._s[946]! } + public var PeopleNearby_NoMembers: String { return self._s[947]! } + public var SettingsSearch_Synonyms_Stickers_Masks: String { return self._s[950]! } + public var ChatSettings_AutoPlayVideos: String { return self._s[951]! } + public var VoiceOver_Chat_OpenLinkHint: String { return self._s[952]! } + public var Settings_ViewVideo: String { return self._s[953]! } + public var Map_ShowPlaces: String { return self._s[955]! } + public var Passport_Phone_UseTelegramNumberHelp: String { return self._s[956]! } + public var InviteLink_Create_Title: String { return self._s[957]! } + public var SettingsSearch_Synonyms_Appearance_ChatBackground_Custom: String { return self._s[958]! } public func PrivacySettings_LastSeenContactsPlus(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[950]!, self._r[950]!, [_0]) + return formatWithArgumentRanges(self._s[959]!, self._r[959]!, [_0]) } - public var Conversation_StatusLeftGroup: String { return self._s[951]! } - public var Theme_Colors_Messages: String { return self._s[952]! } - public var AuthSessions_EmptyText: String { return self._s[953]! } + public var Conversation_StatusLeftGroup: String { return self._s[960]! } + public var Theme_Colors_Messages: String { return self._s[961]! } + public var AuthSessions_EmptyText: String { return self._s[962]! } public func PUSH_MESSAGE_CONTACT(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[954]!, self._r[954]!, [_1]) + return formatWithArgumentRanges(self._s[963]!, self._r[963]!, [_1]) } - public var UserInfo_StartSecretChat: String { return self._s[955]! } - public var ChatListFolderSettings_EditFoldersInfo: String { return self._s[956]! } - public var Channel_Edit_PrivatePublicLinkAlert: String { return self._s[957]! } - public var Conversation_ReportSpamGroupConfirmation: String { return self._s[958]! } - public var Conversation_PrivateMessageLinkCopied: String { return self._s[960]! } - public var PeerInfo_PaneFiles: String { return self._s[961]! } - public var PrivacySettings_AutoArchive: String { return self._s[962]! } - public var Camera_VideoMode: String { return self._s[963]! } - public var NotificationsSound_Alert: String { return self._s[964]! } - public var Privacy_Forwards_NeverAllow_Title: String { return self._s[965]! } - public var Appearance_AutoNightTheme: String { return self._s[966]! } - public var Passport_Language_he: String { return self._s[967]! } - public var Passport_InvalidPasswordError: String { return self._s[968]! } - public var Conversation_PinMessageAlert_OnlyPin: String { return self._s[969]! } - public var UserInfo_InviteBotToGroup: String { return self._s[970]! } - public var Conversation_SilentBroadcastTooltipOff: String { return self._s[971]! } - public var Common_TakePhoto: String { return self._s[972]! } - public var Passport_Email_UseTelegramEmailHelp: String { return self._s[973]! } - public var ChatList_Context_JoinChannel: String { return self._s[974]! } - public var MediaPlayer_UnknownArtist: String { return self._s[975]! } - public var KeyCommand_JumpToPreviousUnreadChat: String { return self._s[978]! } - public var Channel_OwnershipTransfer_Title: String { return self._s[979]! } - public var EditTheme_UploadEditedTheme: String { return self._s[980]! } - public var Settings_SetProfilePhotoOrVideo: String { return self._s[982]! } - public var Passport_FieldOneOf_Delimeter: String { return self._s[983]! } - public var MessagePoll_ViewResults: String { return self._s[984]! } - public var Group_Setup_TypePrivateHelp: String { return self._s[985]! } - public var Passport_Address_OneOfTypeUtilityBill: String { return self._s[986]! } - public var ChatList_Search_ShowLess: String { return self._s[987]! } - public var UserInfo_ShareBot: String { return self._s[988]! } - public var Privacy_Calls_P2P: String { return self._s[990]! } - public var WebBrowser_InAppSafari: String { return self._s[991]! } - public var SharedMedia_EmptyFilesText: String { return self._s[994]! } - public var Channel_AdminLog_MessagePreviousMessage: String { return self._s[995]! } - public var GroupInfo_SetSound: String { return self._s[996]! } - public var Permissions_PeopleNearbyAllowInSettings_v0: String { return self._s[997]! } - public var Channel_AdminLog_MessagePreviousDescription: String { return self._s[998]! } - public var Channel_AdminLogFilter_EventsAll: String { return self._s[999]! } - public var CallSettings_UseLessData: String { return self._s[1000]! } - public var InfoPlist_NSCameraUsageDescription: String { return self._s[1001]! } - public var NotificationsSound_Chord: String { return self._s[1002]! } - public var PhotoEditor_CurvesTool: String { return self._s[1003]! } - public var Appearance_ThemePreview_Chat_2_Text: String { return self._s[1004]! } - public var Resolve_ErrorNotFound: String { return self._s[1005]! } - public var Activity_PlayingGame: String { return self._s[1006]! } + public var UserInfo_StartSecretChat: String { return self._s[964]! } + public var ChatListFolderSettings_EditFoldersInfo: String { return self._s[965]! } + public var Channel_Edit_PrivatePublicLinkAlert: String { return self._s[966]! } + public var Conversation_ReportSpamGroupConfirmation: String { return self._s[967]! } + public var Conversation_PrivateMessageLinkCopied: String { return self._s[969]! } + public var PeerInfo_PaneFiles: String { return self._s[970]! } + public var PrivacySettings_AutoArchive: String { return self._s[971]! } + public var Camera_VideoMode: String { return self._s[972]! } + public var NotificationsSound_Alert: String { return self._s[973]! } + public var Privacy_Forwards_NeverAllow_Title: String { return self._s[974]! } + public var Appearance_AutoNightTheme: String { return self._s[975]! } + public var Passport_Language_he: String { return self._s[976]! } + public var Passport_InvalidPasswordError: String { return self._s[977]! } + public var Conversation_PinMessageAlert_OnlyPin: String { return self._s[978]! } + public var UserInfo_InviteBotToGroup: String { return self._s[979]! } + public var Conversation_SilentBroadcastTooltipOff: String { return self._s[980]! } + public var Common_TakePhoto: String { return self._s[981]! } + public var Passport_Email_UseTelegramEmailHelp: String { return self._s[982]! } + public var ChatList_Context_JoinChannel: String { return self._s[983]! } + public var MediaPlayer_UnknownArtist: String { return self._s[984]! } + public var KeyCommand_JumpToPreviousUnreadChat: String { return self._s[987]! } + public var Channel_OwnershipTransfer_Title: String { return self._s[988]! } + public var EditTheme_UploadEditedTheme: String { return self._s[989]! } + public var Settings_SetProfilePhotoOrVideo: String { return self._s[991]! } + public var Passport_FieldOneOf_Delimeter: String { return self._s[992]! } + public var MessagePoll_ViewResults: String { return self._s[993]! } + public var Group_Setup_TypePrivateHelp: String { return self._s[994]! } + public var Passport_Address_OneOfTypeUtilityBill: String { return self._s[995]! } + public var ChatList_Search_ShowLess: String { return self._s[996]! } + public var InviteLink_Create_UsersLimitNoLimit: String { return self._s[997]! } + public var UserInfo_ShareBot: String { return self._s[998]! } + public var Privacy_Calls_P2P: String { return self._s[1000]! } + public var WebBrowser_InAppSafari: String { return self._s[1001]! } + public var SharedMedia_EmptyFilesText: String { return self._s[1004]! } + public var Channel_AdminLog_MessagePreviousMessage: String { return self._s[1005]! } + public var GroupInfo_SetSound: String { return self._s[1006]! } + public var Permissions_PeopleNearbyAllowInSettings_v0: String { return self._s[1007]! } + public var Channel_AdminLog_MessagePreviousDescription: String { return self._s[1008]! } + public var Channel_AdminLogFilter_EventsAll: String { return self._s[1009]! } + public var CallSettings_UseLessData: String { return self._s[1010]! } + public var InfoPlist_NSCameraUsageDescription: String { return self._s[1011]! } + public var NotificationsSound_Chord: String { return self._s[1012]! } + public var PhotoEditor_CurvesTool: String { return self._s[1013]! } + public var Appearance_ThemePreview_Chat_2_Text: String { return self._s[1014]! } + public var Resolve_ErrorNotFound: String { return self._s[1015]! } + public var Activity_PlayingGame: String { return self._s[1016]! } public func VoiceChat_InvitedPeerText(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1009]!, self._r[1009]!, [_0]) + return formatWithArgumentRanges(self._s[1019]!, self._r[1019]!, [_0]) } - public var StickerPacksSettings_AnimatedStickersInfo: String { return self._s[1010]! } + public var StickerPacksSettings_AnimatedStickersInfo: String { return self._s[1020]! } public func PUSH_CHANNEL_MESSAGE_GEO(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1011]!, self._r[1011]!, [_1]) + return formatWithArgumentRanges(self._s[1021]!, self._r[1021]!, [_1]) } - public var Conversation_ShareBotContactConfirmationTitle: String { return self._s[1012]! } - public var Notification_CallIncoming: String { return self._s[1013]! } - public var Stats_EnabledNotifications: String { return self._s[1014]! } - public var Notifications_PermissionsOpenSettings: String { return self._s[1015]! } - public var Checkout_ErrorProviderAccountTimeout: String { return self._s[1016]! } + public var Conversation_ShareBotContactConfirmationTitle: String { return self._s[1022]! } + public var Notification_CallIncoming: String { return self._s[1023]! } + public var Stats_EnabledNotifications: String { return self._s[1024]! } + public var Notifications_PermissionsOpenSettings: String { return self._s[1025]! } + public var Checkout_ErrorProviderAccountTimeout: String { return self._s[1026]! } public func Activity_RemindAboutChannel(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1017]!, self._r[1017]!, [_0]) + return formatWithArgumentRanges(self._s[1027]!, self._r[1027]!, [_0]) } - public var VoiceChat_StatusMutedYou: String { return self._s[1018]! } - public var VoiceOver_Chat_ReplyToYourMessage: String { return self._s[1019]! } - public var Channel_DiscussionGroup_MakeHistoryPublic: String { return self._s[1020]! } - public var StickerPacksSettings_Title: String { return self._s[1021]! } + public var VoiceChat_StatusMutedYou: String { return self._s[1028]! } + public var VoiceOver_Chat_ReplyToYourMessage: String { return self._s[1029]! } + public var Channel_DiscussionGroup_MakeHistoryPublic: String { return self._s[1030]! } + public var StickerPacksSettings_Title: String { return self._s[1031]! } public func Channel_AdminLog_MessageGroupPreHistoryVisible(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1022]!, self._r[1022]!, [_0]) + return formatWithArgumentRanges(self._s[1032]!, self._r[1032]!, [_0]) } - public var Watch_NoConnection: String { return self._s[1023]! } - public var EncryptionKey_Title: String { return self._s[1024]! } - public var Widget_AuthRequired: String { return self._s[1025]! } + public var Watch_NoConnection: String { return self._s[1033]! } + public var EncryptionKey_Title: String { return self._s[1034]! } + public var Widget_AuthRequired: String { return self._s[1035]! } public func PUSH_MESSAGE_ROUND(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1026]!, self._r[1026]!, [_1]) + return formatWithArgumentRanges(self._s[1036]!, self._r[1036]!, [_1]) } - public var Notifications_ExceptionsTitle: String { return self._s[1027]! } - public var EditTheme_Expand_TopInfo: String { return self._s[1028]! } + public var Notifications_ExceptionsTitle: String { return self._s[1037]! } + public var EditTheme_Expand_TopInfo: String { return self._s[1038]! } public func Contacts_AddPhoneNumber(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1029]!, self._r[1029]!, [_0]) + return formatWithArgumentRanges(self._s[1039]!, self._r[1039]!, [_0]) } - public var Channel_AdminLogFilter_EventsRestrictions: String { return self._s[1031]! } - public var Notifications_GroupNotificationsSound: String { return self._s[1032]! } - public var VoiceChat_SpeakPermissionAdmin: String { return self._s[1033]! } - public var Passport_Email_EnterOtherEmail: String { return self._s[1034]! } + public var Channel_AdminLogFilter_EventsRestrictions: String { return self._s[1041]! } + public var Notifications_GroupNotificationsSound: String { return self._s[1042]! } + public var VoiceChat_SpeakPermissionAdmin: String { return self._s[1043]! } + public var Passport_Email_EnterOtherEmail: String { return self._s[1044]! } public func VoiceChat_RemovePeerConfirmation(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1037]!, self._r[1037]!, [_0]) + return formatWithArgumentRanges(self._s[1047]!, self._r[1047]!, [_0]) } - public var Conversation_AddToContacts: String { return self._s[1038]! } - public var AutoDownloadSettings_DataUsageMedium: String { return self._s[1039]! } - public var AuthSessions_LogOutApplications: String { return self._s[1041]! } - public var ChatList_Context_Unpin: String { return self._s[1042]! } - public var PeopleNearby_DiscoverDescription: String { return self._s[1043]! } - public var Notification_MessageLifetime1d: String { return self._s[1044]! } - public var PrivacyLastSeenSettings_NeverShareWith_Title: String { return self._s[1045]! } - public var ChatListFolder_CategoryChannels: String { return self._s[1046]! } - public var VoiceOver_Chat_SeenByRecipient: String { return self._s[1047]! } - public var Notifications_PermissionsAllow: String { return self._s[1048]! } - public var Undo_ScheduledMessagesCleared: String { return self._s[1049]! } - public var AutoDownloadSettings_PrivateChats: String { return self._s[1051]! } - public var ApplyLanguage_ChangeLanguageAction: String { return self._s[1052]! } + public var Conversation_AddToContacts: String { return self._s[1048]! } + public var AutoDownloadSettings_DataUsageMedium: String { return self._s[1049]! } + public var AuthSessions_LogOutApplications: String { return self._s[1051]! } + public var ChatList_Context_Unpin: String { return self._s[1052]! } + public var PeopleNearby_DiscoverDescription: String { return self._s[1053]! } + public var Notification_MessageLifetime1d: String { return self._s[1054]! } + public var PrivacyLastSeenSettings_NeverShareWith_Title: String { return self._s[1055]! } + public var ChatListFolder_CategoryChannels: String { return self._s[1056]! } + public var VoiceOver_Chat_SeenByRecipient: String { return self._s[1057]! } + public var Notifications_PermissionsAllow: String { return self._s[1058]! } + public var Undo_ScheduledMessagesCleared: String { return self._s[1059]! } + public var AutoDownloadSettings_PrivateChats: String { return self._s[1061]! } + public var ApplyLanguage_ChangeLanguageAction: String { return self._s[1062]! } public func PrivacySettings_LastSeenNobodyPlus(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1053]!, self._r[1053]!, [_0]) + return formatWithArgumentRanges(self._s[1063]!, self._r[1063]!, [_0]) } - public var Notifications_MessageNotificationsHelp: String { return self._s[1056]! } - public var WallpaperSearch_ColorPink: String { return self._s[1057]! } - public var ContactInfo_PhoneNumberHidden: String { return self._s[1058]! } - public var Passport_Identity_IssueDate: String { return self._s[1060]! } + public var Notifications_MessageNotificationsHelp: String { return self._s[1066]! } + public var WallpaperSearch_ColorPink: String { return self._s[1067]! } + public var ContactInfo_PhoneNumberHidden: String { return self._s[1068]! } + public var Passport_Identity_IssueDate: String { return self._s[1070]! } public func PUSH_CHAT_MESSAGE_GIF(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1061]!, self._r[1061]!, [_1, _2]) + return formatWithArgumentRanges(self._s[1071]!, self._r[1071]!, [_1, _2]) } - public var PrivacySettings_DeleteAccountIfAwayFor: String { return self._s[1062]! } - public var Channel_Info_Description: String { return self._s[1063]! } - public var Common_Back: String { return self._s[1064]! } - public var Weekday_ShortTuesday: String { return self._s[1065]! } - public var Chat_PinnedMessagesHiddenTitle: String { return self._s[1067]! } - public var ChatListFolder_AddChats: String { return self._s[1068]! } - public var Common_Close: String { return self._s[1070]! } - public var Map_OpenIn: String { return self._s[1071]! } - public var Group_Setup_HistoryTitle: String { return self._s[1072]! } - public var SettingsSearch_Synonyms_Data_AutoDownloadUsingWifi: String { return self._s[1073]! } - public var Notification_MessageLifetime1h: String { return self._s[1074]! } + public var PrivacySettings_DeleteAccountIfAwayFor: String { return self._s[1072]! } + public var Channel_Info_Description: String { return self._s[1073]! } + public var Common_Back: String { return self._s[1074]! } + public var Weekday_ShortTuesday: String { return self._s[1075]! } + public var Chat_PinnedMessagesHiddenTitle: String { return self._s[1077]! } + public var ChatListFolder_AddChats: String { return self._s[1078]! } + public var Common_Close: String { return self._s[1080]! } + public var Map_OpenIn: String { return self._s[1081]! } + public var Group_Setup_HistoryTitle: String { return self._s[1082]! } + public var SettingsSearch_Synonyms_Data_AutoDownloadUsingWifi: String { return self._s[1083]! } + public var Notification_MessageLifetime1h: String { return self._s[1084]! } public func CancelResetAccount_Success(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1075]!, self._r[1075]!, [_0]) + return formatWithArgumentRanges(self._s[1085]!, self._r[1085]!, [_0]) } - public var Watch_Contacts_NoResults: String { return self._s[1077]! } - public var TwoStepAuth_SetupResendEmailCode: String { return self._s[1078]! } - public var Checkout_Phone: String { return self._s[1079]! } - public var OwnershipTransfer_ComeBackLater: String { return self._s[1080]! } + public var Watch_Contacts_NoResults: String { return self._s[1087]! } + public var TwoStepAuth_SetupResendEmailCode: String { return self._s[1088]! } + public var Checkout_Phone: String { return self._s[1089]! } + public var OwnershipTransfer_ComeBackLater: String { return self._s[1090]! } public func Channel_CommentsGroup_HeaderGroupSet(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1081]!, self._r[1081]!, [_0]) + return formatWithArgumentRanges(self._s[1091]!, self._r[1091]!, [_0]) } public func DialogList_MultipleTypingSuffix(_ _0: Int) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1082]!, self._r[1082]!, ["\(_0)"]) + return formatWithArgumentRanges(self._s[1092]!, self._r[1092]!, ["\(_0)"]) } - public var ChatAdmins_Title: String { return self._s[1083]! } - public var Appearance_ThemePreview_Chat_7_Text: String { return self._s[1084]! } + public var ChatAdmins_Title: String { return self._s[1093]! } + public var Appearance_ThemePreview_Chat_7_Text: String { return self._s[1094]! } public func PUSH_CHANNEL_MESSAGE_POLL(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1085]!, self._r[1085]!, [_1]) + return formatWithArgumentRanges(self._s[1095]!, self._r[1095]!, [_1]) } - public var Common_Done: String { return self._s[1086]! } - public var Appearance_ThemeCarouselNight: String { return self._s[1089]! } + public var Common_Done: String { return self._s[1096]! } + public var Appearance_ThemeCarouselNight: String { return self._s[1099]! } public func PUSH_PINNED_VIDEO(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1091]!, self._r[1091]!, [_1]) + return formatWithArgumentRanges(self._s[1101]!, self._r[1101]!, [_1]) } - public var Preview_OpenInInstagram: String { return self._s[1093]! } - public var VoiceChat_StartRecordingStop: String { return self._s[1097]! } - public var Wallpaper_SetColor: String { return self._s[1098]! } - public var VoiceOver_Media_PlaybackRate: String { return self._s[1099]! } - public var ChatSettings_Groups: String { return self._s[1100]! } + public var InviteLink_Expired: String { return self._s[1103]! } + public var Preview_OpenInInstagram: String { return self._s[1104]! } + public var VoiceChat_StartRecordingStop: String { return self._s[1108]! } + public var Wallpaper_SetColor: String { return self._s[1109]! } + public var VoiceOver_Media_PlaybackRate: String { return self._s[1110]! } + public var ChatSettings_Groups: String { return self._s[1111]! } public func VoiceOver_Chat_VoiceMessageFrom(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1101]!, self._r[1101]!, [_0]) + return formatWithArgumentRanges(self._s[1112]!, self._r[1112]!, [_0]) } - public var Contacts_SortedByName: String { return self._s[1102]! } - public var SettingsSearch_Synonyms_Notifications_ContactJoined: String { return self._s[1103]! } - public var Channel_Management_LabelCreator: String { return self._s[1104]! } - public var Contacts_PermissionsSuppressWarningTitle: String { return self._s[1105]! } + public var Contacts_SortedByName: String { return self._s[1113]! } + public var SettingsSearch_Synonyms_Notifications_ContactJoined: String { return self._s[1114]! } + public var Channel_Management_LabelCreator: String { return self._s[1115]! } + public var Contacts_PermissionsSuppressWarningTitle: String { return self._s[1116]! } public func PrivacySettings_LastSeenContactsMinusPlus(_ _0: String, _ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1106]!, self._r[1106]!, [_0, _1]) + return formatWithArgumentRanges(self._s[1117]!, self._r[1117]!, [_0, _1]) } - public var Group_GroupMembersHeader: String { return self._s[1107]! } - public var Group_PublicLink_Title: String { return self._s[1108]! } - public var Channel_OwnershipTransfer_ErrorAdminsTooMuch: String { return self._s[1109]! } - public var VoiceOver_Chat_Photo: String { return self._s[1110]! } - public var TwoFactorSetup_EmailVerification_Placeholder: String { return self._s[1111]! } - public var IntentsSettings_SuggestBy: String { return self._s[1112]! } - public var Privacy_Calls_AlwaysAllow_Placeholder: String { return self._s[1113]! } - public var Appearance_ThemePreview_ChatList_1_Name: String { return self._s[1114]! } - public var PhoneNumberHelp_ChangeNumber: String { return self._s[1115]! } - public var LogoutOptions_SetPasscodeText: String { return self._s[1116]! } - public var Map_OpenInMaps: String { return self._s[1117]! } - public var ContactInfo_PhoneLabelWorkFax: String { return self._s[1118]! } - public var BlockedUsers_Unblock: String { return self._s[1119]! } + public var Group_GroupMembersHeader: String { return self._s[1118]! } + public var Group_PublicLink_Title: String { return self._s[1119]! } + public var Channel_OwnershipTransfer_ErrorAdminsTooMuch: String { return self._s[1120]! } + public var VoiceOver_Chat_Photo: String { return self._s[1121]! } + public var TwoFactorSetup_EmailVerification_Placeholder: String { return self._s[1122]! } + public var IntentsSettings_SuggestBy: String { return self._s[1123]! } + public var Privacy_Calls_AlwaysAllow_Placeholder: String { return self._s[1124]! } + public var Appearance_ThemePreview_ChatList_1_Name: String { return self._s[1125]! } + public var PhoneNumberHelp_ChangeNumber: String { return self._s[1126]! } + public var LogoutOptions_SetPasscodeText: String { return self._s[1127]! } + public var Map_OpenInMaps: String { return self._s[1128]! } + public var ContactInfo_PhoneLabelWorkFax: String { return self._s[1129]! } + public var BlockedUsers_Unblock: String { return self._s[1130]! } public func Settings_ApplyProxyAlert(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1120]!, self._r[1120]!, [_1, _2]) + return formatWithArgumentRanges(self._s[1131]!, self._r[1131]!, [_1, _2]) } public func Channel_AdminLog_MessageRestrictedNameUsername(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1121]!, self._r[1121]!, [_1, _2]) - } - public var Conversation_Block: String { return self._s[1123]! } - public var Passport_Scans_UploadNew: String { return self._s[1124]! } - public var Share_Title: String { return self._s[1125]! } - public var Conversation_ApplyLocalization: String { return self._s[1126]! } - public var SharedMedia_EmptyLinksText: String { return self._s[1127]! } - public var Settings_NotificationsAndSounds: String { return self._s[1128]! } - public var Stats_ViewsByHoursTitle: String { return self._s[1129]! } - public var PhotoEditor_QualityMedium: String { return self._s[1130]! } - public var Conversation_ContextMenuCancelSending: String { return self._s[1131]! } - public func PUSH_CHANNEL_MESSAGE_GAME(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { return formatWithArgumentRanges(self._s[1132]!, self._r[1132]!, [_1, _2]) } - public var Conversation_RestrictedInline: String { return self._s[1133]! } - public var Passport_Language_tr: String { return self._s[1134]! } - public var Call_Mute: String { return self._s[1135]! } + public var Conversation_Block: String { return self._s[1134]! } + public var Passport_Scans_UploadNew: String { return self._s[1135]! } + public var Share_Title: String { return self._s[1136]! } + public var Conversation_ApplyLocalization: String { return self._s[1137]! } + public var SharedMedia_EmptyLinksText: String { return self._s[1138]! } + public var Settings_NotificationsAndSounds: String { return self._s[1139]! } + public var Stats_ViewsByHoursTitle: String { return self._s[1140]! } + public var PhotoEditor_QualityMedium: String { return self._s[1141]! } + public var Conversation_ContextMenuCancelSending: String { return self._s[1142]! } + public func PUSH_CHANNEL_MESSAGE_GAME(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[1143]!, self._r[1143]!, [_1, _2]) + } + public var Conversation_RestrictedInline: String { return self._s[1144]! } + public var Passport_Language_tr: String { return self._s[1145]! } + public var Call_Mute: String { return self._s[1146]! } public func Conversation_NoticeInvitedByInGroup(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1136]!, self._r[1136]!, [_0]) + return formatWithArgumentRanges(self._s[1147]!, self._r[1147]!, [_0]) } - public var Passport_Language_bn: String { return self._s[1137]! } - public var AccessDenied_LocationTracking: String { return self._s[1139]! } - public var Month_ShortOctober: String { return self._s[1140]! } - public var AutoDownloadSettings_WiFi: String { return self._s[1141]! } - public var ProfilePhoto_SetMainPhoto: String { return self._s[1143]! } - public var ChangePhoneNumberNumber_NewNumber: String { return self._s[1144]! } + public var Passport_Language_bn: String { return self._s[1148]! } + public var AccessDenied_LocationTracking: String { return self._s[1151]! } + public var Month_ShortOctober: String { return self._s[1152]! } + public var AutoDownloadSettings_WiFi: String { return self._s[1153]! } + public var ProfilePhoto_SetMainPhoto: String { return self._s[1155]! } + public var ChangePhoneNumberNumber_NewNumber: String { return self._s[1156]! } public func Time_MonthOfYear_m3(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1145]!, self._r[1145]!, [_0]) + return formatWithArgumentRanges(self._s[1157]!, self._r[1157]!, [_0]) } - public var Watch_ChannelInfo_Title: String { return self._s[1146]! } - public var State_Updating: String { return self._s[1147]! } - public var Conversation_UnblockUser: String { return self._s[1148]! } - public var Notifications_ChannelNotificationsSound: String { return self._s[1149]! } - public var Map_GetDirections: String { return self._s[1150]! } - public var Watch_Compose_AddContact: String { return self._s[1152]! } - public var Conversation_Dice_u26BD: String { return self._s[1153]! } - public var AccessDenied_PhotosRestricted: String { return self._s[1154]! } + public var Watch_ChannelInfo_Title: String { return self._s[1158]! } + public var State_Updating: String { return self._s[1159]! } + public var Conversation_UnblockUser: String { return self._s[1160]! } + public var Notifications_ChannelNotificationsSound: String { return self._s[1161]! } + public var Map_GetDirections: String { return self._s[1162]! } + public var Watch_Compose_AddContact: String { return self._s[1164]! } + public var Conversation_Dice_u26BD: String { return self._s[1165]! } + public var AccessDenied_PhotosRestricted: String { return self._s[1166]! } public func Channel_AdminLog_MessageRank(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1155]!, self._r[1155]!, [_1]) - } - public var Map_LoadError: String { return self._s[1157]! } - public var SettingsSearch_Synonyms_Privacy_Calls: String { return self._s[1158]! } - public var PhotoEditor_CropAuto: String { return self._s[1159]! } - public func Target_ShareGameConfirmationPrivate(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1162]!, self._r[1162]!, [_0]) - } - public var Username_TooManyPublicUsernamesError: String { return self._s[1164]! } - public func PUSH_PINNED_GAME(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1165]!, self._r[1165]!, [_1]) - } - public var Settings_PhoneNumber: String { return self._s[1166]! } - public func Channel_AdminLog_MessageTransferedName(_ _1: String) -> (String, [(Int, NSRange)]) { return formatWithArgumentRanges(self._s[1167]!, self._r[1167]!, [_1]) } - public var Month_GenJune: String { return self._s[1169]! } - public var Notifications_ExceptionsGroupPlaceholder: String { return self._s[1170]! } - public var ChatListFolder_CategoryRead: String { return self._s[1171]! } - public var LoginPassword_ResetAccount: String { return self._s[1172]! } + public var Map_LoadError: String { return self._s[1169]! } + public var SettingsSearch_Synonyms_Privacy_Calls: String { return self._s[1170]! } + public var PhotoEditor_CropAuto: String { return self._s[1171]! } + public func Target_ShareGameConfirmationPrivate(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[1174]!, self._r[1174]!, [_0]) + } + public var Username_TooManyPublicUsernamesError: String { return self._s[1176]! } + public func PUSH_PINNED_GAME(_ _1: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[1177]!, self._r[1177]!, [_1]) + } + public var Settings_PhoneNumber: String { return self._s[1178]! } + public func Channel_AdminLog_MessageTransferedName(_ _1: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[1179]!, self._r[1179]!, [_1]) + } + public var Month_GenJune: String { return self._s[1181]! } + public var Notifications_ExceptionsGroupPlaceholder: String { return self._s[1182]! } + public var ChatListFolder_CategoryRead: String { return self._s[1183]! } + public var LoginPassword_ResetAccount: String { return self._s[1184]! } public func DialogList_SingleUploadingFileSuffix(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1173]!, self._r[1173]!, [_0]) + return formatWithArgumentRanges(self._s[1185]!, self._r[1185]!, [_0]) } - public var Call_CameraConfirmationConfirm: String { return self._s[1174]! } - public var Notification_RenamedChannel: String { return self._s[1175]! } + public var Call_CameraConfirmationConfirm: String { return self._s[1186]! } + public var Notification_RenamedChannel: String { return self._s[1187]! } public func Channel_AdminLog_MessageUnpinned(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1176]!, self._r[1176]!, [_0]) + return formatWithArgumentRanges(self._s[1188]!, self._r[1188]!, [_0]) } - public var Channel_AdminLogFilter_EventsAdmins: String { return self._s[1177]! } - public var IntentsSettings_Title: String { return self._s[1179]! } - public var Settings_AppleWatch: String { return self._s[1180]! } - public var DialogList_NoMessagesText: String { return self._s[1181]! } - public var GroupPermission_NoChangeInfo: String { return self._s[1182]! } - public var Channel_ErrorAccessDenied: String { return self._s[1184]! } - public var ScheduledMessages_EmptyPlaceholder: String { return self._s[1185]! } + public var Channel_AdminLogFilter_EventsAdmins: String { return self._s[1189]! } + public var IntentsSettings_Title: String { return self._s[1191]! } + public var Settings_AppleWatch: String { return self._s[1192]! } + public var DialogList_NoMessagesText: String { return self._s[1193]! } + public var GroupPermission_NoChangeInfo: String { return self._s[1194]! } + public var Channel_ErrorAccessDenied: String { return self._s[1196]! } + public var ScheduledMessages_EmptyPlaceholder: String { return self._s[1197]! } public func Message_StickerText(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1186]!, self._r[1186]!, [_0]) + return formatWithArgumentRanges(self._s[1198]!, self._r[1198]!, [_0]) } - public var AuthSessions_TerminateOtherSessionsHelp: String { return self._s[1187]! } - public var StickerPacksSettings_AnimatedStickers: String { return self._s[1188]! } - public var Month_ShortJanuary: String { return self._s[1189]! } - public var Conversation_UnreadMessages: String { return self._s[1190]! } - public var Call_VoiceOver_VideoCallCanceled: String { return self._s[1192]! } - public var Conversation_PrivateChannelTooltip: String { return self._s[1193]! } - public var PrivacySettings_DeleteAccountTitle: String { return self._s[1195]! } - public var Channel_Members_AddBannedErrorAdmin: String { return self._s[1196]! } + public var AuthSessions_TerminateOtherSessionsHelp: String { return self._s[1199]! } + public var StickerPacksSettings_AnimatedStickers: String { return self._s[1200]! } + public var Month_ShortJanuary: String { return self._s[1201]! } + public var Conversation_UnreadMessages: String { return self._s[1202]! } + public var Call_VoiceOver_VideoCallCanceled: String { return self._s[1204]! } + public var Conversation_PrivateChannelTooltip: String { return self._s[1205]! } + public var PrivacySettings_DeleteAccountTitle: String { return self._s[1207]! } + public var Channel_Members_AddBannedErrorAdmin: String { return self._s[1208]! } public func Conversation_ShareMyPhoneNumberConfirmation(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1199]!, self._r[1199]!, [_1, _2]) + return formatWithArgumentRanges(self._s[1212]!, self._r[1212]!, [_1, _2]) } - public var Widget_ApplicationLocked: String { return self._s[1200]! } + public var Widget_ApplicationLocked: String { return self._s[1213]! } public func TextFormat_AddLinkText(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1201]!, self._r[1201]!, [_0]) + return formatWithArgumentRanges(self._s[1214]!, self._r[1214]!, [_0]) } - public var Common_TakePhotoOrVideo: String { return self._s[1202]! } - public var Passport_Language_ru: String { return self._s[1203]! } - public var MediaPicker_VideoMuteDescription: String { return self._s[1204]! } - public var EditTheme_ErrorLinkTaken: String { return self._s[1205]! } + public var Common_TakePhotoOrVideo: String { return self._s[1215]! } + public var Passport_Language_ru: String { return self._s[1216]! } + public var MediaPicker_VideoMuteDescription: String { return self._s[1217]! } + public var EditTheme_ErrorLinkTaken: String { return self._s[1218]! } public func Group_EditAdmin_RankInfo(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1207]!, self._r[1207]!, [_0]) + return formatWithArgumentRanges(self._s[1220]!, self._r[1220]!, [_0]) } - public var Channel_Members_AddAdminErrorBlacklisted: String { return self._s[1208]! } - public var Conversation_Owner: String { return self._s[1210]! } - public var Settings_FAQ_Intro: String { return self._s[1211]! } - public var PhotoEditor_QualityLow: String { return self._s[1213]! } - public var Widget_GalleryTitle: String { return self._s[1214]! } - public var Call_End: String { return self._s[1215]! } - public var StickerPacksSettings_FeaturedPacks: String { return self._s[1217]! } - public var Privacy_ContactsSyncHelp: String { return self._s[1218]! } - public var OldChannels_NoticeUpgradeText: String { return self._s[1222]! } - public var Conversation_Call: String { return self._s[1224]! } - public var Watch_MessageView_Title: String { return self._s[1225]! } + public var Channel_Members_AddAdminErrorBlacklisted: String { return self._s[1221]! } + public var Conversation_Owner: String { return self._s[1223]! } + public var Settings_FAQ_Intro: String { return self._s[1224]! } + public var PhotoEditor_QualityLow: String { return self._s[1226]! } + public var Widget_GalleryTitle: String { return self._s[1227]! } + public var Call_End: String { return self._s[1228]! } + public var StickerPacksSettings_FeaturedPacks: String { return self._s[1230]! } + public var Privacy_ContactsSyncHelp: String { return self._s[1231]! } + public var OldChannels_NoticeUpgradeText: String { return self._s[1235]! } + public var Conversation_Call: String { return self._s[1237]! } + public var Watch_MessageView_Title: String { return self._s[1238]! } public func Notification_RenamedChat(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1226]!, self._r[1226]!, [_0]) + return formatWithArgumentRanges(self._s[1239]!, self._r[1239]!, [_0]) } - public var Passport_PasswordCompleteSetup: String { return self._s[1227]! } + public var Passport_PasswordCompleteSetup: String { return self._s[1240]! } public func Notification_ChangedGroupVideo(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1228]!, self._r[1228]!, [_0]) + return formatWithArgumentRanges(self._s[1241]!, self._r[1241]!, [_0]) } public func TwoFactorSetup_EmailVerification_Text(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1230]!, self._r[1230]!, [_0]) + return formatWithArgumentRanges(self._s[1243]!, self._r[1243]!, [_0]) } - public var Map_Location: String { return self._s[1231]! } - public var Watch_MessageView_ViewOnPhone: String { return self._s[1232]! } - public var Login_CountryCode: String { return self._s[1233]! } - public var Channel_DiscussionGroup_PrivateGroup: String { return self._s[1235]! } - public var ChatState_ConnectingToProxy: String { return self._s[1236]! } - public var Login_CallRequestState3: String { return self._s[1237]! } - public var NetworkUsageSettings_MediaAudioDataSection: String { return self._s[1239]! } - public var SocksProxySetup_ProxyStatusConnecting: String { return self._s[1240]! } - public var PrivacyLastSeenSettings_NeverShareWith_Placeholder: String { return self._s[1243]! } - public var Call_StatusEnded: String { return self._s[1244]! } - public var MusicPlayer_VoiceNote: String { return self._s[1247]! } + public var Map_Location: String { return self._s[1244]! } + public var Watch_MessageView_ViewOnPhone: String { return self._s[1245]! } + public var Login_CountryCode: String { return self._s[1246]! } + public var Channel_DiscussionGroup_PrivateGroup: String { return self._s[1248]! } + public var ChatState_ConnectingToProxy: String { return self._s[1249]! } + public var Login_CallRequestState3: String { return self._s[1250]! } + public var NetworkUsageSettings_MediaAudioDataSection: String { return self._s[1252]! } + public var SocksProxySetup_ProxyStatusConnecting: String { return self._s[1253]! } + public var PrivacyLastSeenSettings_NeverShareWith_Placeholder: String { return self._s[1256]! } + public var Call_StatusEnded: String { return self._s[1257]! } + public var MusicPlayer_VoiceNote: String { return self._s[1260]! } public func PUSH_CHANNEL_MESSAGE_TEXT(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1248]!, self._r[1248]!, [_1, _2]) + return formatWithArgumentRanges(self._s[1261]!, self._r[1261]!, [_1, _2]) } - public var VoiceOver_MessageContextShare: String { return self._s[1249]! } - public var ProfilePhoto_SearchWeb: String { return self._s[1250]! } - public var EditProfile_Title: String { return self._s[1251]! } + public var VoiceOver_MessageContextShare: String { return self._s[1262]! } + public var ProfilePhoto_SearchWeb: String { return self._s[1263]! } + public var EditProfile_Title: String { return self._s[1264]! } public func Notification_PinnedQuizMessage(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1252]!, self._r[1252]!, [_0]) + return formatWithArgumentRanges(self._s[1265]!, self._r[1265]!, [_0]) } - public var VoiceChat_Unmute: String { return self._s[1253]! } - public var ChangePhoneNumberCode_CodePlaceholder: String { return self._s[1254]! } - public var NetworkUsageSettings_ResetStats: String { return self._s[1256]! } - public var NetworkUsageSettings_GeneralDataSection: String { return self._s[1257]! } - public var StickerPackActionInfo_AddedTitle: String { return self._s[1258]! } - public var Channel_BanUser_PermissionSendStickersAndGifs: String { return self._s[1259]! } + public var VoiceChat_Unmute: String { return self._s[1266]! } + public var ChangePhoneNumberCode_CodePlaceholder: String { return self._s[1267]! } + public var NetworkUsageSettings_ResetStats: String { return self._s[1269]! } + public var NetworkUsageSettings_GeneralDataSection: String { return self._s[1270]! } + public var StickerPackActionInfo_AddedTitle: String { return self._s[1271]! } + public var Channel_BanUser_PermissionSendStickersAndGifs: String { return self._s[1272]! } public func Call_ParticipantVideoVersionOutdatedError(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1260]!, self._r[1260]!, [_0]) + return formatWithArgumentRanges(self._s[1273]!, self._r[1273]!, [_0]) } - public var Location_ProximityNotification_Title: String { return self._s[1261]! } - public var AuthSessions_AddDeviceIntro_Text1: String { return self._s[1262]! } - public var Passport_Identity_LatinNameHelp: String { return self._s[1265]! } - public var AuthSessions_AddDeviceIntro_Text2: String { return self._s[1266]! } - public var Stats_GroupMembersTitle: String { return self._s[1267]! } - public var AuthSessions_AddDeviceIntro_Text3: String { return self._s[1268]! } - public var Contacts_PermissionsSuppressWarningText: String { return self._s[1269]! } - public var OpenFile_PotentiallyDangerousContentAlert: String { return self._s[1270]! } - public var Settings_SetUsername: String { return self._s[1271]! } - public var GroupInfo_ActionRestrict: String { return self._s[1272]! } - public var SettingsSearch_Synonyms_SavedMessages: String { return self._s[1273]! } + public var Location_ProximityNotification_Title: String { return self._s[1274]! } + public var AuthSessions_AddDeviceIntro_Text1: String { return self._s[1275]! } + public var Passport_Identity_LatinNameHelp: String { return self._s[1278]! } + public var AuthSessions_AddDeviceIntro_Text2: String { return self._s[1279]! } + public var Stats_GroupMembersTitle: String { return self._s[1280]! } + public var AuthSessions_AddDeviceIntro_Text3: String { return self._s[1281]! } + public var Contacts_PermissionsSuppressWarningText: String { return self._s[1282]! } + public var OpenFile_PotentiallyDangerousContentAlert: String { return self._s[1283]! } + public var Settings_SetUsername: String { return self._s[1284]! } + public var GroupInfo_ActionRestrict: String { return self._s[1285]! } + public var SettingsSearch_Synonyms_SavedMessages: String { return self._s[1286]! } public func Time_PreciseDate_m2(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1274]!, self._r[1274]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[1287]!, self._r[1287]!, [_1, _2, _3]) } - public var Notifications_DisplayNamesOnLockScreenInfoWithLink: String { return self._s[1275]! } - public var Notification_Exceptions_AlwaysOff: String { return self._s[1276]! } - public var Conversation_ContextMenuDelete: String { return self._s[1277]! } - public var Privacy_Calls_WhoCanCallMe: String { return self._s[1278]! } - public var ChatList_PsaAlert_covid: String { return self._s[1281]! } - public var DialogList_Pin: String { return self._s[1282]! } - public var PrivacySettings_SecurityTitle: String { return self._s[1283]! } - public var GroupPermission_NotAvailableInPublicGroups: String { return self._s[1284]! } - public var PeopleNearby_Groups: String { return self._s[1285]! } - public var Message_File: String { return self._s[1286]! } - public var Calls_NoCallsPlaceholder: String { return self._s[1287]! } - public var ChatList_GenericPsaLabel: String { return self._s[1289]! } - public var UserInfo_LastNamePlaceholder: String { return self._s[1290]! } - public var IntentsSettings_Reset: String { return self._s[1292]! } - public var Call_ConnectionErrorTitle: String { return self._s[1293]! } - public var PhotoEditor_SaturationTool: String { return self._s[1294]! } - public var ChatSettings_AutomaticVideoMessageDownload: String { return self._s[1295]! } - public var SettingsSearch_Synonyms_Stickers_ArchivedPacks: String { return self._s[1296]! } - public var Conversation_SearchNoResults: String { return self._s[1297]! } - public var Channel_DiscussionGroup_PrivateChannel: String { return self._s[1298]! } - public var Map_OpenInWaze: String { return self._s[1299]! } - public var WallpaperPreview_Title: String { return self._s[1300]! } + public var Notifications_DisplayNamesOnLockScreenInfoWithLink: String { return self._s[1288]! } + public var Notification_Exceptions_AlwaysOff: String { return self._s[1289]! } + public var Conversation_ContextMenuDelete: String { return self._s[1290]! } + public var Privacy_Calls_WhoCanCallMe: String { return self._s[1291]! } + public var ChatList_PsaAlert_covid: String { return self._s[1294]! } + public var DialogList_Pin: String { return self._s[1295]! } + public var PrivacySettings_SecurityTitle: String { return self._s[1296]! } + public var GroupPermission_NotAvailableInPublicGroups: String { return self._s[1297]! } + public var PeopleNearby_Groups: String { return self._s[1298]! } + public var Message_File: String { return self._s[1299]! } + public var Calls_NoCallsPlaceholder: String { return self._s[1300]! } + public var ChatList_GenericPsaLabel: String { return self._s[1302]! } + public var UserInfo_LastNamePlaceholder: String { return self._s[1303]! } + public var IntentsSettings_Reset: String { return self._s[1305]! } + public var Call_ConnectionErrorTitle: String { return self._s[1306]! } + public var PhotoEditor_SaturationTool: String { return self._s[1307]! } + public var ChatSettings_AutomaticVideoMessageDownload: String { return self._s[1308]! } + public var SettingsSearch_Synonyms_Stickers_ArchivedPacks: String { return self._s[1309]! } + public var Conversation_SearchNoResults: String { return self._s[1310]! } + public var Channel_DiscussionGroup_PrivateChannel: String { return self._s[1311]! } + public var Map_OpenInWaze: String { return self._s[1312]! } + public var InviteLink_PeopleJoinedNone: String { return self._s[1313]! } + public var WallpaperPreview_Title: String { return self._s[1314]! } public func Passport_AcceptHelp(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1302]!, self._r[1302]!, [_1, _2]) + return formatWithArgumentRanges(self._s[1316]!, self._r[1316]!, [_1, _2]) } - public var AuthSessions_AddDeviceIntro_Title: String { return self._s[1303]! } - public var VoiceOver_Chat_RecordModeVideoMessageInfo: String { return self._s[1304]! } - public var Passport_Identity_OneOfTypeInternalPassport: String { return self._s[1305]! } - public var Notifications_PermissionsUnreachableTitle: String { return self._s[1307]! } - public var Stats_Total: String { return self._s[1310]! } - public var Stats_GroupMessages: String { return self._s[1311]! } - public var TwoFactorSetup_Email_SkipAction: String { return self._s[1312]! } - public var CheckoutInfo_ErrorPhoneInvalid: String { return self._s[1313]! } - public var Passport_Identity_Translation: String { return self._s[1314]! } - public var Notifications_TextTone: String { return self._s[1317]! } - public var Settings_RemoveConfirmation: String { return self._s[1319]! } - public var ScheduledMessages_Delete: String { return self._s[1320]! } - public var Channel_AdminLog_BanEmbedLinks: String { return self._s[1321]! } - public var Passport_PasswordNext: String { return self._s[1322]! } + public var AuthSessions_AddDeviceIntro_Title: String { return self._s[1317]! } + public var VoiceOver_Chat_RecordModeVideoMessageInfo: String { return self._s[1318]! } + public var Passport_Identity_OneOfTypeInternalPassport: String { return self._s[1319]! } + public var Notifications_PermissionsUnreachableTitle: String { return self._s[1321]! } + public var Stats_Total: String { return self._s[1324]! } + public var Stats_GroupMessages: String { return self._s[1325]! } + public var TwoFactorSetup_Email_SkipAction: String { return self._s[1326]! } + public var CheckoutInfo_ErrorPhoneInvalid: String { return self._s[1327]! } + public var Passport_Identity_Translation: String { return self._s[1328]! } + public var Notifications_TextTone: String { return self._s[1331]! } + public var Settings_RemoveConfirmation: String { return self._s[1333]! } + public var ScheduledMessages_Delete: String { return self._s[1334]! } + public var Channel_AdminLog_BanEmbedLinks: String { return self._s[1335]! } + public var Passport_PasswordNext: String { return self._s[1336]! } public func PUSH_ENCRYPTED_MESSAGE(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1323]!, self._r[1323]!, [_1]) + return formatWithArgumentRanges(self._s[1337]!, self._r[1337]!, [_1]) } - public var Passport_Address_EditBankStatement: String { return self._s[1324]! } - public var PhotoEditor_ShadowsTool: String { return self._s[1325]! } - public var Notification_VideoCallMissed: String { return self._s[1326]! } - public var AccessDenied_CameraDisabled: String { return self._s[1327]! } - public var AuthSessions_AddDevice_ScanInfo: String { return self._s[1328]! } - public var Notifications_ExceptionsMuted: String { return self._s[1329]! } - public var Conversation_ScheduleMessage_SendWhenOnline: String { return self._s[1330]! } - public var Channel_BlackList_Title: String { return self._s[1331]! } - public var PasscodeSettings_4DigitCode: String { return self._s[1332]! } - public var NotificationsSound_Bamboo: String { return self._s[1333]! } - public var PrivacySettings_LastSeenContacts: String { return self._s[1334]! } - public var Passport_Address_TypeUtilityBill: String { return self._s[1335]! } - public var Passport_Address_CountryPlaceholder: String { return self._s[1336]! } - public var GroupPermission_SectionTitle: String { return self._s[1337]! } + public var Passport_Address_EditBankStatement: String { return self._s[1338]! } + public var PhotoEditor_ShadowsTool: String { return self._s[1339]! } + public var Notification_VideoCallMissed: String { return self._s[1340]! } + public var AccessDenied_CameraDisabled: String { return self._s[1341]! } + public var AuthSessions_AddDevice_ScanInfo: String { return self._s[1342]! } + public var Notifications_ExceptionsMuted: String { return self._s[1343]! } + public var Conversation_ScheduleMessage_SendWhenOnline: String { return self._s[1344]! } + public var Channel_BlackList_Title: String { return self._s[1345]! } + public var PasscodeSettings_4DigitCode: String { return self._s[1346]! } + public var NotificationsSound_Bamboo: String { return self._s[1347]! } + public var PrivacySettings_LastSeenContacts: String { return self._s[1348]! } + public var Passport_Address_TypeUtilityBill: String { return self._s[1349]! } + public var Passport_Address_CountryPlaceholder: String { return self._s[1350]! } + public var GroupPermission_SectionTitle: String { return self._s[1351]! } + public var InviteLink_ContextRevoke: String { return self._s[1352]! } public func Notification_InvitedMultiple(_ _0: String, _ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1338]!, self._r[1338]!, [_0, _1]) + return formatWithArgumentRanges(self._s[1353]!, self._r[1353]!, [_0, _1]) } - public var CheckoutInfo_ShippingInfoStatePlaceholder: String { return self._s[1339]! } - public var Channel_LeaveChannel: String { return self._s[1340]! } - public var Watch_Notification_Joined: String { return self._s[1341]! } - public var PeerInfo_ButtonMore: String { return self._s[1342]! } - public var Passport_FieldEmailHelp: String { return self._s[1343]! } - public var ChatList_Context_Pin: String { return self._s[1344]! } + public var CheckoutInfo_ShippingInfoStatePlaceholder: String { return self._s[1354]! } + public var Channel_LeaveChannel: String { return self._s[1355]! } + public var Watch_Notification_Joined: String { return self._s[1356]! } + public var PeerInfo_ButtonMore: String { return self._s[1357]! } + public var Passport_FieldEmailHelp: String { return self._s[1358]! } + public var ChatList_Context_Pin: String { return self._s[1359]! } public func Time_MonthOfYear_m9(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1345]!, self._r[1345]!, [_0]) + return formatWithArgumentRanges(self._s[1360]!, self._r[1360]!, [_0]) } - public var Group_Location_CreateInThisPlace: String { return self._s[1346]! } - public var PhotoEditor_QualityVeryHigh: String { return self._s[1347]! } - public var Tour_Title5: String { return self._s[1348]! } + public var Group_Location_CreateInThisPlace: String { return self._s[1361]! } + public var PhotoEditor_QualityVeryHigh: String { return self._s[1362]! } + public var Tour_Title5: String { return self._s[1363]! } public func PUSH_CHAT_MESSAGE_FWD(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1349]!, self._r[1349]!, [_1, _2]) + return formatWithArgumentRanges(self._s[1364]!, self._r[1364]!, [_1, _2]) } - public var Passport_Language_en: String { return self._s[1350]! } - public var Checkout_Name: String { return self._s[1351]! } + public var Passport_Language_en: String { return self._s[1365]! } + public var Checkout_Name: String { return self._s[1366]! } public func NetworkUsageSettings_WifiUsageSince(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1352]!, self._r[1352]!, [_0]) + return formatWithArgumentRanges(self._s[1367]!, self._r[1367]!, [_0]) } - public var PhotoEditor_EnhanceTool: String { return self._s[1353]! } + public var PhotoEditor_EnhanceTool: String { return self._s[1368]! } public func PUSH_CHAT_DELETE_YOU(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1354]!, self._r[1354]!, [_1, _2]) + return formatWithArgumentRanges(self._s[1369]!, self._r[1369]!, [_1, _2]) } public func Login_TermsOfService_ProceedBot(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1355]!, self._r[1355]!, [_0]) + return formatWithArgumentRanges(self._s[1370]!, self._r[1370]!, [_0]) } - public var Group_ErrorSendRestrictedMedia: String { return self._s[1356]! } + public var Group_ErrorSendRestrictedMedia: String { return self._s[1371]! } public func UserInfo_NotificationsDefaultSound(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1357]!, self._r[1357]!, [_0]) + return formatWithArgumentRanges(self._s[1372]!, self._r[1372]!, [_0]) } - public var Login_UnknownError: String { return self._s[1358]! } - public var Passport_Identity_TypeDriversLicense: String { return self._s[1361]! } - public var ChatList_AutoarchiveSuggestion_Title: String { return self._s[1362]! } - public var Watch_PhotoView_Title: String { return self._s[1363]! } - public var Appearance_ThemePreview_ChatList_3_Text: String { return self._s[1364]! } - public var Checkout_TotalAmount: String { return self._s[1365]! } - public var ChatList_RemoveFolderAction: String { return self._s[1366]! } - public var GroupInfo_SetGroupPhoto: String { return self._s[1367]! } - public var Watch_AppName: String { return self._s[1368]! } + public var Login_UnknownError: String { return self._s[1373]! } + public var Passport_Identity_TypeDriversLicense: String { return self._s[1376]! } + public var ChatList_AutoarchiveSuggestion_Title: String { return self._s[1377]! } + public var Watch_PhotoView_Title: String { return self._s[1378]! } + public var Appearance_ThemePreview_ChatList_3_Text: String { return self._s[1379]! } + public var Checkout_TotalAmount: String { return self._s[1380]! } + public var ChatList_RemoveFolderAction: String { return self._s[1381]! } + public var GroupInfo_SetGroupPhoto: String { return self._s[1382]! } + public var Watch_AppName: String { return self._s[1383]! } public func PUSH_PINNED_GAME_SCORE(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1369]!, self._r[1369]!, [_1]) + return formatWithArgumentRanges(self._s[1384]!, self._r[1384]!, [_1]) } - public var Channel_Username_CheckingUsername: String { return self._s[1370]! } - public var ContactList_Context_Call: String { return self._s[1371]! } - public var ChatList_ReorderTabs: String { return self._s[1372]! } - public var Watch_ChatList_Compose: String { return self._s[1373]! } + public var Channel_Username_CheckingUsername: String { return self._s[1385]! } + public var ContactList_Context_Call: String { return self._s[1386]! } + public var ChatList_ReorderTabs: String { return self._s[1387]! } + public var Watch_ChatList_Compose: String { return self._s[1388]! } public func Conversation_LiveLocationYouAnd(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1374]!, self._r[1374]!, [_0]) - } - public var Channel_AdminLog_EmptyFilterTitle: String { return self._s[1375]! } - public var ArchivedChats_IntroTitle1: String { return self._s[1376]! } - public func PUSH_ENCRYPTION_ACCEPT(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1377]!, self._r[1377]!, [_1]) - } - public var Call_StatusRequesting: String { return self._s[1379]! } - public var Checkout_TotalPaidAmount: String { return self._s[1380]! } - public var Weekday_Friday: String { return self._s[1382]! } - public var CreateGroup_ChannelsTooMuch: String { return self._s[1383]! } - public var Watch_ChatList_NoConversationsText: String { return self._s[1384]! } - public func Channel_AdminLog_MessageChangedGroupStickerPack(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1385]!, self._r[1385]!, [_0]) - } - public var SecretVideo_Title: String { return self._s[1386]! } - public func Notification_PinnedStickerMessage(_ _0: String) -> (String, [(Int, NSRange)]) { return formatWithArgumentRanges(self._s[1389]!, self._r[1389]!, [_0]) } - public var Undo_Undo: String { return self._s[1390]! } - public var Watch_Microphone_Access: String { return self._s[1391]! } + public var Channel_AdminLog_EmptyFilterTitle: String { return self._s[1390]! } + public var ArchivedChats_IntroTitle1: String { return self._s[1391]! } + public func PUSH_ENCRYPTION_ACCEPT(_ _1: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[1392]!, self._r[1392]!, [_1]) + } + public var Call_StatusRequesting: String { return self._s[1394]! } + public var Checkout_TotalPaidAmount: String { return self._s[1395]! } + public var Weekday_Friday: String { return self._s[1397]! } + public var CreateGroup_ChannelsTooMuch: String { return self._s[1398]! } + public var Watch_ChatList_NoConversationsText: String { return self._s[1399]! } + public func Channel_AdminLog_MessageChangedGroupStickerPack(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[1400]!, self._r[1400]!, [_0]) + } + public var SecretVideo_Title: String { return self._s[1401]! } + public func Notification_PinnedStickerMessage(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[1404]!, self._r[1404]!, [_0]) + } + public var Undo_Undo: String { return self._s[1405]! } + public var Watch_Microphone_Access: String { return self._s[1406]! } public func PUSH_CHAT_MESSAGE_PHOTO(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1392]!, self._r[1392]!, [_1, _2]) + return formatWithArgumentRanges(self._s[1407]!, self._r[1407]!, [_1, _2]) } public func ChatList_Search_NoResultsQueryDescription(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1393]!, self._r[1393]!, [_0]) + return formatWithArgumentRanges(self._s[1408]!, self._r[1408]!, [_0]) } - public var Checkout_NewCard_PostcodeTitle: String { return self._s[1395]! } - public var TwoFactorSetup_Intro_Action: String { return self._s[1396]! } - public var Passport_Language_ne: String { return self._s[1397]! } - public var TwoStepAuth_EmailHelp: String { return self._s[1399]! } - public var Profile_MessageLifetime2s: String { return self._s[1400]! } + public var Checkout_NewCard_PostcodeTitle: String { return self._s[1410]! } + public var TwoFactorSetup_Intro_Action: String { return self._s[1411]! } + public var Passport_Language_ne: String { return self._s[1412]! } + public var TwoStepAuth_EmailHelp: String { return self._s[1414]! } + public var Profile_MessageLifetime2s: String { return self._s[1415]! } public func Conversation_MessageDialogRetryAll(_ _1: Int) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1401]!, self._r[1401]!, ["\(_1)"]) + return formatWithArgumentRanges(self._s[1416]!, self._r[1416]!, ["\(_1)"]) } public func Items_NOfM(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1402]!, self._r[1402]!, [_1, _2]) + return formatWithArgumentRanges(self._s[1417]!, self._r[1417]!, [_1, _2]) } - public var Media_LimitedAccessText: String { return self._s[1403]! } + public var Media_LimitedAccessText: String { return self._s[1418]! } public func PUSH_CHAT_TITLE_EDITED(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1404]!, self._r[1404]!, [_1, _2]) + return formatWithArgumentRanges(self._s[1419]!, self._r[1419]!, [_1, _2]) } - public var GroupPermission_NoPinMessages: String { return self._s[1405]! } + public var GroupPermission_NoPinMessages: String { return self._s[1420]! } public func Notification_VoiceChatStarted(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1406]!, self._r[1406]!, [_1]) + return formatWithArgumentRanges(self._s[1421]!, self._r[1421]!, [_1]) } public func Notification_CreatedChat(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1407]!, self._r[1407]!, [_0]) + return formatWithArgumentRanges(self._s[1422]!, self._r[1422]!, [_0]) } - public var FastTwoStepSetup_HintHelp: String { return self._s[1408]! } - public var WallpaperSearch_ColorRed: String { return self._s[1409]! } - public var Watch_ConnectionDescription: String { return self._s[1410]! } - public var Notification_Exceptions_AddException: String { return self._s[1411]! } - public var LocalGroup_IrrelevantWarning: String { return self._s[1412]! } - public var VoiceOver_MessageContextDelete: String { return self._s[1413]! } - public var LogoutOptions_AlternativeOptionsSection: String { return self._s[1414]! } - public var Passport_PasswordPlaceholder: String { return self._s[1415]! } - public var TwoStepAuth_RecoveryEmailAddDescription: String { return self._s[1416]! } - public var Stats_MessageInteractionsTitle: String { return self._s[1417]! } - public var Appearance_ThemeCarouselClassic: String { return self._s[1418]! } - public var TwoFactorSetup_Email_SkipConfirmationText: String { return self._s[1420]! } - public var Channel_AdminLog_PinMessages: String { return self._s[1421]! } - public var Passport_Address_AddRentalAgreement: String { return self._s[1422]! } - public var Watch_Message_Game: String { return self._s[1423]! } - public var PrivacyLastSeenSettings_NeverShareWith: String { return self._s[1424]! } - public var PrivacyPolicy_DeclineLastWarning: String { return self._s[1425]! } - public var EditTheme_FileReadError: String { return self._s[1426]! } - public var Group_ErrorAddBlocked: String { return self._s[1427]! } - public var CallSettings_UseLessDataLongDescription: String { return self._s[1428]! } + public var FastTwoStepSetup_HintHelp: String { return self._s[1423]! } + public var WallpaperSearch_ColorRed: String { return self._s[1424]! } + public var Watch_ConnectionDescription: String { return self._s[1425]! } + public var Notification_Exceptions_AddException: String { return self._s[1426]! } + public var LocalGroup_IrrelevantWarning: String { return self._s[1427]! } + public var VoiceOver_MessageContextDelete: String { return self._s[1428]! } + public var LogoutOptions_AlternativeOptionsSection: String { return self._s[1429]! } + public var Passport_PasswordPlaceholder: String { return self._s[1430]! } + public var TwoStepAuth_RecoveryEmailAddDescription: String { return self._s[1431]! } + public var Stats_MessageInteractionsTitle: String { return self._s[1432]! } + public var Appearance_ThemeCarouselClassic: String { return self._s[1433]! } + public var TwoFactorSetup_Email_SkipConfirmationText: String { return self._s[1435]! } + public var Channel_AdminLog_PinMessages: String { return self._s[1436]! } + public var Passport_Address_AddRentalAgreement: String { return self._s[1437]! } + public var Watch_Message_Game: String { return self._s[1438]! } + public var PrivacyLastSeenSettings_NeverShareWith: String { return self._s[1439]! } + public var PrivacyPolicy_DeclineLastWarning: String { return self._s[1440]! } + public var EditTheme_FileReadError: String { return self._s[1441]! } + public var Group_ErrorAddBlocked: String { return self._s[1442]! } + public var CallSettings_UseLessDataLongDescription: String { return self._s[1443]! } public func PUSH_MESSAGE_PHOTO(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1430]!, self._r[1430]!, [_1]) + return formatWithArgumentRanges(self._s[1445]!, self._r[1445]!, [_1]) } public func UserInfo_BlockConfirmation(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1431]!, self._r[1431]!, [_0]) + return formatWithArgumentRanges(self._s[1446]!, self._r[1446]!, [_0]) } - public var CheckoutInfo_ShippingInfoAddress2Placeholder: String { return self._s[1432]! } - public var TwoFactorSetup_EmailVerification_Action: String { return self._s[1433]! } + public var CheckoutInfo_ShippingInfoAddress2Placeholder: String { return self._s[1447]! } + public var TwoFactorSetup_EmailVerification_Action: String { return self._s[1448]! } public func Username_LinkHint(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1434]!, self._r[1434]!, [_0]) + return formatWithArgumentRanges(self._s[1449]!, self._r[1449]!, [_0]) } - public var ConversationProfile_ErrorCreatingConversation: String { return self._s[1435]! } - public var Bot_GroupStatusReadsHistory: String { return self._s[1436]! } - public var PhotoEditor_CurvesRed: String { return self._s[1437]! } - public var InstantPage_TapToOpenLink: String { return self._s[1438]! } - public var FastTwoStepSetup_PasswordHelp: String { return self._s[1439]! } - public var Conversation_DiscussionNotStarted: String { return self._s[1440]! } - public var Notification_CallMissedShort: String { return self._s[1441]! } + public var ConversationProfile_ErrorCreatingConversation: String { return self._s[1450]! } + public var Bot_GroupStatusReadsHistory: String { return self._s[1451]! } + public var PhotoEditor_CurvesRed: String { return self._s[1452]! } + public var InstantPage_TapToOpenLink: String { return self._s[1453]! } + public var InviteLink_PeopleJoinedShortNoneExpired: String { return self._s[1454]! } + public var FastTwoStepSetup_PasswordHelp: String { return self._s[1455]! } + public var Conversation_DiscussionNotStarted: String { return self._s[1456]! } + public var Notification_CallMissedShort: String { return self._s[1457]! } public func Notification_JoinedGroupByLink(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1442]!, self._r[1442]!, [_0]) + return formatWithArgumentRanges(self._s[1458]!, self._r[1458]!, [_0]) } - public var Conversation_DeleteMessagesForEveryone: String { return self._s[1443]! } - public var Permissions_SiriTitle_v0: String { return self._s[1444]! } - public var GroupInfo_AddUserLeftError: String { return self._s[1445]! } - public var Conversation_SendMessage_SendSilently: String { return self._s[1446]! } - public var Paint_Duplicate: String { return self._s[1447]! } - public var AttachmentMenu_WebSearch: String { return self._s[1448]! } - public var Bot_Stop: String { return self._s[1450]! } - public var Conversation_PrivateChannelTimeLimitedAlertTitle: String { return self._s[1451]! } - public var ReportGroupLocation_Report: String { return self._s[1452]! } - public var Compose_Create: String { return self._s[1453]! } - public var Stats_GroupViewers: String { return self._s[1454]! } - public var AutoDownloadSettings_Channels: String { return self._s[1455]! } - public var PhotoEditor_QualityHigh: String { return self._s[1456]! } - public var VoiceChat_Leave: String { return self._s[1457]! } - public var Call_Speaker: String { return self._s[1458]! } + public var Conversation_DeleteMessagesForEveryone: String { return self._s[1459]! } + public var Permissions_SiriTitle_v0: String { return self._s[1460]! } + public var GroupInfo_AddUserLeftError: String { return self._s[1461]! } + public var Conversation_SendMessage_SendSilently: String { return self._s[1462]! } + public var Paint_Duplicate: String { return self._s[1463]! } + public var AttachmentMenu_WebSearch: String { return self._s[1464]! } + public var Bot_Stop: String { return self._s[1466]! } + public var Conversation_PrivateChannelTimeLimitedAlertTitle: String { return self._s[1467]! } + public var ReportGroupLocation_Report: String { return self._s[1468]! } + public var Compose_Create: String { return self._s[1469]! } + public var Stats_GroupViewers: String { return self._s[1470]! } + public var AutoDownloadSettings_Channels: String { return self._s[1471]! } + public var PhotoEditor_QualityHigh: String { return self._s[1472]! } + public var VoiceChat_Leave: String { return self._s[1473]! } + public var Call_Speaker: String { return self._s[1474]! } public func ChatList_LeaveGroupConfirmation(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1459]!, self._r[1459]!, [_0]) + return formatWithArgumentRanges(self._s[1475]!, self._r[1475]!, [_0]) } - public var Conversation_CloudStorage_ChatStatus: String { return self._s[1460]! } - public var Chat_AttachmentMultipleFilesDisabled: String { return self._s[1461]! } - public var ChatList_Context_AddToFolder: String { return self._s[1462]! } - public var Conversation_Unblock: String { return self._s[1463]! } - public var SettingsSearch_Synonyms_Proxy_UseForCalls: String { return self._s[1464]! } + public var Conversation_CloudStorage_ChatStatus: String { return self._s[1476]! } + public var Chat_AttachmentMultipleFilesDisabled: String { return self._s[1477]! } + public var ChatList_Context_AddToFolder: String { return self._s[1478]! } + public var InviteLink_QRCode_Info: String { return self._s[1479]! } + public var Conversation_Unblock: String { return self._s[1480]! } + public var SettingsSearch_Synonyms_Proxy_UseForCalls: String { return self._s[1481]! } public func Time_PreciseDate_m8(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1465]!, self._r[1465]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[1482]!, self._r[1482]!, [_1, _2, _3]) } - public var Conversation_ContextMenuReply: String { return self._s[1466]! } - public var Contacts_SearchLabel: String { return self._s[1467]! } - public var Forward_ErrorPublicQuizDisabledInChannels: String { return self._s[1468]! } - public var Stats_GroupMessagesTitle: String { return self._s[1470]! } - public var Notification_CallCanceled: String { return self._s[1471]! } - public var VoiceOver_Chat_Selected: String { return self._s[1472]! } - public var NotificationsSound_Tremolo: String { return self._s[1474]! } - public var ChatList_Search_NoResultsDescription: String { return self._s[1475]! } - public var AccessDenied_PhotosAndVideos: String { return self._s[1476]! } - public var LogoutOptions_ClearCacheText: String { return self._s[1477]! } - public var ChatListFolder_NameUnread: String { return self._s[1479]! } - public var PeerInfo_ButtonMessage: String { return self._s[1481]! } - public var InfoPlist_NSPhotoLibraryAddUsageDescription: String { return self._s[1482]! } - public var BlockedUsers_SelectUserTitle: String { return self._s[1483]! } - public var ChatSettings_Other: String { return self._s[1484]! } - public var UserInfo_NotificationsEnabled: String { return self._s[1485]! } - public var CreatePoll_OptionsHeader: String { return self._s[1486]! } - public var Appearance_RemoveThemeColorConfirmation: String { return self._s[1489]! } - public var Channel_Moderator_Title: String { return self._s[1490]! } - public var Channel_AdminLog_MessageRestrictedForever: String { return self._s[1491]! } - public var WallpaperColors_Title: String { return self._s[1492]! } - public var PrivacyPolicy_DeclineMessage: String { return self._s[1494]! } - public var AutoDownloadSettings_VoiceMessagesTitle: String { return self._s[1495]! } - public var Your_card_was_declined: String { return self._s[1496]! } - public var SettingsSearch_FAQ: String { return self._s[1498]! } - public var EditTheme_Expand_Preview_IncomingReplyName: String { return self._s[1499]! } - public var Conversation_ReportSpamConfirmation: String { return self._s[1500]! } - public var OwnershipTransfer_SecurityCheck: String { return self._s[1502]! } - public var PrivacySettings_DataSettingsHelp: String { return self._s[1503]! } - public var Settings_About_Help: String { return self._s[1504]! } + public var Conversation_ContextMenuReply: String { return self._s[1483]! } + public var Contacts_SearchLabel: String { return self._s[1484]! } + public var Forward_ErrorPublicQuizDisabledInChannels: String { return self._s[1485]! } + public var Stats_GroupMessagesTitle: String { return self._s[1487]! } + public var Notification_CallCanceled: String { return self._s[1488]! } + public var VoiceOver_Chat_Selected: String { return self._s[1489]! } + public var NotificationsSound_Tremolo: String { return self._s[1491]! } + public var ChatList_Search_NoResultsDescription: String { return self._s[1492]! } + public var AccessDenied_PhotosAndVideos: String { return self._s[1493]! } + public var LogoutOptions_ClearCacheText: String { return self._s[1494]! } + public var ChatListFolder_NameUnread: String { return self._s[1496]! } + public var PeerInfo_ButtonMessage: String { return self._s[1498]! } + public var InfoPlist_NSPhotoLibraryAddUsageDescription: String { return self._s[1499]! } + public var BlockedUsers_SelectUserTitle: String { return self._s[1500]! } + public var ChatSettings_Other: String { return self._s[1501]! } + public var UserInfo_NotificationsEnabled: String { return self._s[1502]! } + public var CreatePoll_OptionsHeader: String { return self._s[1503]! } + public var Appearance_RemoveThemeColorConfirmation: String { return self._s[1506]! } + public var Channel_Moderator_Title: String { return self._s[1507]! } + public var Channel_AdminLog_MessageRestrictedForever: String { return self._s[1508]! } + public var WallpaperColors_Title: String { return self._s[1509]! } + public var InviteLink_InviteLink: String { return self._s[1511]! } + public var PrivacyPolicy_DeclineMessage: String { return self._s[1512]! } + public var AutoDownloadSettings_VoiceMessagesTitle: String { return self._s[1513]! } + public var Your_card_was_declined: String { return self._s[1514]! } + public var SettingsSearch_FAQ: String { return self._s[1516]! } + public var EditTheme_Expand_Preview_IncomingReplyName: String { return self._s[1517]! } + public var Conversation_ReportSpamConfirmation: String { return self._s[1518]! } + public var OwnershipTransfer_SecurityCheck: String { return self._s[1520]! } + public var PrivacySettings_DataSettingsHelp: String { return self._s[1521]! } + public var Settings_About_Help: String { return self._s[1522]! } public func Channel_DiscussionGroup_HeaderGroupSet(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1505]!, self._r[1505]!, [_0]) + return formatWithArgumentRanges(self._s[1523]!, self._r[1523]!, [_0]) } - public var Settings_Proxy: String { return self._s[1506]! } - public var TwoStepAuth_ResetAccountConfirmation: String { return self._s[1507]! } - public var Passport_Identity_TypePassportUploadScan: String { return self._s[1509]! } - public var NotificationsSound_Bell: String { return self._s[1510]! } - public var PrivacySettings_Title: String { return self._s[1512]! } - public var PrivacySettings_DataSettings: String { return self._s[1513]! } - public var ConversationMedia_Title: String { return self._s[1514]! } + public var Settings_Proxy: String { return self._s[1524]! } + public var TwoStepAuth_ResetAccountConfirmation: String { return self._s[1525]! } + public var Passport_Identity_TypePassportUploadScan: String { return self._s[1527]! } + public var NotificationsSound_Bell: String { return self._s[1528]! } + public var PrivacySettings_Title: String { return self._s[1530]! } + public var PrivacySettings_DataSettings: String { return self._s[1531]! } + public var ConversationMedia_Title: String { return self._s[1532]! } public func Conversation_EncryptedPlaceholderTitleIncoming(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1515]!, self._r[1515]!, [_0]) + return formatWithArgumentRanges(self._s[1533]!, self._r[1533]!, [_0]) } - public var PrivacySettings_BlockedPeersEmpty: String { return self._s[1516]! } - public var ReportPeer_ReasonPornography: String { return self._s[1518]! } - public var Privacy_Calls: String { return self._s[1519]! } - public var TwoFactorSetup_Email_Text: String { return self._s[1520]! } - public var Conversation_EncryptedDescriptionTitle: String { return self._s[1521]! } + public var PrivacySettings_BlockedPeersEmpty: String { return self._s[1534]! } + public var ReportPeer_ReasonPornography: String { return self._s[1536]! } + public var Privacy_Calls: String { return self._s[1537]! } + public var TwoFactorSetup_Email_Text: String { return self._s[1538]! } + public var Conversation_EncryptedDescriptionTitle: String { return self._s[1539]! } public func VoiceOver_Chat_MusicTitle(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1522]!, self._r[1522]!, [_1, _2]) + return formatWithArgumentRanges(self._s[1540]!, self._r[1540]!, [_1, _2]) } - public var Passport_Identity_FrontSideHelp: String { return self._s[1523]! } - public var GroupInfo_Permissions_SlowmodeHeader: String { return self._s[1525]! } - public var ContactList_Context_VideoCall: String { return self._s[1526]! } - public var Settings_SaveIncomingPhotos: String { return self._s[1527]! } - public var Passport_Identity_MiddleName: String { return self._s[1528]! } - public var MessagePoll_QuizNoUsers: String { return self._s[1529]! } + public var Passport_Identity_FrontSideHelp: String { return self._s[1541]! } + public var GroupInfo_Permissions_SlowmodeHeader: String { return self._s[1543]! } + public var ContactList_Context_VideoCall: String { return self._s[1544]! } + public var Settings_SaveIncomingPhotos: String { return self._s[1545]! } + public var Passport_Identity_MiddleName: String { return self._s[1546]! } + public var MessagePoll_QuizNoUsers: String { return self._s[1547]! } public func Channel_AdminLog_MutedParticipant(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1530]!, self._r[1530]!, [_1, _2]) + return formatWithArgumentRanges(self._s[1548]!, self._r[1548]!, [_1, _2]) } - public var OldChannels_ChannelFormat: String { return self._s[1531]! } - public var Watch_Message_Call: String { return self._s[1532]! } - public var Wallpaper_Title: String { return self._s[1533]! } - public var PasscodeSettings_TurnPasscodeOff: String { return self._s[1534]! } - public var IntentsSettings_SuggestedChatsSavedMessages: String { return self._s[1535]! } - public var ReportGroupLocation_Text: String { return self._s[1536]! } - public var InviteText_URL: String { return self._s[1537]! } - public var ClearCache_StorageServiceFiles: String { return self._s[1538]! } - public var MessageTimer_Custom: String { return self._s[1539]! } - public var Message_PinnedLocationMessage: String { return self._s[1540]! } + public var OldChannels_ChannelFormat: String { return self._s[1549]! } + public var Watch_Message_Call: String { return self._s[1550]! } + public var Wallpaper_Title: String { return self._s[1551]! } + public var PasscodeSettings_TurnPasscodeOff: String { return self._s[1552]! } + public var IntentsSettings_SuggestedChatsSavedMessages: String { return self._s[1553]! } + public var ReportGroupLocation_Text: String { return self._s[1554]! } + public var InviteText_URL: String { return self._s[1555]! } + public var ClearCache_StorageServiceFiles: String { return self._s[1556]! } + public var MessageTimer_Custom: String { return self._s[1557]! } + public var Message_PinnedLocationMessage: String { return self._s[1558]! } public func VoiceOver_Chat_ContactOrganization(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1541]!, self._r[1541]!, [_0]) + return formatWithArgumentRanges(self._s[1559]!, self._r[1559]!, [_0]) } - public var EditTheme_UploadNewTheme: String { return self._s[1542]! } + public var EditTheme_UploadNewTheme: String { return self._s[1560]! } public func AutoDownloadSettings_UpToForAll(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1545]!, self._r[1545]!, [_0]) + return formatWithArgumentRanges(self._s[1563]!, self._r[1563]!, [_0]) } - public var Login_CodeSentCall: String { return self._s[1547]! } - public var Conversation_Report: String { return self._s[1548]! } - public var NotificationSettings_ContactJoined: String { return self._s[1549]! } + public var Login_CodeSentCall: String { return self._s[1565]! } + public var Conversation_Report: String { return self._s[1566]! } + public var NotificationSettings_ContactJoined: String { return self._s[1567]! } public func PUSH_MESSAGE_SCREENSHOT(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1550]!, self._r[1550]!, [_1]) + return formatWithArgumentRanges(self._s[1568]!, self._r[1568]!, [_1]) } - public var StickerPacksSettings_ShowStickersButtonHelp: String { return self._s[1551]! } - public var IntentsSettings_SuggestByAll: String { return self._s[1552]! } - public var StickerPacksSettings_ShowStickersButton: String { return self._s[1553]! } - public var AuthSessions_Title: String { return self._s[1554]! } + public var StickerPacksSettings_ShowStickersButtonHelp: String { return self._s[1569]! } + public var IntentsSettings_SuggestByAll: String { return self._s[1570]! } + public var StickerPacksSettings_ShowStickersButton: String { return self._s[1571]! } + public var AuthSessions_Title: String { return self._s[1572]! } public func Notification_VoiceChatEnded(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1555]!, self._r[1555]!, [_0]) + return formatWithArgumentRanges(self._s[1573]!, self._r[1573]!, [_0]) } - public var Channel_AdminLog_TitleAllEvents: String { return self._s[1556]! } - public var KeyCommand_JumpToNextUnreadChat: String { return self._s[1557]! } - public var Passport_Address_AddPassportRegistration: String { return self._s[1561]! } - public var AutoDownloadSettings_MaxVideoSize: String { return self._s[1562]! } - public var ExplicitContent_AlertTitle: String { return self._s[1563]! } - public var Channel_UpdatePhotoItem: String { return self._s[1564]! } - public var ChatList_AutoarchiveSuggestion_Text: String { return self._s[1566]! } - public var Channel_DiscussionGroup_LinkGroup: String { return self._s[1567]! } + public var Channel_AdminLog_TitleAllEvents: String { return self._s[1574]! } + public var KeyCommand_JumpToNextUnreadChat: String { return self._s[1575]! } + public var Passport_Address_AddPassportRegistration: String { return self._s[1579]! } + public var AutoDownloadSettings_MaxVideoSize: String { return self._s[1580]! } + public var ExplicitContent_AlertTitle: String { return self._s[1581]! } + public var Channel_UpdatePhotoItem: String { return self._s[1582]! } + public var ChatList_AutoarchiveSuggestion_Text: String { return self._s[1584]! } + public var Channel_DiscussionGroup_LinkGroup: String { return self._s[1585]! } public func Call_BatteryLow(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1568]!, self._r[1568]!, [_0]) + return formatWithArgumentRanges(self._s[1586]!, self._r[1586]!, [_0]) } - public var Login_HaveNotReceivedCodeInternal: String { return self._s[1569]! } - public var WallpaperPreview_PatternPaternApply: String { return self._s[1570]! } - public var Notifications_MessageNotificationsSound: String { return self._s[1571]! } - public var CommentsGroup_ErrorAccessDenied: String { return self._s[1572]! } - public var Appearance_AccentColor: String { return self._s[1574]! } - public var GroupInfo_SharedMedia: String { return self._s[1575]! } - public var Login_PhonePlaceholder: String { return self._s[1576]! } - public var Appearance_TextSize_Automatic: String { return self._s[1577]! } - public var EmptyGroupInfo_Line2: String { return self._s[1578]! } + public var Login_HaveNotReceivedCodeInternal: String { return self._s[1587]! } + public var WallpaperPreview_PatternPaternApply: String { return self._s[1588]! } + public var Notifications_MessageNotificationsSound: String { return self._s[1589]! } + public var CommentsGroup_ErrorAccessDenied: String { return self._s[1590]! } + public var Appearance_AccentColor: String { return self._s[1592]! } + public var GroupInfo_SharedMedia: String { return self._s[1593]! } + public var Login_PhonePlaceholder: String { return self._s[1594]! } + public var Appearance_TextSize_Automatic: String { return self._s[1595]! } + public var EmptyGroupInfo_Line2: String { return self._s[1596]! } public func PUSH_CHAT_CREATED(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1579]!, self._r[1579]!, [_1, _2]) + return formatWithArgumentRanges(self._s[1597]!, self._r[1597]!, [_1, _2]) } - public var Appearance_AppIconDefaultX: String { return self._s[1581]! } - public var EditProfile_NameAndPhotoOrVideoHelp: String { return self._s[1582]! } - public var CheckoutInfo_ShippingInfoPostcodePlaceholder: String { return self._s[1583]! } - public var Notifications_GroupNotificationsHelp: String { return self._s[1584]! } + public var Appearance_AppIconDefaultX: String { return self._s[1599]! } + public var EditProfile_NameAndPhotoOrVideoHelp: String { return self._s[1600]! } + public var CheckoutInfo_ShippingInfoPostcodePlaceholder: String { return self._s[1601]! } + public var Notifications_GroupNotificationsHelp: String { return self._s[1602]! } public func PUSH_CHAT_MESSAGE_NOTEXT(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1585]!, self._r[1585]!, [_1, _2]) + return formatWithArgumentRanges(self._s[1603]!, self._r[1603]!, [_1, _2]) } - public var ChatList_EmptyChatListEditFilter: String { return self._s[1586]! } - public var ChatSettings_ConnectionType_UseProxy: String { return self._s[1589]! } - public var Chat_PinnedMessagesHiddenText: String { return self._s[1590]! } + public var ChatList_EmptyChatListEditFilter: String { return self._s[1604]! } + public var ChatSettings_ConnectionType_UseProxy: String { return self._s[1607]! } + public var Chat_PinnedMessagesHiddenText: String { return self._s[1608]! } public func Message_PinnedGenericMessage(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1591]!, self._r[1591]!, [_0]) + return formatWithArgumentRanges(self._s[1609]!, self._r[1609]!, [_0]) } public func Location_ProximityTip(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1592]!, self._r[1592]!, [_0]) + return formatWithArgumentRanges(self._s[1610]!, self._r[1610]!, [_0]) } - public var UserInfo_NotificationsEnable: String { return self._s[1593]! } - public var Checkout_PayWithTouchId: String { return self._s[1594]! } - public var SharedMedia_ViewInChat: String { return self._s[1595]! } + public var UserInfo_NotificationsEnable: String { return self._s[1611]! } + public var Checkout_PayWithTouchId: String { return self._s[1612]! } + public var SharedMedia_ViewInChat: String { return self._s[1613]! } public func Notification_CreatedChatWithTitle(_ _0: String, _ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1596]!, self._r[1596]!, [_0, _1]) + return formatWithArgumentRanges(self._s[1614]!, self._r[1614]!, [_0, _1]) } - public var ChatSettings_AutoDownloadSettings_OffForAll: String { return self._s[1597]! } + public var ChatSettings_AutoDownloadSettings_OffForAll: String { return self._s[1615]! } public func Channel_DiscussionGroup_PublicChannelLink(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1598]!, self._r[1598]!, [_1, _2]) + return formatWithArgumentRanges(self._s[1616]!, self._r[1616]!, [_1, _2]) } public func Cache_Clear(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1600]!, self._r[1600]!, [_0]) + return formatWithArgumentRanges(self._s[1618]!, self._r[1618]!, [_0]) } - public var Conversation_PeerNearbyText: String { return self._s[1602]! } - public var Conversation_StopPollConfirmationTitle: String { return self._s[1603]! } - public var PhotoEditor_Skip: String { return self._s[1604]! } - public var SettingsSearch_Synonyms_Appearance_ChatBackground_SetColor: String { return self._s[1605]! } - public var ChatList_EmptyChatList: String { return self._s[1606]! } - public var Channel_BanUser_Unban: String { return self._s[1607]! } + public var Conversation_PeerNearbyText: String { return self._s[1620]! } + public var Conversation_StopPollConfirmationTitle: String { return self._s[1621]! } + public var PhotoEditor_Skip: String { return self._s[1622]! } + public var SettingsSearch_Synonyms_Appearance_ChatBackground_SetColor: String { return self._s[1623]! } + public var ChatList_EmptyChatList: String { return self._s[1624]! } + public var Channel_BanUser_Unban: String { return self._s[1625]! } public func Message_GenericForwardedPsa(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1608]!, self._r[1608]!, [_0]) + return formatWithArgumentRanges(self._s[1626]!, self._r[1626]!, [_0]) } - public var Appearance_TextSize_Apply: String { return self._s[1609]! } + public var Appearance_TextSize_Apply: String { return self._s[1627]! } public func Conversation_MessageViewCommentsFormat(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1610]!, self._r[1610]!, [_1, _2]) + return formatWithArgumentRanges(self._s[1628]!, self._r[1628]!, [_1, _2]) } - public var Login_InfoFirstNamePlaceholder: String { return self._s[1611]! } - public var TwoStepAuth_HintPlaceholder: String { return self._s[1612]! } - public var TwoStepAuth_EmailSkip: String { return self._s[1614]! } - public var ChatList_UndoArchiveMultipleTitle: String { return self._s[1615]! } - public var TwoFactorSetup_Email_SkipConfirmationTitle: String { return self._s[1616]! } + public var Login_InfoFirstNamePlaceholder: String { return self._s[1629]! } + public var TwoStepAuth_HintPlaceholder: String { return self._s[1630]! } + public var TwoStepAuth_EmailSkip: String { return self._s[1632]! } + public var ChatList_UndoArchiveMultipleTitle: String { return self._s[1633]! } + public var TwoFactorSetup_Email_SkipConfirmationTitle: String { return self._s[1634]! } public func PUSH_MESSAGE_QUIZ(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1617]!, self._r[1617]!, [_1]) + return formatWithArgumentRanges(self._s[1635]!, self._r[1635]!, [_1]) } - public var State_WaitingForNetwork: String { return self._s[1619]! } - public var AccessDenied_CameraRestricted: String { return self._s[1620]! } - public var ChatSettings_Appearance: String { return self._s[1621]! } - public var ScheduledMessages_BotActionUnavailable: String { return self._s[1622]! } - public var GroupInfo_InviteLink_CopyAlert_Success: String { return self._s[1623]! } - public var Channel_DiscussionGroupAdd: String { return self._s[1624]! } - public var Map_NoPlacesNearby: String { return self._s[1626]! } - public var AuthSessions_IncompleteAttemptsInfo: String { return self._s[1627]! } - public var GroupRemoved_Title: String { return self._s[1628]! } - public var TwoStepAuth_EnterPasswordHelp: String { return self._s[1630]! } - public var VoiceChat_Mute: String { return self._s[1631]! } - public var Paint_Marker: String { return self._s[1632]! } + public var State_WaitingForNetwork: String { return self._s[1637]! } + public var AccessDenied_CameraRestricted: String { return self._s[1638]! } + public var ChatSettings_Appearance: String { return self._s[1639]! } + public var ScheduledMessages_BotActionUnavailable: String { return self._s[1640]! } + public var GroupInfo_InviteLink_CopyAlert_Success: String { return self._s[1641]! } + public var Channel_DiscussionGroupAdd: String { return self._s[1642]! } + public var Map_NoPlacesNearby: String { return self._s[1644]! } + public var AuthSessions_IncompleteAttemptsInfo: String { return self._s[1645]! } + public var GroupRemoved_Title: String { return self._s[1646]! } + public var TwoStepAuth_EnterPasswordHelp: String { return self._s[1648]! } + public var VoiceChat_Mute: String { return self._s[1649]! } + public var Paint_Marker: String { return self._s[1650]! } public func AddContact_ContactWillBeSharedAfterMutual(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1633]!, self._r[1633]!, [_1]) + return formatWithArgumentRanges(self._s[1651]!, self._r[1651]!, [_1]) } - public var SocksProxySetup_ShareProxyList: String { return self._s[1634]! } - public var GroupInfo_InvitationLinkDoesNotExist: String { return self._s[1635]! } + public var SocksProxySetup_ShareProxyList: String { return self._s[1652]! } + public var GroupInfo_InvitationLinkDoesNotExist: String { return self._s[1653]! } public func VoiceOver_Chat_Size(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1636]!, self._r[1636]!, [_0]) + return formatWithArgumentRanges(self._s[1654]!, self._r[1654]!, [_0]) } - public var EditTheme_ErrorInvalidCharacters: String { return self._s[1637]! } - public var Appearance_ThemePreview_ChatList_7_Name: String { return self._s[1638]! } - public var Notifications_GroupNotificationsAlert: String { return self._s[1639]! } - public var SocksProxySetup_ShareQRCode: String { return self._s[1640]! } - public var Compose_NewGroup: String { return self._s[1641]! } + public var EditTheme_ErrorInvalidCharacters: String { return self._s[1655]! } + public var Appearance_ThemePreview_ChatList_7_Name: String { return self._s[1656]! } + public var Notifications_GroupNotificationsAlert: String { return self._s[1657]! } + public var SocksProxySetup_ShareQRCode: String { return self._s[1658]! } + public var Compose_NewGroup: String { return self._s[1659]! } public func Passport_Address_UploadOneOfScan(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1642]!, self._r[1642]!, [_0]) - } - public var Location_LiveLocationRequired_Description: String { return self._s[1644]! } - public var Conversation_ClearGroupHistory: String { return self._s[1645]! } - public var GroupInfo_InviteLink_Help: String { return self._s[1648]! } - public var Channel_BanUser_BlockFor: String { return self._s[1649]! } - public var Bot_Start: String { return self._s[1650]! } - public var Your_card_has_expired: String { return self._s[1651]! } - public var Channel_About_Title: String { return self._s[1652]! } - public var Passport_Identity_ExpiryDatePlaceholder: String { return self._s[1653]! } - public var SettingsSearch_Synonyms_Notifications_MessageNotificationsExceptions: String { return self._s[1655]! } - public var Conversation_FileDropbox: String { return self._s[1656]! } - public var ChatList_Search_NoResultsFitlerMusic: String { return self._s[1657]! } - public var Month_GenNovember: String { return self._s[1658]! } - public var IntentsSettings_SuggestByShare: String { return self._s[1659]! } - public func Call_PrivacyErrorMessage(_ _0: String) -> (String, [(Int, NSRange)]) { return formatWithArgumentRanges(self._s[1660]!, self._r[1660]!, [_0]) } - public var StickerPack_Add: String { return self._s[1661]! } - public var Theme_ErrorNotFound: String { return self._s[1662]! } - public var Wallpaper_SearchShort: String { return self._s[1664]! } - public var Channel_BanUser_PermissionsHeader: String { return self._s[1665]! } - public var ConversationProfile_UsersTooMuchError: String { return self._s[1666]! } - public var ChatList_FolderAllChats: String { return self._s[1667]! } - public var VoiceChat_EndConfirmationEnd: String { return self._s[1668]! } - public var Passport_Authorize: String { return self._s[1669]! } + public var Location_LiveLocationRequired_Description: String { return self._s[1662]! } + public var Conversation_ClearGroupHistory: String { return self._s[1663]! } + public var GroupInfo_InviteLink_Help: String { return self._s[1666]! } + public var Channel_BanUser_BlockFor: String { return self._s[1667]! } + public var Bot_Start: String { return self._s[1668]! } + public var Your_card_has_expired: String { return self._s[1669]! } + public var Channel_About_Title: String { return self._s[1670]! } + public var Passport_Identity_ExpiryDatePlaceholder: String { return self._s[1671]! } + public var SettingsSearch_Synonyms_Notifications_MessageNotificationsExceptions: String { return self._s[1673]! } + public var Conversation_FileDropbox: String { return self._s[1674]! } + public var ChatList_Search_NoResultsFitlerMusic: String { return self._s[1675]! } + public var Month_GenNovember: String { return self._s[1676]! } + public var IntentsSettings_SuggestByShare: String { return self._s[1677]! } + public func Call_PrivacyErrorMessage(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[1678]!, self._r[1678]!, [_0]) + } + public var StickerPack_Add: String { return self._s[1679]! } + public var Theme_ErrorNotFound: String { return self._s[1680]! } + public var Wallpaper_SearchShort: String { return self._s[1682]! } + public var Channel_BanUser_PermissionsHeader: String { return self._s[1683]! } + public var ConversationProfile_UsersTooMuchError: String { return self._s[1684]! } + public var ChatList_FolderAllChats: String { return self._s[1685]! } + public var VoiceChat_EndConfirmationEnd: String { return self._s[1686]! } + public var Passport_Authorize: String { return self._s[1687]! } public func Channel_AdminLog_MessageChangedLinkedChannel(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1670]!, self._r[1670]!, [_1, _2]) + return formatWithArgumentRanges(self._s[1688]!, self._r[1688]!, [_1, _2]) } - public var GroupInfo_GroupHistoryVisible: String { return self._s[1671]! } + public var GroupInfo_GroupHistoryVisible: String { return self._s[1689]! } public func PUSH_MESSAGE_VIDEO(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1672]!, self._r[1672]!, [_1]) + return formatWithArgumentRanges(self._s[1690]!, self._r[1690]!, [_1]) } - public var LocalGroup_ButtonTitle: String { return self._s[1673]! } - public var UserInfo_GroupsInCommon: String { return self._s[1675]! } - public var LoginPassword_Title: String { return self._s[1677]! } - public var Wallpaper_Set: String { return self._s[1678]! } - public var Stats_InteractionsTitle: String { return self._s[1679]! } + public var LocalGroup_ButtonTitle: String { return self._s[1691]! } + public var UserInfo_GroupsInCommon: String { return self._s[1693]! } + public var LoginPassword_Title: String { return self._s[1695]! } + public var Wallpaper_Set: String { return self._s[1696]! } + public var Stats_InteractionsTitle: String { return self._s[1697]! } public func SecretGIF_NotViewedYet(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1681]!, self._r[1681]!, [_0]) + return formatWithArgumentRanges(self._s[1699]!, self._r[1699]!, [_0]) } - public var Conversation_MessageDialogEdit: String { return self._s[1682]! } - public var Paint_Outlined: String { return self._s[1683]! } - public var VoiceChat_Rec: String { return self._s[1684]! } + public var Conversation_MessageDialogEdit: String { return self._s[1700]! } + public var Paint_Outlined: String { return self._s[1701]! } + public var VoiceChat_Rec: String { return self._s[1702]! } public func Login_ResetAccountProtected_Text(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1685]!, self._r[1685]!, [_0]) + return formatWithArgumentRanges(self._s[1703]!, self._r[1703]!, [_0]) } public func Conversation_SetReminder_RemindTomorrow(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1686]!, self._r[1686]!, [_0]) + return formatWithArgumentRanges(self._s[1704]!, self._r[1704]!, [_0]) } - public var Invite_LargeRecipientsCountWarning: String { return self._s[1687]! } - public var Passport_Address_Street1Placeholder: String { return self._s[1688]! } - public var Appearance_ColorThemeNight: String { return self._s[1689]! } - public var ChannelInfo_Stats: String { return self._s[1690]! } - public var TwoStepAuth_RecoveryTitle: String { return self._s[1691]! } - public var MediaPicker_TimerTooltip: String { return self._s[1692]! } - public var Common_ChoosePhoto: String { return self._s[1693]! } - public var Media_LimitedAccessTitle: String { return self._s[1694]! } - public var ChatSettings_AutoDownloadVideos: String { return self._s[1695]! } - public var PeerInfo_PaneGroups: String { return self._s[1696]! } - public var SocksProxySetup_UsernamePlaceholder: String { return self._s[1698]! } - public var ChangePhoneNumberNumber_Title: String { return self._s[1699]! } - public var ContactInfo_PhoneLabelMobile: String { return self._s[1700]! } - public var OldChannels_ChannelsHeader: String { return self._s[1701]! } - public var MuteFor_Forever: String { return self._s[1702]! } - public var Passport_Address_PostcodePlaceholder: String { return self._s[1703]! } - public var SettingsSearch_Synonyms_Appearance_ChatBackground: String { return self._s[1704]! } - public var MessagePoll_LabelAnonymous: String { return self._s[1705]! } - public var ContactInfo_Job: String { return self._s[1706]! } - public var Passport_Language_mk: String { return self._s[1707]! } - public var EditTheme_ShortLink: String { return self._s[1708]! } - public var AutoDownloadSettings_PhotosTitle: String { return self._s[1710]! } - public var Month_GenApril: String { return self._s[1712]! } - public var Channel_DiscussionGroup_HeaderLabel: String { return self._s[1714]! } - public var NetworkUsageSettings_TotalSection: String { return self._s[1715]! } - public var EditTheme_Create_Preview_OutgoingText: String { return self._s[1716]! } - public var EditTheme_Title: String { return self._s[1717]! } - public var Conversation_LinkDialogCopy: String { return self._s[1718]! } + public var Invite_LargeRecipientsCountWarning: String { return self._s[1705]! } + public var Passport_Address_Street1Placeholder: String { return self._s[1706]! } + public var Appearance_ColorThemeNight: String { return self._s[1707]! } + public var ChannelInfo_Stats: String { return self._s[1708]! } + public var TwoStepAuth_RecoveryTitle: String { return self._s[1709]! } + public var MediaPicker_TimerTooltip: String { return self._s[1710]! } + public var Common_ChoosePhoto: String { return self._s[1711]! } + public var Media_LimitedAccessTitle: String { return self._s[1712]! } + public var ChatSettings_AutoDownloadVideos: String { return self._s[1713]! } + public var PeerInfo_PaneGroups: String { return self._s[1714]! } + public var SocksProxySetup_UsernamePlaceholder: String { return self._s[1716]! } + public var ChangePhoneNumberNumber_Title: String { return self._s[1717]! } + public var ContactInfo_PhoneLabelMobile: String { return self._s[1718]! } + public var OldChannels_ChannelsHeader: String { return self._s[1719]! } + public var MuteFor_Forever: String { return self._s[1720]! } + public var Passport_Address_PostcodePlaceholder: String { return self._s[1721]! } + public var SettingsSearch_Synonyms_Appearance_ChatBackground: String { return self._s[1722]! } + public var MessagePoll_LabelAnonymous: String { return self._s[1723]! } + public var ContactInfo_Job: String { return self._s[1724]! } + public var Passport_Language_mk: String { return self._s[1725]! } + public var EditTheme_ShortLink: String { return self._s[1726]! } + public var AutoDownloadSettings_PhotosTitle: String { return self._s[1728]! } + public var Month_GenApril: String { return self._s[1730]! } + public var Channel_DiscussionGroup_HeaderLabel: String { return self._s[1732]! } + public var NetworkUsageSettings_TotalSection: String { return self._s[1733]! } + public var EditTheme_Create_Preview_OutgoingText: String { return self._s[1734]! } + public var EditTheme_Title: String { return self._s[1735]! } + public var Conversation_LinkDialogCopy: String { return self._s[1736]! } public func Channel_AdminLog_MessageInvitedNameUsername(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1719]!, self._r[1719]!, [_1, _2]) + return formatWithArgumentRanges(self._s[1737]!, self._r[1737]!, [_1, _2]) } - public var Passport_ForgottenPassword: String { return self._s[1720]! } - public var WallpaperSearch_Recent: String { return self._s[1721]! } - public var ChatSettings_Title: String { return self._s[1726]! } - public var Appearance_ReduceMotionInfo: String { return self._s[1727]! } + public var Passport_ForgottenPassword: String { return self._s[1738]! } + public var WallpaperSearch_Recent: String { return self._s[1739]! } + public var ChatSettings_Title: String { return self._s[1744]! } + public var Appearance_ReduceMotionInfo: String { return self._s[1745]! } public func StickerPackActionInfo_AddedText(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1728]!, self._r[1728]!, [_0]) + return formatWithArgumentRanges(self._s[1746]!, self._r[1746]!, [_0]) } - public var SocksProxySetup_UseForCallsHelp: String { return self._s[1729]! } - public var LastSeen_WithinAMonth: String { return self._s[1730]! } - public var VoiceChat_Live: String { return self._s[1731]! } - public var PeerInfo_ButtonCall: String { return self._s[1732]! } - public var SettingsSearch_Synonyms_Appearance_Title: String { return self._s[1733]! } - public var Group_Username_InvalidStartsWithNumber: String { return self._s[1734]! } - public var Call_AudioRouteHide: String { return self._s[1735]! } - public var DialogList_SavedMessages: String { return self._s[1736]! } - public var ChatList_Context_Mute: String { return self._s[1737]! } - public var Conversation_StatusKickedFromChannel: String { return self._s[1738]! } + public var SocksProxySetup_UseForCallsHelp: String { return self._s[1747]! } + public var LastSeen_WithinAMonth: String { return self._s[1748]! } + public var VoiceChat_Live: String { return self._s[1749]! } + public var PeerInfo_ButtonCall: String { return self._s[1750]! } + public var SettingsSearch_Synonyms_Appearance_Title: String { return self._s[1751]! } + public var Group_Username_InvalidStartsWithNumber: String { return self._s[1752]! } + public var Call_AudioRouteHide: String { return self._s[1753]! } + public var DialogList_SavedMessages: String { return self._s[1754]! } + public var ChatList_Context_Mute: String { return self._s[1755]! } + public var Conversation_StatusKickedFromChannel: String { return self._s[1756]! } public func Notification_Exceptions_MutedUntil(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1739]!, self._r[1739]!, [_0]) + return formatWithArgumentRanges(self._s[1757]!, self._r[1757]!, [_0]) } - public var VoiceChat_StatusMutedForYou: String { return self._s[1740]! } - public var Passport_Language_et: String { return self._s[1741]! } - public var Conversation_MessageLeaveCommentShort: String { return self._s[1742]! } - public var PhotoEditor_CropReset: String { return self._s[1743]! } - public var Privacy_GroupsAndChannels_AlwaysAllow: String { return self._s[1744]! } - public var SocksProxySetup_HostnamePlaceholder: String { return self._s[1745]! } - public var CreateGroup_ErrorLocatedGroupsTooMuch: String { return self._s[1746]! } - public var WallpaperSearch_ColorWhite: String { return self._s[1749]! } - public var Channel_AdminLog_CanEditMessages: String { return self._s[1751]! } - public var Privacy_PaymentsClearInfoDoneHelp: String { return self._s[1752]! } - public var Channel_Username_InvalidStartsWithNumber: String { return self._s[1754]! } - public var CheckoutInfo_ReceiverInfoName: String { return self._s[1756]! } - public var Map_YouAreHere: String { return self._s[1758]! } - public var Core_ServiceUserStatus: String { return self._s[1759]! } - public var Channel_Setup_TypePrivateHelp: String { return self._s[1762]! } - public var VoiceChat_StartRecording: String { return self._s[1763]! } - public var SettingsSearch_Synonyms_Notifications_BadgeCountUnreadMessages: String { return self._s[1764]! } - public var MediaPicker_Videos: String { return self._s[1766]! } - public var Map_LiveLocationFor15Minutes: String { return self._s[1768]! } - public var Passport_Identity_TranslationsHelp: String { return self._s[1769]! } - public var SharedMedia_CategoryMedia: String { return self._s[1770]! } + public var VoiceChat_StatusMutedForYou: String { return self._s[1758]! } + public var Passport_Language_et: String { return self._s[1759]! } + public var Conversation_MessageLeaveCommentShort: String { return self._s[1760]! } + public var PhotoEditor_CropReset: String { return self._s[1761]! } + public var Privacy_GroupsAndChannels_AlwaysAllow: String { return self._s[1762]! } + public var SocksProxySetup_HostnamePlaceholder: String { return self._s[1763]! } + public var CreateGroup_ErrorLocatedGroupsTooMuch: String { return self._s[1764]! } + public var WallpaperSearch_ColorWhite: String { return self._s[1767]! } + public var Channel_AdminLog_CanEditMessages: String { return self._s[1769]! } + public var Privacy_PaymentsClearInfoDoneHelp: String { return self._s[1770]! } + public var Channel_Username_InvalidStartsWithNumber: String { return self._s[1772]! } + public var CheckoutInfo_ReceiverInfoName: String { return self._s[1774]! } + public var Map_YouAreHere: String { return self._s[1776]! } + public var Core_ServiceUserStatus: String { return self._s[1777]! } + public var Channel_Setup_TypePrivateHelp: String { return self._s[1780]! } + public var VoiceChat_StartRecording: String { return self._s[1781]! } + public var SettingsSearch_Synonyms_Notifications_BadgeCountUnreadMessages: String { return self._s[1782]! } + public var MediaPicker_Videos: String { return self._s[1784]! } + public var Map_LiveLocationFor15Minutes: String { return self._s[1786]! } + public var Passport_Identity_TranslationsHelp: String { return self._s[1787]! } + public var SharedMedia_CategoryMedia: String { return self._s[1788]! } public func MediaPicker_Nof(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1771]!, self._r[1771]!, [_0]) - } - public var ChatSettings_AutoPlayGifs: String { return self._s[1772]! } - public var Passport_Identity_CountryPlaceholder: String { return self._s[1773]! } - public var Bot_GroupStatusDoesNotReadHistory: String { return self._s[1774]! } - public var Notification_Exceptions_RemoveFromExceptions: String { return self._s[1775]! } - public func Chat_SlowmodeTooltip(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1776]!, self._r[1776]!, [_0]) - } - public var Web_Error: String { return self._s[1777]! } - public var PhotoEditor_SkinTool: String { return self._s[1778]! } - public var ApplyLanguage_UnsufficientDataTitle: String { return self._s[1779]! } - public var ChatSettings_ConnectionType_UseSocks5: String { return self._s[1781]! } - public var PasscodeSettings_Help: String { return self._s[1782]! } - public var Appearance_ColorTheme: String { return self._s[1783]! } - public func Channel_AdminLog_MessageRestrictedNewSetting(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1784]!, self._r[1784]!, [_0]) - } - public func PUSH_PINNED_GEO(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1785]!, self._r[1785]!, [_1]) - } - public var GroupInfo_LeftStatus: String { return self._s[1786]! } - public var EditTheme_Preview: String { return self._s[1787]! } - public var Watch_Suggestion_WhatsUp: String { return self._s[1788]! } - public func AutoDownloadSettings_PreloadVideoInfo(_ _0: String) -> (String, [(Int, NSRange)]) { return formatWithArgumentRanges(self._s[1789]!, self._r[1789]!, [_0]) } - public var NotificationsSound_Keys: String { return self._s[1790]! } - public var PasscodeSettings_UnlockWithTouchId: String { return self._s[1791]! } - public var ChatList_Context_MarkAsUnread: String { return self._s[1792]! } - public var DialogList_AdNoticeAlert: String { return self._s[1793]! } - public var UserInfo_Invite: String { return self._s[1794]! } - public var Checkout_Email: String { return self._s[1795]! } - public var Stats_GroupActionsTitle: String { return self._s[1796]! } - public var Coub_TapForSound: String { return self._s[1797]! } - public var Theme_ThemeChangedText: String { return self._s[1798]! } - public var Call_ExternalCallInProgressMessage: String { return self._s[1799]! } - public var Settings_ApplyProxyAlertEnable: String { return self._s[1800]! } - public var ScheduledMessages_ScheduledToday: String { return self._s[1801]! } - public var Channel_AdminLog_DefaultRestrictionsUpdated: String { return self._s[1802]! } - public var Call_ReportIncludeLogDescription: String { return self._s[1803]! } - public var Settings_FrequentlyAskedQuestions: String { return self._s[1805]! } - public var Call_VoiceOver_VoiceCallMissed: String { return self._s[1806]! } - public var Channel_MessagePhotoRemoved: String { return self._s[1807]! } - public var Passport_Email_Delete: String { return self._s[1808]! } + public var ChatSettings_AutoPlayGifs: String { return self._s[1790]! } + public var Passport_Identity_CountryPlaceholder: String { return self._s[1791]! } + public var Bot_GroupStatusDoesNotReadHistory: String { return self._s[1792]! } + public var Notification_Exceptions_RemoveFromExceptions: String { return self._s[1793]! } + public func Chat_SlowmodeTooltip(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[1794]!, self._r[1794]!, [_0]) + } + public var Web_Error: String { return self._s[1795]! } + public var PhotoEditor_SkinTool: String { return self._s[1796]! } + public var ApplyLanguage_UnsufficientDataTitle: String { return self._s[1797]! } + public var ChatSettings_ConnectionType_UseSocks5: String { return self._s[1799]! } + public var PasscodeSettings_Help: String { return self._s[1800]! } + public var Appearance_ColorTheme: String { return self._s[1801]! } + public func Channel_AdminLog_MessageRestrictedNewSetting(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[1802]!, self._r[1802]!, [_0]) + } + public var InviteLink_DeleteAllRevokedLinks: String { return self._s[1803]! } + public func PUSH_PINNED_GEO(_ _1: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[1804]!, self._r[1804]!, [_1]) + } + public var InviteLink_QRCode_Title: String { return self._s[1805]! } + public var GroupInfo_LeftStatus: String { return self._s[1806]! } + public var EditTheme_Preview: String { return self._s[1807]! } + public var Watch_Suggestion_WhatsUp: String { return self._s[1808]! } + public func AutoDownloadSettings_PreloadVideoInfo(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[1809]!, self._r[1809]!, [_0]) + } + public var NotificationsSound_Keys: String { return self._s[1810]! } + public var PasscodeSettings_UnlockWithTouchId: String { return self._s[1811]! } + public var ChatList_Context_MarkAsUnread: String { return self._s[1812]! } + public var DialogList_AdNoticeAlert: String { return self._s[1813]! } + public var UserInfo_Invite: String { return self._s[1814]! } + public var Checkout_Email: String { return self._s[1815]! } + public var Stats_GroupActionsTitle: String { return self._s[1816]! } + public var Coub_TapForSound: String { return self._s[1817]! } + public var Theme_ThemeChangedText: String { return self._s[1818]! } + public var Call_ExternalCallInProgressMessage: String { return self._s[1819]! } + public var Settings_ApplyProxyAlertEnable: String { return self._s[1820]! } + public var ScheduledMessages_ScheduledToday: String { return self._s[1821]! } + public var Channel_AdminLog_DefaultRestrictionsUpdated: String { return self._s[1822]! } + public var Call_ReportIncludeLogDescription: String { return self._s[1823]! } + public var Settings_FrequentlyAskedQuestions: String { return self._s[1825]! } + public var Call_VoiceOver_VoiceCallMissed: String { return self._s[1826]! } + public var Channel_MessagePhotoRemoved: String { return self._s[1827]! } + public var Passport_Email_Delete: String { return self._s[1828]! } public func PUSH_PINNED_PHOTO(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1809]!, self._r[1809]!, [_1]) + return formatWithArgumentRanges(self._s[1829]!, self._r[1829]!, [_1]) } - public var NotificationSettings_ShowNotificationsAllAccountsInfoOn: String { return self._s[1810]! } - public var Channel_AdminLog_CanAddAdmins: String { return self._s[1811]! } - public var SocksProxySetup_FailedToConnect: String { return self._s[1813]! } - public var SettingsSearch_Synonyms_Data_NetworkUsage: String { return self._s[1814]! } - public var Common_of: String { return self._s[1815]! } - public var VoiceChat_StartRecordingStart: String { return self._s[1816]! } - public var VoiceChat_CreateNewVoiceChatText: String { return self._s[1817]! } - public var PeerInfo_ButtonUnmute: String { return self._s[1820]! } + public var NotificationSettings_ShowNotificationsAllAccountsInfoOn: String { return self._s[1830]! } + public var Channel_AdminLog_CanAddAdmins: String { return self._s[1831]! } + public var SocksProxySetup_FailedToConnect: String { return self._s[1833]! } + public var SettingsSearch_Synonyms_Data_NetworkUsage: String { return self._s[1834]! } + public var Common_of: String { return self._s[1835]! } + public var VoiceChat_StartRecordingStart: String { return self._s[1836]! } + public var VoiceChat_CreateNewVoiceChatText: String { return self._s[1837]! } + public var PeerInfo_ButtonUnmute: String { return self._s[1840]! } public func ChatSettings_AutoDownloadSettings_TypeFile(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1821]!, self._r[1821]!, [_0]) + return formatWithArgumentRanges(self._s[1841]!, self._r[1841]!, [_0]) } - public var ChatList_AddChatsToFolder: String { return self._s[1822]! } - public var Login_ResetAccountProtected_LimitExceeded: String { return self._s[1823]! } - public var Settings_Title: String { return self._s[1825]! } - public var AutoDownloadSettings_Contacts: String { return self._s[1827]! } - public var Appearance_BubbleCornersSetting: String { return self._s[1828]! } - public var Privacy_Calls_AlwaysAllow: String { return self._s[1829]! } - public var Privacy_Forwards_AlwaysAllow_Title: String { return self._s[1831]! } - public var WallpaperPreview_CropBottomText: String { return self._s[1832]! } - public var SecretTimer_VideoDescription: String { return self._s[1833]! } - public var WallpaperPreview_Blurred: String { return self._s[1834]! } - public var SettingsSearch_Synonyms_Notifications_GroupNotificationsExceptions: String { return self._s[1835]! } - public var ChatListFolder_ExcludedSectionHeader: String { return self._s[1837]! } - public var DialogList_PasscodeLockHelp: String { return self._s[1838]! } - public var SocksProxySetup_SecretPlaceholder: String { return self._s[1839]! } - public var NetworkUsageSettings_CallDataSection: String { return self._s[1840]! } - public var TwoStepAuth_PasswordRemovePassportConfirmation: String { return self._s[1841]! } - public var Passport_FieldAddressTranslationHelp: String { return self._s[1842]! } - public var SocksProxySetup_Connection: String { return self._s[1843]! } - public var Passport_Address_TypePassportRegistration: String { return self._s[1844]! } - public var Contacts_PermissionsAllowInSettings: String { return self._s[1845]! } - public var Conversation_Unpin: String { return self._s[1846]! } - public var Notifications_MessageNotificationsExceptionsHelp: String { return self._s[1847]! } - public var TwoFactorSetup_Hint_Placeholder: String { return self._s[1848]! } - public var Call_ReportSkip: String { return self._s[1849]! } + public var ChatList_AddChatsToFolder: String { return self._s[1842]! } + public var Login_ResetAccountProtected_LimitExceeded: String { return self._s[1843]! } + public var Settings_Title: String { return self._s[1845]! } + public var AutoDownloadSettings_Contacts: String { return self._s[1847]! } + public var Appearance_BubbleCornersSetting: String { return self._s[1848]! } + public var Privacy_Calls_AlwaysAllow: String { return self._s[1849]! } + public var Privacy_Forwards_AlwaysAllow_Title: String { return self._s[1851]! } + public var WallpaperPreview_CropBottomText: String { return self._s[1852]! } + public var SecretTimer_VideoDescription: String { return self._s[1853]! } + public var WallpaperPreview_Blurred: String { return self._s[1854]! } + public var SettingsSearch_Synonyms_Notifications_GroupNotificationsExceptions: String { return self._s[1855]! } + public var ChatListFolder_ExcludedSectionHeader: String { return self._s[1857]! } + public var DialogList_PasscodeLockHelp: String { return self._s[1858]! } + public var SocksProxySetup_SecretPlaceholder: String { return self._s[1859]! } + public var NetworkUsageSettings_CallDataSection: String { return self._s[1860]! } + public var TwoStepAuth_PasswordRemovePassportConfirmation: String { return self._s[1861]! } + public var Passport_FieldAddressTranslationHelp: String { return self._s[1862]! } + public var SocksProxySetup_Connection: String { return self._s[1863]! } + public var Passport_Address_TypePassportRegistration: String { return self._s[1864]! } + public var Contacts_PermissionsAllowInSettings: String { return self._s[1865]! } + public var Conversation_Unpin: String { return self._s[1866]! } + public var Notifications_MessageNotificationsExceptionsHelp: String { return self._s[1867]! } + public var TwoFactorSetup_Hint_Placeholder: String { return self._s[1868]! } + public var Call_ReportSkip: String { return self._s[1869]! } public func VoiceOver_Chat_PhotoFrom(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1850]!, self._r[1850]!, [_0]) + return formatWithArgumentRanges(self._s[1870]!, self._r[1870]!, [_0]) } public func VoiceOver_Chat_Caption(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1852]!, self._r[1852]!, [_0]) + return formatWithArgumentRanges(self._s[1872]!, self._r[1872]!, [_0]) } - public var AutoNightTheme_Automatic: String { return self._s[1853]! } - public var Passport_Language_az: String { return self._s[1854]! } - public var SettingsSearch_Synonyms_Data_Storage_ClearCache: String { return self._s[1855]! } - public var Watch_UserInfo_Unmute: String { return self._s[1856]! } - public var Channel_Stickers_YourStickers: String { return self._s[1857]! } - public var Channel_DiscussionGroup_UnlinkChannel: String { return self._s[1858]! } - public var Tour_Text1: String { return self._s[1859]! } - public var Common_Delete: String { return self._s[1860]! } - public var Settings_EditPhoto: String { return self._s[1861]! } - public var Common_Edit: String { return self._s[1862]! } + public var AutoNightTheme_Automatic: String { return self._s[1873]! } + public var Passport_Language_az: String { return self._s[1874]! } + public var SettingsSearch_Synonyms_Data_Storage_ClearCache: String { return self._s[1875]! } + public var Watch_UserInfo_Unmute: String { return self._s[1876]! } + public var Channel_Stickers_YourStickers: String { return self._s[1877]! } + public var Channel_DiscussionGroup_UnlinkChannel: String { return self._s[1878]! } + public var Tour_Text1: String { return self._s[1879]! } + public var Common_Delete: String { return self._s[1880]! } + public var Settings_EditPhoto: String { return self._s[1881]! } + public var Common_Edit: String { return self._s[1882]! } public func Channel_AdminLog_MutedNewMembers(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1864]!, self._r[1864]!, [_1]) + return formatWithArgumentRanges(self._s[1884]!, self._r[1884]!, [_1]) } - public var Passport_Identity_ExpiryDate: String { return self._s[1865]! } - public var ShareMenu_ShareTo: String { return self._s[1866]! } - public var Preview_DeleteGif: String { return self._s[1867]! } - public var WallpaperPreview_PatternPaternDiscard: String { return self._s[1868]! } - public var ChatSettings_AutoDownloadUsingCellular: String { return self._s[1869]! } - public var Conversation_ViewReply: String { return self._s[1870]! } - public var Stats_LoadingText: String { return self._s[1871]! } - public var Channel_EditAdmin_PermissinAddAdminOn: String { return self._s[1872]! } - public var CheckoutInfo_ReceiverInfoEmailPlaceholder: String { return self._s[1873]! } - public var Channel_AdminLog_CanChangeInfo: String { return self._s[1874]! } + public var Passport_Identity_ExpiryDate: String { return self._s[1885]! } + public var ShareMenu_ShareTo: String { return self._s[1886]! } + public var Preview_DeleteGif: String { return self._s[1887]! } + public var WallpaperPreview_PatternPaternDiscard: String { return self._s[1888]! } + public var ChatSettings_AutoDownloadUsingCellular: String { return self._s[1889]! } + public var Conversation_ViewReply: String { return self._s[1890]! } + public var Stats_LoadingText: String { return self._s[1891]! } + public var Channel_EditAdmin_PermissinAddAdminOn: String { return self._s[1892]! } + public var CheckoutInfo_ReceiverInfoEmailPlaceholder: String { return self._s[1893]! } + public var Channel_AdminLog_CanChangeInfo: String { return self._s[1894]! } public func Passport_Phone_UseTelegramNumber(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1875]!, self._r[1875]!, [_0]) + return formatWithArgumentRanges(self._s[1895]!, self._r[1895]!, [_0]) } public func Time_MonthOfYear_m2(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1876]!, self._r[1876]!, [_0]) + return formatWithArgumentRanges(self._s[1896]!, self._r[1896]!, [_0]) } public func VoiceOver_Chat_VideoMessageFrom(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1878]!, self._r[1878]!, [_0]) + return formatWithArgumentRanges(self._s[1898]!, self._r[1898]!, [_0]) } - public var Passport_Address_OneOfTypeRentalAgreement: String { return self._s[1879]! } - public var IntentsSettings_MainAccount: String { return self._s[1882]! } - public var Group_MessagePhotoRemoved: String { return self._s[1885]! } - public var Conversation_ContextMenuSelect: String { return self._s[1886]! } - public var GroupInfo_Permissions_Exceptions: String { return self._s[1888]! } - public var GroupRemoved_UsersSectionTitle: String { return self._s[1889]! } - public var Contacts_PermissionsEnable: String { return self._s[1890]! } - public var Channel_EditAdmin_PermissionDeleteMessagesOfOthers: String { return self._s[1891]! } - public var Common_NotNow: String { return self._s[1892]! } - public var Notification_CreatedChannel: String { return self._s[1893]! } - public var Stats_ViewsBySourceTitle: String { return self._s[1895]! } - public var Appearance_AppIconClassic: String { return self._s[1896]! } - public var PhotoEditor_QualityTool: String { return self._s[1897]! } - public var ClearCache_ClearCache: String { return self._s[1898]! } - public var TwoFactorSetup_Password_PlaceholderConfirmPassword: String { return self._s[1899]! } - public var AutoDownloadSettings_Videos: String { return self._s[1900]! } - public var GroupPermission_Duration: String { return self._s[1901]! } - public var ChatList_Read: String { return self._s[1902]! } + public var Passport_Address_OneOfTypeRentalAgreement: String { return self._s[1899]! } + public var InviteLink_Share: String { return self._s[1901]! } + public var IntentsSettings_MainAccount: String { return self._s[1903]! } + public var Group_MessagePhotoRemoved: String { return self._s[1906]! } + public var Conversation_ContextMenuSelect: String { return self._s[1907]! } + public var GroupInfo_Permissions_Exceptions: String { return self._s[1909]! } + public var GroupRemoved_UsersSectionTitle: String { return self._s[1910]! } + public var Contacts_PermissionsEnable: String { return self._s[1911]! } + public var Channel_EditAdmin_PermissionDeleteMessagesOfOthers: String { return self._s[1912]! } + public var Common_NotNow: String { return self._s[1913]! } + public var Notification_CreatedChannel: String { return self._s[1914]! } + public var Stats_ViewsBySourceTitle: String { return self._s[1916]! } + public var InviteLink_ContextShare: String { return self._s[1917]! } + public var Appearance_AppIconClassic: String { return self._s[1918]! } + public var PhotoEditor_QualityTool: String { return self._s[1919]! } + public var ClearCache_ClearCache: String { return self._s[1920]! } + public var TwoFactorSetup_Password_PlaceholderConfirmPassword: String { return self._s[1921]! } + public var AutoDownloadSettings_Videos: String { return self._s[1922]! } + public var GroupPermission_Duration: String { return self._s[1923]! } + public var ChatList_Read: String { return self._s[1924]! } public func Group_OwnershipTransfer_DescriptionInfo(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1903]!, self._r[1903]!, [_1, _2]) + return formatWithArgumentRanges(self._s[1925]!, self._r[1925]!, [_1, _2]) } - public var CallFeedback_Send: String { return self._s[1904]! } - public var Channel_Stickers_Searching: String { return self._s[1905]! } - public var ScheduledMessages_ReminderNotification: String { return self._s[1906]! } - public var FastTwoStepSetup_HintSection: String { return self._s[1907]! } - public var ChatSettings_AutoDownloadVideoMessages: String { return self._s[1908]! } - public var EditTheme_CreateTitle: String { return self._s[1909]! } - public var Application_Name: String { return self._s[1910]! } - public var Paint_Stickers: String { return self._s[1911]! } - public var Appearance_ThemePreview_Chat_1_Text: String { return self._s[1912]! } - public var Call_StatusFailed: String { return self._s[1913]! } - public var Stickers_FavoriteStickers: String { return self._s[1914]! } - public var ClearCache_Clear: String { return self._s[1915]! } - public var Passport_Language_mn: String { return self._s[1916]! } - public var WallpaperPreview_PreviewTopText: String { return self._s[1917]! } - public var LogoutOptions_ClearCacheTitle: String { return self._s[1918]! } - public var Call_VoiceOver_VideoCallOutgoing: String { return self._s[1920]! } - public var TwoFactorSetup_Hint_Text: String { return self._s[1922]! } - public var WallpaperPreview_PatternIntensity: String { return self._s[1923]! } - public var CheckoutInfo_ErrorShippingNotAvailable: String { return self._s[1924]! } - public var Passport_Address_AddBankStatement: String { return self._s[1925]! } + public var CallFeedback_Send: String { return self._s[1926]! } + public var Channel_Stickers_Searching: String { return self._s[1927]! } + public var ScheduledMessages_ReminderNotification: String { return self._s[1928]! } + public var FastTwoStepSetup_HintSection: String { return self._s[1929]! } + public var ChatSettings_AutoDownloadVideoMessages: String { return self._s[1930]! } + public var EditTheme_CreateTitle: String { return self._s[1931]! } + public var Application_Name: String { return self._s[1932]! } + public var Paint_Stickers: String { return self._s[1933]! } + public var Appearance_ThemePreview_Chat_1_Text: String { return self._s[1934]! } + public var Call_StatusFailed: String { return self._s[1935]! } + public var Stickers_FavoriteStickers: String { return self._s[1936]! } + public var ClearCache_Clear: String { return self._s[1937]! } + public var Passport_Language_mn: String { return self._s[1938]! } + public var WallpaperPreview_PreviewTopText: String { return self._s[1939]! } + public var LogoutOptions_ClearCacheTitle: String { return self._s[1940]! } + public var Call_VoiceOver_VideoCallOutgoing: String { return self._s[1942]! } + public var TwoFactorSetup_Hint_Text: String { return self._s[1944]! } + public var WallpaperPreview_PatternIntensity: String { return self._s[1945]! } + public var CheckoutInfo_ErrorShippingNotAvailable: String { return self._s[1946]! } + public var Passport_Address_AddBankStatement: String { return self._s[1947]! } public func Conversation_TitleRepliesFormat(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1927]!, self._r[1927]!, [_1, _2]) + return formatWithArgumentRanges(self._s[1949]!, self._r[1949]!, [_1, _2]) } - public var ChatListFolderSettings_RecommendedNewFolder: String { return self._s[1928]! } - public var UserInfo_ShareContact: String { return self._s[1929]! } - public var Passport_Identity_NamePlaceholder: String { return self._s[1930]! } - public var Call_RateCall: String { return self._s[1932]! } - public var Contacts_AccessDeniedError: String { return self._s[1933]! } - public var Invite_ChannelsTooMuch: String { return self._s[1934]! } - public var CheckoutInfo_ShippingInfoPostcode: String { return self._s[1935]! } - public var Channel_BanUser_PermissionReadMessages: String { return self._s[1936]! } - public var Cache_NoLimit: String { return self._s[1938]! } - public var Conversation_EmptyPlaceholder: String { return self._s[1942]! } - public var Privacy_GroupsAndChannels_AlwaysAllow_Placeholder: String { return self._s[1943]! } - public var GroupRemoved_RemoveInfo: String { return self._s[1945]! } - public var Privacy_Calls_IntegrationHelp: String { return self._s[1946]! } + public var ChatListFolderSettings_RecommendedNewFolder: String { return self._s[1950]! } + public var UserInfo_ShareContact: String { return self._s[1951]! } + public var Passport_Identity_NamePlaceholder: String { return self._s[1952]! } + public var Call_RateCall: String { return self._s[1954]! } + public var Contacts_AccessDeniedError: String { return self._s[1955]! } + public var Invite_ChannelsTooMuch: String { return self._s[1956]! } + public var CheckoutInfo_ShippingInfoPostcode: String { return self._s[1957]! } + public var Channel_BanUser_PermissionReadMessages: String { return self._s[1958]! } + public var InviteLink_Create_TimeLimitInfo: String { return self._s[1959]! } + public var Cache_NoLimit: String { return self._s[1961]! } + public var Conversation_EmptyPlaceholder: String { return self._s[1965]! } + public var Privacy_GroupsAndChannels_AlwaysAllow_Placeholder: String { return self._s[1966]! } + public var GroupRemoved_RemoveInfo: String { return self._s[1968]! } + public var Privacy_Calls_IntegrationHelp: String { return self._s[1969]! } public func PUSH_VIDEO_CALL_MISSED(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1947]!, self._r[1947]!, [_1]) + return formatWithArgumentRanges(self._s[1970]!, self._r[1970]!, [_1]) } - public var VoiceOver_Media_PlaybackRateFast: String { return self._s[1948]! } - public var Theme_ThemeChanged: String { return self._s[1949]! } - public var Privacy_GroupsAndChannels_NeverAllow: String { return self._s[1951]! } - public var AutoDownloadSettings_MediaTypes: String { return self._s[1952]! } + public var VoiceOver_Media_PlaybackRateFast: String { return self._s[1971]! } + public var Theme_ThemeChanged: String { return self._s[1972]! } + public var Privacy_GroupsAndChannels_NeverAllow: String { return self._s[1974]! } + public var AutoDownloadSettings_MediaTypes: String { return self._s[1975]! } public func Notification_PinnedDocumentMessage(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1953]!, self._r[1953]!, [_0]) + return formatWithArgumentRanges(self._s[1976]!, self._r[1976]!, [_0]) } - public var Channel_AdminLog_InfoPanelTitle: String { return self._s[1954]! } - public var Passport_Language_da: String { return self._s[1956]! } - public var Chat_SlowmodeSendError: String { return self._s[1957]! } - public var Application_Update: String { return self._s[1959]! } - public var SocksProxySetup_SaveProxy: String { return self._s[1960]! } + public var Channel_AdminLog_InfoPanelTitle: String { return self._s[1977]! } + public var Passport_Language_da: String { return self._s[1979]! } + public var Chat_SlowmodeSendError: String { return self._s[1980]! } + public var Application_Update: String { return self._s[1982]! } + public var SocksProxySetup_SaveProxy: String { return self._s[1983]! } public func PUSH_AUTH_REGION(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1961]!, self._r[1961]!, [_1, _2]) + return formatWithArgumentRanges(self._s[1984]!, self._r[1984]!, [_1, _2]) } - public var Privacy_AddNewPeer: String { return self._s[1963]! } - public var Channel_DiscussionGroup_MakeHistoryPublicProceed: String { return self._s[1965]! } - public var Channel_Members_Title: String { return self._s[1966]! } - public var Settings_LogoutConfirmationText: String { return self._s[1967]! } - public var Chat_UnsendMyMessages: String { return self._s[1968]! } - public var Conversation_EditingMessageMediaEditCurrentVideo: String { return self._s[1970]! } - public var ChatListFilter_AddChatsTitle: String { return self._s[1971]! } - public var Passport_FloodError: String { return self._s[1972]! } - public var NotificationSettings_ContactJoinedInfo: String { return self._s[1973]! } - public var SettingsSearch_Synonyms_Privacy_Data_SecretChatLinkPreview: String { return self._s[1974]! } - public var CallSettings_TabIconDescription: String { return self._s[1975]! } - public var Group_Setup_HistoryHeader: String { return self._s[1977]! } + public var Privacy_AddNewPeer: String { return self._s[1986]! } + public var Channel_DiscussionGroup_MakeHistoryPublicProceed: String { return self._s[1988]! } + public var Channel_Members_Title: String { return self._s[1989]! } + public var Settings_LogoutConfirmationText: String { return self._s[1990]! } + public var Chat_UnsendMyMessages: String { return self._s[1991]! } + public var Conversation_EditingMessageMediaEditCurrentVideo: String { return self._s[1993]! } + public var ChatListFilter_AddChatsTitle: String { return self._s[1994]! } + public var Passport_FloodError: String { return self._s[1995]! } + public var NotificationSettings_ContactJoinedInfo: String { return self._s[1996]! } + public var SettingsSearch_Synonyms_Privacy_Data_SecretChatLinkPreview: String { return self._s[1997]! } + public var CallSettings_TabIconDescription: String { return self._s[1998]! } + public var Group_Setup_HistoryHeader: String { return self._s[2000]! } public func Channel_AdminLog_AllowedNewMembersToSpeak(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1978]!, self._r[1978]!, [_1]) + return formatWithArgumentRanges(self._s[2001]!, self._r[2001]!, [_1]) } - public var TwoStepAuth_EmailTitle: String { return self._s[1979]! } - public var GroupInfo_Permissions_Removed: String { return self._s[1980]! } - public var DialogList_ClearHistoryConfirmation: String { return self._s[1981]! } - public var Contacts_Title: String { return self._s[1983]! } + public var TwoStepAuth_EmailTitle: String { return self._s[2002]! } + public var GroupInfo_Permissions_Removed: String { return self._s[2003]! } + public var DialogList_ClearHistoryConfirmation: String { return self._s[2004]! } + public var Contacts_Title: String { return self._s[2006]! } public func Notification_Invited(_ _0: String, _ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1984]!, self._r[1984]!, [_0, _1]) + return formatWithArgumentRanges(self._s[2007]!, self._r[2007]!, [_0, _1]) } - public var ChatList_PeerTypeBot: String { return self._s[1987]! } + public var ChatList_PeerTypeBot: String { return self._s[2010]! } public func Channel_AdminLog_SetSlowmode(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1988]!, self._r[1988]!, [_1, _2]) + return formatWithArgumentRanges(self._s[2011]!, self._r[2011]!, [_1, _2]) } - public var Appearance_ThemePreview_Chat_6_Text: String { return self._s[1989]! } + public var Appearance_ThemePreview_Chat_6_Text: String { return self._s[2012]! } public func Time_PreciseDate_m1(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1990]!, self._r[1990]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[2013]!, self._r[2013]!, [_1, _2, _3]) } - public var Camera_PhotoMode: String { return self._s[1992]! } + public var Camera_PhotoMode: String { return self._s[2015]! } public func PUSH_MESSAGE_GAME_SCORE(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1993]!, self._r[1993]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[2016]!, self._r[2016]!, [_1, _2, _3]) } - public var ContactInfo_PhoneLabelPager: String { return self._s[1994]! } - public var SettingsSearch_Synonyms_FAQ: String { return self._s[1995]! } - public var Call_CallAgain: String { return self._s[1996]! } - public var TwoStepAuth_PasswordSet: String { return self._s[1997]! } + public var ContactInfo_PhoneLabelPager: String { return self._s[2017]! } + public var SettingsSearch_Synonyms_FAQ: String { return self._s[2018]! } + public var Call_CallAgain: String { return self._s[2019]! } + public var TwoStepAuth_PasswordSet: String { return self._s[2020]! } public func Channel_Management_RestrictedBy(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[1998]!, self._r[1998]!, [_0]) + return formatWithArgumentRanges(self._s[2021]!, self._r[2021]!, [_0]) } - public var GroupInfo_InviteLink_RevokeAlert_Success: String { return self._s[1999]! } - public var ClearCache_FreeSpaceDescription: String { return self._s[2000]! } - public var Permissions_ContactsAllowInSettings_v0: String { return self._s[2001]! } - public var Group_LeaveGroup: String { return self._s[2002]! } - public var GroupInfo_LabelAdmin: String { return self._s[2005]! } - public var CheckoutInfo_ErrorStateInvalid: String { return self._s[2007]! } - public var Notification_PassportValuePersonalDetails: String { return self._s[2008]! } + public var GroupInfo_InviteLink_RevokeAlert_Success: String { return self._s[2022]! } + public var ClearCache_FreeSpaceDescription: String { return self._s[2023]! } + public var Permissions_ContactsAllowInSettings_v0: String { return self._s[2024]! } + public var Group_LeaveGroup: String { return self._s[2025]! } + public var GroupInfo_LabelAdmin: String { return self._s[2028]! } + public var CheckoutInfo_ErrorStateInvalid: String { return self._s[2030]! } + public var Notification_PassportValuePersonalDetails: String { return self._s[2031]! } public func WebSearch_SearchNoResultsDescription(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2009]!, self._r[2009]!, [_0]) + return formatWithArgumentRanges(self._s[2032]!, self._r[2032]!, [_0]) } - public var Stats_GroupNewMembersBySourceTitle: String { return self._s[2010]! } - public var Appearance_Preview: String { return self._s[2011]! } - public var VoiceOver_Chat_Contact: String { return self._s[2012]! } - public var Passport_Language_th: String { return self._s[2013]! } - public var PhotoEditor_CropAspectRatioOriginal: String { return self._s[2015]! } - public var LastSeen_Offline: String { return self._s[2018]! } - public var Map_OpenInHereMaps: String { return self._s[2019]! } - public var SettingsSearch_Synonyms_Data_AutoplayVideos: String { return self._s[2020]! } - public var AutoDownloadSettings_Reset: String { return self._s[2022]! } - public var Conversation_SendMessage_SetReminder: String { return self._s[2023]! } - public var Channel_AdminLog_EmptyMessageText: String { return self._s[2024]! } + public var Stats_GroupNewMembersBySourceTitle: String { return self._s[2033]! } + public var Appearance_Preview: String { return self._s[2034]! } + public var VoiceOver_Chat_Contact: String { return self._s[2035]! } + public var Passport_Language_th: String { return self._s[2036]! } + public var PhotoEditor_CropAspectRatioOriginal: String { return self._s[2038]! } + public var LastSeen_Offline: String { return self._s[2041]! } + public var Map_OpenInHereMaps: String { return self._s[2042]! } + public var SettingsSearch_Synonyms_Data_AutoplayVideos: String { return self._s[2043]! } + public var InviteLink_ContextEdit: String { return self._s[2045]! } + public var AutoDownloadSettings_Reset: String { return self._s[2046]! } + public var Conversation_SendMessage_SetReminder: String { return self._s[2047]! } + public var Channel_AdminLog_EmptyMessageText: String { return self._s[2048]! } public func AddContact_StatusSuccess(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2025]!, self._r[2025]!, [_0]) + return formatWithArgumentRanges(self._s[2049]!, self._r[2049]!, [_0]) } public func AuthCode_Alert(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2026]!, self._r[2026]!, [_0]) + return formatWithArgumentRanges(self._s[2050]!, self._r[2050]!, [_0]) } - public var Passport_Identity_EditDriversLicense: String { return self._s[2027]! } - public var ChatListFolder_NameNonMuted: String { return self._s[2028]! } - public var Username_Placeholder: String { return self._s[2029]! } + public var Passport_Identity_EditDriversLicense: String { return self._s[2051]! } + public var ChatListFolder_NameNonMuted: String { return self._s[2052]! } + public var Username_Placeholder: String { return self._s[2053]! } public func PUSH_ALBUM(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2030]!, self._r[2030]!, [_1]) + return formatWithArgumentRanges(self._s[2054]!, self._r[2054]!, [_1]) } - public var Passport_Language_it: String { return self._s[2031]! } - public var Checkout_NewCard_SaveInfo: String { return self._s[2032]! } + public var Passport_Language_it: String { return self._s[2055]! } + public var Checkout_NewCard_SaveInfo: String { return self._s[2056]! } public func Channel_OwnershipTransfer_DescriptionInfo(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2033]!, self._r[2033]!, [_1, _2]) + return formatWithArgumentRanges(self._s[2057]!, self._r[2057]!, [_1, _2]) } - public var NotificationsSound_Pulse: String { return self._s[2034]! } - public var VoiceOver_DismissContextMenu: String { return self._s[2036]! } - public var MessagePoll_NoVotes: String { return self._s[2039]! } - public var Message_Wallpaper: String { return self._s[2040]! } - public var Appearance_Other: String { return self._s[2041]! } - public var Passport_Identity_NativeNameHelp: String { return self._s[2043]! } - public var Group_PublicLink_Placeholder: String { return self._s[2046]! } - public var Appearance_ThemePreview_ChatList_2_Text: String { return self._s[2047]! } - public var VoiceOver_Recording_StopAndPreview: String { return self._s[2048]! } - public var ChatListFolder_NameBots: String { return self._s[2049]! } - public var Conversation_StopPollConfirmation: String { return self._s[2050]! } - public var UserInfo_DeleteContact: String { return self._s[2051]! } + public var NotificationsSound_Pulse: String { return self._s[2058]! } + public var VoiceOver_DismissContextMenu: String { return self._s[2060]! } + public var MessagePoll_NoVotes: String { return self._s[2063]! } + public var Message_Wallpaper: String { return self._s[2064]! } + public var Appearance_Other: String { return self._s[2065]! } + public var Passport_Identity_NativeNameHelp: String { return self._s[2067]! } + public var Group_PublicLink_Placeholder: String { return self._s[2070]! } + public var Appearance_ThemePreview_ChatList_2_Text: String { return self._s[2071]! } + public var VoiceOver_Recording_StopAndPreview: String { return self._s[2072]! } + public var ChatListFolder_NameBots: String { return self._s[2073]! } + public var Conversation_StopPollConfirmation: String { return self._s[2074]! } + public var UserInfo_DeleteContact: String { return self._s[2075]! } public func Time_MonthOfYear_m11(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2052]!, self._r[2052]!, [_0]) + return formatWithArgumentRanges(self._s[2076]!, self._r[2076]!, [_0]) } - public var Wallpaper_Wallpaper: String { return self._s[2054]! } + public var Wallpaper_Wallpaper: String { return self._s[2078]! } public func PUSH_MESSAGE_NOTEXT(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2055]!, self._r[2055]!, [_1]) + return formatWithArgumentRanges(self._s[2079]!, self._r[2079]!, [_1]) } - public var LoginPassword_ForgotPassword: String { return self._s[2056]! } - public var FeaturedStickerPacks_Title: String { return self._s[2057]! } - public var Paint_Pen: String { return self._s[2058]! } - public var Channel_AdminLogFilter_EventsInfo: String { return self._s[2059]! } - public var ChatListFolderSettings_Info: String { return self._s[2060]! } - public var FastTwoStepSetup_HintPlaceholder: String { return self._s[2061]! } - public var PhotoEditor_CurvesAll: String { return self._s[2063]! } + public var LoginPassword_ForgotPassword: String { return self._s[2080]! } + public var FeaturedStickerPacks_Title: String { return self._s[2081]! } + public var Paint_Pen: String { return self._s[2082]! } + public var Channel_AdminLogFilter_EventsInfo: String { return self._s[2083]! } + public var ChatListFolderSettings_Info: String { return self._s[2084]! } + public var FastTwoStepSetup_HintPlaceholder: String { return self._s[2085]! } + public var PhotoEditor_CurvesAll: String { return self._s[2087]! } public func Time_PreciseDate_m12(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2065]!, self._r[2065]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[2089]!, self._r[2089]!, [_1, _2, _3]) } - public var Passport_Address_TypeRentalAgreement: String { return self._s[2067]! } - public var Message_ImageExpired: String { return self._s[2068]! } - public var Call_ConnectionErrorMessage: String { return self._s[2069]! } - public var SearchImages_NoImagesFound: String { return self._s[2071]! } - public var PeerInfo_PaneGifs: String { return self._s[2072]! } - public var Passport_DeletePersonalDetailsConfirmation: String { return self._s[2073]! } - public var EnterPasscode_RepeatNewPasscode: String { return self._s[2074]! } - public var PhotoEditor_VignetteTool: String { return self._s[2075]! } - public var Passport_Language_dz: String { return self._s[2076]! } - public var Notifications_ChannelNotificationsHelp: String { return self._s[2077]! } - public var Conversation_BlockUser: String { return self._s[2078]! } - public var GroupPermission_PermissionDisabledByDefault: String { return self._s[2081]! } - public var Group_OwnershipTransfer_ErrorAdminsTooMuch: String { return self._s[2083]! } + public var Passport_Address_TypeRentalAgreement: String { return self._s[2091]! } + public var Message_ImageExpired: String { return self._s[2092]! } + public var Call_ConnectionErrorMessage: String { return self._s[2093]! } + public var SearchImages_NoImagesFound: String { return self._s[2095]! } + public var PeerInfo_PaneGifs: String { return self._s[2096]! } + public var Passport_DeletePersonalDetailsConfirmation: String { return self._s[2097]! } + public var EnterPasscode_RepeatNewPasscode: String { return self._s[2098]! } + public var PhotoEditor_VignetteTool: String { return self._s[2099]! } + public var Passport_Language_dz: String { return self._s[2100]! } + public var Notifications_ChannelNotificationsHelp: String { return self._s[2101]! } + public var Conversation_BlockUser: String { return self._s[2102]! } + public var GroupPermission_PermissionDisabledByDefault: String { return self._s[2105]! } + public var Group_OwnershipTransfer_ErrorAdminsTooMuch: String { return self._s[2107]! } public func Time_MonthOfYear_m8(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2084]!, self._r[2084]!, [_0]) + return formatWithArgumentRanges(self._s[2108]!, self._r[2108]!, [_0]) } - public var KeyCommand_NewMessage: String { return self._s[2085]! } - public var EditTheme_Edit_Preview_IncomingReplyText: String { return self._s[2087]! } + public var KeyCommand_NewMessage: String { return self._s[2109]! } + public var EditTheme_Edit_Preview_IncomingReplyText: String { return self._s[2111]! } public func PUSH_CHAT_MESSAGE_GEO(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2089]!, self._r[2089]!, [_1, _2]) + return formatWithArgumentRanges(self._s[2113]!, self._r[2113]!, [_1, _2]) } - public var ContactList_Context_StartSecretChat: String { return self._s[2090]! } - public var VoiceOver_Chat_File: String { return self._s[2091]! } - public var ChatList_EditFolder: String { return self._s[2093]! } - public var Appearance_BubbleCorners_Title: String { return self._s[2094]! } - public var PeerInfo_PaneAudio: String { return self._s[2095]! } - public var ChatListFolder_CategoryContacts: String { return self._s[2097]! } + public var ContactList_Context_StartSecretChat: String { return self._s[2114]! } + public var VoiceOver_Chat_File: String { return self._s[2115]! } + public var ChatList_EditFolder: String { return self._s[2117]! } + public var Appearance_BubbleCorners_Title: String { return self._s[2118]! } + public var PeerInfo_PaneAudio: String { return self._s[2119]! } + public var ChatListFolder_CategoryContacts: String { return self._s[2121]! } public func Login_InvalidPhoneEmailBody(_ _1: String, _ _2: String, _ _3: String, _ _4: String, _ _5: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2098]!, self._r[2098]!, [_1, _2, _3, _4, _5]) + return formatWithArgumentRanges(self._s[2122]!, self._r[2122]!, [_1, _2, _3, _4, _5]) } - public var ChatList_PeerTypeChannel: String { return self._s[2099]! } - public var VoiceOver_Navigation_Search: String { return self._s[2100]! } - public var Settings_Search: String { return self._s[2101]! } - public var WallpaperSearch_ColorYellow: String { return self._s[2102]! } - public var Login_PhoneBannedError: String { return self._s[2103]! } - public var KeyCommand_JumpToNextChat: String { return self._s[2104]! } - public var Passport_Language_fa: String { return self._s[2105]! } - public var Settings_About: String { return self._s[2106]! } - public var AutoDownloadSettings_MaxFileSize: String { return self._s[2107]! } - public var Channel_AdminLog_InfoPanelChannelAlertText: String { return self._s[2108]! } - public var AutoDownloadSettings_DataUsageHigh: String { return self._s[2109]! } + public var ChatList_PeerTypeChannel: String { return self._s[2123]! } + public var VoiceOver_Navigation_Search: String { return self._s[2124]! } + public var Settings_Search: String { return self._s[2125]! } + public var WallpaperSearch_ColorYellow: String { return self._s[2126]! } + public var Login_PhoneBannedError: String { return self._s[2127]! } + public var KeyCommand_JumpToNextChat: String { return self._s[2128]! } + public var Passport_Language_fa: String { return self._s[2129]! } + public var Settings_About: String { return self._s[2130]! } + public var AutoDownloadSettings_MaxFileSize: String { return self._s[2131]! } + public var Channel_AdminLog_InfoPanelChannelAlertText: String { return self._s[2132]! } + public var AutoDownloadSettings_DataUsageHigh: String { return self._s[2133]! } public func PUSH_CHAT_MESSAGE_TEXT(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2110]!, self._r[2110]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[2134]!, self._r[2134]!, [_1, _2, _3]) } - public var Common_OK: String { return self._s[2111]! } - public var Contacts_SortBy: String { return self._s[2112]! } - public var AutoNightTheme_PreferredTheme: String { return self._s[2113]! } + public var Common_OK: String { return self._s[2135]! } + public var Contacts_SortBy: String { return self._s[2136]! } + public var AutoNightTheme_PreferredTheme: String { return self._s[2137]! } public func AutoDownloadSettings_OnFor(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2115]!, self._r[2115]!, [_0]) + return formatWithArgumentRanges(self._s[2139]!, self._r[2139]!, [_0]) } - public var CallFeedback_IncludeLogs: String { return self._s[2118]! } + public var CallFeedback_IncludeLogs: String { return self._s[2142]! } public func External_OpenIn(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2119]!, self._r[2119]!, [_0]) + return formatWithArgumentRanges(self._s[2143]!, self._r[2143]!, [_0]) } - public var Passcode_AppLockedAlert: String { return self._s[2121]! } - public var TwoStepAuth_SetupPasswordTitle: String { return self._s[2122]! } - public var Channel_NotificationLoading: String { return self._s[2124]! } - public var Passport_Identity_DocumentNumber: String { return self._s[2125]! } - public var VoiceOver_Chat_PagePreview: String { return self._s[2126]! } - public var VoiceOver_Chat_OpenHint: String { return self._s[2127]! } - public var Weekday_ShortFriday: String { return self._s[2128]! } - public var Conversation_TitleMute: String { return self._s[2129]! } - public var SettingsSearch_Synonyms_Notifications_GroupNotificationsSound: String { return self._s[2130]! } - public var ScheduledMessages_PollUnavailable: String { return self._s[2131]! } - public var DialogList_LanguageTooltip: String { return self._s[2133]! } - public var Channel_AdminLogFilter_EventsPinned: String { return self._s[2134]! } + public var Passcode_AppLockedAlert: String { return self._s[2145]! } + public var TwoStepAuth_SetupPasswordTitle: String { return self._s[2146]! } + public var Channel_NotificationLoading: String { return self._s[2148]! } + public var Passport_Identity_DocumentNumber: String { return self._s[2149]! } + public var VoiceOver_Chat_PagePreview: String { return self._s[2150]! } + public var VoiceOver_Chat_OpenHint: String { return self._s[2151]! } + public var Weekday_ShortFriday: String { return self._s[2152]! } + public var Conversation_TitleMute: String { return self._s[2153]! } + public var SettingsSearch_Synonyms_Notifications_GroupNotificationsSound: String { return self._s[2154]! } + public var ScheduledMessages_PollUnavailable: String { return self._s[2155]! } + public var DialogList_LanguageTooltip: String { return self._s[2157]! } + public var Channel_AdminLogFilter_EventsPinned: String { return self._s[2158]! } public func DialogList_SingleUploadingVideoSuffix(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2135]!, self._r[2135]!, [_0]) + return formatWithArgumentRanges(self._s[2159]!, self._r[2159]!, [_0]) } - public var TwoStepAuth_SetupResendEmailCodeAlert: String { return self._s[2137]! } - public var Privacy_Calls_AlwaysAllow_Title: String { return self._s[2138]! } - public var Settings_EditVideo: String { return self._s[2139]! } - public var VoiceOver_Common_Off: String { return self._s[2140]! } - public var Stickers_FrequentlyUsed: String { return self._s[2141]! } - public var GroupPermission_Title: String { return self._s[2142]! } - public var AccessDenied_VideoMessageCamera: String { return self._s[2143]! } - public var Appearance_ThemeCarouselDay: String { return self._s[2144]! } + public var TwoStepAuth_SetupResendEmailCodeAlert: String { return self._s[2161]! } + public var Privacy_Calls_AlwaysAllow_Title: String { return self._s[2162]! } + public var Settings_EditVideo: String { return self._s[2163]! } + public var VoiceOver_Common_Off: String { return self._s[2164]! } + public var Stickers_FrequentlyUsed: String { return self._s[2165]! } + public var GroupPermission_Title: String { return self._s[2166]! } + public var AccessDenied_VideoMessageCamera: String { return self._s[2167]! } + public var Appearance_ThemeCarouselDay: String { return self._s[2168]! } public func PUSH_CHAT_MESSAGE_AUDIO(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2145]!, self._r[2145]!, [_1, _2]) + return formatWithArgumentRanges(self._s[2169]!, self._r[2169]!, [_1, _2]) } - public var Passport_Identity_DocumentNumberPlaceholder: String { return self._s[2146]! } - public var Tour_Title6: String { return self._s[2147]! } - public var EmptyGroupInfo_Title: String { return self._s[2148]! } + public var Passport_Identity_DocumentNumberPlaceholder: String { return self._s[2170]! } + public var Tour_Title6: String { return self._s[2171]! } + public var EmptyGroupInfo_Title: String { return self._s[2172]! } public func Channel_AdminLog_MessageToggleSignaturesOn(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2149]!, self._r[2149]!, [_0]) + return formatWithArgumentRanges(self._s[2173]!, self._r[2173]!, [_0]) } - public var Passport_Language_sk: String { return self._s[2150]! } - public var VoiceOver_Chat_YourAnonymousPoll: String { return self._s[2151]! } - public var Preview_SaveToCameraRoll: String { return self._s[2152]! } - public var LogoutOptions_SetPasscodeTitle: String { return self._s[2153]! } - public var Passport_Address_TypeUtilityBillUploadScan: String { return self._s[2154]! } - public var Conversation_ContextMenuMore: String { return self._s[2155]! } - public var Conversation_ForwardAuthorHiddenTooltip: String { return self._s[2156]! } - public var Channel_AdminLog_CanBeAnonymous: String { return self._s[2157]! } - public var CallFeedback_ReasonSilentLocal: String { return self._s[2159]! } + public var Passport_Language_sk: String { return self._s[2174]! } + public var VoiceOver_Chat_YourAnonymousPoll: String { return self._s[2175]! } + public var Preview_SaveToCameraRoll: String { return self._s[2176]! } + public var LogoutOptions_SetPasscodeTitle: String { return self._s[2177]! } + public var Passport_Address_TypeUtilityBillUploadScan: String { return self._s[2178]! } + public var Conversation_ContextMenuMore: String { return self._s[2179]! } + public var Conversation_ForwardAuthorHiddenTooltip: String { return self._s[2180]! } + public var Channel_AdminLog_CanBeAnonymous: String { return self._s[2181]! } + public var CallFeedback_ReasonSilentLocal: String { return self._s[2183]! } public func Channel_AdminLog_UnmutedMutedParticipant(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2160]!, self._r[2160]!, [_1, _2]) + return formatWithArgumentRanges(self._s[2184]!, self._r[2184]!, [_1, _2]) } - public var UserInfo_NotificationsDisable: String { return self._s[2161]! } + public var UserInfo_NotificationsDisable: String { return self._s[2185]! } public func Channel_AdminLog_EmptyFilterQueryText(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2163]!, self._r[2163]!, [_0]) + return formatWithArgumentRanges(self._s[2187]!, self._r[2187]!, [_0]) } - public var SettingsSearch_Synonyms_EditProfile_Bio: String { return self._s[2164]! } + public var SettingsSearch_Synonyms_EditProfile_Bio: String { return self._s[2188]! } public func Date_ChatDateHeader(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2166]!, self._r[2166]!, [_1, _2]) - } - public var WallpaperSearch_ColorPrefix: String { return self._s[2167]! } - public func Message_ForwardedPsa_covid(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2168]!, self._r[2168]!, [_0]) - } - public var Conversation_RestrictedMedia: String { return self._s[2170]! } - public var Group_MessageVideoUpdated: String { return self._s[2171]! } - public var NetworkUsageSettings_ResetStatsConfirmation: String { return self._s[2172]! } - public var GroupInfo_DeleteAndExit: String { return self._s[2173]! } - public var TwoFactorSetup_Email_Action: String { return self._s[2174]! } - public var Media_ShareThisVideo: String { return self._s[2176]! } - public var DialogList_Replies: String { return self._s[2177]! } - public func Conversation_Moderate_DeleteAllMessages(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2178]!, self._r[2178]!, [_0]) - } - public var CheckoutInfo_ShippingInfoAddress1: String { return self._s[2179]! } - public var Watch_Suggestion_OnMyWay: String { return self._s[2180]! } - public var CheckoutInfo_ShippingInfoAddress2: String { return self._s[2181]! } - public func PUSH_PINNED_POLL(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2182]!, self._r[2182]!, [_1, _2]) - } - public func GroupInfo_InvitationLinkAcceptChannel(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2183]!, self._r[2183]!, [_0]) - } - public var Channel_EditAdmin_PermissinAddAdminOff: String { return self._s[2184]! } - public var ChatAdmins_AllMembersAreAdminsOnHelp: String { return self._s[2185]! } - public var ChatList_Search_NoResultsFitlerMedia: String { return self._s[2186]! } - public var Channel_Members_InviteLink: String { return self._s[2187]! } - public var Conversation_TapAndHoldToRecord: String { return self._s[2188]! } - public var WatchRemote_AlertText: String { return self._s[2189]! } - public func Channel_DiscussionGroup_PrivateChannelLink(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { return formatWithArgumentRanges(self._s[2190]!, self._r[2190]!, [_1, _2]) } - public var Conversation_Pin: String { return self._s[2191]! } - public var InfoPlist_NSMicrophoneUsageDescription: String { return self._s[2192]! } - public var Stickers_RemoveFromFavorites: String { return self._s[2193]! } + public var WallpaperSearch_ColorPrefix: String { return self._s[2191]! } + public func Message_ForwardedPsa_covid(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[2192]!, self._r[2192]!, [_0]) + } + public var Conversation_RestrictedMedia: String { return self._s[2194]! } + public var Group_MessageVideoUpdated: String { return self._s[2195]! } + public var NetworkUsageSettings_ResetStatsConfirmation: String { return self._s[2196]! } + public var GroupInfo_DeleteAndExit: String { return self._s[2197]! } + public var TwoFactorSetup_Email_Action: String { return self._s[2198]! } + public var Media_ShareThisVideo: String { return self._s[2200]! } + public var DialogList_Replies: String { return self._s[2201]! } + public func Conversation_Moderate_DeleteAllMessages(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[2202]!, self._r[2202]!, [_0]) + } + public var CheckoutInfo_ShippingInfoAddress1: String { return self._s[2203]! } + public var Watch_Suggestion_OnMyWay: String { return self._s[2204]! } + public var CheckoutInfo_ShippingInfoAddress2: String { return self._s[2205]! } + public func PUSH_PINNED_POLL(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[2206]!, self._r[2206]!, [_1, _2]) + } + public func GroupInfo_InvitationLinkAcceptChannel(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[2207]!, self._r[2207]!, [_0]) + } + public var Channel_EditAdmin_PermissinAddAdminOff: String { return self._s[2208]! } + public var ChatAdmins_AllMembersAreAdminsOnHelp: String { return self._s[2209]! } + public var ChatList_Search_NoResultsFitlerMedia: String { return self._s[2210]! } + public var Channel_Members_InviteLink: String { return self._s[2211]! } + public var Conversation_TapAndHoldToRecord: String { return self._s[2212]! } + public var WatchRemote_AlertText: String { return self._s[2213]! } + public func Channel_DiscussionGroup_PrivateChannelLink(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[2214]!, self._r[2214]!, [_1, _2]) + } + public var Conversation_Pin: String { return self._s[2215]! } + public var InfoPlist_NSMicrophoneUsageDescription: String { return self._s[2216]! } + public var Stickers_RemoveFromFavorites: String { return self._s[2217]! } public func Notification_PinnedPollMessage(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2194]!, self._r[2194]!, [_0]) + return formatWithArgumentRanges(self._s[2218]!, self._r[2218]!, [_0]) } - public var Appearance_AppIconFilled: String { return self._s[2195]! } - public var StickerPack_ErrorNotFound: String { return self._s[2196]! } + public var Appearance_AppIconFilled: String { return self._s[2219]! } + public var StickerPack_ErrorNotFound: String { return self._s[2220]! } public func Channel_AdminLog_MessageRestrictedName(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2197]!, self._r[2197]!, [_1]) + return formatWithArgumentRanges(self._s[2221]!, self._r[2221]!, [_1]) } - public var Passport_Identity_AddIdentityCard: String { return self._s[2198]! } + public var Passport_Identity_AddIdentityCard: String { return self._s[2222]! } public func PUSH_CHANNEL_MESSAGE_DOC(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2200]!, self._r[2200]!, [_1]) + return formatWithArgumentRanges(self._s[2224]!, self._r[2224]!, [_1]) } - public var Call_Camera: String { return self._s[2201]! } - public var GroupInfo_InviteLink_RevokeAlert_Text: String { return self._s[2202]! } - public var Group_Location_Info: String { return self._s[2203]! } - public var Watch_LastSeen_WithinAMonth: String { return self._s[2204]! } - public var UserInfo_NotificationsDefaultEnabled: String { return self._s[2205]! } + public var Call_Camera: String { return self._s[2225]! } + public var GroupInfo_InviteLink_RevokeAlert_Text: String { return self._s[2226]! } + public var Group_Location_Info: String { return self._s[2227]! } + public var Watch_LastSeen_WithinAMonth: String { return self._s[2228]! } + public var UserInfo_NotificationsDefaultEnabled: String { return self._s[2229]! } public func DialogList_PinLimitError(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2206]!, self._r[2206]!, [_0]) + return formatWithArgumentRanges(self._s[2230]!, self._r[2230]!, [_0]) } - public var Weekday_Yesterday: String { return self._s[2207]! } - public var TwoStepAuth_SetupPasswordEnterPasswordNew: String { return self._s[2208]! } - public var ArchivedPacksAlert_Title: String { return self._s[2209]! } - public var PeerInfo_PaneMembers: String { return self._s[2210]! } - public var PhotoEditor_SelectCoverFrame: String { return self._s[2211]! } + public var Weekday_Yesterday: String { return self._s[2231]! } + public var TwoStepAuth_SetupPasswordEnterPasswordNew: String { return self._s[2232]! } + public var InviteLink_Create_UsersLimit: String { return self._s[2233]! } + public var ArchivedPacksAlert_Title: String { return self._s[2234]! } + public var PeerInfo_PaneMembers: String { return self._s[2235]! } + public var PhotoEditor_SelectCoverFrame: String { return self._s[2236]! } public func Location_ProximityAlertSetTextGroup(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2212]!, self._r[2212]!, [_0]) + return formatWithArgumentRanges(self._s[2237]!, self._r[2237]!, [_0]) } - public var ContactInfo_PhoneLabelMain: String { return self._s[2213]! } + public var ContactInfo_PhoneLabelMain: String { return self._s[2238]! } public func Time_PreciseDate_m7(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2214]!, self._r[2214]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[2239]!, self._r[2239]!, [_1, _2, _3]) } - public var TwoFactorSetup_EmailVerification_ChangeAction: String { return self._s[2215]! } - public var Channel_DiscussionGroup: String { return self._s[2216]! } - public var EditTheme_Edit_Preview_IncomingReplyName: String { return self._s[2217]! } - public var Channel_EditAdmin_PermissionsHeader: String { return self._s[2219]! } - public var VoiceOver_MessageContextForward: String { return self._s[2220]! } - public var SocksProxySetup_TypeNone: String { return self._s[2221]! } - public var CreatePoll_MultipleChoiceQuizAlert: String { return self._s[2223]! } - public var ProfilePhoto_OpenInEditor: String { return self._s[2225]! } - public var WallpaperSearch_ColorPurple: String { return self._s[2226]! } - public var ChatListFolder_IncludeChatsTitle: String { return self._s[2227]! } - public var Group_Username_InvalidTooShort: String { return self._s[2228]! } - public var Location_ProximityNotification_DistanceM: String { return self._s[2229]! } + public var TwoFactorSetup_EmailVerification_ChangeAction: String { return self._s[2240]! } + public var Channel_DiscussionGroup: String { return self._s[2241]! } + public var EditTheme_Edit_Preview_IncomingReplyName: String { return self._s[2242]! } + public var InviteLink_Create_TimeLimit: String { return self._s[2244]! } + public var Channel_EditAdmin_PermissionsHeader: String { return self._s[2245]! } + public var VoiceOver_MessageContextForward: String { return self._s[2246]! } + public var SocksProxySetup_TypeNone: String { return self._s[2247]! } + public var CreatePoll_MultipleChoiceQuizAlert: String { return self._s[2249]! } + public var ProfilePhoto_OpenInEditor: String { return self._s[2251]! } + public var WallpaperSearch_ColorPurple: String { return self._s[2252]! } + public var ChatListFolder_IncludeChatsTitle: String { return self._s[2253]! } + public var Group_Username_InvalidTooShort: String { return self._s[2254]! } + public var Location_ProximityNotification_DistanceM: String { return self._s[2255]! } public func Login_EmailPhoneBody(_ _0: String, _ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2230]!, self._r[2230]!, [_0, _1, _2]) + return formatWithArgumentRanges(self._s[2256]!, self._r[2256]!, [_0, _1, _2]) } - public var Passport_Language_tk: String { return self._s[2231]! } - public var ConvertToSupergroup_Title: String { return self._s[2232]! } - public var Channel_BanUser_PermissionEmbedLinks: String { return self._s[2233]! } - public var Cache_KeepMediaHelp: String { return self._s[2234]! } - public var Channel_Management_Title: String { return self._s[2235]! } + public var Passport_Language_tk: String { return self._s[2257]! } + public var ConvertToSupergroup_Title: String { return self._s[2258]! } + public var Channel_BanUser_PermissionEmbedLinks: String { return self._s[2259]! } + public var Cache_KeepMediaHelp: String { return self._s[2260]! } + public var Channel_Management_Title: String { return self._s[2261]! } public func PUSH_MESSAGE_PHOTO_SECRET(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2236]!, self._r[2236]!, [_1]) + return formatWithArgumentRanges(self._s[2262]!, self._r[2262]!, [_1]) } - public var Conversation_ForwardChats: String { return self._s[2237]! } - public var Passport_Language_bg: String { return self._s[2238]! } - public var SocksProxySetup_TypeSocks: String { return self._s[2239]! } - public var Permissions_PrivacyPolicy: String { return self._s[2240]! } - public var VoiceOver_Chat_YourMusic: String { return self._s[2241]! } - public var SettingsSearch_Synonyms_Notifications_ResetAllNotifications: String { return self._s[2242]! } - public var Conversation_EmptyGifPanelPlaceholder: String { return self._s[2243]! } - public var Conversation_ContextMenuOpenChannel: String { return self._s[2244]! } - public var Activity_UploadingVideo: String { return self._s[2245]! } - public var PrivacyPolicy_AgeVerificationAgree: String { return self._s[2247]! } - public var SocksProxySetup_Credentials: String { return self._s[2249]! } - public var Preview_SaveGif: String { return self._s[2250]! } - public var Cache_Photos: String { return self._s[2251]! } - public var Channel_AdminLogFilter_EventsCalls: String { return self._s[2252]! } - public var Conversation_ContextMenuCancelEditing: String { return self._s[2253]! } - public var Contacts_FailedToSendInvitesMessage: String { return self._s[2254]! } - public var Passport_Language_lt: String { return self._s[2255]! } - public var Passport_DeleteDocument: String { return self._s[2257]! } - public var GroupInfo_SetGroupPhotoStop: String { return self._s[2258]! } + public var Conversation_ForwardChats: String { return self._s[2263]! } + public var Passport_Language_bg: String { return self._s[2264]! } + public var SocksProxySetup_TypeSocks: String { return self._s[2265]! } + public var Permissions_PrivacyPolicy: String { return self._s[2266]! } + public var VoiceOver_Chat_YourMusic: String { return self._s[2267]! } + public var SettingsSearch_Synonyms_Notifications_ResetAllNotifications: String { return self._s[2268]! } + public var Conversation_EmptyGifPanelPlaceholder: String { return self._s[2269]! } + public var Conversation_ContextMenuOpenChannel: String { return self._s[2270]! } + public var Activity_UploadingVideo: String { return self._s[2271]! } + public var PrivacyPolicy_AgeVerificationAgree: String { return self._s[2273]! } + public var SocksProxySetup_Credentials: String { return self._s[2275]! } + public var Preview_SaveGif: String { return self._s[2276]! } + public var Cache_Photos: String { return self._s[2277]! } + public var Channel_AdminLogFilter_EventsCalls: String { return self._s[2278]! } + public var Conversation_ContextMenuCancelEditing: String { return self._s[2279]! } + public var Contacts_FailedToSendInvitesMessage: String { return self._s[2280]! } + public var Passport_Language_lt: String { return self._s[2281]! } + public var Passport_DeleteDocument: String { return self._s[2283]! } + public var GroupInfo_SetGroupPhotoStop: String { return self._s[2284]! } public func Location_ProximityNotification_NotifyLong(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2259]!, self._r[2259]!, [_1, _2]) + return formatWithArgumentRanges(self._s[2285]!, self._r[2285]!, [_1, _2]) } - public var AccessDenied_VideoMessageMicrophone: String { return self._s[2260]! } + public var AccessDenied_VideoMessageMicrophone: String { return self._s[2286]! } public func PeopleNearby_VisibleUntil(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2261]!, self._r[2261]!, [_0]) + return formatWithArgumentRanges(self._s[2287]!, self._r[2287]!, [_0]) } - public var AccessDenied_VideoCallCamera: String { return self._s[2262]! } + public var AccessDenied_VideoCallCamera: String { return self._s[2288]! } public func Channel_AdminLog_MessageDeleted(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2263]!, self._r[2263]!, [_0]) + return formatWithArgumentRanges(self._s[2289]!, self._r[2289]!, [_0]) } - public var PhotoEditor_SharpenTool: String { return self._s[2264]! } + public var PhotoEditor_SharpenTool: String { return self._s[2290]! } public func PUSH_CHANNEL_MESSAGE_AUDIO(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2265]!, self._r[2265]!, [_1]) + return formatWithArgumentRanges(self._s[2291]!, self._r[2291]!, [_1]) } - public var DialogList_Unpin: String { return self._s[2266]! } - public var Stickers_NoStickersFound: String { return self._s[2267]! } - public var UserInfo_AddContact: String { return self._s[2269]! } + public var DialogList_Unpin: String { return self._s[2292]! } + public var Stickers_NoStickersFound: String { return self._s[2293]! } + public var UserInfo_AddContact: String { return self._s[2295]! } public func AddContact_SharedContactExceptionInfo(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2271]!, self._r[2271]!, [_0]) + return formatWithArgumentRanges(self._s[2297]!, self._r[2297]!, [_0]) } public func Notification_PinnedLocationMessage(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2272]!, self._r[2272]!, [_0]) + return formatWithArgumentRanges(self._s[2298]!, self._r[2298]!, [_0]) } - public var CallFeedback_VideoReasonDistorted: String { return self._s[2273]! } - public var Tour_Text2: String { return self._s[2274]! } + public var CallFeedback_VideoReasonDistorted: String { return self._s[2299]! } + public var Tour_Text2: String { return self._s[2300]! } public func Conversation_TitleCommentsFormat(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2276]!, self._r[2276]!, [_1, _2]) + return formatWithArgumentRanges(self._s[2302]!, self._r[2302]!, [_1, _2]) } - public var Paint_Delete: String { return self._s[2278]! } + public var InviteLink_DeleteAllRevokedLinksAlert_Text: String { return self._s[2304]! } + public var Paint_Delete: String { return self._s[2305]! } public func Call_VoiceChatInProgressMessage(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2279]!, self._r[2279]!, [_1, _2]) + return formatWithArgumentRanges(self._s[2306]!, self._r[2306]!, [_1, _2]) } - public var SettingsSearch_Synonyms_Notifications_InAppNotificationsVibrate: String { return self._s[2280]! } + public var SettingsSearch_Synonyms_Notifications_InAppNotificationsVibrate: String { return self._s[2307]! } public func PrivacySettings_LastSeenEverybodyMinus(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2282]!, self._r[2282]!, [_0]) + return formatWithArgumentRanges(self._s[2309]!, self._r[2309]!, [_0]) } - public var Privacy_Calls_NeverAllow_Title: String { return self._s[2283]! } - public var Notification_CallOutgoingShort: String { return self._s[2284]! } - public var Checkout_PasswordEntry_Title: String { return self._s[2285]! } - public var Channel_AdminLogFilter_AdminsAll: String { return self._s[2286]! } - public var Notification_MessageLifetime1m: String { return self._s[2287]! } - public var BlockedUsers_AddNew: String { return self._s[2289]! } - public var FastTwoStepSetup_EmailSection: String { return self._s[2290]! } - public var Settings_SaveEditedPhotos: String { return self._s[2291]! } - public var GroupInfo_GroupNamePlaceholder: String { return self._s[2292]! } - public var Channel_AboutItem: String { return self._s[2293]! } - public var GroupInfo_InviteLink_RevokeLink: String { return self._s[2294]! } - public var Privacy_Calls_P2PNever: String { return self._s[2296]! } - public var Passport_Language_uk: String { return self._s[2297]! } - public var NetworkUsageSettings_Wifi: String { return self._s[2298]! } - public var Conversation_Moderate_Report: String { return self._s[2299]! } - public var Wallpaper_ResetWallpapersConfirmation: String { return self._s[2300]! } - public var VoiceOver_Chat_SeenByRecipients: String { return self._s[2301]! } - public var Permissions_SiriText_v0: String { return self._s[2302]! } - public var Theme_Colors_Background: String { return self._s[2303]! } - public var Notification_CallMissed: String { return self._s[2304]! } - public var Stats_ZoomOut: String { return self._s[2305]! } - public var Profile_AddToExisting: String { return self._s[2306]! } - public var Passport_FieldAddressUploadHelp: String { return self._s[2309]! } - public var VoiceChat_RemovePeerRemove: String { return self._s[2310]! } - public var Undo_DeletedChannel: String { return self._s[2311]! } + public var Privacy_Calls_NeverAllow_Title: String { return self._s[2310]! } + public var Notification_CallOutgoingShort: String { return self._s[2311]! } + public var Checkout_PasswordEntry_Title: String { return self._s[2312]! } + public var Channel_AdminLogFilter_AdminsAll: String { return self._s[2313]! } + public var Notification_MessageLifetime1m: String { return self._s[2314]! } + public var BlockedUsers_AddNew: String { return self._s[2316]! } + public var FastTwoStepSetup_EmailSection: String { return self._s[2317]! } + public var Settings_SaveEditedPhotos: String { return self._s[2318]! } + public var GroupInfo_GroupNamePlaceholder: String { return self._s[2319]! } + public var Channel_AboutItem: String { return self._s[2320]! } + public var GroupInfo_InviteLink_RevokeLink: String { return self._s[2321]! } + public var Privacy_Calls_P2PNever: String { return self._s[2323]! } + public var Passport_Language_uk: String { return self._s[2324]! } + public var NetworkUsageSettings_Wifi: String { return self._s[2325]! } + public var Conversation_Moderate_Report: String { return self._s[2326]! } + public var Wallpaper_ResetWallpapersConfirmation: String { return self._s[2327]! } + public var VoiceOver_Chat_SeenByRecipients: String { return self._s[2328]! } + public var Permissions_SiriText_v0: String { return self._s[2329]! } + public var Theme_Colors_Background: String { return self._s[2330]! } + public var Notification_CallMissed: String { return self._s[2331]! } + public var Stats_ZoomOut: String { return self._s[2332]! } + public var Profile_AddToExisting: String { return self._s[2333]! } + public var Passport_FieldAddressUploadHelp: String { return self._s[2336]! } + public var VoiceChat_RemovePeerRemove: String { return self._s[2337]! } + public var Undo_DeletedChannel: String { return self._s[2338]! } public func Channel_AdminLog_MessagePinned(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2312]!, self._r[2312]!, [_0]) + return formatWithArgumentRanges(self._s[2339]!, self._r[2339]!, [_0]) } - public var Login_ResetAccountProtected_TimerTitle: String { return self._s[2313]! } - public var Map_LiveLocationGroupDescription: String { return self._s[2314]! } - public var Passport_InfoFAQ_URL: String { return self._s[2315]! } - public var IntentsSettings_SuggestedChats: String { return self._s[2317]! } + public var Login_ResetAccountProtected_TimerTitle: String { return self._s[2340]! } + public var Map_LiveLocationGroupDescription: String { return self._s[2341]! } + public var Passport_InfoFAQ_URL: String { return self._s[2342]! } + public var IntentsSettings_SuggestedChats: String { return self._s[2344]! } public func PUSH_MESSAGE_DOC(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2318]!, self._r[2318]!, [_1]) + return formatWithArgumentRanges(self._s[2345]!, self._r[2345]!, [_1]) } - public var State_connecting: String { return self._s[2319]! } - public var Passport_Identity_Country: String { return self._s[2320]! } - public var Passport_PasswordDescription: String { return self._s[2321]! } - public var ChatList_PsaLabel_covid: String { return self._s[2322]! } + public var State_connecting: String { return self._s[2346]! } + public var Passport_Identity_Country: String { return self._s[2347]! } + public var Passport_PasswordDescription: String { return self._s[2348]! } + public var ChatList_PsaLabel_covid: String { return self._s[2349]! } public func PUSH_MESSAGE(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2323]!, self._r[2323]!, [_1]) + return formatWithArgumentRanges(self._s[2350]!, self._r[2350]!, [_1]) } - public var Contacts_AddPeopleNearby: String { return self._s[2324]! } - public var OwnershipTransfer_SetupTwoStepAuth: String { return self._s[2325]! } - public var ClearCache_Description: String { return self._s[2326]! } - public var Localization_LanguageName: String { return self._s[2327]! } + public var Contacts_AddPeopleNearby: String { return self._s[2351]! } + public var OwnershipTransfer_SetupTwoStepAuth: String { return self._s[2352]! } + public var ClearCache_Description: String { return self._s[2353]! } + public var Localization_LanguageName: String { return self._s[2354]! } public func UserInfo_UnblockConfirmation(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2328]!, self._r[2328]!, [_0]) + return formatWithArgumentRanges(self._s[2355]!, self._r[2355]!, [_0]) } - public var ChatList_TabIconFoldersTooltipEmptyFolders: String { return self._s[2329]! } - public var UserInfo_CreateNewContact: String { return self._s[2330]! } - public var Channel_Stickers_NotFound: String { return self._s[2332]! } - public var Watch_Message_Poll: String { return self._s[2333]! } - public var Privacy_Forwards_WhoCanForward: String { return self._s[2334]! } + public var ChatList_TabIconFoldersTooltipEmptyFolders: String { return self._s[2356]! } + public var UserInfo_CreateNewContact: String { return self._s[2357]! } + public var Channel_Stickers_NotFound: String { return self._s[2359]! } + public var Watch_Message_Poll: String { return self._s[2360]! } + public var Privacy_Forwards_WhoCanForward: String { return self._s[2361]! } public func Notification_Kicked(_ _0: String, _ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2335]!, self._r[2335]!, [_0, _1]) + return formatWithArgumentRanges(self._s[2362]!, self._r[2362]!, [_0, _1]) } - public var Login_InfoDeletePhoto: String { return self._s[2336]! } - public var Appearance_ThemePreview_ChatList_6_Name: String { return self._s[2337]! } - public var InstantPage_FeedbackButton: String { return self._s[2338]! } - public var Appearance_PreviewReplyText: String { return self._s[2339]! } - public var Passport_FieldPhoneHelp: String { return self._s[2340]! } - public var Group_ErrorAddTooMuchBots: String { return self._s[2341]! } - public var Media_SendingOptionsTooltip: String { return self._s[2342]! } - public var ScheduledMessages_ScheduledOnline: String { return self._s[2343]! } - public var Notifications_Badge: String { return self._s[2344]! } - public var VoiceOver_Chat_VideoMessage: String { return self._s[2345]! } - public var TwoStepAuth_RecoveryCodeExpired: String { return self._s[2346]! } + public var Login_InfoDeletePhoto: String { return self._s[2363]! } + public var Appearance_ThemePreview_ChatList_6_Name: String { return self._s[2364]! } + public var InstantPage_FeedbackButton: String { return self._s[2365]! } + public var Appearance_PreviewReplyText: String { return self._s[2366]! } + public var Passport_FieldPhoneHelp: String { return self._s[2367]! } + public var Group_ErrorAddTooMuchBots: String { return self._s[2368]! } + public var Media_SendingOptionsTooltip: String { return self._s[2369]! } + public var ScheduledMessages_ScheduledOnline: String { return self._s[2370]! } + public var Notifications_Badge: String { return self._s[2371]! } + public var VoiceOver_Chat_VideoMessage: String { return self._s[2372]! } + public var TwoStepAuth_RecoveryCodeExpired: String { return self._s[2373]! } public func Notification_PinnedPhotoMessage(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2348]!, self._r[2348]!, [_0]) + return formatWithArgumentRanges(self._s[2375]!, self._r[2375]!, [_0]) } - public var Passport_InfoLearnMore: String { return self._s[2349]! } - public var EnterPasscode_EnterTitle: String { return self._s[2350]! } - public var Appearance_EditTheme: String { return self._s[2351]! } - public var EditTheme_Expand_BottomInfo: String { return self._s[2352]! } - public var Stats_FollowersTitle: String { return self._s[2353]! } - public var Passport_Identity_SurnamePlaceholder: String { return self._s[2354]! } - public var Channel_Subscribers_Title: String { return self._s[2355]! } - public var Group_ErrorSupergroupConversionNotPossible: String { return self._s[2356]! } - public var EditTheme_ThemeTemplateAlertTitle: String { return self._s[2357]! } - public var EditTheme_Create_Preview_IncomingText: String { return self._s[2358]! } - public var Conversation_AddToReadingList: String { return self._s[2359]! } + public var Passport_InfoLearnMore: String { return self._s[2376]! } + public var EnterPasscode_EnterTitle: String { return self._s[2377]! } + public var Appearance_EditTheme: String { return self._s[2378]! } + public var EditTheme_Expand_BottomInfo: String { return self._s[2379]! } + public var Stats_FollowersTitle: String { return self._s[2380]! } + public var Passport_Identity_SurnamePlaceholder: String { return self._s[2381]! } + public var Channel_Subscribers_Title: String { return self._s[2382]! } + public var Group_ErrorSupergroupConversionNotPossible: String { return self._s[2383]! } + public var EditTheme_ThemeTemplateAlertTitle: String { return self._s[2384]! } + public var EditTheme_Create_Preview_IncomingText: String { return self._s[2385]! } + public var Conversation_AddToReadingList: String { return self._s[2386]! } public func Notifications_ExceptionsChangeSound(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2360]!, self._r[2360]!, [_0]) + return formatWithArgumentRanges(self._s[2387]!, self._r[2387]!, [_0]) } - public var Group_AdminLog_EmptyText: String { return self._s[2361]! } - public var Passport_Identity_EditInternalPassport: String { return self._s[2362]! } - public var Watch_Location_Current: String { return self._s[2363]! } - public var PrivacyPolicy_Title: String { return self._s[2364]! } - public var Privacy_GroupsAndChannels_CustomHelp: String { return self._s[2371]! } - public var Channel_TypeSetup_Title: String { return self._s[2374]! } - public var Appearance_PreviewReplyAuthor: String { return self._s[2375]! } - public var Passport_Language_ja: String { return self._s[2376]! } - public var ReportPeer_ReasonSpam: String { return self._s[2377]! } - public var Widget_GalleryDescription: String { return self._s[2378]! } - public var Privacy_PaymentsClearInfoHelp: String { return self._s[2379]! } - public var Conversation_EditingMessageMediaEditCurrentPhoto: String { return self._s[2381]! } - public var Channel_AdminLog_ChangeInfo: String { return self._s[2382]! } - public var ChatListFolder_NameNonContacts: String { return self._s[2383]! } - public var Call_Audio: String { return self._s[2384]! } - public var PhotoEditor_CurvesGreen: String { return self._s[2385]! } - public var ChatList_Search_NoResultsFitlerFiles: String { return self._s[2386]! } - public var Settings_PrivacySettings: String { return self._s[2387]! } - public var Stats_Followers: String { return self._s[2388]! } - public var Notifications_AddExceptionTitle: String { return self._s[2389]! } - public var TwoFactorSetup_Password_Title: String { return self._s[2390]! } - public var ChannelMembers_WhoCanAddMembersAllHelp: String { return self._s[2391]! } - public var OldChannels_NoticeText: String { return self._s[2392]! } - public var Conversation_SavedMessages: String { return self._s[2393]! } + public var Group_AdminLog_EmptyText: String { return self._s[2388]! } + public var Passport_Identity_EditInternalPassport: String { return self._s[2389]! } + public var Watch_Location_Current: String { return self._s[2390]! } + public var PrivacyPolicy_Title: String { return self._s[2391]! } + public var Privacy_GroupsAndChannels_CustomHelp: String { return self._s[2398]! } + public var Channel_TypeSetup_Title: String { return self._s[2402]! } + public var Appearance_PreviewReplyAuthor: String { return self._s[2403]! } + public var Passport_Language_ja: String { return self._s[2404]! } + public var ReportPeer_ReasonSpam: String { return self._s[2405]! } + public var Widget_GalleryDescription: String { return self._s[2406]! } + public var Privacy_PaymentsClearInfoHelp: String { return self._s[2407]! } + public var Conversation_EditingMessageMediaEditCurrentPhoto: String { return self._s[2409]! } + public var Channel_AdminLog_ChangeInfo: String { return self._s[2410]! } + public var ChatListFolder_NameNonContacts: String { return self._s[2411]! } + public var Call_Audio: String { return self._s[2412]! } + public var PhotoEditor_CurvesGreen: String { return self._s[2413]! } + public var ChatList_Search_NoResultsFitlerFiles: String { return self._s[2414]! } + public var Settings_PrivacySettings: String { return self._s[2415]! } + public var InviteLink_UsageLimitReached: String { return self._s[2416]! } + public var Stats_Followers: String { return self._s[2417]! } + public var Notifications_AddExceptionTitle: String { return self._s[2418]! } + public var TwoFactorSetup_Password_Title: String { return self._s[2419]! } + public var ChannelMembers_WhoCanAddMembersAllHelp: String { return self._s[2420]! } + public var OldChannels_NoticeText: String { return self._s[2421]! } + public var Conversation_SavedMessages: String { return self._s[2422]! } public func Conversation_PeerNearbyTitle(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2395]!, self._r[2395]!, [_1, _2]) + return formatWithArgumentRanges(self._s[2424]!, self._r[2424]!, [_1, _2]) } - public var Passport_Address_TypeResidentialAddress: String { return self._s[2396]! } - public var Appearance_ThemeNightBlue: String { return self._s[2397]! } - public var Notification_ChannelInviterSelf: String { return self._s[2398]! } - public var Watch_UserInfo_Service: String { return self._s[2400]! } - public var ChatList_Context_Back: String { return self._s[2401]! } - public var Passport_Email_Title: String { return self._s[2402]! } - public var Stats_GroupTopAdmin_Promote: String { return self._s[2403]! } + public var Passport_Address_TypeResidentialAddress: String { return self._s[2425]! } + public var Appearance_ThemeNightBlue: String { return self._s[2426]! } + public var Notification_ChannelInviterSelf: String { return self._s[2427]! } + public var InviteLink_Create_TimeLimitExpiryDateNever: String { return self._s[2429]! } + public var Watch_UserInfo_Service: String { return self._s[2430]! } + public var ChatList_Context_Back: String { return self._s[2431]! } + public var Passport_Email_Title: String { return self._s[2432]! } + public var Stats_GroupTopAdmin_Promote: String { return self._s[2433]! } public func PUSH_PINNED_INVOICE(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2404]!, self._r[2404]!, [_1]) + return formatWithArgumentRanges(self._s[2434]!, self._r[2434]!, [_1]) } - public var Conversation_UnsupportedMedia: String { return self._s[2405]! } - public var Passport_Address_OneOfTypePassportRegistration: String { return self._s[2406]! } - public var Privacy_TopPeersHelp: String { return self._s[2408]! } - public var Privacy_Forwards_AlwaysLink: String { return self._s[2409]! } - public var Notifications_Badge_CountUnreadMessages_InfoOn: String { return self._s[2410]! } - public var Permissions_NotificationsTitle_v0: String { return self._s[2411]! } + public var Conversation_UnsupportedMedia: String { return self._s[2435]! } + public var Passport_Address_OneOfTypePassportRegistration: String { return self._s[2436]! } + public var Privacy_TopPeersHelp: String { return self._s[2438]! } + public var Privacy_Forwards_AlwaysLink: String { return self._s[2439]! } + public var Notifications_Badge_CountUnreadMessages_InfoOn: String { return self._s[2440]! } + public var Permissions_NotificationsTitle_v0: String { return self._s[2441]! } public func Location_ProximityNotification_AlreadyClose(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2412]!, self._r[2412]!, [_0]) + return formatWithArgumentRanges(self._s[2442]!, self._r[2442]!, [_0]) } - public var Notification_PassportValueProofOfAddress: String { return self._s[2413]! } - public var Map_Map: String { return self._s[2414]! } - public var WallpaperSearch_ColorBlue: String { return self._s[2415]! } - public var Privacy_Calls_CustomShareHelp: String { return self._s[2416]! } - public var PhotoEditor_BlurToolRadial: String { return self._s[2417]! } - public var ChatList_Search_FilterMusic: String { return self._s[2418]! } - public var SettingsSearch_Synonyms_Data_AutoplayGifs: String { return self._s[2419]! } - public var Privacy_PaymentsClear_ShippingInfo: String { return self._s[2420]! } - public var Settings_LogoutConfirmationTitle: String { return self._s[2422]! } + public var Notification_PassportValueProofOfAddress: String { return self._s[2443]! } + public var Map_Map: String { return self._s[2444]! } + public var WallpaperSearch_ColorBlue: String { return self._s[2445]! } + public var Privacy_Calls_CustomShareHelp: String { return self._s[2446]! } + public var PhotoEditor_BlurToolRadial: String { return self._s[2447]! } + public var ChatList_Search_FilterMusic: String { return self._s[2448]! } + public var SettingsSearch_Synonyms_Data_AutoplayGifs: String { return self._s[2449]! } + public var Privacy_PaymentsClear_ShippingInfo: String { return self._s[2450]! } + public var Settings_LogoutConfirmationTitle: String { return self._s[2452]! } public func PUSH_CHANNEL_MESSAGE_VIDEOS(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2423]!, self._r[2423]!, [_1, _2]) + return formatWithArgumentRanges(self._s[2453]!, self._r[2453]!, [_1, _2]) } public func Notification_ChangedGroupPhoto(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2424]!, self._r[2424]!, [_0]) + return formatWithArgumentRanges(self._s[2454]!, self._r[2454]!, [_0]) } - public var Channel_Username_RevokeExistingUsernamesInfo: String { return self._s[2425]! } - public var Group_Username_CreatePublicLinkHelp: String { return self._s[2426]! } - public var GroupInfo_Location: String { return self._s[2429]! } - public var Passport_Language_ka: String { return self._s[2430]! } + public var Channel_Username_RevokeExistingUsernamesInfo: String { return self._s[2455]! } + public var Group_Username_CreatePublicLinkHelp: String { return self._s[2456]! } + public var GroupInfo_Location: String { return self._s[2459]! } + public var Passport_Language_ka: String { return self._s[2460]! } public func TwoStepAuth_SetupPendingEmail(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2431]!, self._r[2431]!, [_0]) + return formatWithArgumentRanges(self._s[2461]!, self._r[2461]!, [_0]) } - public var Conversation_ContextMenuOpenChannelProfile: String { return self._s[2432]! } - public var ScheduledMessages_ClearAllConfirmation: String { return self._s[2435]! } - public var DialogList_SearchSectionRecent: String { return self._s[2436]! } - public var Passport_Address_OneOfTypeTemporaryRegistration: String { return self._s[2437]! } - public var Conversation_Timer_Send: String { return self._s[2438]! } - public var ChatState_Updating: String { return self._s[2440]! } - public var ChannelMembers_WhoCanAddMembers: String { return self._s[2441]! } - public var ChannelInfo_DeleteGroup: String { return self._s[2442]! } - public var TwoStepAuth_RecoveryFailed: String { return self._s[2443]! } - public var Channel_OwnershipTransfer_EnterPassword: String { return self._s[2444]! } - public var ChatList_Search_NoResults: String { return self._s[2445]! } - public var ChatListFolderSettings_AddRecommended: String { return self._s[2447]! } - public var ChangePhoneNumberCode_Called: String { return self._s[2448]! } - public var PeerInfo_GroupAboutItem: String { return self._s[2449]! } + public var Conversation_ContextMenuOpenChannelProfile: String { return self._s[2462]! } + public var ScheduledMessages_ClearAllConfirmation: String { return self._s[2465]! } + public var DialogList_SearchSectionRecent: String { return self._s[2466]! } + public var Passport_Address_OneOfTypeTemporaryRegistration: String { return self._s[2467]! } + public var Conversation_Timer_Send: String { return self._s[2468]! } + public var ChatState_Updating: String { return self._s[2470]! } + public var ChannelMembers_WhoCanAddMembers: String { return self._s[2471]! } + public var ChannelInfo_DeleteGroup: String { return self._s[2472]! } + public var TwoStepAuth_RecoveryFailed: String { return self._s[2473]! } + public var Channel_OwnershipTransfer_EnterPassword: String { return self._s[2474]! } + public var InviteLink_Create_TimeLimitExpiryTime: String { return self._s[2475]! } + public var ChatList_Search_NoResults: String { return self._s[2476]! } + public var ChatListFolderSettings_AddRecommended: String { return self._s[2478]! } + public var ChangePhoneNumberCode_Called: String { return self._s[2479]! } + public var PeerInfo_GroupAboutItem: String { return self._s[2480]! } public func LiveLocationUpdated_YesterdayAt(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2451]!, self._r[2451]!, [_0]) + return formatWithArgumentRanges(self._s[2482]!, self._r[2482]!, [_0]) } - public var PrivacySettings_AuthSessions: String { return self._s[2452]! } - public var Passport_Address_Postcode: String { return self._s[2453]! } - public var VoiceOver_Chat_YourVideoMessage: String { return self._s[2454]! } - public var Passport_Address_Street2Placeholder: String { return self._s[2455]! } - public var Group_Location_Title: String { return self._s[2456]! } - public var SettingsSearch_Synonyms_Data_AutoDownloadReset: String { return self._s[2457]! } - public var PeopleNearby_UsersEmpty: String { return self._s[2458]! } - public var SettingsSearch_Synonyms_Data_Title: String { return self._s[2460]! } + public var PrivacySettings_AuthSessions: String { return self._s[2483]! } + public var Passport_Address_Postcode: String { return self._s[2484]! } + public var VoiceOver_Chat_YourVideoMessage: String { return self._s[2485]! } + public var Passport_Address_Street2Placeholder: String { return self._s[2486]! } + public var Group_Location_Title: String { return self._s[2487]! } + public var SettingsSearch_Synonyms_Data_AutoDownloadReset: String { return self._s[2488]! } + public var PeopleNearby_UsersEmpty: String { return self._s[2489]! } + public var SettingsSearch_Synonyms_Data_Title: String { return self._s[2491]! } public func Checkout_PasswordEntry_Text(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2462]!, self._r[2462]!, [_0]) + return formatWithArgumentRanges(self._s[2493]!, self._r[2493]!, [_0]) } - public var Proxy_TooltipUnavailable: String { return self._s[2463]! } - public var Map_Search: String { return self._s[2464]! } - public var AutoDownloadSettings_TypeContacts: String { return self._s[2465]! } - public var Conversation_SearchByName_Prefix: String { return self._s[2466]! } + public var Proxy_TooltipUnavailable: String { return self._s[2494]! } + public var Map_Search: String { return self._s[2495]! } + public var AutoDownloadSettings_TypeContacts: String { return self._s[2496]! } + public var Conversation_SearchByName_Prefix: String { return self._s[2497]! } public func Channel_AdminLog_MessageToggleSignaturesOff(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2467]!, self._r[2467]!, [_0]) + return formatWithArgumentRanges(self._s[2498]!, self._r[2498]!, [_0]) } - public var TwoStepAuth_EmailAddSuccess: String { return self._s[2468]! } - public var ProfilePhoto_MainPhoto: String { return self._s[2469]! } - public var SettingsSearch_Synonyms_Notifications_InAppNotificationsSound: String { return self._s[2470]! } - public var SharedMedia_EmptyMusicText: String { return self._s[2471]! } - public var ChatSettings_AutoDownloadPhotos: String { return self._s[2472]! } - public var NetworkUsageSettings_BytesReceived: String { return self._s[2473]! } - public var Channel_AdminLog_EmptyText: String { return self._s[2474]! } - public var Channel_BanUser_PermissionSendMessages: String { return self._s[2475]! } - public var Undo_ChatDeletedForBothSides: String { return self._s[2476]! } - public var Notifications_GroupNotifications: String { return self._s[2477]! } - public var AccessDenied_SaveMedia: String { return self._s[2478]! } - public var GroupInfo_LabelOwner: String { return self._s[2479]! } - public var Passport_Language_id: String { return self._s[2480]! } - public var ChatSettings_AutoDownloadTitle: String { return self._s[2481]! } - public var Conversation_UnpinMessageAlert: String { return self._s[2482]! } + public var TwoStepAuth_EmailAddSuccess: String { return self._s[2499]! } + public var ProfilePhoto_MainPhoto: String { return self._s[2500]! } + public var SettingsSearch_Synonyms_Notifications_InAppNotificationsSound: String { return self._s[2501]! } + public var SharedMedia_EmptyMusicText: String { return self._s[2502]! } + public var ChatSettings_AutoDownloadPhotos: String { return self._s[2503]! } + public var NetworkUsageSettings_BytesReceived: String { return self._s[2504]! } + public var Channel_AdminLog_EmptyText: String { return self._s[2505]! } + public var Channel_BanUser_PermissionSendMessages: String { return self._s[2506]! } + public var Undo_ChatDeletedForBothSides: String { return self._s[2507]! } + public var Notifications_GroupNotifications: String { return self._s[2508]! } + public var AccessDenied_SaveMedia: String { return self._s[2509]! } + public var InviteLink_Create_Revoke: String { return self._s[2510]! } + public var GroupInfo_LabelOwner: String { return self._s[2511]! } + public var Passport_Language_id: String { return self._s[2512]! } + public var ChatSettings_AutoDownloadTitle: String { return self._s[2513]! } + public var Conversation_UnpinMessageAlert: String { return self._s[2514]! } public func LiveLocationUpdated_TodayAt(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2483]!, self._r[2483]!, [_0]) - } - public func Call_RemoteVideoPaused(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2484]!, self._r[2484]!, [_0]) - } - public var TwoFactorSetup_Done_Text: String { return self._s[2485]! } - public func LastSeen_AtDate(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2486]!, self._r[2486]!, [_0]) - } - public var NetworkUsageSettings_BytesSent: String { return self._s[2487]! } - public var OwnershipTransfer_Transfer: String { return self._s[2488]! } - public func Notification_Exceptions_Sound(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2489]!, self._r[2489]!, [_0]) - } - public var Passport_Language_pt: String { return self._s[2490]! } - public var PrivacySettings_WebSessions: String { return self._s[2491]! } - public var PrivacyPolicy_DeclineDeleteNow: String { return self._s[2493]! } - public var TwoFactorSetup_Hint_Title: String { return self._s[2494]! } - public func Notification_Joined(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2495]!, self._r[2495]!, [_0]) - } - public var Group_Username_RemoveExistingUsernamesInfo: String { return self._s[2496]! } - public var PrivacyLastSeenSettings_CustomShareSettings_Delete: String { return self._s[2497]! } - public var AutoNightTheme_Scheduled: String { return self._s[2498]! } - public var CreatePoll_ExplanationHeader: String { return self._s[2499]! } - public var Calls_TabTitle: String { return self._s[2500]! } - public var VoiceChat_RecordingInProgress: String { return self._s[2501]! } - public var ChatList_UndoArchiveHiddenText: String { return self._s[2502]! } - public var Notification_VideoCallCanceled: String { return self._s[2503]! } - public var Login_CodeSentInternal: String { return self._s[2504]! } - public var SettingsSearch_Synonyms_Proxy_AddProxy: String { return self._s[2505]! } - public var Call_RecordingDisabledMessage: String { return self._s[2507]! } - public func VoiceChat_RemovedPeerText(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2508]!, self._r[2508]!, [_0]) - } - public var AutoDownloadSettings_TypeChannels: String { return self._s[2510]! } - public var Channel_Info_Stickers: String { return self._s[2511]! } - public var Passport_DeleteAddressConfirmation: String { return self._s[2512]! } - public func Conversation_PeerNearbyDistance(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2513]!, self._r[2513]!, [_1, _2]) - } - public var ChannelMembers_WhoCanAddMembers_Admins: String { return self._s[2514]! } - public func Call_StatusOngoing(_ _0: String) -> (String, [(Int, NSRange)]) { return formatWithArgumentRanges(self._s[2515]!, self._r[2515]!, [_0]) } - public var Passport_DiscardMessageTitle: String { return self._s[2516]! } - public var Call_VoiceOver_VideoCallIncoming: String { return self._s[2517]! } - public var Localization_LanguageOther: String { return self._s[2518]! } - public var Conversation_EncryptionCanceled: String { return self._s[2519]! } - public var ChatSettings_AutomaticPhotoDownload: String { return self._s[2520]! } + public func Call_RemoteVideoPaused(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[2516]!, self._r[2516]!, [_0]) + } + public var TwoFactorSetup_Done_Text: String { return self._s[2517]! } + public func LastSeen_AtDate(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[2518]!, self._r[2518]!, [_0]) + } + public var NetworkUsageSettings_BytesSent: String { return self._s[2519]! } + public var OwnershipTransfer_Transfer: String { return self._s[2520]! } + public func Notification_Exceptions_Sound(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[2521]!, self._r[2521]!, [_0]) + } + public var Passport_Language_pt: String { return self._s[2522]! } + public var PrivacySettings_WebSessions: String { return self._s[2523]! } + public var PrivacyPolicy_DeclineDeleteNow: String { return self._s[2525]! } + public var TwoFactorSetup_Hint_Title: String { return self._s[2526]! } + public func Notification_Joined(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[2527]!, self._r[2527]!, [_0]) + } + public var Group_Username_RemoveExistingUsernamesInfo: String { return self._s[2528]! } + public var PrivacyLastSeenSettings_CustomShareSettings_Delete: String { return self._s[2529]! } + public var AutoNightTheme_Scheduled: String { return self._s[2530]! } + public var CreatePoll_ExplanationHeader: String { return self._s[2531]! } + public var Calls_TabTitle: String { return self._s[2532]! } + public var VoiceChat_RecordingInProgress: String { return self._s[2533]! } + public var ChatList_UndoArchiveHiddenText: String { return self._s[2534]! } + public var Notification_VideoCallCanceled: String { return self._s[2535]! } + public var Login_CodeSentInternal: String { return self._s[2536]! } + public var SettingsSearch_Synonyms_Proxy_AddProxy: String { return self._s[2537]! } + public var Call_RecordingDisabledMessage: String { return self._s[2539]! } + public func VoiceChat_RemovedPeerText(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[2540]!, self._r[2540]!, [_0]) + } + public var AutoDownloadSettings_TypeChannels: String { return self._s[2542]! } + public var Channel_Info_Stickers: String { return self._s[2543]! } + public var Passport_DeleteAddressConfirmation: String { return self._s[2544]! } + public func Conversation_PeerNearbyDistance(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[2545]!, self._r[2545]!, [_1, _2]) + } + public var ChannelMembers_WhoCanAddMembers_Admins: String { return self._s[2546]! } + public func Call_StatusOngoing(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[2547]!, self._r[2547]!, [_0]) + } + public var Passport_DiscardMessageTitle: String { return self._s[2548]! } + public var Call_VoiceOver_VideoCallIncoming: String { return self._s[2549]! } + public var Localization_LanguageOther: String { return self._s[2550]! } + public var Conversation_EncryptionCanceled: String { return self._s[2551]! } + public var ChatSettings_AutomaticPhotoDownload: String { return self._s[2552]! } public func Notification_SecretChatMessageScreenshot(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2522]!, self._r[2522]!, [_0]) + return formatWithArgumentRanges(self._s[2554]!, self._r[2554]!, [_0]) } - public var Target_InviteToGroupErrorAlreadyInvited: String { return self._s[2524]! } - public var SocksProxySetup_SavedProxies: String { return self._s[2525]! } + public var Target_InviteToGroupErrorAlreadyInvited: String { return self._s[2556]! } + public var SocksProxySetup_SavedProxies: String { return self._s[2557]! } + public var InviteLink_Create_UsersLimitNumberOfUsers: String { return self._s[2558]! } public func ApplyLanguage_ChangeLanguageAlreadyActive(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2526]!, self._r[2526]!, [_1]) + return formatWithArgumentRanges(self._s[2559]!, self._r[2559]!, [_1]) } - public var Conversation_ScamWarning: String { return self._s[2528]! } - public var Channel_AdminLog_InfoPanelAlertTitle: String { return self._s[2529]! } - public var LocalGroup_Title: String { return self._s[2530]! } - public var SettingsSearch_Synonyms_Notifications_MessageNotificationsAlert: String { return self._s[2532]! } - public var SettingsSearch_Synonyms_Privacy_PasscodeAndFaceId: String { return self._s[2533]! } - public var Login_PhoneFloodError: String { return self._s[2534]! } - public var Conversation_PinMessageAlert_PinAndNotifyMembers: String { return self._s[2535]! } - public var Username_InvalidTaken: String { return self._s[2537]! } - public var SocksProxySetup_AddProxy: String { return self._s[2539]! } - public var PrivacyLastSeenSettings_WhoCanSeeMyTimestamp: String { return self._s[2540]! } - public var MediaPicker_UngroupDescription: String { return self._s[2541]! } - public var Login_CodeExpired: String { return self._s[2542]! } - public var Localization_ChooseLanguage: String { return self._s[2543]! } - public var Checkout_NewCard_PostcodePlaceholder: String { return self._s[2544]! } + public var Conversation_ScamWarning: String { return self._s[2561]! } + public var Channel_AdminLog_InfoPanelAlertTitle: String { return self._s[2562]! } + public var LocalGroup_Title: String { return self._s[2563]! } + public var SettingsSearch_Synonyms_Notifications_MessageNotificationsAlert: String { return self._s[2565]! } + public var SettingsSearch_Synonyms_Privacy_PasscodeAndFaceId: String { return self._s[2566]! } + public var Login_PhoneFloodError: String { return self._s[2567]! } + public var Conversation_PinMessageAlert_PinAndNotifyMembers: String { return self._s[2568]! } + public var Username_InvalidTaken: String { return self._s[2570]! } + public var SocksProxySetup_AddProxy: String { return self._s[2572]! } + public var PrivacyLastSeenSettings_WhoCanSeeMyTimestamp: String { return self._s[2573]! } + public var MediaPicker_UngroupDescription: String { return self._s[2574]! } + public var Login_CodeExpired: String { return self._s[2575]! } + public var Localization_ChooseLanguage: String { return self._s[2576]! } + public var Checkout_NewCard_PostcodePlaceholder: String { return self._s[2577]! } public func ChangePhone_ErrorOccupied(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2545]!, self._r[2545]!, [_0]) + return formatWithArgumentRanges(self._s[2578]!, self._r[2578]!, [_0]) } public func Channel_DiscussionGroup_HeaderSet(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2546]!, self._r[2546]!, [_0]) + return formatWithArgumentRanges(self._s[2579]!, self._r[2579]!, [_0]) } - public var ReportPeer_ReasonOther_Title: String { return self._s[2548]! } - public var Conversation_ScheduleMessage_Title: String { return self._s[2549]! } + public var ReportPeer_ReasonOther_Title: String { return self._s[2581]! } + public var Conversation_ScheduleMessage_Title: String { return self._s[2582]! } public func VoiceChat_UserInvited(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2550]!, self._r[2550]!, [_0]) + return formatWithArgumentRanges(self._s[2583]!, self._r[2583]!, [_0]) } - public var PeerInfo_ButtonDiscuss: String { return self._s[2551]! } - public var SettingsSearch_Synonyms_Notifications_BadgeIncludeMutedPublicGroups: String { return self._s[2552]! } - public var Call_StatusNoAnswer: String { return self._s[2553]! } - public var ScheduledMessages_DeleteMany: String { return self._s[2555]! } - public var Channel_DiscussionGroupInfo: String { return self._s[2556]! } - public var Conversation_UnarchiveDone: String { return self._s[2557]! } - public var LogoutOptions_AddAccountText: String { return self._s[2558]! } - public var Message_PinnedContactMessage: String { return self._s[2559]! } + public var PeerInfo_ButtonDiscuss: String { return self._s[2584]! } + public var SettingsSearch_Synonyms_Notifications_BadgeIncludeMutedPublicGroups: String { return self._s[2585]! } + public var Call_StatusNoAnswer: String { return self._s[2586]! } + public var ScheduledMessages_DeleteMany: String { return self._s[2588]! } + public var Channel_DiscussionGroupInfo: String { return self._s[2589]! } + public var Conversation_UnarchiveDone: String { return self._s[2590]! } + public var LogoutOptions_AddAccountText: String { return self._s[2591]! } + public var Message_PinnedContactMessage: String { return self._s[2592]! } public func FileSize_GB(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2561]!, self._r[2561]!, [_0]) + return formatWithArgumentRanges(self._s[2594]!, self._r[2594]!, [_0]) } - public var Stats_GroupLanguagesTitle: String { return self._s[2562]! } - public var Passport_FieldAddressHelp: String { return self._s[2563]! } + public var Stats_GroupLanguagesTitle: String { return self._s[2595]! } + public var Passport_FieldAddressHelp: String { return self._s[2596]! } public func Passport_FieldOneOf_Or(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2564]!, self._r[2564]!, [_1, _2]) + return formatWithArgumentRanges(self._s[2597]!, self._r[2597]!, [_1, _2]) } - public var ChatSettings_OpenLinksIn: String { return self._s[2566]! } - public var TwoFactorSetup_Hint_SkipAction: String { return self._s[2567]! } - public var Message_Photo: String { return self._s[2568]! } - public var Media_LimitedAccessManage: String { return self._s[2570]! } - public var MediaPicker_AddCaption: String { return self._s[2571]! } - public var LogoutOptions_Title: String { return self._s[2572]! } + public var ChatSettings_OpenLinksIn: String { return self._s[2599]! } + public var TwoFactorSetup_Hint_SkipAction: String { return self._s[2600]! } + public var Message_Photo: String { return self._s[2601]! } + public var Media_LimitedAccessManage: String { return self._s[2603]! } + public var MediaPicker_AddCaption: String { return self._s[2604]! } + public var LogoutOptions_Title: String { return self._s[2605]! } public func PUSH_PINNED_GIF(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2573]!, self._r[2573]!, [_1]) + return formatWithArgumentRanges(self._s[2606]!, self._r[2606]!, [_1]) } - public var Conversation_StatusKickedFromGroup: String { return self._s[2574]! } - public var Channel_AdminLogFilter_AdminsTitle: String { return self._s[2575]! } - public var ChatList_DeleteSavedMessagesConfirmationTitle: String { return self._s[2576]! } - public var Channel_AdminLogFilter_Title: String { return self._s[2577]! } - public var Passport_Address_TypeRentalAgreementUploadScan: String { return self._s[2578]! } - public var Compose_GroupTokenListPlaceholder: String { return self._s[2579]! } - public var Notifications_MessageNotificationsExceptions: String { return self._s[2580]! } - public var ChannelIntro_Title: String { return self._s[2581]! } - public var Stats_Message_Views: String { return self._s[2582]! } - public var Stickers_Install: String { return self._s[2583]! } + public var Conversation_StatusKickedFromGroup: String { return self._s[2607]! } + public var Channel_AdminLogFilter_AdminsTitle: String { return self._s[2608]! } + public var ChatList_DeleteSavedMessagesConfirmationTitle: String { return self._s[2609]! } + public var Channel_AdminLogFilter_Title: String { return self._s[2610]! } + public var Passport_Address_TypeRentalAgreementUploadScan: String { return self._s[2611]! } + public var Compose_GroupTokenListPlaceholder: String { return self._s[2612]! } + public var Notifications_MessageNotificationsExceptions: String { return self._s[2613]! } + public var ChannelIntro_Title: String { return self._s[2614]! } + public var Stats_Message_Views: String { return self._s[2615]! } + public var Stickers_Install: String { return self._s[2616]! } public func VoiceOver_Chat_FileFrom(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2584]!, self._r[2584]!, [_0]) + return formatWithArgumentRanges(self._s[2617]!, self._r[2617]!, [_0]) } - public var EditTheme_Create_Preview_IncomingReplyText: String { return self._s[2585]! } - public var Conversation_SwipeToReplyHintTitle: String { return self._s[2587]! } - public var Settings_Username: String { return self._s[2590]! } - public var FastTwoStepSetup_Title: String { return self._s[2591]! } - public var Notifications_Badge_CountUnreadMessages_InfoOff: String { return self._s[2592]! } - public var SettingsSearch_Synonyms_Privacy_Title: String { return self._s[2593]! } - public var Passport_Identity_IssueDatePlaceholder: String { return self._s[2594]! } - public var CallFeedback_ReasonEcho: String { return self._s[2595]! } + public var EditTheme_Create_Preview_IncomingReplyText: String { return self._s[2618]! } + public var Conversation_SwipeToReplyHintTitle: String { return self._s[2620]! } + public var Settings_Username: String { return self._s[2623]! } + public var FastTwoStepSetup_Title: String { return self._s[2624]! } + public var Notifications_Badge_CountUnreadMessages_InfoOff: String { return self._s[2625]! } + public var SettingsSearch_Synonyms_Privacy_Title: String { return self._s[2626]! } + public var Passport_Identity_IssueDatePlaceholder: String { return self._s[2627]! } + public var CallFeedback_ReasonEcho: String { return self._s[2628]! } public func Time_MonthOfYear_m1(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2596]!, self._r[2596]!, [_0]) + return formatWithArgumentRanges(self._s[2629]!, self._r[2629]!, [_0]) } - public var Conversation_OpenBotLinkTitle: String { return self._s[2597]! } - public var SocksProxySetup_Title: String { return self._s[2598]! } - public var CallFeedback_Success: String { return self._s[2599]! } - public var WallpaperPreview_SwipeTopText: String { return self._s[2601]! } - public var InstantPage_AutoNightTheme: String { return self._s[2603]! } - public var Watch_Conversation_Reply: String { return self._s[2604]! } - public var VoiceChat_Share: String { return self._s[2606]! } - public var Chat_PanelUnpinAllMessages: String { return self._s[2607]! } - public var WallpaperPreview_Pattern: String { return self._s[2608]! } - public var CheckoutInfo_ReceiverInfoEmail: String { return self._s[2609]! } + public var Conversation_OpenBotLinkTitle: String { return self._s[2630]! } + public var SocksProxySetup_Title: String { return self._s[2631]! } + public var CallFeedback_Success: String { return self._s[2632]! } + public var WallpaperPreview_SwipeTopText: String { return self._s[2634]! } + public var InstantPage_AutoNightTheme: String { return self._s[2636]! } + public var Watch_Conversation_Reply: String { return self._s[2637]! } + public var VoiceChat_Share: String { return self._s[2639]! } + public var Chat_PanelUnpinAllMessages: String { return self._s[2640]! } + public var WallpaperPreview_Pattern: String { return self._s[2641]! } + public var CheckoutInfo_ReceiverInfoEmail: String { return self._s[2642]! } public func Conversation_DeleteMessagesFor(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2610]!, self._r[2610]!, [_0]) + return formatWithArgumentRanges(self._s[2643]!, self._r[2643]!, [_0]) } - public var AutoDownloadSettings_TypeGroupChats: String { return self._s[2611]! } - public var DialogList_SavedMessagesTooltip: String { return self._s[2613]! } - public var Update_Title: String { return self._s[2614]! } - public var Conversation_ShareMyPhoneNumber: String { return self._s[2615]! } - public var WallpaperPreview_CropTopText: String { return self._s[2617]! } - public var Channel_EditMessageErrorGeneric: String { return self._s[2618]! } - public var AccessDenied_LocationAlwaysDenied: String { return self._s[2619]! } - public var ChatListFolder_DiscardCancel: String { return self._s[2620]! } - public var Message_PinnedPhotoMessage: String { return self._s[2621]! } - public var Appearance_ThemeDayClassic: String { return self._s[2622]! } - public var SocksProxySetup_ProxySocks5: String { return self._s[2623]! } - public var AccessDenied_Wallpapers: String { return self._s[2629]! } + public var AutoDownloadSettings_TypeGroupChats: String { return self._s[2644]! } + public var DialogList_SavedMessagesTooltip: String { return self._s[2646]! } + public var Update_Title: String { return self._s[2647]! } + public var Conversation_ShareMyPhoneNumber: String { return self._s[2648]! } + public var WallpaperPreview_CropTopText: String { return self._s[2650]! } + public var Channel_EditMessageErrorGeneric: String { return self._s[2651]! } + public var AccessDenied_LocationAlwaysDenied: String { return self._s[2652]! } + public var ChatListFolder_DiscardCancel: String { return self._s[2653]! } + public var Message_PinnedPhotoMessage: String { return self._s[2654]! } + public var Appearance_ThemeDayClassic: String { return self._s[2655]! } + public var SocksProxySetup_ProxySocks5: String { return self._s[2656]! } + public var AccessDenied_Wallpapers: String { return self._s[2662]! } public func Channel_AdminLog_MessageChangedGroupAbout(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2630]!, self._r[2630]!, [_0]) + return formatWithArgumentRanges(self._s[2663]!, self._r[2663]!, [_0]) } - public var Weekday_Sunday: String { return self._s[2631]! } - public var SettingsSearch_Synonyms_Privacy_GroupsAndChannels: String { return self._s[2633]! } - public var PeopleNearby_MakeVisibleDescription: String { return self._s[2634]! } - public var AccessDenied_LocationDisabled: String { return self._s[2635]! } - public var Tour_Text3: String { return self._s[2636]! } - public var AuthSessions_AddDevice_ScanTitle: String { return self._s[2637]! } + public var Weekday_Sunday: String { return self._s[2664]! } + public var SettingsSearch_Synonyms_Privacy_GroupsAndChannels: String { return self._s[2666]! } + public var PeopleNearby_MakeVisibleDescription: String { return self._s[2667]! } + public var AccessDenied_LocationDisabled: String { return self._s[2668]! } + public var Tour_Text3: String { return self._s[2669]! } + public var AuthSessions_AddDevice_ScanTitle: String { return self._s[2670]! } public func Time_TodayAt(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2638]!, self._r[2638]!, [_0]) + return formatWithArgumentRanges(self._s[2671]!, self._r[2671]!, [_0]) } - public var Privacy_SecretChatsLinkPreviewsHelp: String { return self._s[2639]! } - public var Conversation_ClearCache: String { return self._s[2640]! } - public var StickerPacksSettings_ArchivedMasks_Info: String { return self._s[2641]! } - public var ChatList_Tabs_AllChats: String { return self._s[2642]! } - public var DialogList_RecentTitlePeople: String { return self._s[2643]! } - public var Stickers_AddToFavorites: String { return self._s[2644]! } - public var ChatList_Context_RemoveFromFolder: String { return self._s[2645]! } - public var Settings_RemoveVideo: String { return self._s[2646]! } - public var PhotoEditor_CropAspectRatioSquare: String { return self._s[2647]! } - public var ConversationProfile_LeaveDeleteAndExit: String { return self._s[2648]! } - public var VoiceOver_Chat_YourFile: String { return self._s[2649]! } - public var SettingsSearch_Synonyms_Privacy_Forwards: String { return self._s[2650]! } - public var Group_OwnershipTransfer_ErrorPrivacyRestricted: String { return self._s[2651]! } - public var Channel_AdminLog_AddMembers: String { return self._s[2652]! } - public var Map_SendThisLocation: String { return self._s[2654]! } - public var TwoStepAuth_EmailSkipAlert: String { return self._s[2656]! } - public var IntentsSettings_SuggestedChatsPrivateChats: String { return self._s[2657]! } - public var CloudStorage_Title: String { return self._s[2658]! } - public var TwoFactorSetup_Password_Action: String { return self._s[2659]! } - public var TwoStepAuth_ConfirmationText: String { return self._s[2660]! } - public var Passport_Address_EditTemporaryRegistration: String { return self._s[2662]! } - public var Undo_LeftGroup: String { return self._s[2663]! } - public var Conversation_StopLiveLocation: String { return self._s[2664]! } - public var NotificationSettings_ShowNotificationsFromAccountsSection: String { return self._s[2665]! } - public var Message_PinnedInvoice: String { return self._s[2666]! } - public var ApplyLanguage_LanguageNotSupportedError: String { return self._s[2667]! } + public var Privacy_SecretChatsLinkPreviewsHelp: String { return self._s[2672]! } + public var Conversation_ClearCache: String { return self._s[2673]! } + public var StickerPacksSettings_ArchivedMasks_Info: String { return self._s[2674]! } + public var ChatList_Tabs_AllChats: String { return self._s[2675]! } + public var DialogList_RecentTitlePeople: String { return self._s[2676]! } + public var Stickers_AddToFavorites: String { return self._s[2677]! } + public var ChatList_Context_RemoveFromFolder: String { return self._s[2678]! } + public var Settings_RemoveVideo: String { return self._s[2679]! } + public var PhotoEditor_CropAspectRatioSquare: String { return self._s[2680]! } + public var ConversationProfile_LeaveDeleteAndExit: String { return self._s[2681]! } + public var VoiceOver_Chat_YourFile: String { return self._s[2682]! } + public var SettingsSearch_Synonyms_Privacy_Forwards: String { return self._s[2684]! } + public var Group_OwnershipTransfer_ErrorPrivacyRestricted: String { return self._s[2685]! } + public var Channel_AdminLog_AddMembers: String { return self._s[2686]! } + public var Map_SendThisLocation: String { return self._s[2688]! } + public var TwoStepAuth_EmailSkipAlert: String { return self._s[2690]! } + public var IntentsSettings_SuggestedChatsPrivateChats: String { return self._s[2691]! } + public var CloudStorage_Title: String { return self._s[2692]! } + public var TwoFactorSetup_Password_Action: String { return self._s[2693]! } + public var TwoStepAuth_ConfirmationText: String { return self._s[2694]! } + public var Passport_Address_EditTemporaryRegistration: String { return self._s[2696]! } + public var Undo_LeftGroup: String { return self._s[2697]! } + public var Conversation_StopLiveLocation: String { return self._s[2698]! } + public var NotificationSettings_ShowNotificationsFromAccountsSection: String { return self._s[2699]! } + public var Message_PinnedInvoice: String { return self._s[2700]! } + public var ApplyLanguage_LanguageNotSupportedError: String { return self._s[2701]! } public func PUSH_CHAT_MESSAGE(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2668]!, self._r[2668]!, [_1, _2]) + return formatWithArgumentRanges(self._s[2702]!, self._r[2702]!, [_1, _2]) } public func Notification_PinnedAudioMessage(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2669]!, self._r[2669]!, [_0]) + return formatWithArgumentRanges(self._s[2703]!, self._r[2703]!, [_0]) } - public var Weekday_Tuesday: String { return self._s[2670]! } - public var ChangePhoneNumberCode_Code: String { return self._s[2671]! } - public var VoiceOver_Chat_YourMessage: String { return self._s[2672]! } - public var Calls_CallTabDescription: String { return self._s[2673]! } - public var SocksProxySetup_UseProxy: String { return self._s[2675]! } - public var SettingsSearch_Synonyms_Stickers_Title: String { return self._s[2676]! } - public var PasscodeSettings_AlphanumericCode: String { return self._s[2677]! } - public var VoiceOver_Chat_YourVideo: String { return self._s[2678]! } - public var ChannelMembers_WhoCanAddMembersAdminsHelp: String { return self._s[2680]! } - public var SettingsSearch_Synonyms_Privacy_DeleteAccountIfAwayFor: String { return self._s[2681]! } - public var Exceptions_AddToExceptions: String { return self._s[2682]! } - public var UserInfo_Title: String { return self._s[2683]! } - public var Passport_DeleteDocumentConfirmation: String { return self._s[2685]! } - public var ChatList_Unmute: String { return self._s[2687]! } - public var SettingsSearch_Synonyms_Privacy_Data_ContactsSync: String { return self._s[2688]! } - public var Stats_GroupTopPostersTitle: String { return self._s[2689]! } - public var Username_CheckingUsername: String { return self._s[2690]! } - public var WallpaperColors_SetCustomColor: String { return self._s[2691]! } - public var Location_ProximityAlertSetTitle: String { return self._s[2695]! } - public var AuthSessions_AddedDeviceTerminate: String { return self._s[2696]! } - public var Privacy_ProfilePhoto_CustomHelp: String { return self._s[2697]! } - public var Settings_ChangePhoneNumber: String { return self._s[2698]! } - public var PeerInfo_PaneLinks: String { return self._s[2699]! } - public var Appearance_ThemePreview_ChatList_1_Text: String { return self._s[2702]! } - public var Channel_EditAdmin_PermissionInviteSubscribers: String { return self._s[2704]! } + public var Weekday_Tuesday: String { return self._s[2704]! } + public var ChangePhoneNumberCode_Code: String { return self._s[2705]! } + public var VoiceOver_Chat_YourMessage: String { return self._s[2706]! } + public var Calls_CallTabDescription: String { return self._s[2707]! } + public var SocksProxySetup_UseProxy: String { return self._s[2709]! } + public var SettingsSearch_Synonyms_Stickers_Title: String { return self._s[2710]! } + public var PasscodeSettings_AlphanumericCode: String { return self._s[2711]! } + public var VoiceOver_Chat_YourVideo: String { return self._s[2712]! } + public var ChannelMembers_WhoCanAddMembersAdminsHelp: String { return self._s[2714]! } + public var SettingsSearch_Synonyms_Privacy_DeleteAccountIfAwayFor: String { return self._s[2715]! } + public var Exceptions_AddToExceptions: String { return self._s[2716]! } + public var UserInfo_Title: String { return self._s[2717]! } + public var Passport_DeleteDocumentConfirmation: String { return self._s[2719]! } + public var ChatList_Unmute: String { return self._s[2721]! } + public var SettingsSearch_Synonyms_Privacy_Data_ContactsSync: String { return self._s[2722]! } + public var Stats_GroupTopPostersTitle: String { return self._s[2723]! } + public var Username_CheckingUsername: String { return self._s[2724]! } + public var WallpaperColors_SetCustomColor: String { return self._s[2725]! } + public var Location_ProximityAlertSetTitle: String { return self._s[2729]! } + public var AuthSessions_AddedDeviceTerminate: String { return self._s[2730]! } + public var Privacy_ProfilePhoto_CustomHelp: String { return self._s[2731]! } + public var Settings_ChangePhoneNumber: String { return self._s[2732]! } + public var PeerInfo_PaneLinks: String { return self._s[2733]! } + public var Appearance_ThemePreview_ChatList_1_Text: String { return self._s[2736]! } + public var Channel_EditAdmin_PermissionInviteSubscribers: String { return self._s[2738]! } public func PUSH_CHAT_VOICECHAT_INVITE_YOU(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2705]!, self._r[2705]!, [_1]) + return formatWithArgumentRanges(self._s[2739]!, self._r[2739]!, [_1]) } - public var LogoutOptions_ChangePhoneNumberText: String { return self._s[2706]! } - public var VoiceOver_Media_PlaybackPause: String { return self._s[2707]! } - public var Stats_FollowersBySourceTitle: String { return self._s[2709]! } + public var LogoutOptions_ChangePhoneNumberText: String { return self._s[2740]! } + public var VoiceOver_Media_PlaybackPause: String { return self._s[2741]! } + public var Stats_FollowersBySourceTitle: String { return self._s[2743]! } public func Conversation_ScheduleMessage_SendOn(_ _0: String, _ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2710]!, self._r[2710]!, [_0, _1]) + return formatWithArgumentRanges(self._s[2744]!, self._r[2744]!, [_0, _1]) } - public var Compose_NewEncryptedChatTitle: String { return self._s[2711]! } - public var Channel_CommentsGroup_Header: String { return self._s[2713]! } + public var Compose_NewEncryptedChatTitle: String { return self._s[2745]! } + public var Channel_CommentsGroup_Header: String { return self._s[2747]! } public func ShareFileTip_Text(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2717]!, self._r[2717]!, [_0]) + return formatWithArgumentRanges(self._s[2751]!, self._r[2751]!, [_0]) } public func PUSH_MESSAGE_AUDIO(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2718]!, self._r[2718]!, [_1]) + return formatWithArgumentRanges(self._s[2752]!, self._r[2752]!, [_1]) } - public var Group_Setup_BasicHistoryHiddenHelp: String { return self._s[2720]! } + public var Group_Setup_BasicHistoryHiddenHelp: String { return self._s[2754]! } public func TwoStepAuth_RecoveryEmailUnavailable(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2721]!, self._r[2721]!, [_0]) + return formatWithArgumentRanges(self._s[2755]!, self._r[2755]!, [_0]) } - public var Conversation_OpenBotLinkOpen: String { return self._s[2722]! } - public var VoiceOver_Chat_RecordModeVoiceMessage: String { return self._s[2723]! } - public var PrivacySettings_LastSeen: String { return self._s[2725]! } - public var SettingsSearch_Synonyms_Privacy_Passcode: String { return self._s[2726]! } - public var Theme_Colors_Proceed: String { return self._s[2727]! } - public var UserInfo_ScamBotWarning: String { return self._s[2728]! } - public var LogoutOptions_LogOut: String { return self._s[2730]! } - public var Conversation_SendMessage: String { return self._s[2731]! } - public var Passport_Address_Region: String { return self._s[2733]! } - public var MediaPicker_CameraRoll: String { return self._s[2735]! } + public var Conversation_OpenBotLinkOpen: String { return self._s[2756]! } + public var VoiceOver_Chat_RecordModeVoiceMessage: String { return self._s[2757]! } + public var PrivacySettings_LastSeen: String { return self._s[2759]! } + public var SettingsSearch_Synonyms_Privacy_Passcode: String { return self._s[2760]! } + public var Theme_Colors_Proceed: String { return self._s[2761]! } + public var UserInfo_ScamBotWarning: String { return self._s[2762]! } + public var LogoutOptions_LogOut: String { return self._s[2764]! } + public var Conversation_SendMessage: String { return self._s[2765]! } + public var Passport_Address_Region: String { return self._s[2767]! } + public var MediaPicker_CameraRoll: String { return self._s[2769]! } public func VoiceOver_Chat_ForwardedFrom(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2737]!, self._r[2737]!, [_0]) + return formatWithArgumentRanges(self._s[2771]!, self._r[2771]!, [_0]) } - public var Call_ReportSend: String { return self._s[2739]! } - public var Month_ShortJune: String { return self._s[2740]! } - public var AutoDownloadSettings_GroupChats: String { return self._s[2741]! } + public var Call_ReportSend: String { return self._s[2773]! } + public var Month_ShortJune: String { return self._s[2774]! } + public var AutoDownloadSettings_GroupChats: String { return self._s[2775]! } public func Channel_AdminLog_CaptionEdited(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2744]!, self._r[2744]!, [_0]) + return formatWithArgumentRanges(self._s[2778]!, self._r[2778]!, [_0]) } - public var TwoStepAuth_DisableSuccess: String { return self._s[2745]! } - public var Cache_KeepMedia: String { return self._s[2746]! } + public var TwoStepAuth_DisableSuccess: String { return self._s[2779]! } + public var Cache_KeepMedia: String { return self._s[2780]! } public func Date_ChatDateHeaderYear(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2747]!, self._r[2747]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[2781]!, self._r[2781]!, [_1, _2, _3]) } - public var Appearance_LargeEmoji: String { return self._s[2748]! } + public var Appearance_LargeEmoji: String { return self._s[2782]! } public func Notification_NewAuthDetected(_ _1: String, _ _2: String, _ _3: String, _ _4: String, _ _5: String, _ _6: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2749]!, self._r[2749]!, [_1, _2, _3, _4, _5, _6]) + return formatWithArgumentRanges(self._s[2783]!, self._r[2783]!, [_1, _2, _3, _4, _5, _6]) } - public var Chat_AttachmentMultipleForwardDisabled: String { return self._s[2750]! } - public var Call_CameraConfirmationText: String { return self._s[2751]! } + public var Chat_AttachmentMultipleForwardDisabled: String { return self._s[2784]! } + public var Call_CameraConfirmationText: String { return self._s[2785]! } public func AuthSessions_AppUnofficial(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2753]!, self._r[2753]!, [_0]) + return formatWithArgumentRanges(self._s[2787]!, self._r[2787]!, [_0]) } - public var VoiceOver_MessageContextReport: String { return self._s[2755]! } - public var VoiceChat_RemovePeer: String { return self._s[2756]! } - public var ChatListFolder_ExcludeChatsTitle: String { return self._s[2757]! } - public var NotificationsSound_Tritone: String { return self._s[2759]! } - public var Notifications_InAppNotificationsPreview: String { return self._s[2762]! } - public var Stats_GroupTopAdmin_Actions: String { return self._s[2763]! } - public var PeerInfo_AddToContacts: String { return self._s[2764]! } - public var VoiceChat_OpenChat: String { return self._s[2765]! } - public var AccessDenied_Title: String { return self._s[2766]! } - public var Tour_Title1: String { return self._s[2767]! } - public var VoiceOver_AttachMedia: String { return self._s[2768]! } + public var VoiceOver_MessageContextReport: String { return self._s[2789]! } + public var VoiceChat_RemovePeer: String { return self._s[2790]! } + public var ChatListFolder_ExcludeChatsTitle: String { return self._s[2791]! } + public var InviteLink_ContextCopy: String { return self._s[2792]! } + public var NotificationsSound_Tritone: String { return self._s[2794]! } + public var Notifications_InAppNotificationsPreview: String { return self._s[2797]! } + public var Stats_GroupTopAdmin_Actions: String { return self._s[2798]! } + public var PeerInfo_AddToContacts: String { return self._s[2799]! } + public var VoiceChat_OpenChat: String { return self._s[2800]! } + public var AccessDenied_Title: String { return self._s[2801]! } + public var Tour_Title1: String { return self._s[2802]! } + public var VoiceOver_AttachMedia: String { return self._s[2803]! } public func SharedMedia_SearchNoResultsDescription(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2770]!, self._r[2770]!, [_0]) + return formatWithArgumentRanges(self._s[2805]!, self._r[2805]!, [_0]) } - public var Chat_Gifs_SavedSectionHeader: String { return self._s[2771]! } - public var LogoutOptions_ChangePhoneNumberTitle: String { return self._s[2772]! } + public var Chat_Gifs_SavedSectionHeader: String { return self._s[2806]! } + public var LogoutOptions_ChangePhoneNumberTitle: String { return self._s[2807]! } public func Passport_Scans_ScanIndex(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2773]!, self._r[2773]!, [_0]) + return formatWithArgumentRanges(self._s[2808]!, self._r[2808]!, [_0]) } - public var Channel_AdminLog_MessagePreviousLink: String { return self._s[2774]! } - public var OldChannels_Title: String { return self._s[2775]! } - public var LoginPassword_FloodError: String { return self._s[2776]! } - public var Checkout_ErrorPaymentFailed: String { return self._s[2778]! } + public var Channel_AdminLog_MessagePreviousLink: String { return self._s[2809]! } + public var OldChannels_Title: String { return self._s[2810]! } + public var LoginPassword_FloodError: String { return self._s[2811]! } + public var Checkout_ErrorPaymentFailed: String { return self._s[2813]! } public func Time_MonthOfYear_m7(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2779]!, self._r[2779]!, [_0]) + return formatWithArgumentRanges(self._s[2814]!, self._r[2814]!, [_0]) } - public var VoiceOver_Media_PlaybackPlay: String { return self._s[2782]! } - public var Passport_CorrectErrors: String { return self._s[2784]! } + public var VoiceOver_Media_PlaybackPlay: String { return self._s[2817]! } + public var Passport_CorrectErrors: String { return self._s[2819]! } public func PUSH_CHAT_PHOTO_EDITED(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2785]!, self._r[2785]!, [_1, _2]) + return formatWithArgumentRanges(self._s[2820]!, self._r[2820]!, [_1, _2]) } - public var ChatListFolderSettings_Title: String { return self._s[2786]! } + public var ChatListFolderSettings_Title: String { return self._s[2821]! } public func AutoDownloadSettings_UpToFor(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2787]!, self._r[2787]!, [_1, _2]) + return formatWithArgumentRanges(self._s[2822]!, self._r[2822]!, [_1, _2]) } - public var PhotoEditor_HighlightsTool: String { return self._s[2788]! } - public var Contacts_NotRegisteredSection: String { return self._s[2791]! } + public var PhotoEditor_HighlightsTool: String { return self._s[2823]! } + public var Contacts_NotRegisteredSection: String { return self._s[2826]! } public func Call_VoiceChatInProgressCallMessage(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2792]!, self._r[2792]!, [_1, _2]) + return formatWithArgumentRanges(self._s[2827]!, self._r[2827]!, [_1, _2]) } public func PUSH_PINNED_DOC(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2793]!, self._r[2793]!, [_1]) + return formatWithArgumentRanges(self._s[2828]!, self._r[2828]!, [_1]) } - public var User_DeletedAccount: String { return self._s[2794]! } - public var Conversation_ViewContactDetails: String { return self._s[2795]! } - public var Conversation_Dice_u1F3B3: String { return self._s[2796]! } - public var WebSearch_GIFs: String { return self._s[2797]! } - public var ChatList_DeleteSavedMessagesConfirmationAction: String { return self._s[2798]! } - public var Appearance_PreviewOutgoingText: String { return self._s[2799]! } - public var Calls_CallTabTitle: String { return self._s[2800]! } - public var Call_VoiceChatInProgressTitle: String { return self._s[2801]! } + public var InviteLink_Create_UsersLimitInfo: String { return self._s[2829]! } + public var User_DeletedAccount: String { return self._s[2830]! } + public var Conversation_ViewContactDetails: String { return self._s[2831]! } + public var Conversation_Dice_u1F3B3: String { return self._s[2832]! } + public var WebSearch_GIFs: String { return self._s[2833]! } + public var ChatList_DeleteSavedMessagesConfirmationAction: String { return self._s[2834]! } + public var Appearance_PreviewOutgoingText: String { return self._s[2835]! } + public var Calls_CallTabTitle: String { return self._s[2836]! } + public var Call_VoiceChatInProgressTitle: String { return self._s[2837]! } public func LastSeen_YesterdayAt(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2802]!, self._r[2802]!, [_0]) + return formatWithArgumentRanges(self._s[2838]!, self._r[2838]!, [_0]) } - public var Channel_Status: String { return self._s[2803]! } - public var Conversation_SendMessageErrorGroupRestricted: String { return self._s[2805]! } - public var VoiceOver_Chat_OptionSelected: String { return self._s[2806]! } - public var SettingsSearch_Synonyms_Notifications_ChannelNotificationsAlert: String { return self._s[2807]! } + public var Channel_Status: String { return self._s[2839]! } + public var Conversation_SendMessageErrorGroupRestricted: String { return self._s[2841]! } + public var VoiceOver_Chat_OptionSelected: String { return self._s[2842]! } + public var SettingsSearch_Synonyms_Notifications_ChannelNotificationsAlert: String { return self._s[2843]! } public func ClearCache_Success(_ _0: String, _ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2808]!, self._r[2808]!, [_0, _1]) + return formatWithArgumentRanges(self._s[2844]!, self._r[2844]!, [_0, _1]) } - public var Passport_Identity_ExpiryDateNone: String { return self._s[2810]! } - public var Your_cards_expiration_month_is_invalid: String { return self._s[2812]! } - public var Month_ShortDecember: String { return self._s[2813]! } - public var Username_Help: String { return self._s[2814]! } - public var Login_InfoAvatarAdd: String { return self._s[2815]! } - public var Month_ShortMay: String { return self._s[2816]! } - public var DialogList_UnknownPinLimitError: String { return self._s[2817]! } - public var PasscodeSettings_AutoLock_IfAwayFor_5hours: String { return self._s[2818]! } - public var TwoStepAuth_EnabledSuccess: String { return self._s[2819]! } - public var Weekday_ShortSunday: String { return self._s[2820]! } - public var Channel_Username_InvalidTooShort: String { return self._s[2821]! } - public var AuthSessions_TerminateSession: String { return self._s[2822]! } - public var Passport_Identity_FilesTitle: String { return self._s[2823]! } + public var Passport_Identity_ExpiryDateNone: String { return self._s[2846]! } + public var Your_cards_expiration_month_is_invalid: String { return self._s[2848]! } + public var Month_ShortDecember: String { return self._s[2849]! } + public var Username_Help: String { return self._s[2850]! } + public var Login_InfoAvatarAdd: String { return self._s[2851]! } + public var Month_ShortMay: String { return self._s[2852]! } + public var DialogList_UnknownPinLimitError: String { return self._s[2853]! } + public var PasscodeSettings_AutoLock_IfAwayFor_5hours: String { return self._s[2854]! } + public var TwoStepAuth_EnabledSuccess: String { return self._s[2855]! } + public var Weekday_ShortSunday: String { return self._s[2856]! } + public var Channel_Username_InvalidTooShort: String { return self._s[2857]! } + public var AuthSessions_TerminateSession: String { return self._s[2858]! } + public var Passport_Identity_FilesTitle: String { return self._s[2859]! } public func Notification_PinnedRoundMessage(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2824]!, self._r[2824]!, [_0]) + return formatWithArgumentRanges(self._s[2860]!, self._r[2860]!, [_0]) } - public var PeopleNearby_MakeVisible: String { return self._s[2826]! } + public var PeopleNearby_MakeVisible: String { return self._s[2862]! } public func Conversation_RestrictedMediaTimed(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2827]!, self._r[2827]!, [_0]) + return formatWithArgumentRanges(self._s[2863]!, self._r[2863]!, [_0]) } public func Notification_MessageLifetimeChanged(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2828]!, self._r[2828]!, [_1, _2]) + return formatWithArgumentRanges(self._s[2864]!, self._r[2864]!, [_1, _2]) } public func GroupInfo_AddParticipantConfirmation(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2829]!, self._r[2829]!, [_0]) + return formatWithArgumentRanges(self._s[2865]!, self._r[2865]!, [_0]) } - public var PrivacyPolicy_DeclineDeclineAndDelete: String { return self._s[2830]! } - public var Conversation_ContextMenuForward: String { return self._s[2831]! } - public var Channel_AdminLog_CanManageCalls: String { return self._s[2832]! } + public var PrivacyPolicy_DeclineDeclineAndDelete: String { return self._s[2866]! } + public var Conversation_ContextMenuForward: String { return self._s[2867]! } + public var Channel_AdminLog_CanManageCalls: String { return self._s[2868]! } public func PUSH_CHAT_MESSAGE_QUIZ(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2834]!, self._r[2834]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[2870]!, self._r[2870]!, [_1, _2, _3]) } - public var Notification_GroupInviterSelf: String { return self._s[2836]! } - public var Privacy_Forwards_NeverLink: String { return self._s[2837]! } - public var AuthSessions_CurrentSession: String { return self._s[2838]! } - public var Passport_Address_EditPassportRegistration: String { return self._s[2839]! } - public var ChannelInfo_DeleteChannelConfirmation: String { return self._s[2840]! } - public var ChatSearch_ResultsTooltip: String { return self._s[2842]! } - public var CheckoutInfo_Pay: String { return self._s[2843]! } + public var Notification_GroupInviterSelf: String { return self._s[2872]! } + public var Privacy_Forwards_NeverLink: String { return self._s[2873]! } + public var AuthSessions_CurrentSession: String { return self._s[2874]! } + public var Passport_Address_EditPassportRegistration: String { return self._s[2875]! } + public var ChannelInfo_DeleteChannelConfirmation: String { return self._s[2876]! } + public var ChatSearch_ResultsTooltip: String { return self._s[2878]! } + public var CheckoutInfo_Pay: String { return self._s[2879]! } public func Conversation_PinMessagesFor(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2845]!, self._r[2845]!, [_0]) + return formatWithArgumentRanges(self._s[2881]!, self._r[2881]!, [_0]) } - public var GroupInfo_AddParticipant: String { return self._s[2846]! } - public var GroupPermission_ApplyAlertAction: String { return self._s[2847]! } + public var GroupInfo_AddParticipant: String { return self._s[2882]! } + public var GroupPermission_ApplyAlertAction: String { return self._s[2883]! } public func Channel_AdminLog_MessageChangedChannelUsername(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2848]!, self._r[2848]!, [_0]) + return formatWithArgumentRanges(self._s[2884]!, self._r[2884]!, [_0]) } - public var Localization_LanguageCustom: String { return self._s[2849]! } - public var SettingsSearch_Synonyms_Passport: String { return self._s[2850]! } - public var Settings_UsernameEmpty: String { return self._s[2851]! } - public var Settings_FAQ_URL: String { return self._s[2852]! } - public var ChatList_UndoArchiveText1: String { return self._s[2853]! } - public var Common_Select: String { return self._s[2855]! } - public var Notification_MessageLifetimeRemovedOutgoing: String { return self._s[2856]! } - public var Notification_PassportValueAddress: String { return self._s[2857]! } - public var Conversation_MessageDialogDelete: String { return self._s[2858]! } - public var Map_OpenInYandexNavigator: String { return self._s[2860]! } - public var DialogList_SearchSectionDialogs: String { return self._s[2861]! } - public var AccessDenied_Contacts: String { return self._s[2862]! } - public var SettingsSearch_Synonyms_Privacy_Data_DeleteDrafts: String { return self._s[2864]! } - public var Passport_ScanPassportHelp: String { return self._s[2865]! } - public var Chat_PinnedListPreview_HidePinnedMessages: String { return self._s[2866]! } - public var ChatListFolder_NameChannels: String { return self._s[2867]! } - public var Appearance_ThemePreview_Chat_5_Text: String { return self._s[2868]! } + public var Localization_LanguageCustom: String { return self._s[2885]! } + public var SettingsSearch_Synonyms_Passport: String { return self._s[2886]! } + public var Settings_UsernameEmpty: String { return self._s[2887]! } + public var Settings_FAQ_URL: String { return self._s[2888]! } + public var ChatList_UndoArchiveText1: String { return self._s[2889]! } + public var Common_Select: String { return self._s[2891]! } + public var Notification_MessageLifetimeRemovedOutgoing: String { return self._s[2892]! } + public var Notification_PassportValueAddress: String { return self._s[2893]! } + public var Conversation_MessageDialogDelete: String { return self._s[2894]! } + public var Map_OpenInYandexNavigator: String { return self._s[2896]! } + public var DialogList_SearchSectionDialogs: String { return self._s[2897]! } + public var AccessDenied_Contacts: String { return self._s[2898]! } + public var SettingsSearch_Synonyms_Privacy_Data_DeleteDrafts: String { return self._s[2900]! } + public var Passport_ScanPassportHelp: String { return self._s[2901]! } + public var Chat_PinnedListPreview_HidePinnedMessages: String { return self._s[2902]! } + public var ChatListFolder_NameChannels: String { return self._s[2903]! } + public var Appearance_ThemePreview_Chat_5_Text: String { return self._s[2904]! } public func Channel_OwnershipTransfer_TransferCompleted(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2869]!, self._r[2869]!, [_1, _2]) + return formatWithArgumentRanges(self._s[2905]!, self._r[2905]!, [_1, _2]) } - public var Checkout_ErrorInvoiceAlreadyPaid: String { return self._s[2870]! } + public var Checkout_ErrorInvoiceAlreadyPaid: String { return self._s[2906]! } public func VoiceChat_InviteMemberToGroupFirstText(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2871]!, self._r[2871]!, [_1, _2]) + return formatWithArgumentRanges(self._s[2907]!, self._r[2907]!, [_1, _2]) } - public var Conversation_GifTooltip: String { return self._s[2872]! } - public var Passport_Identity_TypeDriversLicenseUploadScan: String { return self._s[2874]! } - public var VoiceChat_Connecting: String { return self._s[2875]! } - public var AutoDownloadSettings_OffForAll: String { return self._s[2876]! } - public var Privacy_GroupsAndChannels_InviteToChannelMultipleError: String { return self._s[2877]! } - public var AutoDownloadSettings_PreloadVideo: String { return self._s[2878]! } - public var CreatePoll_Quiz: String { return self._s[2879]! } - public var TwoFactorSetup_Email_Placeholder: String { return self._s[2880]! } - public var Watch_Message_Invoice: String { return self._s[2881]! } - public var Settings_AddAnotherAccount_Help: String { return self._s[2882]! } - public var Watch_Message_Unsupported: String { return self._s[2883]! } + public var Conversation_GifTooltip: String { return self._s[2908]! } + public var Passport_Identity_TypeDriversLicenseUploadScan: String { return self._s[2910]! } + public var VoiceChat_Connecting: String { return self._s[2911]! } + public var AutoDownloadSettings_OffForAll: String { return self._s[2912]! } + public var Privacy_GroupsAndChannels_InviteToChannelMultipleError: String { return self._s[2913]! } + public var AutoDownloadSettings_PreloadVideo: String { return self._s[2914]! } + public var CreatePoll_Quiz: String { return self._s[2915]! } + public var TwoFactorSetup_Email_Placeholder: String { return self._s[2917]! } + public var Watch_Message_Invoice: String { return self._s[2918]! } + public var Settings_AddAnotherAccount_Help: String { return self._s[2919]! } + public var Watch_Message_Unsupported: String { return self._s[2920]! } public func Call_CameraOff(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2885]!, self._r[2885]!, [_0]) + return formatWithArgumentRanges(self._s[2922]!, self._r[2922]!, [_0]) } - public var AuthSessions_TerminateOtherSessions: String { return self._s[2886]! } - public var CreatePoll_AllOptionsAdded: String { return self._s[2888]! } - public var TwoStepAuth_RecoveryEmailTitle: String { return self._s[2889]! } - public var Call_IncomingVoiceCall: String { return self._s[2890]! } + public var AuthSessions_TerminateOtherSessions: String { return self._s[2923]! } + public var CreatePoll_AllOptionsAdded: String { return self._s[2925]! } + public var TwoStepAuth_RecoveryEmailTitle: String { return self._s[2926]! } + public var Call_IncomingVoiceCall: String { return self._s[2927]! } public func Channel_AdminLog_MessageTransferedNameUsername(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2891]!, self._r[2891]!, [_1, _2]) + return formatWithArgumentRanges(self._s[2928]!, self._r[2928]!, [_1, _2]) } - public var PrivacySettings_DeleteAccountHelp: String { return self._s[2892]! } - public var Passport_Address_TypePassportRegistrationUploadScan: String { return self._s[2893]! } - public var Group_EditAdmin_RankOwnerPlaceholder: String { return self._s[2894]! } - public var Group_ErrorAccessDenied: String { return self._s[2895]! } - public var PasscodeSettings_HelpTop: String { return self._s[2896]! } - public var Watch_ChatList_NoConversationsTitle: String { return self._s[2897]! } - public var AddContact_SharedContactException: String { return self._s[2898]! } - public var AccessDenied_MicrophoneRestricted: String { return self._s[2899]! } - public var Privacy_TopPeers: String { return self._s[2900]! } - public var Web_OpenExternal: String { return self._s[2901]! } - public var Group_ErrorSendRestrictedStickers: String { return self._s[2902]! } - public var Channel_Management_LabelAdministrator: String { return self._s[2903]! } + public var PrivacySettings_DeleteAccountHelp: String { return self._s[2929]! } + public var Passport_Address_TypePassportRegistrationUploadScan: String { return self._s[2930]! } + public var Group_EditAdmin_RankOwnerPlaceholder: String { return self._s[2931]! } + public var Group_ErrorAccessDenied: String { return self._s[2932]! } + public var PasscodeSettings_HelpTop: String { return self._s[2933]! } + public var Watch_ChatList_NoConversationsTitle: String { return self._s[2934]! } + public var AddContact_SharedContactException: String { return self._s[2935]! } + public var AccessDenied_MicrophoneRestricted: String { return self._s[2936]! } + public var Privacy_TopPeers: String { return self._s[2937]! } + public var Web_OpenExternal: String { return self._s[2938]! } + public var Group_ErrorSendRestrictedStickers: String { return self._s[2939]! } + public var Channel_Management_LabelAdministrator: String { return self._s[2940]! } public func ChangePhoneNumberCode_CallTimer(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2904]!, self._r[2904]!, [_0]) + return formatWithArgumentRanges(self._s[2941]!, self._r[2941]!, [_0]) } - public var Permissions_Skip: String { return self._s[2905]! } - public var Notifications_GroupNotificationsExceptions: String { return self._s[2906]! } - public var PeopleNearby_Title: String { return self._s[2907]! } - public var GroupInfo_SharedMediaNone: String { return self._s[2908]! } + public var Permissions_Skip: String { return self._s[2942]! } + public var Notifications_GroupNotificationsExceptions: String { return self._s[2943]! } + public var PeopleNearby_Title: String { return self._s[2944]! } + public var GroupInfo_SharedMediaNone: String { return self._s[2945]! } public func PUSH_MESSAGE_GEOLIVE(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2910]!, self._r[2910]!, [_1]) + return formatWithArgumentRanges(self._s[2947]!, self._r[2947]!, [_1]) } - public var Profile_MessageLifetime1w: String { return self._s[2911]! } + public var Profile_MessageLifetime1w: String { return self._s[2948]! } public func Time_PreciseDate_m6(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2912]!, self._r[2912]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[2949]!, self._r[2949]!, [_1, _2, _3]) } - public var WebBrowser_DefaultBrowser: String { return self._s[2913]! } - public var Conversation_PinOlderMessageAlertTitle: String { return self._s[2915]! } - public var EditTheme_Edit_BottomInfo: String { return self._s[2916]! } - public var Privacy_Forwards_Preview: String { return self._s[2917]! } - public var Settings_EditAccount: String { return self._s[2918]! } + public var WebBrowser_DefaultBrowser: String { return self._s[2950]! } + public var Conversation_PinOlderMessageAlertTitle: String { return self._s[2952]! } + public var EditTheme_Edit_BottomInfo: String { return self._s[2953]! } + public var Privacy_Forwards_Preview: String { return self._s[2954]! } + public var Settings_EditAccount: String { return self._s[2955]! } public func Conversation_RestrictedInlineTimed(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2919]!, self._r[2919]!, [_0]) + return formatWithArgumentRanges(self._s[2956]!, self._r[2956]!, [_0]) } - public var TwoFactorSetup_Intro_Title: String { return self._s[2920]! } + public var TwoFactorSetup_Intro_Title: String { return self._s[2957]! } public func Channel_AdminLog_MessagePromotedName(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2922]!, self._r[2922]!, [_1]) + return formatWithArgumentRanges(self._s[2959]!, self._r[2959]!, [_1]) } - public var PeerInfo_ButtonVideoCall: String { return self._s[2923]! } + public var PeerInfo_ButtonVideoCall: String { return self._s[2960]! } public func DialogList_SingleUploadingPhotoSuffix(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2924]!, self._r[2924]!, [_0]) + return formatWithArgumentRanges(self._s[2961]!, self._r[2961]!, [_0]) } - public var Login_InfoHelp: String { return self._s[2925]! } - public var Notification_SecretChatMessageScreenshotSelf: String { return self._s[2926]! } - public var VoiceChat_SpeakPermissionEveryone: String { return self._s[2927]! } - public var Profile_MessageLifetime1d: String { return self._s[2928]! } - public var Group_UpgradeConfirmation: String { return self._s[2929]! } + public var Login_InfoHelp: String { return self._s[2962]! } + public var Notification_SecretChatMessageScreenshotSelf: String { return self._s[2963]! } + public var VoiceChat_SpeakPermissionEveryone: String { return self._s[2964]! } + public var Profile_MessageLifetime1d: String { return self._s[2965]! } + public var Group_UpgradeConfirmation: String { return self._s[2966]! } public func PUSH_PINNED_STICKER(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2930]!, self._r[2930]!, [_1, _2]) + return formatWithArgumentRanges(self._s[2967]!, self._r[2967]!, [_1, _2]) } - public var Appearance_RemoveThemeColor: String { return self._s[2931]! } - public var Channel_AdminLog_TitleSelectedEvents: String { return self._s[2932]! } + public var Appearance_RemoveThemeColor: String { return self._s[2968]! } + public var Channel_AdminLog_TitleSelectedEvents: String { return self._s[2969]! } public func Call_AnsweringWithAccount(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2933]!, self._r[2933]!, [_0]) + return formatWithArgumentRanges(self._s[2970]!, self._r[2970]!, [_0]) } - public var UserInfo_BotSettings: String { return self._s[2934]! } + public var UserInfo_BotSettings: String { return self._s[2971]! } public func Notification_ChannelInviter(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2936]!, self._r[2936]!, [_0]) + return formatWithArgumentRanges(self._s[2973]!, self._r[2973]!, [_0]) } - public var Permissions_ContactsText_v0: String { return self._s[2937]! } - public var Conversation_PinMessagesForMe: String { return self._s[2938]! } - public var VoiceChat_PanelJoin: String { return self._s[2939]! } - public var Conversation_DiscussionStarted: String { return self._s[2941]! } - public var SettingsSearch_Synonyms_Privacy_TwoStepAuth: String { return self._s[2942]! } - public var SharedMedia_SearchNoResults: String { return self._s[2944]! } + public var Permissions_ContactsText_v0: String { return self._s[2974]! } + public var Conversation_PinMessagesForMe: String { return self._s[2975]! } + public var VoiceChat_PanelJoin: String { return self._s[2976]! } + public var Conversation_DiscussionStarted: String { return self._s[2978]! } + public var SettingsSearch_Synonyms_Privacy_TwoStepAuth: String { return self._s[2979]! } + public var SharedMedia_SearchNoResults: String { return self._s[2981]! } public func Login_EmailPhoneSubject(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2946]!, self._r[2946]!, [_0]) + return formatWithArgumentRanges(self._s[2983]!, self._r[2983]!, [_0]) } public func Conversation_ShareMyPhoneNumber_StatusSuccess(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2948]!, self._r[2948]!, [_0]) + return formatWithArgumentRanges(self._s[2985]!, self._r[2985]!, [_0]) } - public var ReportPeer_ReasonOther_Placeholder: String { return self._s[2949]! } - public var ContactInfo_PhoneLabelHomeFax: String { return self._s[2950]! } - public var Call_AudioRouteHeadphones: String { return self._s[2951]! } + public var ReportPeer_ReasonOther_Placeholder: String { return self._s[2986]! } + public var ContactInfo_PhoneLabelHomeFax: String { return self._s[2987]! } + public var Call_AudioRouteHeadphones: String { return self._s[2988]! } public func PUSH_AUTH_UNKNOWN(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2953]!, self._r[2953]!, [_1]) + return formatWithArgumentRanges(self._s[2990]!, self._r[2990]!, [_1]) } - public var Passport_Identity_FilesView: String { return self._s[2954]! } - public var TwoStepAuth_SetupEmail: String { return self._s[2955]! } - public var Widget_ApplicationStartRequired: String { return self._s[2956]! } - public var PhotoEditor_Original: String { return self._s[2957]! } - public var Call_YourMicrophoneOff: String { return self._s[2958]! } - public var Permissions_ContactsAllow_v0: String { return self._s[2959]! } - public var Notification_Exceptions_PreviewAlwaysOn: String { return self._s[2960]! } - public var PrivacyPolicy_Decline: String { return self._s[2961]! } - public var SettingsSearch_Synonyms_ChatFolders: String { return self._s[2962]! } - public var TwoStepAuth_PasswordRemoveConfirmation: String { return self._s[2963]! } - public var ChatListFolder_IncludeSectionInfo: String { return self._s[2964]! } + public var Passport_Identity_FilesView: String { return self._s[2991]! } + public var TwoStepAuth_SetupEmail: String { return self._s[2992]! } + public var Widget_ApplicationStartRequired: String { return self._s[2993]! } + public var PhotoEditor_Original: String { return self._s[2994]! } + public var Call_YourMicrophoneOff: String { return self._s[2995]! } + public var Permissions_ContactsAllow_v0: String { return self._s[2996]! } + public var Notification_Exceptions_PreviewAlwaysOn: String { return self._s[2997]! } + public var PrivacyPolicy_Decline: String { return self._s[2998]! } + public var SettingsSearch_Synonyms_ChatFolders: String { return self._s[2999]! } + public var TwoStepAuth_PasswordRemoveConfirmation: String { return self._s[3000]! } + public var ChatListFolder_IncludeSectionInfo: String { return self._s[3001]! } public func Map_DirectionsDriveEta(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2965]!, self._r[2965]!, [_0]) + return formatWithArgumentRanges(self._s[3002]!, self._r[3002]!, [_0]) } - public var Passport_Identity_Name: String { return self._s[2966]! } - public var WallpaperPreview_PatternTitle: String { return self._s[2968]! } - public var VoiceOver_Chat_RecordModeVideoMessage: String { return self._s[2969]! } - public var WallpaperSearch_ColorOrange: String { return self._s[2971]! } - public var Appearance_ThemePreview_ChatList_5_Name: String { return self._s[2972]! } - public var GroupInfo_Permissions_SlowmodeInfo: String { return self._s[2973]! } - public var Your_cards_security_code_is_invalid: String { return self._s[2974]! } - public var IntentsSettings_ResetAll: String { return self._s[2975]! } - public var SettingsSearch_Synonyms_Calls_CallTab: String { return self._s[2977]! } - public var Group_EditAdmin_TransferOwnership: String { return self._s[2978]! } - public var Notification_Exceptions_Add: String { return self._s[2979]! } - public var Cache_Help: String { return self._s[2980]! } - public var Call_AudioRouteMute: String { return self._s[2981]! } - public var VoiceOver_Chat_YourVoiceMessage: String { return self._s[2982]! } - public var SocksProxySetup_ProxyEnabled: String { return self._s[2983]! } + public var Passport_Identity_Name: String { return self._s[3003]! } + public var WallpaperPreview_PatternTitle: String { return self._s[3005]! } + public var VoiceOver_Chat_RecordModeVideoMessage: String { return self._s[3006]! } + public var WallpaperSearch_ColorOrange: String { return self._s[3008]! } + public var Appearance_ThemePreview_ChatList_5_Name: String { return self._s[3009]! } + public var GroupInfo_Permissions_SlowmodeInfo: String { return self._s[3010]! } + public var Your_cards_security_code_is_invalid: String { return self._s[3011]! } + public var IntentsSettings_ResetAll: String { return self._s[3012]! } + public var SettingsSearch_Synonyms_Calls_CallTab: String { return self._s[3014]! } + public var Group_EditAdmin_TransferOwnership: String { return self._s[3015]! } + public var Notification_Exceptions_Add: String { return self._s[3016]! } + public var Cache_Help: String { return self._s[3017]! } + public var Call_AudioRouteMute: String { return self._s[3018]! } + public var VoiceOver_Chat_YourVoiceMessage: String { return self._s[3019]! } + public var SocksProxySetup_ProxyEnabled: String { return self._s[3020]! } public func VoiceChat_Status_MembersFormat(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2984]!, self._r[2984]!, [_1, _2]) + return formatWithArgumentRanges(self._s[3021]!, self._r[3021]!, [_1, _2]) } public func ApplyLanguage_UnsufficientDataText(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2985]!, self._r[2985]!, [_1]) + return formatWithArgumentRanges(self._s[3022]!, self._r[3022]!, [_1]) } public func Call_CallInProgressMessage(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2986]!, self._r[2986]!, [_1, _2]) + return formatWithArgumentRanges(self._s[3023]!, self._r[3023]!, [_1, _2]) } - public var AutoDownloadSettings_VideoMessagesTitle: String { return self._s[2987]! } - public var Channel_BanUser_PermissionAddMembers: String { return self._s[2988]! } + public var AutoDownloadSettings_VideoMessagesTitle: String { return self._s[3024]! } + public var Channel_BanUser_PermissionAddMembers: String { return self._s[3025]! } public func PUSH_CHAT_VOICECHAT_INVITE(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2989]!, self._r[2989]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[3026]!, self._r[3026]!, [_1, _2, _3]) } - public var Contacts_MemberSearchSectionTitleGroup: String { return self._s[2990]! } - public var TwoStepAuth_RecoveryCodeHelp: String { return self._s[2991]! } - public var ClearCache_StorageFree: String { return self._s[2992]! } + public var Contacts_MemberSearchSectionTitleGroup: String { return self._s[3027]! } + public var TwoStepAuth_RecoveryCodeHelp: String { return self._s[3028]! } + public var ClearCache_StorageFree: String { return self._s[3029]! } public func DialogList_SingleRecordingVideoMessageSuffix(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2993]!, self._r[2993]!, [_0]) + return formatWithArgumentRanges(self._s[3030]!, self._r[3030]!, [_0]) } - public var Privacy_Forwards_CustomHelp: String { return self._s[2994]! } - public var Group_ErrorAddTooMuchAdmins: String { return self._s[2996]! } - public var DialogList_Typing: String { return self._s[2997]! } + public var Privacy_Forwards_CustomHelp: String { return self._s[3031]! } + public var Group_ErrorAddTooMuchAdmins: String { return self._s[3033]! } + public var DialogList_Typing: String { return self._s[3034]! } public func Login_EmailCodeSubject(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[2998]!, self._r[2998]!, [_0]) + return formatWithArgumentRanges(self._s[3035]!, self._r[3035]!, [_0]) } - public var Target_SelectGroup: String { return self._s[2999]! } - public var AuthSessions_IncompleteAttempts: String { return self._s[3000]! } + public var Target_SelectGroup: String { return self._s[3036]! } + public var AuthSessions_IncompleteAttempts: String { return self._s[3037]! } public func Notification_ProximityReached(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3001]!, self._r[3001]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[3038]!, self._r[3038]!, [_1, _2, _3]) } - public var Chat_PinnedListPreview_ShowAllMessages: String { return self._s[3002]! } - public var TwoStepAuth_EmailChangeSuccess: String { return self._s[3003]! } + public var Chat_PinnedListPreview_ShowAllMessages: String { return self._s[3039]! } + public var TwoStepAuth_EmailChangeSuccess: String { return self._s[3040]! } public func Settings_CheckPhoneNumberTitle(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3004]!, self._r[3004]!, [_0]) + return formatWithArgumentRanges(self._s[3041]!, self._r[3041]!, [_0]) } - public var Channel_AdminLog_CanSendMessages: String { return self._s[3005]! } - public var TwoFactorSetup_EmailVerification_Title: String { return self._s[3006]! } - public var ChatSettings_TextSize: String { return self._s[3007]! } - public var Channel_AdminLogFilter_EventsEditedMessages: String { return self._s[3009]! } - public var Map_SendThisPlace: String { return self._s[3010]! } - public var Conversation_TextCopied: String { return self._s[3011]! } - public var Login_PhoneNumberAlreadyAuthorized: String { return self._s[3012]! } - public var ContactInfo_BirthdayLabel: String { return self._s[3013]! } - public var Call_ShareStats: String { return self._s[3014]! } - public var ChatList_UndoArchiveRevealedText: String { return self._s[3016]! } - public var Notifications_GroupNotificationsPreview: String { return self._s[3017]! } - public var Settings_Support: String { return self._s[3018]! } - public var GroupInfo_ChannelListNamePlaceholder: String { return self._s[3019]! } + public var Channel_AdminLog_CanSendMessages: String { return self._s[3042]! } + public var TwoFactorSetup_EmailVerification_Title: String { return self._s[3043]! } + public var ChatSettings_TextSize: String { return self._s[3044]! } + public var Channel_AdminLogFilter_EventsEditedMessages: String { return self._s[3046]! } + public var Map_SendThisPlace: String { return self._s[3047]! } + public var Conversation_TextCopied: String { return self._s[3048]! } + public var Login_PhoneNumberAlreadyAuthorized: String { return self._s[3049]! } + public var ContactInfo_BirthdayLabel: String { return self._s[3050]! } + public var Call_ShareStats: String { return self._s[3051]! } + public var ChatList_UndoArchiveRevealedText: String { return self._s[3053]! } + public var Notifications_GroupNotificationsPreview: String { return self._s[3054]! } + public var Settings_Support: String { return self._s[3055]! } + public var GroupInfo_ChannelListNamePlaceholder: String { return self._s[3056]! } public func EmptyGroupInfo_Line1(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3021]!, self._r[3021]!, [_0]) + return formatWithArgumentRanges(self._s[3058]!, self._r[3058]!, [_0]) } - public var Watch_Conversation_GroupInfo: String { return self._s[3022]! } - public var Tour_Text4: String { return self._s[3023]! } - public var PasscodeSettings_AutoLock: String { return self._s[3025]! } - public var Channel_BanList_BlockedTitle: String { return self._s[3026]! } - public var Bot_DescriptionTitle: String { return self._s[3027]! } - public var Map_LocationTitle: String { return self._s[3028]! } - public var ChatListFolder_ExcludeSectionInfo: String { return self._s[3029]! } + public var Watch_Conversation_GroupInfo: String { return self._s[3059]! } + public var Tour_Text4: String { return self._s[3060]! } + public var PasscodeSettings_AutoLock: String { return self._s[3062]! } + public var Channel_BanList_BlockedTitle: String { return self._s[3063]! } + public var Bot_DescriptionTitle: String { return self._s[3064]! } + public var Map_LocationTitle: String { return self._s[3065]! } + public var ChatListFolder_ExcludeSectionInfo: String { return self._s[3066]! } public func Notification_MessageLifetimeChangedOutgoing(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3030]!, self._r[3030]!, [_1]) + return formatWithArgumentRanges(self._s[3067]!, self._r[3067]!, [_1]) } - public var Login_EmailNotConfiguredError: String { return self._s[3031]! } - public var AutoDownloadSettings_LimitBySize: String { return self._s[3032]! } - public var PrivacySettings_LastSeenNobody: String { return self._s[3033]! } - public var Permissions_CellularDataText_v0: String { return self._s[3034]! } - public var Conversation_EncryptionProcessing: String { return self._s[3035]! } - public var GroupPermission_Delete: String { return self._s[3036]! } - public var Contacts_SortByName: String { return self._s[3037]! } - public var TwoStepAuth_RecoveryUnavailable: String { return self._s[3038]! } - public var Compose_ChannelTokenListPlaceholder: String { return self._s[3039]! } - public var Group_Management_AddModeratorHelp: String { return self._s[3041]! } - public var SettingsSearch_Synonyms_EditProfile_Logout: String { return self._s[3042]! } - public var Forward_ErrorPublicPollDisabledInChannels: String { return self._s[3043]! } - public var CallFeedback_IncludeLogsInfo: String { return self._s[3045]! } + public var Login_EmailNotConfiguredError: String { return self._s[3068]! } + public var AutoDownloadSettings_LimitBySize: String { return self._s[3069]! } + public var PrivacySettings_LastSeenNobody: String { return self._s[3070]! } + public var Permissions_CellularDataText_v0: String { return self._s[3071]! } + public var Conversation_EncryptionProcessing: String { return self._s[3072]! } + public var GroupPermission_Delete: String { return self._s[3073]! } + public var Contacts_SortByName: String { return self._s[3074]! } + public var TwoStepAuth_RecoveryUnavailable: String { return self._s[3075]! } + public var Compose_ChannelTokenListPlaceholder: String { return self._s[3076]! } + public var Group_Management_AddModeratorHelp: String { return self._s[3078]! } + public var SettingsSearch_Synonyms_EditProfile_Logout: String { return self._s[3079]! } + public var Forward_ErrorPublicPollDisabledInChannels: String { return self._s[3080]! } + public var CallFeedback_IncludeLogsInfo: String { return self._s[3082]! } public func PUSH_CHANNEL_MESSAGE_QUIZ(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3046]!, self._r[3046]!, [_1]) + return formatWithArgumentRanges(self._s[3083]!, self._r[3083]!, [_1]) } public func SecretVideo_NotViewedYet(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3047]!, self._r[3047]!, [_0]) + return formatWithArgumentRanges(self._s[3084]!, self._r[3084]!, [_0]) } - public var ChatList_Context_Delete: String { return self._s[3048]! } - public var VoiceChat_InviteMember: String { return self._s[3049]! } - public var PrivacyPhoneNumberSettings_CustomDisabledHelp: String { return self._s[3050]! } - public var Conversation_Processing: String { return self._s[3051]! } - public var TwoStepAuth_EmailCodeExpired: String { return self._s[3052]! } - public var ChatSettings_Stickers: String { return self._s[3053]! } - public var AppleWatch_ReplyPresetsHelp: String { return self._s[3054]! } - public var Passport_Language_cs: String { return self._s[3055]! } - public var GroupInfo_InvitationLinkGroupFull: String { return self._s[3057]! } - public var Conversation_Contact: String { return self._s[3058]! } - public var Passport_Identity_ReverseSideHelp: String { return self._s[3059]! } - public var SocksProxySetup_PasteFromClipboard: String { return self._s[3060]! } - public var Theme_Unsupported: String { return self._s[3061]! } - public var Privacy_TopPeersWarning: String { return self._s[3062]! } + public var ChatList_Context_Delete: String { return self._s[3085]! } + public var VoiceChat_InviteMember: String { return self._s[3086]! } + public var PrivacyPhoneNumberSettings_CustomDisabledHelp: String { return self._s[3087]! } + public var Conversation_Processing: String { return self._s[3088]! } + public var TwoStepAuth_EmailCodeExpired: String { return self._s[3089]! } + public var ChatSettings_Stickers: String { return self._s[3090]! } + public var AppleWatch_ReplyPresetsHelp: String { return self._s[3091]! } + public var Passport_Language_cs: String { return self._s[3092]! } + public var GroupInfo_InvitationLinkGroupFull: String { return self._s[3094]! } + public var Conversation_Contact: String { return self._s[3095]! } + public var Passport_Identity_ReverseSideHelp: String { return self._s[3096]! } + public var SocksProxySetup_PasteFromClipboard: String { return self._s[3097]! } + public var Theme_Unsupported: String { return self._s[3098]! } + public var Privacy_TopPeersWarning: String { return self._s[3099]! } + public var InviteLink_Title: String { return self._s[3101]! } public func UserInfo_BlockConfirmationTitle(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3064]!, self._r[3064]!, [_0]) + return formatWithArgumentRanges(self._s[3102]!, self._r[3102]!, [_0]) } - public var Conversation_SilentBroadcastTooltipOn: String { return self._s[3065]! } - public var TwoStepAuth_RemovePassword: String { return self._s[3066]! } - public var Settings_CheckPhoneNumberText: String { return self._s[3067]! } - public var PeopleNearby_Users: String { return self._s[3068]! } - public var Appearance_TextSize_UseSystem: String { return self._s[3069]! } - public var Settings_SetProfilePhoto: String { return self._s[3070]! } - public var Conversation_ContextMenuBan: String { return self._s[3071]! } - public var KeyCommand_ScrollUp: String { return self._s[3072]! } - public var Settings_ChatSettings: String { return self._s[3074]! } - public var CallList_RecentCallsHeader: String { return self._s[3075]! } + public var Conversation_SilentBroadcastTooltipOn: String { return self._s[3103]! } + public var TwoStepAuth_RemovePassword: String { return self._s[3104]! } + public var Settings_CheckPhoneNumberText: String { return self._s[3105]! } + public var PeopleNearby_Users: String { return self._s[3106]! } + public var Appearance_TextSize_UseSystem: String { return self._s[3107]! } + public var Settings_SetProfilePhoto: String { return self._s[3108]! } + public var Conversation_ContextMenuBan: String { return self._s[3109]! } + public var KeyCommand_ScrollUp: String { return self._s[3110]! } + public var Settings_ChatSettings: String { return self._s[3112]! } + public var CallList_RecentCallsHeader: String { return self._s[3113]! } public func PUSH_CHAT_MESSAGE_VIDEO(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3076]!, self._r[3076]!, [_1, _2]) + return formatWithArgumentRanges(self._s[3114]!, self._r[3114]!, [_1, _2]) } - public var Stats_GroupTopInvitersTitle: String { return self._s[3077]! } - public var Passport_Phone_EnterOtherNumber: String { return self._s[3078]! } - public var VoiceChat_StartRecordingTitle: String { return self._s[3079]! } - public var Passport_Identity_MiddleNamePlaceholder: String { return self._s[3081]! } - public var Passport_Address_OneOfTypeBankStatement: String { return self._s[3082]! } - public var Stats_GroupTopPoster_Promote: String { return self._s[3083]! } - public var Cache_Title: String { return self._s[3084]! } - public var Clipboard_SendPhoto: String { return self._s[3085]! } - public var Notifications_ExceptionsMessagePlaceholder: String { return self._s[3087]! } - public var TwoStepAuth_EnterPasswordForgot: String { return self._s[3088]! } - public var WatchRemote_AlertTitle: String { return self._s[3089]! } - public var Appearance_ReduceMotion: String { return self._s[3090]! } + public var Stats_GroupTopInvitersTitle: String { return self._s[3115]! } + public var Passport_Phone_EnterOtherNumber: String { return self._s[3116]! } + public var VoiceChat_StartRecordingTitle: String { return self._s[3117]! } + public var Passport_Identity_MiddleNamePlaceholder: String { return self._s[3119]! } + public var Passport_Address_OneOfTypeBankStatement: String { return self._s[3120]! } + public var Stats_GroupTopPoster_Promote: String { return self._s[3121]! } + public var Cache_Title: String { return self._s[3122]! } + public var Clipboard_SendPhoto: String { return self._s[3123]! } + public var Notifications_ExceptionsMessagePlaceholder: String { return self._s[3125]! } + public var TwoStepAuth_EnterPasswordForgot: String { return self._s[3126]! } + public var WatchRemote_AlertTitle: String { return self._s[3127]! } + public var Appearance_ReduceMotion: String { return self._s[3128]! } public func PUSH_CHAT_MESSAGE_ROUND(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3093]!, self._r[3093]!, [_1, _2]) + return formatWithArgumentRanges(self._s[3131]!, self._r[3131]!, [_1, _2]) } - public var Notifications_PermissionsSuppressWarningText: String { return self._s[3094]! } - public var ChatList_UndoArchiveHiddenTitle: String { return self._s[3095]! } - public var Passport_Identity_TypePersonalDetails: String { return self._s[3096]! } + public var Notifications_PermissionsSuppressWarningText: String { return self._s[3132]! } + public var ChatList_UndoArchiveHiddenTitle: String { return self._s[3133]! } + public var Passport_Identity_TypePersonalDetails: String { return self._s[3134]! } public func Call_CallInProgressVoiceChatMessage(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3097]!, self._r[3097]!, [_1, _2]) + return formatWithArgumentRanges(self._s[3135]!, self._r[3135]!, [_1, _2]) } public func Passport_Identity_UploadOneOfScan(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3099]!, self._r[3099]!, [_0]) + return formatWithArgumentRanges(self._s[3137]!, self._r[3137]!, [_0]) } - public var ChatListFolder_DiscardConfirmation: String { return self._s[3100]! } + public var ChatListFolder_DiscardConfirmation: String { return self._s[3138]! } public func Conversation_RestrictedStickersTimed(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3101]!, self._r[3101]!, [_0]) - } - public var ChatState_WaitingForNetwork: String { return self._s[3102]! } - public var GroupInfo_Sound: String { return self._s[3103]! } - public var NotificationsSound_Telegraph: String { return self._s[3104]! } - public var NotificationsSound_Hello: String { return self._s[3105]! } - public var Passport_FieldIdentityDetailsHelp: String { return self._s[3106]! } - public var Group_Members_AddMemberBotErrorNotAllowed: String { return self._s[3107]! } - public var Conversation_HoldForVideo: String { return self._s[3108]! } - public var Conversation_PinOlderMessageAlertText: String { return self._s[3109]! } - public var Appearance_ShareTheme: String { return self._s[3110]! } - public var TwoStepAuth_SetupHint: String { return self._s[3111]! } - public var Stats_GrowthTitle: String { return self._s[3114]! } - public var GroupInfo_InviteLink_ShareLink: String { return self._s[3115]! } - public var Conversation_DefaultRestrictedMedia: String { return self._s[3116]! } - public var Channel_EditAdmin_PermissionPostMessages: String { return self._s[3117]! } - public var GroupPermission_NoSendMessages: String { return self._s[3120]! } - public var Conversation_SetReminder_Title: String { return self._s[3121]! } - public var Privacy_Calls_CustomHelp: String { return self._s[3122]! } - public var CheckoutInfo_ErrorPostcodeInvalid: String { return self._s[3123]! } - public func ClearCache_StorageTitle(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3124]!, self._r[3124]!, [_0]) - } - public var Undo_SecretChatDeleted: String { return self._s[3126]! } - public var PhotoEditor_ContrastTool: String { return self._s[3127]! } - public var Privacy_Forwards: String { return self._s[3128]! } - public var AuthSessions_LoggedInWithTelegram: String { return self._s[3129]! } - public var KeyCommand_SendMessage: String { return self._s[3131]! } - public func InstantPage_RelatedArticleAuthorAndDateTitle(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3132]!, self._r[3132]!, [_1, _2]) - } - public var GroupPermission_NoSendGifs: String { return self._s[3133]! } - public var Notification_MessageLifetime2s: String { return self._s[3134]! } - public var Message_Theme: String { return self._s[3135]! } - public var Conversation_Dice_u1F3AF: String { return self._s[3138]! } - public func DialogList_SinglePlayingGameSuffix(_ _0: String) -> (String, [(Int, NSRange)]) { return formatWithArgumentRanges(self._s[3139]!, self._r[3139]!, [_0]) } - public var Group_UpgradeNoticeHeader: String { return self._s[3141]! } - public var PeerInfo_BioExpand: String { return self._s[3142]! } - public var Passport_DeletePersonalDetails: String { return self._s[3143]! } - public var Widget_NoUsers: String { return self._s[3144]! } - public var TwoStepAuth_AddHintTitle: String { return self._s[3145]! } - public var Login_TermsOfServiceDecline: String { return self._s[3146]! } - public var CreatePoll_QuizTip: String { return self._s[3148]! } - public var Watch_LastSeen_WithinAWeek: String { return self._s[3149]! } - public var MessagePoll_SubmitVote: String { return self._s[3151]! } - public var ChatSettings_AutoDownloadEnabled: String { return self._s[3152]! } - public var Passport_Address_EditRentalAgreement: String { return self._s[3153]! } - public var Conversation_SearchByName_Placeholder: String { return self._s[3154]! } - public var Conversation_UpdateTelegram: String { return self._s[3155]! } - public func FileSize_KB(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3156]!, self._r[3156]!, [_0]) + public var ChatState_WaitingForNetwork: String { return self._s[3140]! } + public var GroupInfo_Sound: String { return self._s[3141]! } + public var NotificationsSound_Telegraph: String { return self._s[3142]! } + public var NotificationsSound_Hello: String { return self._s[3143]! } + public var Passport_FieldIdentityDetailsHelp: String { return self._s[3144]! } + public var Group_Members_AddMemberBotErrorNotAllowed: String { return self._s[3145]! } + public var Conversation_HoldForVideo: String { return self._s[3146]! } + public var Conversation_PinOlderMessageAlertText: String { return self._s[3147]! } + public var Appearance_ShareTheme: String { return self._s[3148]! } + public var TwoStepAuth_SetupHint: String { return self._s[3149]! } + public var Stats_GrowthTitle: String { return self._s[3152]! } + public var GroupInfo_InviteLink_ShareLink: String { return self._s[3153]! } + public var Conversation_DefaultRestrictedMedia: String { return self._s[3154]! } + public var Channel_EditAdmin_PermissionPostMessages: String { return self._s[3155]! } + public var GroupPermission_NoSendMessages: String { return self._s[3158]! } + public var Conversation_SetReminder_Title: String { return self._s[3159]! } + public var Privacy_Calls_CustomHelp: String { return self._s[3160]! } + public var CheckoutInfo_ErrorPostcodeInvalid: String { return self._s[3161]! } + public func ClearCache_StorageTitle(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3162]!, self._r[3162]!, [_0]) } - public var UserInfo_About_Placeholder: String { return self._s[3157]! } - public var CallSettings_Always: String { return self._s[3158]! } - public var ChannelInfo_ScamChannelWarning: String { return self._s[3159]! } - public var Login_TermsOfServiceHeader: String { return self._s[3160]! } - public var KeyCommand_ChatInfo: String { return self._s[3161]! } - public var MessagePoll_LabelPoll: String { return self._s[3162]! } - public var Paint_Clear: String { return self._s[3163]! } - public var PeerInfo_ButtonMute: String { return self._s[3164]! } - public var LastSeen_WithinAWeek: String { return self._s[3165]! } - public var Passport_Identity_FrontSide: String { return self._s[3166]! } - public var Stickers_GroupStickers: String { return self._s[3167]! } - public var ChangePhoneNumberNumber_NumberPlaceholder: String { return self._s[3168]! } + public var Undo_SecretChatDeleted: String { return self._s[3164]! } + public var PhotoEditor_ContrastTool: String { return self._s[3165]! } + public var Privacy_Forwards: String { return self._s[3166]! } + public var AuthSessions_LoggedInWithTelegram: String { return self._s[3167]! } + public var KeyCommand_SendMessage: String { return self._s[3169]! } + public func InstantPage_RelatedArticleAuthorAndDateTitle(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3170]!, self._r[3170]!, [_1, _2]) + } + public var GroupPermission_NoSendGifs: String { return self._s[3171]! } + public var Notification_MessageLifetime2s: String { return self._s[3172]! } + public var Message_Theme: String { return self._s[3173]! } + public var Conversation_Dice_u1F3AF: String { return self._s[3176]! } + public func DialogList_SinglePlayingGameSuffix(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3177]!, self._r[3177]!, [_0]) + } + public var Group_UpgradeNoticeHeader: String { return self._s[3179]! } + public var PeerInfo_BioExpand: String { return self._s[3180]! } + public var Passport_DeletePersonalDetails: String { return self._s[3181]! } + public var Widget_NoUsers: String { return self._s[3182]! } + public var TwoStepAuth_AddHintTitle: String { return self._s[3183]! } + public var Login_TermsOfServiceDecline: String { return self._s[3184]! } + public var CreatePoll_QuizTip: String { return self._s[3186]! } + public var Watch_LastSeen_WithinAWeek: String { return self._s[3187]! } + public var MessagePoll_SubmitVote: String { return self._s[3189]! } + public var ChatSettings_AutoDownloadEnabled: String { return self._s[3190]! } + public var Passport_Address_EditRentalAgreement: String { return self._s[3191]! } + public var Conversation_SearchByName_Placeholder: String { return self._s[3192]! } + public var Conversation_UpdateTelegram: String { return self._s[3193]! } + public func FileSize_KB(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3194]!, self._r[3194]!, [_0]) + } + public var UserInfo_About_Placeholder: String { return self._s[3195]! } + public var CallSettings_Always: String { return self._s[3196]! } + public var ChannelInfo_ScamChannelWarning: String { return self._s[3197]! } + public var Login_TermsOfServiceHeader: String { return self._s[3198]! } + public var KeyCommand_ChatInfo: String { return self._s[3199]! } + public var MessagePoll_LabelPoll: String { return self._s[3200]! } + public var Paint_Clear: String { return self._s[3201]! } + public var PeerInfo_ButtonMute: String { return self._s[3202]! } + public var LastSeen_WithinAWeek: String { return self._s[3203]! } + public var Passport_Identity_FrontSide: String { return self._s[3204]! } + public var Stickers_GroupStickers: String { return self._s[3205]! } + public var ChangePhoneNumberNumber_NumberPlaceholder: String { return self._s[3206]! } public func Map_SearchNoResultsDescription(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3169]!, self._r[3169]!, [_0]) + return formatWithArgumentRanges(self._s[3207]!, self._r[3207]!, [_0]) } public func PUSH_MESSAGE_GEO(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3172]!, self._r[3172]!, [_1]) + return formatWithArgumentRanges(self._s[3210]!, self._r[3210]!, [_1]) } - public var SocksProxySetup_ProxyStatusConnected: String { return self._s[3173]! } - public var Chat_MultipleTextMessagesDisabled: String { return self._s[3174]! } + public var SocksProxySetup_ProxyStatusConnected: String { return self._s[3211]! } + public var Chat_MultipleTextMessagesDisabled: String { return self._s[3212]! } + public var InviteLink_ContextDelete: String { return self._s[3213]! } public func Notification_LeftChat(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3175]!, self._r[3175]!, [_0]) + return formatWithArgumentRanges(self._s[3214]!, self._r[3214]!, [_0]) } - public var WebSearch_SearchNoResults: String { return self._s[3177]! } - public var Channel_DiscussionGroup_Create: String { return self._s[3178]! } - public var Passport_Language_es: String { return self._s[3179]! } - public var EnterPasscode_EnterCurrentPasscode: String { return self._s[3180]! } - public var Map_LiveLocationShowAll: String { return self._s[3181]! } - public var Cache_MaximumCacheSizeHelp: String { return self._s[3183]! } - public var Map_OpenInGoogleMaps: String { return self._s[3184]! } - public var CheckoutInfo_ErrorNameInvalid: String { return self._s[3186]! } - public var EditTheme_Create_BottomInfo: String { return self._s[3187]! } - public var PhotoEditor_BlurToolLinear: String { return self._s[3188]! } + public var WebSearch_SearchNoResults: String { return self._s[3216]! } + public var Channel_DiscussionGroup_Create: String { return self._s[3217]! } + public var Passport_Language_es: String { return self._s[3218]! } + public var EnterPasscode_EnterCurrentPasscode: String { return self._s[3219]! } + public var Map_LiveLocationShowAll: String { return self._s[3220]! } + public var Cache_MaximumCacheSizeHelp: String { return self._s[3222]! } + public var Map_OpenInGoogleMaps: String { return self._s[3223]! } + public var CheckoutInfo_ErrorNameInvalid: String { return self._s[3225]! } + public var EditTheme_Create_BottomInfo: String { return self._s[3226]! } + public var PhotoEditor_BlurToolLinear: String { return self._s[3227]! } public func Channel_AdminLog_MessageEdited(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3189]!, self._r[3189]!, [_0]) + return formatWithArgumentRanges(self._s[3228]!, self._r[3228]!, [_0]) } - public var Passport_Phone_Delete: String { return self._s[3190]! } - public var Channel_Username_CreatePrivateLinkHelp: String { return self._s[3191]! } - public var PrivacySettings_PrivacyTitle: String { return self._s[3192]! } - public var CheckoutInfo_ReceiverInfoNamePlaceholder: String { return self._s[3193]! } + public var Passport_Phone_Delete: String { return self._s[3229]! } + public var Channel_Username_CreatePrivateLinkHelp: String { return self._s[3230]! } + public var PrivacySettings_PrivacyTitle: String { return self._s[3231]! } + public var CheckoutInfo_ReceiverInfoNamePlaceholder: String { return self._s[3232]! } public func EncryptionKey_Description(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3194]!, self._r[3194]!, [_1, _2]) + return formatWithArgumentRanges(self._s[3233]!, self._r[3233]!, [_1, _2]) } - public var LogoutOptions_LogOutInfo: String { return self._s[3195]! } - public var Cache_ByPeerHeader: String { return self._s[3197]! } - public var Username_InvalidCharacters: String { return self._s[3198]! } - public var Checkout_ShippingAddress: String { return self._s[3199]! } + public var LogoutOptions_LogOutInfo: String { return self._s[3234]! } + public var Cache_ByPeerHeader: String { return self._s[3236]! } + public var Username_InvalidCharacters: String { return self._s[3237]! } + public var Checkout_ShippingAddress: String { return self._s[3238]! } public func PUSH_CHAT_MESSAGE_GAME_SCORE(_ _1: String, _ _2: String, _ _3: String, _ _4: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3200]!, self._r[3200]!, [_1, _2, _3, _4]) + return formatWithArgumentRanges(self._s[3239]!, self._r[3239]!, [_1, _2, _3, _4]) } - public var Conversation_AddContact: String { return self._s[3202]! } - public var Passport_Address_EditUtilityBill: String { return self._s[3203]! } - public var Message_Video: String { return self._s[3204]! } + public var Conversation_AddContact: String { return self._s[3241]! } + public var Passport_Address_EditUtilityBill: String { return self._s[3242]! } + public var InviteLink_ContextGetQRCode: String { return self._s[3243]! } + public var Conversation_ChecksTooltip_Delivered: String { return self._s[3244]! } + public var Message_Video: String { return self._s[3245]! } public func Watch_Time_ShortYesterdayAt(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3205]!, self._r[3205]!, [_0]) + return formatWithArgumentRanges(self._s[3246]!, self._r[3246]!, [_0]) } public func Conversation_Megabytes(_ _0: Float) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3206]!, self._r[3206]!, ["\(_0)"]) + return formatWithArgumentRanges(self._s[3247]!, self._r[3247]!, ["\(_0)"]) } - public var Passport_Language_km: String { return self._s[3207]! } + public var Passport_Language_km: String { return self._s[3248]! } public func PUSH_MESSAGE_CHANNEL_MESSAGE_GAME_SCORE(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3208]!, self._r[3208]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[3249]!, self._r[3249]!, [_1, _2, _3]) } - public var EmptyGroupInfo_Line4: String { return self._s[3209]! } - public var Conversation_SendMessageErrorTooMuchScheduled: String { return self._s[3211]! } - public var Notification_CallCanceledShort: String { return self._s[3212]! } - public var PhotoEditor_FadeTool: String { return self._s[3213]! } - public var Group_PublicLink_Info: String { return self._s[3214]! } - public var Contacts_DeselectAll: String { return self._s[3215]! } - public var Conversation_Moderate_Delete: String { return self._s[3216]! } - public var TwoStepAuth_RecoveryCodeInvalid: String { return self._s[3217]! } - public var NotificationsSound_Note: String { return self._s[3220]! } + public var EmptyGroupInfo_Line4: String { return self._s[3250]! } + public var Conversation_SendMessageErrorTooMuchScheduled: String { return self._s[3252]! } + public var Notification_CallCanceledShort: String { return self._s[3253]! } + public var PhotoEditor_FadeTool: String { return self._s[3254]! } + public var Group_PublicLink_Info: String { return self._s[3255]! } + public var Contacts_DeselectAll: String { return self._s[3256]! } + public var Conversation_Moderate_Delete: String { return self._s[3257]! } + public var TwoStepAuth_RecoveryCodeInvalid: String { return self._s[3258]! } + public var NotificationsSound_Note: String { return self._s[3261]! } public func Message_PaymentSent(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3221]!, self._r[3221]!, [_0]) + return formatWithArgumentRanges(self._s[3262]!, self._r[3262]!, [_0]) } - public var Appearance_ThemePreview_ChatList_7_Text: String { return self._s[3222]! } - public var Channel_EditAdmin_PermissionInviteViaLink: String { return self._s[3224]! } - public var DialogList_SearchSectionGlobal: String { return self._s[3225]! } - public var AccessDenied_Settings: String { return self._s[3226]! } - public var Passport_Identity_TypeIdentityCardUploadScan: String { return self._s[3227]! } - public var AuthSessions_EmptyTitle: String { return self._s[3228]! } - public var TwoStepAuth_PasswordChangeSuccess: String { return self._s[3229]! } - public var GroupInfo_GroupType: String { return self._s[3230]! } - public var Calls_Missed: String { return self._s[3231]! } - public var UserInfo_GenericPhoneLabel: String { return self._s[3232]! } - public var Passport_Language_uz: String { return self._s[3233]! } - public var Conversation_StopQuizConfirmationTitle: String { return self._s[3234]! } - public var PhotoEditor_BlurToolPortrait: String { return self._s[3235]! } - public var Map_ChooseLocationTitle: String { return self._s[3236]! } - public var Checkout_EnterPassword: String { return self._s[3237]! } - public var GroupInfo_ConvertToSupergroup: String { return self._s[3238]! } - public var AutoNightTheme_UpdateLocation: String { return self._s[3239]! } - public var NetworkUsageSettings_Title: String { return self._s[3240]! } - public var Location_ProximityAlertCancelled: String { return self._s[3241]! } - public var SettingsSearch_Synonyms_ChatSettings_IntentsSettings: String { return self._s[3242]! } - public var Message_PinnedLiveLocationMessage: String { return self._s[3243]! } - public var Compose_NewChannel: String { return self._s[3244]! } - public var Privacy_PaymentsClearInfo: String { return self._s[3246]! } + public var Appearance_ThemePreview_ChatList_7_Text: String { return self._s[3263]! } + public var Channel_EditAdmin_PermissionInviteViaLink: String { return self._s[3265]! } + public var DialogList_SearchSectionGlobal: String { return self._s[3266]! } + public var AccessDenied_Settings: String { return self._s[3267]! } + public var Passport_Identity_TypeIdentityCardUploadScan: String { return self._s[3268]! } + public var AuthSessions_EmptyTitle: String { return self._s[3269]! } + public var TwoStepAuth_PasswordChangeSuccess: String { return self._s[3270]! } + public var GroupInfo_GroupType: String { return self._s[3271]! } + public var Calls_Missed: String { return self._s[3272]! } + public var UserInfo_GenericPhoneLabel: String { return self._s[3273]! } + public var Passport_Language_uz: String { return self._s[3274]! } + public var Conversation_StopQuizConfirmationTitle: String { return self._s[3275]! } + public var PhotoEditor_BlurToolPortrait: String { return self._s[3276]! } + public var Map_ChooseLocationTitle: String { return self._s[3277]! } + public var Checkout_EnterPassword: String { return self._s[3278]! } + public var GroupInfo_ConvertToSupergroup: String { return self._s[3279]! } + public var AutoNightTheme_UpdateLocation: String { return self._s[3280]! } + public var NetworkUsageSettings_Title: String { return self._s[3281]! } + public var Location_ProximityAlertCancelled: String { return self._s[3282]! } + public var SettingsSearch_Synonyms_ChatSettings_IntentsSettings: String { return self._s[3283]! } + public var Message_PinnedLiveLocationMessage: String { return self._s[3284]! } + public var Compose_NewChannel: String { return self._s[3285]! } + public var Privacy_PaymentsClearInfo: String { return self._s[3287]! } public func PUSH_MESSAGE_POLL(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3247]!, self._r[3247]!, [_1]) + return formatWithArgumentRanges(self._s[3288]!, self._r[3288]!, [_1]) } - public var Notification_Exceptions_AlwaysOn: String { return self._s[3248]! } - public var Privacy_GroupsAndChannels_WhoCanAddMe: String { return self._s[3249]! } - public var AutoNightTheme_AutomaticSection: String { return self._s[3252]! } - public var WallpaperSearch_ColorBrown: String { return self._s[3253]! } - public var Appearance_AppIconDefault: String { return self._s[3254]! } - public var StickerSettings_ContextInfo: String { return self._s[3257]! } - public var Channel_AddBotErrorNoRights: String { return self._s[3258]! } - public var Passport_FieldPhone: String { return self._s[3260]! } - public var Contacts_PermissionsTitle: String { return self._s[3261]! } - public var TwoFactorSetup_Email_SkipConfirmationSkip: String { return self._s[3262]! } + public var Notification_Exceptions_AlwaysOn: String { return self._s[3289]! } + public var Privacy_GroupsAndChannels_WhoCanAddMe: String { return self._s[3290]! } + public var AutoNightTheme_AutomaticSection: String { return self._s[3293]! } + public var WallpaperSearch_ColorBrown: String { return self._s[3294]! } + public var Appearance_AppIconDefault: String { return self._s[3295]! } + public var StickerSettings_ContextInfo: String { return self._s[3298]! } + public var Channel_AddBotErrorNoRights: String { return self._s[3299]! } + public var Passport_FieldPhone: String { return self._s[3301]! } + public var Contacts_PermissionsTitle: String { return self._s[3302]! } + public var TwoFactorSetup_Email_SkipConfirmationSkip: String { return self._s[3303]! } public func Notification_JoinedChat(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3263]!, self._r[3263]!, [_0]) + return formatWithArgumentRanges(self._s[3304]!, self._r[3304]!, [_0]) } - public var Bot_Unblock: String { return self._s[3264]! } - public var PasscodeSettings_SimplePasscode: String { return self._s[3265]! } - public var Passport_PasswordHelp: String { return self._s[3266]! } - public var Watch_Conversation_UserInfo: String { return self._s[3267]! } + public var Bot_Unblock: String { return self._s[3305]! } + public var PasscodeSettings_SimplePasscode: String { return self._s[3306]! } + public var Passport_PasswordHelp: String { return self._s[3307]! } + public var Watch_Conversation_UserInfo: String { return self._s[3308]! } public func Channel_AdminLog_MessageChangedGroupGeoLocation(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3271]!, self._r[3271]!, [_0]) + return formatWithArgumentRanges(self._s[3312]!, self._r[3312]!, [_0]) } - public var State_Connecting: String { return self._s[3273]! } - public var Passport_Address_TypeTemporaryRegistration: String { return self._s[3274]! } - public var TextFormat_AddLinkPlaceholder: String { return self._s[3275]! } - public var Conversation_Dice_u1F3B2: String { return self._s[3276]! } + public var State_Connecting: String { return self._s[3314]! } + public var Passport_Address_TypeTemporaryRegistration: String { return self._s[3315]! } + public var TextFormat_AddLinkPlaceholder: String { return self._s[3316]! } + public var Conversation_Dice_u1F3B2: String { return self._s[3317]! } public func Call_StatusBar(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3277]!, self._r[3277]!, [_0]) + return formatWithArgumentRanges(self._s[3318]!, self._r[3318]!, [_0]) } - public var Conversation_SendingOptionsTooltip: String { return self._s[3278]! } - public var ChatList_UndoArchiveTitle: String { return self._s[3279]! } - public var ChatList_EmptyChatListNewMessage: String { return self._s[3280]! } - public var WallpaperSearch_ColorGreen: String { return self._s[3282]! } - public var PhotoEditor_BlurToolOff: String { return self._s[3283]! } - public var SocksProxySetup_PortPlaceholder: String { return self._s[3284]! } - public var Weekday_Saturday: String { return self._s[3285]! } - public var DialogList_Unread: String { return self._s[3286]! } - public var Watch_LastSeen_ALongTimeAgo: String { return self._s[3287]! } - public var Stats_GroupPosters: String { return self._s[3288]! } + public var Conversation_SendingOptionsTooltip: String { return self._s[3319]! } + public var ChatList_UndoArchiveTitle: String { return self._s[3320]! } + public var ChatList_EmptyChatListNewMessage: String { return self._s[3321]! } + public var WallpaperSearch_ColorGreen: String { return self._s[3323]! } + public var PhotoEditor_BlurToolOff: String { return self._s[3324]! } + public var SocksProxySetup_PortPlaceholder: String { return self._s[3325]! } + public var Weekday_Saturday: String { return self._s[3326]! } + public var DialogList_Unread: String { return self._s[3327]! } + public var Watch_LastSeen_ALongTimeAgo: String { return self._s[3328]! } + public var Stats_GroupPosters: String { return self._s[3329]! } public func PUSH_ENCRYPTION_REQUEST(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3289]!, self._r[3289]!, [_1]) + return formatWithArgumentRanges(self._s[3330]!, self._r[3330]!, [_1]) } public func Target_ShareGameConfirmationGroup(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3292]!, self._r[3292]!, [_0]) + return formatWithArgumentRanges(self._s[3333]!, self._r[3333]!, [_0]) } - public var ReportPeer_ReasonChildAbuse: String { return self._s[3293]! } + public var ReportPeer_ReasonChildAbuse: String { return self._s[3334]! } public func Channel_AdminLog_MessageUnkickedNameUsername(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3294]!, self._r[3294]!, [_1, _2]) + return formatWithArgumentRanges(self._s[3335]!, self._r[3335]!, [_1, _2]) } - public var InfoPlist_NSContactsUsageDescription: String { return self._s[3295]! } - public var AutoNightTheme_UseSunsetSunrise: String { return self._s[3297]! } - public var Channel_OwnershipTransfer_ChangeOwner: String { return self._s[3298]! } - public var Call_VoiceOver_VoiceCallCanceled: String { return self._s[3299]! } - public var Passport_Language_dv: String { return self._s[3300]! } - public var GroupPermission_AddSuccess: String { return self._s[3302]! } - public var Passport_Email_Help: String { return self._s[3303]! } - public var Call_ReportPlaceholder: String { return self._s[3304]! } - public var CreatePoll_AddOption: String { return self._s[3305]! } - public var MessagePoll_LabelAnonymousQuiz: String { return self._s[3306]! } - public var PeerInfo_ButtonLeave: String { return self._s[3307]! } - public var PhotoEditor_TiltShift: String { return self._s[3310]! } - public var SecretGif_Title: String { return self._s[3312]! } - public var PhotoEditor_QualityVeryLow: String { return self._s[3313]! } - public var SocksProxySetup_Connecting: String { return self._s[3314]! } - public var PrivacySettings_PasscodeAndFaceId: String { return self._s[3315]! } - public var ContactInfo_PhoneLabelWork: String { return self._s[3316]! } - public var Stats_GroupTopHoursTitle: String { return self._s[3317]! } - public var Compose_NewMessage: String { return self._s[3318]! } - public var VoiceOver_Common_SwitchHint: String { return self._s[3319]! } - public var NotificationsSound_Synth: String { return self._s[3320]! } - public var Conversation_FileOpenIn: String { return self._s[3321]! } - public var AutoDownloadSettings_WifiTitle: String { return self._s[3322]! } - public var UserInfo_SendMessage: String { return self._s[3323]! } - public var Checkout_PayWithFaceId: String { return self._s[3324]! } + public var InfoPlist_NSContactsUsageDescription: String { return self._s[3336]! } + public var AutoNightTheme_UseSunsetSunrise: String { return self._s[3338]! } + public var Channel_OwnershipTransfer_ChangeOwner: String { return self._s[3339]! } + public var Call_VoiceOver_VoiceCallCanceled: String { return self._s[3340]! } + public var Passport_Language_dv: String { return self._s[3341]! } + public var GroupPermission_AddSuccess: String { return self._s[3343]! } + public var Passport_Email_Help: String { return self._s[3344]! } + public var Call_ReportPlaceholder: String { return self._s[3345]! } + public var CreatePoll_AddOption: String { return self._s[3346]! } + public var MessagePoll_LabelAnonymousQuiz: String { return self._s[3348]! } + public var PeerInfo_ButtonLeave: String { return self._s[3349]! } + public var PhotoEditor_TiltShift: String { return self._s[3352]! } + public var SecretGif_Title: String { return self._s[3354]! } + public var GroupInfo_InviteLinks: String { return self._s[3355]! } + public var PhotoEditor_QualityVeryLow: String { return self._s[3356]! } + public var SocksProxySetup_Connecting: String { return self._s[3357]! } + public var PrivacySettings_PasscodeAndFaceId: String { return self._s[3358]! } + public var ContactInfo_PhoneLabelWork: String { return self._s[3359]! } + public var Stats_GroupTopHoursTitle: String { return self._s[3360]! } + public var Compose_NewMessage: String { return self._s[3361]! } + public var VoiceOver_Common_SwitchHint: String { return self._s[3362]! } + public var NotificationsSound_Synth: String { return self._s[3363]! } + public var Conversation_FileOpenIn: String { return self._s[3364]! } + public var AutoDownloadSettings_WifiTitle: String { return self._s[3365]! } + public var UserInfo_SendMessage: String { return self._s[3366]! } + public var Checkout_PayWithFaceId: String { return self._s[3367]! } public func Map_LiveLocationShortHour(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3325]!, self._r[3325]!, [_0]) + return formatWithArgumentRanges(self._s[3368]!, self._r[3368]!, [_0]) } - public var TextFormat_Strikethrough: String { return self._s[3326]! } - public var SettingsSearch_Synonyms_Notifications_DisplayNamesOnLockScreen: String { return self._s[3327]! } - public var Conversation_ViewChannel: String { return self._s[3328]! } + public var TextFormat_Strikethrough: String { return self._s[3369]! } + public var SettingsSearch_Synonyms_Notifications_DisplayNamesOnLockScreen: String { return self._s[3370]! } + public var Conversation_ViewChannel: String { return self._s[3371]! } public func Message_ForwardedMessage(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3329]!, self._r[3329]!, [_0]) + return formatWithArgumentRanges(self._s[3372]!, self._r[3372]!, [_0]) } - public var Channel_Stickers_Placeholder: String { return self._s[3330]! } - public var Channel_OwnershipTransfer_PasswordPlaceholder: String { return self._s[3331]! } - public var Camera_FlashAuto: String { return self._s[3332]! } - public var Conversation_EncryptedDescription1: String { return self._s[3333]! } - public var LocalGroup_Text: String { return self._s[3334]! } - public var SettingsSearch_Synonyms_Data_Storage_KeepMedia: String { return self._s[3335]! } - public var UserInfo_FirstNamePlaceholder: String { return self._s[3336]! } - public var Conversation_SendMessageErrorFlood: String { return self._s[3337]! } - public var Conversation_EncryptedDescription2: String { return self._s[3338]! } - public var Notification_GroupActivated: String { return self._s[3339]! } - public var LastSeen_Lately: String { return self._s[3340]! } - public var Conversation_EncryptedDescription3: String { return self._s[3341]! } - public var SettingsSearch_Synonyms_Privacy_ProfilePhoto: String { return self._s[3342]! } - public var Conversation_SwipeToReplyHintText: String { return self._s[3343]! } - public var Conversation_EncryptedDescription4: String { return self._s[3344]! } - public var SharedMedia_EmptyTitle: String { return self._s[3345]! } - public var Appearance_CreateTheme: String { return self._s[3346]! } - public var Stats_SharesPerPost: String { return self._s[3347]! } - public var Contacts_TabTitle: String { return self._s[3348]! } - public var Weekday_ShortThursday: String { return self._s[3349]! } - public var MessageTimer_Forever: String { return self._s[3350]! } - public var ChatListFolder_CategoryArchived: String { return self._s[3351]! } - public var Channel_EditAdmin_PermissionDeleteMessages: String { return self._s[3352]! } - public var EditTheme_Create_TopInfo: String { return self._s[3354]! } - public var Month_GenDecember: String { return self._s[3355]! } - public var EnterPasscode_EnterPasscode: String { return self._s[3356]! } - public var SettingsSearch_Synonyms_Appearance_LargeEmoji: String { return self._s[3357]! } - public var PeopleNearby_CreateGroup: String { return self._s[3359]! } - public var Group_EditAdmin_PermissionChangeInfo: String { return self._s[3360]! } - public var Paint_ClearConfirm: String { return self._s[3361]! } - public var ChatList_ReadAll: String { return self._s[3362]! } - public var ChatSettings_IntentsSettings: String { return self._s[3363]! } - public var Passport_PassportInformation: String { return self._s[3365]! } - public var Login_CheckOtherSessionMessages: String { return self._s[3367]! } - public var Location_ProximityNotification_DistanceMI: String { return self._s[3370]! } - public var PhotoEditor_ExposureTool: String { return self._s[3371]! } - public var Group_Username_CreatePrivateLinkHelp: String { return self._s[3372]! } - public var SettingsSearch_Synonyms_Watch: String { return self._s[3373]! } - public var Stats_GroupTopPoster_History: String { return self._s[3374]! } - public var UserInfo_AddPhone: String { return self._s[3375]! } - public var Media_SendWithTimer: String { return self._s[3377]! } - public var SettingsSearch_Synonyms_Notifications_Title: String { return self._s[3378]! } - public var Channel_EditAdmin_PermissionEnabledByDefault: String { return self._s[3379]! } - public var PasscodeSettings_AutoLock_Disabled: String { return self._s[3380]! } - public var ChatList_Context_Unarchive: String { return self._s[3382]! } + public var Channel_Stickers_Placeholder: String { return self._s[3373]! } + public var Channel_OwnershipTransfer_PasswordPlaceholder: String { return self._s[3374]! } + public var Camera_FlashAuto: String { return self._s[3375]! } + public var Conversation_EncryptedDescription1: String { return self._s[3376]! } + public var LocalGroup_Text: String { return self._s[3377]! } + public var SettingsSearch_Synonyms_Data_Storage_KeepMedia: String { return self._s[3378]! } + public var UserInfo_FirstNamePlaceholder: String { return self._s[3379]! } + public var Conversation_SendMessageErrorFlood: String { return self._s[3380]! } + public var Conversation_EncryptedDescription2: String { return self._s[3381]! } + public var Notification_GroupActivated: String { return self._s[3382]! } + public var LastSeen_Lately: String { return self._s[3383]! } + public var Conversation_EncryptedDescription3: String { return self._s[3384]! } + public var SettingsSearch_Synonyms_Privacy_ProfilePhoto: String { return self._s[3385]! } + public var Conversation_SwipeToReplyHintText: String { return self._s[3386]! } + public var Conversation_EncryptedDescription4: String { return self._s[3387]! } + public var SharedMedia_EmptyTitle: String { return self._s[3388]! } + public var Appearance_CreateTheme: String { return self._s[3389]! } + public var Stats_SharesPerPost: String { return self._s[3390]! } + public var Contacts_TabTitle: String { return self._s[3391]! } + public var Weekday_ShortThursday: String { return self._s[3392]! } + public var MessageTimer_Forever: String { return self._s[3393]! } + public var ChatListFolder_CategoryArchived: String { return self._s[3394]! } + public var Channel_EditAdmin_PermissionDeleteMessages: String { return self._s[3395]! } + public var EditTheme_Create_TopInfo: String { return self._s[3397]! } + public var Month_GenDecember: String { return self._s[3398]! } + public var EnterPasscode_EnterPasscode: String { return self._s[3399]! } + public var SettingsSearch_Synonyms_Appearance_LargeEmoji: String { return self._s[3400]! } + public var PeopleNearby_CreateGroup: String { return self._s[3402]! } + public var Group_EditAdmin_PermissionChangeInfo: String { return self._s[3403]! } + public var Paint_ClearConfirm: String { return self._s[3404]! } + public var ChatList_ReadAll: String { return self._s[3405]! } + public var ChatSettings_IntentsSettings: String { return self._s[3406]! } + public var Passport_PassportInformation: String { return self._s[3408]! } + public var Login_CheckOtherSessionMessages: String { return self._s[3410]! } + public var Location_ProximityNotification_DistanceMI: String { return self._s[3413]! } + public var PhotoEditor_ExposureTool: String { return self._s[3414]! } + public var Group_Username_CreatePrivateLinkHelp: String { return self._s[3415]! } + public var SettingsSearch_Synonyms_Watch: String { return self._s[3416]! } + public var Stats_GroupTopPoster_History: String { return self._s[3417]! } + public var UserInfo_AddPhone: String { return self._s[3418]! } + public var Media_SendWithTimer: String { return self._s[3420]! } + public var SettingsSearch_Synonyms_Notifications_Title: String { return self._s[3421]! } + public var Channel_EditAdmin_PermissionEnabledByDefault: String { return self._s[3422]! } + public var PasscodeSettings_AutoLock_Disabled: String { return self._s[3423]! } + public var ChatList_Context_Unarchive: String { return self._s[3425]! } public func DialogList_LiveLocationSharingTo(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3383]!, self._r[3383]!, [_0]) + return formatWithArgumentRanges(self._s[3426]!, self._r[3426]!, [_0]) } - public var BlockedUsers_Title: String { return self._s[3385]! } - public var TwoStepAuth_EmailPlaceholder: String { return self._s[3386]! } - public var Media_ShareThisPhoto: String { return self._s[3387]! } - public var Notifications_DisplayNamesOnLockScreen: String { return self._s[3388]! } - public var Conversation_FilePhotoOrVideo: String { return self._s[3389]! } - public var Appearance_ThemePreview_Chat_2_ReplyName: String { return self._s[3393]! } - public var CallFeedback_ReasonNoise: String { return self._s[3395]! } - public var WebBrowser_Title: String { return self._s[3396]! } + public var BlockedUsers_Title: String { return self._s[3428]! } + public var TwoStepAuth_EmailPlaceholder: String { return self._s[3429]! } + public var Media_ShareThisPhoto: String { return self._s[3430]! } + public var Notifications_DisplayNamesOnLockScreen: String { return self._s[3431]! } + public var Conversation_FilePhotoOrVideo: String { return self._s[3432]! } + public var Appearance_ThemePreview_Chat_2_ReplyName: String { return self._s[3436]! } + public var CallFeedback_ReasonNoise: String { return self._s[3438]! } + public var WebBrowser_Title: String { return self._s[3439]! } public func Checkout_SavePasswordTimeoutAndTouchId(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3397]!, self._r[3397]!, [_0]) + return formatWithArgumentRanges(self._s[3440]!, self._r[3440]!, [_0]) } - public var Notification_MessageLifetime5s: String { return self._s[3399]! } - public var Passport_Address_AddResidentialAddress: String { return self._s[3400]! } - public var Profile_MessageLifetime1m: String { return self._s[3402]! } - public var Passport_ScanPassport: String { return self._s[3403]! } - public var Stats_LoadingTitle: String { return self._s[3404]! } - public var Passport_Address_AddTemporaryRegistration: String { return self._s[3406]! } - public var Permissions_NotificationsAllow_v0: String { return self._s[3407]! } - public var Login_InvalidFirstNameError: String { return self._s[3408]! } - public var Undo_ChatCleared: String { return self._s[3410]! } + public var Notification_MessageLifetime5s: String { return self._s[3442]! } + public var Passport_Address_AddResidentialAddress: String { return self._s[3443]! } + public var Profile_MessageLifetime1m: String { return self._s[3445]! } + public var Passport_ScanPassport: String { return self._s[3446]! } + public var Stats_LoadingTitle: String { return self._s[3447]! } + public var Passport_Address_AddTemporaryRegistration: String { return self._s[3449]! } + public var Permissions_NotificationsAllow_v0: String { return self._s[3450]! } + public var Login_InvalidFirstNameError: String { return self._s[3451]! } + public var Undo_ChatCleared: String { return self._s[3453]! } public func ApplyLanguage_ChangeLanguageUnofficialText(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3412]!, self._r[3412]!, [_1, _2]) + return formatWithArgumentRanges(self._s[3455]!, self._r[3455]!, [_1, _2]) } - public var Conversation_PinMessageAlertPin: String { return self._s[3413]! } + public var Conversation_PinMessageAlertPin: String { return self._s[3456]! } public func Login_PhoneBannedEmailBody(_ _1: String, _ _2: String, _ _3: String, _ _4: String, _ _5: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3414]!, self._r[3414]!, [_1, _2, _3, _4, _5]) + return formatWithArgumentRanges(self._s[3457]!, self._r[3457]!, [_1, _2, _3, _4, _5]) } public func PUSH_MESSAGE_FWD(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3415]!, self._r[3415]!, [_1]) + return formatWithArgumentRanges(self._s[3458]!, self._r[3458]!, [_1]) } - public var Share_MultipleMessagesDisabled: String { return self._s[3416]! } - public var TwoStepAuth_EmailInvalid: String { return self._s[3417]! } - public var EnterPasscode_ChangeTitle: String { return self._s[3419]! } - public var CallSettings_RecentCalls: String { return self._s[3420]! } - public var GroupInfo_DeactivatedStatus: String { return self._s[3421]! } - public var AuthSessions_OtherSessions: String { return self._s[3422]! } - public var PrivacyLastSeenSettings_CustomHelp: String { return self._s[3423]! } - public var Tour_Text5: String { return self._s[3424]! } - public var Login_PadPhoneHelp: String { return self._s[3425]! } - public var Wallpaper_PhotoLibrary: String { return self._s[3427]! } - public var Conversation_ViewGroup: String { return self._s[3428]! } - public var PeopleNearby_MakeVisibleTitle: String { return self._s[3430]! } - public var VoiceOver_Chat_YourContact: String { return self._s[3431]! } - public var Watch_AuthRequired: String { return self._s[3432]! } - public var VoiceOver_Chat_ForwardedFromYou: String { return self._s[3433]! } - public var Conversation_ForwardContacts: String { return self._s[3434]! } - public var Conversation_InputTextPlaceholder: String { return self._s[3435]! } + public var Share_MultipleMessagesDisabled: String { return self._s[3459]! } + public var TwoStepAuth_EmailInvalid: String { return self._s[3460]! } + public var EnterPasscode_ChangeTitle: String { return self._s[3462]! } + public var CallSettings_RecentCalls: String { return self._s[3463]! } + public var GroupInfo_DeactivatedStatus: String { return self._s[3464]! } + public var AuthSessions_OtherSessions: String { return self._s[3465]! } + public var PrivacyLastSeenSettings_CustomHelp: String { return self._s[3466]! } + public var Tour_Text5: String { return self._s[3467]! } + public var Login_PadPhoneHelp: String { return self._s[3468]! } + public var Wallpaper_PhotoLibrary: String { return self._s[3470]! } + public var Conversation_ViewGroup: String { return self._s[3471]! } + public var PeopleNearby_MakeVisibleTitle: String { return self._s[3473]! } + public var VoiceOver_Chat_YourContact: String { return self._s[3474]! } + public var Watch_AuthRequired: String { return self._s[3475]! } + public var VoiceOver_Chat_ForwardedFromYou: String { return self._s[3476]! } + public var Conversation_ForwardContacts: String { return self._s[3477]! } + public var Conversation_InputTextPlaceholder: String { return self._s[3478]! } public func PUSH_CHANNEL_MESSAGE_PHOTO(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3436]!, self._r[3436]!, [_1]) + return formatWithArgumentRanges(self._s[3479]!, self._r[3479]!, [_1]) } public func Conversation_MessageViaUser(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3437]!, self._r[3437]!, [_0]) + return formatWithArgumentRanges(self._s[3480]!, self._r[3480]!, [_0]) } - public var Channel_Setup_TypePrivate: String { return self._s[3438]! } + public var Channel_Setup_TypePrivate: String { return self._s[3481]! } public func Conversation_NoticeInvitedByInChannel(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3439]!, self._r[3439]!, [_0]) - } - public var InfoPlist_NSSiriUsageDescription: String { return self._s[3440]! } - public var AutoDownloadSettings_Delimeter: String { return self._s[3441]! } - public var EmptyGroupInfo_Subtitle: String { return self._s[3442]! } - public var UserInfo_StartSecretChatStart: String { return self._s[3443]! } - public func GroupPermission_AddedInfo(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3444]!, self._r[3444]!, [_1, _2]) - } - public func Channel_AdminLog_MessageRestricted(_ _0: String, _ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3445]!, self._r[3445]!, [_0, _1, _2]) - } - public var PrivacySettings_AutoArchiveTitle: String { return self._s[3446]! } - public var GroupInfo_InviteLink_LinkSection: String { return self._s[3447]! } - public var FastTwoStepSetup_EmailPlaceholder: String { return self._s[3448]! } - public var StickerPacksSettings_ArchivedMasks: String { return self._s[3450]! } - public var NewContact_Title: String { return self._s[3453]! } - public var Appearance_ThemeCarouselTintedNight: String { return self._s[3454]! } - public var VoiceChat_StatusSpeaking: String { return self._s[3455]! } - public var Notifications_PermissionsKeepDisabled: String { return self._s[3456]! } - public func Time_YesterdayAt(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3457]!, self._r[3457]!, [_0]) - } - public func AutoNightTheme_LocationHelp(_ _0: String, _ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3458]!, self._r[3458]!, [_0, _1]) - } - public var Chat_SlowmodeTooltipPending: String { return self._s[3459]! } - public var CallFeedback_ReasonInterruption: String { return self._s[3461]! } - public var ContactInfo_PhoneLabelHome: String { return self._s[3462]! } - public var Passport_Identity_OneOfTypeDriversLicense: String { return self._s[3463]! } - public func PUSH_MESSAGE_DOCS(_ _1: String, _ _2: Int) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3465]!, self._r[3465]!, [_1, "\(_2)"]) - } - public var Conversation_MessageEditedLabel: String { return self._s[3466]! } - public var CallList_ActiveVoiceChatsHeader: String { return self._s[3467]! } - public var SocksProxySetup_PasswordPlaceholder: String { return self._s[3468]! } - public var ChatList_Context_AddToContacts: String { return self._s[3469]! } - public var Passport_Language_is: String { return self._s[3470]! } - public var Notification_PassportValueProofOfIdentity: String { return self._s[3471]! } - public var PhotoEditor_CurvesBlue: String { return self._s[3472]! } - public func FileSize_MB(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3473]!, self._r[3473]!, [_0]) - } - public var SocksProxySetup_Username: String { return self._s[3474]! } - public var Login_SmsRequestState3: String { return self._s[3475]! } - public var Message_PinnedVideoMessage: String { return self._s[3476]! } - public var SharedMedia_TitleLink: String { return self._s[3477]! } - public var Passport_FieldIdentity: String { return self._s[3478]! } - public func Conversation_EncryptedPlaceholderTitleOutgoing(_ _0: String) -> (String, [(Int, NSRange)]) { return formatWithArgumentRanges(self._s[3482]!, self._r[3482]!, [_0]) } - public var DialogList_ProxyConnectionIssuesTooltip: String { return self._s[3485]! } - public var ReportSpam_DeleteThisChat: String { return self._s[3486]! } - public var Checkout_NewCard_CardholderNamePlaceholder: String { return self._s[3487]! } - public var Passport_Identity_DateOfBirth: String { return self._s[3488]! } - public var Call_StatusIncoming: String { return self._s[3489]! } - public var ChatAdmins_AdminLabel: String { return self._s[3490]! } - public func Time_MonthOfYear_m10(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3492]!, self._r[3492]!, [_0]) + public var InviteLink_Create_TimeLimitExpiryDate: String { return self._s[3483]! } + public var InfoPlist_NSSiriUsageDescription: String { return self._s[3484]! } + public var AutoDownloadSettings_Delimeter: String { return self._s[3485]! } + public var EmptyGroupInfo_Subtitle: String { return self._s[3486]! } + public var UserInfo_StartSecretChatStart: String { return self._s[3487]! } + public func GroupPermission_AddedInfo(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3488]!, self._r[3488]!, [_1, _2]) } - public var Message_PinnedAnimationMessage: String { return self._s[3493]! } - public var Conversation_ReportSpamAndLeave: String { return self._s[3494]! } - public var Preview_CopyAddress: String { return self._s[3495]! } - public var MediaPlayer_UnknownTrack: String { return self._s[3496]! } - public var Login_CancelSignUpConfirmation: String { return self._s[3497]! } - public var Map_OpenInYandexMaps: String { return self._s[3499]! } - public func Time_PreciseDate_m11(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3502]!, self._r[3502]!, [_1, _2, _3]) + public func Channel_AdminLog_MessageRestricted(_ _0: String, _ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3489]!, self._r[3489]!, [_0, _1, _2]) } - public var GroupRemoved_Remove: String { return self._s[3503]! } - public var ChatListFolder_TitleCreate: String { return self._s[3504]! } - public func InstantPage_AuthorAndDateTitle(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3506]!, self._r[3506]!, [_1, _2]) + public var PrivacySettings_AutoArchiveTitle: String { return self._s[3490]! } + public var GroupInfo_InviteLink_LinkSection: String { return self._s[3491]! } + public var FastTwoStepSetup_EmailPlaceholder: String { return self._s[3492]! } + public var StickerPacksSettings_ArchivedMasks: String { return self._s[3494]! } + public var NewContact_Title: String { return self._s[3497]! } + public var Appearance_ThemeCarouselTintedNight: String { return self._s[3498]! } + public var VoiceChat_StatusSpeaking: String { return self._s[3499]! } + public var Notifications_PermissionsKeepDisabled: String { return self._s[3500]! } + public func Time_YesterdayAt(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3501]!, self._r[3501]!, [_0]) } - public var Watch_UserInfo_MuteTitle: String { return self._s[3507]! } - public var Group_UpgradeNoticeText2: String { return self._s[3509]! } - public var Stats_GroupGrowthTitle: String { return self._s[3510]! } - public var CreatePoll_CancelConfirmation: String { return self._s[3513]! } - public var Month_GenOctober: String { return self._s[3514]! } - public var Conversation_TitleCommentsEmpty: String { return self._s[3515]! } - public var Settings_Appearance: String { return self._s[3516]! } - public func Time_MonthOfYear_m6(_ _0: String) -> (String, [(Int, NSRange)]) { + public func AutoNightTheme_LocationHelp(_ _0: String, _ _1: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3502]!, self._r[3502]!, [_0, _1]) + } + public var Chat_SlowmodeTooltipPending: String { return self._s[3503]! } + public var CallFeedback_ReasonInterruption: String { return self._s[3505]! } + public var ContactInfo_PhoneLabelHome: String { return self._s[3506]! } + public var Passport_Identity_OneOfTypeDriversLicense: String { return self._s[3507]! } + public func PUSH_MESSAGE_DOCS(_ _1: String, _ _2: Int) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3509]!, self._r[3509]!, [_1, "\(_2)"]) + } + public var Conversation_MessageEditedLabel: String { return self._s[3510]! } + public var CallList_ActiveVoiceChatsHeader: String { return self._s[3511]! } + public var SocksProxySetup_PasswordPlaceholder: String { return self._s[3512]! } + public var ChatList_Context_AddToContacts: String { return self._s[3513]! } + public var Passport_Language_is: String { return self._s[3514]! } + public var Notification_PassportValueProofOfIdentity: String { return self._s[3515]! } + public var PhotoEditor_CurvesBlue: String { return self._s[3516]! } + public func FileSize_MB(_ _0: String) -> (String, [(Int, NSRange)]) { return formatWithArgumentRanges(self._s[3517]!, self._r[3517]!, [_0]) } - public var UserInfo_AddToExisting: String { return self._s[3518]! } - public var Call_PhoneCallInProgressMessage: String { return self._s[3519]! } - public var Map_HomeAndWorkInfo: String { return self._s[3520]! } - public var Paint_Arrow: String { return self._s[3521]! } - public var CancelResetAccount_Title: String { return self._s[3522]! } - public var NotificationsSound_Circles: String { return self._s[3523]! } - public var Notifications_GroupNotificationsExceptionsHelp: String { return self._s[3524]! } - public var ChatState_Connecting: String { return self._s[3526]! } - public var Profile_MessageLifetime5s: String { return self._s[3527]! } + public var SocksProxySetup_Username: String { return self._s[3518]! } + public var Login_SmsRequestState3: String { return self._s[3519]! } + public var Message_PinnedVideoMessage: String { return self._s[3520]! } + public var SharedMedia_TitleLink: String { return self._s[3521]! } + public var Passport_FieldIdentity: String { return self._s[3522]! } + public func Conversation_EncryptedPlaceholderTitleOutgoing(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3526]!, self._r[3526]!, [_0]) + } + public var DialogList_ProxyConnectionIssuesTooltip: String { return self._s[3529]! } + public var ReportSpam_DeleteThisChat: String { return self._s[3530]! } + public var Checkout_NewCard_CardholderNamePlaceholder: String { return self._s[3531]! } + public var Passport_Identity_DateOfBirth: String { return self._s[3532]! } + public var Call_StatusIncoming: String { return self._s[3533]! } + public var ChatAdmins_AdminLabel: String { return self._s[3534]! } + public func Time_MonthOfYear_m10(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3536]!, self._r[3536]!, [_0]) + } + public var Message_PinnedAnimationMessage: String { return self._s[3537]! } + public var Conversation_ReportSpamAndLeave: String { return self._s[3538]! } + public var Preview_CopyAddress: String { return self._s[3539]! } + public var MediaPlayer_UnknownTrack: String { return self._s[3540]! } + public var Login_CancelSignUpConfirmation: String { return self._s[3541]! } + public var Map_OpenInYandexMaps: String { return self._s[3543]! } + public func Time_PreciseDate_m11(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3546]!, self._r[3546]!, [_1, _2, _3]) + } + public var GroupRemoved_Remove: String { return self._s[3547]! } + public var ChatListFolder_TitleCreate: String { return self._s[3548]! } + public func InstantPage_AuthorAndDateTitle(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3550]!, self._r[3550]!, [_1, _2]) + } + public var Watch_UserInfo_MuteTitle: String { return self._s[3551]! } + public var Group_UpgradeNoticeText2: String { return self._s[3553]! } + public var Stats_GroupGrowthTitle: String { return self._s[3554]! } + public var CreatePoll_CancelConfirmation: String { return self._s[3557]! } + public var Month_GenOctober: String { return self._s[3558]! } + public var Conversation_TitleCommentsEmpty: String { return self._s[3559]! } + public var Settings_Appearance: String { return self._s[3560]! } + public func Time_MonthOfYear_m6(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3561]!, self._r[3561]!, [_0]) + } + public var UserInfo_AddToExisting: String { return self._s[3562]! } + public var Call_PhoneCallInProgressMessage: String { return self._s[3563]! } + public var Map_HomeAndWorkInfo: String { return self._s[3564]! } + public var Paint_Arrow: String { return self._s[3565]! } + public var InviteLink_CreatePrivateLinkHelp: String { return self._s[3566]! } + public func DialogList_MultipleTypingPair(_ _0: String, _ _1: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3567]!, self._r[3567]!, [_0, _1]) + } + public var CancelResetAccount_Title: String { return self._s[3568]! } + public var NotificationsSound_Circles: String { return self._s[3569]! } + public var Notifications_GroupNotificationsExceptionsHelp: String { return self._s[3570]! } + public var ChatState_Connecting: String { return self._s[3572]! } + public var Profile_MessageLifetime5s: String { return self._s[3573]! } public func DialogList_AwaitingEncryption(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3528]!, self._r[3528]!, [_0]) + return formatWithArgumentRanges(self._s[3574]!, self._r[3574]!, [_0]) } - public var PrivacyPolicy_AgeVerificationTitle: String { return self._s[3529]! } - public var Channel_Username_CreatePublicLinkHelp: String { return self._s[3530]! } - public var AutoNightTheme_ScheduledTo: String { return self._s[3531]! } - public var Conversation_DefaultRestrictedStickers: String { return self._s[3532]! } - public var TwoStepAuth_ConfirmationTitle: String { return self._s[3533]! } + public var PrivacyPolicy_AgeVerificationTitle: String { return self._s[3575]! } + public var Channel_Username_CreatePublicLinkHelp: String { return self._s[3576]! } + public var AutoNightTheme_ScheduledTo: String { return self._s[3577]! } + public var Conversation_DefaultRestrictedStickers: String { return self._s[3578]! } + public var TwoStepAuth_ConfirmationTitle: String { return self._s[3579]! } public func Chat_UnsendMyMessagesAlertTitle(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3534]!, self._r[3534]!, [_0]) + return formatWithArgumentRanges(self._s[3580]!, self._r[3580]!, [_0]) } - public var Passport_Phone_Help: String { return self._s[3535]! } - public var Privacy_ContactsSync: String { return self._s[3536]! } - public var CheckoutInfo_ReceiverInfoPhone: String { return self._s[3537]! } - public var Channel_AdminLogFilter_EventsLeavingSubscribers: String { return self._s[3538]! } - public var Map_SendMyCurrentLocation: String { return self._s[3539]! } - public var Map_AddressOnMap: String { return self._s[3540]! } - public var DialogList_SearchLabel: String { return self._s[3542]! } - public var Notification_Exceptions_NewException_NotificationHeader: String { return self._s[3543]! } - public var ConversationProfile_UnknownAddMemberError: String { return self._s[3544]! } - public var ChatList_Search_ShowMore: String { return self._s[3545]! } - public var DialogList_EncryptionRejected: String { return self._s[3546]! } - public var VoiceChat_InviteLinkCopiedText: String { return self._s[3547]! } - public var DialogList_DeleteBotConfirmation: String { return self._s[3548]! } - public var VoiceChat_StartRecordingText: String { return self._s[3549]! } - public var Privacy_TopPeersDelete: String { return self._s[3550]! } - public var AttachmentMenu_SendAsFile: String { return self._s[3551]! } - public var ChatList_GenericPsaAlert: String { return self._s[3553]! } - public var SecretTimer_ImageDescription: String { return self._s[3555]! } + public var Passport_Phone_Help: String { return self._s[3581]! } + public var Privacy_ContactsSync: String { return self._s[3582]! } + public var CheckoutInfo_ReceiverInfoPhone: String { return self._s[3583]! } + public var Channel_AdminLogFilter_EventsLeavingSubscribers: String { return self._s[3584]! } + public var Map_SendMyCurrentLocation: String { return self._s[3585]! } + public var Map_AddressOnMap: String { return self._s[3586]! } + public var DialogList_SearchLabel: String { return self._s[3588]! } + public var Notification_Exceptions_NewException_NotificationHeader: String { return self._s[3589]! } + public var Conversation_ChecksTooltip_Read: String { return self._s[3590]! } + public var ConversationProfile_UnknownAddMemberError: String { return self._s[3591]! } + public var ChatList_Search_ShowMore: String { return self._s[3592]! } + public var DialogList_EncryptionRejected: String { return self._s[3593]! } + public var VoiceChat_InviteLinkCopiedText: String { return self._s[3594]! } + public var DialogList_DeleteBotConfirmation: String { return self._s[3595]! } + public var VoiceChat_StartRecordingText: String { return self._s[3596]! } + public var Privacy_TopPeersDelete: String { return self._s[3597]! } + public var AttachmentMenu_SendAsFile: String { return self._s[3599]! } + public var ChatList_GenericPsaAlert: String { return self._s[3601]! } + public var SecretTimer_ImageDescription: String { return self._s[3603]! } public func Conversation_SetReminder_RemindOn(_ _0: String, _ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3556]!, self._r[3556]!, [_0, _1]) + return formatWithArgumentRanges(self._s[3604]!, self._r[3604]!, [_0, _1]) } - public var ChatSettings_TextSizeUnits: String { return self._s[3557]! } - public var Notification_RenamedGroup: String { return self._s[3558]! } - public var Tour_Title2: String { return self._s[3559]! } - public var Settings_CopyUsername: String { return self._s[3560]! } - public var Compose_NewEncryptedChat: String { return self._s[3561]! } - public var Conversation_CloudStorageInfo_Title: String { return self._s[3562]! } - public var Month_ShortSeptember: String { return self._s[3563]! } - public var AutoDownloadSettings_OnForAll: String { return self._s[3564]! } - public var ChatList_DeleteForEveryoneConfirmationText: String { return self._s[3565]! } - public var Call_StatusConnecting: String { return self._s[3567]! } - public var Privacy_GroupsAndChannels_NeverAllow_Placeholder: String { return self._s[3568]! } - public var Map_ShareLiveLocationHelp: String { return self._s[3569]! } - public var Cache_Files: String { return self._s[3570]! } - public var Notifications_Reset: String { return self._s[3571]! } + public var ChatSettings_TextSizeUnits: String { return self._s[3605]! } + public var Notification_RenamedGroup: String { return self._s[3606]! } + public var Tour_Title2: String { return self._s[3607]! } + public var Settings_CopyUsername: String { return self._s[3608]! } + public var Compose_NewEncryptedChat: String { return self._s[3609]! } + public var Conversation_CloudStorageInfo_Title: String { return self._s[3610]! } + public var Month_ShortSeptember: String { return self._s[3611]! } + public var AutoDownloadSettings_OnForAll: String { return self._s[3612]! } + public var ChatList_DeleteForEveryoneConfirmationText: String { return self._s[3613]! } + public var Call_StatusConnecting: String { return self._s[3615]! } + public var Privacy_GroupsAndChannels_NeverAllow_Placeholder: String { return self._s[3616]! } + public var Map_ShareLiveLocationHelp: String { return self._s[3617]! } + public var Cache_Files: String { return self._s[3618]! } + public var Notifications_Reset: String { return self._s[3619]! } public func Settings_KeepPhoneNumber(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3572]!, self._r[3572]!, [_0]) + return formatWithArgumentRanges(self._s[3620]!, self._r[3620]!, [_0]) } - public var Privacy_GroupsAndChannels_AlwaysAllow_Title: String { return self._s[3573]! } + public var Privacy_GroupsAndChannels_AlwaysAllow_Title: String { return self._s[3621]! } public func Conversation_OpenBotLinkLogin(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3574]!, self._r[3574]!, [_1, _2]) + return formatWithArgumentRanges(self._s[3622]!, self._r[3622]!, [_1, _2]) } - public var Notification_CallIncomingShort: String { return self._s[3575]! } - public var UserInfo_BotPrivacy: String { return self._s[3577]! } - public var Appearance_BubbleCorners_Apply: String { return self._s[3578]! } - public var WebSearch_RecentClearConfirmation: String { return self._s[3579]! } - public var Conversation_ContextMenuLookUp: String { return self._s[3580]! } - public var Calls_RatingTitle: String { return self._s[3581]! } - public var SecretImage_Title: String { return self._s[3582]! } - public var Weekday_Monday: String { return self._s[3583]! } + public var Notification_CallIncomingShort: String { return self._s[3623]! } + public var UserInfo_BotPrivacy: String { return self._s[3625]! } + public var Appearance_BubbleCorners_Apply: String { return self._s[3626]! } + public var WebSearch_RecentClearConfirmation: String { return self._s[3627]! } + public var Conversation_ContextMenuLookUp: String { return self._s[3628]! } + public var Calls_RatingTitle: String { return self._s[3629]! } + public var SecretImage_Title: String { return self._s[3630]! } + public var Weekday_Monday: String { return self._s[3631]! } public func Passport_PrivacyPolicy(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3584]!, self._r[3584]!, [_1, _2]) + return formatWithArgumentRanges(self._s[3632]!, self._r[3632]!, [_1, _2]) } - public var KeyCommand_JumpToPreviousChat: String { return self._s[3585]! } + public var KeyCommand_JumpToPreviousChat: String { return self._s[3633]! } public func DialogList_SearchSubtitleFormat(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3586]!, self._r[3586]!, [_1, _2]) + return formatWithArgumentRanges(self._s[3634]!, self._r[3634]!, [_1, _2]) } - public var Stats_GroupMembers: String { return self._s[3587]! } - public var Camera_Retake: String { return self._s[3588]! } - public var Conversation_SearchPlaceholder: String { return self._s[3590]! } + public var Stats_GroupMembers: String { return self._s[3635]! } + public var Camera_Retake: String { return self._s[3636]! } + public var Conversation_SearchPlaceholder: String { return self._s[3638]! } public func Passport_Identity_NativeNameGenericHelp(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3591]!, self._r[3591]!, [_0]) + return formatWithArgumentRanges(self._s[3639]!, self._r[3639]!, [_0]) } - public var Channel_DiscussionGroup_Info: String { return self._s[3592]! } - public var SocksProxySetup_Hostname: String { return self._s[3593]! } - public var PrivacyLastSeenSettings_EmpryUsersPlaceholder: String { return self._s[3594]! } - public var Privacy_DeleteDrafts: String { return self._s[3596]! } + public var Channel_DiscussionGroup_Info: String { return self._s[3640]! } + public var SocksProxySetup_Hostname: String { return self._s[3641]! } + public var PrivacyLastSeenSettings_EmpryUsersPlaceholder: String { return self._s[3642]! } + public var Privacy_DeleteDrafts: String { return self._s[3644]! } public func Checkout_LiabilityAlert(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3597]!, self._r[3597]!, [_1, _1, _1, _2]) + return formatWithArgumentRanges(self._s[3645]!, self._r[3645]!, [_1, _1, _1, _2]) } - public var Login_CancelPhoneVerification: String { return self._s[3599]! } - public var TwoStepAuth_ResetAccountHelp: String { return self._s[3600]! } + public var Login_CancelPhoneVerification: String { return self._s[3647]! } + public var TwoStepAuth_ResetAccountHelp: String { return self._s[3648]! } public func SocksProxySetup_ProxyStatusPing(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3601]!, self._r[3601]!, [_0]) + return formatWithArgumentRanges(self._s[3649]!, self._r[3649]!, [_0]) } - public var TwoStepAuth_EmailSent: String { return self._s[3602]! } - public var Cache_Indexing: String { return self._s[3603]! } - public var Notifications_ExceptionsNone: String { return self._s[3604]! } - public var MessagePoll_LabelQuiz: String { return self._s[3605]! } - public var Call_EncryptionKey_Title: String { return self._s[3606]! } - public var Common_Yes: String { return self._s[3607]! } - public var Channel_ErrorAddBlocked: String { return self._s[3608]! } - public var Month_GenJanuary: String { return self._s[3609]! } - public var Checkout_NewCard_Title: String { return self._s[3610]! } + public var TwoStepAuth_EmailSent: String { return self._s[3650]! } + public var Cache_Indexing: String { return self._s[3651]! } + public var Notifications_ExceptionsNone: String { return self._s[3652]! } + public var MessagePoll_LabelQuiz: String { return self._s[3653]! } + public var Call_EncryptionKey_Title: String { return self._s[3654]! } + public var Common_Yes: String { return self._s[3655]! } + public var Channel_ErrorAddBlocked: String { return self._s[3656]! } + public var Month_GenJanuary: String { return self._s[3657]! } + public var Checkout_NewCard_Title: String { return self._s[3658]! } public func TwoStepAuth_EnterPasswordHint(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3611]!, self._r[3611]!, [_0]) + return formatWithArgumentRanges(self._s[3659]!, self._r[3659]!, [_0]) } - public var Conversation_InputTextPlaceholderReply: String { return self._s[3613]! } - public var PasscodeSettings_AutoLock_IfAwayFor_1hour: String { return self._s[3614]! } - public var Conversation_SendDice: String { return self._s[3615]! } + public var Conversation_InputTextPlaceholderReply: String { return self._s[3661]! } + public var PasscodeSettings_AutoLock_IfAwayFor_1hour: String { return self._s[3662]! } + public var Conversation_SendDice: String { return self._s[3663]! } public func ChatSettings_AutoDownloadSettings_TypeVideo(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3616]!, self._r[3616]!, [_0]) + return formatWithArgumentRanges(self._s[3664]!, self._r[3664]!, [_0]) } public func VoiceOver_Chat_VideoFrom(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3617]!, self._r[3617]!, [_0]) + return formatWithArgumentRanges(self._s[3665]!, self._r[3665]!, [_0]) } - public var Weekday_Wednesday: String { return self._s[3618]! } - public var ReportPeer_ReasonOther_Send: String { return self._s[3619]! } - public var PasscodeSettings_EncryptDataHelp: String { return self._s[3620]! } - public var PrivacyLastSeenSettings_CustomShareSettingsHelp: String { return self._s[3621]! } - public var OldChannels_NoticeTitle: String { return self._s[3622]! } - public var TwoStepAuth_ChangeEmail: String { return self._s[3623]! } - public var PasscodeSettings_PasscodeOptions: String { return self._s[3624]! } - public var InfoPlist_NSPhotoLibraryUsageDescription: String { return self._s[3625]! } - public var Passport_Address_AddUtilityBill: String { return self._s[3626]! } + public var Weekday_Wednesday: String { return self._s[3666]! } + public var ReportPeer_ReasonOther_Send: String { return self._s[3667]! } + public var PasscodeSettings_EncryptDataHelp: String { return self._s[3668]! } + public var PrivacyLastSeenSettings_CustomShareSettingsHelp: String { return self._s[3669]! } + public var OldChannels_NoticeTitle: String { return self._s[3670]! } + public var TwoStepAuth_ChangeEmail: String { return self._s[3671]! } + public var PasscodeSettings_PasscodeOptions: String { return self._s[3672]! } + public var InfoPlist_NSPhotoLibraryUsageDescription: String { return self._s[3673]! } + public var Passport_Address_AddUtilityBill: String { return self._s[3674]! } public func Time_PreciseDate_m5(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3628]!, self._r[3628]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[3676]!, self._r[3676]!, [_1, _2, _3]) } - public var TwoFactorSetup_EmailVerification_ResendAction: String { return self._s[3630]! } - public var Stats_GroupTopAdminsTitle: String { return self._s[3631]! } - public var Paint_Regular: String { return self._s[3632]! } - public var Message_Contact: String { return self._s[3633]! } - public var NetworkUsageSettings_MediaVideoDataSection: String { return self._s[3634]! } - public var VoiceOver_Chat_YourPhoto: String { return self._s[3635]! } - public var Notification_Mute1hMin: String { return self._s[3636]! } + public var TwoFactorSetup_EmailVerification_ResendAction: String { return self._s[3678]! } + public var Stats_GroupTopAdminsTitle: String { return self._s[3679]! } + public var Paint_Regular: String { return self._s[3680]! } + public var Message_Contact: String { return self._s[3681]! } + public var NetworkUsageSettings_MediaVideoDataSection: String { return self._s[3682]! } + public var VoiceOver_Chat_YourPhoto: String { return self._s[3683]! } + public var Notification_Mute1hMin: String { return self._s[3684]! } public func Login_BannedPhoneSubject(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3637]!, self._r[3637]!, [_0]) + return formatWithArgumentRanges(self._s[3685]!, self._r[3685]!, [_0]) } - public var Profile_MessageLifetime1h: String { return self._s[3638]! } - public var TwoStepAuth_GenericHelp: String { return self._s[3639]! } - public var TextFormat_Monospace: String { return self._s[3640]! } - public var VoiceOver_Media_PlaybackRateChange: String { return self._s[3642]! } - public var Conversation_DeleteMessagesForMe: String { return self._s[3643]! } - public var ChatList_DeleteChat: String { return self._s[3644]! } - public var Channel_OwnershipTransfer_EnterPasswordText: String { return self._s[3647]! } + public var Profile_MessageLifetime1h: String { return self._s[3686]! } + public var TwoStepAuth_GenericHelp: String { return self._s[3687]! } + public var TextFormat_Monospace: String { return self._s[3688]! } + public var VoiceOver_Media_PlaybackRateChange: String { return self._s[3690]! } + public var Conversation_DeleteMessagesForMe: String { return self._s[3691]! } + public var ChatList_DeleteChat: String { return self._s[3692]! } + public var Channel_OwnershipTransfer_EnterPasswordText: String { return self._s[3695]! } public func Settings_ApplyProxyAlertCredentials(_ _1: String, _ _2: String, _ _3: String, _ _4: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3648]!, self._r[3648]!, [_1, _2, _3, _4]) + return formatWithArgumentRanges(self._s[3696]!, self._r[3696]!, [_1, _2, _3, _4]) } - public var Login_CancelPhoneVerificationStop: String { return self._s[3649]! } - public var Appearance_ThemePreview_ChatList_4_Name: String { return self._s[3650]! } - public var MediaPicker_MomentsDateRangeSameMonthYearFormat: String { return self._s[3651]! } + public var Login_CancelPhoneVerificationStop: String { return self._s[3697]! } + public var Appearance_ThemePreview_ChatList_4_Name: String { return self._s[3698]! } + public var MediaPicker_MomentsDateRangeSameMonthYearFormat: String { return self._s[3699]! } public func Channel_AdminLog_MessageToggleInvitesOn(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3652]!, self._r[3652]!, [_0]) + return formatWithArgumentRanges(self._s[3700]!, self._r[3700]!, [_0]) } - public var Notifications_Badge_IncludeChannels: String { return self._s[3653]! } - public var StickerPack_ViewPack: String { return self._s[3656]! } - public var FastTwoStepSetup_PasswordConfirmationPlaceholder: String { return self._s[3658]! } - public var EditTheme_Expand_Preview_IncomingText: String { return self._s[3659]! } - public var Notifications_Title: String { return self._s[3660]! } - public var Conversation_InputTextPlaceholderComment: String { return self._s[3661]! } - public var GroupInfo_PublicLink: String { return self._s[3662]! } - public var VoiceOver_DiscardPreparedContent: String { return self._s[3663]! } - public var Conversation_Moderate_Ban: String { return self._s[3667]! } + public var Notifications_Badge_IncludeChannels: String { return self._s[3701]! } + public var StickerPack_ViewPack: String { return self._s[3704]! } + public var FastTwoStepSetup_PasswordConfirmationPlaceholder: String { return self._s[3706]! } + public var EditTheme_Expand_Preview_IncomingText: String { return self._s[3707]! } + public var Notifications_Title: String { return self._s[3708]! } + public var Conversation_InputTextPlaceholderComment: String { return self._s[3709]! } + public var GroupInfo_PublicLink: String { return self._s[3710]! } + public var VoiceOver_DiscardPreparedContent: String { return self._s[3711]! } + public var Conversation_Moderate_Ban: String { return self._s[3715]! } + public var InviteLink_Manage: String { return self._s[3716]! } public func Activity_RemindAboutGroup(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3668]!, self._r[3668]!, [_0]) + return formatWithArgumentRanges(self._s[3717]!, self._r[3717]!, [_0]) } - public var TextFormat_Underline: String { return self._s[3669]! } + public var TextFormat_Underline: String { return self._s[3718]! } public func DownloadingStatus(_ _0: String, _ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3670]!, self._r[3670]!, [_0, _1]) + return formatWithArgumentRanges(self._s[3719]!, self._r[3719]!, [_0, _1]) } public func PUSH_PINNED_ROUND(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3671]!, self._r[3671]!, [_1]) + return formatWithArgumentRanges(self._s[3720]!, self._r[3720]!, [_1]) } - public var PollResults_Collapse: String { return self._s[3673]! } - public var Contacts_GlobalSearch: String { return self._s[3674]! } + public var PollResults_Collapse: String { return self._s[3722]! } + public var Contacts_GlobalSearch: String { return self._s[3723]! } public func Conversation_EncryptionWaiting(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3675]!, self._r[3675]!, [_0]) + return formatWithArgumentRanges(self._s[3725]!, self._r[3725]!, [_0]) } - public var Channel_Management_LabelEditor: String { return self._s[3676]! } - public var Conversation_VoiceChatMediaRecordingRestricted: String { return self._s[3677]! } - public var SettingsSearch_Synonyms_Stickers_FeaturedPacks: String { return self._s[3678]! } - public var Conversation_Theme: String { return self._s[3679]! } + public var Channel_Management_LabelEditor: String { return self._s[3726]! } + public var Conversation_VoiceChatMediaRecordingRestricted: String { return self._s[3727]! } + public var SettingsSearch_Synonyms_Stickers_FeaturedPacks: String { return self._s[3728]! } + public var Conversation_Theme: String { return self._s[3729]! } public func PUSH_CHANNEL_MESSAGE_DOCS(_ _1: String, _ _2: Int) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3680]!, self._r[3680]!, [_1, "\(_2)"]) + return formatWithArgumentRanges(self._s[3730]!, self._r[3730]!, [_1, "\(_2)"]) } - public var Conversation_LinkDialogSave: String { return self._s[3681]! } - public var EnterPasscode_TouchId: String { return self._s[3682]! } - public var Stats_MessageOverview: String { return self._s[3684]! } - public var Privacy_Calls_P2PAlways: String { return self._s[3686]! } - public var Message_Sticker: String { return self._s[3687]! } - public var Conversation_Mute: String { return self._s[3689]! } - public var VoiceChat_AnonymousDisabledAlertText: String { return self._s[3690]! } - public var ContactInfo_Title: String { return self._s[3691]! } + public var Conversation_LinkDialogSave: String { return self._s[3731]! } + public var EnterPasscode_TouchId: String { return self._s[3732]! } + public var Stats_MessageOverview: String { return self._s[3734]! } + public var Privacy_Calls_P2PAlways: String { return self._s[3736]! } + public var Message_Sticker: String { return self._s[3737]! } + public var Conversation_Mute: String { return self._s[3739]! } + public var VoiceChat_AnonymousDisabledAlertText: String { return self._s[3740]! } + public var ContactInfo_Title: String { return self._s[3741]! } public func PUSH_CHANNEL_MESSAGE_CONTACT(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3692]!, self._r[3692]!, [_1]) + return formatWithArgumentRanges(self._s[3742]!, self._r[3742]!, [_1]) } - public var Channel_Setup_TypeHeader: String { return self._s[3693]! } - public var AuthSessions_LogOut: String { return self._s[3694]! } - public var ChatSettings_AutoDownloadReset: String { return self._s[3695]! } - public var ChatListFolderSettings_NewFolder: String { return self._s[3697]! } - public var Appearance_ThemePreview_ChatList_3_AuthorName: String { return self._s[3698]! } - public var CreatePoll_Title: String { return self._s[3699]! } - public var EditTheme_EditTitle: String { return self._s[3700]! } - public var ChatListFolderSettings_RecommendedFoldersSection: String { return self._s[3701]! } - public var TwoStepAuth_SetPassword: String { return self._s[3702]! } + public var Channel_Setup_TypeHeader: String { return self._s[3743]! } + public var AuthSessions_LogOut: String { return self._s[3744]! } + public var ChatSettings_AutoDownloadReset: String { return self._s[3745]! } + public var ChatListFolderSettings_NewFolder: String { return self._s[3747]! } + public var Appearance_ThemePreview_ChatList_3_AuthorName: String { return self._s[3748]! } + public var CreatePoll_Title: String { return self._s[3749]! } + public var EditTheme_EditTitle: String { return self._s[3750]! } + public var ChatListFolderSettings_RecommendedFoldersSection: String { return self._s[3751]! } + public var TwoStepAuth_SetPassword: String { return self._s[3752]! } public func Login_InvalidPhoneEmailSubject(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3703]!, self._r[3703]!, [_0]) - } - public var BlockedUsers_Info: String { return self._s[3704]! } - public var AuthSessions_Sessions: String { return self._s[3705]! } - public var Group_EditAdmin_RankTitle: String { return self._s[3706]! } - public var Common_ActionNotAllowedError: String { return self._s[3707]! } - public var WebPreview_GettingLinkInfo: String { return self._s[3708]! } - public var Appearance_AppIconFilledX: String { return self._s[3709]! } - public var Passport_Email_EmailPlaceholder: String { return self._s[3710]! } - public var FeaturedStickers_OtherSection: String { return self._s[3711]! } - public var VoiceChat_RecordingStarted: String { return self._s[3712]! } - public var EditTheme_Edit_Preview_OutgoingText: String { return self._s[3713]! } - public var Profile_Username: String { return self._s[3714]! } - public var Appearance_RemoveTheme: String { return self._s[3715]! } - public var TwoStepAuth_SetupPasswordConfirmPassword: String { return self._s[3716]! } - public var Message_PinnedStickerMessage: String { return self._s[3717]! } - public var AccessDenied_VideoMicrophone: String { return self._s[3718]! } - public var WallpaperPreview_CustomColorBottomText: String { return self._s[3719]! } - public var Passport_Address_RegionPlaceholder: String { return self._s[3720]! } - public var SettingsSearch_Synonyms_Data_Storage_Title: String { return self._s[3721]! } - public var TwoStepAuth_Title: String { return self._s[3722]! } - public var Checkout_WebConfirmation_Title: String { return self._s[3723]! } - public var AutoDownloadSettings_VoiceMessagesInfo: String { return self._s[3724]! } - public var ChatListFolder_CategoryGroups: String { return self._s[3726]! } - public var Stats_GroupTopInviter_Promote: String { return self._s[3727]! } - public var Conversation_EditingPhotoPanelTitle: String { return self._s[3728]! } - public var Month_GenJuly: String { return self._s[3729]! } - public var Passport_Identity_Gender: String { return self._s[3730]! } - public var Channel_DiscussionGroup_UnlinkGroup: String { return self._s[3731]! } - public var Notification_Exceptions_DeleteAll: String { return self._s[3732]! } - public var VoiceChat_StopRecording: String { return self._s[3733]! } - public func Conversation_FileHowToText(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3734]!, self._r[3734]!, [_0]) - } - public func Channel_AdminLog_MessageAdmin(_ _0: String, _ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3735]!, self._r[3735]!, [_0, _1, _2]) - } - public var Login_CodeSentSms: String { return self._s[3736]! } - public func VoiceOver_Chat_ReplyFrom(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3737]!, self._r[3737]!, [_0]) - } - public var Login_CallRequestState2: String { return self._s[3738]! } - public var Channel_DiscussionGroup_Header: String { return self._s[3739]! } - public func Channel_AdminLog_MessageToggleInvitesOff(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3740]!, self._r[3740]!, [_0]) - } - public var Passport_Language_ms: String { return self._s[3741]! } - public var PeopleNearby_MakeInvisible: String { return self._s[3743]! } - public var ChatList_Search_FilterVoice: String { return self._s[3745]! } - public var Camera_TapAndHoldForVideo: String { return self._s[3747]! } - public var Permissions_NotificationsAllowInSettings_v0: String { return self._s[3748]! } - public func Notification_LeftChannel(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3749]!, self._r[3749]!, [_0]) - } - public func Call_VoiceChatInProgressMessageCall(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3750]!, self._r[3750]!, [_1, _2]) - } - public var Map_Locating: String { return self._s[3751]! } - public func Checkout_SavePasswordTimeout(_ _0: String) -> (String, [(Int, NSRange)]) { return formatWithArgumentRanges(self._s[3753]!, self._r[3753]!, [_0]) } - public var Passport_Identity_TypeInternalPassport: String { return self._s[3755]! } - public var Appearance_ThemePreview_Chat_4_Text: String { return self._s[3756]! } - public var SettingsSearch_Synonyms_EditProfile_Username: String { return self._s[3757]! } - public var Stickers_Installed: String { return self._s[3758]! } - public var Notifications_PermissionsAllowInSettings: String { return self._s[3759]! } - public var StickerPackActionInfo_RemovedTitle: String { return self._s[3760]! } - public var CallSettings_Never: String { return self._s[3762]! } - public var Channel_Setup_TypePublicHelp: String { return self._s[3763]! } - public func ChatList_DeleteForEveryone(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3765]!, self._r[3765]!, [_0]) + public var BlockedUsers_Info: String { return self._s[3754]! } + public var AuthSessions_Sessions: String { return self._s[3755]! } + public var Group_EditAdmin_RankTitle: String { return self._s[3756]! } + public var Common_ActionNotAllowedError: String { return self._s[3757]! } + public var WebPreview_GettingLinkInfo: String { return self._s[3758]! } + public var Appearance_AppIconFilledX: String { return self._s[3759]! } + public var Passport_Email_EmailPlaceholder: String { return self._s[3760]! } + public var FeaturedStickers_OtherSection: String { return self._s[3761]! } + public var VoiceChat_RecordingStarted: String { return self._s[3762]! } + public var EditTheme_Edit_Preview_OutgoingText: String { return self._s[3763]! } + public var Profile_Username: String { return self._s[3764]! } + public var Appearance_RemoveTheme: String { return self._s[3765]! } + public var TwoStepAuth_SetupPasswordConfirmPassword: String { return self._s[3766]! } + public var Message_PinnedStickerMessage: String { return self._s[3767]! } + public var AccessDenied_VideoMicrophone: String { return self._s[3768]! } + public var WallpaperPreview_CustomColorBottomText: String { return self._s[3769]! } + public var Passport_Address_RegionPlaceholder: String { return self._s[3770]! } + public var SettingsSearch_Synonyms_Data_Storage_Title: String { return self._s[3771]! } + public var TwoStepAuth_Title: String { return self._s[3772]! } + public var Checkout_WebConfirmation_Title: String { return self._s[3773]! } + public var AutoDownloadSettings_VoiceMessagesInfo: String { return self._s[3774]! } + public var ChatListFolder_CategoryGroups: String { return self._s[3776]! } + public var Stats_GroupTopInviter_Promote: String { return self._s[3777]! } + public var Conversation_EditingPhotoPanelTitle: String { return self._s[3778]! } + public var Month_GenJuly: String { return self._s[3779]! } + public var Passport_Identity_Gender: String { return self._s[3780]! } + public var Channel_DiscussionGroup_UnlinkGroup: String { return self._s[3781]! } + public var Notification_Exceptions_DeleteAll: String { return self._s[3782]! } + public var VoiceChat_StopRecording: String { return self._s[3783]! } + public func Conversation_FileHowToText(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3784]!, self._r[3784]!, [_0]) } - public var Message_Game: String { return self._s[3766]! } - public var Call_Message: String { return self._s[3767]! } - public func PUSH_CHANNEL_MESSAGE_VIDEO(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3768]!, self._r[3768]!, [_1]) + public func Channel_AdminLog_MessageAdmin(_ _0: String, _ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3785]!, self._r[3785]!, [_0, _1, _2]) } - public var ChannelIntro_Text: String { return self._s[3769]! } - public var StickerPack_Send: String { return self._s[3770]! } - public var Share_AuthDescription: String { return self._s[3771]! } - public var PasscodeSettings_AutoLock_IfAwayFor_5minutes: String { return self._s[3772]! } - public var CallFeedback_WhatWentWrong: String { return self._s[3773]! } - public var Common_Create: String { return self._s[3776]! } - public var Passport_Language_hy: String { return self._s[3777]! } - public var CreatePoll_Explanation: String { return self._s[3778]! } - public var GroupPermission_AddMembersNotAvailable: String { return self._s[3779]! } - public var PeerInfo_ButtonVoiceChat: String { return self._s[3780]! } - public var Undo_ChatClearedForBothSides: String { return self._s[3781]! } - public var DialogList_NoMessagesTitle: String { return self._s[3782]! } - public var GroupInfo_Title: String { return self._s[3784]! } - public var Channel_AdminLog_CanBanUsers: String { return self._s[3785]! } - public var PhoneNumberHelp_Help: String { return self._s[3786]! } - public var TwoStepAuth_AdditionalPassword: String { return self._s[3787]! } - public var Settings_Logout: String { return self._s[3788]! } - public var Privacy_PaymentsTitle: String { return self._s[3789]! } - public var StickerPacksSettings_StickerPacksSection: String { return self._s[3790]! } - public var Tour_Text6: String { return self._s[3791]! } - public var Channel_Username_Help: String { return self._s[3793]! } - public var VoiceOver_Chat_RecordModeVoiceMessageInfo: String { return self._s[3794]! } - public var AttachmentMenu_Poll: String { return self._s[3795]! } - public var EditTheme_Create_Preview_IncomingReplyName: String { return self._s[3796]! } - public var Conversation_ReportSpamChannelConfirmation: String { return self._s[3797]! } - public var Passport_DeletePassport: String { return self._s[3798]! } - public var Login_Code: String { return self._s[3799]! } - public var Notification_SecretChatScreenshot: String { return self._s[3800]! } - public var Login_CodeFloodError: String { return self._s[3801]! } - public func Notification_PinnedAnimationMessage(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3802]!, self._r[3802]!, [_0]) + public var Login_CodeSentSms: String { return self._s[3786]! } + public func VoiceOver_Chat_ReplyFrom(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3787]!, self._r[3787]!, [_0]) } - public func Channel_Username_UsernameIsAvailable(_ _0: String) -> (String, [(Int, NSRange)]) { + public var Login_CallRequestState2: String { return self._s[3788]! } + public var Channel_DiscussionGroup_Header: String { return self._s[3789]! } + public func Channel_AdminLog_MessageToggleInvitesOff(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3790]!, self._r[3790]!, [_0]) + } + public var Passport_Language_ms: String { return self._s[3791]! } + public var PeopleNearby_MakeInvisible: String { return self._s[3793]! } + public var ChatList_Search_FilterVoice: String { return self._s[3795]! } + public var Camera_TapAndHoldForVideo: String { return self._s[3797]! } + public var Permissions_NotificationsAllowInSettings_v0: String { return self._s[3798]! } + public func Notification_LeftChannel(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3799]!, self._r[3799]!, [_0]) + } + public func Call_VoiceChatInProgressMessageCall(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3800]!, self._r[3800]!, [_1, _2]) + } + public var Map_Locating: String { return self._s[3801]! } + public func Checkout_SavePasswordTimeout(_ _0: String) -> (String, [(Int, NSRange)]) { return formatWithArgumentRanges(self._s[3803]!, self._r[3803]!, [_0]) } - public var Watch_Stickers_Recents: String { return self._s[3804]! } - public var Generic_ErrorMoreInfo: String { return self._s[3805]! } + public var Passport_Identity_TypeInternalPassport: String { return self._s[3805]! } + public var Appearance_ThemePreview_Chat_4_Text: String { return self._s[3806]! } + public var SettingsSearch_Synonyms_EditProfile_Username: String { return self._s[3807]! } + public var Stickers_Installed: String { return self._s[3808]! } + public var Notifications_PermissionsAllowInSettings: String { return self._s[3809]! } + public var StickerPackActionInfo_RemovedTitle: String { return self._s[3810]! } + public var CallSettings_Never: String { return self._s[3812]! } + public var Channel_Setup_TypePublicHelp: String { return self._s[3813]! } + public func ChatList_DeleteForEveryone(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3815]!, self._r[3815]!, [_0]) + } + public var Message_Game: String { return self._s[3816]! } + public var Call_Message: String { return self._s[3817]! } + public func PUSH_CHANNEL_MESSAGE_VIDEO(_ _1: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3818]!, self._r[3818]!, [_1]) + } + public var ChannelIntro_Text: String { return self._s[3819]! } + public var StickerPack_Send: String { return self._s[3820]! } + public var Share_AuthDescription: String { return self._s[3821]! } + public var PasscodeSettings_AutoLock_IfAwayFor_5minutes: String { return self._s[3822]! } + public var CallFeedback_WhatWentWrong: String { return self._s[3823]! } + public var Common_Create: String { return self._s[3826]! } + public var Passport_Language_hy: String { return self._s[3827]! } + public var CreatePoll_Explanation: String { return self._s[3828]! } + public var GroupPermission_AddMembersNotAvailable: String { return self._s[3829]! } + public var PeerInfo_ButtonVoiceChat: String { return self._s[3830]! } + public var Undo_ChatClearedForBothSides: String { return self._s[3831]! } + public var DialogList_NoMessagesTitle: String { return self._s[3832]! } + public var GroupInfo_Title: String { return self._s[3834]! } + public var Channel_AdminLog_CanBanUsers: String { return self._s[3835]! } + public var PhoneNumberHelp_Help: String { return self._s[3836]! } + public var TwoStepAuth_AdditionalPassword: String { return self._s[3837]! } + public var Settings_Logout: String { return self._s[3838]! } + public var Privacy_PaymentsTitle: String { return self._s[3839]! } + public var StickerPacksSettings_StickerPacksSection: String { return self._s[3840]! } + public var Tour_Text6: String { return self._s[3841]! } + public var Channel_Username_Help: String { return self._s[3843]! } + public var VoiceOver_Chat_RecordModeVoiceMessageInfo: String { return self._s[3844]! } + public var AttachmentMenu_Poll: String { return self._s[3845]! } + public var EditTheme_Create_Preview_IncomingReplyName: String { return self._s[3846]! } + public var Conversation_ReportSpamChannelConfirmation: String { return self._s[3847]! } + public var Passport_DeletePassport: String { return self._s[3848]! } + public var Login_Code: String { return self._s[3849]! } + public var Notification_SecretChatScreenshot: String { return self._s[3850]! } + public var Login_CodeFloodError: String { return self._s[3851]! } + public func Notification_PinnedAnimationMessage(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3852]!, self._r[3852]!, [_0]) + } + public func Channel_Username_UsernameIsAvailable(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3853]!, self._r[3853]!, [_0]) + } + public var Watch_Stickers_Recents: String { return self._s[3854]! } + public var Generic_ErrorMoreInfo: String { return self._s[3855]! } public func Call_AccountIsLoggedOnCurrentDevice(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3806]!, self._r[3806]!, [_0]) + return formatWithArgumentRanges(self._s[3856]!, self._r[3856]!, [_0]) } - public var AutoDownloadSettings_DataUsage: String { return self._s[3807]! } - public var Conversation_ViewTheme: String { return self._s[3808]! } - public var Contacts_InviteSearchLabel: String { return self._s[3809]! } - public var Settings_CancelUpload: String { return self._s[3811]! } - public var Settings_AppLanguage_Unofficial: String { return self._s[3812]! } + public var AutoDownloadSettings_DataUsage: String { return self._s[3857]! } + public var Conversation_ViewTheme: String { return self._s[3858]! } + public var Contacts_InviteSearchLabel: String { return self._s[3859]! } + public var Settings_CancelUpload: String { return self._s[3861]! } + public var Settings_AppLanguage_Unofficial: String { return self._s[3862]! } public func ChatList_ClearChatConfirmation(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3813]!, self._r[3813]!, [_0]) + return formatWithArgumentRanges(self._s[3863]!, self._r[3863]!, [_0]) } - public var ChatList_AddFolder: String { return self._s[3814]! } - public var Conversation_Location: String { return self._s[3816]! } - public var Appearance_BubbleCorners_AdjustAdjacent: String { return self._s[3817]! } - public var DialogList_AdLabel: String { return self._s[3818]! } + public var ChatList_AddFolder: String { return self._s[3864]! } + public var Conversation_Location: String { return self._s[3866]! } + public var Appearance_BubbleCorners_AdjustAdjacent: String { return self._s[3867]! } + public var DialogList_AdLabel: String { return self._s[3868]! } public func Time_TomorrowAt(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3820]!, self._r[3820]!, [_0]) + return formatWithArgumentRanges(self._s[3870]!, self._r[3870]!, [_0]) } - public var Message_InvoiceLabel: String { return self._s[3821]! } - public var Channel_TooMuchBots: String { return self._s[3822]! } + public var Message_InvoiceLabel: String { return self._s[3871]! } + public var Channel_TooMuchBots: String { return self._s[3872]! } public func Channel_AdminLog_MessageRemovedChannelUsername(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3824]!, self._r[3824]!, [_0]) + return formatWithArgumentRanges(self._s[3874]!, self._r[3874]!, [_0]) } - public var Call_IncomingVideoCall: String { return self._s[3825]! } - public var Conversation_LiveLocation: String { return self._s[3826]! } - public var TwoStepAuth_SetupPasswordEnterPasswordChange: String { return self._s[3827]! } - public var Passport_Identity_EditPassport: String { return self._s[3828]! } - public var Permissions_CellularDataTitle_v0: String { return self._s[3830]! } - public var ChatList_Search_NoResultsFitlerVoice: String { return self._s[3831]! } - public var GroupInfo_Permissions_AddException: String { return self._s[3832]! } - public var Channel_AdminLog_CanInviteUsers: String { return self._s[3834]! } - public var Channel_MessageVideoUpdated: String { return self._s[3835]! } - public var GroupInfo_Permissions_EditingDisabled: String { return self._s[3836]! } - public var AccessDenied_Camera: String { return self._s[3839]! } + public var Call_IncomingVideoCall: String { return self._s[3875]! } + public var Conversation_LiveLocation: String { return self._s[3876]! } + public var TwoStepAuth_SetupPasswordEnterPasswordChange: String { return self._s[3877]! } + public var Passport_Identity_EditPassport: String { return self._s[3878]! } + public var Permissions_CellularDataTitle_v0: String { return self._s[3880]! } + public var ChatList_Search_NoResultsFitlerVoice: String { return self._s[3881]! } + public var GroupInfo_Permissions_AddException: String { return self._s[3882]! } + public var Channel_AdminLog_CanInviteUsers: String { return self._s[3884]! } + public var Channel_MessageVideoUpdated: String { return self._s[3885]! } + public var GroupInfo_Permissions_EditingDisabled: String { return self._s[3886]! } + public var AccessDenied_Camera: String { return self._s[3889]! } public func Target_InviteToGroupConfirmation(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3840]!, self._r[3840]!, [_0]) + return formatWithArgumentRanges(self._s[3890]!, self._r[3890]!, [_0]) } - public var Theme_Context_ChangeColors: String { return self._s[3841]! } - public var PrivacySettings_TwoStepAuth: String { return self._s[3842]! } - public var Privacy_Forwards_PreviewMessageText: String { return self._s[3843]! } - public var Login_CodeExpiredError: String { return self._s[3844]! } - public var State_ConnectingToProxy: String { return self._s[3845]! } - public var TextFormat_Link: String { return self._s[3846]! } - public var Passport_Language_lv: String { return self._s[3847]! } - public var AccessDenied_VoiceMicrophone: String { return self._s[3848]! } - public var WallpaperPreview_SwipeBottomText: String { return self._s[3849]! } - public var ProfilePhoto_SetMainVideo: String { return self._s[3850]! } - public var AutoDownloadSettings_Cellular: String { return self._s[3852]! } - public var ChatSettings_AutoDownloadVoiceMessages: String { return self._s[3853]! } + public var Theme_Context_ChangeColors: String { return self._s[3891]! } + public var PrivacySettings_TwoStepAuth: String { return self._s[3892]! } + public var Privacy_Forwards_PreviewMessageText: String { return self._s[3893]! } + public var Login_CodeExpiredError: String { return self._s[3894]! } + public var State_ConnectingToProxy: String { return self._s[3895]! } + public var TextFormat_Link: String { return self._s[3896]! } + public var Passport_Language_lv: String { return self._s[3897]! } + public var AccessDenied_VoiceMicrophone: String { return self._s[3898]! } + public var WallpaperPreview_SwipeBottomText: String { return self._s[3899]! } + public var ProfilePhoto_SetMainVideo: String { return self._s[3900]! } + public var AutoDownloadSettings_Cellular: String { return self._s[3902]! } + public var ChatSettings_AutoDownloadVoiceMessages: String { return self._s[3903]! } public func Channel_AdminLog_MessageKickedNameUsername(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3854]!, self._r[3854]!, [_1, _2]) - } - public var ChatList_EmptyChatListFilterTitle: String { return self._s[3855]! } - public var Checkout_PayNone: String { return self._s[3856]! } - public var NotificationsSound_Complete: String { return self._s[3858]! } - public var TwoStepAuth_ConfirmEmailCodePlaceholder: String { return self._s[3859]! } - public var AuthSessions_DevicesTitle: String { return self._s[3860]! } - public func DialogList_MultipleTyping(_ _0: String, _ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3861]!, self._r[3861]!, [_0, _1]) - } - public var Message_LiveLocation: String { return self._s[3862]! } - public var Watch_Suggestion_BRB: String { return self._s[3863]! } - public var Channel_BanUser_Title: String { return self._s[3864]! } - public var SettingsSearch_Synonyms_Privacy_Data_Title: String { return self._s[3865]! } - public var Conversation_Dice_u1F3C0: String { return self._s[3866]! } - public var Conversation_ClearSelfHistory: String { return self._s[3867]! } - public var ProfilePhoto_OpenGallery: String { return self._s[3868]! } - public var PrivacySettings_LastSeenTitle: String { return self._s[3869]! } - public var Weekday_Thursday: String { return self._s[3870]! } - public var BroadcastListInfo_AddRecipient: String { return self._s[3871]! } - public var Privacy_ProfilePhoto: String { return self._s[3873]! } - public var StickerPacksSettings_ArchivedPacks_Info: String { return self._s[3874]! } - public func Channel_AdminLog_MessageChangedUnlinkedGroup(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3875]!, self._r[3875]!, [_1, _2]) - } - public var Message_Audio: String { return self._s[3876]! } - public var Conversation_Info: String { return self._s[3877]! } - public var Cache_Videos: String { return self._s[3878]! } - public var Appearance_ThemePreview_ChatList_6_Text: String { return self._s[3879]! } - public var Channel_ErrorAddTooMuch: String { return self._s[3880]! } - public func ChatList_DeleteSecretChatConfirmation(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3881]!, self._r[3881]!, [_0]) - } - public var ChannelMembers_ChannelAdminsTitle: String { return self._s[3883]! } - public var ScheduledMessages_Title: String { return self._s[3885]! } - public var ShareFileTip_Title: String { return self._s[3888]! } - public var Chat_Gifs_TrendingSectionHeader: String { return self._s[3889]! } - public var ChatList_RemoveFolderConfirmation: String { return self._s[3890]! } - public func PUSH_CHAT_MESSAGE_GEOLIVE(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3891]!, self._r[3891]!, [_1, _2]) - } - public var Conversation_ContextViewStats: String { return self._s[3893]! } - public var Channel_DiscussionGroup_SearchPlaceholder: String { return self._s[3894]! } - public var PasscodeSettings_Title: String { return self._s[3895]! } - public var Channel_AdminLog_SendPolls: String { return self._s[3896]! } - public var LastSeen_ALongTimeAgo: String { return self._s[3897]! } - public func PUSH_CHANNEL_MESSAGE_GIF(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3898]!, self._r[3898]!, [_1]) - } - public var SettingsSearch_Synonyms_Notifications_BadgeIncludeMutedChannels: String { return self._s[3899]! } - public var CallFeedback_VideoReasonLowQuality: String { return self._s[3900]! } - public var Conversation_PinnedPreviousMessage: String { return self._s[3901]! } - public var SocksProxySetup_AddProxyTitle: String { return self._s[3902]! } - public var Passport_Identity_AddInternalPassport: String { return self._s[3903]! } - public func ChatList_RemovedFromFolderTooltip(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { return formatWithArgumentRanges(self._s[3904]!, self._r[3904]!, [_1, _2]) } - public func Conversation_SetReminder_RemindToday(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3905]!, self._r[3905]!, [_0]) + public var ChatList_EmptyChatListFilterTitle: String { return self._s[3905]! } + public var Checkout_PayNone: String { return self._s[3906]! } + public var NotificationsSound_Complete: String { return self._s[3908]! } + public var TwoStepAuth_ConfirmEmailCodePlaceholder: String { return self._s[3909]! } + public var InviteLink_CreateInfo: String { return self._s[3910]! } + public var AuthSessions_DevicesTitle: String { return self._s[3911]! } + public func DialogList_MultipleTyping(_ _0: String, _ _1: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3912]!, self._r[3912]!, [_0, _1]) } - public var Passport_Identity_GenderFemale: String { return self._s[3906]! } - public var Location_ProximityNotification_DistanceKM: String { return self._s[3909]! } - public var ConvertToSupergroup_HelpTitle: String { return self._s[3910]! } - public var VoiceChat_Audio: String { return self._s[3911]! } - public var SharedMedia_TitleAll: String { return self._s[3912]! } - public var Settings_Context_Logout: String { return self._s[3913]! } - public var GroupInfo_SetGroupPhotoDelete: String { return self._s[3915]! } - public var Settings_About_Title: String { return self._s[3916]! } - public var StickerSettings_ContextHide: String { return self._s[3917]! } + public var Message_LiveLocation: String { return self._s[3913]! } + public var Watch_Suggestion_BRB: String { return self._s[3914]! } + public var Channel_BanUser_Title: String { return self._s[3915]! } + public var SettingsSearch_Synonyms_Privacy_Data_Title: String { return self._s[3916]! } + public var Conversation_Dice_u1F3C0: String { return self._s[3917]! } + public var Conversation_ClearSelfHistory: String { return self._s[3918]! } + public var ProfilePhoto_OpenGallery: String { return self._s[3919]! } + public var PrivacySettings_LastSeenTitle: String { return self._s[3920]! } + public var Weekday_Thursday: String { return self._s[3921]! } + public var BroadcastListInfo_AddRecipient: String { return self._s[3922]! } + public var Privacy_ProfilePhoto: String { return self._s[3924]! } + public var StickerPacksSettings_ArchivedPacks_Info: String { return self._s[3925]! } + public func Channel_AdminLog_MessageChangedUnlinkedGroup(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3926]!, self._r[3926]!, [_1, _2]) + } + public var Message_Audio: String { return self._s[3927]! } + public var Conversation_Info: String { return self._s[3928]! } + public var Cache_Videos: String { return self._s[3929]! } + public var Appearance_ThemePreview_ChatList_6_Text: String { return self._s[3930]! } + public var Channel_ErrorAddTooMuch: String { return self._s[3931]! } + public func ChatList_DeleteSecretChatConfirmation(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3932]!, self._r[3932]!, [_0]) + } + public var ChannelMembers_ChannelAdminsTitle: String { return self._s[3934]! } + public var ScheduledMessages_Title: String { return self._s[3936]! } + public var ShareFileTip_Title: String { return self._s[3939]! } + public var Chat_Gifs_TrendingSectionHeader: String { return self._s[3940]! } + public var ChatList_RemoveFolderConfirmation: String { return self._s[3941]! } + public func PUSH_CHAT_MESSAGE_GEOLIVE(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3942]!, self._r[3942]!, [_1, _2]) + } + public var Conversation_ContextViewStats: String { return self._s[3944]! } + public var Channel_DiscussionGroup_SearchPlaceholder: String { return self._s[3945]! } + public var PasscodeSettings_Title: String { return self._s[3946]! } + public var Channel_AdminLog_SendPolls: String { return self._s[3947]! } + public var LastSeen_ALongTimeAgo: String { return self._s[3948]! } + public func PUSH_CHANNEL_MESSAGE_GIF(_ _1: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3949]!, self._r[3949]!, [_1]) + } + public var SettingsSearch_Synonyms_Notifications_BadgeIncludeMutedChannels: String { return self._s[3950]! } + public var CallFeedback_VideoReasonLowQuality: String { return self._s[3951]! } + public var Conversation_PinnedPreviousMessage: String { return self._s[3952]! } + public var SocksProxySetup_AddProxyTitle: String { return self._s[3953]! } + public var Passport_Identity_AddInternalPassport: String { return self._s[3954]! } + public func ChatList_RemovedFromFolderTooltip(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3955]!, self._r[3955]!, [_1, _2]) + } + public func Conversation_SetReminder_RemindToday(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[3956]!, self._r[3956]!, [_0]) + } + public var Passport_Identity_GenderFemale: String { return self._s[3957]! } + public var Location_ProximityNotification_DistanceKM: String { return self._s[3960]! } + public var ConvertToSupergroup_HelpTitle: String { return self._s[3961]! } + public var VoiceChat_Audio: String { return self._s[3962]! } + public var SharedMedia_TitleAll: String { return self._s[3963]! } + public var Settings_Context_Logout: String { return self._s[3964]! } + public var GroupInfo_SetGroupPhotoDelete: String { return self._s[3966]! } + public var Settings_About_Title: String { return self._s[3967]! } + public var StickerSettings_ContextHide: String { return self._s[3968]! } public func AutoDownloadSettings_UpTo(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3918]!, self._r[3918]!, [_0]) + return formatWithArgumentRanges(self._s[3969]!, self._r[3969]!, [_0]) } public func Conversation_LiveLocationYouAndOther(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3919]!, self._r[3919]!, [_0]) + return formatWithArgumentRanges(self._s[3970]!, self._r[3970]!, [_0]) } - public var Common_Cancel: String { return self._s[3921]! } - public var CallFeedback_Title: String { return self._s[3923]! } + public var Common_Cancel: String { return self._s[3972]! } + public var CallFeedback_Title: String { return self._s[3974]! } public func Notification_PinnedContactMessage(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3924]!, self._r[3924]!, [_0]) + return formatWithArgumentRanges(self._s[3975]!, self._r[3975]!, [_0]) } - public var Activity_UploadingVideoMessage: String { return self._s[3925]! } - public var MediaPicker_Send: String { return self._s[3926]! } - public var PasscodeSettings_AutoLock_IfAwayFor_1minute: String { return self._s[3927]! } - public var Conversation_LiveLocationYou: String { return self._s[3928]! } - public var Notifications_ExceptionsUnmuted: String { return self._s[3929]! } + public var Activity_UploadingVideoMessage: String { return self._s[3976]! } + public var MediaPicker_Send: String { return self._s[3977]! } + public var PasscodeSettings_AutoLock_IfAwayFor_1minute: String { return self._s[3978]! } + public var Conversation_LiveLocationYou: String { return self._s[3979]! } + public var Notifications_ExceptionsUnmuted: String { return self._s[3980]! } public func Channel_AdminLog_MessageGroupPreHistoryHidden(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3930]!, self._r[3930]!, [_0]) + return formatWithArgumentRanges(self._s[3981]!, self._r[3981]!, [_0]) } public func PUSH_CHAT_ADD_YOU(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3931]!, self._r[3931]!, [_1, _2]) + return formatWithArgumentRanges(self._s[3982]!, self._r[3982]!, [_1, _2]) } - public var Conversation_ViewBackground: String { return self._s[3932]! } - public var ChatSettings_PrivateChats: String { return self._s[3935]! } - public var Conversation_ErrorInaccessibleMessage: String { return self._s[3936]! } - public var Appearance_ThemeNight: String { return self._s[3937]! } - public var Common_Search: String { return self._s[3938]! } - public var TwoStepAuth_ReEnterPasswordTitle: String { return self._s[3939]! } - public var ChangePhoneNumberNumber_Help: String { return self._s[3941]! } - public var Stickers_SuggestAdded: String { return self._s[3942]! } - public var Conversation_DiscardVoiceMessageDescription: String { return self._s[3945]! } - public var NetworkUsageSettings_Cellular: String { return self._s[3946]! } - public var CheckoutInfo_Title: String { return self._s[3947]! } - public var Conversation_ShareBotLocationConfirmationTitle: String { return self._s[3948]! } - public var Channel_BotDoesntSupportGroups: String { return self._s[3949]! } + public var Conversation_ViewBackground: String { return self._s[3983]! } + public var ChatSettings_PrivateChats: String { return self._s[3986]! } + public var Conversation_ErrorInaccessibleMessage: String { return self._s[3987]! } + public var Appearance_ThemeNight: String { return self._s[3988]! } + public var Common_Search: String { return self._s[3989]! } + public var TwoStepAuth_ReEnterPasswordTitle: String { return self._s[3990]! } + public var ChangePhoneNumberNumber_Help: String { return self._s[3992]! } + public var InviteLink_QRCode_Share: String { return self._s[3993]! } + public var Stickers_SuggestAdded: String { return self._s[3994]! } + public var Conversation_DiscardVoiceMessageDescription: String { return self._s[3997]! } + public var NetworkUsageSettings_Cellular: String { return self._s[3998]! } + public var CheckoutInfo_Title: String { return self._s[3999]! } + public var Conversation_ShareBotLocationConfirmationTitle: String { return self._s[4000]! } + public var Channel_BotDoesntSupportGroups: String { return self._s[4001]! } public func DialogList_SingleRecordingAudioSuffix(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3950]!, self._r[3950]!, [_0]) + return formatWithArgumentRanges(self._s[4002]!, self._r[4002]!, [_0]) } - public var MaskStickerSettings_Info: String { return self._s[3952]! } - public var GroupRemoved_DeleteUser: String { return self._s[3954]! } - public var Contacts_ShareTelegram: String { return self._s[3955]! } - public var Group_UpgradeNoticeText1: String { return self._s[3956]! } + public var MaskStickerSettings_Info: String { return self._s[4004]! } + public var GroupRemoved_DeleteUser: String { return self._s[4006]! } + public var Contacts_ShareTelegram: String { return self._s[4007]! } + public var Group_UpgradeNoticeText1: String { return self._s[4008]! } public func PUSH_PHONE_CALL_REQUEST(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3957]!, self._r[3957]!, [_1]) + return formatWithArgumentRanges(self._s[4009]!, self._r[4009]!, [_1]) } - public var PrivacyLastSeenSettings_Title: String { return self._s[3958]! } - public var SettingsSearch_Synonyms_Support: String { return self._s[3962]! } - public var PhotoEditor_TintTool: String { return self._s[3963]! } - public var GroupPermission_NoSendPolls: String { return self._s[3965]! } - public var NotificationsSound_None: String { return self._s[3966]! } + public var PrivacyLastSeenSettings_Title: String { return self._s[4010]! } + public var SettingsSearch_Synonyms_Support: String { return self._s[4014]! } + public var PhotoEditor_TintTool: String { return self._s[4015]! } + public var GroupPermission_NoSendPolls: String { return self._s[4017]! } + public var NotificationsSound_None: String { return self._s[4018]! } public func LOCAL_CHANNEL_MESSAGE_FWDS(_ _1: String, _ _2: Int) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3967]!, self._r[3967]!, [_1, "\(_2)"]) + return formatWithArgumentRanges(self._s[4019]!, self._r[4019]!, [_1, "\(_2)"]) } - public var CheckoutInfo_ShippingInfoCityPlaceholder: String { return self._s[3969]! } - public var ExplicitContent_AlertChannel: String { return self._s[3971]! } - public var Conversation_ClousStorageInfo_Description1: String { return self._s[3972]! } - public var Contacts_SortedByPresence: String { return self._s[3973]! } - public var WallpaperSearch_ColorGray: String { return self._s[3974]! } - public var Channel_AdminLogFilter_EventsNewSubscribers: String { return self._s[3975]! } - public var Conversation_ReportSpam: String { return self._s[3976]! } - public var ChatList_Search_NoResultsFilter: String { return self._s[3979]! } - public var WallpaperSearch_ColorBlack: String { return self._s[3980]! } - public var ArchivedChats_IntroTitle3: String { return self._s[3981]! } + public var CheckoutInfo_ShippingInfoCityPlaceholder: String { return self._s[4021]! } + public var ExplicitContent_AlertChannel: String { return self._s[4023]! } + public var Conversation_ClousStorageInfo_Description1: String { return self._s[4024]! } + public var Contacts_SortedByPresence: String { return self._s[4025]! } + public var WallpaperSearch_ColorGray: String { return self._s[4026]! } + public var Channel_AdminLogFilter_EventsNewSubscribers: String { return self._s[4027]! } + public var Conversation_ReportSpam: String { return self._s[4028]! } + public var ChatList_Search_NoResultsFilter: String { return self._s[4031]! } + public var WallpaperSearch_ColorBlack: String { return self._s[4032]! } + public var ArchivedChats_IntroTitle3: String { return self._s[4033]! } + public var InviteLink_DeleteAllRevokedLinksAlert_Action: String { return self._s[4034]! } public func VoiceChat_PeerJoinedText(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[3982]!, self._r[3982]!, [_0]) + return formatWithArgumentRanges(self._s[4035]!, self._r[4035]!, [_0]) } - public var Conversation_DefaultRestrictedText: String { return self._s[3983]! } - public var Settings_Devices: String { return self._s[3984]! } - public var Call_AudioRouteSpeaker: String { return self._s[3985]! } - public var GroupInfo_InviteLink_CopyLink: String { return self._s[3986]! } - public var Passport_Address_Country: String { return self._s[3988]! } - public var Cache_MaximumCacheSize: String { return self._s[3989]! } - public var Chat_PanelHidePinnedMessages: String { return self._s[3990]! } - public var Notifications_Badge_IncludePublicGroups: String { return self._s[3991]! } - public var ChatSettings_AutoDownloadUsingWiFi: String { return self._s[3993]! } - public var Login_TermsOfServiceLabel: String { return self._s[3994]! } - public var Calls_NoMissedCallsPlacehoder: String { return self._s[3995]! } - public var SocksProxySetup_RequiredCredentials: String { return self._s[3996]! } - public var VoiceOver_MessageContextOpenMessageMenu: String { return self._s[3997]! } - public var AutoNightTheme_ScheduledFrom: String { return self._s[3998]! } - public var ChatSettings_AutoDownloadDocuments: String { return self._s[3999]! } - public var ConvertToSupergroup_Note: String { return self._s[4001]! } - public var Settings_SetNewProfilePhotoOrVideo: String { return self._s[4002]! } - public var PrivacySettings_PasscodeAndTouchId: String { return self._s[4003]! } - public var Common_More: String { return self._s[4004]! } - public var ShareMenu_SelectChats: String { return self._s[4006]! } + public var Conversation_DefaultRestrictedText: String { return self._s[4036]! } + public var Settings_Devices: String { return self._s[4037]! } + public var Call_AudioRouteSpeaker: String { return self._s[4038]! } + public var GroupInfo_InviteLink_CopyLink: String { return self._s[4039]! } + public var Passport_Address_Country: String { return self._s[4041]! } + public var Cache_MaximumCacheSize: String { return self._s[4042]! } + public var Chat_PanelHidePinnedMessages: String { return self._s[4043]! } + public var Notifications_Badge_IncludePublicGroups: String { return self._s[4044]! } + public var ChatSettings_AutoDownloadUsingWiFi: String { return self._s[4046]! } + public var Login_TermsOfServiceLabel: String { return self._s[4047]! } + public var Calls_NoMissedCallsPlacehoder: String { return self._s[4048]! } + public var SocksProxySetup_RequiredCredentials: String { return self._s[4049]! } + public var VoiceOver_MessageContextOpenMessageMenu: String { return self._s[4050]! } + public var AutoNightTheme_ScheduledFrom: String { return self._s[4051]! } + public var ChatSettings_AutoDownloadDocuments: String { return self._s[4052]! } + public var ConvertToSupergroup_Note: String { return self._s[4054]! } + public var Settings_SetNewProfilePhotoOrVideo: String { return self._s[4055]! } + public var PrivacySettings_PasscodeAndTouchId: String { return self._s[4056]! } + public var Common_More: String { return self._s[4057]! } + public var ShareMenu_SelectChats: String { return self._s[4059]! } public func Conversation_ScheduleMessage_SendToday(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4007]!, self._r[4007]!, [_0]) + return formatWithArgumentRanges(self._s[4060]!, self._r[4060]!, [_0]) } public func Channel_AdminLog_MessageRemovedGroupStickerPack(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4008]!, self._r[4008]!, [_0]) + return formatWithArgumentRanges(self._s[4061]!, self._r[4061]!, [_0]) } - public var Contacts_PermissionsKeepDisabled: String { return self._s[4010]! } + public var Contacts_PermissionsKeepDisabled: String { return self._s[4063]! } public func Call_ParticipantVersionOutdatedError(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4011]!, self._r[4011]!, [_0]) + return formatWithArgumentRanges(self._s[4064]!, self._r[4064]!, [_0]) } - public var WatchRemote_AlertOpen: String { return self._s[4012]! } + public var WatchRemote_AlertOpen: String { return self._s[4065]! } public func PUSH_CHAT_ADD_MEMBER(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4013]!, self._r[4013]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[4066]!, self._r[4066]!, [_1, _2, _3]) } - public var Channel_Members_AddMembersHelp: String { return self._s[4014]! } - public var Shortcut_SwitchAccount: String { return self._s[4015]! } - public var Map_LiveLocationFor8Hours: String { return self._s[4016]! } + public var Channel_Members_AddMembersHelp: String { return self._s[4067]! } + public var Shortcut_SwitchAccount: String { return self._s[4068]! } + public var Map_LiveLocationFor8Hours: String { return self._s[4069]! } public func AutoNightTheme_AutomaticHelp(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4017]!, self._r[4017]!, [_0]) + return formatWithArgumentRanges(self._s[4070]!, self._r[4070]!, [_0]) } - public var Compose_NewGroupTitle: String { return self._s[4018]! } - public var Call_VoiceOver_VoiceCallOutgoing: String { return self._s[4019]! } - public var DialogList_You: String { return self._s[4020]! } - public var ReportPeer_ReasonViolence: String { return self._s[4021]! } + public var Compose_NewGroupTitle: String { return self._s[4071]! } + public var Call_VoiceOver_VoiceCallOutgoing: String { return self._s[4072]! } + public var DialogList_You: String { return self._s[4073]! } + public var ReportPeer_ReasonViolence: String { return self._s[4074]! } public func PUSH_CHANNEL_MESSAGE_STICKER(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4022]!, self._r[4022]!, [_1, _2]) + return formatWithArgumentRanges(self._s[4075]!, self._r[4075]!, [_1, _2]) } - public var VoiceChat_Reconnecting: String { return self._s[4024]! } - public var KeyCommand_ScrollDown: String { return self._s[4027]! } - public var ChatSettings_DownloadInBackground: String { return self._s[4028]! } - public var Wallpaper_ResetWallpapers: String { return self._s[4029]! } - public var Channel_BanList_RestrictedTitle: String { return self._s[4030]! } - public var ArchivedChats_IntroText3: String { return self._s[4031]! } - public var HashtagSearch_AllChats: String { return self._s[4033]! } - public var VoiceChat_EndVoiceChat: String { return self._s[4034]! } - public var Channel_Info_BlackList: String { return self._s[4036]! } - public var Contacts_SearchUsersAndGroupsLabel: String { return self._s[4037]! } - public var PrivacyPhoneNumberSettings_DiscoveryHeader: String { return self._s[4038]! } - public var Paint_Neon: String { return self._s[4040]! } - public var SettingsSearch_Synonyms_AppLanguage: String { return self._s[4041]! } - public var AutoDownloadSettings_AutoDownload: String { return self._s[4042]! } + public var VoiceChat_Reconnecting: String { return self._s[4077]! } + public var KeyCommand_ScrollDown: String { return self._s[4080]! } + public var ChatSettings_DownloadInBackground: String { return self._s[4081]! } + public var Wallpaper_ResetWallpapers: String { return self._s[4082]! } + public var Channel_BanList_RestrictedTitle: String { return self._s[4083]! } + public var ArchivedChats_IntroText3: String { return self._s[4084]! } + public var HashtagSearch_AllChats: String { return self._s[4086]! } + public var VoiceChat_EndVoiceChat: String { return self._s[4087]! } + public var Channel_Info_BlackList: String { return self._s[4089]! } + public var Contacts_SearchUsersAndGroupsLabel: String { return self._s[4090]! } + public var PrivacyPhoneNumberSettings_DiscoveryHeader: String { return self._s[4091]! } + public var Paint_Neon: String { return self._s[4093]! } + public var SettingsSearch_Synonyms_AppLanguage: String { return self._s[4094]! } + public var AutoDownloadSettings_AutoDownload: String { return self._s[4095]! } public func Notification_PinnedVideoMessage(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4044]!, self._r[4044]!, [_0]) + return formatWithArgumentRanges(self._s[4097]!, self._r[4097]!, [_0]) } - public var Map_StopLiveLocation: String { return self._s[4045]! } - public var SettingsSearch_Synonyms_Data_SaveEditedPhotos: String { return self._s[4046]! } - public var Channel_Username_InvalidCharacters: String { return self._s[4047]! } - public var InstantPage_Reference: String { return self._s[4048]! } - public var ChatList_HideAction: String { return self._s[4050]! } - public var Conversation_FileICloudDrive: String { return self._s[4052]! } + public var Map_StopLiveLocation: String { return self._s[4098]! } + public var SettingsSearch_Synonyms_Data_SaveEditedPhotos: String { return self._s[4099]! } + public var Channel_Username_InvalidCharacters: String { return self._s[4100]! } + public var InstantPage_Reference: String { return self._s[4101]! } + public var ChatList_HideAction: String { return self._s[4103]! } + public var Conversation_FileICloudDrive: String { return self._s[4105]! } public func PUSH_PINNED_GEOLIVE(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4053]!, self._r[4053]!, [_1]) + return formatWithArgumentRanges(self._s[4106]!, self._r[4106]!, [_1]) } - public var Passport_PasswordReset: String { return self._s[4055]! } - public var ChatList_Context_UnhideArchive: String { return self._s[4057]! } - public var ConvertToSupergroup_HelpText: String { return self._s[4058]! } - public var Calls_AddTab: String { return self._s[4059]! } - public var TwoStepAuth_ConfirmEmailResendCode: String { return self._s[4060]! } - public var SettingsSearch_Synonyms_Stickers_SuggestStickers: String { return self._s[4061]! } - public var Privacy_GroupsAndChannels: String { return self._s[4064]! } - public var AutoNightTheme_Disabled: String { return self._s[4065]! } - public var CreatePoll_MultipleChoice: String { return self._s[4066]! } + public var Passport_PasswordReset: String { return self._s[4108]! } + public var ChatList_Context_UnhideArchive: String { return self._s[4110]! } + public var ConvertToSupergroup_HelpText: String { return self._s[4111]! } + public var Calls_AddTab: String { return self._s[4112]! } + public var TwoStepAuth_ConfirmEmailResendCode: String { return self._s[4113]! } + public var SettingsSearch_Synonyms_Stickers_SuggestStickers: String { return self._s[4114]! } + public var Privacy_GroupsAndChannels: String { return self._s[4117]! } + public var AutoNightTheme_Disabled: String { return self._s[4118]! } + public var CreatePoll_MultipleChoice: String { return self._s[4119]! } public func PINNED_INVOICE(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4067]!, self._r[4067]!, [_1]) + return formatWithArgumentRanges(self._s[4120]!, self._r[4120]!, [_1]) } - public var Watch_Bot_Restart: String { return self._s[4069]! } + public var Watch_Bot_Restart: String { return self._s[4122]! } public func Conversation_Kilobytes(_ _0: Int) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4070]!, self._r[4070]!, ["\(_0)"]) + return formatWithArgumentRanges(self._s[4123]!, self._r[4123]!, ["\(_0)"]) } - public var GroupInfo_ScamGroupWarning: String { return self._s[4072]! } - public var Conversation_EditingMessagePanelMedia: String { return self._s[4073]! } - public var Appearance_PreviewIncomingText: String { return self._s[4074]! } - public var ChatSettings_WidgetSettings: String { return self._s[4075]! } - public var Notifications_ChannelNotificationsExceptionsHelp: String { return self._s[4076]! } - public var ChatList_UndoArchiveRevealedTitle: String { return self._s[4078]! } - public var Stats_GroupOverview: String { return self._s[4080]! } - public var ScheduledMessages_EditTime: String { return self._s[4083]! } - public var Month_GenFebruary: String { return self._s[4084]! } - public var ChatList_AutoarchiveSuggestion_OpenSettings: String { return self._s[4085]! } - public var Stickers_ClearRecent: String { return self._s[4086]! } - public var TwoStepAuth_EnterPasswordPassword: String { return self._s[4087]! } - public var Stats_Message_PublicShares: String { return self._s[4088]! } + public var GroupInfo_ScamGroupWarning: String { return self._s[4125]! } + public var Conversation_EditingMessagePanelMedia: String { return self._s[4126]! } + public var Appearance_PreviewIncomingText: String { return self._s[4127]! } + public var ChatSettings_WidgetSettings: String { return self._s[4128]! } + public var Notifications_ChannelNotificationsExceptionsHelp: String { return self._s[4129]! } + public var ChatList_UndoArchiveRevealedTitle: String { return self._s[4131]! } + public var Stats_GroupOverview: String { return self._s[4133]! } + public var ScheduledMessages_EditTime: String { return self._s[4136]! } + public var Month_GenFebruary: String { return self._s[4137]! } + public var ChatList_AutoarchiveSuggestion_OpenSettings: String { return self._s[4138]! } + public var Stickers_ClearRecent: String { return self._s[4139]! } + public var InviteLink_Create_UsersLimitNumberOfUsersUnlimited: String { return self._s[4140]! } + public var TwoStepAuth_EnterPasswordPassword: String { return self._s[4141]! } + public var Stats_Message_PublicShares: String { return self._s[4142]! } public func Checkout_PayPrice(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4089]!, self._r[4089]!, [_0]) + return formatWithArgumentRanges(self._s[4143]!, self._r[4143]!, [_0]) } - public var Login_TermsOfServiceSignupDecline: String { return self._s[4090]! } - public var CheckoutInfo_ErrorCityInvalid: String { return self._s[4091]! } - public var VoiceOver_Chat_PlayHint: String { return self._s[4092]! } - public var ChatAdmins_AllMembersAreAdminsOffHelp: String { return self._s[4093]! } - public var CheckoutInfo_ShippingInfoTitle: String { return self._s[4095]! } - public var CreatePoll_Create: String { return self._s[4096]! } - public var ChatList_Search_FilterLinks: String { return self._s[4097]! } - public var Your_cards_number_is_invalid: String { return self._s[4098]! } - public var Month_ShortApril: String { return self._s[4099]! } - public var SocksProxySetup_UseForCalls: String { return self._s[4100]! } - public var Conversation_EditingCaptionPanelTitle: String { return self._s[4101]! } - public var VoiceChat_UnmuteForMe: String { return self._s[4102]! } - public var SocksProxySetup_Status: String { return self._s[4103]! } - public var ChannelInfo_DeleteGroupConfirmation: String { return self._s[4104]! } - public var ChatListFolder_CategoryBots: String { return self._s[4105]! } - public var Passport_FieldIdentitySelfieHelp: String { return self._s[4107]! } - public var GroupInfo_BroadcastListNamePlaceholder: String { return self._s[4108]! } - public var Chat_PinnedListPreview_UnpinAllMessages: String { return self._s[4109]! } - public var Wallpaper_ResetWallpapersInfo: String { return self._s[4110]! } - public var Conversation_TitleUnmute: String { return self._s[4111]! } - public var Group_Setup_TypeHeader: String { return self._s[4112]! } - public var Stats_ViewsPerPost: String { return self._s[4113]! } - public var CheckoutInfo_ShippingInfoCountry: String { return self._s[4114]! } - public var Passport_Identity_TranslationHelp: String { return self._s[4115]! } + public var Login_TermsOfServiceSignupDecline: String { return self._s[4144]! } + public var CheckoutInfo_ErrorCityInvalid: String { return self._s[4145]! } + public var VoiceOver_Chat_PlayHint: String { return self._s[4146]! } + public var ChatAdmins_AllMembersAreAdminsOffHelp: String { return self._s[4147]! } + public var CheckoutInfo_ShippingInfoTitle: String { return self._s[4149]! } + public var CreatePoll_Create: String { return self._s[4150]! } + public var ChatList_Search_FilterLinks: String { return self._s[4151]! } + public var Your_cards_number_is_invalid: String { return self._s[4152]! } + public var Month_ShortApril: String { return self._s[4153]! } + public var SocksProxySetup_UseForCalls: String { return self._s[4154]! } + public var Conversation_EditingCaptionPanelTitle: String { return self._s[4155]! } + public var VoiceChat_UnmuteForMe: String { return self._s[4156]! } + public var SocksProxySetup_Status: String { return self._s[4157]! } + public var ChannelInfo_DeleteGroupConfirmation: String { return self._s[4158]! } + public var ChatListFolder_CategoryBots: String { return self._s[4159]! } + public var Passport_FieldIdentitySelfieHelp: String { return self._s[4161]! } + public var GroupInfo_BroadcastListNamePlaceholder: String { return self._s[4162]! } + public var Chat_PinnedListPreview_UnpinAllMessages: String { return self._s[4163]! } + public var Wallpaper_ResetWallpapersInfo: String { return self._s[4164]! } + public var Conversation_TitleUnmute: String { return self._s[4165]! } + public var Group_Setup_TypeHeader: String { return self._s[4166]! } + public var Stats_ViewsPerPost: String { return self._s[4167]! } + public var CheckoutInfo_ShippingInfoCountry: String { return self._s[4168]! } + public var Passport_Identity_TranslationHelp: String { return self._s[4169]! } public func PUSH_CHANNEL_MESSAGE_FWD(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4116]!, self._r[4116]!, [_1]) + return formatWithArgumentRanges(self._s[4170]!, self._r[4170]!, [_1]) } - public var GroupInfo_Administrators_Title: String { return self._s[4117]! } + public var GroupInfo_Administrators_Title: String { return self._s[4171]! } public func Channel_AdminLog_MessageRankName(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4118]!, self._r[4118]!, [_1, _2]) + return formatWithArgumentRanges(self._s[4172]!, self._r[4172]!, [_1, _2]) } public func PUSH_CHAT_MESSAGE_POLL(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4119]!, self._r[4119]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[4173]!, self._r[4173]!, [_1, _2, _3]) } - public var CheckoutInfo_ShippingInfoState: String { return self._s[4120]! } - public var Passport_Language_my: String { return self._s[4122]! } - public var PrivacyLastSeenSettings_AlwaysShareWith_Title: String { return self._s[4123]! } - public var Map_PlacesNearby: String { return self._s[4124]! } - public var Channel_About_Help: String { return self._s[4125]! } - public var LogoutOptions_AddAccountTitle: String { return self._s[4126]! } - public var ChatSettings_AutomaticAudioDownload: String { return self._s[4127]! } - public var Channel_Username_Title: String { return self._s[4128]! } - public var Activity_RecordingVideoMessage: String { return self._s[4129]! } + public var CheckoutInfo_ShippingInfoState: String { return self._s[4174]! } + public var Passport_Language_my: String { return self._s[4176]! } + public var PrivacyLastSeenSettings_AlwaysShareWith_Title: String { return self._s[4177]! } + public var Map_PlacesNearby: String { return self._s[4178]! } + public var Channel_About_Help: String { return self._s[4179]! } + public var LogoutOptions_AddAccountTitle: String { return self._s[4180]! } + public var ChatSettings_AutomaticAudioDownload: String { return self._s[4181]! } + public var Channel_Username_Title: String { return self._s[4182]! } + public var Activity_RecordingVideoMessage: String { return self._s[4183]! } public func StickerPackActionInfo_RemovedText(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4130]!, self._r[4130]!, [_0]) + return formatWithArgumentRanges(self._s[4184]!, self._r[4184]!, [_0]) } - public var CheckoutInfo_ShippingInfoCity: String { return self._s[4131]! } - public var Passport_DiscardMessageDescription: String { return self._s[4132]! } - public var Conversation_LinkDialogOpen: String { return self._s[4133]! } - public var ChatList_Context_HideArchive: String { return self._s[4134]! } + public var CheckoutInfo_ShippingInfoCity: String { return self._s[4185]! } + public var Passport_DiscardMessageDescription: String { return self._s[4186]! } + public var Conversation_LinkDialogOpen: String { return self._s[4187]! } + public var ChatList_Context_HideArchive: String { return self._s[4188]! } public func Message_AuthorPinnedGame(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4135]!, self._r[4135]!, [_0]) + return formatWithArgumentRanges(self._s[4189]!, self._r[4189]!, [_0]) } - public var Privacy_GroupsAndChannels_CustomShareHelp: String { return self._s[4136]! } - public var Conversation_Admin: String { return self._s[4137]! } - public var DialogList_TabTitle: String { return self._s[4138]! } + public var Privacy_GroupsAndChannels_CustomShareHelp: String { return self._s[4190]! } + public var Conversation_Admin: String { return self._s[4191]! } + public var DialogList_TabTitle: String { return self._s[4192]! } public func PUSH_CHAT_ALBUM(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4139]!, self._r[4139]!, [_1, _2]) + return formatWithArgumentRanges(self._s[4193]!, self._r[4193]!, [_1, _2]) } - public var Notifications_PermissionsUnreachableText: String { return self._s[4140]! } - public var Passport_Identity_GenderMale: String { return self._s[4142]! } - public var SettingsSearch_Synonyms_Privacy_BlockedUsers: String { return self._s[4144]! } - public var PhoneNumberHelp_Alert: String { return self._s[4145]! } - public var EnterPasscode_EnterNewPasscodeChange: String { return self._s[4146]! } - public var Notifications_InAppNotifications: String { return self._s[4147]! } + public var Notifications_PermissionsUnreachableText: String { return self._s[4194]! } + public var Passport_Identity_GenderMale: String { return self._s[4196]! } + public var SettingsSearch_Synonyms_Privacy_BlockedUsers: String { return self._s[4198]! } + public var PhoneNumberHelp_Alert: String { return self._s[4199]! } + public var EnterPasscode_EnterNewPasscodeChange: String { return self._s[4200]! } + public var Notifications_InAppNotifications: String { return self._s[4201]! } public func Update_AppVersion(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4148]!, self._r[4148]!, [_0]) + return formatWithArgumentRanges(self._s[4202]!, self._r[4202]!, [_0]) } - public var Notification_VideoCallOutgoing: String { return self._s[4149]! } - public var Login_InvalidCodeError: String { return self._s[4150]! } - public var Conversation_PrivateChannelTimeLimitedAlertJoin: String { return self._s[4151]! } + public var Notification_VideoCallOutgoing: String { return self._s[4203]! } + public var Login_InvalidCodeError: String { return self._s[4204]! } + public var Conversation_PrivateChannelTimeLimitedAlertJoin: String { return self._s[4205]! } public func LastSeen_TodayAt(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4153]!, self._r[4153]!, [_0]) + return formatWithArgumentRanges(self._s[4207]!, self._r[4207]!, [_0]) } - public var Conversation_InputTextCaptionPlaceholder: String { return self._s[4154]! } - public var ReportPeer_Report: String { return self._s[4155]! } - public var Camera_FlashOff: String { return self._s[4158]! } - public var Conversation_InputTextBroadcastPlaceholder: String { return self._s[4161]! } - public var PrivacyPolicy_DeclineTitle: String { return self._s[4164]! } - public var SettingsSearch_Synonyms_Privacy_PasscodeAndTouchId: String { return self._s[4165]! } - public var Passport_FieldEmail: String { return self._s[4166]! } + public var Conversation_InputTextCaptionPlaceholder: String { return self._s[4208]! } + public var ReportPeer_Report: String { return self._s[4209]! } + public var Camera_FlashOff: String { return self._s[4212]! } + public var Conversation_InputTextBroadcastPlaceholder: String { return self._s[4215]! } + public var PrivacyPolicy_DeclineTitle: String { return self._s[4218]! } + public var SettingsSearch_Synonyms_Privacy_PasscodeAndTouchId: String { return self._s[4219]! } + public var Passport_FieldEmail: String { return self._s[4220]! } public func Channel_AdminLog_MessageKickedName(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4167]!, self._r[4167]!, [_1]) + return formatWithArgumentRanges(self._s[4221]!, self._r[4221]!, [_1]) } - public var Notifications_ExceptionsResetToDefaults: String { return self._s[4168]! } - public var PeerInfo_PaneVoiceAndVideo: String { return self._s[4169]! } - public var Group_OwnershipTransfer_Title: String { return self._s[4170]! } - public var Conversation_DefaultRestrictedInline: String { return self._s[4171]! } - public var Login_PhoneNumberHelp: String { return self._s[4173]! } - public var Channel_AdminLogFilter_EventsNewMembers: String { return self._s[4174]! } - public var Conversation_PinnedQuiz: String { return self._s[4175]! } - public var CreateGroup_SoftUserLimitAlert: String { return self._s[4176]! } - public var Login_PhoneNumberAlreadyAuthorizedSwitch: String { return self._s[4177]! } - public var Group_MessagePhotoUpdated: String { return self._s[4178]! } - public var LoginPassword_PasswordPlaceholder: String { return self._s[4179]! } - public var Passport_Identity_Translations: String { return self._s[4181]! } - public var ChatAdmins_AllMembersAreAdmins: String { return self._s[4182]! } - public var ChannelInfo_DeleteChannel: String { return self._s[4184]! } - public var PasscodeSettings_HelpBottom: String { return self._s[4185]! } - public var Channel_Members_AddMembers: String { return self._s[4186]! } - public var AutoDownloadSettings_LastDelimeter: String { return self._s[4187]! } - public var Notification_Exceptions_DeleteAllConfirmation: String { return self._s[4189]! } - public var Conversation_HoldForAudio: String { return self._s[4190]! } - public var Media_LimitedAccessChangeSettings: String { return self._s[4192]! } - public var Watch_LastSeen_Lately: String { return self._s[4193]! } - public var ChatList_Context_MarkAsRead: String { return self._s[4194]! } - public var Conversation_PinnedMessage: String { return self._s[4195]! } - public var SettingsSearch_Synonyms_Appearance_ColorTheme: String { return self._s[4196]! } - public var Passport_UpdateRequiredError: String { return self._s[4198]! } - public var PrivacySettings_Passcode: String { return self._s[4199]! } + public var Notifications_ExceptionsResetToDefaults: String { return self._s[4222]! } + public var PeerInfo_PaneVoiceAndVideo: String { return self._s[4223]! } + public var Group_OwnershipTransfer_Title: String { return self._s[4224]! } + public var Conversation_DefaultRestrictedInline: String { return self._s[4225]! } + public var Login_PhoneNumberHelp: String { return self._s[4227]! } + public var Channel_AdminLogFilter_EventsNewMembers: String { return self._s[4228]! } + public var Conversation_PinnedQuiz: String { return self._s[4229]! } + public var CreateGroup_SoftUserLimitAlert: String { return self._s[4230]! } + public var Login_PhoneNumberAlreadyAuthorizedSwitch: String { return self._s[4231]! } + public var Group_MessagePhotoUpdated: String { return self._s[4232]! } + public var LoginPassword_PasswordPlaceholder: String { return self._s[4233]! } + public var Passport_Identity_Translations: String { return self._s[4235]! } + public var ChatAdmins_AllMembersAreAdmins: String { return self._s[4236]! } + public var ChannelInfo_DeleteChannel: String { return self._s[4238]! } + public var PasscodeSettings_HelpBottom: String { return self._s[4239]! } + public var Channel_Members_AddMembers: String { return self._s[4240]! } + public var AutoDownloadSettings_LastDelimeter: String { return self._s[4241]! } + public var Notification_Exceptions_DeleteAllConfirmation: String { return self._s[4243]! } + public var Conversation_HoldForAudio: String { return self._s[4244]! } + public var Media_LimitedAccessChangeSettings: String { return self._s[4246]! } + public var Watch_LastSeen_Lately: String { return self._s[4247]! } + public var ChatList_Context_MarkAsRead: String { return self._s[4248]! } + public var Conversation_PinnedMessage: String { return self._s[4249]! } + public var SettingsSearch_Synonyms_Appearance_ColorTheme: String { return self._s[4250]! } + public var Passport_UpdateRequiredError: String { return self._s[4252]! } + public var PrivacySettings_Passcode: String { return self._s[4253]! } public func Call_EmojiDescription(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4200]!, self._r[4200]!, [_0]) + return formatWithArgumentRanges(self._s[4254]!, self._r[4254]!, [_0]) } - public var AutoNightTheme_NotAvailable: String { return self._s[4201]! } - public var Conversation_PressVolumeButtonForSound: String { return self._s[4202]! } - public var VoiceOver_Common_On: String { return self._s[4203]! } - public var LoginPassword_InvalidPasswordError: String { return self._s[4204]! } - public var ChatListFolder_IncludedSectionHeader: String { return self._s[4205]! } - public var Channel_SignMessages_Help: String { return self._s[4206]! } - public var ChatList_DeleteForEveryoneConfirmationTitle: String { return self._s[4207]! } - public var Conversation_TitleNoComments: String { return self._s[4208]! } - public var MediaPicker_LivePhotoDescription: String { return self._s[4209]! } - public var GroupInfo_Permissions: String { return self._s[4210]! } - public var GroupPermission_NoSendLinks: String { return self._s[4213]! } - public var Passport_Identity_ResidenceCountry: String { return self._s[4214]! } - public var Appearance_ThemeCarouselNightBlue: String { return self._s[4216]! } - public var ChatList_ArchiveAction: String { return self._s[4217]! } + public var AutoNightTheme_NotAvailable: String { return self._s[4255]! } + public var Conversation_PressVolumeButtonForSound: String { return self._s[4256]! } + public var VoiceOver_Common_On: String { return self._s[4257]! } + public var LoginPassword_InvalidPasswordError: String { return self._s[4258]! } + public var ChatListFolder_IncludedSectionHeader: String { return self._s[4259]! } + public var Channel_SignMessages_Help: String { return self._s[4260]! } + public var ChatList_DeleteForEveryoneConfirmationTitle: String { return self._s[4261]! } + public var Conversation_TitleNoComments: String { return self._s[4262]! } + public var MediaPicker_LivePhotoDescription: String { return self._s[4263]! } + public var GroupInfo_Permissions: String { return self._s[4264]! } + public var GroupPermission_NoSendLinks: String { return self._s[4267]! } + public var Passport_Identity_ResidenceCountry: String { return self._s[4268]! } + public var Appearance_ThemeCarouselNightBlue: String { return self._s[4270]! } + public var ChatList_ArchiveAction: String { return self._s[4271]! } public func Channel_AdminLog_DisabledSlowmode(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4218]!, self._r[4218]!, [_0]) + return formatWithArgumentRanges(self._s[4272]!, self._r[4272]!, [_0]) } - public var GroupInfo_GroupHistory: String { return self._s[4219]! } + public var GroupInfo_GroupHistory: String { return self._s[4273]! } public func Channel_Management_ErrorNotMember(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4221]!, self._r[4221]!, [_0]) + return formatWithArgumentRanges(self._s[4275]!, self._r[4275]!, [_0]) } - public var Privacy_Forwards_LinkIfAllowed: String { return self._s[4223]! } - public var Channel_Info_Banned: String { return self._s[4224]! } - public var Paint_RecentStickers: String { return self._s[4225]! } - public var VoiceOver_MessageContextSend: String { return self._s[4226]! } - public var Group_ErrorNotMutualContact: String { return self._s[4227]! } - public var ReportPeer_ReasonOther: String { return self._s[4229]! } - public var Channel_BanUser_PermissionChangeGroupInfo: String { return self._s[4230]! } - public var SocksProxySetup_ShareQRCodeInfo: String { return self._s[4232]! } - public var KeyCommand_Find: String { return self._s[4233]! } + public var Privacy_Forwards_LinkIfAllowed: String { return self._s[4277]! } + public var Channel_Info_Banned: String { return self._s[4278]! } + public var Paint_RecentStickers: String { return self._s[4279]! } + public var VoiceOver_MessageContextSend: String { return self._s[4280]! } + public var Group_ErrorNotMutualContact: String { return self._s[4281]! } + public var ReportPeer_ReasonOther: String { return self._s[4283]! } + public var Channel_BanUser_PermissionChangeGroupInfo: String { return self._s[4284]! } + public var SocksProxySetup_ShareQRCodeInfo: String { return self._s[4286]! } + public var KeyCommand_Find: String { return self._s[4287]! } public func Channel_MessageTitleUpdated(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4234]!, self._r[4234]!, [_0]) + return formatWithArgumentRanges(self._s[4288]!, self._r[4288]!, [_0]) } - public var ChatList_Context_Unmute: String { return self._s[4235]! } - public var Chat_SlowmodeAttachmentLimitReached: String { return self._s[4236]! } - public var Stickers_GroupStickersHelp: String { return self._s[4237]! } - public var Checkout_Title: String { return self._s[4238]! } - public var Activity_RecordingAudio: String { return self._s[4239]! } - public var SettingsSearch_Synonyms_Notifications_GroupNotificationsPreview: String { return self._s[4240]! } - public var BlockedUsers_BlockTitle: String { return self._s[4241]! } - public var DialogList_SavedMessagesHelp: String { return self._s[4243]! } - public var Calls_All: String { return self._s[4244]! } - public var Settings_FAQ_Button: String { return self._s[4246]! } - public var Conversation_Dice_u1F3B0: String { return self._s[4248]! } + public var ChatList_Context_Unmute: String { return self._s[4289]! } + public var Chat_SlowmodeAttachmentLimitReached: String { return self._s[4290]! } + public var Stickers_GroupStickersHelp: String { return self._s[4291]! } + public var Checkout_Title: String { return self._s[4292]! } + public var Activity_RecordingAudio: String { return self._s[4293]! } + public var SettingsSearch_Synonyms_Notifications_GroupNotificationsPreview: String { return self._s[4294]! } + public var BlockedUsers_BlockTitle: String { return self._s[4295]! } + public var DialogList_SavedMessagesHelp: String { return self._s[4297]! } + public var Calls_All: String { return self._s[4298]! } + public var Settings_FAQ_Button: String { return self._s[4300]! } + public var Conversation_Dice_u1F3B0: String { return self._s[4302]! } public func Time_MonthOfYear_m5(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4249]!, self._r[4249]!, [_0]) + return formatWithArgumentRanges(self._s[4303]!, self._r[4303]!, [_0]) } - public var Conversation_ReportGroupLocation: String { return self._s[4250]! } - public var Passport_Scans_Upload: String { return self._s[4251]! } - public var Channel_EditAdmin_PermissionPinMessages: String { return self._s[4253]! } - public var ChatList_UnarchiveAction: String { return self._s[4254]! } - public var Stats_GroupTopInviter_History: String { return self._s[4255]! } - public var GroupInfo_Permissions_Title: String { return self._s[4256]! } - public var VoiceChat_CreateNewVoiceChatStart: String { return self._s[4257]! } - public var Passport_Language_el: String { return self._s[4258]! } - public var Channel_DiscussionMessageUnavailable: String { return self._s[4259]! } - public var GroupInfo_ActionPromote: String { return self._s[4260]! } - public var Group_OwnershipTransfer_ErrorLocatedGroupsTooMuch: String { return self._s[4261]! } - public var Media_LimitedAccessSelectMore: String { return self._s[4262]! } + public var Conversation_ReportGroupLocation: String { return self._s[4304]! } + public var Passport_Scans_Upload: String { return self._s[4305]! } + public var Channel_EditAdmin_PermissionPinMessages: String { return self._s[4307]! } + public var ChatList_UnarchiveAction: String { return self._s[4308]! } + public var Stats_GroupTopInviter_History: String { return self._s[4309]! } + public var GroupInfo_Permissions_Title: String { return self._s[4310]! } + public var VoiceChat_CreateNewVoiceChatStart: String { return self._s[4311]! } + public var Passport_Language_el: String { return self._s[4312]! } + public var Channel_DiscussionMessageUnavailable: String { return self._s[4313]! } + public var GroupInfo_ActionPromote: String { return self._s[4314]! } + public var Group_OwnershipTransfer_ErrorLocatedGroupsTooMuch: String { return self._s[4315]! } + public var Media_LimitedAccessSelectMore: String { return self._s[4316]! } public func TwoStepAuth_PendingEmailHelp(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4263]!, self._r[4263]!, [_0]) - } - public var VoiceOver_Chat_Reply: String { return self._s[4264]! } - public var Month_GenMay: String { return self._s[4265]! } - public var DialogList_DeleteBotConversationConfirmation: String { return self._s[4266]! } - public var Chat_PsaTooltip_covid: String { return self._s[4267]! } - public var Watch_Suggestion_CantTalk: String { return self._s[4268]! } - public var Privacy_GroupsAndChannels_NeverAllow_Title: String { return self._s[4269]! } - public var AppUpgrade_Running: String { return self._s[4270]! } - public var PasscodeSettings_UnlockWithFaceId: String { return self._s[4273]! } - public var Notification_Exceptions_PreviewAlwaysOff: String { return self._s[4274]! } - public var SharedMedia_EmptyText: String { return self._s[4275]! } - public var Passport_Address_EditResidentialAddress: String { return self._s[4276]! } - public var SettingsSearch_Synonyms_Notifications_GroupNotificationsAlert: String { return self._s[4277]! } - public var Message_PinnedGame: String { return self._s[4278]! } - public var KeyCommand_SearchInChat: String { return self._s[4279]! } - public var Appearance_ThemeCarouselNewNight: String { return self._s[4280]! } - public var ChatList_Search_FilterMedia: String { return self._s[4281]! } - public var Message_PinnedAudioMessage: String { return self._s[4282]! } - public var ChannelInfo_ConfirmLeave: String { return self._s[4283]! } - public func Channel_AdminLog_MessagePromotedNameUsername(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4284]!, self._r[4284]!, [_1, _2]) - } - public var SocksProxySetup_ProxyStatusUnavailable: String { return self._s[4285]! } - public func Passport_Email_CodeHelp(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4286]!, self._r[4286]!, [_0]) - } - public func Message_PinnedTextMessage(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4287]!, self._r[4287]!, [_0]) - } - public var Settings_AddAccount: String { return self._s[4288]! } - public var Channel_AdminLog_CanDeleteMessages: String { return self._s[4289]! } - public var Conversation_DiscardVoiceMessageTitle: String { return self._s[4290]! } - public var Channel_JoinChannel: String { return self._s[4291]! } - public var Watch_UserInfo_Unblock: String { return self._s[4292]! } - public var PhoneLabel_Title: String { return self._s[4293]! } - public var Group_Setup_HistoryHiddenHelp: String { return self._s[4295]! } - public var Privacy_ProfilePhoto_AlwaysShareWith_Title: String { return self._s[4296]! } - public func Login_PhoneGenericEmailBody(_ _1: String, _ _2: String, _ _3: String, _ _4: String, _ _5: String, _ _6: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4297]!, self._r[4297]!, [_1, _2, _3, _4, _5, _6]) - } - public var Channel_AddBotErrorHaveRights: String { return self._s[4298]! } - public var ChatList_TabIconFoldersTooltipNonEmptyFolders: String { return self._s[4299]! } - public var DialogList_EncryptionProcessing: String { return self._s[4300]! } - public var ChatList_Search_FilterChats: String { return self._s[4301]! } - public var WatchRemote_NotificationText: String { return self._s[4302]! } - public var EditTheme_ChangeColors: String { return self._s[4303]! } - public var GroupRemoved_ViewUserInfo: String { return self._s[4304]! } - public var CallSettings_OnMobile: String { return self._s[4306]! } - public var Month_ShortFebruary: String { return self._s[4308]! } - public var VoiceOver_MessageContextReply: String { return self._s[4309]! } - public var Group_Location_ChangeLocation: String { return self._s[4311]! } - public func PUSH_VIDEO_CALL_REQUEST(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4312]!, self._r[4312]!, [_1]) - } - public var Passport_Address_TypeBankStatementUploadScan: String { return self._s[4313]! } - public var VoiceOver_Media_PlaybackStop: String { return self._s[4314]! } - public var SettingsSearch_Synonyms_Data_SaveIncomingPhotos: String { return self._s[4315]! } - public func Channel_AdminLog_MessageRestrictedUntil(_ _0: String) -> (String, [(Int, NSRange)]) { return formatWithArgumentRanges(self._s[4317]!, self._r[4317]!, [_0]) } - public var PhotoEditor_WarmthTool: String { return self._s[4318]! } - public var Login_InfoAvatarPhoto: String { return self._s[4319]! } - public var Notification_Exceptions_NewException_MessagePreviewHeader: String { return self._s[4320]! } - public var Permissions_CellularDataAllowInSettings_v0: String { return self._s[4321]! } - public var Map_PlacesInThisArea: String { return self._s[4322]! } - public var VoiceOver_Chat_ContactEmail: String { return self._s[4323]! } - public var Notifications_InAppNotificationsSounds: String { return self._s[4324]! } + public var VoiceOver_Chat_Reply: String { return self._s[4318]! } + public var Month_GenMay: String { return self._s[4319]! } + public var DialogList_DeleteBotConversationConfirmation: String { return self._s[4320]! } + public var Chat_PsaTooltip_covid: String { return self._s[4321]! } + public var Watch_Suggestion_CantTalk: String { return self._s[4322]! } + public var Privacy_GroupsAndChannels_NeverAllow_Title: String { return self._s[4323]! } + public var AppUpgrade_Running: String { return self._s[4324]! } + public var PasscodeSettings_UnlockWithFaceId: String { return self._s[4327]! } + public var Notification_Exceptions_PreviewAlwaysOff: String { return self._s[4328]! } + public var SharedMedia_EmptyText: String { return self._s[4329]! } + public var Passport_Address_EditResidentialAddress: String { return self._s[4330]! } + public var SettingsSearch_Synonyms_Notifications_GroupNotificationsAlert: String { return self._s[4331]! } + public var Message_PinnedGame: String { return self._s[4332]! } + public var KeyCommand_SearchInChat: String { return self._s[4333]! } + public var Appearance_ThemeCarouselNewNight: String { return self._s[4334]! } + public var ChatList_Search_FilterMedia: String { return self._s[4335]! } + public var Message_PinnedAudioMessage: String { return self._s[4336]! } + public var ChannelInfo_ConfirmLeave: String { return self._s[4337]! } + public func Channel_AdminLog_MessagePromotedNameUsername(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[4338]!, self._r[4338]!, [_1, _2]) + } + public var SocksProxySetup_ProxyStatusUnavailable: String { return self._s[4339]! } + public var InviteLink_Create: String { return self._s[4340]! } + public func Passport_Email_CodeHelp(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[4341]!, self._r[4341]!, [_0]) + } + public func Message_PinnedTextMessage(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[4342]!, self._r[4342]!, [_0]) + } + public var Settings_AddAccount: String { return self._s[4343]! } + public var Channel_AdminLog_CanDeleteMessages: String { return self._s[4344]! } + public var Conversation_DiscardVoiceMessageTitle: String { return self._s[4345]! } + public var Channel_JoinChannel: String { return self._s[4346]! } + public var Watch_UserInfo_Unblock: String { return self._s[4347]! } + public var PhoneLabel_Title: String { return self._s[4348]! } + public var Group_Setup_HistoryHiddenHelp: String { return self._s[4350]! } + public var Privacy_ProfilePhoto_AlwaysShareWith_Title: String { return self._s[4351]! } + public func Login_PhoneGenericEmailBody(_ _1: String, _ _2: String, _ _3: String, _ _4: String, _ _5: String, _ _6: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[4352]!, self._r[4352]!, [_1, _2, _3, _4, _5, _6]) + } + public var Channel_AddBotErrorHaveRights: String { return self._s[4353]! } + public var ChatList_TabIconFoldersTooltipNonEmptyFolders: String { return self._s[4354]! } + public var DialogList_EncryptionProcessing: String { return self._s[4355]! } + public var ChatList_Search_FilterChats: String { return self._s[4356]! } + public var WatchRemote_NotificationText: String { return self._s[4357]! } + public var EditTheme_ChangeColors: String { return self._s[4358]! } + public var GroupRemoved_ViewUserInfo: String { return self._s[4359]! } + public var CallSettings_OnMobile: String { return self._s[4361]! } + public var Month_ShortFebruary: String { return self._s[4363]! } + public var VoiceOver_MessageContextReply: String { return self._s[4364]! } + public var Group_Location_ChangeLocation: String { return self._s[4366]! } + public func PUSH_VIDEO_CALL_REQUEST(_ _1: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[4367]!, self._r[4367]!, [_1]) + } + public var Passport_Address_TypeBankStatementUploadScan: String { return self._s[4368]! } + public var VoiceOver_Media_PlaybackStop: String { return self._s[4369]! } + public var SettingsSearch_Synonyms_Data_SaveIncomingPhotos: String { return self._s[4370]! } + public func Channel_AdminLog_MessageRestrictedUntil(_ _0: String) -> (String, [(Int, NSRange)]) { + return formatWithArgumentRanges(self._s[4372]!, self._r[4372]!, [_0]) + } + public var PhotoEditor_WarmthTool: String { return self._s[4373]! } + public var Login_InfoAvatarPhoto: String { return self._s[4374]! } + public var Notification_Exceptions_NewException_MessagePreviewHeader: String { return self._s[4375]! } + public var Permissions_CellularDataAllowInSettings_v0: String { return self._s[4376]! } + public var Map_PlacesInThisArea: String { return self._s[4377]! } + public var VoiceOver_Chat_ContactEmail: String { return self._s[4378]! } + public var Notifications_InAppNotificationsSounds: String { return self._s[4379]! } public func PUSH_PINNED_NOTEXT(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4325]!, self._r[4325]!, [_1]) + return formatWithArgumentRanges(self._s[4380]!, self._r[4380]!, [_1]) } - public var ShareMenu_Send: String { return self._s[4326]! } - public var Username_InvalidStartsWithNumber: String { return self._s[4327]! } + public var ShareMenu_Send: String { return self._s[4381]! } + public var Username_InvalidStartsWithNumber: String { return self._s[4382]! } public func Channel_AdminLog_StartedVoiceChat(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4328]!, self._r[4328]!, [_1]) + return formatWithArgumentRanges(self._s[4383]!, self._r[4383]!, [_1]) } - public var Appearance_AppIconClassicX: String { return self._s[4329]! } + public var Appearance_AppIconClassicX: String { return self._s[4384]! } public func PUSH_CHANNEL_MESSAGE_ROUND(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4330]!, self._r[4330]!, [_1]) + return formatWithArgumentRanges(self._s[4385]!, self._r[4385]!, [_1]) } - public var Conversation_StopPoll: String { return self._s[4331]! } - public var InfoPlist_NSLocationAlwaysUsageDescription: String { return self._s[4333]! } - public var Passport_Identity_EditIdentityCard: String { return self._s[4334]! } - public var Appearance_ThemePreview_ChatList_3_Name: String { return self._s[4335]! } - public var Conversation_Timer_Title: String { return self._s[4336]! } - public var Common_Next: String { return self._s[4337]! } - public var Notification_Exceptions_NewException: String { return self._s[4338]! } + public var Conversation_StopPoll: String { return self._s[4386]! } + public var InfoPlist_NSLocationAlwaysUsageDescription: String { return self._s[4388]! } + public var Passport_Identity_EditIdentityCard: String { return self._s[4389]! } + public var Appearance_ThemePreview_ChatList_3_Name: String { return self._s[4390]! } + public var Conversation_Timer_Title: String { return self._s[4391]! } + public var Common_Next: String { return self._s[4392]! } + public var Notification_Exceptions_NewException: String { return self._s[4393]! } public func Generic_OpenHiddenLinkAlert(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4339]!, self._r[4339]!, [_0]) + return formatWithArgumentRanges(self._s[4394]!, self._r[4394]!, [_0]) } - public var AccessDenied_CallMicrophone: String { return self._s[4340]! } - public var VoiceChat_UnmutePeer: String { return self._s[4341]! } - public var SettingsSearch_Synonyms_Data_AutoDownloadUsingCellular: String { return self._s[4342]! } - public var ChangePhoneNumberCode_Help: String { return self._s[4343]! } - public var Passport_Identity_OneOfTypeIdentityCard: String { return self._s[4344]! } - public var Channel_AdminLogFilter_EventsLeaving: String { return self._s[4345]! } - public var BlockedUsers_LeavePrefix: String { return self._s[4346]! } + public var AccessDenied_CallMicrophone: String { return self._s[4395]! } + public var VoiceChat_UnmutePeer: String { return self._s[4396]! } + public var SettingsSearch_Synonyms_Data_AutoDownloadUsingCellular: String { return self._s[4397]! } + public var ChangePhoneNumberCode_Help: String { return self._s[4398]! } + public var Passport_Identity_OneOfTypeIdentityCard: String { return self._s[4399]! } + public var Channel_AdminLogFilter_EventsLeaving: String { return self._s[4400]! } + public var BlockedUsers_LeavePrefix: String { return self._s[4401]! } public func Passport_RequestHeader(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4347]!, self._r[4347]!, [_0]) + return formatWithArgumentRanges(self._s[4402]!, self._r[4402]!, [_0]) } - public var Group_About_Help: String { return self._s[4348]! } - public var TwoStepAuth_ChangePasswordDescription: String { return self._s[4349]! } - public var Tour_Title3: String { return self._s[4350]! } - public var Watch_Conversation_Unblock: String { return self._s[4351]! } - public var Watch_UserInfo_Block: String { return self._s[4352]! } - public var Notifications_ChannelNotificationsAlert: String { return self._s[4353]! } - public var TwoFactorSetup_Hint_Action: String { return self._s[4354]! } - public var IntentsSettings_SuggestedChatsInfo: String { return self._s[4355]! } - public var TextFormat_AddLinkTitle: String { return self._s[4356]! } - public var GroupInfo_InviteLink_RevokeAlert_Revoke: String { return self._s[4357]! } - public var TwoStepAuth_EnterPasswordTitle: String { return self._s[4358]! } - public var FastTwoStepSetup_PasswordSection: String { return self._s[4359]! } - public var Compose_ChannelMembers: String { return self._s[4360]! } - public var Conversation_ForwardTitle: String { return self._s[4361]! } - public var Conversation_PinnedPoll: String { return self._s[4363]! } + public var Group_About_Help: String { return self._s[4403]! } + public var TwoStepAuth_ChangePasswordDescription: String { return self._s[4404]! } + public var Tour_Title3: String { return self._s[4405]! } + public var Watch_Conversation_Unblock: String { return self._s[4406]! } + public var Watch_UserInfo_Block: String { return self._s[4407]! } + public var Notifications_ChannelNotificationsAlert: String { return self._s[4408]! } + public var TwoFactorSetup_Hint_Action: String { return self._s[4409]! } + public var IntentsSettings_SuggestedChatsInfo: String { return self._s[4410]! } + public var TextFormat_AddLinkTitle: String { return self._s[4411]! } + public var GroupInfo_InviteLink_RevokeAlert_Revoke: String { return self._s[4412]! } + public var TwoStepAuth_EnterPasswordTitle: String { return self._s[4413]! } + public var FastTwoStepSetup_PasswordSection: String { return self._s[4414]! } + public var Compose_ChannelMembers: String { return self._s[4415]! } + public var Conversation_ForwardTitle: String { return self._s[4416]! } + public var Conversation_PinnedPoll: String { return self._s[4418]! } public func VoiceOver_Chat_AnonymousPollFrom(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4364]!, self._r[4364]!, [_0]) + return formatWithArgumentRanges(self._s[4419]!, self._r[4419]!, [_0]) } - public var SettingsSearch_Synonyms_EditProfile_AddAccount: String { return self._s[4365]! } - public var Conversation_ContextMenuStickerPackAdd: String { return self._s[4366]! } - public var Stats_Overview: String { return self._s[4367]! } - public var Map_HomeAndWorkTitle: String { return self._s[4368]! } + public var SettingsSearch_Synonyms_EditProfile_AddAccount: String { return self._s[4420]! } + public var Conversation_ContextMenuStickerPackAdd: String { return self._s[4421]! } + public var Stats_Overview: String { return self._s[4422]! } + public var Map_HomeAndWorkTitle: String { return self._s[4423]! } public func Time_PreciseDate_m4(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4369]!, self._r[4369]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[4424]!, self._r[4424]!, [_1, _2, _3]) } - public var Passport_Address_CityPlaceholder: String { return self._s[4370]! } - public var InfoPlist_NSLocationAlwaysAndWhenInUseUsageDescription: String { return self._s[4371]! } - public var Privacy_PhoneNumber: String { return self._s[4372]! } - public var ChatList_Search_FilterFiles: String { return self._s[4373]! } - public var ChatList_DeleteForEveryoneConfirmationAction: String { return self._s[4374]! } - public var ChannelIntro_CreateChannel: String { return self._s[4375]! } - public var Conversation_InputTextAnonymousPlaceholder: String { return self._s[4376]! } + public var Passport_Address_CityPlaceholder: String { return self._s[4425]! } + public var InfoPlist_NSLocationAlwaysAndWhenInUseUsageDescription: String { return self._s[4426]! } + public var Privacy_PhoneNumber: String { return self._s[4427]! } + public var ChatList_Search_FilterFiles: String { return self._s[4428]! } + public var ChatList_DeleteForEveryoneConfirmationAction: String { return self._s[4429]! } + public var ChannelIntro_CreateChannel: String { return self._s[4430]! } + public var Conversation_InputTextAnonymousPlaceholder: String { return self._s[4431]! } public func Login_EmailCodeBody(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4377]!, self._r[4377]!, [_0]) + return formatWithArgumentRanges(self._s[4432]!, self._r[4432]!, [_0]) } - public var Weekday_ShortMonday: String { return self._s[4378]! } - public var Passport_Language_ar: String { return self._s[4380]! } - public var SettingsSearch_Synonyms_EditProfile_Title: String { return self._s[4381]! } - public var TwoFactorSetup_Done_Title: String { return self._s[4382]! } - public var Calls_RatingFeedback: String { return self._s[4383]! } - public var SettingsSearch_Synonyms_Notifications_ChannelNotificationsPreview: String { return self._s[4384]! } - public var AutoDownloadSettings_ResetSettings: String { return self._s[4387]! } - public var Watch_Compose_Send: String { return self._s[4388]! } - public var PasscodeSettings_ChangePasscode: String { return self._s[4389]! } - public var WebSearch_RecentSectionClear: String { return self._s[4390]! } + public var Weekday_ShortMonday: String { return self._s[4433]! } + public var Passport_Language_ar: String { return self._s[4435]! } + public var SettingsSearch_Synonyms_EditProfile_Title: String { return self._s[4436]! } + public var TwoFactorSetup_Done_Title: String { return self._s[4437]! } + public var Calls_RatingFeedback: String { return self._s[4438]! } + public var SettingsSearch_Synonyms_Notifications_ChannelNotificationsPreview: String { return self._s[4439]! } + public var AutoDownloadSettings_ResetSettings: String { return self._s[4442]! } + public var Watch_Compose_Send: String { return self._s[4443]! } + public var PasscodeSettings_ChangePasscode: String { return self._s[4444]! } + public var WebSearch_RecentSectionClear: String { return self._s[4445]! } public func Contacts_AccessDeniedHelpPortrait(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4391]!, self._r[4391]!, [_0]) + return formatWithArgumentRanges(self._s[4446]!, self._r[4446]!, [_0]) } - public var WallpaperSearch_ColorTeal: String { return self._s[4392]! } - public var Wallpaper_SetCustomBackgroundInfo: String { return self._s[4393]! } - public var Permissions_ContactsTitle_v0: String { return self._s[4394]! } - public var Checkout_PasswordEntry_Pay: String { return self._s[4396]! } - public var Settings_SavedMessages: String { return self._s[4397]! } - public var TwoStepAuth_ReEnterPasswordDescription: String { return self._s[4398]! } - public var Month_ShortMarch: String { return self._s[4399]! } - public var Message_Location: String { return self._s[4400]! } + public var WallpaperSearch_ColorTeal: String { return self._s[4447]! } + public var Wallpaper_SetCustomBackgroundInfo: String { return self._s[4448]! } + public var Permissions_ContactsTitle_v0: String { return self._s[4449]! } + public var Checkout_PasswordEntry_Pay: String { return self._s[4451]! } + public var Settings_SavedMessages: String { return self._s[4452]! } + public var TwoStepAuth_ReEnterPasswordDescription: String { return self._s[4453]! } + public var Month_ShortMarch: String { return self._s[4454]! } + public var Message_Location: String { return self._s[4455]! } public func PUSH_MESSAGE_GIF(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4401]!, self._r[4401]!, [_1]) + return formatWithArgumentRanges(self._s[4456]!, self._r[4456]!, [_1]) } public func Notification_CallTimeFormat(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4402]!, self._r[4402]!, [_1, _2]) + return formatWithArgumentRanges(self._s[4457]!, self._r[4457]!, [_1, _2]) } - public var VoiceOver_Chat_VoiceMessage: String { return self._s[4404]! } + public var VoiceOver_Chat_VoiceMessage: String { return self._s[4459]! } public func Channel_AdminLog_MessageChangedUnlinkedChannel(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4405]!, self._r[4405]!, [_1, _2]) + return formatWithArgumentRanges(self._s[4460]!, self._r[4460]!, [_1, _2]) } - public var GroupPermission_NoSendMedia: String { return self._s[4406]! } - public var Conversation_ClousStorageInfo_Description2: String { return self._s[4407]! } - public var SharedMedia_CategoryDocs: String { return self._s[4408]! } - public var Appearance_RemoveThemeConfirmation: String { return self._s[4409]! } - public var Paint_Framed: String { return self._s[4410]! } - public var Channel_EditAdmin_PermissionAddAdmins: String { return self._s[4411]! } - public var Passport_Identity_DoesNotExpire: String { return self._s[4412]! } - public var Channel_SignMessages: String { return self._s[4413]! } - public var Contacts_AccessDeniedHelpON: String { return self._s[4414]! } - public var Conversation_ContextMenuStickerPackInfo: String { return self._s[4415]! } + public var GroupPermission_NoSendMedia: String { return self._s[4461]! } + public var Conversation_ClousStorageInfo_Description2: String { return self._s[4462]! } + public var SharedMedia_CategoryDocs: String { return self._s[4463]! } + public var Appearance_RemoveThemeConfirmation: String { return self._s[4464]! } + public var Paint_Framed: String { return self._s[4465]! } + public var Channel_EditAdmin_PermissionAddAdmins: String { return self._s[4466]! } + public var Passport_Identity_DoesNotExpire: String { return self._s[4467]! } + public var Channel_SignMessages: String { return self._s[4468]! } + public var Contacts_AccessDeniedHelpON: String { return self._s[4469]! } + public var Conversation_ContextMenuStickerPackInfo: String { return self._s[4470]! } public func PUSH_CHAT_LEFT(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4416]!, self._r[4416]!, [_1, _2]) + return formatWithArgumentRanges(self._s[4471]!, self._r[4471]!, [_1, _2]) } - public var GroupInfo_UpgradeButton: String { return self._s[4417]! } - public var Channel_EditAdmin_PermissionInviteMembers: String { return self._s[4418]! } - public var AutoDownloadSettings_Files: String { return self._s[4419]! } + public var InviteLink_Create_TimeLimitNoLimit: String { return self._s[4472]! } + public var GroupInfo_UpgradeButton: String { return self._s[4473]! } + public var Channel_EditAdmin_PermissionInviteMembers: String { return self._s[4474]! } + public var AutoDownloadSettings_Files: String { return self._s[4475]! } public func Notification_ChangedGroupName(_ _0: String, _ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4420]!, self._r[4420]!, [_0, _1]) + return formatWithArgumentRanges(self._s[4476]!, self._r[4476]!, [_0, _1]) } - public var Login_SendCodeViaSms: String { return self._s[4422]! } - public var Update_UpdateApp: String { return self._s[4423]! } - public var Channel_Setup_TypePublic: String { return self._s[4424]! } - public var Watch_Compose_CreateMessage: String { return self._s[4425]! } + public var Login_SendCodeViaSms: String { return self._s[4478]! } + public var Update_UpdateApp: String { return self._s[4479]! } + public var Channel_Setup_TypePublic: String { return self._s[4480]! } + public var Watch_Compose_CreateMessage: String { return self._s[4481]! } public func PUSH_CHAT_MESSAGE_VIDEOS(_ _1: String, _ _2: String, _ _3: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4426]!, self._r[4426]!, [_1, _2, _3]) + return formatWithArgumentRanges(self._s[4482]!, self._r[4482]!, [_1, _2, _3]) } - public var StickerPacksSettings_ManagingHelp: String { return self._s[4427]! } - public var VoiceOver_Chat_Video: String { return self._s[4428]! } - public var Forward_ChannelReadOnly: String { return self._s[4429]! } - public var StickerPack_HideStickers: String { return self._s[4430]! } - public var ChatListFolder_NameContacts: String { return self._s[4431]! } - public var Profile_BotInfo: String { return self._s[4432]! } - public var Document_TargetConfirmationFormat: String { return self._s[4433]! } - public var GroupInfo_InviteByLink: String { return self._s[4434]! } - public var Channel_AdminLog_BanSendStickersAndGifs: String { return self._s[4435]! } - public var Watch_Stickers_RecentPlaceholder: String { return self._s[4436]! } - public var Broadcast_AdminLog_EmptyText: String { return self._s[4437]! } - public var Passport_NotLoggedInMessage: String { return self._s[4438]! } - public var Conversation_StopQuizConfirmation: String { return self._s[4439]! } - public var Checkout_PaymentMethod: String { return self._s[4440]! } - public var ChatList_ArchivedChatsTitle: String { return self._s[4444]! } - public var TwoStepAuth_SetupPasswordConfirmFailed: String { return self._s[4445]! } - public var VoiceOver_Chat_RecordPreviewVoiceMessage: String { return self._s[4446]! } - public var PrivacyLastSeenSettings_GroupsAndChannelsHelp: String { return self._s[4447]! } - public var SettingsSearch_Synonyms_Privacy_Data_ContactsReset: String { return self._s[4448]! } - public var Camera_Title: String { return self._s[4449]! } - public var Map_Directions: String { return self._s[4450]! } - public var Stats_MessagePublicForwardsTitle: String { return self._s[4452]! } - public var Privacy_ProfilePhoto_WhoCanSeeMyPhoto: String { return self._s[4453]! } - public var Profile_EncryptionKey: String { return self._s[4454]! } + public var StickerPacksSettings_ManagingHelp: String { return self._s[4483]! } + public var VoiceOver_Chat_Video: String { return self._s[4484]! } + public var Forward_ChannelReadOnly: String { return self._s[4485]! } + public var StickerPack_HideStickers: String { return self._s[4486]! } + public var ChatListFolder_NameContacts: String { return self._s[4487]! } + public var Profile_BotInfo: String { return self._s[4488]! } + public var Document_TargetConfirmationFormat: String { return self._s[4489]! } + public var GroupInfo_InviteByLink: String { return self._s[4490]! } + public var Channel_AdminLog_BanSendStickersAndGifs: String { return self._s[4491]! } + public var Watch_Stickers_RecentPlaceholder: String { return self._s[4492]! } + public var Broadcast_AdminLog_EmptyText: String { return self._s[4493]! } + public var Passport_NotLoggedInMessage: String { return self._s[4494]! } + public var Conversation_StopQuizConfirmation: String { return self._s[4495]! } + public var Checkout_PaymentMethod: String { return self._s[4496]! } + public var ChatList_ArchivedChatsTitle: String { return self._s[4500]! } + public var TwoStepAuth_SetupPasswordConfirmFailed: String { return self._s[4501]! } + public var VoiceOver_Chat_RecordPreviewVoiceMessage: String { return self._s[4502]! } + public var PrivacyLastSeenSettings_GroupsAndChannelsHelp: String { return self._s[4503]! } + public var SettingsSearch_Synonyms_Privacy_Data_ContactsReset: String { return self._s[4504]! } + public var Camera_Title: String { return self._s[4505]! } + public var Map_Directions: String { return self._s[4506]! } + public var Stats_MessagePublicForwardsTitle: String { return self._s[4508]! } + public var Privacy_ProfilePhoto_WhoCanSeeMyPhoto: String { return self._s[4509]! } + public var Profile_EncryptionKey: String { return self._s[4510]! } public func LOCAL_CHAT_MESSAGE_FWDS(_ _1: String, _ _2: Int) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4455]!, self._r[4455]!, [_1, "\(_2)"]) + return formatWithArgumentRanges(self._s[4511]!, self._r[4511]!, [_1, "\(_2)"]) } public func Compatibility_SecretMediaVersionTooLow(_ _0: String, _ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4456]!, self._r[4456]!, [_0, _1]) + return formatWithArgumentRanges(self._s[4512]!, self._r[4512]!, [_0, _1]) } - public var Passport_Identity_TypePassport: String { return self._s[4457]! } - public var CreatePoll_QuizOptionsHeader: String { return self._s[4459]! } - public var Common_No: String { return self._s[4460]! } - public var Conversation_SendMessage_ScheduleMessage: String { return self._s[4461]! } - public var SettingsSearch_Synonyms_Privacy_LastSeen: String { return self._s[4462]! } - public var Settings_AboutEmpty: String { return self._s[4463]! } - public var TwoStepAuth_FloodError: String { return self._s[4465]! } - public var SettingsSearch_Synonyms_Appearance_TextSize: String { return self._s[4466]! } + public var Passport_Identity_TypePassport: String { return self._s[4513]! } + public var CreatePoll_QuizOptionsHeader: String { return self._s[4515]! } + public var Common_No: String { return self._s[4516]! } + public var Conversation_SendMessage_ScheduleMessage: String { return self._s[4517]! } + public var SettingsSearch_Synonyms_Privacy_LastSeen: String { return self._s[4518]! } + public var Settings_AboutEmpty: String { return self._s[4519]! } + public var TwoStepAuth_FloodError: String { return self._s[4521]! } + public var SettingsSearch_Synonyms_Appearance_TextSize: String { return self._s[4522]! } public func Channel_AdminLog_MessageUnkickedName(_ _1: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4468]!, self._r[4468]!, [_1]) + return formatWithArgumentRanges(self._s[4524]!, self._r[4524]!, [_1]) } - public var Conversation_Edit: String { return self._s[4471]! } - public var CheckoutInfo_SaveInfo: String { return self._s[4472]! } - public var VoiceOver_Chat_AnonymousPoll: String { return self._s[4473]! } - public var Call_CameraTooltip: String { return self._s[4475]! } - public var InstantPage_FeedbackButtonShort: String { return self._s[4476]! } - public var Contacts_InviteToTelegram: String { return self._s[4477]! } - public var Notifications_ResetAllNotifications: String { return self._s[4478]! } - public var Calls_NewCall: String { return self._s[4479]! } - public var VoiceOver_Chat_Music: String { return self._s[4482]! } - public var Channel_Members_AddAdminErrorNotAMember: String { return self._s[4483]! } - public var Channel_Edit_AboutItem: String { return self._s[4484]! } - public var Message_VideoExpired: String { return self._s[4485]! } - public var Passport_Address_TypeTemporaryRegistrationUploadScan: String { return self._s[4486]! } + public var Conversation_Edit: String { return self._s[4527]! } + public var CheckoutInfo_SaveInfo: String { return self._s[4528]! } + public var VoiceOver_Chat_AnonymousPoll: String { return self._s[4529]! } + public var Call_CameraTooltip: String { return self._s[4531]! } + public var InstantPage_FeedbackButtonShort: String { return self._s[4532]! } + public var Contacts_InviteToTelegram: String { return self._s[4533]! } + public var Notifications_ResetAllNotifications: String { return self._s[4534]! } + public var Calls_NewCall: String { return self._s[4535]! } + public var VoiceOver_Chat_Music: String { return self._s[4538]! } + public var Channel_Members_AddAdminErrorNotAMember: String { return self._s[4539]! } + public var Channel_Edit_AboutItem: String { return self._s[4540]! } + public var Message_VideoExpired: String { return self._s[4541]! } + public var Passport_Address_TypeTemporaryRegistrationUploadScan: String { return self._s[4542]! } public func PUSH_CHAT_RETURNED(_ _1: String, _ _2: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4487]!, self._r[4487]!, [_1, _2]) + return formatWithArgumentRanges(self._s[4543]!, self._r[4543]!, [_1, _2]) } - public var NotificationsSound_Input: String { return self._s[4489]! } - public var Notifications_ClassicTones: String { return self._s[4490]! } - public var Conversation_StatusTyping: String { return self._s[4491]! } - public var Checkout_ErrorProviderAccountInvalid: String { return self._s[4492]! } - public var ChatSettings_AutoDownloadSettings_Delimeter: String { return self._s[4493]! } - public var SettingsSearch_Synonyms_Notifications_BadgeIncludeMutedChats: String { return self._s[4494]! } - public var Conversation_MessageLeaveComment: String { return self._s[4495]! } - public var UserInfo_TapToCall: String { return self._s[4496]! } - public var EnterPasscode_EnterNewPasscodeNew: String { return self._s[4497]! } - public var Conversation_ClearAll: String { return self._s[4499]! } - public var UserInfo_NotificationsDefault: String { return self._s[4500]! } - public var Location_ProximityGroupTip: String { return self._s[4501]! } - public var Map_ChooseAPlace: String { return self._s[4502]! } - public var GroupInfo_AddParticipantTitle: String { return self._s[4503]! } - public var ChatList_PeerTypeNonContact: String { return self._s[4504]! } - public var Conversation_SlideToCancel: String { return self._s[4505]! } - public var Month_ShortJuly: String { return self._s[4506]! } - public var SocksProxySetup_ProxyType: String { return self._s[4507]! } + public var NotificationsSound_Input: String { return self._s[4545]! } + public var Notifications_ClassicTones: String { return self._s[4546]! } + public var Conversation_StatusTyping: String { return self._s[4547]! } + public var Checkout_ErrorProviderAccountInvalid: String { return self._s[4548]! } + public var ChatSettings_AutoDownloadSettings_Delimeter: String { return self._s[4549]! } + public var SettingsSearch_Synonyms_Notifications_BadgeIncludeMutedChats: String { return self._s[4550]! } + public var Conversation_MessageLeaveComment: String { return self._s[4551]! } + public var UserInfo_TapToCall: String { return self._s[4552]! } + public var EnterPasscode_EnterNewPasscodeNew: String { return self._s[4553]! } + public var Conversation_ClearAll: String { return self._s[4555]! } + public var UserInfo_NotificationsDefault: String { return self._s[4556]! } + public var Location_ProximityGroupTip: String { return self._s[4557]! } + public var Map_ChooseAPlace: String { return self._s[4558]! } + public var GroupInfo_AddParticipantTitle: String { return self._s[4559]! } + public var ChatList_PeerTypeNonContact: String { return self._s[4560]! } + public var Conversation_SlideToCancel: String { return self._s[4561]! } + public var Month_ShortJuly: String { return self._s[4562]! } + public var SocksProxySetup_ProxyType: String { return self._s[4563]! } public func ChatList_DeleteChatConfirmation(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4508]!, self._r[4508]!, [_0]) + return formatWithArgumentRanges(self._s[4564]!, self._r[4564]!, [_0]) } - public var ChatList_EditFolders: String { return self._s[4509]! } - public var TwoStepAuth_SetPasswordHelp: String { return self._s[4510]! } - public var ScheduledMessages_RemindersTitle: String { return self._s[4512]! } + public var ChatList_EditFolders: String { return self._s[4565]! } + public var TwoStepAuth_SetPasswordHelp: String { return self._s[4566]! } + public var ScheduledMessages_RemindersTitle: String { return self._s[4568]! } public func GroupPermission_ApplyAlertText(_ _0: String) -> (String, [(Int, NSRange)]) { - return formatWithArgumentRanges(self._s[4513]!, self._r[4513]!, [_0]) + return formatWithArgumentRanges(self._s[4569]!, self._r[4569]!, [_0]) } - public var Permissions_PeopleNearbyTitle_v0: String { return self._s[4514]! } - public var Your_cards_expiration_year_is_invalid: String { return self._s[4515]! } - public var UserInfo_ShareMyContactInfo: String { return self._s[4517]! } - public var Passport_DeleteAddress: String { return self._s[4519]! } - public var Passport_DeletePassportConfirmation: String { return self._s[4520]! } - public var Passport_Identity_ReverseSide: String { return self._s[4521]! } - public var CheckoutInfo_ErrorEmailInvalid: String { return self._s[4522]! } - public var Login_InfoLastNamePlaceholder: String { return self._s[4523]! } - public var Passport_FieldAddress: String { return self._s[4524]! } - public var SettingsSearch_Synonyms_Calls_Title: String { return self._s[4525]! } - public var Passport_Identity_ResidenceCountryPlaceholder: String { return self._s[4528]! } - public var VoiceChat_Panel_TapToJoin: String { return self._s[4529]! } - public var Map_Home: String { return self._s[4530]! } - public var PollResults_Title: String { return self._s[4532]! } - public var ArchivedChats_IntroText2: String { return self._s[4534]! } - public var PasscodeSettings_SimplePasscodeHelp: String { return self._s[4535]! } - public var VoiceOver_Chat_ContactPhoneNumber: String { return self._s[4536]! } - public var VoiceChat_Muted: String { return self._s[4538]! } - public var CallFeedback_ReasonSilentRemote: String { return self._s[4539]! } - public var Passport_Identity_AddPersonalDetails: String { return self._s[4540]! } - public var Group_Info_AdminLog: String { return self._s[4542]! } - public var ChatSettings_AutoPlayTitle: String { return self._s[4543]! } - public var Appearance_Animations: String { return self._s[4544]! } - public var Appearance_TextSizeSetting: String { return self._s[4545]! } - public func SharedMedia_Video(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[0 * 6 + Int(form.rawValue)]!, stringValue) + public var Permissions_PeopleNearbyTitle_v0: String { return self._s[4570]! } + public var Your_cards_expiration_year_is_invalid: String { return self._s[4571]! } + public var UserInfo_ShareMyContactInfo: String { return self._s[4573]! } + public var Passport_DeleteAddress: String { return self._s[4575]! } + public var Passport_DeletePassportConfirmation: String { return self._s[4576]! } + public var Passport_Identity_ReverseSide: String { return self._s[4577]! } + public var CheckoutInfo_ErrorEmailInvalid: String { return self._s[4578]! } + public var Login_InfoLastNamePlaceholder: String { return self._s[4579]! } + public var InviteLink_CreatedBy: String { return self._s[4580]! } + public var Passport_FieldAddress: String { return self._s[4581]! } + public var SettingsSearch_Synonyms_Calls_Title: String { return self._s[4582]! } + public var Passport_Identity_ResidenceCountryPlaceholder: String { return self._s[4585]! } + public var VoiceChat_Panel_TapToJoin: String { return self._s[4586]! } + public var Map_Home: String { return self._s[4587]! } + public var PollResults_Title: String { return self._s[4589]! } + public var ArchivedChats_IntroText2: String { return self._s[4591]! } + public var PasscodeSettings_SimplePasscodeHelp: String { return self._s[4592]! } + public var VoiceOver_Chat_ContactPhoneNumber: String { return self._s[4593]! } + public var VoiceChat_Muted: String { return self._s[4595]! } + public var CallFeedback_ReasonSilentRemote: String { return self._s[4596]! } + public var Passport_Identity_AddPersonalDetails: String { return self._s[4597]! } + public var Group_Info_AdminLog: String { return self._s[4599]! } + public var ChatSettings_AutoPlayTitle: String { return self._s[4600]! } + public var Appearance_Animations: String { return self._s[4601]! } + public var Appearance_TextSizeSetting: String { return self._s[4602]! } + public func PUSH_CHANNEL_MESSAGE_PHOTOS(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { + let form = getPluralizationForm(self.lc, selector) + return String(format: self._ps[0 * 6 + Int(form.rawValue)]!, _1, _2) } - public func InstantPage_Views(_ value: Int32) -> String { + public func MessageTimer_Years(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[1 * 6 + Int(form.rawValue)]!, stringValue) } - public func PUSH_CHAT_MESSAGE_FWDS(_ selector: Int32, _ _2: String, _ _1: String, _ _3: Int32) -> String { + public func PUSH_CHANNEL_MESSAGE_FWDS(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { let form = getPluralizationForm(self.lc, selector) - return String(format: self._ps[2 * 6 + Int(form.rawValue)]!, _2, _1, _3) + return String(format: self._ps[2 * 6 + Int(form.rawValue)]!, _1, _2) } - public func Stats_GroupTopInviterInvites(_ value: Int32) -> String { + public func ChatList_MessageVideos(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[3 * 6 + Int(form.rawValue)]!, stringValue) } - public func PUSH_MESSAGE_PHOTOS(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { - let form = getPluralizationForm(self.lc, selector) - return String(format: self._ps[4 * 6 + Int(form.rawValue)]!, _1, _2) + public func CreatePoll_AddMoreOptions(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[4 * 6 + Int(form.rawValue)]!, stringValue) } - public func OldChannels_GroupFormat(_ value: Int32) -> String { + public func Map_ETAHours(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[5 * 6 + Int(form.rawValue)]!, stringValue) } - public func MessageTimer_Hours(_ value: Int32) -> String { + public func MessageTimer_ShortMinutes(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[6 * 6 + Int(form.rawValue)]!, stringValue) } - public func Conversation_StatusOnline(_ value: Int32) -> String { + public func MessageTimer_Days(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[7 * 6 + Int(form.rawValue)]!, stringValue) } - public func PollResults_ShowMore(_ value: Int32) -> String { + public func LiveLocationUpdated_MinutesAgo(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[8 * 6 + Int(form.rawValue)]!, stringValue) } - public func Stats_GroupShowMoreTopPosters(_ value: Int32) -> String { + public func ChatList_DeleteConfirmation(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[9 * 6 + Int(form.rawValue)]!, stringValue) } - public func ForwardedPolls(_ value: Int32) -> String { + public func Media_SharePhoto(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[10 * 6 + Int(form.rawValue)]!, stringValue) } - public func MuteFor_Days(_ value: Int32) -> String { + public func InviteLink_PeopleJoinedShort(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[11 * 6 + Int(form.rawValue)]!, stringValue) } - public func Chat_TitlePinnedMessages(_ value: Int32) -> String { + public func PrivacyLastSeenSettings_AddUsers(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[12 * 6 + Int(form.rawValue)]!, stringValue) } - public func PUSH_MESSAGE_FWDS(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { - let form = getPluralizationForm(self.lc, selector) - return String(format: self._ps[13 * 6 + Int(form.rawValue)]!, _1, _2) - } - public func MessageTimer_Weeks(_ value: Int32) -> String { + public func Chat_DeleteMessagesConfirmation(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[14 * 6 + Int(form.rawValue)]!, stringValue) + return String(format: self._ps[13 * 6 + Int(form.rawValue)]!, stringValue) } - public func Passport_Scans(_ value: Int32) -> String { + public func PUSH_CHAT_MESSAGE_FWDS(_ selector: Int32, _ _2: String, _ _1: String, _ _3: Int32) -> String { + let form = getPluralizationForm(self.lc, selector) + return String(format: self._ps[14 * 6 + Int(form.rawValue)]!, _2, _1, _3) + } + public func ChatList_SelectedChats(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[15 * 6 + Int(form.rawValue)]!, stringValue) } - public func Stats_GroupTopAdminKicks(_ value: Int32) -> String { + public func Notification_GameScoreSelfSimple(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[16 * 6 + Int(form.rawValue)]!, stringValue) } - public func VoiceOver_Chat_ContactPhoneNumberCount(_ value: Int32) -> String { + public func SharedMedia_Link(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[17 * 6 + Int(form.rawValue)]!, stringValue) } - public func Call_ShortMinutes(_ value: Int32) -> String { + public func MessageTimer_ShortSeconds(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[18 * 6 + Int(form.rawValue)]!, stringValue) } - public func LastSeen_HoursAgo(_ value: Int32) -> String { + public func Call_ShortSeconds(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[19 * 6 + Int(form.rawValue)]!, stringValue) @@ -5116,141 +5165,139 @@ public final class PresentationStrings: Equatable { let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[20 * 6 + Int(form.rawValue)]!, stringValue) } - public func Notifications_ExceptionMuteExpires_Hours(_ value: Int32) -> String { + public func Call_Seconds(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[21 * 6 + Int(form.rawValue)]!, stringValue) } - public func SharedMedia_Photo(_ value: Int32) -> String { + public func MuteFor_Hours(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[22 * 6 + Int(form.rawValue)]!, stringValue) } - public func Notifications_ExceptionMuteExpires_Minutes(_ value: Int32) -> String { + public func StickerPack_AddMaskCount(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[23 * 6 + Int(form.rawValue)]!, stringValue) } - public func ServiceMessage_GameScoreSimple(_ value: Int32) -> String { + public func SharedMedia_Generic(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[24 * 6 + Int(form.rawValue)]!, stringValue) } - public func Invitation_Members(_ value: Int32) -> String { + public func Watch_LastSeen_MinutesAgo(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[25 * 6 + Int(form.rawValue)]!, stringValue) } - public func ForwardedStickers(_ value: Int32) -> String { + public func SharedMedia_Video(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[26 * 6 + Int(form.rawValue)]!, stringValue) } - public func ChatList_DeleteConfirmation(_ value: Int32) -> String { + public func Conversation_SelectedMessages(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[27 * 6 + Int(form.rawValue)]!, stringValue) } - public func MessageTimer_ShortHours(_ value: Int32) -> String { + public func Invitation_Members(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[28 * 6 + Int(form.rawValue)]!, stringValue) } - public func Contacts_InviteContacts(_ value: Int32) -> String { + public func Conversation_StatusMembers(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[29 * 6 + Int(form.rawValue)]!, stringValue) } - public func PasscodeSettings_FailedAttempts(_ value: Int32) -> String { + public func PeopleNearby_ShowMorePeople(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[30 * 6 + Int(form.rawValue)]!, stringValue) } - public func ChatList_MessageVideos(_ value: Int32) -> String { + public func Conversation_StatusSubscribers(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[31 * 6 + Int(form.rawValue)]!, stringValue) } - public func PUSH_MESSAGE_DOCS(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { - let form = getPluralizationForm(self.lc, selector) - return String(format: self._ps[32 * 6 + Int(form.rawValue)]!, _1, _2) + public func VoiceChat_Status_Members(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[32 * 6 + Int(form.rawValue)]!, stringValue) } - public func StickerPack_StickerCount(_ value: Int32) -> String { + public func Notification_GameScoreExtended(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[33 * 6 + Int(form.rawValue)]!, stringValue) } - public func Conversation_TitleComments(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[34 * 6 + Int(form.rawValue)]!, stringValue) + public func PUSH_CHAT_MESSAGE_ROUNDS(_ selector: Int32, _ _2: String, _ _1: String, _ _3: Int32) -> String { + let form = getPluralizationForm(self.lc, selector) + return String(format: self._ps[34 * 6 + Int(form.rawValue)]!, _2, _1, _3) } - public func MessageTimer_ShortMinutes(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[35 * 6 + Int(form.rawValue)]!, stringValue) + public func PUSH_CHANNEL_MESSAGES(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { + let form = getPluralizationForm(self.lc, selector) + return String(format: self._ps[35 * 6 + Int(form.rawValue)]!, _1, _2) } - public func Call_Hours(_ value: Int32) -> String { + public func Wallpaper_DeleteConfirmation(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[36 * 6 + Int(form.rawValue)]!, stringValue) } - public func Conversation_SelectedMessages(_ value: Int32) -> String { + public func Conversation_TitleReplies(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[37 * 6 + Int(form.rawValue)]!, stringValue) } - public func MessageTimer_Months(_ value: Int32) -> String { + public func InviteText_ContactsCountText(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[38 * 6 + Int(form.rawValue)]!, stringValue) } - public func MessageTimer_ShortSeconds(_ value: Int32) -> String { + public func OldChannels_InactiveMonth(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[39 * 6 + Int(form.rawValue)]!, stringValue) } - public func SharedMedia_Generic(_ value: Int32) -> String { + public func Conversation_StatusOnline(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[40 * 6 + Int(form.rawValue)]!, stringValue) } - public func MuteExpires_Minutes(_ value: Int32) -> String { + public func Notifications_ExceptionMuteExpires_Hours(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[41 * 6 + Int(form.rawValue)]!, stringValue) } - public func Watch_UserInfo_Mute(_ value: Int32) -> String { + public func ChatList_MessagePhotos(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[42 * 6 + Int(form.rawValue)]!, stringValue) } - public func Notification_GameScoreSelfSimple(_ value: Int32) -> String { + public func ForwardedAudios(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[43 * 6 + Int(form.rawValue)]!, stringValue) } - public func CreatePoll_AddMoreOptions(_ value: Int32) -> String { + public func Stats_GroupShowMoreTopAdmins(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[44 * 6 + Int(form.rawValue)]!, stringValue) } - public func OldChannels_InactiveYear(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[45 * 6 + Int(form.rawValue)]!, stringValue) + public func PUSH_MESSAGE_FWDS(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { + let form = getPluralizationForm(self.lc, selector) + return String(format: self._ps[45 * 6 + Int(form.rawValue)]!, _1, _2) } - public func Watch_LastSeen_HoursAgo(_ value: Int32) -> String { + public func Call_ShortMinutes(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[46 * 6 + Int(form.rawValue)]!, stringValue) } - public func VoiceOver_Chat_ContactEmailCount(_ value: Int32) -> String { + public func Call_Days(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[47 * 6 + Int(form.rawValue)]!, stringValue) } - public func Stats_GroupTopPosterChars(_ value: Int32) -> String { + public func MessageTimer_Minutes(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[48 * 6 + Int(form.rawValue)]!, stringValue) @@ -5260,469 +5307,471 @@ public final class PresentationStrings: Equatable { let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[49 * 6 + Int(form.rawValue)]!, stringValue) } - public func Stats_MessageForwards(_ value: Int32) -> String { + public func ForwardedAuthorsOthers(_ selector: Int32, _ _0: String, _ _1: String) -> String { + let form = getPluralizationForm(self.lc, selector) + return String(format: self._ps[50 * 6 + Int(form.rawValue)]!, _0, _1) + } + public func MessageTimer_ShortHours(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[50 * 6 + Int(form.rawValue)]!, stringValue) + return String(format: self._ps[51 * 6 + Int(form.rawValue)]!, stringValue) } - public func PUSH_CHANNEL_MESSAGE_VIDEOS(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { - let form = getPluralizationForm(self.lc, selector) - return String(format: self._ps[51 * 6 + Int(form.rawValue)]!, _1, _2) - } - public func GroupInfo_ShowMoreMembers(_ value: Int32) -> String { + public func ForwardedStickers(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[52 * 6 + Int(form.rawValue)]!, stringValue) } - public func AttachmentMenu_SendGif(_ value: Int32) -> String { + public func ForwardedVideos(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[53 * 6 + Int(form.rawValue)]!, stringValue) } - public func SharedMedia_DeleteItemsConfirmation(_ value: Int32) -> String { + public func Stats_GroupShowMoreTopInviters(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[54 * 6 + Int(form.rawValue)]!, stringValue) } - public func StickerPack_RemoveStickerCount(_ value: Int32) -> String { + public func Stats_GroupTopAdminKicks(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[55 * 6 + Int(form.rawValue)]!, stringValue) } - public func Notification_GameScoreSelfExtended(_ value: Int32) -> String { + public func MessageTimer_ShortWeeks(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[56 * 6 + Int(form.rawValue)]!, stringValue) } - public func MessageTimer_Years(_ value: Int32) -> String { + public func OldChannels_InactiveWeek(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[57 * 6 + Int(form.rawValue)]!, stringValue) } - public func VoiceOver_Chat_PollVotes(_ value: Int32) -> String { + public func ServiceMessage_GameScoreExtended(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[58 * 6 + Int(form.rawValue)]!, stringValue) } - public func ChatList_SelectedChats(_ value: Int32) -> String { + public func Stats_GroupTopPosterMessages(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[59 * 6 + Int(form.rawValue)]!, stringValue) } - public func Conversation_LiveLocationMembersCount(_ value: Int32) -> String { + public func MuteFor_Days(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[60 * 6 + Int(form.rawValue)]!, stringValue) } - public func MessagePoll_QuizCount(_ value: Int32) -> String { + public func ServiceMessage_GameScoreSelfExtended(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[61 * 6 + Int(form.rawValue)]!, stringValue) } - public func PrivacyLastSeenSettings_AddUsers(_ value: Int32) -> String { + public func Conversation_ContextMenuSelectAll(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[62 * 6 + Int(form.rawValue)]!, stringValue) } - public func Notification_GameScoreSimple(_ value: Int32) -> String { + public func SharedMedia_Photo(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[63 * 6 + Int(form.rawValue)]!, stringValue) } - public func AttachmentMenu_SendVideo(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[64 * 6 + Int(form.rawValue)]!, stringValue) + public func PUSH_MESSAGES(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { + let form = getPluralizationForm(self.lc, selector) + return String(format: self._ps[64 * 6 + Int(form.rawValue)]!, _1, _2) } - public func StickerPack_AddMaskCount(_ value: Int32) -> String { + public func Chat_MessagesUnpinned(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[65 * 6 + Int(form.rawValue)]!, stringValue) } - public func MessageTimer_Days(_ value: Int32) -> String { + public func AttachmentMenu_SendItem(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[66 * 6 + Int(form.rawValue)]!, stringValue) } - public func GroupInfo_ParticipantCount(_ value: Int32) -> String { + public func VoiceOver_Chat_ContactEmailCount(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[67 * 6 + Int(form.rawValue)]!, stringValue) } - public func Call_Seconds(_ value: Int32) -> String { + public func InviteLink_PeopleJoined(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[68 * 6 + Int(form.rawValue)]!, stringValue) } - public func PUSH_CHAT_MESSAGE_ROUNDS(_ selector: Int32, _ _2: String, _ _1: String, _ _3: Int32) -> String { + public func PUSH_CHAT_MESSAGES(_ selector: Int32, _ _2: String, _ _1: String, _ _3: Int32) -> String { let form = getPluralizationForm(self.lc, selector) return String(format: self._ps[69 * 6 + Int(form.rawValue)]!, _2, _1, _3) } - public func Conversation_MessageViewComments(_ value: Int32) -> String { + public func Contacts_ImportersCount(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[70 * 6 + Int(form.rawValue)]!, stringValue) } - public func MuteExpires_Days(_ value: Int32) -> String { + public func Conversation_LiveLocationMembersCount(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[71 * 6 + Int(form.rawValue)]!, stringValue) } - public func Stats_GroupShowMoreTopInviters(_ value: Int32) -> String { + public func ForwardedPhotos(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[72 * 6 + Int(form.rawValue)]!, stringValue) } - public func PUSH_CHANNEL_MESSAGE_PHOTOS(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { - let form = getPluralizationForm(self.lc, selector) - return String(format: self._ps[73 * 6 + Int(form.rawValue)]!, _1, _2) + public func StickerPack_AddStickerCount(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[73 * 6 + Int(form.rawValue)]!, stringValue) } - public func Conversation_StatusMembers(_ value: Int32) -> String { + public func Call_Minutes(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[74 * 6 + Int(form.rawValue)]!, stringValue) } - public func Chat_DeleteMessagesConfirmation(_ value: Int32) -> String { + public func ServiceMessage_GameScoreSelfSimple(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[75 * 6 + Int(form.rawValue)]!, stringValue) } - public func Stats_GroupShowMoreTopAdmins(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[76 * 6 + Int(form.rawValue)]!, stringValue) - } - public func MuteExpires_Hours(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[77 * 6 + Int(form.rawValue)]!, stringValue) - } - public func PUSH_CHANNEL_MESSAGE_ROUNDS(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { + public func PUSH_CHANNEL_MESSAGE_VIDEOS(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { let form = getPluralizationForm(self.lc, selector) - return String(format: self._ps[78 * 6 + Int(form.rawValue)]!, _1, _2) + return String(format: self._ps[76 * 6 + Int(form.rawValue)]!, _1, _2) } public func Stats_MessageViews(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[77 * 6 + Int(form.rawValue)]!, stringValue) + } + public func MessageTimer_Seconds(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[78 * 6 + Int(form.rawValue)]!, stringValue) + } + public func StickerPack_RemoveMaskCount(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[79 * 6 + Int(form.rawValue)]!, stringValue) } - public func PUSH_MESSAGE_VIDEOS(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { + public func PUSH_CHANNEL_MESSAGE_ROUNDS(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { let form = getPluralizationForm(self.lc, selector) return String(format: self._ps[80 * 6 + Int(form.rawValue)]!, _1, _2) } - public func MessageTimer_ShortWeeks(_ value: Int32) -> String { + public func VoiceChat_Panel_Members(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[81 * 6 + Int(form.rawValue)]!, stringValue) } - public func VoiceOver_Chat_MessagesSelected(_ value: Int32) -> String { + public func MessageTimer_Months(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[82 * 6 + Int(form.rawValue)]!, stringValue) } - public func LastSeen_MinutesAgo(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[83 * 6 + Int(form.rawValue)]!, stringValue) + public func PUSH_CHAT_MESSAGE_DOCS_FIX1(_ selector: Int32, _ _2: String, _ _1: String, _ _3: Int32) -> String { + let form = getPluralizationForm(self.lc, selector) + return String(format: self._ps[83 * 6 + Int(form.rawValue)]!, _2, _1, _3) } - public func ForwardedLocations(_ value: Int32) -> String { + public func Watch_UserInfo_Mute(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[84 * 6 + Int(form.rawValue)]!, stringValue) } - public func SharedMedia_Link(_ value: Int32) -> String { + public func PollResults_ShowMore(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[85 * 6 + Int(form.rawValue)]!, stringValue) } - public func Notifications_Exceptions(_ value: Int32) -> String { + public func SharedMedia_File(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[86 * 6 + Int(form.rawValue)]!, stringValue) } - public func ChatList_MessageFiles(_ value: Int32) -> String { + public func Notifications_Exceptions(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[87 * 6 + Int(form.rawValue)]!, stringValue) } - public func ChatList_DeletedChats(_ value: Int32) -> String { + public func Stats_GroupTopInviterInvites(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[88 * 6 + Int(form.rawValue)]!, stringValue) } - public func LiveLocation_MenuChatsCount(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[89 * 6 + Int(form.rawValue)]!, stringValue) - } - public func PUSH_MESSAGE_ROUNDS(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { + public func PUSH_MESSAGE_PHOTOS(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { let form = getPluralizationForm(self.lc, selector) - return String(format: self._ps[90 * 6 + Int(form.rawValue)]!, _1, _2) - } - public func VoiceOver_Chat_PollOptionCount(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[91 * 6 + Int(form.rawValue)]!, stringValue) - } - public func LiveLocationUpdated_MinutesAgo(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[92 * 6 + Int(form.rawValue)]!, stringValue) - } - public func Stats_GroupTopAdminDeletions(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[93 * 6 + Int(form.rawValue)]!, stringValue) - } - public func ServiceMessage_GameScoreSelfExtended(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[94 * 6 + Int(form.rawValue)]!, stringValue) - } - public func StickerPack_AddStickerCount(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[95 * 6 + Int(form.rawValue)]!, stringValue) - } - public func InviteText_ContactsCountText(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[96 * 6 + Int(form.rawValue)]!, stringValue) - } - public func OldChannels_InactiveMonth(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[97 * 6 + Int(form.rawValue)]!, stringValue) - } - public func Notification_GameScoreExtended(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[98 * 6 + Int(form.rawValue)]!, stringValue) - } - public func Stats_GroupTopPosterMessages(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[99 * 6 + Int(form.rawValue)]!, stringValue) - } - public func Call_Days(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[100 * 6 + Int(form.rawValue)]!, stringValue) - } - public func AttachmentMenu_SendItem(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[101 * 6 + Int(form.rawValue)]!, stringValue) + return String(format: self._ps[89 * 6 + Int(form.rawValue)]!, _1, _2) } public func QuickSend_Photos(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[90 * 6 + Int(form.rawValue)]!, stringValue) + } + public func Forward_ConfirmMultipleFiles(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[91 * 6 + Int(form.rawValue)]!, stringValue) + } + public func VoiceOver_Chat_MessagesSelected(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[92 * 6 + Int(form.rawValue)]!, stringValue) + } + public func GroupInfo_ParticipantCount(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[93 * 6 + Int(form.rawValue)]!, stringValue) + } + public func Stats_MessageForwards(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[94 * 6 + Int(form.rawValue)]!, stringValue) + } + public func LastSeen_MinutesAgo(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[95 * 6 + Int(form.rawValue)]!, stringValue) + } + public func PUSH_CHAT_MESSAGE_VIDEOS(_ selector: Int32, _ _2: String, _ _1: String, _ _3: Int32) -> String { + let form = getPluralizationForm(self.lc, selector) + return String(format: self._ps[96 * 6 + Int(form.rawValue)]!, _2, _1, _3) + } + public func MuteExpires_Days(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[97 * 6 + Int(form.rawValue)]!, stringValue) + } + public func OldChannels_Leave(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[98 * 6 + Int(form.rawValue)]!, stringValue) + } + public func MessagePoll_VotedCount(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[99 * 6 + Int(form.rawValue)]!, stringValue) + } + public func PUSH_MESSAGE_DOCS(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { + let form = getPluralizationForm(self.lc, selector) + return String(format: self._ps[100 * 6 + Int(form.rawValue)]!, _1, _2) + } + public func ChatList_Search_Messages(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[101 * 6 + Int(form.rawValue)]!, stringValue) + } + public func Media_ShareItem(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[102 * 6 + Int(form.rawValue)]!, stringValue) } - public func PUSH_CHANNEL_MESSAGE_FWDS(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { - let form = getPluralizationForm(self.lc, selector) - return String(format: self._ps[103 * 6 + Int(form.rawValue)]!, _1, _2) + public func VoiceOver_Chat_PollVotes(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[103 * 6 + Int(form.rawValue)]!, stringValue) } public func Stats_GroupTopAdminBans(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[104 * 6 + Int(form.rawValue)]!, stringValue) } - public func VoiceChat_Panel_Members(_ value: Int32) -> String { + public func LastSeen_HoursAgo(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[105 * 6 + Int(form.rawValue)]!, stringValue) } - public func MessageTimer_Minutes(_ value: Int32) -> String { + public func Notification_GameScoreSelfExtended(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[106 * 6 + Int(form.rawValue)]!, stringValue) } - public func ForwardedContacts(_ value: Int32) -> String { + public func Passport_Scans(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[107 * 6 + Int(form.rawValue)]!, stringValue) } - public func Conversation_StatusSubscribers(_ value: Int32) -> String { + public func OldChannels_InactiveYear(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[108 * 6 + Int(form.rawValue)]!, stringValue) } - public func PUSH_CHAT_MESSAGE_PHOTOS(_ selector: Int32, _ _2: String, _ _1: String, _ _3: Int32) -> String { - let form = getPluralizationForm(self.lc, selector) - return String(format: self._ps[109 * 6 + Int(form.rawValue)]!, _2, _1, _3) - } - public func PUSH_CHAT_MESSAGES(_ selector: Int32, _ _2: String, _ _1: String, _ _3: Int32) -> String { - let form = getPluralizationForm(self.lc, selector) - return String(format: self._ps[110 * 6 + Int(form.rawValue)]!, _2, _1, _3) - } - public func Conversation_ContextViewReplies(_ value: Int32) -> String { + public func PasscodeSettings_FailedAttempts(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[111 * 6 + Int(form.rawValue)]!, stringValue) - } - public func ChatList_Search_Messages(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[112 * 6 + Int(form.rawValue)]!, stringValue) - } - public func ChatList_MessageMusic(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[113 * 6 + Int(form.rawValue)]!, stringValue) - } - public func SharedMedia_File(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[114 * 6 + Int(form.rawValue)]!, stringValue) - } - public func OldChannels_InactiveWeek(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[115 * 6 + Int(form.rawValue)]!, stringValue) - } - public func Watch_LastSeen_MinutesAgo(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[116 * 6 + Int(form.rawValue)]!, stringValue) - } - public func Map_ETAMinutes(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[117 * 6 + Int(form.rawValue)]!, stringValue) - } - public func ForwardedMessages(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[118 * 6 + Int(form.rawValue)]!, stringValue) - } - public func Map_ETAHours(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[119 * 6 + Int(form.rawValue)]!, stringValue) - } - public func Call_ShortSeconds(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[120 * 6 + Int(form.rawValue)]!, stringValue) - } - public func VoiceChat_Status_Members(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[121 * 6 + Int(form.rawValue)]!, stringValue) + return String(format: self._ps[109 * 6 + Int(form.rawValue)]!, stringValue) } public func UserCount(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[122 * 6 + Int(form.rawValue)]!, stringValue) + return String(format: self._ps[110 * 6 + Int(form.rawValue)]!, stringValue) } - public func PUSH_CHANNEL_MESSAGES(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { - let form = getPluralizationForm(self.lc, selector) - return String(format: self._ps[123 * 6 + Int(form.rawValue)]!, _1, _2) - } - public func ForwardedPhotos(_ value: Int32) -> String { + public func AttachmentMenu_SendGif(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[124 * 6 + Int(form.rawValue)]!, stringValue) + return String(format: self._ps[111 * 6 + Int(form.rawValue)]!, stringValue) } - public func Chat_MessagesUnpinned(_ value: Int32) -> String { + public func MessagePoll_QuizCount(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[125 * 6 + Int(form.rawValue)]!, stringValue) + return String(format: self._ps[112 * 6 + Int(form.rawValue)]!, stringValue) } - public func ForwardedGifs(_ value: Int32) -> String { + public func Contacts_InviteContacts(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[126 * 6 + Int(form.rawValue)]!, stringValue) + return String(format: self._ps[113 * 6 + Int(form.rawValue)]!, stringValue) } - public func ChatList_MessagePhotos(_ value: Int32) -> String { + public func ServiceMessage_GameScoreSimple(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[127 * 6 + Int(form.rawValue)]!, stringValue) + return String(format: self._ps[114 * 6 + Int(form.rawValue)]!, stringValue) } - public func MessagePoll_VotedCount(_ value: Int32) -> String { + public func SharedMedia_DeleteItemsConfirmation(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[128 * 6 + Int(form.rawValue)]!, stringValue) + return String(format: self._ps[115 * 6 + Int(form.rawValue)]!, stringValue) } - public func PUSH_MESSAGES(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { - let form = getPluralizationForm(self.lc, selector) - return String(format: self._ps[129 * 6 + Int(form.rawValue)]!, _1, _2) - } - public func ForwardedVideoMessages(_ value: Int32) -> String { + public func MessageTimer_Hours(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[130 * 6 + Int(form.rawValue)]!, stringValue) + return String(format: self._ps[116 * 6 + Int(form.rawValue)]!, stringValue) } - public func MessageTimer_Seconds(_ value: Int32) -> String { + public func Conversation_TitleComments(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[131 * 6 + Int(form.rawValue)]!, stringValue) + return String(format: self._ps[117 * 6 + Int(form.rawValue)]!, stringValue) } - public func ForwardedAuthorsOthers(_ selector: Int32, _ _0: String, _ _1: String) -> String { - let form = getPluralizationForm(self.lc, selector) - return String(format: self._ps[132 * 6 + Int(form.rawValue)]!, _0, _1) - } - public func MuteFor_Hours(_ value: Int32) -> String { + public func AttachmentMenu_SendPhoto(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[133 * 6 + Int(form.rawValue)]!, stringValue) + return String(format: self._ps[118 * 6 + Int(form.rawValue)]!, stringValue) + } + public func StickerPack_RemoveStickerCount(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[119 * 6 + Int(form.rawValue)]!, stringValue) + } + public func ChatList_MessageMusic(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[120 * 6 + Int(form.rawValue)]!, stringValue) } public func Media_ShareVideo(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[134 * 6 + Int(form.rawValue)]!, stringValue) - } - public func ForwardedFiles(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[135 * 6 + Int(form.rawValue)]!, stringValue) - } - public func PeopleNearby_ShowMorePeople(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[136 * 6 + Int(form.rawValue)]!, stringValue) - } - public func Forward_ConfirmMultipleFiles(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[137 * 6 + Int(form.rawValue)]!, stringValue) - } - public func Conversation_TitleReplies(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[138 * 6 + Int(form.rawValue)]!, stringValue) - } - public func Media_SharePhoto(_ value: Int32) -> String { - let form = getPluralizationForm(self.lc, value) - let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[139 * 6 + Int(form.rawValue)]!, stringValue) + return String(format: self._ps[121 * 6 + Int(form.rawValue)]!, stringValue) } public func DialogList_LiveLocationChatsCount(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[140 * 6 + Int(form.rawValue)]!, stringValue) + return String(format: self._ps[122 * 6 + Int(form.rawValue)]!, stringValue) + } + public func LiveLocation_MenuChatsCount(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[123 * 6 + Int(form.rawValue)]!, stringValue) + } + public func Notifications_ExceptionMuteExpires_Minutes(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[124 * 6 + Int(form.rawValue)]!, stringValue) + } + public func MuteExpires_Hours(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[125 * 6 + Int(form.rawValue)]!, stringValue) + } + public func GroupInfo_ShowMoreMembers(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[126 * 6 + Int(form.rawValue)]!, stringValue) + } + public func Call_Hours(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[127 * 6 + Int(form.rawValue)]!, stringValue) + } + public func Stats_GroupShowMoreTopPosters(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[128 * 6 + Int(form.rawValue)]!, stringValue) + } + public func StickerPack_StickerCount(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[129 * 6 + Int(form.rawValue)]!, stringValue) + } + public func Conversation_ContextViewReplies(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[130 * 6 + Int(form.rawValue)]!, stringValue) + } + public func Conversation_MessageViewComments(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[131 * 6 + Int(form.rawValue)]!, stringValue) + } + public func InstantPage_Views(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[132 * 6 + Int(form.rawValue)]!, stringValue) + } + public func ChatList_MessageFiles(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[133 * 6 + Int(form.rawValue)]!, stringValue) + } + public func PUSH_MESSAGE_ROUNDS(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { + let form = getPluralizationForm(self.lc, selector) + return String(format: self._ps[134 * 6 + Int(form.rawValue)]!, _1, _2) + } + public func OldChannels_GroupFormat(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[135 * 6 + Int(form.rawValue)]!, stringValue) + } + public func Notification_GameScoreSimple(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[136 * 6 + Int(form.rawValue)]!, stringValue) + } + public func ForwardedMessages(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[137 * 6 + Int(form.rawValue)]!, stringValue) + } + public func MuteExpires_Minutes(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[138 * 6 + Int(form.rawValue)]!, stringValue) + } + public func ForwardedFiles(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[139 * 6 + Int(form.rawValue)]!, stringValue) } public func PUSH_CHANNEL_MESSAGE_DOCS(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { let form = getPluralizationForm(self.lc, selector) - return String(format: self._ps[141 * 6 + Int(form.rawValue)]!, _1, _2) + return String(format: self._ps[140 * 6 + Int(form.rawValue)]!, _1, _2) } - public func OldChannels_Leave(_ value: Int32) -> String { + public func ForwardedVideoMessages(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[141 * 6 + Int(form.rawValue)]!, stringValue) + } + public func Chat_TitlePinnedMessages(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[142 * 6 + Int(form.rawValue)]!, stringValue) } - public func StickerPack_RemoveMaskCount(_ value: Int32) -> String { + public func VoiceOver_Chat_PollOptionCount(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[143 * 6 + Int(form.rawValue)]!, stringValue) } - public func ForwardedVideos(_ value: Int32) -> String { + public func ForwardedGifs(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[144 * 6 + Int(form.rawValue)]!, stringValue) } - public func Contacts_ImportersCount(_ value: Int32) -> String { + public func Watch_LastSeen_HoursAgo(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[145 * 6 + Int(form.rawValue)]!, stringValue) @@ -5732,58 +5781,68 @@ public final class PresentationStrings: Equatable { let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[146 * 6 + Int(form.rawValue)]!, stringValue) } - public func ServiceMessage_GameScoreSelfSimple(_ value: Int32) -> String { + public func ForwardedLocations(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[147 * 6 + Int(form.rawValue)]!, stringValue) } - public func Theme_UsersCount(_ value: Int32) -> String { + public func Stats_GroupTopAdminDeletions(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[148 * 6 + Int(form.rawValue)]!, stringValue) } - public func AttachmentMenu_SendPhoto(_ value: Int32) -> String { + public func Map_ETAMinutes(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[149 * 6 + Int(form.rawValue)]!, stringValue) } - public func Wallpaper_DeleteConfirmation(_ value: Int32) -> String { + public func Theme_UsersCount(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[150 * 6 + Int(form.rawValue)]!, stringValue) } - public func PUSH_CHAT_MESSAGE_VIDEOS(_ selector: Int32, _ _2: String, _ _1: String, _ _3: Int32) -> String { - let form = getPluralizationForm(self.lc, selector) - return String(format: self._ps[151 * 6 + Int(form.rawValue)]!, _2, _1, _3) + public func VoiceOver_Chat_ContactPhoneNumberCount(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[151 * 6 + Int(form.rawValue)]!, stringValue) } - public func ForwardedAudios(_ value: Int32) -> String { + public func AttachmentMenu_SendVideo(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[152 * 6 + Int(form.rawValue)]!, stringValue) } - public func Call_Minutes(_ value: Int32) -> String { + public func MessageTimer_Weeks(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[153 * 6 + Int(form.rawValue)]!, stringValue) } - public func Media_ShareItem(_ value: Int32) -> String { + public func ForwardedContacts(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[154 * 6 + Int(form.rawValue)]!, stringValue) } - public func PUSH_CHAT_MESSAGE_DOCS_FIX1(_ selector: Int32, _ _2: String, _ _1: String, _ _3: Int32) -> String { + public func PUSH_MESSAGE_VIDEOS(_ selector: Int32, _ _1: String, _ _2: Int32) -> String { let form = getPluralizationForm(self.lc, selector) - return String(format: self._ps[155 * 6 + Int(form.rawValue)]!, _2, _1, _3) + return String(format: self._ps[155 * 6 + Int(form.rawValue)]!, _1, _2) } - public func Conversation_ContextMenuSelectAll(_ value: Int32) -> String { + public func Stats_GroupTopPosterChars(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) return String(format: self._ps[156 * 6 + Int(form.rawValue)]!, stringValue) } - public func ServiceMessage_GameScoreExtended(_ value: Int32) -> String { + public func PUSH_CHAT_MESSAGE_PHOTOS(_ selector: Int32, _ _2: String, _ _1: String, _ _3: Int32) -> String { + let form = getPluralizationForm(self.lc, selector) + return String(format: self._ps[157 * 6 + Int(form.rawValue)]!, _2, _1, _3) + } + public func ForwardedPolls(_ value: Int32) -> String { let form = getPluralizationForm(self.lc, value) let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) - return String(format: self._ps[157 * 6 + Int(form.rawValue)]!, stringValue) + return String(format: self._ps[158 * 6 + Int(form.rawValue)]!, stringValue) + } + public func ChatList_DeletedChats(_ value: Int32) -> String { + let form = getPluralizationForm(self.lc, value) + let stringValue = presentationStringsFormattedNumber(value, self.groupingSeparator) + return String(format: self._ps[159 * 6 + Int(form.rawValue)]!, stringValue) } public init(primaryComponent: PresentationStringsComponent, secondaryComponent: PresentationStringsComponent?, groupingSeparator: String) { diff --git a/submodules/TelegramPresentationData/Sources/Resources/PresentationResourceKey.swift b/submodules/TelegramPresentationData/Sources/Resources/PresentationResourceKey.swift index 0c77121d4f..6f50f080b7 100644 --- a/submodules/TelegramPresentationData/Sources/Resources/PresentationResourceKey.swift +++ b/submodules/TelegramPresentationData/Sources/Resources/PresentationResourceKey.swift @@ -38,8 +38,10 @@ public enum PresentationResourceKey: Int32 { case itemListCheckIcon case itemListSecondaryCheckIcon case itemListPlusIcon + case itemListDeleteIcon case itemListDeleteIndicatorIcon case itemListReorderIndicatorIcon + case itemListLinkIcon case itemListAddPersonIcon case itemListCreateGroupIcon case itemListAddExceptionIcon diff --git a/submodules/TelegramPresentationData/Sources/Resources/PresentationResourcesItemList.swift b/submodules/TelegramPresentationData/Sources/Resources/PresentationResourcesItemList.swift index aaf11406da..e9ae56fc13 100644 --- a/submodules/TelegramPresentationData/Sources/Resources/PresentationResourcesItemList.swift +++ b/submodules/TelegramPresentationData/Sources/Resources/PresentationResourcesItemList.swift @@ -51,6 +51,12 @@ public struct PresentationResourcesItemList { }) } + public static func deleteIconImage(_ theme: PresentationTheme) -> UIImage? { + return theme.image(PresentationResourceKey.itemListDeleteIcon.rawValue, { theme in + return generateTintedImage(image: UIImage(bundleImageName: "Chat/Input/Accessory Panels/MessageSelectionTrash"), color: theme.list.itemDestructiveColor) + }) + } + public static func stickerUnreadDotImage(_ theme: PresentationTheme) -> UIImage? { return theme.image(PresentationResourceKey.itemListStickerItemUnreadDot.rawValue, { theme in return generateFilledCircleImage(diameter: 6.0, color: theme.list.itemAccentColor) @@ -90,6 +96,12 @@ public struct PresentationResourcesItemList { }) } + public static func linkIcon(_ theme: PresentationTheme) -> UIImage? { + return theme.image(PresentationResourceKey.itemListLinkIcon.rawValue, { theme in + return generateTintedImage(image: UIImage(bundleImageName: "Contact List/LinkActionIcon"), color: theme.list.itemAccentColor) + }) + } + public static func addPersonIcon(_ theme: PresentationTheme) -> UIImage? { return theme.image(PresentationResourceKey.itemListAddPersonIcon.rawValue, { theme in return generateTintedImage(image: UIImage(bundleImageName: "Contact List/AddMemberIcon"), color: theme.list.itemAccentColor) diff --git a/submodules/TelegramUI/Images.xcassets/Chat/Contents.json b/submodules/TelegramUI/Images.xcassets/Chat/Contents.json index 38f0c81fc2..6e965652df 100644 --- a/submodules/TelegramUI/Images.xcassets/Chat/Contents.json +++ b/submodules/TelegramUI/Images.xcassets/Chat/Contents.json @@ -1,9 +1,9 @@ { "info" : { - "version" : 1, - "author" : "xcode" + "author" : "xcode", + "version" : 1 }, "properties" : { "provides-namespace" : true } -} \ No newline at end of file +} diff --git a/submodules/TelegramUI/Images.xcassets/Chat/Links/Contents.json b/submodules/TelegramUI/Images.xcassets/Chat/Links/Contents.json new file mode 100644 index 0000000000..6e965652df --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Chat/Links/Contents.json @@ -0,0 +1,9 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "provides-namespace" : true + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Settings/EditAccount.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Chat/Links/Expired.imageset/Contents.json similarity index 75% rename from submodules/TelegramUI/Images.xcassets/Settings/EditAccount.imageset/Contents.json rename to submodules/TelegramUI/Images.xcassets/Chat/Links/Expired.imageset/Contents.json index 101f377488..3048d94181 100644 --- a/submodules/TelegramUI/Images.xcassets/Settings/EditAccount.imageset/Contents.json +++ b/submodules/TelegramUI/Images.xcassets/Chat/Links/Expired.imageset/Contents.json @@ -1,7 +1,7 @@ { "images" : [ { - "filename" : "ic_editaccount.pdf", + "filename" : "ic_linkexpired.pdf", "idiom" : "universal" } ], diff --git a/submodules/TelegramUI/Images.xcassets/Wallet/SendButtonIcon.imageset/Group.pdf b/submodules/TelegramUI/Images.xcassets/Chat/Links/Expired.imageset/ic_linkexpired.pdf similarity index 54% rename from submodules/TelegramUI/Images.xcassets/Wallet/SendButtonIcon.imageset/Group.pdf rename to submodules/TelegramUI/Images.xcassets/Chat/Links/Expired.imageset/ic_linkexpired.pdf index 7d261978993cc449ecf87f182616be59081f69e3..d040d312a16aa9c2e708c9899cd354e8665a47d6 100644 GIT binary patch delta 2295 zcmah~2~bm67G+a31XN^^X8W=zsGa;QBm_th!rDSaRAdnh%NG<$APE>ij3{A~B_Sw{ zEvu+30zwc+1Pm=83XUj(EFznr=~2{xVE~yK(oIb{JySKWUcI_?-Z}T(y7yI`$NCNC zJS`Xo00DZ?NtlfdAECuTxm7jFrl$gg+R6RGifRua!#mmZsxOH@8vyEsqQs!Cr8U9MOel1LlsyoAxu#4{g_ z>-2(ich4B`9ZqK6a}vge8sbtvyr~+*(+x)@as91#mae9A@>Q#(B^?pnF;2p>&XP-m zP0K%ZKbI11UQxWh&J(q5tN96wcRq>rrXzM$qpDhdOU)P!)m`{mM?ikmG7#{~VdKtm zR!x@08D@wtQbl6ac)ERu9sZY#2^#l*Jl{1x`c$LhBSm!5=~xrO`z?V`sX%3CRYt8l z%{^G4)jsSvz_Ipv`#9=N|GUAbpP(V@ktKf;u40tfD4lL$#AbJY`m7L+xF+{xI-uB3 zn7Y^!I`~2|>_bJA+3^G)Vk#R$4GYpQZgxJOH(o97dz^iApm6Gh;@(ZZ zK9dLoy>O2VtxYCY$bnuL1znX56f1qr=rCKoF;41VS7zTUOC;+WnbeYp?nl1$7S%Kd z1gRfNXlye+_^M9s&^mTBQ^nmlS+_l|Tv;v5-ec!W>#Bj*rk@}GWP4!6_h^F17g4|) z7%#tReBZ_SjNiHy!bk=>H?QRKk43yuot(q)(G5D4+}MVT>FSl~B{%<~1VhCcot{wU z4!$qvJXmgOJ(Afl&s@G1{g2GKg;&V^<5|-9_=ShMD{S+yFRZc%OkK_3Am7v2+!uIr$J4V2O(|#P4q;-p|_)hwih5oUVP76m_(2O)%Aq|qN%cH<0eJ$d)f>{@>b;`GgUII zLRT)H*t}QUEa^==@7fPr7sE(8oVMaqY_)GzSCYoj8nZM0Nrl2%Ic`JbP_VVbZAO&z zeiP-+o$q*h<0MMm4E> z&%lRLr?J47T}t}R)_RzFZgHw*f902RL(05vOvxS}%y_3#|B)aoyDF^H5XaRwa-}`X zumFgSA{0cHrEVu=kw#Qd(11DYn^ykq52wbzI&%vpU6fKx@>>+-`A&0xa3U+j@N@BP zKVS)eDm%b?2ghTDwVZP_@a*F*UU>yn`0EWhzv7~3U20_xY-{nAC%Iklo&Fsz z2k0^PFT&D+JdF;W>UTN>&ICT?tG==IMvn(&Q3z9J72?~X^$U{y&XW20PmKF=T}J|o9Zus!wbe!O_DukDuY z5#r)+oWparcAR3Da$lSbqH!oRA7Lb{k#~UU%HEeH^aGHtWU_rA6AA%jGEcyt-(n!J zg>Ny$fB**c2asfXG>rwI0OYzic3jATZbh8-x!I z3br5+EJ2io1s=lVp&$z=7>mJ>VE>;2_{)0AWCb!nkX92Lyni#3jHCfOQ3+NoV?eYibke!0kvJnW6VZi4 z7u~tA%rm&K3qkY+T)J{0zJMrQxbYFZH)%9!7Y_4tIp3M@JLlZ{vG{rZ_3Rl9F!=3< zYPAYzm3!OK1JDIz@4{@nm(C29nkSltCbkQAar1ZLdr3(yvz=At`F`6{_BWLz&OAw7 zqBE?2@@2-dc7C)JD|ssMHKm#@cw?%cHk0q)Dn;eji-J0HO#gDWrgF~>qkub)IR_Zv&S6JnzKE82i%K*NrixH9{T-Yi-%y00L%9xQ4c}7vY|E{CR-cv zw-jC~AD|P3ihvH;Hzh0XrRR6g=RfSF*VKwcB#9b?jwQ-o5?cq%CJBi34fB z5c8_8SKRn#wup#{O^UIpn~dDY_@cOrfNuG2-HA99>Q^Y9q{Pd)x#J|hmzygws)?NB z1suMtNl*0qjiYc)+sf_c#Q dn)-jrXkWs}38JR~@2E_}6?Jj(+U>-3^$-8J&{F^a diff --git a/submodules/TelegramUI/Images.xcassets/Chat/Links/Flame.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Chat/Links/Flame.imageset/Contents.json new file mode 100644 index 0000000000..82d4b2d05d --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Chat/Links/Flame.imageset/Contents.json @@ -0,0 +1,12 @@ +{ + "images" : [ + { + "filename" : "ic_linkfire.pdf", + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Wallet/ReceiveButtonIcon.imageset/Group2.pdf b/submodules/TelegramUI/Images.xcassets/Chat/Links/Flame.imageset/ic_linkfire.pdf similarity index 72% rename from submodules/TelegramUI/Images.xcassets/Wallet/ReceiveButtonIcon.imageset/Group2.pdf rename to submodules/TelegramUI/Images.xcassets/Chat/Links/Flame.imageset/ic_linkfire.pdf index 1f90a5f60cafb567fc516b9e29edba6825b4b297..f5c3521b478cc228a893a21184b42e8cc0beddb1 100644 GIT binary patch delta 954 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 Iv%eb`0Dv(=3;+NC diff --git a/submodules/TelegramUI/Images.xcassets/Chat/Links/QrLogo.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Chat/Links/QrLogo.imageset/Contents.json new file mode 100644 index 0000000000..de3bb69a15 --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Chat/Links/QrLogo.imageset/Contents.json @@ -0,0 +1,12 @@ +{ + "images" : [ + { + "filename" : "ic_qrlogo.pdf", + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Settings/EditAccount.imageset/ic_editaccount.pdf b/submodules/TelegramUI/Images.xcassets/Chat/Links/QrLogo.imageset/ic_qrlogo.pdf similarity index 60% rename from submodules/TelegramUI/Images.xcassets/Settings/EditAccount.imageset/ic_editaccount.pdf rename to submodules/TelegramUI/Images.xcassets/Chat/Links/QrLogo.imageset/ic_qrlogo.pdf index 005582c532fe753afd1a785145152b5e9cd220ac..cbf4e3cb353e959026176275da3e33d1f4dbb25a 100644 GIT binary patch delta 1841 zcmah|dpML?7-!4n`D#T;E?o{KDrUZK?&Dfx=pvhR!%{nDzG2!i!_3ggQ`1F6Ax%%3 zijBmmc5I@;>Sk3+E^XxUSS#fzDbki&R(r-}OZ~Cm^L^(#=Xc)soZtJtzqj7@o$Vnr zgbYw14%v#hxTF!x6vWlIC{z{(XbdKBb3>#u3C;;eN{XY#AsJi7HzL-ox9$Et2ob9+npV58S1VXEV5n@x;)Dxb^mecc~T z4nks+<-s}|qvxhyn?7vlI_b`E`Ny*Uw@%I?8!*0-Ooq*p1sj&E>^2G zMZs9va{t$fu8OBI@`1zLN`JfJXRbxfM>}$(eZlqzni5&-j)oZ1;QYqbxWjs&^0H zAv1H-ubxlggX=lgT7Z1M72qE7jC!%D7!TSB$(v5(uz zvIc4^>Nn3dauEE~dcQR^{<&(?>7BJFLrw;zn%68#pOyN$aMPc1xh`dY*fiqnYgb>j z=0y+AaAFx2Ah?J(CSW+zY8+Ey#xtD((U&8^MKVBw()R;IKU^w~l5lY;po{_@Vv!8) zOM$jG{Iq>A=1)=1(L)uqF%%Hjg>iTSQ7CY6(ateI9k>7FqJ>7=3lKfTLa}6B1c!@* z(KE&mSRWICLpIPrEa6GP_kfH-H#kt41S*vZf&q~&4(A9&a0_)uFfHIIha7njkV&HO z(E*}fQ6jEPAQl1oIF&K#jQ@Bj<$80bjI}w5hLO?;HW!Qa*oZDoX=H4p&^nG29zSh_ z9}0*pflTU)OV)A1BZN3$z@P=&aYz9p)4m`HMmYL$M4>nWIm6}n2T6`yjxnf^us|ro zCGaog$Z$5!O!hzZYk&C}4e9 zn^UMM!lVCMTpg^1qeJA2%gH|D73%gMZ*`6^YvS8i2%SbPheC44&~lZk>OLBTBy zrjr1j1mpG)M?uwLfaon4sj@aHez2CG)WZBM`6EdudT?YMp*U2l*-)s^QOSOCz))q8 z@q!r`0VQA{$etA?5b^*CWs#W5m&S_;7>Z#ej6$YR=z%C|2j4}2=p*K_VX}ce+lhpd zJ~L5zn$XQrIs?jH`v4I@MkXi)V~~|S4OQquCG$y4CYOiOxJ*8Z@+mYv opMsGoG!B_YBa@kK$p4#+mP0DzNMv#e&PQP8=!mVY=Nd2MU!cvrZU6uP delta 1792 zcmai#eN;?&9LAN9lqQ*SguSMz3A5dM=e3w-r5dV9H6&AwnrZGR%nb7qsSFh+MlWun zs8-%rNNc!C(L&8By+0E<+c8`d(XCyR3(C!cl%j2!Mio{7u$b-dDof>$<feljiQz+anz0-Bu5k#QfdmQFoH+CzPB;+z-M4zFlP_`ofV2;kk_tVmtdLiiA)%S|j{iWiD8qVUjyDMw~- z=ay!RYMracD2mlW6PhkBYHF13!h_^en-uFbhqayiL-M_{*Kfvl2YG64WaE;%{8ZSK zqZem1^HSRI_c`^y&v?K!?Fwgc%AFhUFiR(PG=K#_hi^KuPN5ODx7$v{+fAKtY+KNF z-ogv6c_yhZU2ko5m~`k!JA0+zRBJ)4I??<=$KLUoMM~aD3@{QttF35HaB^txG2#hM zyv*`8ZnlG?Z}Xn@TK$U7t(x=Uo?)tu{mF9fTMx6?C4*6c!;P4F?>)hmjomV<-0M=y z1%DiJR95QUG-sEx#J90NcUyvqrdQGCCcEcbV|wsn!qFSqXRLU*b&J4}EQ-WI*G=)4Nvm%FU-= z&AU}JP87Y|v@dewyk+P2AahrhP%DFETVQ+ro3MUj``X;Hy_!0(Vu|Wj-@dxeov8=k z&?IFy!32VSZ`VES3t~!OsH)?+M6{~5Dp$@p-Y@kQNTW7A8%UZt^N>LT@xZ%AFBLOE$!gR;Ykg7i;^)9e14*A+dvy6-;>ay-Fz&lVbSMMt7%@HF zANfJ|XoAn5F5;1->V4I8MN)$*M|I=f z4^88ry|O?gLb*bQ@WsZJv4@Q1V|>CeOfoQ(gK<~@fI)m2B2fTjV>1CD2a!vavS>sO z&^`k%sYGFz%K>AXaR4UspWSF^)L3X9{crlkz|DaEQ+w`46}r@HW)^O0nZ_*00&65j z5s5bHWz-)XeKHwmLPjxTv8Whh^Boxi$PfAUZ^bsZ@yS9tJ@K!#xK8 z1EfN4z5)SUyyuf3(iakRjSZbhA<+%#881KrSd1eq|{Qz+;~8@vi8LEj8v z^0$lvLxw26-UKGo4dH*yD3IGXItq#Og(-!MAG;2bjs9(xq3fqu#lz5OP@u0wBsIi0 zdcMBih5#7?l1LzcAQ2TIQRpOuOs9%S?l3HHbECUa-DwEZ;{PQ8&v@o?1z)DvBtt|N P6bc2haB}hqWLf+LK$h1$ diff --git a/submodules/TelegramUI/Images.xcassets/Settings/Contents.json b/submodules/TelegramUI/Images.xcassets/Settings/Contents.json index 38f0c81fc2..6e965652df 100644 --- a/submodules/TelegramUI/Images.xcassets/Settings/Contents.json +++ b/submodules/TelegramUI/Images.xcassets/Settings/Contents.json @@ -1,9 +1,9 @@ { "info" : { - "version" : 1, - "author" : "xcode" + "author" : "xcode", + "version" : 1 }, "properties" : { "provides-namespace" : true } -} \ No newline at end of file +} diff --git a/submodules/TelegramUI/Images.xcassets/Wallet/Contents.json b/submodules/TelegramUI/Images.xcassets/Wallet/Contents.json index 38f0c81fc2..6e965652df 100644 --- a/submodules/TelegramUI/Images.xcassets/Wallet/Contents.json +++ b/submodules/TelegramUI/Images.xcassets/Wallet/Contents.json @@ -1,9 +1,9 @@ { "info" : { - "version" : 1, - "author" : "xcode" + "author" : "xcode", + "version" : 1 }, "properties" : { "provides-namespace" : true } -} \ No newline at end of file +} diff --git a/submodules/TelegramUI/Images.xcassets/Wallet/NavigationSettingsIcon.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Wallet/NavigationSettingsIcon.imageset/Contents.json deleted file mode 100644 index 31daa63843..0000000000 --- a/submodules/TelegramUI/Images.xcassets/Wallet/NavigationSettingsIcon.imageset/Contents.json +++ /dev/null @@ -1,12 +0,0 @@ -{ - "images" : [ - { - "idiom" : "universal", - "filename" : "NavigationSettingsIcon.pdf" - } - ], - "info" : { - "version" : 1, - "author" : "xcode" - } -} \ No newline at end of file diff --git a/submodules/TelegramUI/Images.xcassets/Wallet/NavigationSettingsIcon.imageset/NavigationSettingsIcon.pdf b/submodules/TelegramUI/Images.xcassets/Wallet/NavigationSettingsIcon.imageset/NavigationSettingsIcon.pdf deleted file mode 100644 index 4f188553a0ed649a4c4805c8fbee47b03f09ced1..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 5839 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 O{5(V~EHY}cME?VP++NWD diff --git a/submodules/TelegramUI/Images.xcassets/Wallet/QrGem.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Wallet/QrGem.imageset/Contents.json deleted file mode 100644 index ae81ff437c..0000000000 --- a/submodules/TelegramUI/Images.xcassets/Wallet/QrGem.imageset/Contents.json +++ /dev/null @@ -1,22 +0,0 @@ -{ - "images" : [ - { - "idiom" : "universal", - "scale" : "1x" - }, - { - "idiom" : "universal", - "filename" : "QrGem@2x.png", - "scale" : "2x" - }, - { - "idiom" : "universal", - "filename" : "QrGem@3x.png", - "scale" : "3x" - } - ], - "info" : { - "version" : 1, - "author" : "xcode" - } -} \ No newline at end of file diff --git a/submodules/TelegramUI/Images.xcassets/Wallet/QrGem.imageset/QrGem@2x.png b/submodules/TelegramUI/Images.xcassets/Wallet/QrGem.imageset/QrGem@2x.png deleted file mode 100644 index 35d164f8b318bb1bee29ead067bdf02332cc2656..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 15212 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 j1x*SzFt~x2p92Z9FkN7nR=C$!Rv diff --git a/submodules/TelegramUI/Images.xcassets/Wallet/ReceiveButtonIcon.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Wallet/ReceiveButtonIcon.imageset/Contents.json deleted file mode 100644 index aca2fcc163..0000000000 --- a/submodules/TelegramUI/Images.xcassets/Wallet/ReceiveButtonIcon.imageset/Contents.json +++ /dev/null @@ -1,12 +0,0 @@ -{ - "images" : [ - { - "idiom" : "universal", - "filename" : "Group2.pdf" - } - ], - "info" : { - "version" : 1, - "author" : "xcode" - } -} \ No newline at end of file diff --git a/submodules/TelegramUI/Images.xcassets/Wallet/RefreshIcon.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Wallet/RefreshIcon.imageset/Contents.json deleted file mode 100644 index 0940c4ae6d..0000000000 --- a/submodules/TelegramUI/Images.xcassets/Wallet/RefreshIcon.imageset/Contents.json +++ /dev/null @@ -1,12 +0,0 @@ -{ - "images" : [ - { - "idiom" : "universal", - "filename" : "ic_walletupdate.pdf" - } - ], - "info" : { - "version" : 1, - "author" : "xcode" - } -} \ No newline at end of file diff --git a/submodules/TelegramUI/Images.xcassets/Wallet/SendButtonIcon.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Wallet/SendButtonIcon.imageset/Contents.json deleted file mode 100644 index 10de2f9a6b..0000000000 --- a/submodules/TelegramUI/Images.xcassets/Wallet/SendButtonIcon.imageset/Contents.json +++ /dev/null @@ -1,12 +0,0 @@ -{ - "images" : [ - { - "idiom" : "universal", - "filename" : "Group.pdf" - } - ], - "info" : { - "version" : 1, - "author" : "xcode" - } -} \ No newline at end of file diff --git a/submodules/TelegramUI/Images.xcassets/Wallet/TransactionGem.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Wallet/TransactionGem.imageset/Contents.json deleted file mode 100644 index b5a8966f3f..0000000000 --- a/submodules/TelegramUI/Images.xcassets/Wallet/TransactionGem.imageset/Contents.json +++ /dev/null @@ -1,12 +0,0 @@ -{ - "images" : [ - { - "idiom" : "universal", - "filename" : "SmallGem.pdf" - } - ], - "info" : { - "version" : 1, - "author" : "xcode" - } -} \ No newline at end of file diff --git a/submodules/TelegramUI/Images.xcassets/Wallet/TransactionGem.imageset/SmallGem.pdf b/submodules/TelegramUI/Images.xcassets/Wallet/TransactionGem.imageset/SmallGem.pdf deleted file mode 100644 index cf1ac9ab53974c48c959f74355f1349826fe5522..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 96214 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

)48Kg&M*Nl` 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�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 diff --git a/submodules/TelegramUI/Resources/PresentationStrings.mapping b/submodules/TelegramUI/Resources/PresentationStrings.mapping index df910f75e334c3ba999f199222a710003c54fccb..dc2e7eef736afad3ebd34af1d1cce4bdf0532b6b 100644 GIT binary patch delta 50051 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 - if let peerName = peerName { - resolveSignal = resolvePeerByName(account: strongSelf.context.account, name: peerName) - |> mapToSignal { peerId -> Signal in - if let peerId = peerId { - return account.postbox.loadedPeerWithId(peerId) - |> map(Optional.init) - } else { - return .single(nil) + let _ = strongSelf.presentVoiceMessageDiscardAlert(action: { + if strongSelf.resolvePeerByNameDisposable == nil { + strongSelf.resolvePeerByNameDisposable = MetaDisposable() + } + let account = strongSelf.context.account + var resolveSignal: Signal + if let peerName = peerName { + resolveSignal = resolvePeerByName(account: strongSelf.context.account, name: peerName) + |> mapToSignal { peerId -> Signal in + if let peerId = peerId { + return account.postbox.loadedPeerWithId(peerId) + |> map(Optional.init) + } else { + return .single(nil) + } + } + } else { + resolveSignal = context.account.postbox.loadedPeerWithId(strongSelf.chatLocation.peerId) + |> map(Optional.init) + } + var cancelImpl: (() -> Void)? + let presentationData = strongSelf.presentationData + let progressSignal = Signal { subscriber in + let controller = OverlayStatusController(theme: presentationData.theme, type: .loading(cancelled: { + cancelImpl?() + })) + self?.present(controller, in: .window(.root)) + return ActionDisposable { [weak controller] in + Queue.mainQueue().async() { + controller?.dismiss() + } } } - } else { - resolveSignal = context.account.postbox.loadedPeerWithId(strongSelf.chatLocation.peerId) - |> map(Optional.init) - } - var cancelImpl: (() -> Void)? - let presentationData = strongSelf.presentationData - let progressSignal = Signal { subscriber in - let controller = OverlayStatusController(theme: presentationData.theme, type: .loading(cancelled: { - cancelImpl?() + |> runOn(Queue.mainQueue()) + |> delay(0.15, queue: Queue.mainQueue()) + let progressDisposable = progressSignal.start() + + resolveSignal = resolveSignal + |> afterDisposed { + Queue.mainQueue().async { + progressDisposable.dispose() + } + } + cancelImpl = { + self?.resolvePeerByNameDisposable?.set(nil) + } + strongSelf.resolvePeerByNameDisposable?.set((resolveSignal + |> deliverOnMainQueue).start(next: { peer in + if let strongSelf = self, !hashtag.isEmpty { + let searchController = HashtagSearchController(context: strongSelf.context, peer: peer, query: hashtag) + strongSelf.effectiveNavigationController?.pushViewController(searchController) + } })) - self?.present(controller, in: .window(.root)) - return ActionDisposable { [weak controller] in - Queue.mainQueue().async() { - controller?.dismiss() - } - } - } - |> runOn(Queue.mainQueue()) - |> delay(0.15, queue: Queue.mainQueue()) - let progressDisposable = progressSignal.start() - - resolveSignal = resolveSignal - |> afterDisposed { - Queue.mainQueue().async { - progressDisposable.dispose() - } - } - cancelImpl = { - self?.resolvePeerByNameDisposable?.set(nil) - } - strongSelf.resolvePeerByNameDisposable?.set((resolveSignal - |> deliverOnMainQueue).start(next: { peer in - if let strongSelf = self, !hashtag.isEmpty { - let searchController = HashtagSearchController(context: strongSelf.context, peer: peer, query: hashtag) - strongSelf.effectiveNavigationController?.pushViewController(searchController) - } - })) + }) }, updateInputState: { [weak self] f in if let strongSelf = self { strongSelf.updateChatPresentationInterfaceState(animated: true, interactive: true, { @@ -1396,14 +1399,7 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G return $0.updatedInputMode(f) }) }, openMessageShareMenu: { [weak self] id in - if let strongSelf = self, let messages = strongSelf.chatDisplayNode.historyNode.messageGroupInCurrentHistoryView(id), let message = messages.first { - var shares: Int = 0 - for attribute in message.attributes { - if let forwardsAttribute = attribute as? ForwardCountMessageAttribute { - shares = forwardsAttribute.count - break - } - } + if let strongSelf = self, let messages = strongSelf.chatDisplayNode.historyNode.messageGroupInCurrentHistoryView(id), let _ = messages.first { let shareController = ShareController(context: strongSelf.context, subject: .messages(messages)) shareController.dismissed = { shared in if shared { @@ -1425,26 +1421,28 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G self?.presentInGlobalOverlay(controller, with: arguments) }, callPeer: { [weak self] peerId, isVideo in if let strongSelf = self { - strongSelf.commitPurposefulAction() - - let _ = (context.account.viewTracker.peerView(peerId) - |> take(1) - |> map { view -> Peer? in - return peerViewMainPeer(view) - } - |> deliverOnMainQueue).start(next: { peer in - guard let peer = peer else { - return - } + let _ = strongSelf.presentVoiceMessageDiscardAlert(action: { + strongSelf.commitPurposefulAction() - if let cachedUserData = strongSelf.peerView?.cachedData as? CachedUserData, cachedUserData.callsPrivate { - let presentationData = context.sharedContext.currentPresentationData.with { $0 } + let _ = (context.account.viewTracker.peerView(peerId) + |> take(1) + |> map { view -> Peer? in + return peerViewMainPeer(view) + } + |> deliverOnMainQueue).start(next: { peer in + guard let peer = peer else { + return + } - strongSelf.present(textAlertController(context: strongSelf.context, title: presentationData.strings.Call_ConnectionErrorTitle, text: presentationData.strings.Call_PrivacyErrorMessage(peer.compactDisplayTitle).0, actions: [TextAlertAction(type: .defaultAction, title: presentationData.strings.Common_OK, action: {})]), in: .window(.root)) - return - } - - context.requestCall(peerId: peer.id, isVideo: isVideo, completion: {}) + if let cachedUserData = strongSelf.peerView?.cachedData as? CachedUserData, cachedUserData.callsPrivate { + let presentationData = context.sharedContext.currentPresentationData.with { $0 } + + strongSelf.present(textAlertController(context: strongSelf.context, title: presentationData.strings.Call_ConnectionErrorTitle, text: presentationData.strings.Call_PrivacyErrorMessage(peer.compactDisplayTitle).0, actions: [TextAlertAction(type: .defaultAction, title: presentationData.strings.Common_OK, action: {})]), in: .window(.root)) + return + } + + context.requestCall(peerId: peer.id, isVideo: isVideo, completion: {}) + }) }) } }, longTap: { [weak self] action, message in @@ -1900,13 +1898,15 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G } }, addContact: { [weak self] phoneNumber in if let strongSelf = self { - strongSelf.context.sharedContext.openAddContact(context: strongSelf.context, firstName: "", lastName: "", phoneNumber: phoneNumber, label: defaultContactLabel, present: { [weak self] controller, arguments in - self?.present(controller, in: .window(.root), with: arguments) - }, pushController: { [weak self] controller in - if let strongSelf = self { - strongSelf.effectiveNavigationController?.pushViewController(controller) - } - }, completed: {}) + let _ = strongSelf.presentVoiceMessageDiscardAlert(action: { + strongSelf.context.sharedContext.openAddContact(context: strongSelf.context, firstName: "", lastName: "", phoneNumber: phoneNumber, label: defaultContactLabel, present: { [weak self] controller, arguments in + self?.present(controller, in: .window(.root), with: arguments) + }, pushController: { [weak self] controller in + if let strongSelf = self { + strongSelf.effectiveNavigationController?.pushViewController(controller) + } + }, completed: {}) + }) } }, rateCall: { [weak self] message, callId, isVideo in if let strongSelf = self { @@ -2019,20 +2019,22 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G guard let strongSelf = self, pollId.namespace == Namespaces.Media.CloudPoll else { return } - let _ = (strongSelf.context.account.postbox.transaction { transaction -> Message? in - return transaction.getMessage(messageId) - } - |> deliverOnMainQueue).start(next: { message in - guard let message = message else { - return + let _ = strongSelf.presentVoiceMessageDiscardAlert(action: { + let _ = (strongSelf.context.account.postbox.transaction { transaction -> Message? in + return transaction.getMessage(messageId) } - for media in message.media { - if let poll = media as? TelegramMediaPoll, poll.pollId == pollId { - strongSelf.push(pollResultsController(context: strongSelf.context, messageId: messageId, poll: poll)) - break + |> deliverOnMainQueue).start(next: { message in + guard let message = message else { + return } - } - }) + for media in message.media { + if let poll = media as? TelegramMediaPoll, poll.pollId == pollId { + strongSelf.push(pollResultsController(context: strongSelf.context, messageId: messageId, poll: poll)) + break + } + } + }) + }, delay: true) }, openAppStorePage: { [weak self] in if let strongSelf = self { strongSelf.context.sharedContext.applicationBindings.openAppStorePage() @@ -2083,17 +2085,21 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G if let strongSelf = self { strongSelf.presentScheduleTimePicker(completion: { [weak self] time in if let strongSelf = self { - strongSelf.chatDisplayNode.sendCurrentMessage(scheduleTime: time, completion: { [weak self] in - if let strongSelf = self { - strongSelf.updateChatPresentationInterfaceState(animated: true, interactive: false, { - $0.updatedInterfaceState { $0.withUpdatedReplyMessageId(nil).withUpdatedComposeInputState(ChatTextInputState(inputText: NSAttributedString(string: ""))) } - }) - - if strongSelf.presentationInterfaceState.subject != .scheduledMessages && time != scheduleWhenOnlineTimestamp { - strongSelf.openScheduledMessages() + if let _ = strongSelf.presentationInterfaceState.recordedMediaPreview { + strongSelf.sendMediaRecording(scheduleTime: time) + } else { + strongSelf.chatDisplayNode.sendCurrentMessage(scheduleTime: time, completion: { [weak self] in + if let strongSelf = self { + strongSelf.updateChatPresentationInterfaceState(animated: true, interactive: false, { + $0.updatedInterfaceState { $0.withUpdatedReplyMessageId(nil).withUpdatedComposeInputState(ChatTextInputState(inputText: NSAttributedString(string: ""))) } + }) + + if strongSelf.presentationInterfaceState.subject != .scheduledMessages && time != scheduleWhenOnlineTimestamp { + strongSelf.openScheduledMessages() + } } - } - }) + }) + } } }) } @@ -2110,27 +2116,29 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G } }, editScheduledMessagesTime: { [weak self] messageIds in if let strongSelf = self, let messageId = messageIds.first { - let _ = (strongSelf.context.account.postbox.transaction { transaction -> Message? in - return transaction.getMessage(messageId) - } |> deliverOnMainQueue).start(next: { [weak self] message in - guard let strongSelf = self, let message = message else { - return - } - strongSelf.presentScheduleTimePicker(selectedTime: message.timestamp, completion: { [weak self] time in - if let strongSelf = self { - var entities: TextEntitiesMessageAttribute? - for attribute in message.attributes { - if let attribute = attribute as? TextEntitiesMessageAttribute { - entities = attribute - break - } - } - strongSelf.editMessageDisposable.set((requestEditMessage(account: strongSelf.context.account, messageId: messageId, text: message.text, media: .keep, entities: entities, disableUrlPreview: false, scheduleTime: time) |> deliverOnMainQueue).start(next: { result in - }, error: { error in - })) + let _ = strongSelf.presentVoiceMessageDiscardAlert(action: { + let _ = (strongSelf.context.account.postbox.transaction { transaction -> Message? in + return transaction.getMessage(messageId) + } |> deliverOnMainQueue).start(next: { [weak self] message in + guard let strongSelf = self, let message = message else { + return } + strongSelf.presentScheduleTimePicker(selectedTime: message.timestamp, completion: { [weak self] time in + if let strongSelf = self { + var entities: TextEntitiesMessageAttribute? + for attribute in message.attributes { + if let attribute = attribute as? TextEntitiesMessageAttribute { + entities = attribute + break + } + } + strongSelf.editMessageDisposable.set((requestEditMessage(account: strongSelf.context.account, messageId: messageId, text: message.text, media: .keep, entities: entities, disableUrlPreview: false, scheduleTime: time) |> deliverOnMainQueue).start(next: { result in + }, error: { error in + })) + } + }) }) - }) + }, delay: true) } }, performTextSelectionAction: { [weak self] _, text, action in guard let strongSelf = self else { @@ -2218,25 +2226,29 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G guard let strongSelf = self else { return } - let _ = (strongSelf.context.account.postbox.transaction { transaction -> Message? in - return transaction.getMessage(messageId) - } - |> deliverOnMainQueue).start(next: { message in - guard let message = message else { - return + let _ = strongSelf.presentVoiceMessageDiscardAlert(action: { + let _ = (strongSelf.context.account.postbox.transaction { transaction -> Message? in + return transaction.getMessage(messageId) } - for media in message.media { - if let poll = media as? TelegramMediaPoll, poll.pollId.namespace == Namespaces.Media.CloudPoll { - strongSelf.push(pollResultsController(context: strongSelf.context, messageId: messageId, poll: poll, focusOnOptionWithOpaqueIdentifier: optionOpaqueIdentifier)) - break + |> deliverOnMainQueue).start(next: { message in + guard let message = message else { + return } - } + for media in message.media { + if let poll = media as? TelegramMediaPoll, poll.pollId.namespace == Namespaces.Media.CloudPoll { + strongSelf.push(pollResultsController(context: strongSelf.context, messageId: messageId, poll: poll, focusOnOptionWithOpaqueIdentifier: optionOpaqueIdentifier)) + break + } + } + }) }) }, openPollCreation: { [weak self] isQuiz in guard let strongSelf = self else { return } - strongSelf.presentPollCreation(isQuiz: isQuiz) + let _ = strongSelf.presentVoiceMessageDiscardAlert(action: { + strongSelf.presentPollCreation(isQuiz: isQuiz) + }) }, displayPollSolution: { [weak self] solution, sourceNode in self?.displayPollSolution(solution: solution, sourceNode: sourceNode, isAutomatic: false) }, displayPsa: { [weak self] type, sourceNode in @@ -2302,13 +2314,19 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G }, action: { _, f in f(.dismissWithoutContent) - self?.interfaceInteraction?.updateTextInputStateAndMode { current, inputMode in - var inputMode = inputMode - if inputMode == .none { - inputMode = .text - } - return (chatTextInputAddMentionAttribute(current, peer: peer), inputMode) + guard let strongSelf = self else { + return } + + let _ = strongSelf.presentVoiceMessageDiscardAlert(action: { + strongSelf.interfaceInteraction?.updateTextInputStateAndMode { current, inputMode in + var inputMode = inputMode + if inputMode == .none { + inputMode = .text + } + return (chatTextInputAddMentionAttribute(current, peer: peer), inputMode) + } + }, delay: true) }))) } return items @@ -2347,18 +2365,24 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G } } }, openMessageStats: { [weak self] id in - let _ = (context.account.postbox.transaction { transaction -> (MessageId, CachedPeerData?)? in - if let message = transaction.getMessage(id), let sourceMessageId = message.forwardInfo?.sourceMessageId { - return (sourceMessageId, transaction.getPeerCachedData(peerId: sourceMessageId.peerId)) - } else { - return (id, transaction.getPeerCachedData(peerId: id.peerId)) - } - } |> deliverOnMainQueue).start(next: { [weak self] messageIdAndCachedPeerData in - guard let strongSelf = self, let (id, cachedPeerDataValue) = messageIdAndCachedPeerData, let cachedPeerData = cachedPeerDataValue else { - return - } - strongSelf.push(messageStatsController(context: context, messageId: id, cachedPeerData: cachedPeerData)) - }) + guard let strongSelf = self else { + return + } + + let _ = strongSelf.presentVoiceMessageDiscardAlert(action: { + let _ = (context.account.postbox.transaction { transaction -> (MessageId, CachedPeerData?)? in + if let message = transaction.getMessage(id), let sourceMessageId = message.forwardInfo?.sourceMessageId { + return (sourceMessageId, transaction.getPeerCachedData(peerId: sourceMessageId.peerId)) + } else { + return (id, transaction.getPeerCachedData(peerId: id.peerId)) + } + } |> deliverOnMainQueue).start(next: { [weak self] messageIdAndCachedPeerData in + guard let strongSelf = self, let (id, cachedPeerDataValue) = messageIdAndCachedPeerData, let cachedPeerData = cachedPeerDataValue else { + return + } + strongSelf.push(messageStatsController(context: context, messageId: id, cachedPeerData: cachedPeerData)) + }) + }, delay: true) }, editMessageMedia: { [weak self] messageId, draw in guard let strongSelf = self else { return @@ -4351,6 +4375,8 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G switch data { case let .images(images): self?.displayPasteMenu(images) + case let .video(data): + self?.enqueueVideoData(data) case let .gif(data): self?.enqueueGifData(data) case let .sticker(image, isMemoji): @@ -4453,13 +4479,17 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G let interfaceInteraction = ChatPanelInterfaceInteraction(setupReplyMessage: { [weak self] messageId, completion in if let strongSelf = self, strongSelf.isNodeLoaded, canSendMessagesToChat(strongSelf.presentationInterfaceState) { - if let message = strongSelf.chatDisplayNode.historyNode.messageInCurrentHistoryView(messageId) { - strongSelf.updateChatPresentationInterfaceState(animated: true, interactive: true, { $0.updatedInterfaceState({ $0.withUpdatedReplyMessageId(message.id) }).updatedSearch(nil) }, completion: completion) - strongSelf.updateItemNodesSearchTextHighlightStates() - strongSelf.chatDisplayNode.ensureInputViewFocused() - } else { + let _ = strongSelf.presentVoiceMessageDiscardAlert(action: { + if let message = strongSelf.chatDisplayNode.historyNode.messageInCurrentHistoryView(messageId) { + strongSelf.updateChatPresentationInterfaceState(animated: true, interactive: true, { $0.updatedInterfaceState({ $0.withUpdatedReplyMessageId(message.id) }).updatedSearch(nil) }, completion: completion) + strongSelf.updateItemNodesSearchTextHighlightStates() + strongSelf.chatDisplayNode.ensureInputViewFocused() + } else { + completion(.immediate) + } + }, alertAction: { completion(.immediate) - } + }, delay: true) } else { completion(.immediate) } @@ -4477,44 +4507,52 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G return } - if let message = strongSelf.chatDisplayNode.historyNode.messageInCurrentHistoryView(messageId) { - strongSelf.updateChatPresentationInterfaceState(animated: true, interactive: true, { state in - var updated = state.updatedInterfaceState { - var entities: [MessageTextEntity] = [] - for attribute in message.attributes { - if let attribute = attribute as? TextEntitiesMessageAttribute { - entities = attribute.entities - break + let _ = strongSelf.presentVoiceMessageDiscardAlert(action: { + if let message = strongSelf.chatDisplayNode.historyNode.messageInCurrentHistoryView(messageId) { + strongSelf.updateChatPresentationInterfaceState(animated: true, interactive: true, { state in + var updated = state.updatedInterfaceState { + var entities: [MessageTextEntity] = [] + for attribute in message.attributes { + if let attribute = attribute as? TextEntitiesMessageAttribute { + entities = attribute.entities + break + } } - } - var inputTextMaxLength: Int32 = 4096 - for media in message.media { - if media is TelegramMediaImage || media is TelegramMediaFile { - inputTextMaxLength = strongSelf.context.currentLimitsConfiguration.with { $0 }.maxMediaCaptionLength - break + var inputTextMaxLength: Int32 = 4096 + for media in message.media { + if media is TelegramMediaImage || media is TelegramMediaFile { + inputTextMaxLength = strongSelf.context.currentLimitsConfiguration.with { $0 }.maxMediaCaptionLength + break + } } + return $0.withUpdatedEditMessage(ChatEditMessageState(messageId: messageId, inputState: ChatTextInputState(inputText: chatInputStateStringWithAppliedEntities(message.text, entities: entities)), disableUrlPreview: nil, inputTextMaxLength: inputTextMaxLength)) } - return $0.withUpdatedEditMessage(ChatEditMessageState(messageId: messageId, inputState: ChatTextInputState(inputText: chatInputStateStringWithAppliedEntities(message.text, entities: entities)), disableUrlPreview: nil, inputTextMaxLength: inputTextMaxLength)) - } - - updated = updatedChatEditInterfaceMessageState(state: updated, message: message) - updated = updated.updatedInputMode({ _ in - return .text - }) - - return updated - }, completion: completion) - } + + updated = updatedChatEditInterfaceMessageState(state: updated, message: message) + updated = updated.updatedInputMode({ _ in + return .text + }) + + return updated + }, completion: completion) + } + }, alertAction: { + completion(.immediate) + }, delay: true) } }, beginMessageSelection: { [weak self] messageIds, completion in if let strongSelf = self, strongSelf.isNodeLoaded { - strongSelf.updateChatPresentationInterfaceState(animated: true, interactive: true, { $0.updatedInterfaceState { $0.withUpdatedSelectedMessages(messageIds) } }, completion: completion) - + let _ = strongSelf.presentVoiceMessageDiscardAlert(action: { + strongSelf.updateChatPresentationInterfaceState(animated: true, interactive: true, { $0.updatedInterfaceState { $0.withUpdatedSelectedMessages(messageIds) } }, completion: completion) + if let selectionState = strongSelf.presentationInterfaceState.interfaceState.selectionState { - let count = selectionState.selectedIds.count - let text = strongSelf.presentationData.strings.VoiceOver_Chat_MessagesSelected(Int32(count)) - UIAccessibility.post(notification: UIAccessibility.Notification.announcement, argument: text) - } + let count = selectionState.selectedIds.count + let text = strongSelf.presentationData.strings.VoiceOver_Chat_MessagesSelected(Int32(count)) + UIAccessibility.post(notification: UIAccessibility.Notification.announcement, argument: text) + } + }, alertAction: { + completion(.immediate) + }, delay: true) } else { completion(.immediate) } @@ -4751,31 +4789,34 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G guard let strongSelf = self else { return } - var interactive = true - if strongSelf.chatDisplayNode.isInputViewFocused { - interactive = false - strongSelf.context.sharedContext.mainWindow?.doNotAnimateLikelyKeyboardAutocorrectionSwitch() - } - strongSelf.updateChatPresentationInterfaceState(animated: true, interactive: interactive, { current in - return current.updatedTitlePanelContext { - if let index = $0.firstIndex(where: { - switch $0 { - case .chatInfo: - return true - default: - return false + let _ = strongSelf.presentVoiceMessageDiscardAlert(action: { + var interactive = true + if strongSelf.chatDisplayNode.isInputViewFocused { + interactive = false + strongSelf.context.sharedContext.mainWindow?.doNotAnimateLikelyKeyboardAutocorrectionSwitch() + } + + strongSelf.updateChatPresentationInterfaceState(animated: true, interactive: interactive, { current in + return current.updatedTitlePanelContext { + if let index = $0.firstIndex(where: { + switch $0 { + case .chatInfo: + return true + default: + return false + } + }) { + var updatedContexts = $0 + updatedContexts.remove(at: index) + return updatedContexts + } else { + return $0 } - }) { - var updatedContexts = $0 - updatedContexts.remove(at: index) - return updatedContexts - } else { - return $0 - } - }.updatedSearch(current.search == nil ? ChatSearchData(domain: domain).withUpdatedQuery(query) : current.search?.withUpdatedDomain(domain).withUpdatedQuery(query)) + }.updatedSearch(current.search == nil ? ChatSearchData(domain: domain).withUpdatedQuery(query) : current.search?.withUpdatedDomain(domain).withUpdatedQuery(query)) + }) + strongSelf.updateItemNodesSearchTextHighlightStates() }) - strongSelf.updateItemNodesSearchTextHighlightStates() }, dismissMessageSearch: { [weak self] in if let strongSelf = self { strongSelf.updateChatPresentationInterfaceState(animated: true, interactive: true, { current in @@ -5090,8 +5131,8 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G strongSelf.lockMediaRecorder() }, deleteRecordedMedia: { [weak self] in self?.deleteMediaRecording() - }, sendRecordedMedia: { [weak self] silently in - self?.sendMediaRecording(silently: silently) + }, sendRecordedMedia: { [weak self] silentPosting in + self?.sendMediaRecording(silentPosting: silentPosting) }, displayRestrictedInfo: { [weak self] subject, displayType in guard let strongSelf = self else { return @@ -6194,6 +6235,8 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G if strongSelf.presentationInterfaceState.subject != .scheduledMessages && namespace == Namespaces.Message.ScheduledCloud { strongSelf.openScheduledMessages() } + +// strongSelf.displayChecksTooltip() } })) @@ -7379,33 +7422,35 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G self.present(actionSheet, in: .window(.root)) } case let .openChatInfo(expandAvatar): - switch self.chatLocationInfoData { - case let .peer(peerView): - self.navigationActionDisposable.set((peerView.get() - |> take(1) - |> deliverOnMainQueue).start(next: { [weak self] peerView in - if let strongSelf = self, let peer = peerView.peers[peerView.peerId], peer.restrictionText(platform: "ios", contentSettings: strongSelf.context.currentContentSettings.with { $0 }) == nil && !strongSelf.presentationInterfaceState.isNotAccessible { - if peer.id == strongSelf.context.account.peerId { - if let peer = strongSelf.presentationInterfaceState.renderedPeer?.chatMainPeer, let infoController = strongSelf.context.sharedContext.makePeerInfoController(context: strongSelf.context, peer: peer, mode: .generic, avatarInitiallyExpanded: false, fromChat: true) { - strongSelf.effectiveNavigationController?.pushViewController(infoController) - } - } else { - var expandAvatar = expandAvatar - if peer.smallProfileImage == nil { - expandAvatar = false - } - if let validLayout = strongSelf.validLayout, validLayout.deviceMetrics.type == .tablet { - expandAvatar = false - } - if let infoController = strongSelf.context.sharedContext.makePeerInfoController(context: strongSelf.context, peer: peer, mode: .generic, avatarInitiallyExpanded: expandAvatar, fromChat: true) { - strongSelf.effectiveNavigationController?.pushViewController(infoController) + let _ = self.presentVoiceMessageDiscardAlert(action: { + switch self.chatLocationInfoData { + case let .peer(peerView): + self.navigationActionDisposable.set((peerView.get() + |> take(1) + |> deliverOnMainQueue).start(next: { [weak self] peerView in + if let strongSelf = self, let peer = peerView.peers[peerView.peerId], peer.restrictionText(platform: "ios", contentSettings: strongSelf.context.currentContentSettings.with { $0 }) == nil && !strongSelf.presentationInterfaceState.isNotAccessible { + if peer.id == strongSelf.context.account.peerId { + if let peer = strongSelf.presentationInterfaceState.renderedPeer?.chatMainPeer, let infoController = strongSelf.context.sharedContext.makePeerInfoController(context: strongSelf.context, peer: peer, mode: .generic, avatarInitiallyExpanded: false, fromChat: true) { + strongSelf.effectiveNavigationController?.pushViewController(infoController) + } + } else { + var expandAvatar = expandAvatar + if peer.smallProfileImage == nil { + expandAvatar = false + } + if let validLayout = strongSelf.validLayout, validLayout.deviceMetrics.type == .tablet { + expandAvatar = false + } + if let infoController = strongSelf.context.sharedContext.makePeerInfoController(context: strongSelf.context, peer: peer, mode: .generic, avatarInitiallyExpanded: expandAvatar, fromChat: true) { + strongSelf.effectiveNavigationController?.pushViewController(infoController) + } } } - } - })) - case .replyThread: - break - } + })) + case .replyThread: + break + } + }) case .search: self.interfaceInteraction?.beginMessageSearch(.everything, "") case .dismiss: @@ -8074,6 +8119,18 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G } }) })) + controller.presentStickers = { [weak self] completion in + if let strongSelf = self { + let controller = DrawingStickersScreen(context: strongSelf.context, selectSticker: { fileReference, node, rect in + completion(fileReference.media, fileReference.media.isAnimatedSticker, node.view, rect) + return true + }) + strongSelf.present(controller, in: .window(.root)) + return controller + } else { + return nil + } + } strongSelf.effectiveNavigationController?.pushViewController(controller) } }, presentSelectionLimitExceeded: { @@ -8166,6 +8223,18 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G } }) })) + controller.presentStickers = { [weak self] completion in + if let strongSelf = self { + let controller = DrawingStickersScreen(context: strongSelf.context, selectSticker: { fileReference, node, rect in + completion(fileReference.media, fileReference.media.isAnimatedSticker, node.view, rect) + return true + }) + strongSelf.present(controller, in: .window(.root)) + return controller + } else { + return nil + } + } strongSelf.present(controller, in: .window(.root), with: ViewControllerPresentationArguments(presentationAnimation: .modalSheet)) } }) @@ -8719,7 +8788,7 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G } private func sendMessages(_ messages: [EnqueueMessage], commit: Bool = false) { - let peerId: PeerId = self.chatLocation.peerId + let peerId = self.chatLocation.peerId var isScheduledMessages = false if case .scheduledMessages = self.presentationInterfaceState.subject { @@ -8772,8 +8841,25 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G |> deliverOnMainQueue).start(next: { [weak self] settings in if let strongSelf = self, let peer = strongSelf.presentationInterfaceState.renderedPeer?.peer { strongSelf.chatDisplayNode.dismissInput() - let _ = presentLegacyPasteMenu(context: strongSelf.context, peer: peer, chatLocation: strongSelf.chatLocation, saveEditedPhotos: settings.storeEditedPhotos, allowGrouping: true, presentationData: strongSelf.presentationData, images: images, sendMessagesWithSignals: { signals in - self?.enqueueMediaMessages(signals: signals, silentPosting: false) + let _ = presentLegacyPasteMenu(context: strongSelf.context, peer: peer, chatLocation: strongSelf.chatLocation, saveEditedPhotos: settings.storeEditedPhotos, allowGrouping: true, hasSchedule: strongSelf.presentationInterfaceState.subject != .scheduledMessages && peer.id.namespace != Namespaces.Peer.SecretChat, presentationData: strongSelf.presentationData, images: images, presentSchedulePicker: { [weak self] done in + if let strongSelf = self { + strongSelf.presentScheduleTimePicker(style: .media, completion: { [weak self] time in + if let strongSelf = self { + done(time) + if strongSelf.presentationInterfaceState.subject != .scheduledMessages && time != scheduleWhenOnlineTimestamp { + strongSelf.openScheduledMessages() + } + } + }) + } + }, presentTimerPicker: { [weak self] done in + if let strongSelf = self { + strongSelf.presentTimerPicker(style: .media, completion: { time in + done(time) + }) + } + }, sendMessagesWithSignals: { signals, silentPosting, scheduleTime in + self?.enqueueMediaMessages(signals: signals, silentPosting: silentPosting, scheduleTime: scheduleTime > 0 ? scheduleTime : nil) }, presentStickers: { [weak self] completion in if let strongSelf = self { let controller = DrawingStickersScreen(context: strongSelf.context, selectSticker: { fileReference, node, rect in @@ -8810,6 +8896,22 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G })) } + private func enqueueVideoData(_ data: Data) { + self.enqueueMediaMessageDisposable.set((legacyEnqueueGifMessage(account: self.context.account, data: data) |> deliverOnMainQueue).start(next: { [weak self] message in + if let strongSelf = self { + let replyMessageId = strongSelf.presentationInterfaceState.interfaceState.replyMessageId + strongSelf.chatDisplayNode.setupSendActionOnViewUpdate({ + if let strongSelf = self { + strongSelf.updateChatPresentationInterfaceState(animated: true, interactive: false, { + $0.updatedInterfaceState { $0.withUpdatedReplyMessageId(nil) } + }) + } + }) + strongSelf.sendMessages([message].map { $0.withUpdatedReplyToMessageId(replyMessageId) }) + } + })) + } + private func enqueueStickerImage(_ image: UIImage, isMemoji: Bool) { let size = image.size.aspectFitted(CGSize(width: 512.0, height: 512.0)) self.enqueueMediaMessageDisposable.set((convertToWebP(image: image, targetSize: size, targetBoundingSize: size, quality: 0.9) |> deliverOnMainQueue).start(next: { [weak self] data in @@ -9110,7 +9212,7 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G }) } - private func sendMediaRecording(silently: Bool) { + private func sendMediaRecording(silentPosting: Bool? = nil, scheduleTime: Int32? = nil) { self.chatDisplayNode.updateRecordedMediaDeleted(false) if let recordedMediaPreview = self.presentationInterfaceState.recordedMediaPreview { var isScheduledMessages = false @@ -9135,12 +9237,26 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G } }) - var attributes: [MessageAttribute] = [] - if silently { - attributes.append(NotificationInfoMessageAttribute(flags: .muted)) + let messages: [EnqueueMessage] = [.message(text: "", attributes: [], mediaReference: .standalone(media: TelegramMediaFile(fileId: MediaId(namespace: Namespaces.Media.LocalFile, id: arc4random64()), partialReference: nil, resource: recordedMediaPreview.resource, previewRepresentations: [], videoThumbnails: [], immediateThumbnailData: nil, mimeType: "audio/ogg", size: Int(recordedMediaPreview.fileSize), attributes: [.Audio(isVoice: true, duration: Int(recordedMediaPreview.duration), title: nil, performer: nil, waveform: waveformBuffer)])), replyToMessageId: self.presentationInterfaceState.interfaceState.replyMessageId, localGroupingKey: nil)] + + let transformedMessages: [EnqueueMessage] + if let silentPosting = silentPosting { + transformedMessages = self.transformEnqueueMessages(messages, silentPosting: silentPosting) + } else if let scheduleTime = scheduleTime { + transformedMessages = self.transformEnqueueMessages(messages, silentPosting: false, scheduleTime: scheduleTime) + } else { + transformedMessages = self.transformEnqueueMessages(messages) } - self.sendMessages([.message(text: "", attributes: attributes, mediaReference: .standalone(media: TelegramMediaFile(fileId: MediaId(namespace: Namespaces.Media.LocalFile, id: arc4random64()), partialReference: nil, resource: recordedMediaPreview.resource, previewRepresentations: [], videoThumbnails: [], immediateThumbnailData: nil, mimeType: "audio/ogg", size: Int(recordedMediaPreview.fileSize), attributes: [.Audio(isVoice: true, duration: Int(recordedMediaPreview.duration), title: nil, performer: nil, waveform: waveformBuffer)])), replyToMessageId: self.presentationInterfaceState.interfaceState.replyMessageId, localGroupingKey: nil)]) + let peerId = self.chatLocation.peerId + let _ = (enqueueMessages(account: self.context.account, peerId: peerId, messages: transformedMessages) + |> deliverOnMainQueue).start(next: { [weak self] _ in + if let strongSelf = self, strongSelf.presentationInterfaceState.subject != .scheduledMessages { + strongSelf.chatDisplayNode.historyNode.scrollToEndOfHistory() + } + }) + + donateSendMessageIntent(account: self.context.account, sharedContext: self.context.sharedContext, intentContext: .chat, peerIds: [peerId]) } } @@ -9453,43 +9569,45 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G return } - let progressSignal: Signal = Signal { [weak self] _ in - guard let strongSelf = self, let controllerInteraction = strongSelf.controllerInteraction else { - return EmptyDisposable - } - - if let displayProgressInMessage = displayProgressInMessage, controllerInteraction.currentMessageWithLoadingReplyThread != displayProgressInMessage { - let previousId = controllerInteraction.currentMessageWithLoadingReplyThread - controllerInteraction.currentMessageWithLoadingReplyThread = displayProgressInMessage - strongSelf.chatDisplayNode.historyNode.requestMessageUpdate(displayProgressInMessage) - if let previousId = previousId { - strongSelf.chatDisplayNode.historyNode.requestMessageUpdate(previousId) + let _ = self.presentVoiceMessageDiscardAlert(action: { + let progressSignal: Signal = Signal { [weak self] _ in + guard let strongSelf = self, let controllerInteraction = strongSelf.controllerInteraction else { + return EmptyDisposable } - } - - return ActionDisposable { - Queue.mainQueue().async { - guard let strongSelf = self, let controllerInteraction = strongSelf.controllerInteraction else { - return + + if let displayProgressInMessage = displayProgressInMessage, controllerInteraction.currentMessageWithLoadingReplyThread != displayProgressInMessage { + let previousId = controllerInteraction.currentMessageWithLoadingReplyThread + controllerInteraction.currentMessageWithLoadingReplyThread = displayProgressInMessage + strongSelf.chatDisplayNode.historyNode.requestMessageUpdate(displayProgressInMessage) + if let previousId = previousId { + strongSelf.chatDisplayNode.historyNode.requestMessageUpdate(previousId) } - if let displayProgressInMessage = displayProgressInMessage, controllerInteraction.currentMessageWithLoadingReplyThread == displayProgressInMessage { - controllerInteraction.currentMessageWithLoadingReplyThread = nil - strongSelf.chatDisplayNode.historyNode.requestMessageUpdate(displayProgressInMessage) + } + + return ActionDisposable { + Queue.mainQueue().async { + guard let strongSelf = self, let controllerInteraction = strongSelf.controllerInteraction else { + return + } + if let displayProgressInMessage = displayProgressInMessage, controllerInteraction.currentMessageWithLoadingReplyThread == displayProgressInMessage { + controllerInteraction.currentMessageWithLoadingReplyThread = nil + strongSelf.chatDisplayNode.historyNode.requestMessageUpdate(displayProgressInMessage) + } } } } - } - |> runOn(.mainQueue()) - - let progress = (progressSignal - |> delay(0.15, queue: .mainQueue())).start() - - self.navigationActionDisposable.set((ChatControllerImpl.openMessageReplies(context: self.context, navigationController: navigationController, present: { [weak self] c, a in - self?.present(c, in: .window(.root), with: a) - }, messageId: messageId, isChannelPost: isChannelPost, atMessage: atMessageId, displayModalProgress: displayModalProgress) - |> afterDisposed { - progress.dispose() - }).start()) + |> runOn(.mainQueue()) + + let progress = (progressSignal + |> delay(0.15, queue: .mainQueue())).start() + + self.navigationActionDisposable.set((ChatControllerImpl.openMessageReplies(context: self.context, navigationController: navigationController, present: { [weak self] c, a in + self?.present(c, in: .window(.root), with: a) + }, messageId: messageId, isChannelPost: isChannelPost, atMessage: atMessageId, displayModalProgress: displayModalProgress) + |> afterDisposed { + progress.dispose() + }).start()) + }) } static func openMessageReplies(context: AccountContext, navigationController: NavigationController, present: @escaping (ViewController, Any?) -> Void, messageId: MessageId, isChannelPost: Bool, atMessage atMessageId: MessageId?, displayModalProgress: Bool) -> Signal { @@ -9775,341 +9893,349 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G } private func forwardMessages(messages: [Message], resetCurrent: Bool) { - var filter: ChatListNodePeersFilter = [.onlyWriteable, .includeSavedMessages, .excludeDisabled, .doNotSearchMessages] - var hasPublicPolls = false - var hasPublicQuiz = false - for message in messages { - for media in message.media { - if let poll = media as? TelegramMediaPoll, case .public = poll.publicity { - hasPublicPolls = true - if case .quiz = poll.kind { - hasPublicQuiz = true + let _ = self.presentVoiceMessageDiscardAlert(action: { + var filter: ChatListNodePeersFilter = [.onlyWriteable, .includeSavedMessages, .excludeDisabled, .doNotSearchMessages] + var hasPublicPolls = false + var hasPublicQuiz = false + for message in messages { + for media in message.media { + if let poll = media as? TelegramMediaPoll, case .public = poll.publicity { + hasPublicPolls = true + if case .quiz = poll.kind { + hasPublicQuiz = true + } + filter.insert(.excludeChannels) + break } - filter.insert(.excludeChannels) - break } } - } - var attemptSelectionImpl: ((Peer) -> Void)? - let controller = self.context.sharedContext.makePeerSelectionController(PeerSelectionControllerParams(context: self.context, filter: filter, attemptSelection: { peer in - attemptSelectionImpl?(peer) - })) - let context = self.context - attemptSelectionImpl = { [weak controller] peer in - guard let controller = controller else { - return - } - let presentationData = context.sharedContext.currentPresentationData.with { $0 } - if hasPublicPolls { - if let channel = peer as? TelegramChannel, case .broadcast = channel.info { - controller.present(textAlertController(context: context, title: nil, text: hasPublicQuiz ? presentationData.strings.Forward_ErrorPublicQuizDisabledInChannels : presentationData.strings.Forward_ErrorPublicPollDisabledInChannels, actions: [TextAlertAction(type: .defaultAction, title: presentationData.strings.Common_OK, action: {})]), in: .window(.root)) + var attemptSelectionImpl: ((Peer) -> Void)? + let controller = self.context.sharedContext.makePeerSelectionController(PeerSelectionControllerParams(context: self.context, filter: filter, attemptSelection: { peer in + attemptSelectionImpl?(peer) + })) + let context = self.context + attemptSelectionImpl = { [weak controller] peer in + guard let controller = controller else { return } - } - controller.present(textAlertController(context: context, title: nil, text: presentationData.strings.Forward_ErrorDisabledForChat, actions: [TextAlertAction(type: .defaultAction, title: presentationData.strings.Common_OK, action: {})]), in: .window(.root)) - } - controller.peerSelected = { [weak self, weak controller] peerId in - guard let strongSelf = self, let strongController = controller else { - return - } - - if resetCurrent { - strongSelf.updateChatPresentationInterfaceState(animated: false, interactive: true, { $0.updatedInterfaceState({ $0.withUpdatedForwardMessageIds(nil) }) }) - } - - var isPinnedMessages = false - if case .pinnedMessages = strongSelf.presentationInterfaceState.subject { - isPinnedMessages = true - } - - if case .peer(peerId) = strongSelf.chatLocation, strongSelf.parentController == nil, !isPinnedMessages { - strongSelf.updateChatPresentationInterfaceState(animated: false, interactive: true, { $0.updatedInterfaceState({ $0.withUpdatedForwardMessageIds(messages.map { $0.id }).withoutSelectionState() }) }) - strongController.dismiss() - } else if peerId == strongSelf.context.account.peerId { - let _ = (enqueueMessages(account: strongSelf.context.account, peerId: peerId, messages: messages.map { message -> EnqueueMessage in - return .forward(source: message.id, grouping: .auto, attributes: []) - }) - |> deliverOnMainQueue).start(next: { messageIds in - if let strongSelf = self { - let signals: [Signal] = messageIds.compactMap({ id -> Signal? in - guard let id = id else { - return nil - } - return strongSelf.context.account.pendingMessageManager.pendingMessageStatus(id) - |> mapToSignal { status, _ -> Signal in - if status != nil { - return .never() - } else { - return .single(true) - } - } - |> take(1) - }) - if strongSelf.shareStatusDisposable == nil { - strongSelf.shareStatusDisposable = MetaDisposable() - } - strongSelf.shareStatusDisposable?.set((combineLatest(signals) - |> deliverOnMainQueue).start(completed: { - guard let strongSelf = self else { - return - } - strongSelf.present(OverlayStatusController(theme: strongSelf.presentationData.theme, type: .success), in: .window(.root)) - })) - } - }) - strongSelf.updateChatPresentationInterfaceState(animated: false, interactive: true, { $0.updatedInterfaceState({ $0.withoutSelectionState() }) }) - strongController.dismiss() - } else { - if let navigationController = strongSelf.navigationController as? NavigationController { - for controller in navigationController.viewControllers { - if let maybeChat = controller as? ChatControllerImpl { - if case .peer(peerId) = maybeChat.chatLocation { - var isChatPinnedMessages = false - if case .pinnedMessages = maybeChat.presentationInterfaceState.subject { - isChatPinnedMessages = true - } - if !isChatPinnedMessages { - maybeChat.updateChatPresentationInterfaceState(animated: false, interactive: true, { $0.updatedInterfaceState({ $0.withUpdatedForwardMessageIds(messages.map { $0.id }).withoutSelectionState() }) }) - strongSelf.dismiss() - strongController.dismiss() - return - } - } - } + let presentationData = context.sharedContext.currentPresentationData.with { $0 } + if hasPublicPolls { + if let channel = peer as? TelegramChannel, case .broadcast = channel.info { + controller.present(textAlertController(context: context, title: nil, text: hasPublicQuiz ? presentationData.strings.Forward_ErrorPublicQuizDisabledInChannels : presentationData.strings.Forward_ErrorPublicPollDisabledInChannels, actions: [TextAlertAction(type: .defaultAction, title: presentationData.strings.Common_OK, action: {})]), in: .window(.root)) + return } } + controller.present(textAlertController(context: context, title: nil, text: presentationData.strings.Forward_ErrorDisabledForChat, actions: [TextAlertAction(type: .defaultAction, title: presentationData.strings.Common_OK, action: {})]), in: .window(.root)) + } + controller.peerSelected = { [weak self, weak controller] peerId in + guard let strongSelf = self, let strongController = controller else { + return + } - let _ = (strongSelf.context.account.postbox.transaction({ transaction -> Void in - transaction.updatePeerChatInterfaceState(peerId, update: { currentState in - if let currentState = currentState as? ChatInterfaceState { - return currentState.withUpdatedForwardMessageIds(messages.map { $0.id }) - } else { - return ChatInterfaceState().withUpdatedForwardMessageIds(messages.map { $0.id }) + if resetCurrent { + strongSelf.updateChatPresentationInterfaceState(animated: false, interactive: true, { $0.updatedInterfaceState({ $0.withUpdatedForwardMessageIds(nil) }) }) + } + + var isPinnedMessages = false + if case .pinnedMessages = strongSelf.presentationInterfaceState.subject { + isPinnedMessages = true + } + + if case .peer(peerId) = strongSelf.chatLocation, strongSelf.parentController == nil, !isPinnedMessages { + strongSelf.updateChatPresentationInterfaceState(animated: false, interactive: true, { $0.updatedInterfaceState({ $0.withUpdatedForwardMessageIds(messages.map { $0.id }).withoutSelectionState() }).updatedSearch(nil) }) + strongSelf.updateItemNodesSearchTextHighlightStates() + strongSelf.searchResultsController = nil + strongController.dismiss() + } else if peerId == strongSelf.context.account.peerId { + let _ = (enqueueMessages(account: strongSelf.context.account, peerId: peerId, messages: messages.map { message -> EnqueueMessage in + return .forward(source: message.id, grouping: .auto, attributes: []) + }) + |> deliverOnMainQueue).start(next: { messageIds in + if let strongSelf = self { + let signals: [Signal] = messageIds.compactMap({ id -> Signal? in + guard let id = id else { + return nil + } + return strongSelf.context.account.pendingMessageManager.pendingMessageStatus(id) + |> mapToSignal { status, _ -> Signal in + if status != nil { + return .never() + } else { + return .single(true) + } + } + |> take(1) + }) + if strongSelf.shareStatusDisposable == nil { + strongSelf.shareStatusDisposable = MetaDisposable() + } + strongSelf.shareStatusDisposable?.set((combineLatest(signals) + |> deliverOnMainQueue).start(completed: { + guard let strongSelf = self else { + return + } + strongSelf.present(OverlayStatusController(theme: strongSelf.presentationData.theme, type: .success), in: .window(.root)) + })) } }) - }) |> deliverOnMainQueue).start(completed: { - if let strongSelf = self { - strongSelf.updateChatPresentationInterfaceState(animated: false, interactive: true, { $0.updatedInterfaceState({ $0.withoutSelectionState() }) }) - - let ready = Promise() - - strongSelf.controllerNavigationDisposable.set((ready.get() - |> SwiftSignalKit.filter { $0 } - |> take(1) - |> deliverOnMainQueue).start(next: { _ in - if let strongController = controller { - strongController.dismiss() + strongSelf.updateChatPresentationInterfaceState(animated: false, interactive: true, { $0.updatedInterfaceState({ $0.withoutSelectionState() }) }) + strongController.dismiss() + } else { + if let navigationController = strongSelf.navigationController as? NavigationController { + for controller in navigationController.viewControllers { + if let maybeChat = controller as? ChatControllerImpl { + if case .peer(peerId) = maybeChat.chatLocation { + var isChatPinnedMessages = false + if case .pinnedMessages = maybeChat.presentationInterfaceState.subject { + isChatPinnedMessages = true + } + if !isChatPinnedMessages { + maybeChat.updateChatPresentationInterfaceState(animated: false, interactive: true, { $0.updatedInterfaceState({ $0.withUpdatedForwardMessageIds(messages.map { $0.id }).withoutSelectionState() }) }) + strongSelf.dismiss() + strongController.dismiss() + return + } + } } - })) - - if let parentController = strongSelf.parentController { - (parentController.navigationController as? NavigationController)?.replaceTopController(ChatControllerImpl(context: strongSelf.context, chatLocation: .peer(peerId)), animated: false, ready: ready) - } else { - strongSelf.effectiveNavigationController?.replaceTopController(ChatControllerImpl(context: strongSelf.context, chatLocation: .peer(peerId)), animated: false, ready: ready) } } - }) + + let _ = (strongSelf.context.account.postbox.transaction({ transaction -> Void in + transaction.updatePeerChatInterfaceState(peerId, update: { currentState in + if let currentState = currentState as? ChatInterfaceState { + return currentState.withUpdatedForwardMessageIds(messages.map { $0.id }) + } else { + return ChatInterfaceState().withUpdatedForwardMessageIds(messages.map { $0.id }) + } + }) + }) |> deliverOnMainQueue).start(completed: { + if let strongSelf = self { + strongSelf.updateChatPresentationInterfaceState(animated: false, interactive: true, { $0.updatedInterfaceState({ $0.withoutSelectionState() }) }) + + let ready = Promise() + + strongSelf.controllerNavigationDisposable.set((ready.get() + |> SwiftSignalKit.filter { $0 } + |> take(1) + |> deliverOnMainQueue).start(next: { _ in + if let strongController = controller { + strongController.dismiss() + } + })) + + if let parentController = strongSelf.parentController { + (parentController.navigationController as? NavigationController)?.replaceTopController(ChatControllerImpl(context: strongSelf.context, chatLocation: .peer(peerId)), animated: false, ready: ready) + } else { + strongSelf.effectiveNavigationController?.replaceTopController(ChatControllerImpl(context: strongSelf.context, chatLocation: .peer(peerId)), animated: false, ready: ready) + } + } + }) + } } - } - self.chatDisplayNode.dismissInput() - self.effectiveNavigationController?.pushViewController(controller) + self.chatDisplayNode.dismissInput() + self.effectiveNavigationController?.pushViewController(controller) + }) } private func openPeer(peerId: PeerId?, navigation: ChatControllerInteractionNavigateToPeer, fromMessage: Message?, expandAvatar: Bool = false) { - if case let .peer(currentPeerId) = self.chatLocation, peerId == currentPeerId { - switch navigation { - case .info: - self.navigationButtonAction(.openChatInfo(expandAvatar: expandAvatar)) - case let .chat(textInputState, _, _): - if let textInputState = textInputState { - self.updateChatPresentationInterfaceState(animated: true, interactive: true, { - return ($0.updatedInterfaceState { - return $0.withUpdatedComposeInputState(textInputState) - }).updatedInputMode({ _ in - return .text + let _ = self.presentVoiceMessageDiscardAlert(action: { + if case let .peer(currentPeerId) = self.chatLocation, peerId == currentPeerId { + switch navigation { + case .info: + self.navigationButtonAction(.openChatInfo(expandAvatar: expandAvatar)) + case let .chat(textInputState, _, _): + if let textInputState = textInputState { + self.updateChatPresentationInterfaceState(animated: true, interactive: true, { + return ($0.updatedInterfaceState { + return $0.withUpdatedComposeInputState(textInputState) + }).updatedInputMode({ _ in + return .text + }) }) + } + case let .withBotStartPayload(botStart): + self.updateChatPresentationInterfaceState(animated: true, interactive: true, { + $0.updatedBotStartPayload(botStart.payload) }) - } - case let .withBotStartPayload(botStart): - self.updateChatPresentationInterfaceState(animated: true, interactive: true, { - $0.updatedBotStartPayload(botStart.payload) - }) - default: - break - } - } else { - if let peerId = peerId { - do { - let selfPeerId = self.chatLocation.peerId - switch navigation { - case .info, .default: - let peerSignal: Signal - if let fromMessage = fromMessage { - peerSignal = loadedPeerFromMessage(account: self.context.account, peerId: peerId, messageId: fromMessage.id) - } else { - peerSignal = self.context.account.postbox.loadedPeerWithId(peerId) |> map(Optional.init) - } - self.navigationActionDisposable.set((peerSignal |> take(1) |> deliverOnMainQueue).start(next: { [weak self] peer in - if let strongSelf = self, let peer = peer { - var mode: PeerInfoControllerMode = .generic - if let _ = fromMessage { - mode = .group(selfPeerId) - } - var expandAvatar = expandAvatar - if peer.smallProfileImage == nil { - expandAvatar = false - } - if let validLayout = strongSelf.validLayout, validLayout.deviceMetrics.type == .tablet { - expandAvatar = false - } - if let infoController = strongSelf.context.sharedContext.makePeerInfoController(context: strongSelf.context, peer: peer, mode: mode, avatarInitiallyExpanded: expandAvatar, fromChat: false) { - strongSelf.effectiveNavigationController?.pushViewController(infoController) - } + default: + break + } + } else { + if let peerId = peerId { + do { + let selfPeerId = self.chatLocation.peerId + switch navigation { + case .info, .default: + let peerSignal: Signal + if let fromMessage = fromMessage { + peerSignal = loadedPeerFromMessage(account: self.context.account, peerId: peerId, messageId: fromMessage.id) + } else { + peerSignal = self.context.account.postbox.loadedPeerWithId(peerId) |> map(Optional.init) } - })) - case let .chat(textInputState, subject, peekData): - if let textInputState = textInputState { - let _ = (self.context.account.postbox.transaction({ transaction -> Void in - transaction.updatePeerChatInterfaceState(peerId, update: { currentState in - if let currentState = currentState as? ChatInterfaceState { - return currentState.withUpdatedComposeInputState(textInputState) - } else { - return ChatInterfaceState().withUpdatedComposeInputState(textInputState) + self.navigationActionDisposable.set((peerSignal |> take(1) |> deliverOnMainQueue).start(next: { [weak self] peer in + if let strongSelf = self, let peer = peer { + var mode: PeerInfoControllerMode = .generic + if let _ = fromMessage { + mode = .group(selfPeerId) + } + var expandAvatar = expandAvatar + if peer.smallProfileImage == nil { + expandAvatar = false + } + if let validLayout = strongSelf.validLayout, validLayout.deviceMetrics.type == .tablet { + expandAvatar = false + } + if let infoController = strongSelf.context.sharedContext.makePeerInfoController(context: strongSelf.context, peer: peer, mode: mode, avatarInitiallyExpanded: expandAvatar, fromChat: false) { + strongSelf.effectiveNavigationController?.pushViewController(infoController) + } + } + })) + case let .chat(textInputState, subject, peekData): + if let textInputState = textInputState { + let _ = (self.context.account.postbox.transaction({ transaction -> Void in + transaction.updatePeerChatInterfaceState(peerId, update: { currentState in + if let currentState = currentState as? ChatInterfaceState { + return currentState.withUpdatedComposeInputState(textInputState) + } else { + return ChatInterfaceState().withUpdatedComposeInputState(textInputState) + } + }) + }) + |> deliverOnMainQueue).start(completed: { [weak self] in + if let strongSelf = self, let navigationController = strongSelf.effectiveNavigationController { + strongSelf.context.sharedContext.navigateToChatController(NavigateToChatControllerParams(navigationController: navigationController, context: strongSelf.context, chatLocation: .peer(peerId), subject: subject, updateTextInputState: textInputState, peekData: peekData)) } }) - }) - |> deliverOnMainQueue).start(completed: { [weak self] in - if let strongSelf = self, let navigationController = strongSelf.effectiveNavigationController { - strongSelf.context.sharedContext.navigateToChatController(NavigateToChatControllerParams(navigationController: navigationController, context: strongSelf.context, chatLocation: .peer(peerId), subject: subject, updateTextInputState: textInputState, peekData: peekData)) + } else { + self.effectiveNavigationController?.pushViewController(ChatControllerImpl(context: self.context, chatLocation: .peer(peerId), subject: subject)) + } + case let .withBotStartPayload(botStart): + self.effectiveNavigationController?.pushViewController(ChatControllerImpl(context: self.context, chatLocation: .peer(peerId), botStart: botStart)) + default: + break + } + } + } else { + switch navigation { + case .info: + break + case let .chat(textInputState, _, _): + if let textInputState = textInputState { + let controller = self.context.sharedContext.makePeerSelectionController(PeerSelectionControllerParams(context: self.context)) + controller.peerSelected = { [weak self, weak controller] peerId in + if let strongSelf = self, let strongController = controller { + if case let .peer(currentPeerId) = strongSelf.chatLocation, peerId == currentPeerId { + strongSelf.updateChatPresentationInterfaceState(animated: true, interactive: true, { + return ($0.updatedInterfaceState { + return $0.withUpdatedComposeInputState(textInputState) + }).updatedInputMode({ _ in + return .text + }) + }) + strongController.dismiss() + } else { + let _ = (strongSelf.context.account.postbox.transaction({ transaction -> Void in + transaction.updatePeerChatInterfaceState(peerId, update: { currentState in + if let currentState = currentState as? ChatInterfaceState { + return currentState.withUpdatedComposeInputState(textInputState) + } else { + return ChatInterfaceState().withUpdatedComposeInputState(textInputState) + } + }) + }) |> deliverOnMainQueue).start(completed: { + if let strongSelf = self { + strongSelf.updateChatPresentationInterfaceState(animated: false, interactive: true, { $0.updatedInterfaceState({ $0.withoutSelectionState() }) }) + + let ready = Promise() + + strongSelf.controllerNavigationDisposable.set((ready.get() |> filter { $0 } |> take(1) |> deliverOnMainQueue).start(next: { _ in + if let strongController = controller { + strongController.dismiss() + } + })) + + strongSelf.effectiveNavigationController?.replaceTopController(ChatControllerImpl(context: strongSelf.context, chatLocation: .peer(peerId)), animated: false, ready: ready) + } + }) + } } - }) - } else { - self.effectiveNavigationController?.pushViewController(ChatControllerImpl(context: self.context, chatLocation: .peer(peerId), subject: subject)) + } + self.chatDisplayNode.dismissInput() + self.effectiveNavigationController?.pushViewController(controller) } - case let .withBotStartPayload(botStart): - self.effectiveNavigationController?.pushViewController(ChatControllerImpl(context: self.context, chatLocation: .peer(peerId), botStart: botStart)) default: break } } - } else { - switch navigation { - case .info: - break - case let .chat(textInputState, _, _): - if let textInputState = textInputState { - let controller = self.context.sharedContext.makePeerSelectionController(PeerSelectionControllerParams(context: self.context)) - controller.peerSelected = { [weak self, weak controller] peerId in - if let strongSelf = self, let strongController = controller { - if case let .peer(currentPeerId) = strongSelf.chatLocation, peerId == currentPeerId { - strongSelf.updateChatPresentationInterfaceState(animated: true, interactive: true, { - return ($0.updatedInterfaceState { - return $0.withUpdatedComposeInputState(textInputState) - }).updatedInputMode({ _ in - return .text - }) - }) - strongController.dismiss() - } else { - let _ = (strongSelf.context.account.postbox.transaction({ transaction -> Void in - transaction.updatePeerChatInterfaceState(peerId, update: { currentState in - if let currentState = currentState as? ChatInterfaceState { - return currentState.withUpdatedComposeInputState(textInputState) - } else { - return ChatInterfaceState().withUpdatedComposeInputState(textInputState) - } - }) - }) |> deliverOnMainQueue).start(completed: { - if let strongSelf = self { - strongSelf.updateChatPresentationInterfaceState(animated: false, interactive: true, { $0.updatedInterfaceState({ $0.withoutSelectionState() }) }) - - let ready = Promise() - - strongSelf.controllerNavigationDisposable.set((ready.get() |> filter { $0 } |> take(1) |> deliverOnMainQueue).start(next: { _ in - if let strongController = controller { - strongController.dismiss() - } - })) - - strongSelf.effectiveNavigationController?.replaceTopController(ChatControllerImpl(context: strongSelf.context, chatLocation: .peer(peerId)), animated: false, ready: ready) - } - }) - } - } - } - self.chatDisplayNode.dismissInput() - self.effectiveNavigationController?.pushViewController(controller) - } - default: - break - } } - } + }) } private func openPeerMention(_ name: String, navigation: ChatControllerInteractionNavigateToPeer = .default) { - let disposable: MetaDisposable - if let resolvePeerByNameDisposable = self.resolvePeerByNameDisposable { - disposable = resolvePeerByNameDisposable - } else { - disposable = MetaDisposable() - self.resolvePeerByNameDisposable = disposable - } - var resolveSignal = resolvePeerByName(account: self.context.account, name: name, ageLimit: 10) - - var cancelImpl: (() -> Void)? - let presentationData = self.presentationData - let progressSignal = Signal { [weak self] subscriber in - let controller = OverlayStatusController(theme: presentationData.theme, type: .loading(cancelled: { - cancelImpl?() - })) - self?.present(controller, in: .window(.root)) - return ActionDisposable { [weak controller] in - Queue.mainQueue().async() { - controller?.dismiss() - } + let _ = self.presentVoiceMessageDiscardAlert(action: { + let disposable: MetaDisposable + if let resolvePeerByNameDisposable = self.resolvePeerByNameDisposable { + disposable = resolvePeerByNameDisposable + } else { + disposable = MetaDisposable() + self.resolvePeerByNameDisposable = disposable } - } - |> runOn(Queue.mainQueue()) - |> delay(0.15, queue: Queue.mainQueue()) - let progressDisposable = progressSignal.start() - - resolveSignal = resolveSignal - |> afterDisposed { - Queue.mainQueue().async { - progressDisposable.dispose() - } - } - cancelImpl = { [weak self] in - self?.resolvePeerByNameDisposable?.set(nil) - } - let account = self.context.account - disposable.set((resolveSignal - |> take(1) - |> mapToSignal { peerId -> Signal in - return account.postbox.transaction { transaction -> Peer? in - if let peerId = peerId { - return transaction.getPeer(peerId) - } else { - return nil - } - } - } - |> deliverOnMainQueue).start(next: { [weak self] peer in - if let strongSelf = self { - if let peer = peer { - var navigation = navigation - if case .default = navigation { - if let peer = peer as? TelegramUser, peer.botInfo != nil { - navigation = .chat(textInputState: nil, subject: nil, peekData: nil) - } + var resolveSignal = resolvePeerByName(account: self.context.account, name: name, ageLimit: 10) + + var cancelImpl: (() -> Void)? + let presentationData = self.presentationData + let progressSignal = Signal { [weak self] subscriber in + let controller = OverlayStatusController(theme: presentationData.theme, type: .loading(cancelled: { + cancelImpl?() + })) + self?.present(controller, in: .window(.root)) + return ActionDisposable { [weak controller] in + Queue.mainQueue().async() { + controller?.dismiss() } - strongSelf.openResolved(.peer(peer.id, navigation)) - } else { - strongSelf.present(textAlertController(context: strongSelf.context, title: nil, text: strongSelf.presentationData.strings.Resolve_ErrorNotFound, actions: [TextAlertAction(type: .defaultAction, title: strongSelf.presentationData.strings.Common_OK, action: {})]), in: .window(.root)) } } - })) + |> runOn(Queue.mainQueue()) + |> delay(0.15, queue: Queue.mainQueue()) + let progressDisposable = progressSignal.start() + + resolveSignal = resolveSignal + |> afterDisposed { + Queue.mainQueue().async { + progressDisposable.dispose() + } + } + cancelImpl = { [weak self] in + self?.resolvePeerByNameDisposable?.set(nil) + } + let account = self.context.account + disposable.set((resolveSignal + |> take(1) + |> mapToSignal { peerId -> Signal in + return account.postbox.transaction { transaction -> Peer? in + if let peerId = peerId { + return transaction.getPeer(peerId) + } else { + return nil + } + } + } + |> deliverOnMainQueue).start(next: { [weak self] peer in + if let strongSelf = self { + if let peer = peer { + var navigation = navigation + if case .default = navigation { + if let peer = peer as? TelegramUser, peer.botInfo != nil { + navigation = .chat(textInputState: nil, subject: nil, peekData: nil) + } + } + strongSelf.openResolved(.peer(peer.id, navigation)) + } else { + strongSelf.present(textAlertController(context: strongSelf.context, title: nil, text: strongSelf.presentationData.strings.Resolve_ErrorNotFound, actions: [TextAlertAction(type: .defaultAction, title: strongSelf.presentationData.strings.Common_OK, action: {})]), in: .window(.root)) + } + } + })) + }) } private func unblockPeer() { @@ -10406,63 +10532,65 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G private func openUrl(_ url: String, concealed: Bool, message: Message? = nil) { self.commitPurposefulAction() - if self.context.sharedContext.immediateExperimentalUISettings.playlistPlayback { - if url.hasSuffix(".m3u8") { - let navigationController = self.navigationController as? NavigationController - - let webPage = TelegramMediaWebpage( - webpageId: MediaId(namespace: 0, id: 0), - content: .Loaded(TelegramMediaWebpageLoadedContent( - url: url, - displayUrl: url, - hash: 0, - type: "video", - websiteName: nil, - title: nil, - text: nil, - embedUrl: url, - embedType: "video", - embedSize: nil, - duration: nil, - author: nil, - image: nil, - file: nil, - attributes: [], - instantPage: nil - )) - ) - let entry = InstantPageGalleryEntry( - index: 0, - pageId: webPage.webpageId, - media: InstantPageMedia( + self.presentVoiceMessageDiscardAlert(action: { + if self.context.sharedContext.immediateExperimentalUISettings.playlistPlayback { + if url.hasSuffix(".m3u8") { + let navigationController = self.navigationController as? NavigationController + + let webPage = TelegramMediaWebpage( + webpageId: MediaId(namespace: 0, id: 0), + content: .Loaded(TelegramMediaWebpageLoadedContent( + url: url, + displayUrl: url, + hash: 0, + type: "video", + websiteName: nil, + title: nil, + text: nil, + embedUrl: url, + embedType: "video", + embedSize: nil, + duration: nil, + author: nil, + image: nil, + file: nil, + attributes: [], + instantPage: nil + )) + ) + let entry = InstantPageGalleryEntry( index: 0, - media: webPage, - url: nil, + pageId: webPage.webpageId, + media: InstantPageMedia( + index: 0, + media: webPage, + url: nil, + caption: nil, + credit: nil + ), caption: nil, - credit: nil - ), - caption: nil, - credit: nil, - location: nil - ) - - let gallery = InstantPageGalleryController(context: context, webPage: webPage, entries: [entry], centralIndex: 0, replaceRootController: { [weak navigationController] controller, ready in - if let navigationController = navigationController { - navigationController.replaceTopController(controller, animated: false, ready: ready) - } - }, baseNavigationController: navigationController) - self.present(gallery, in: .window(.root), with: InstantPageGalleryControllerPresentationArguments(transitionArguments: { entry -> GalleryTransitionArguments? in - return nil - })) - return; + credit: nil, + location: nil + ) + + let gallery = InstantPageGalleryController(context: self.context, webPage: webPage, entries: [entry], centralIndex: 0, replaceRootController: { [weak navigationController] controller, ready in + if let navigationController = navigationController { + navigationController.replaceTopController(controller, animated: false, ready: ready) + } + }, baseNavigationController: navigationController) + self.present(gallery, in: .window(.root), with: InstantPageGalleryControllerPresentationArguments(transitionArguments: { entry -> GalleryTransitionArguments? in + return nil + })) + return; + } } - } - - openUserGeneratedUrl(context: self.context, url: url, concealed: concealed, present: { [weak self] c in - self?.present(c, in: .window(.root)) - }, openResolved: { [weak self] resolved in - self?.openResolved(resolved) - }) + + openUserGeneratedUrl(context: self.context, url: url, concealed: concealed, present: { [weak self] c in + self?.present(c, in: .window(.root)) + }, openResolved: { [weak self] resolved in + self?.openResolved(resolved) + }) + }, performAction: true) } private func openUrlIn(_ url: String) { @@ -10924,6 +11052,44 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G })) } + private func displayChecksTooltip() { + self.checksTooltipController?.dismiss() + + var latestNode: (Int32, ASDisplayNode)? + self.chatDisplayNode.historyNode.forEachVisibleItemNode { itemNode in + if let itemNode = itemNode as? ChatMessageItemView, let item = itemNode.item, let statusNode = itemNode.getStatusNode() { + if let (latestTimestamp, _) = latestNode { + if item.message.timestamp > latestTimestamp { + latestNode = (item.message.timestamp, statusNode) + } + } else { + latestNode = (item.message.timestamp, statusNode) + } + } + } + + if let (_, latestStatusNode) = latestNode { + let bounds = latestStatusNode.view.convert(latestStatusNode.view.bounds, to: self.chatDisplayNode.view) + let location = CGPoint(x: bounds.maxX - 7.0, y: bounds.minY + 2.0) + + let contentNode = ChatStatusChecksTooltipContentNode(presentationData: self.presentationData) + let tooltipController = TooltipController(content: .custom(contentNode), baseFontSize: self.presentationData.listsFontSize.baseDisplaySize, timeout: 3.5, dismissByTapOutside: true, dismissImmediatelyOnLayoutUpdate: true) + self.checksTooltipController = tooltipController + tooltipController.dismissed = { [weak self, weak tooltipController] _ in + if let strongSelf = self, let tooltipController = tooltipController, strongSelf.checksTooltipController === tooltipController { + strongSelf.checksTooltipController = nil + // ApplicationSpecificNotice.setVolumeButtonToUnmute(accountManager: strongSelf.context.sharedContext.accountManager) + } + } + self.present(tooltipController, in: .window(.root), with: TooltipControllerPresentationArguments(sourceNodeAndRect: { [weak self] in + if let strongSelf = self { + return (strongSelf.chatDisplayNode, CGRect(origin: location, size: CGSize())) + } + return nil + })) + } + } + private func dismissAllTooltips() { self.sendingOptionsTooltipController?.dismiss() self.searchResultsTooltipController?.dismiss() @@ -10932,6 +11098,7 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G self.silentPostTooltipController?.dismiss() self.mediaRecordingModeTooltipController?.dismiss() self.mediaRestrictedTooltipController?.dismiss() + self.checksTooltipController?.dismiss() self.window?.forEachController({ controller in if let controller = controller as? UndoOverlayController { @@ -11071,16 +11238,6 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G } let otherShortcuts: [KeyShortcut] = [ -// KeyShortcut(title: strings.KeyCommand_ScrollUp, input: UIKeyCommand.inputUpArrow, modifiers: [.shift], action: { [weak self] in -// if let strongSelf = self { -// _ = strongSelf.chatDisplayNode.historyNode.scrollWithDirection(.down, distance: 75.0) -// } -// }), -// KeyShortcut(title: strings.KeyCommand_ScrollDown, input: UIKeyCommand.inputDownArrow, modifiers: [.shift], action: { [weak self] in -// if let strongSelf = self { -// _ = strongSelf.chatDisplayNode.historyNode.scrollWithDirection(.up, distance: 75.0) -// } -// }), KeyShortcut(title: strings.KeyCommand_ChatInfo, input: "I", modifiers: [.command, .control], action: { [weak self] in if let strongSelf = self { strongSelf.interfaceInteraction?.openPeerInfo() @@ -11105,6 +11262,12 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G return inputShortcuts + otherShortcuts } + public override func joinGroupCall(peerId: PeerId, info: GroupCallInfo) { + let _ = self.presentVoiceMessageDiscardAlert(action: { + super.joinGroupCall(peerId: peerId, info: info) + }) + } + public func getTransitionInfo(messageId: MessageId, media: Media) -> ((UIView) -> Void, ASDisplayNode, () -> (UIView?, UIView?))? { var selectedNode: (ASDisplayNode, CGRect, () -> (UIView?, UIView?))? self.chatDisplayNode.historyNode.forEachItemNode { itemNode in @@ -11374,14 +11537,19 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G self.present(controller, in: .window(.root)) } - private func presentVoiceMessageDiscardAlert(action: @escaping () -> Void = {}) -> Bool { + private func presentVoiceMessageDiscardAlert(action: @escaping () -> Void = {}, alertAction: (() -> Void)? = nil, delay: Bool = false, performAction: Bool = true) -> Bool { if let _ = self.presentationInterfaceState.inputTextPanelState.mediaRecordingState { - self.present(textAlertController(context: self.context, title: nil, text: self.presentationData.strings.Conversation_DiscardVoiceMessageDescription, actions: [TextAlertAction(type: .genericAction, title: self.presentationData.strings.Common_Cancel, action: {}), TextAlertAction(type: .defaultAction, title: self.presentationData.strings.Conversation_DiscardVoiceMessageAction, action: { [weak self] in - self?.stopMediaRecorder() - action() - })]), in: .window(.root)) + alertAction?() + Queue.mainQueue().after(delay ? 0.2 : 0.0) { + self.present(textAlertController(context: self.context, title: nil, text: self.presentationData.strings.Conversation_DiscardVoiceMessageDescription, actions: [TextAlertAction(type: .genericAction, title: self.presentationData.strings.Common_Cancel, action: {}), TextAlertAction(type: .defaultAction, title: self.presentationData.strings.Conversation_DiscardVoiceMessageAction, action: { [weak self] in + self?.stopMediaRecorder() + action() + })]), in: .window(.root)) + } return true + } else if performAction { + action() } return false } diff --git a/submodules/TelegramUI/Sources/ChatInterfaceStateContextMenus.swift b/submodules/TelegramUI/Sources/ChatInterfaceStateContextMenus.swift index 960b5ba8a3..5d795a1980 100644 --- a/submodules/TelegramUI/Sources/ChatInterfaceStateContextMenus.swift +++ b/submodules/TelegramUI/Sources/ChatInterfaceStateContextMenus.swift @@ -579,7 +579,7 @@ func contextMenuForChatPresentationInterfaceState(chatPresentationInterfaceState if resourceAvailable, !message.containsSecretMedia { var mediaReference: AnyMediaReference? for media in message.media { - if let image = media as? TelegramMediaImage, let largest = largestImageRepresentation(image.representations) { + if let image = media as? TelegramMediaImage, let _ = largestImageRepresentation(image.representations) { mediaReference = ImageMediaReference.standalone(media: image).abstract break } else if let file = media as? TelegramMediaFile, file.isVideo { @@ -641,8 +641,15 @@ func contextMenuForChatPresentationInterfaceState(chatPresentationInterfaceState }) }))) } + + let isMigrated: Bool + if chatPresentationInterfaceState.renderedPeer?.peer is TelegramChannel && message.id.peerId.namespace == Namespaces.Peer.CloudGroup { + isMigrated = true + } else { + isMigrated = false + } - if data.canEdit && !isPinnedMessages { + if data.canEdit && !isPinnedMessages && !isMigrated { actions.append(.action(ContextMenuActionItem(text: chatPresentationInterfaceState.strings.Conversation_MessageDialogEdit, icon: { theme in return generateTintedImage(image: UIImage(bundleImageName: "Chat/Context Menu/Edit"), color: theme.actionSheet.primaryTextColor) }, action: { c, f in @@ -678,7 +685,7 @@ func contextMenuForChatPresentationInterfaceState(chatPresentationInterfaceState } } - if data.canPin, case .peer = chatPresentationInterfaceState.chatLocation { + if data.canPin && !isMigrated, case .peer = chatPresentationInterfaceState.chatLocation { var pinnedSelectedMessageId: MessageId? for message in messages { if message.tags.contains(.pinned) { @@ -744,7 +751,7 @@ func contextMenuForChatPresentationInterfaceState(chatPresentationInterfaceState } } - if let message = messages.first, message.id.namespace == Namespaces.Message.Cloud, let channel = message.peers[message.id.peerId] as? TelegramChannel, !(message.media.first is TelegramMediaAction), !isReplyThreadHead { + if let message = messages.first, message.id.namespace == Namespaces.Message.Cloud, let channel = message.peers[message.id.peerId] as? TelegramChannel, !(message.media.first is TelegramMediaAction), !isReplyThreadHead, !isMigrated { actions.append(.action(ContextMenuActionItem(text: chatPresentationInterfaceState.strings.Conversation_ContextMenuCopyLink, icon: { theme in return generateTintedImage(image: UIImage(bundleImageName: "Chat/Context Menu/Link"), color: theme.actionSheet.primaryTextColor) }, action: { _, f in @@ -850,7 +857,7 @@ func contextMenuForChatPresentationInterfaceState(chatPresentationInterfaceState } var clearCacheAsDelete = false - if message.id.peerId.namespace == Namespaces.Peer.CloudChannel { + if message.id.peerId.namespace == Namespaces.Peer.CloudChannel && !isMigrated { var views: Int = 0 for attribute in message.attributes { if let attribute = attribute as? ViewCountMessageAttribute { diff --git a/submodules/TelegramUI/Sources/ChatMessageBubbleContentNode.swift b/submodules/TelegramUI/Sources/ChatMessageBubbleContentNode.swift index 323e2a7b45..8f716ff1eb 100644 --- a/submodules/TelegramUI/Sources/ChatMessageBubbleContentNode.swift +++ b/submodules/TelegramUI/Sources/ChatMessageBubbleContentNode.swift @@ -205,4 +205,8 @@ class ChatMessageBubbleContentNode: ASDisplayNode { func reactionTargetNode(value: String) -> (ASDisplayNode, ASDisplayNode)? { return nil } + + func getStatusNode() -> ASDisplayNode? { + return nil + } } diff --git a/submodules/TelegramUI/Sources/ChatMessageBubbleItemNode.swift b/submodules/TelegramUI/Sources/ChatMessageBubbleItemNode.swift index 9675518bbe..f343c2cddb 100644 --- a/submodules/TelegramUI/Sources/ChatMessageBubbleItemNode.swift +++ b/submodules/TelegramUI/Sources/ChatMessageBubbleItemNode.swift @@ -3764,4 +3764,16 @@ class ChatMessageBubbleItemNode: ChatMessageItemView, ChatMessagePreviewItemNode forwardInfoNode.updatePsaButtonDisplay(isVisible: item.controllerInteraction.currentPsaMessageWithTooltip != item.message.id, animated: animated) } } + + override func getStatusNode() -> ASDisplayNode? { + for contentNode in self.contentNodes { + if let statusNode = contentNode.getStatusNode() { + return statusNode + } + } + if let statusNode = self.mosaicStatusNode { + return statusNode + } + return nil + } } diff --git a/submodules/TelegramUI/Sources/ChatMessageItemView.swift b/submodules/TelegramUI/Sources/ChatMessageItemView.swift index 2baabb1006..d56b6d58ee 100644 --- a/submodules/TelegramUI/Sources/ChatMessageItemView.swift +++ b/submodules/TelegramUI/Sources/ChatMessageItemView.swift @@ -835,4 +835,8 @@ public class ChatMessageItemView: ListViewItemNode { func targetReactionNode(value: String) -> (ASDisplayNode, ASDisplayNode)? { return nil } + + func getStatusNode() -> ASDisplayNode? { + return nil + } } diff --git a/submodules/TelegramUI/Sources/ChatMessageTextBubbleContentNode.swift b/submodules/TelegramUI/Sources/ChatMessageTextBubbleContentNode.swift index f54ab56e51..d693d31514 100644 --- a/submodules/TelegramUI/Sources/ChatMessageTextBubbleContentNode.swift +++ b/submodules/TelegramUI/Sources/ChatMessageTextBubbleContentNode.swift @@ -617,4 +617,8 @@ class ChatMessageTextBubbleContentNode: ChatMessageBubbleContentNode { } return nil } + + override func getStatusNode() -> ASDisplayNode? { + return self.statusNode + } } diff --git a/submodules/TelegramUI/Sources/ChatRecentActionsControllerNode.swift b/submodules/TelegramUI/Sources/ChatRecentActionsControllerNode.swift index 8b910bda2a..86b987ae05 100644 --- a/submodules/TelegramUI/Sources/ChatRecentActionsControllerNode.swift +++ b/submodules/TelegramUI/Sources/ChatRecentActionsControllerNode.swift @@ -157,6 +157,7 @@ final class ChatRecentActionsControllerNode: ViewControllerTracingNode { break } } + let gallerySource = GalleryControllerItemSource.standaloneMessage(message) return context.sharedContext.openChatMessage(OpenChatMessageParams(context: context, chatLocation: nil, chatLocationContextHolder: nil, message: message, standalone: true, reverseMessageGalleryOrder: false, navigationController: navigationController, dismissInput: { //self?.chatDisplayNode.dismissInput() }, present: { c, a in @@ -204,7 +205,7 @@ final class ChatRecentActionsControllerNode: ViewControllerTracingNode { } })) } - })) + }, gallerySource: gallerySource)) } return false }, openPeer: { [weak self] peerId, _, message in diff --git a/submodules/TelegramUI/Sources/ChatRecentActionsFilterController.swift b/submodules/TelegramUI/Sources/ChatRecentActionsFilterController.swift index caa24c5850..ae772865e1 100644 --- a/submodules/TelegramUI/Sources/ChatRecentActionsFilterController.swift +++ b/submodules/TelegramUI/Sources/ChatRecentActionsFilterController.swift @@ -285,10 +285,10 @@ private func channelRecentActionsFilterControllerEntries(presentationData: Prese let order: [([AdminLogEventsFlags], String)] if isGroup { order = [ - ([.ban, .unban], presentationData.strings.Channel_AdminLogFilter_EventsRestrictions), + ([.ban, .unban, .kick, .unkick], presentationData.strings.Channel_AdminLogFilter_EventsRestrictions), ([.promote, .demote], presentationData.strings.Channel_AdminLogFilter_EventsAdmins), ([.invite, .join], presentationData.strings.Channel_AdminLogFilter_EventsNewMembers), - ([.info], isGroup ? presentationData.strings.Channel_AdminLogFilter_EventsInfo : presentationData.strings.Channel_AdminLogFilter_ChannelEventsInfo), + ([.info, .settings], isGroup ? presentationData.strings.Channel_AdminLogFilter_EventsInfo : presentationData.strings.Channel_AdminLogFilter_ChannelEventsInfo), ([.deleteMessages], presentationData.strings.Channel_AdminLogFilter_EventsDeletedMessages), ([.editMessages], presentationData.strings.Channel_AdminLogFilter_EventsEditedMessages), ([.pinnedMessages], presentationData.strings.Channel_AdminLogFilter_EventsPinned), @@ -299,9 +299,10 @@ private func channelRecentActionsFilterControllerEntries(presentationData: Prese order = [ ([.promote, .demote], presentationData.strings.Channel_AdminLogFilter_EventsAdmins), ([.invite, .join], presentationData.strings.Channel_AdminLogFilter_EventsNewMembers), - ([.info], isGroup ? presentationData.strings.Channel_AdminLogFilter_EventsInfo : presentationData.strings.Channel_AdminLogFilter_ChannelEventsInfo), + ([.info, .settings], isGroup ? presentationData.strings.Channel_AdminLogFilter_EventsInfo : presentationData.strings.Channel_AdminLogFilter_ChannelEventsInfo), ([.deleteMessages], presentationData.strings.Channel_AdminLogFilter_EventsDeletedMessages), ([.editMessages], presentationData.strings.Channel_AdminLogFilter_EventsEditedMessages), + ([.pinnedMessages], presentationData.strings.Channel_AdminLogFilter_EventsPinned), ([.leave], presentationData.strings.Channel_AdminLogFilter_EventsLeaving), ] } diff --git a/submodules/TelegramUI/Sources/ChatSearchResultsContollerNode.swift b/submodules/TelegramUI/Sources/ChatSearchResultsContollerNode.swift index d0cae8cb5c..22efb1cc2a 100644 --- a/submodules/TelegramUI/Sources/ChatSearchResultsContollerNode.swift +++ b/submodules/TelegramUI/Sources/ChatSearchResultsContollerNode.swift @@ -174,7 +174,11 @@ class ChatSearchResultsControllerNode: ViewControllerTracingNode, UIScrollViewDe }, messageSelected: { [weak self] peer, message, _ in if let strongSelf = self { if let index = strongSelf.searchResult.messages.firstIndex(where: { $0.index == message.index }) { - strongSelf.resultSelected?(strongSelf.searchResult.messages.count - index - 1) + if message.id.peerId.namespace == Namespaces.Peer.SecretChat { + strongSelf.resultSelected?(index) + } else { + strongSelf.resultSelected?(strongSelf.searchResult.messages.count - index - 1) + } } strongSelf.listNode.clearHighlightAnimated(true) } diff --git a/submodules/TelegramUI/Sources/ChatStatusChecksTooltipContentNode.swift b/submodules/TelegramUI/Sources/ChatStatusChecksTooltipContentNode.swift new file mode 100644 index 0000000000..21d17d7d6d --- /dev/null +++ b/submodules/TelegramUI/Sources/ChatStatusChecksTooltipContentNode.swift @@ -0,0 +1,245 @@ +import Foundation +import UIKit +import AsyncDisplayKit +import Display +import SwiftSignalKit +import LegacyComponents +import TelegramPresentationData +import TooltipUI + +private final class ChecksNodeParameters: NSObject { + let color: UIColor + let progress: CGFloat + + init(color: UIColor, progress: CGFloat) { + self.color = color + self.progress = progress + + super.init() + } +} + +private class ChecksNode: ASDisplayNode { + var state: Bool? = nil + + var color: UIColor { + didSet { + self.setNeedsDisplay() + } + } + + private var effectiveProgress: CGFloat = 1.0 { + didSet { + self.setNeedsDisplay() + } + } + + init(color: UIColor) { + self.color = color + + super.init() + + self.backgroundColor = .clear + self.isOpaque = false + } + + func animateProgress(from: CGFloat, to: CGFloat) { + self.pop_removeAllAnimations() + + let animation = POPBasicAnimation() + animation.property = (POPAnimatableProperty.property(withName: "progress", initializer: { property in + property?.readBlock = { node, values in + values?.pointee = (node as! ChecksNode).effectiveProgress + } + property?.writeBlock = { node, values in + (node as! ChecksNode).effectiveProgress = values!.pointee + } + property?.threshold = 0.01 + }) as! POPAnimatableProperty) + animation.fromValue = from as NSNumber + animation.toValue = to as NSNumber + animation.timingFunction = CAMediaTimingFunction(name: CAMediaTimingFunctionName.linear) + animation.duration = 0.2 + self.pop_add(animation, forKey: "progress") + } + + override func drawParameters(forAsyncLayer layer: _ASDisplayLayer) -> NSObjectProtocol? { + return ChecksNodeParameters(color: self.color, progress: self.effectiveProgress) + } + + override func didEnterHierarchy() { + super.didEnterHierarchy() + } + + @objc override class func draw(_ bounds: CGRect, withParameters parameters: Any?, isCancelled: () -> Bool, isRasterizing: Bool) { + let context = UIGraphicsGetCurrentContext()! + + if !isRasterizing { + context.setBlendMode(.copy) + context.setFillColor(UIColor.clear.cgColor) + context.fill(bounds) + } + + guard let parameters = parameters as? ChecksNodeParameters else { + return + } + + let scaleFactor: CGFloat = 1.0 + context.translateBy(x: bounds.width / 2.0, y: bounds.height / 2.0) + context.scaleBy(x: scaleFactor, y: scaleFactor) + context.translateBy(x: -bounds.width / 2.0, y: -bounds.height / 2.0) + + let progress = parameters.progress + + context.setStrokeColor(parameters.color.cgColor) + context.setLineWidth(1.0 + UIScreenPixel) + context.setLineCap(.round) + context.setLineJoin(.round) + context.setMiterLimit(10.0) + + context.saveGState() + var s1 = CGPoint(x: 9.0, y: 13.0) + var s2 = CGPoint(x: 5.0, y: 13.0) + let p1 = CGPoint(x: 3.5, y: 3.5) + let p2 = CGPoint(x: 7.5 - UIScreenPixel, y: -8.0) + + let check1FirstSegment: CGFloat = max(0.0, min(1.0, progress * 3.0)) + let check2FirstSegment: CGFloat = max(0.0, min(1.0, (progress - 1.0) * 3.0)) + + let firstProgress = max(0.0, min(1.0, progress)) + let secondProgress = max(0.0, min(1.0, progress - 1.0)) + + let scale: CGFloat = 1.2 + context.translateBy(x: 16.0, y: 13.0) + context.scaleBy(x: scale - abs((scale - 1.0) * (firstProgress - 0.5) / 0.5), y: scale - abs((scale - 1.0) * (firstProgress - 0.5) / 0.5)) + s1 = s1.offsetBy(dx: -16.0, dy: -13.0) + + if !check1FirstSegment.isZero { + if check1FirstSegment < 1.0 { + context.move(to: CGPoint(x: s1.x + p1.x * check1FirstSegment, y: s1.y + p1.y * check1FirstSegment)) + context.addLine(to: s1) + } else { + let secondSegment = (min(1.0, progress) - 0.33) * 1.5 + context.move(to: CGPoint(x: s1.x + p1.x + p2.x * secondSegment, y: s1.y + p1.y + p2.y * secondSegment)) + context.addLine(to: CGPoint(x: s1.x + p1.x, y: s1.y + p1.y)) + context.addLine(to: CGPoint(x: s1.x + p1.x * min(1.0, check2FirstSegment), y: s1.y + p1.y * min(1.0, check2FirstSegment))) + } + } + context.strokePath() + + context.restoreGState() + + context.translateBy(x: 12.0, y: 13.0) + context.scaleBy(x: scale - abs((scale - 1.0) * (secondProgress - 0.5) / 0.5), y: scale - abs((scale - 1.0) * (secondProgress - 0.5) / 0.5)) + s2 = s2.offsetBy(dx: -12.0, dy: -13.0) + + if !check2FirstSegment.isZero { + if check2FirstSegment < 1.0 { + context.move(to: CGPoint(x: s2.x + p1.x * check2FirstSegment, y: s2.y + p1.y * check2FirstSegment)) + context.addLine(to: s2) + } else { + let secondSegment = (max(0.0, (progress - 1.0)) - 0.33) * 1.5 + context.move(to: CGPoint(x: s2.x + p1.x + p2.x * secondSegment, y: s2.y + p1.y + p2.y * secondSegment)) + context.addLine(to: CGPoint(x: s2.x + p1.x, y: s2.y + p1.y)) + context.addLine(to: s2) + } + } + context.strokePath() + } + + func updateState(_ state: Bool, animated: Bool) { + guard state != self.state else { + return + } + let previousState = self.state + self.state = state + if animated { + if previousState == nil && self.state == false { + self.animateProgress(from: 0.0, to: 1.0) + } else if previousState == false && self.state == true { + self.animateProgress(from: 1.0, to: 2.0) + } + } else { + if let state = self.state { + self.effectiveProgress = state ? 2.0 : 1.0 + } else { + self.effectiveProgress = 0.0 + } + } + } +} + +class ChatStatusChecksTooltipContentNode: ASDisplayNode, TooltipControllerCustomContentNode { + private let deliveredChecksNode: ChecksNode + private let deliveredTextNode: ImmediateTextNode + private let readChecksNode: ChecksNode + private let readTextNode: ImmediateTextNode + + init(presentationData: PresentationData) { + self.deliveredChecksNode = ChecksNode(color: .white) + self.deliveredTextNode = ImmediateTextNode() + self.readChecksNode = ChecksNode(color: .white) + self.readTextNode = ImmediateTextNode() + + self.deliveredTextNode.attributedText = NSAttributedString(string: presentationData.strings.Conversation_ChecksTooltip_Delivered, font: Font.regular(14.0), textColor: UIColor.white) + self.readTextNode.attributedText = NSAttributedString(string: presentationData.strings.Conversation_ChecksTooltip_Read, font: Font.regular(14.0), textColor: UIColor.white) + + super.init() + + self.addSubnode(self.deliveredChecksNode) + self.addSubnode(self.deliveredTextNode) + self.addSubnode(self.readChecksNode) + self.addSubnode(self.readTextNode) + } + + func animateIn() { + self.deliveredChecksNode.updateState(false, animated: true) + self.readChecksNode.updateState(false, animated: true) + + self.deliveredChecksNode.layer.animateScale(from: 1.0, to: 1.12, duration: 0.25, delay: 0.1, removeOnCompletion: false, completion: { [weak self] _ in + if let strongSelf = self { + strongSelf.deliveredChecksNode.layer.animateScale(from: 1.12, to: 1.0, duration: 0.25) + } + }) + + self.deliveredTextNode.layer.animateScale(from: 1.0, to: 1.12, duration: 0.25, delay: 0.1, removeOnCompletion: false, completion: { [weak self] _ in + if let strongSelf = self { + strongSelf.deliveredTextNode.layer.animateScale(from: 1.12, to: 1.0, duration: 0.25) + } + }) + + Queue.mainQueue().after(0.5) { + self.readChecksNode.updateState(true, animated: true) + + self.readChecksNode.layer.animateScale(from: 1.0, to: 1.12, duration: 0.25, removeOnCompletion: false, completion: { [weak self] _ in + if let strongSelf = self { + strongSelf.readChecksNode.layer.animateScale(from: 1.12, to: 1.0, duration: 0.25) + } + }) + + self.readTextNode.layer.animateScale(from: 1.0, to: 1.12, duration: 0.25, removeOnCompletion: false, completion: { [weak self] _ in + if let strongSelf = self { + strongSelf.readTextNode.layer.animateScale(from: 1.12, to: 1.0, duration: 0.25) + } + }) + } + } + + func updateLayout(size: CGSize) -> CGSize { + let deliveredSize = self.deliveredTextNode.updateLayout(size) + let readSize = self.readTextNode.updateLayout(size) + + let checksInset: CGFloat = 8.0 + let checksSize = CGSize(width: 24.0, height: 24.0) + + self.deliveredChecksNode.frame = CGRect(origin: CGPoint(x: checksInset, y: 15.0), size: checksSize) + self.deliveredTextNode.frame = CGRect(origin: CGPoint(x: checksInset + checksSize.width + 5.0, y: 19.0), size: deliveredSize) + self.readChecksNode.frame = CGRect(origin: CGPoint(x: checksInset, y: 38.0), size: checksSize) + self.readTextNode.frame = CGRect(origin: CGPoint(x: checksInset + checksSize.width + 5.0, y: 43.0), size: readSize) + + let contentWidth = max(deliveredSize.width, readSize.width) + checksInset + checksSize.width + 18.0 + let contentHeight: CGFloat = 77.0 + + return CGSize(width: contentWidth, height: contentHeight) + } +} diff --git a/submodules/TelegramUI/Sources/ChatTextInputPanelNode.swift b/submodules/TelegramUI/Sources/ChatTextInputPanelNode.swift index 7151db41a5..f99cde87bc 100644 --- a/submodules/TelegramUI/Sources/ChatTextInputPanelNode.swift +++ b/submodules/TelegramUI/Sources/ChatTextInputPanelNode.swift @@ -191,6 +191,7 @@ private func textInputBackgroundImage(backgroundColor: UIColor, strokeColor: UIC enum ChatTextInputPanelPasteData { case images([UIImage]) + case video(Data) case gif(Data) case sticker(UIImage, Bool) } @@ -1911,6 +1912,9 @@ class ChatTextInputPanelNode: ChatInputPanelNode, ASEditableTextNodeDelegate { if let data = pasteboard.data(forPasteboardType: "com.compuserve.gif") { self.paste(.gif(data)) return false + } else if let data = pasteboard.data(forPasteboardType: "public.mpeg-4") { + self.paste(.video(data)) + return false } else { var isPNG = false var isMemoji = false diff --git a/submodules/TelegramUI/Sources/OpenUrl.swift b/submodules/TelegramUI/Sources/OpenUrl.swift index 2d1e643577..e85b6a7b6f 100644 --- a/submodules/TelegramUI/Sources/OpenUrl.swift +++ b/submodules/TelegramUI/Sources/OpenUrl.swift @@ -12,10 +12,6 @@ import AccountContext import UrlEscaping import PassportUI import UrlHandling -#if ENABLE_WALLET -import WalletUI -import WalletUrl -#endif import OpenInExternalAppUI public struct ParsedSecureIdUrl { @@ -142,18 +138,6 @@ func formattedConfirmationCode(_ code: Int) -> String { } func openExternalUrlImpl(context: AccountContext, urlContext: OpenURLContext, url: String, forceExternal: Bool, presentationData: PresentationData, navigationController: NavigationController?, dismissInput: @escaping () -> Void) { - #if ENABLE_WALLET - if url.hasPrefix("ton://") { - if let url = URL(string: url), let parsedUrl = parseWalletUrl(url) { - context.sharedContext.openWallet(context: context, walletContext: .send(address: parsedUrl.address, amount: parsedUrl.amount, comment: parsedUrl.comment)) { c in - navigationController?.pushViewController(c) - } - } - - return - } - #endif - if forceExternal || url.lowercased().hasPrefix("tel:") || url.lowercased().hasPrefix("calshow:") { context.sharedContext.applicationBindings.openUrl(url) return diff --git a/submodules/TelegramUI/Sources/OverlayAudioPlayerControllerNode.swift b/submodules/TelegramUI/Sources/OverlayAudioPlayerControllerNode.swift index aed16bc4e0..4a3f3d162e 100644 --- a/submodules/TelegramUI/Sources/OverlayAudioPlayerControllerNode.swift +++ b/submodules/TelegramUI/Sources/OverlayAudioPlayerControllerNode.swift @@ -15,6 +15,7 @@ final class OverlayAudioPlayerControllerNode: ViewControllerTracingNode, UIGestu let ready = Promise() private let context: AccountContext + private let peerId: PeerId private var presentationData: PresentationData private let type: MediaManagerPlayerType diff --git a/submodules/TelegramUI/Sources/PeerInfo/PeerInfoData.swift b/submodules/TelegramUI/Sources/PeerInfo/PeerInfoData.swift index 7a9bf48665..029d1b7ade 100644 --- a/submodules/TelegramUI/Sources/PeerInfo/PeerInfoData.swift +++ b/submodules/TelegramUI/Sources/PeerInfo/PeerInfoData.swift @@ -861,6 +861,9 @@ func availableActionsForMemberOfPeer(accountPeerId: PeerId, peer: Peer?, member: if channel.hasPermission(.banMembers) { result.insert(.restrict) } + if channel.hasPermission(.addAdmins) { + result.insert(.promote) + } } } case .legacyGroupMember: @@ -904,7 +907,7 @@ func availableActionsForMemberOfPeer(accountPeerId: PeerId, peer: Peer?, member: return result } -func peerInfoHeaderButtons(peer: Peer?, cachedData: CachedPeerData?, isOpenedFromChat: Bool, videoCallsEnabled: Bool) -> [PeerInfoHeaderButtonKey] { +func peerInfoHeaderButtons(peer: Peer?, cachedData: CachedPeerData?, isOpenedFromChat: Bool, videoCallsEnabled: Bool, isSecretChat: Bool, isContact: Bool) -> [PeerInfoHeaderButtonKey] { var result: [PeerInfoHeaderButtonKey] = [] if let user = peer as? TelegramUser { if !isOpenedFromChat { @@ -931,7 +934,10 @@ func peerInfoHeaderButtons(peer: Peer?, cachedData: CachedPeerData?, isOpenedFro if isOpenedFromChat { result.append(.search) } - result.append(.more) + if isSecretChat && !isContact { + } else { + result.append(.more) + } } else if let channel = peer as? TelegramChannel { var displayLeave = !channel.flags.contains(.isCreator) var canViewStats = false diff --git a/submodules/TelegramUI/Sources/PeerInfo/PeerInfoHeaderNode.swift b/submodules/TelegramUI/Sources/PeerInfo/PeerInfoHeaderNode.swift index 3360d09e62..d83b5c5b0e 100644 --- a/submodules/TelegramUI/Sources/PeerInfo/PeerInfoHeaderNode.swift +++ b/submodules/TelegramUI/Sources/PeerInfo/PeerInfoHeaderNode.swift @@ -2754,7 +2754,7 @@ final class PeerInfoHeaderNode: ASDisplayNode { self.avatarListNode.listContainerNode.updateEntryIsHidden(entry: entry) } - func update(width: CGFloat, containerHeight: CGFloat, containerInset: CGFloat, statusBarHeight: CGFloat, navigationHeight: CGFloat, isModalOverlay: Bool, isMediaOnly: Bool, contentOffset: CGFloat, presentationData: PresentationData, peer: Peer?, cachedData: CachedPeerData?, notificationSettings: TelegramPeerNotificationSettings?, statusData: PeerInfoStatusData?, isContact: Bool, isSettings: Bool, state: PeerInfoState, transition: ContainedViewLayoutTransition, additive: Bool) -> CGFloat { + func update(width: CGFloat, containerHeight: CGFloat, containerInset: CGFloat, statusBarHeight: CGFloat, navigationHeight: CGFloat, isModalOverlay: Bool, isMediaOnly: Bool, contentOffset: CGFloat, presentationData: PresentationData, peer: Peer?, cachedData: CachedPeerData?, notificationSettings: TelegramPeerNotificationSettings?, statusData: PeerInfoStatusData?, isSecretChat: Bool, isContact: Bool, isSettings: Bool, state: PeerInfoState, transition: ContainedViewLayoutTransition, additive: Bool) -> CGFloat { self.state = state self.peer = peer self.avatarListNode.listContainerNode.peer = peer @@ -2848,7 +2848,7 @@ final class PeerInfoHeaderNode: ASDisplayNode { let expandedAvatarListHeight = min(width, containerHeight - expandedAvatarControlsHeight) let expandedAvatarListSize = CGSize(width: width, height: expandedAvatarListHeight) - let buttonKeys: [PeerInfoHeaderButtonKey] = self.isSettings ? [] : peerInfoHeaderButtons(peer: peer, cachedData: cachedData, isOpenedFromChat: self.isOpenedFromChat, videoCallsEnabled: self.videoCallsEnabled) + let buttonKeys: [PeerInfoHeaderButtonKey] = self.isSettings ? [] : peerInfoHeaderButtons(peer: peer, cachedData: cachedData, isOpenedFromChat: self.isOpenedFromChat, videoCallsEnabled: self.videoCallsEnabled, isSecretChat: isSecretChat, isContact: isContact) var isVerified = false let titleString: NSAttributedString diff --git a/submodules/TelegramUI/Sources/PeerInfo/PeerInfoScreen.swift b/submodules/TelegramUI/Sources/PeerInfo/PeerInfoScreen.swift index 355bce3314..92a2200a07 100644 --- a/submodules/TelegramUI/Sources/PeerInfo/PeerInfoScreen.swift +++ b/submodules/TelegramUI/Sources/PeerInfo/PeerInfoScreen.swift @@ -53,6 +53,7 @@ import ListMessageItem import GalleryData import ChatInterfaceState import TelegramVoip +import InviteLinksUI protocol PeerInfoScreenItem: class { var id: AnyHashable { get } @@ -536,6 +537,7 @@ private final class PeerInfoInteraction { let openAddBotToGroup: () -> Void let performBotCommand: (PeerInfoBotCommand) -> Void let editingOpenPublicLinkSetup: () -> Void + let editingOpenInviteLinksSetup: () -> Void let editingOpenDiscussionGroupSetup: () -> Void let editingToggleMessageSignatures: (Bool) -> Void let openParticipantsSection: (PeerInfoParticipantsSection) -> Void @@ -571,6 +573,7 @@ private final class PeerInfoInteraction { openAddBotToGroup: @escaping () -> Void, performBotCommand: @escaping (PeerInfoBotCommand) -> Void, editingOpenPublicLinkSetup: @escaping () -> Void, + editingOpenInviteLinksSetup: @escaping () -> Void, editingOpenDiscussionGroupSetup: @escaping () -> Void, editingToggleMessageSignatures: @escaping (Bool) -> Void, openParticipantsSection: @escaping (PeerInfoParticipantsSection) -> Void, @@ -605,6 +608,7 @@ private final class PeerInfoInteraction { self.openAddBotToGroup = openAddBotToGroup self.performBotCommand = performBotCommand self.editingOpenPublicLinkSetup = editingOpenPublicLinkSetup + self.editingOpenInviteLinksSetup = editingOpenInviteLinksSetup self.editingOpenDiscussionGroupSetup = editingOpenDiscussionGroupSetup self.editingToggleMessageSignatures = editingToggleMessageSignatures self.openParticipantsSection = openParticipantsSection @@ -1146,9 +1150,10 @@ private func editingItems(data: PeerInfoScreenData?, context: AccountContext, pr } } else if let channel = data.peer as? TelegramChannel { let ItemUsername = 1 - let ItemDiscussionGroup = 2 - let ItemSignMessages = 3 - let ItemSignMessagesHelp = 4 + let ItemInviteLinks = 2 + let ItemDiscussionGroup = 3 + let ItemSignMessages = 4 + let ItemSignMessagesHelp = 5 switch channel.info { case .broadcast: @@ -1250,6 +1255,12 @@ private func editingItems(data: PeerInfoScreenData?, context: AccountContext, pr interaction.editingOpenPublicLinkSetup() })) + if !isPublic { + items[.peerPublicSettings]!.append(PeerInfoScreenDisclosureItem(id: ItemInviteLinks, label: .text(""), text: presentationData.strings.GroupInfo_InviteLinks, action: { + interaction.editingOpenInviteLinksSetup() + })) + } + if let linkedDiscussionPeer = data.linkedDiscussionPeer { let peerTitle: String if let addressName = linkedDiscussionPeer.addressName, !addressName.isEmpty { @@ -1357,7 +1368,7 @@ private final class PeerInfoScreenNode: ViewControllerTracingNode, UIScrollViewD private weak var controller: PeerInfoScreen? private let context: AccountContext - private let peerId: PeerId + let peerId: PeerId private let isOpenedFromChat: Bool private let videoCallsEnabled: Bool private let callMessages: [Message] @@ -1499,6 +1510,9 @@ private final class PeerInfoScreenNode: ViewControllerTracingNode, UIScrollViewD editingOpenPublicLinkSetup: { [weak self] in self?.editingOpenPublicLinkSetup() }, + editingOpenInviteLinksSetup: { [weak self] in + self?.editingOpenInviteLinksSetup() + }, editingOpenDiscussionGroupSetup: { [weak self] in self?.editingOpenDiscussionGroupSetup() }, @@ -2899,7 +2913,7 @@ private final class PeerInfoScreenNode: ViewControllerTracingNode, UIScrollViewD actionSheet?.dismissAnimated() } var items: [ActionSheetItem] = [] - if !peerInfoHeaderButtons(peer: peer, cachedData: data.cachedData, isOpenedFromChat: self.isOpenedFromChat, videoCallsEnabled: self.videoCallsEnabled).contains(.search) || (self.headerNode.isAvatarExpanded && self.peerId.namespace == Namespaces.Peer.CloudUser) { + if !peerInfoHeaderButtons(peer: peer, cachedData: data.cachedData, isOpenedFromChat: self.isOpenedFromChat, videoCallsEnabled: self.videoCallsEnabled, isSecretChat: self.peerId.namespace == Namespaces.Peer.SecretChat, isContact: self.data?.isContact ?? false).contains(.search) || (self.headerNode.isAvatarExpanded && self.peerId.namespace == Namespaces.Peer.CloudUser) { items.append(ActionSheetButtonItem(title: presentationData.strings.ChatSearch_SearchPlaceholder, color: .accent, action: { [weak self] in dismissAction() self?.openChatWithMessageSearch() @@ -2955,13 +2969,24 @@ private final class PeerInfoScreenNode: ViewControllerTracingNode, UIScrollViewD } })) } - + if self.peerId.namespace == Namespaces.Peer.CloudUser && user.botInfo == nil && !user.flags.contains(.isSupport) { items.append(ActionSheetButtonItem(title: presentationData.strings.UserInfo_StartSecretChat, color: .accent, action: { [weak self] in dismissAction() self?.openStartSecretChat() })) if data.isContact { + if let cachedData = data.cachedData as? CachedUserData, cachedData.isBlocked { + } else { + items.append(ActionSheetButtonItem(title: presentationData.strings.Conversation_BlockUser, color: .destructive, action: { [weak self] in + dismissAction() + self?.updateBlocked(block: true) + })) + } + } + } else if self.peerId.namespace == Namespaces.Peer.SecretChat && data.isContact { + if let cachedData = data.cachedData as? CachedUserData, cachedData.isBlocked { + } else { items.append(ActionSheetButtonItem(title: presentationData.strings.Conversation_BlockUser, color: .destructive, action: { [weak self] in dismissAction() self?.updateBlocked(block: true) @@ -3010,7 +3035,7 @@ private final class PeerInfoScreenNode: ViewControllerTracingNode, UIScrollViewD self?.openDeletePeer() })) } else { - if !peerInfoHeaderButtons(peer: peer, cachedData: data.cachedData, isOpenedFromChat: self.isOpenedFromChat, videoCallsEnabled: self.videoCallsEnabled).contains(.leave) { + if !peerInfoHeaderButtons(peer: peer, cachedData: data.cachedData, isOpenedFromChat: self.isOpenedFromChat, videoCallsEnabled: self.videoCallsEnabled, isSecretChat: self.peerId.namespace == Namespaces.Peer.SecretChat, isContact: self.data?.isContact ?? false).contains(.leave) { if case .member = channel.participationStatus { items.append(ActionSheetButtonItem(title: presentationData.strings.Channel_LeaveChannel, color: .destructive, action: { [weak self] in dismissAction() @@ -3653,6 +3678,10 @@ private final class PeerInfoScreenNode: ViewControllerTracingNode, UIScrollViewD self.controller?.push(channelVisibilityController(context: self.context, peerId: self.peerId, mode: .generic, upgradedToSupergroup: { _, f in f() })) } + private func editingOpenInviteLinksSetup() { + self.controller?.push(inviteLinkListController(context: self.context, peerId: self.peerId)) + } + private func editingOpenDiscussionGroupSetup() { guard let data = self.data, let peer = data.peer else { return @@ -4513,23 +4542,13 @@ private final class PeerInfoScreenNode: ViewControllerTracingNode, UIScrollViewD } let mode: ChannelVisibilityControllerMode if groupPeer.addressName != nil { - mode = .generic + let visibilityController = channelVisibilityController(context: strongSelf.context, peerId: groupPeer.id, mode: .generic, upgradedToSupergroup: { _, f in f() }, onDismissRemoveController: contactsController) + //visibilityController.navigationPresentation = .modal + + contactsController?.push(visibilityController) } else { - mode = .privateLink + contactsController?.push(InviteLinkInviteController(context: context, peerId: groupPeer.id)) } - let visibilityController = channelVisibilityController(context: strongSelf.context, peerId: groupPeer.id, mode: mode, upgradedToSupergroup: { _, f in f() }, onDismissRemoveController: contactsController) - //visibilityController.navigationPresentation = .modal - - contactsController?.push(visibilityController) - - /*if let navigationController = strongSelf.controller?.navigationController as? NavigationController { - var controllers = navigationController.viewControllers - if let contactsController = contactsController { - controllers.removeAll(where: { $0 === contactsController }) - } - controllers.append(visibilityController) - navigationController.setViewControllers(controllers, animated: true) - }*/ } strongSelf.controller?.push(contactsController) @@ -4953,7 +4972,7 @@ private final class PeerInfoScreenNode: ViewControllerTracingNode, UIScrollViewD } }) } - }, resolvedFaqUrl: self.cachedFaq.get(), exceptionsList: .single(settings.notificationExceptions), archivedStickerPacks: .single(settings.archivedStickerPacks), privacySettings: .single(settings.privacySettings), hasWallet: .single(false), activeSessionsContext: self.activeSessionsContextAndCount.get() |> map { $0?.0 }, webSessionsContext: self.activeSessionsContextAndCount.get() |> map { $0?.2 }), cancel: { [weak self] in + }, resolvedFaqUrl: self.cachedFaq.get(), exceptionsList: .single(settings.notificationExceptions), archivedStickerPacks: .single(settings.archivedStickerPacks), privacySettings: .single(settings.privacySettings), hasTwoStepAuth: self.hasTwoStepAuth.get(), activeSessionsContext: self.activeSessionsContextAndCount.get() |> map { $0?.0 }, webSessionsContext: self.activeSessionsContextAndCount.get() |> map { $0?.2 }), cancel: { [weak self] in self?.deactivateSearch() }) } @@ -5047,7 +5066,7 @@ private final class PeerInfoScreenNode: ViewControllerTracingNode, UIScrollViewD var contentHeight: CGFloat = 0.0 - let headerHeight = self.headerNode.update(width: layout.size.width, containerHeight: layout.size.height, containerInset: layout.safeInsets.left, statusBarHeight: layout.statusBarHeight ?? 0.0, navigationHeight: navigationHeight, isModalOverlay: layout.isModalOverlay, isMediaOnly: self.isMediaOnly, contentOffset: self.isMediaOnly ? 212.0 : self.scrollNode.view.contentOffset.y, presentationData: self.presentationData, peer: self.data?.peer, cachedData: self.data?.cachedData, notificationSettings: self.data?.notificationSettings, statusData: self.data?.status, isContact: self.data?.isContact ?? false, isSettings: self.isSettings, state: self.state, transition: transition, additive: additive) + let headerHeight = self.headerNode.update(width: layout.size.width, containerHeight: layout.size.height, containerInset: layout.safeInsets.left, statusBarHeight: layout.statusBarHeight ?? 0.0, navigationHeight: navigationHeight, isModalOverlay: layout.isModalOverlay, isMediaOnly: self.isMediaOnly, contentOffset: self.isMediaOnly ? 212.0 : self.scrollNode.view.contentOffset.y, presentationData: self.presentationData, peer: self.data?.peer, cachedData: self.data?.cachedData, notificationSettings: self.data?.notificationSettings, statusData: self.data?.status, isSecretChat: self.peerId.namespace == Namespaces.Peer.SecretChat, isContact: self.data?.isContact ?? false, isSettings: self.isSettings, state: self.state, transition: transition, additive: additive) let headerFrame = CGRect(origin: CGPoint(x: 0.0, y: contentHeight), size: CGSize(width: layout.size.width, height: headerHeight)) if additive { transition.updateFrameAdditive(node: self.headerNode, frame: headerFrame) @@ -5294,7 +5313,7 @@ private final class PeerInfoScreenNode: ViewControllerTracingNode, UIScrollViewD if let (layout, navigationHeight) = self.validLayout { if !additive { - let _ = self.headerNode.update(width: layout.size.width, containerHeight: layout.size.height, containerInset: layout.safeInsets.left, statusBarHeight: layout.statusBarHeight ?? 0.0, navigationHeight: navigationHeight, isModalOverlay: layout.isModalOverlay, isMediaOnly: self.isMediaOnly, contentOffset: self.isMediaOnly ? 212.0 : offsetY, presentationData: self.presentationData, peer: self.data?.peer, cachedData: self.data?.cachedData, notificationSettings: self.data?.notificationSettings, statusData: self.data?.status, isContact: self.data?.isContact ?? false, isSettings: self.isSettings, state: self.state, transition: transition, additive: additive) + let _ = self.headerNode.update(width: layout.size.width, containerHeight: layout.size.height, containerInset: layout.safeInsets.left, statusBarHeight: layout.statusBarHeight ?? 0.0, navigationHeight: navigationHeight, isModalOverlay: layout.isModalOverlay, isMediaOnly: self.isMediaOnly, contentOffset: self.isMediaOnly ? 212.0 : offsetY, presentationData: self.presentationData, peer: self.data?.peer, cachedData: self.data?.cachedData, notificationSettings: self.data?.notificationSettings, statusData: self.data?.status, isSecretChat: self.peerId.namespace == Namespaces.Peer.SecretChat, isContact: self.data?.isContact ?? false, isSettings: self.isSettings, state: self.state, transition: transition, additive: additive) } let paneAreaExpansionDistance: CGFloat = 32.0 @@ -6134,7 +6153,7 @@ private final class PeerInfoNavigationTransitionNode: ASDisplayNode, CustomNavig self.headerNode.navigationTransition = PeerInfoHeaderNavigationTransition(sourceNavigationBar: bottomNavigationBar, sourceTitleView: previousTitleView, sourceTitleFrame: previousTitleFrame, sourceSubtitleFrame: previousStatusFrame, fraction: fraction) if let (layout, _) = self.screenNode.validLayout { - let _ = self.headerNode.update(width: layout.size.width, containerHeight: layout.size.height, containerInset: layout.safeInsets.left, statusBarHeight: layout.statusBarHeight ?? 0.0, navigationHeight: topNavigationBar.bounds.height, isModalOverlay: layout.isModalOverlay, isMediaOnly: false, contentOffset: 0.0, presentationData: self.presentationData, peer: self.screenNode.data?.peer, cachedData: self.screenNode.data?.cachedData, notificationSettings: self.screenNode.data?.notificationSettings, statusData: self.screenNode.data?.status, isContact: self.screenNode.data?.isContact ?? false, isSettings: self.screenNode.isSettings, state: self.screenNode.state, transition: transition, additive: false) + let _ = self.headerNode.update(width: layout.size.width, containerHeight: layout.size.height, containerInset: layout.safeInsets.left, statusBarHeight: layout.statusBarHeight ?? 0.0, navigationHeight: topNavigationBar.bounds.height, isModalOverlay: layout.isModalOverlay, isMediaOnly: false, contentOffset: 0.0, presentationData: self.presentationData, peer: self.screenNode.data?.peer, cachedData: self.screenNode.data?.cachedData, notificationSettings: self.screenNode.data?.notificationSettings, statusData: self.screenNode.data?.status, isSecretChat: self.screenNode.peerId.namespace == Namespaces.Peer.SecretChat, isContact: self.screenNode.data?.isContact ?? false, isSettings: self.screenNode.isSettings, state: self.screenNode.state, transition: transition, additive: false) } let titleScale = (fraction * previousTitleNode.bounds.height + (1.0 - fraction) * self.headerNode.titleNodeRawContainer.bounds.height) / previousTitleNode.bounds.height diff --git a/submodules/TelegramUI/Sources/SharedAccountContext.swift b/submodules/TelegramUI/Sources/SharedAccountContext.swift index f1ef91bb18..950c18fdd2 100644 --- a/submodules/TelegramUI/Sources/SharedAccountContext.swift +++ b/submodules/TelegramUI/Sources/SharedAccountContext.swift @@ -17,10 +17,6 @@ import PeersNearbyUI import PeerInfoUI import SettingsUI import UrlHandling -#if ENABLE_WALLET -import WalletUI -import WalletCore -#endif import LegacyMediaPickerUI import LocalMediaResources import OverlayStatusController diff --git a/submodules/TelegramUniversalVideoContent/Sources/GenericEmbedImplementation.swift b/submodules/TelegramUniversalVideoContent/Sources/GenericEmbedImplementation.swift index ce2f67df30..731c1364b5 100644 --- a/submodules/TelegramUniversalVideoContent/Sources/GenericEmbedImplementation.swift +++ b/submodules/TelegramUniversalVideoContent/Sources/GenericEmbedImplementation.swift @@ -43,7 +43,7 @@ final class GenericEmbedImplementation: WebEmbedImplementation { self.onPlaybackStarted = onPlaybackStarted updateStatus(self.status) - if self.url.contains("player.twitch.tv/"), let url = URL(string: self.url) { + if self.url.contains(".twitch.tv/"), let url = URL(string: self.url) { webView.load(URLRequest(url: url)) } else { let html = String(format: htmlTemplate, self.url) diff --git a/submodules/TooltipUI/Sources/TooltipScreen.swift b/submodules/TooltipUI/Sources/TooltipScreen.swift index 236e74cb63..a61186f50b 100644 --- a/submodules/TooltipUI/Sources/TooltipScreen.swift +++ b/submodules/TooltipUI/Sources/TooltipScreen.swift @@ -11,6 +11,11 @@ import TextFormat import Postbox import UrlEscaping +public protocol TooltipCustomContentNode: ASDisplayNode { + func animateIn() + func updateLayout(size: CGSize) -> CGSize +} + public enum TooltipActiveTextItem { case url(String, Bool) case mention(PeerId, String) @@ -27,6 +32,7 @@ public enum TooltipActiveTextAction { private final class TooltipScreenNode: ViewControllerTracingNode { private let tooltipStyle: TooltipScreen.Style private let icon: TooltipScreen.Icon? + private let customContentNode: TooltipCustomContentNode? private let location: TooltipScreen.Location private let displayDuration: TooltipScreen.DisplayDuration private let shouldDismissOnTouch: (CGPoint) -> TooltipScreen.DismissOnTouch @@ -47,9 +53,10 @@ private final class TooltipScreenNode: ViewControllerTracingNode { private var validLayout: ContainerViewLayout? - init(text: String, textEntities: [MessageTextEntity], style: TooltipScreen.Style, icon: TooltipScreen.Icon?, location: TooltipScreen.Location, displayDuration: TooltipScreen.DisplayDuration, shouldDismissOnTouch: @escaping (CGPoint) -> TooltipScreen.DismissOnTouch, requestDismiss: @escaping () -> Void, openActiveTextItem: @escaping (TooltipActiveTextItem, TooltipActiveTextAction) -> Void) { + init(text: String, textEntities: [MessageTextEntity], style: TooltipScreen.Style, icon: TooltipScreen.Icon?, customContentNode: TooltipCustomContentNode? = nil, location: TooltipScreen.Location, displayDuration: TooltipScreen.DisplayDuration, shouldDismissOnTouch: @escaping (CGPoint) -> TooltipScreen.DismissOnTouch, requestDismiss: @escaping () -> Void, openActiveTextItem: @escaping (TooltipActiveTextItem, TooltipActiveTextAction) -> Void) { self.tooltipStyle = style self.icon = icon + self.customContentNode = customContentNode self.location = location self.displayDuration = displayDuration self.shouldDismissOnTouch = shouldDismissOnTouch @@ -500,6 +507,7 @@ public final class TooltipScreen: ViewController { public let textEntities: [MessageTextEntity] private let style: TooltipScreen.Style private let icon: TooltipScreen.Icon? + private let customContentNode: TooltipCustomContentNode? private let location: TooltipScreen.Location private let displayDuration: DisplayDuration private let shouldDismissOnTouch: (CGPoint) -> TooltipScreen.DismissOnTouch @@ -517,11 +525,12 @@ public final class TooltipScreen: ViewController { private var dismissTimer: Foundation.Timer? - public init(text: String, textEntities: [MessageTextEntity] = [], style: TooltipScreen.Style = .default, icon: TooltipScreen.Icon?, location: TooltipScreen.Location, displayDuration: DisplayDuration = .default, shouldDismissOnTouch: @escaping (CGPoint) -> TooltipScreen.DismissOnTouch, openActiveTextItem: @escaping (TooltipActiveTextItem, TooltipActiveTextAction) -> Void = { _, _ in }) { + public init(text: String, textEntities: [MessageTextEntity] = [], style: TooltipScreen.Style = .default, icon: TooltipScreen.Icon?, customContentNode: TooltipCustomContentNode? = nil, location: TooltipScreen.Location, displayDuration: DisplayDuration = .default, shouldDismissOnTouch: @escaping (CGPoint) -> TooltipScreen.DismissOnTouch, openActiveTextItem: @escaping (TooltipActiveTextItem, TooltipActiveTextAction) -> Void = { _, _ in }) { self.text = text self.textEntities = textEntities self.style = style self.icon = icon + self.customContentNode = customContentNode self.location = location self.displayDuration = displayDuration self.shouldDismissOnTouch = shouldDismissOnTouch @@ -580,7 +589,7 @@ public final class TooltipScreen: ViewController { } override public func loadDisplayNode() { - self.displayNode = TooltipScreenNode(text: self.text, textEntities: self.textEntities, style: self.style, icon: self.icon, location: self.location, displayDuration: self.displayDuration, shouldDismissOnTouch: self.shouldDismissOnTouch, requestDismiss: { [weak self] in + self.displayNode = TooltipScreenNode(text: self.text, textEntities: self.textEntities, style: self.style, icon: self.icon, customContentNode: self.customContentNode, location: self.location, displayDuration: self.displayDuration, shouldDismissOnTouch: self.shouldDismissOnTouch, requestDismiss: { [weak self] in guard let strongSelf = self else { return } diff --git a/submodules/UrlHandling/Sources/UrlHandling.swift b/submodules/UrlHandling/Sources/UrlHandling.swift index f814af7b40..d24349be69 100644 --- a/submodules/UrlHandling/Sources/UrlHandling.swift +++ b/submodules/UrlHandling/Sources/UrlHandling.swift @@ -8,9 +8,6 @@ import MtProtoKit import TelegramPresentationData import TelegramUIPreferences import AccountContext -#if ENABLE_WALLET -import WalletUrl -#endif private let baseTelegramMePaths = ["telegram.me", "t.me", "telegram.dog"] private let baseTelegraPhPaths = ["telegra.ph/", "te.legra.ph/", "graph.org/", "t.me/iv?"] @@ -494,13 +491,6 @@ public func parseWallpaperUrl(_ url: String) -> WallpaperUrlParameter? { } public func resolveUrlImpl(account: Account, url: String) -> Signal { - #if ENABLE_WALLET - if url.hasPrefix("ton://") { - if let url = URL(string: url), let parsedUrl = parseWalletUrl(url) { - return .single(.wallet(address: parsedUrl.address, amount: parsedUrl.amount, comment: parsedUrl.comment)) - } - } - #endif let schemes = ["http://", "https://", ""] for basePath in baseTelegramMePaths { for scheme in schemes { diff --git a/submodules/WebPBinding/Sources/UIImage+WebP.m b/submodules/WebPBinding/Sources/UIImage+WebP.m index 19f4c6037d..3f541a37c7 100644 --- a/submodules/WebPBinding/Sources/UIImage+WebP.m +++ b/submodules/WebPBinding/Sources/UIImage+WebP.m @@ -72,17 +72,24 @@ + (NSData * _Nullable)convertToWebP:(UIImage * _Nonnull)image quality:(CGFloat)quality error:(NSError ** _Nullable)error { WebPPreset preset = WEBP_PRESET_DEFAULT; - CGImageRef webPImageRef = image.CGImage; - size_t webPBytesPerRow = CGImageGetBytesPerRow(webPImageRef); - - size_t webPImageWidth = CGImageGetWidth(webPImageRef); - size_t webPImageHeight = CGImageGetHeight(webPImageRef); - - CGDataProviderRef webPDataProviderRef = CGImageGetDataProvider(webPImageRef); - CFDataRef webPImageDatRef = CGDataProviderCopyData(webPDataProviderRef); - - uint8_t *webPImageData = (uint8_t *)CFDataGetBytePtr(webPImageDatRef); - + CGImageRef imageRef = [image CGImage]; + + NSUInteger width = CGImageGetWidth(imageRef); + NSUInteger height = CGImageGetHeight(imageRef); + + CGColorSpaceRef colorSpace = CGColorSpaceCreateDeviceRGB(); + + uint8_t *rawData = malloc(height * width * 4); + + NSUInteger bytesPerPixel = 4; + NSUInteger bytesPerRow = bytesPerPixel * width; + NSUInteger bitsPerComponent = 8; + CGContextRef context = CGBitmapContextCreate(rawData, width, height, bitsPerComponent, bytesPerRow, colorSpace, kCGImageAlphaPremultipliedLast | kCGBitmapByteOrder32Big); + CGColorSpaceRelease(colorSpace); + + CGContextDrawImage(context, CGRectMake(0, 0, width, height), imageRef); + CGContextRelease(context); + WebPConfig config; if (!WebPConfigPreset(&config, preset, quality)) { NSMutableDictionary *errorDetail = [NSMutableDictionary dictionary]; @@ -90,7 +97,7 @@ if(error != NULL) *error = [NSError errorWithDomain:[NSString stringWithFormat:@"%@.errorDomain", [[NSBundle mainBundle] bundleIdentifier]] code:-101 userInfo:errorDetail]; - CFRelease(webPImageDatRef); + free(rawData); return nil; } @@ -102,7 +109,7 @@ if(error != NULL) *error = [NSError errorWithDomain:[NSString stringWithFormat:@"%@.errorDomain", [[NSBundle mainBundle] bundleIdentifier]] code:-101 userInfo:errorDetail]; - CFRelease(webPImageDatRef); + free(rawData); return nil; } @@ -113,19 +120,19 @@ if(error != NULL) *error = [NSError errorWithDomain:[NSString stringWithFormat:@"%@.errorDomain", [[NSBundle mainBundle] bundleIdentifier]] code:-101 userInfo:errorDetail]; - CFRelease(webPImageDatRef); + free(rawData); return nil; } - pic.width = (int)webPImageWidth; - pic.height = (int)webPImageHeight; + pic.width = (int)width; + pic.height = (int)height; pic.colorspace = WEBP_YUV420; - if ([[NSProcessInfo processInfo] isOperatingSystemAtLeastVersion:(NSOperatingSystemVersion){12, 0, 0}]) { - WebPPictureImportRGBA(&pic, webPImageData, (int)webPBytesPerRow); - } else { - WebPPictureImportBGRA(&pic, webPImageData, (int)webPBytesPerRow); - } +// if ([[NSProcessInfo processInfo] isOperatingSystemAtLeastVersion:(NSOperatingSystemVersion){12, 0, 0}]) { + WebPPictureImportRGBA(&pic, rawData, (int)bytesPerRow); +// } else { +// WebPPictureImportBGRA(&pic, webPImageData, (int)webPBytesPerRow); +// } WebPPictureARGBToYUVA(&pic, WEBP_YUV420); WebPCleanupTransparentArea(&pic); @@ -140,7 +147,7 @@ free(writer.mem); WebPPictureFree(&pic); - CFRelease(webPImageDatRef); + free(rawData); return webPFinalData; } diff --git a/submodules/WebSearchUI/Sources/WebSearchController.swift b/submodules/WebSearchUI/Sources/WebSearchController.swift index d15c663dc4..2e872a9aa8 100644 --- a/submodules/WebSearchUI/Sources/WebSearchController.swift +++ b/submodules/WebSearchUI/Sources/WebSearchController.swift @@ -150,7 +150,7 @@ public final class WebSearchController: ViewController { private var navigationContentNode: WebSearchNavigationContentNode? - var presentStickers: ((@escaping (TelegramMediaFile, Bool, UIView, CGRect) -> Void) -> TGPhotoPaintStickersScreen?)? { + public var presentStickers: ((@escaping (TelegramMediaFile, Bool, UIView, CGRect) -> Void) -> TGPhotoPaintStickersScreen?)? { didSet { self.controllerNode.presentStickers = self.presentStickers }