From 4783892aabfc4daca55cc5dc14f60b4d3a33c1c8 Mon Sep 17 00:00:00 2001 From: Denis Shilovich Date: Mon, 11 Mar 2024 11:02:33 +0000 Subject: [PATCH] Release 1.5.7 (229) --- .../NGLottie/Sources/LottieViewImpl.swift | 27 + Nicegram/NGStats/BUILD | 4 +- Nicegram/NGStats/Sources/ChatsSharing.swift | 278 +++ Nicegram/NGStats/Sources/DTO.swift | 137 -- Nicegram/NGStats/Sources/MetaStorage.swift | 41 - Nicegram/NGStats/Sources/NGStats.swift | 182 -- .../NGStats/Sources/StickersSharing.swift | 60 + Nicegram/NGStats/Sources/Throttling.swift | 53 - Nicegram/NGUI/BUILD | 2 + .../Sources/NicegramSettingsController.swift | 121 +- Package.resolved | 19 +- Telegram/SiriIntents/IntentMessages.swift | 2 +- .../ar.lproj/NiceLocalizable.strings | 8 +- .../de.lproj/NiceLocalizable.strings | 8 +- .../Telegram-iOS/en.lproj/Localizable.strings | 279 +++ .../en.lproj/NiceLocalizable.strings | 8 +- .../es.lproj/NiceLocalizable.strings | 8 +- .../fr.lproj/NiceLocalizable.strings | 8 +- .../it.lproj/NiceLocalizable.strings | 8 +- .../km.lproj/NiceLocalizable.strings | 12 +- .../ko.lproj/NiceLocalizable.strings | 8 +- .../pl.lproj/NiceLocalizable.strings | 8 +- .../pt.lproj/NiceLocalizable.strings | 8 +- .../ru.lproj/NiceLocalizable.strings | 8 +- .../tr.lproj/NiceLocalizable.strings | 8 +- .../uk.lproj/NiceLocalizable.strings | 2 - .../vi.lproj/NiceLocalizable.strings | 34 + .../zh-hans.lproj/NiceLocalizable.strings | 8 +- .../zh-hant.lproj/NiceLocalizable.strings | 8 +- .../Sources/AccountContext.swift | 23 +- .../Sources/ChatController.swift | 55 +- .../ContactMultiselectionController.swift | 5 +- .../AccountContext/Sources/MediaManager.swift | 4 +- .../Sources/PeerNameColors.swift | 19 + .../AccountContext/Sources/Premium.swift | 9 + .../Sources/AttachmentController.swift | 103 +- .../Sources/AttachmentPanel.swift | 7 + .../AuthorizationSequenceController.swift | 13 +- .../BrowserUI/Sources/BrowserScreen.swift | 1 + submodules/ChatListUI/BUILD | 1 + .../Sources/ChatListController.swift | 5 +- .../Sources/ChatListControllerNode.swift | 1 + .../ChatListFilterPresetController.swift | 84 +- .../ChatListFilterPresetListController.swift | 117 +- .../ChatListFilterPresetListItem.swift | 62 +- .../Sources/ChatListSearchListPaneNode.swift | 2 + .../Sources/ChatListShimmerNode.swift | 1 + .../Sources/Node/ChatListItem.swift | 471 ++++- .../Sources/Node/ChatListNode.swift | 68 +- .../Sources/Node/ChatListNodeLocation.swift | 4 +- .../ChatPanelInterfaceInteraction.swift | 8 + .../ChatPresentationInterfaceState.swift | 6 +- ...SendMessageActionSheetControllerNode.swift | 10 +- .../Source/Base/CombinedComponent.swift | 4 +- .../ComponentFlow/Source/Base/Component.swift | 6 +- .../Source/Components/RoundedRectangle.swift | 18 +- .../Source/Host/ComponentHostView.swift | 6 +- .../Sources/ViewControllerComponent.swift | 7 + .../Sources/ContactListNode.swift | 38 +- .../Sources/ContactsController.swift | 34 +- .../Sources/ContactsControllerNode.swift | 1 + .../Display/Source/AlertController.swift | 2 +- submodules/Display/Source/TextNode.swift | 19 +- submodules/Display/Source/WindowContent.swift | 43 +- .../Sources/DrawingEntitiesView.swift | 50 +- .../DrawingUI/Sources/DrawingScreen.swift | 1 + .../Sources/ImageObjectSeparation.swift | 84 - .../Sources/StickerPickerScreen.swift | 1 + submodules/Emoji/Sources/EmojiUtils.swift | 2 +- .../GalleryData/Sources/GalleryData.swift | 5 + .../ChatItemGalleryFooterContentNode.swift | 7 +- .../GalleryUI/Sources/GalleryController.swift | 25 +- .../Items/UniversalVideoGalleryItem.swift | 2 +- .../Sources/HashtagSearchController.swift | 1 + .../Sources/ItemListAddressItem.swift | 17 +- .../Sources/ItemListPeerActionItem.swift | 6 + .../Sources/ItemListStickerPackItem.swift | 2 +- .../Sources/ItemListController.swift | 8 + .../Sources/ItemListControllerNode.swift | 3 + .../ItemListSelectableControlNode.swift | 35 +- .../Items/ItemListTextWithLabelItem.swift | 2 +- .../Sources/TGMediaAssetsController.m | 4 + .../Sources/LegacyAttachmentMenu.swift | 31 +- .../Sources/ListMessageFileItemNode.swift | 2 +- .../Sources/ListMessageSnippetItemNode.swift | 2 +- .../Sources/LocationPickerController.swift | 2 +- .../LocationPickerControllerNode.swift | 6 +- .../Sources/LegacyMediaPickerGallery.swift | 6 +- .../Sources/MediaPickerScreen.swift | 13 +- .../SecureIdPlaintextFormControllerNode.swift | 11 +- .../Sources/DeviceContactInfoController.swift | 2 +- .../AdditionalMessageHistoryViewData.swift | 2 +- submodules/Postbox/Sources/ChatLocation.swift | 6 +- submodules/Postbox/Sources/Message.swift | 34 +- .../Postbox/Sources/MessageHistoryView.swift | 24 +- .../Sources/MessageOfInterestHolesView.swift | 6 +- submodules/Postbox/Sources/Postbox.swift | 32 +- submodules/Postbox/Sources/PostboxView.swift | 71 +- submodules/Postbox/Sources/ViewTracker.swift | 37 +- submodules/Postbox/Sources/Views.swift | 75 + submodules/PremiumUI/BUILD | 4 + submodules/PremiumUI/Resources/badge | Bin 0 -> 13043 bytes submodules/PremiumUI/Resources/badge.scn | Bin 27949 -> 0 bytes submodules/PremiumUI/Resources/boost | Bin 0 -> 17568 bytes submodules/PremiumUI/Resources/boost.scn | Bin 43534 -> 0 bytes submodules/PremiumUI/Resources/business.png | Bin 0 -> 2160 bytes submodules/PremiumUI/Resources/business.scn | Bin 0 -> 28482 bytes submodules/PremiumUI/Resources/coin | Bin 0 -> 110086 bytes submodules/PremiumUI/Resources/coin_anim.png | Bin 0 -> 61046 bytes submodules/PremiumUI/Resources/coin_edge.png | Bin 0 -> 332 bytes .../PremiumUI/Resources/darkerTexture.jpg | Bin 0 -> 9624 bytes .../PremiumUI/Resources/diagonal_shine.png | Bin 0 -> 11771 bytes submodules/PremiumUI/Resources/emoji | Bin 0 -> 20580 bytes submodules/PremiumUI/Resources/emoji.scn | Bin 58392 -> 0 bytes submodules/PremiumUI/Resources/gift | Bin 0 -> 21826 bytes submodules/PremiumUI/Resources/gift.scn | Bin 53497 -> 0 bytes .../PremiumUI/Resources/lighterTexture.jpg | Bin 0 -> 9199 bytes submodules/PremiumUI/Resources/lightspeed | Bin 0 -> 9780 bytes submodules/PremiumUI/Resources/lightspeed.scn | Bin 14629 -> 0 bytes submodules/PremiumUI/Resources/star | Bin 58585 -> 59131 bytes submodules/PremiumUI/Resources/swirl | Bin 0 -> 9213 bytes submodules/PremiumUI/Resources/swirl.scn | Bin 16919 -> 0 bytes submodules/PremiumUI/Resources/tag | Bin 0 -> 20564 bytes submodules/PremiumUI/Resources/tag.scn | Bin 58387 -> 0 bytes submodules/PremiumUI/Resources/texture.jpg | Bin 2902 -> 9195 bytes .../PremiumUI/Sources/BadgeBusinessView.swift | 61 + .../PremiumUI/Sources/BadgeStarsView.swift | 12 +- .../BoostHeaderBackgroundComponent.swift | 4 +- .../Sources/BusinessPageComponent.swift | 652 ++++++ .../Sources/EmojiHeaderComponent.swift | 2 - .../PremiumUI/Sources/FasterStarsView.swift | 6 +- .../Sources/GiftAvatarComponent.swift | 4 +- .../PremiumUI/Sources/GiftOptionItem.swift | 6 +- .../Sources/PhoneDemoComponent.swift | 8 + .../Sources/PremiumBoostLevelsScreen.swift | 2 +- .../Sources/PremiumCoinComponent.swift | 509 +++++ .../PremiumUI/Sources/PremiumDemoScreen.swift | 39 +- .../PremiumUI/Sources/PremiumGiftScreen.swift | 6 + .../Sources/PremiumIntroScreen.swift | 1504 +++++++++----- .../Sources/PremiumLimitsListScreen.swift | 265 ++- .../Sources/PremiumOptionComponent.swift | 340 +++ .../Sources/PremiumStarComponent.swift | 47 +- .../Sources/ReplaceBoostScreen.swift | 1 + .../PremiumUI/Sources/SwirlStarsView.swift | 6 +- .../Sources/SearchPeerMembers.swift | 2 +- .../Sources/ChangePhoneNumberController.swift | 22 +- .../ChangePhoneNumberControllerNode.swift | 14 +- .../Sources/DeleteAccountPhoneItem.swift | 14 +- .../Search/SettingsSearchableItems.swift | 4 +- .../TextSizeSelectionController.swift | 1 + .../Themes/ThemePreviewControllerNode.swift | 1 + submodules/ShareController/BUILD | 1 - .../Sources/ShareController.swift | 4 +- .../Sources/ChannelStatsController.swift | 8 +- submodules/StickerPeekUI/BUILD | 1 - submodules/TelegramApi/Sources/Api0.swift | 91 +- submodules/TelegramApi/Sources/Api1.swift | 52 + submodules/TelegramApi/Sources/Api10.swift | 136 +- submodules/TelegramApi/Sources/Api11.swift | 242 +-- submodules/TelegramApi/Sources/Api12.swift | 180 +- submodules/TelegramApi/Sources/Api18.swift | 48 + submodules/TelegramApi/Sources/Api2.swift | 344 ++++ submodules/TelegramApi/Sources/Api21.swift | 44 + submodules/TelegramApi/Sources/Api22.swift | 304 +++ submodules/TelegramApi/Sources/Api23.swift | 146 +- submodules/TelegramApi/Sources/Api24.swift | 104 +- submodules/TelegramApi/Sources/Api25.swift | 52 + submodules/TelegramApi/Sources/Api27.swift | 192 +- submodules/TelegramApi/Sources/Api28.swift | 414 ++-- submodules/TelegramApi/Sources/Api29.swift | 486 +++-- submodules/TelegramApi/Sources/Api30.swift | 528 +++-- submodules/TelegramApi/Sources/Api31.swift | 168 ++ submodules/TelegramApi/Sources/Api32.swift | 441 +++- submodules/TelegramApi/Sources/Api4.swift | 102 +- submodules/TelegramApi/Sources/Api7.swift | 144 ++ .../Account/AccountIntermediateState.swift | 18 + .../Sources/Account/AccountManager.swift | 1 + .../ApiUtils/StoreMessage_Telegram.swift | 21 +- .../PendingMessages/EnqueueMessage.swift | 27 +- .../PendingMessages/RequestEditMessage.swift | 10 +- .../StandaloneSendMessage.swift | 12 +- .../State/AccountStateManagementUtils.swift | 38 + .../Sources/State/AccountViewTracker.swift | 151 +- .../Sources/State/ApplyUpdateMessage.swift | 50 +- .../CloudChatRemoveMessagesOperation.swift | 25 +- .../State/HistoryViewStateValidation.swift | 164 +- ...gedCloudChatRemoveMessagesOperations.swift | 55 +- ...anagedConsumePersonalMessagesActions.swift | 2 +- .../Sources/State/PendingMessageManager.swift | 99 +- .../Sources/State/Serialization.swift | 2 +- .../Sources/State/UpdateMessageService.swift | 4 +- .../Sources/State/UpdatesApiUtils.swift | 18 +- .../SyncCore/QuickReplyMessageAttribute.swift | 23 + .../SyncCore/SyncCore_CachedUserData.swift | 336 ++- ...ore_CloudChatRemoveMessagesOperation.swift | 11 +- .../SyncCore/SyncCore_MediaReference.swift | 38 +- .../SyncCore/SyncCore_Namespaces.swift | 18 + .../TelegramEngineAccountData.swift | 100 + .../TelegramEngine/Data/PeersData.swift | 225 ++ .../Data/TelegramEngineData.swift | 67 + .../DeleteMessagesInteractively.swift | 56 +- .../TelegramEngine/Messages/ForwardGame.swift | 2 +- .../TelegramEngine/Messages/Polls.swift | 2 +- .../Messages/QuickReplyMessages.swift | 899 ++++++++ .../Messages/ReplyThreadHistory.swift | 2 +- .../Messages/SearchMessages.swift | 8 +- .../Messages/SparseMessageList.swift | 2 +- .../Messages/TelegramEngineMessages.swift | 2 +- .../TelegramEngine/Messages/TimeZones.swift | 106 + .../Peers/ChatListFiltering.swift | 276 ++- .../Peers/TelegramEnginePeers.swift | 14 +- .../Peers/UpdateCachedPeerData.swift | 65 +- .../Stickers/TelegramEngineStickers.swift | 6 + .../Sources/Utils/MessageUtils.swift | 30 + .../TelegramNotices/Sources/Notices.swift | 26 + .../PresentationResourcesSettings.swift | 34 +- .../Sources/DateFormat.swift | 14 +- .../Sources/PresenceStrings.swift | 71 +- submodules/TelegramUI/BUILD | 6 +- .../AudioTranscriptionButtonComponent.swift | 10 + .../CameraScreen/Sources/CameraScreen.swift | 1 + ...ChatInlineSearchResultsListComponent.swift | 22 +- .../ChatMessageAnimatedStickerItemNode.swift | 15 +- .../ChatMessageAttachedContentNode.swift | 7 +- .../Sources/ChatMessageBubbleItemNode.swift | 11 +- .../ChatMessageContactBubbleContentNode.swift | 6 +- .../ChatMessageFileBubbleContentNode.swift | 8 +- ...MessageInstantVideoBubbleContentNode.swift | 28 +- .../ChatMessageInstantVideoItemNode.swift | 8 +- .../ChatMessageInteractiveFileNode.swift | 12 +- ...atMessageInteractiveInstantVideoNode.swift | 6 +- .../Sources/ChatMessageDateHeader.swift | 4 +- .../Sources/ChatMessageItemImpl.swift | 5 +- .../ChatMessageMapBubbleContentNode.swift | 6 +- .../ChatMessageMediaBubbleContentNode.swift | 6 +- .../ChatMessagePollBubbleContentNode.swift | 18 +- ...atMessageRestrictedBubbleContentNode.swift | 28 +- .../Sources/ChatMessageStickerItemNode.swift | 13 +- .../ChatMessageTextBubbleContentNode.swift | 3 + .../Sources/ChatNavigationButton.swift | 1 + .../Sources/ChatRecentActionsController.swift | 2 + .../ChatRecentActionsControllerNode.swift | 1 + .../ChatRecentActionsHistoryTransition.swift | 3 + .../Sources/ChatControllerInteraction.swift | 3 + .../Sources/ChatEntityKeyboardInputNode.swift | 7 + .../Sources/ChatListHeaderComponent.swift | 21 +- .../Sources/ChatListNavigationBar.swift | 10 + .../EmptyStateIndicatorComponent.swift | 63 +- .../Sources/EmojiPagerContentComponent.swift | 10 + .../Sources/EmojiPagerContentSignals.swift | 2 +- .../LegacyCamera/Sources/LegacyCamera.swift | 19 +- .../Sources/LegacyMessageInputPanel.swift | 2 +- .../Sources/ListActionItemComponent.swift | 266 ++- .../ListItemSliderSelectorComponent/BUILD | 23 + .../ListItemSliderSelectorComponent.swift | 151 ++ .../ListItemSwipeOptionContainer/BUILD | 21 + .../ListItemSwipeOptionContainer.swift | 905 ++++++++ .../ListMultilineTextFieldItemComponent/BUILD | 24 + .../ListMultilineTextFieldItemComponent.swift | 281 +++ .../Sources/ListSectionComponent.swift | 1 + .../ListTextFieldItemComponent/BUILD | 3 + .../Sources/ListTextFieldItemComponent.swift | 108 +- .../MetalResources/EditorDefault.metal | 10 +- .../MetalResources/EditorDual.metal | 22 +- .../Sources/ImageObjectSeparation.swift | 153 ++ .../MediaEditor/Sources/MediaEditor.swift | 80 +- .../Sources/MediaEditorComposer.swift | 19 +- .../Sources/MediaEditorRenderer.swift | 3 +- .../Sources/MediaEditorValues.swift | 8 + .../MediaEditor/Sources/RenderPass.swift | 1 + .../Sources/UniversalTextureSource.swift | 16 +- .../MediaEditor/Sources/VideoFinishPass.swift | 105 +- .../Components/MediaEditorScreen/BUILD | 2 + .../Sources/MediaCutoutScreen.swift | 437 ++++ .../Sources/MediaEditorScreen.swift | 1821 ++++++++++------- .../Sources/MediaToolsScreen.swift | 1 + .../Sources/SaveProgressScreen.swift | 1 + .../Components/PeerInfo/PeerInfoScreen/BUILD | 1 + .../ListItems/PeerInfoScreenAddressItem.swift | 20 +- .../PeerInfoScreenBusinessHoursItem.swift | 624 ++++++ .../Sources/PeerInfoScreen.swift | 115 +- .../Sources/PeerInfoStoryGridScreen.swift | 76 +- .../Sources/PeerInfoStoryPaneNode.swift | 17 +- .../Sources/PeerSelectionControllerNode.swift | 2 + .../Sources/PlainButtonComponent.swift | 41 +- .../AutomaticBusinessMessageSetupScreen/BUILD | 55 + ...aticBusinessMessageListItemComponent.swift | 318 +++ ...aticBusinessMessageSetupChatContents.swift | 252 +++ .../AutomaticBusinessMessageSetupScreen.swift | 1667 +++++++++++++++ .../Sources/BottomPanelComponent.swift | 117 ++ .../QuickReplyEmptyStateComponent.swift | 186 ++ .../Sources/QuickReplySetupScreen.swift | 1384 +++++++++++++ .../Settings/BusinessHoursSetupScreen/BUILD | 43 + .../Sources/BusinessDaySetupScreen.swift | 677 ++++++ .../Sources/BusinessHoursSetupScreen.swift | 871 ++++++++ .../BUILD | 16 +- .../Sources/BusinessLocationSetupScreen.swift | 682 ++++++ .../Sources/MapPreviewComponent.swift | 139 ++ .../Sources/BusinessSetupScreen.swift | 426 ---- .../Settings/ChatbotSetupScreen/BUILD | 4 + .../ChatbotSearchResultItemComponent.swift | 402 ++++ .../Sources/ChatbotSetupScreen.swift | 679 +++++- .../Sources/PeerNameColorItem.swift | 135 +- .../Sources/ChannelAppearanceScreen.swift | 24 +- .../Sources/PeerNameColorScreen.swift | 6 +- .../QuickReplyNameAlertController/BUILD | 28 + .../QuickReplyNameAlertController.swift | 543 +++++ .../ThemeAccentColorControllerNode.swift | 1 + .../Settings/TimezoneSelectionScreen/BUILD | 32 + .../Sources/TimezoneSelectionScreen.swift | 155 ++ .../Sources/TimezoneSelectionScreenNode.swift | 550 +++++ .../Sources/ShareWithPeersScreen.swift | 30 +- .../Components/SliderComponent/BUILD | 22 + .../Sources/SliderComponent.swift | 152 ++ .../AvatarStoryIndicatorComponent.swift | 9 +- .../Stories/PeerListItemComponent/BUILD | 2 + .../Sources/PeerListItemComponent.swift | 179 +- .../Sources/StoryAuthorInfoComponent.swift | 3 + .../Sources/StoryChatContent.swift | 6 +- .../Sources/StoryItemContentComponent.swift | 2 +- .../StoryItemSetContainerComponent.swift | 3 +- ...StoryItemSetContainerViewSendMessage.swift | 1 + .../Sources/StoryPeerListComponent.swift | 2 +- .../Sources/TextFieldComponent.swift | 32 +- .../Components/TimeSelectionActionSheet/BUILD | 25 + .../Sources/TimeSelectionActionSheet.swift | 132 ++ .../Sources/TokenListTextField.swift | 6 +- .../Sources/VideoMessageCameraScreen.swift | 37 +- .../Filters/Chats.imageset/Contents.json | 12 + .../Filters/Chats.imageset/existing.pdf | Bin 0 -> 2451 bytes .../Filters/NewChats.imageset/Contents.json | 12 + .../Filters/NewChats.imageset/unread.pdf | Bin 0 -> 2015 bytes .../Attach Menu/Reply.imageset/Contents.json | 12 + .../Attach Menu/Reply.imageset/replies.pdf | Bin 0 -> 1237 bytes .../AwayShortcut.imageset/Contents.json | 12 + .../AwayShortcut.imageset/awaydemo.pdf | Bin 0 -> 2756 bytes .../GreetingShortcut.imageset/Contents.json | 12 + .../greetingdemo.pdf | Bin 0 -> 3042 bytes .../QuickReplies.imageset/Contents.json | 12 + .../quickrepliesdemo.pdf | Bin 0 -> 3460 bytes .../AddTimeIcon.imageset/Contents.json | 12 + .../AddTimeIcon.imageset/addclock_30.pdf | Bin 0 -> 2808 bytes .../DownArrow.imageset/Contents.json | 12 + .../DownArrow.imageset/arrow (1).pdf} | Bin 2635 -> 2584 bytes .../Media Editor/Apply.imageset/Contents.json | 2 +- .../Apply.imageset/arrowhead_30.pdf | Bin 0 -> 1479 bytes .../Cutout.imageset/Contents.json | 12 + .../Media Editor/Cutout.imageset/magic_30.pdf | Bin 0 -> 3780 bytes .../CutoutUndo.imageset/Contents.json | 12 + .../CutoutUndo.imageset/undo2_30.pdf | Bin 0 -> 2314 bytes .../Business/Away.imageset/Contents.json | 12 + .../Business/Away.imageset/sleep_30.pdf | Bin 0 -> 6376 bytes .../Business/Chatbots.imageset/Contents.json | 12 + .../Business/Chatbots.imageset/bot_30.pdf | Bin 0 -> 8380 bytes .../Premium/Business/Contents.json | 9 + .../Business/Greetings.imageset/Contents.json | 12 + .../Greetings.imageset/greeting_30.pdf | Bin 0 -> 3542 bytes .../Business/Hours.imageset/Contents.json | 12 + .../Business/Hours.imageset/clock_30.pdf | Bin 0 -> 1766 bytes .../Business/Location.imageset/Contents.json | 12 + .../Location.imageset/location_30.pdf | Bin 0 -> 2987 bytes .../Business/Replies.imageset/Contents.json | 12 + .../Replies.imageset/arrowshape_30.pdf | Bin 0 -> 3326 bytes .../BusinessPerk/Away.imageset/Contents.json | 12 + .../BusinessPerk/Away.imageset/bubble_30.pdf | Bin 0 -> 4561 bytes .../Chatbots.imageset/Contents.json | 12 + .../Chatbots.imageset/bots_30.pdf | Bin 0 -> 4886 bytes .../Premium/BusinessPerk/Contents.json | 9 + .../Greetings.imageset/Contents.json | 12 + .../Greetings.imageset/hand_30.pdf | Bin 0 -> 4059 bytes .../BusinessPerk/Hours.imageset/Contents.json | 12 + .../BusinessPerk/Hours.imageset/clock_30.pdf | Bin 0 -> 3302 bytes .../Location.imageset/Contents.json | 12 + .../Location.imageset/location_30.pdf | Bin 0 -> 3164 bytes .../Replies.imageset/Contents.json | 12 + .../Replies.imageset/stories_30.pdf | Bin 0 -> 3118 bytes .../Status.imageset/Contents.json | 12 + .../BusinessPerk/Status.imageset/work_30.pdf | Bin 0 -> 3904 bytes .../BusinessPerk/Tag.imageset/Contents.json | 12 + .../BusinessPerk/Tag.imageset/tag_30 (3).pdf | Bin 0 -> 3332 bytes .../Perk/Business.imageset/Contents.json | 12 + .../Perk/Business.imageset/business.pdf | Bin 0 -> 4649 bytes .../Menu/Business.imageset/Contents.json | 12 + .../Menu/Business.imageset/business_30.pdf | Bin 0 -> 4653 bytes .../Animations/BusinessHoursEmoji.tgs | Bin 0 -> 33321 bytes .../Resources/Animations/HandWaveEmoji.tgs | Bin 0 -> 11908 bytes .../Resources/Animations/MapEmoji.tgs | Bin 0 -> 64974 bytes .../Resources/Animations/WriteEmoji.tgs | Bin 0 -> 64148 bytes .../Resources/Animations/ZzzEmoji.tgs | Bin 0 -> 38806 bytes .../TelegramUI/Sources/AccountContext.swift | 8 +- .../TelegramUI/Sources/AppDelegate.swift | 44 +- .../ChatControllerNavigateToMessage.swift | 6 +- ...ChatControllerOpenMessageContextMenu.swift | 2 +- .../TelegramUI/Sources/ChatController.swift | 394 +++- .../Sources/ChatControllerEditChat.swift | 64 + .../Sources/ChatControllerNode.swift | 58 +- .../ChatControllerOpenAttachmentMenu.swift | 484 +++-- .../ChatControllerOpenCalendarSearch.swift | 4 +- .../ChatControllerOpenMessageShareMenu.swift | 2 +- .../Sources/ChatControllerUpdateSearch.swift | 4 +- .../TelegramUI/Sources/ChatEmptyNode.swift | 100 +- .../Sources/ChatHistoryEntriesForView.swift | 58 + .../Sources/ChatHistoryListNode.swift | 166 +- .../Sources/ChatHistoryViewForLocation.swift | 2 +- .../ChatInterfaceInputContextPanels.swift | 12 +- .../ChatInterfaceStateAccessoryPanels.swift | 19 +- .../ChatInterfaceStateContextMenus.swift | 96 +- .../ChatInterfaceStateContextQueries.swift | 146 +- .../ChatInterfaceStateInputPanels.swift | 20 +- .../ChatInterfaceStateNavigationButtons.swift | 34 +- ...essageContextControllerContentSource.swift | 7 +- ...hatMessageThrottledProcessingManager.swift | 24 +- .../ChatRestrictedInputPanelNode.swift | 12 +- .../ChatSearchNavigationContentNode.swift | 4 +- .../ChatSearchResultsContollerNode.swift | 1 + .../Sources/ChatTextInputPanelNode.swift | 72 +- .../CommandChatInputContextPanelNode.swift | 358 +++- .../ContactMultiselectionControllerNode.swift | 45 +- .../Sources/EditAccessoryPanelNode.swift | 4 +- .../Sources/Nicegram/NGDeeplinkHandler.swift | 51 +- .../TelegramUI/Sources/OpenResolvedUrl.swift | 48 +- .../OverlayAudioPlayerControllerNode.swift | 1 + .../Sources/PeerMessagesMediaPlaylist.swift | 4 +- .../TelegramUI/Sources/PrefetchManager.swift | 6 +- .../Sources/SharedAccountContext.swift | 51 +- .../Sources/TelegramRootController.swift | 16 + ...textResultsChatInputContextPanelNode.swift | 2 +- ...ntextResultsChatInputPanelButtonItem.swift | 50 +- .../Sources/GroupCallContext.swift | 2 +- .../macOS/OngoingCallVideoCapturer.swift | 21 +- submodules/TgVoipWebrtc/tgcalls | 2 +- .../TranslateUI/Sources/TranslateScreen.swift | 1 + submodules/WebUI/Sources/WebAppWebView.swift | 2 +- swift_deps.bzl | 16 +- swift_deps_index.json | 127 +- versions.json | 2 +- 436 files changed, 27691 insertions(+), 5182 deletions(-) create mode 100644 Nicegram/NGStats/Sources/ChatsSharing.swift delete mode 100644 Nicegram/NGStats/Sources/DTO.swift delete mode 100644 Nicegram/NGStats/Sources/MetaStorage.swift delete mode 100644 Nicegram/NGStats/Sources/NGStats.swift create mode 100644 Nicegram/NGStats/Sources/StickersSharing.swift delete mode 100644 Nicegram/NGStats/Sources/Throttling.swift delete mode 100644 submodules/DrawingUI/Sources/ImageObjectSeparation.swift create mode 100644 submodules/PremiumUI/Resources/badge delete mode 100644 submodules/PremiumUI/Resources/badge.scn create mode 100644 submodules/PremiumUI/Resources/boost delete mode 100644 submodules/PremiumUI/Resources/boost.scn create mode 100644 submodules/PremiumUI/Resources/business.png create mode 100644 submodules/PremiumUI/Resources/business.scn create mode 100644 submodules/PremiumUI/Resources/coin create mode 100644 submodules/PremiumUI/Resources/coin_anim.png create mode 100644 submodules/PremiumUI/Resources/coin_edge.png create mode 100644 submodules/PremiumUI/Resources/darkerTexture.jpg create mode 100644 submodules/PremiumUI/Resources/diagonal_shine.png create mode 100644 submodules/PremiumUI/Resources/emoji delete mode 100644 submodules/PremiumUI/Resources/emoji.scn create mode 100644 submodules/PremiumUI/Resources/gift delete mode 100644 submodules/PremiumUI/Resources/gift.scn create mode 100644 submodules/PremiumUI/Resources/lighterTexture.jpg create mode 100644 submodules/PremiumUI/Resources/lightspeed delete mode 100644 submodules/PremiumUI/Resources/lightspeed.scn create mode 100644 submodules/PremiumUI/Resources/swirl delete mode 100644 submodules/PremiumUI/Resources/swirl.scn create mode 100644 submodules/PremiumUI/Resources/tag delete mode 100644 submodules/PremiumUI/Resources/tag.scn create mode 100644 submodules/PremiumUI/Sources/BadgeBusinessView.swift create mode 100644 submodules/PremiumUI/Sources/BusinessPageComponent.swift create mode 100644 submodules/PremiumUI/Sources/PremiumCoinComponent.swift create mode 100644 submodules/PremiumUI/Sources/PremiumOptionComponent.swift create mode 100644 submodules/TelegramCore/Sources/SyncCore/QuickReplyMessageAttribute.swift create mode 100644 submodules/TelegramCore/Sources/TelegramEngine/Messages/QuickReplyMessages.swift create mode 100644 submodules/TelegramCore/Sources/TelegramEngine/Messages/TimeZones.swift create mode 100644 submodules/TelegramUI/Components/ListItemSliderSelectorComponent/BUILD create mode 100644 submodules/TelegramUI/Components/ListItemSliderSelectorComponent/Sources/ListItemSliderSelectorComponent.swift create mode 100644 submodules/TelegramUI/Components/ListItemSwipeOptionContainer/BUILD create mode 100644 submodules/TelegramUI/Components/ListItemSwipeOptionContainer/Sources/ListItemSwipeOptionContainer.swift create mode 100644 submodules/TelegramUI/Components/ListMultilineTextFieldItemComponent/BUILD create mode 100644 submodules/TelegramUI/Components/ListMultilineTextFieldItemComponent/Sources/ListMultilineTextFieldItemComponent.swift create mode 100644 submodules/TelegramUI/Components/MediaEditor/Sources/ImageObjectSeparation.swift create mode 100644 submodules/TelegramUI/Components/MediaEditorScreen/Sources/MediaCutoutScreen.swift create mode 100644 submodules/TelegramUI/Components/PeerInfo/PeerInfoScreen/Sources/ListItems/PeerInfoScreenBusinessHoursItem.swift create mode 100644 submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/BUILD create mode 100644 submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/AutomaticBusinessMessageListItemComponent.swift create mode 100644 submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/AutomaticBusinessMessageSetupChatContents.swift create mode 100644 submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/AutomaticBusinessMessageSetupScreen.swift create mode 100644 submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/BottomPanelComponent.swift create mode 100644 submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/QuickReplyEmptyStateComponent.swift create mode 100644 submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/QuickReplySetupScreen.swift create mode 100644 submodules/TelegramUI/Components/Settings/BusinessHoursSetupScreen/BUILD create mode 100644 submodules/TelegramUI/Components/Settings/BusinessHoursSetupScreen/Sources/BusinessDaySetupScreen.swift create mode 100644 submodules/TelegramUI/Components/Settings/BusinessHoursSetupScreen/Sources/BusinessHoursSetupScreen.swift rename submodules/TelegramUI/Components/Settings/{BusinessSetupScreen => BusinessLocationSetupScreen}/BUILD (66%) create mode 100644 submodules/TelegramUI/Components/Settings/BusinessLocationSetupScreen/Sources/BusinessLocationSetupScreen.swift create mode 100644 submodules/TelegramUI/Components/Settings/BusinessLocationSetupScreen/Sources/MapPreviewComponent.swift delete mode 100644 submodules/TelegramUI/Components/Settings/BusinessSetupScreen/Sources/BusinessSetupScreen.swift create mode 100644 submodules/TelegramUI/Components/Settings/ChatbotSetupScreen/Sources/ChatbotSearchResultItemComponent.swift create mode 100644 submodules/TelegramUI/Components/Settings/QuickReplyNameAlertController/BUILD create mode 100644 submodules/TelegramUI/Components/Settings/QuickReplyNameAlertController/Sources/QuickReplyNameAlertController.swift create mode 100644 submodules/TelegramUI/Components/Settings/TimezoneSelectionScreen/BUILD create mode 100644 submodules/TelegramUI/Components/Settings/TimezoneSelectionScreen/Sources/TimezoneSelectionScreen.swift create mode 100644 submodules/TelegramUI/Components/Settings/TimezoneSelectionScreen/Sources/TimezoneSelectionScreenNode.swift create mode 100644 submodules/TelegramUI/Components/SliderComponent/BUILD create mode 100644 submodules/TelegramUI/Components/SliderComponent/Sources/SliderComponent.swift create mode 100644 submodules/TelegramUI/Components/TimeSelectionActionSheet/BUILD create mode 100644 submodules/TelegramUI/Components/TimeSelectionActionSheet/Sources/TimeSelectionActionSheet.swift create mode 100644 submodules/TelegramUI/Images.xcassets/Chat List/Filters/Chats.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Chat List/Filters/Chats.imageset/existing.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Chat List/Filters/NewChats.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Chat List/Filters/NewChats.imageset/unread.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Chat/Attach Menu/Reply.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Chat/Attach Menu/Reply.imageset/replies.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Chat/Empty Chat/AwayShortcut.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Chat/Empty Chat/AwayShortcut.imageset/awaydemo.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Chat/Empty Chat/GreetingShortcut.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Chat/Empty Chat/GreetingShortcut.imageset/greetingdemo.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Chat/Empty Chat/QuickReplies.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Chat/Empty Chat/QuickReplies.imageset/quickrepliesdemo.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Item List/AddTimeIcon.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Item List/AddTimeIcon.imageset/addclock_30.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Item List/DownArrow.imageset/Contents.json rename submodules/TelegramUI/Images.xcassets/{Media Editor/Apply.imageset/check.pdf => Item List/DownArrow.imageset/arrow (1).pdf} (53%) create mode 100644 submodules/TelegramUI/Images.xcassets/Media Editor/Apply.imageset/arrowhead_30.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Media Editor/Cutout.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Media Editor/Cutout.imageset/magic_30.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Media Editor/CutoutUndo.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Media Editor/CutoutUndo.imageset/undo2_30.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/Business/Away.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/Business/Away.imageset/sleep_30.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/Business/Chatbots.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/Business/Chatbots.imageset/bot_30.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/Business/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/Business/Greetings.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/Business/Greetings.imageset/greeting_30.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/Business/Hours.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/Business/Hours.imageset/clock_30.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/Business/Location.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/Business/Location.imageset/location_30.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/Business/Replies.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/Business/Replies.imageset/arrowshape_30.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Away.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Away.imageset/bubble_30.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Chatbots.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Chatbots.imageset/bots_30.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Greetings.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Greetings.imageset/hand_30.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Hours.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Hours.imageset/clock_30.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Location.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Location.imageset/location_30.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Replies.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Replies.imageset/stories_30.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Status.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Status.imageset/work_30.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Tag.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Tag.imageset/tag_30 (3).pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/Perk/Business.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Premium/Perk/Business.imageset/business.pdf create mode 100644 submodules/TelegramUI/Images.xcassets/Settings/Menu/Business.imageset/Contents.json create mode 100644 submodules/TelegramUI/Images.xcassets/Settings/Menu/Business.imageset/business_30.pdf create mode 100644 submodules/TelegramUI/Resources/Animations/BusinessHoursEmoji.tgs create mode 100644 submodules/TelegramUI/Resources/Animations/HandWaveEmoji.tgs create mode 100644 submodules/TelegramUI/Resources/Animations/MapEmoji.tgs create mode 100644 submodules/TelegramUI/Resources/Animations/WriteEmoji.tgs create mode 100644 submodules/TelegramUI/Resources/Animations/ZzzEmoji.tgs create mode 100644 submodules/TelegramUI/Sources/ChatControllerEditChat.swift diff --git a/Nicegram/NGLottie/Sources/LottieViewImpl.swift b/Nicegram/NGLottie/Sources/LottieViewImpl.swift index 2d0428e9f57..ed8fc94457c 100644 --- a/Nicegram/NGLottie/Sources/LottieViewImpl.swift +++ b/Nicegram/NGLottie/Sources/LottieViewImpl.swift @@ -40,6 +40,19 @@ extension LottieViewImpl: LottieViewProtocol { ) } + public func setImageProvider(_ imageProvider: ImageProvider?) { + if let imageProvider { + animationView.imageProvider = AnonymousImageProvider( + provider: imageProvider + ) + } else { + animationView.imageProvider = BundleImageProvider( + bundle: .main, + searchPath: nil + ) + } + } + public func setLoopMode(_ loopMode: LoopMode) { let mode: LottieLoopMode switch loopMode { @@ -62,3 +75,17 @@ extension LottieViewImpl: LottieViewProtocol { animationView.stop() } } + +private class AnonymousImageProvider { + let provider: LottieViewProtocol.ImageProvider + + init(provider: LottieViewProtocol.ImageProvider) { + self.provider = provider + } +} + +extension AnonymousImageProvider: AnimationImageProvider { + func imageForAsset(asset: ImageAsset) -> CGImage? { + provider.image(asset.id) + } +} diff --git a/Nicegram/NGStats/BUILD b/Nicegram/NGStats/BUILD index 5cca9fcaa4f..cece22c3d14 100644 --- a/Nicegram/NGStats/BUILD +++ b/Nicegram/NGStats/BUILD @@ -7,10 +7,8 @@ swift_library( "Sources/**/*.swift", ]), deps = [ - "@swiftpkg_nicegram_assistant_ios//:NGApi", + "@swiftpkg_nicegram_assistant_ios//:FeatNicegramHub", "//submodules/AccountContext:AccountContext", - "//Nicegram/NGData:NGData", - "//Nicegram/NGRemoteConfig:NGRemoteConfig", "//Nicegram/NGUtils:NGUtils", ], visibility = [ diff --git a/Nicegram/NGStats/Sources/ChatsSharing.swift b/Nicegram/NGStats/Sources/ChatsSharing.swift new file mode 100644 index 00000000000..74898560026 --- /dev/null +++ b/Nicegram/NGStats/Sources/ChatsSharing.swift @@ -0,0 +1,278 @@ +import AccountContext +import FeatNicegramHub +import Foundation +import NGUtils +import Postbox +import SwiftSignalKit +import TelegramCore + +@available(iOS 13.0, *) +public func sharePeerData( + peerId: PeerId, + context: AccountContext +) { + _ = (context.account.viewTracker.peerView(peerId, updateData: true) + |> take(1)) + .start(next: { peerView in + if let peer = peerView.peers[peerView.peerId] { + Task { + await sharePeerData( + peer: peer, + cachedData: peerView.cachedData, + context: context + ) + } + } + }) +} + +@available(iOS 13.0, *) +private func sharePeerData( + peer: Peer, + cachedData: CachedPeerData?, + context: AccountContext +) async { + let peerId = extractPeerId(peer: peer) + + let type: PeerType + switch EnginePeer(peer) { + case let .channel(channel): + switch channel.info { + case .group: + type = .group + case .broadcast: + type = .channel + } + case .legacyGroup: + type = .group + case .user(let user): + if user.botInfo != nil { + type = .bot + } else { + return + } + default: + return + } + + let sharePeerDataUseCase = NicegramHubContainer.shared.sharePeerDataUseCase() + + await sharePeerDataUseCase( + id: peerId, + participantsCount: extractParticipantsCount( + peer: peer, + cachedData: cachedData + ), + type: type, + peerDataProvider: { + await withCheckedContinuation { continuation in + let avatarImageSignal = fetchAvatarImage(peer: peer, context: context) + let inviteLinksSignal = context.engine.peers.direct_peerExportedInvitations(peerId: peer.id, revoked: false) + let interlocutorLanguageSignal = wrapped_detectInterlocutorLanguage(forChatWith: peer.id, context: context) + + _ = (combineLatest(avatarImageSignal, inviteLinksSignal, interlocutorLanguageSignal) + |> take(1)).start(next: { avatarImageData, inviteLinks, interlocutorLanguage in + let peerData = PeerData( + avatarImageData: avatarImageData?.base64EncodedString(), + id: peerId, + inviteLinks: extractInviteLinks(inviteLinks), + payload: extractPayload( + peer: peer, + cachedData: cachedData, + interlocutorLanguage: interlocutorLanguage + ), + type: type + ) + continuation.resume(returning: peerData) + }) + } + } + ) +} + +private func extractPayload( + peer: Peer, + cachedData: CachedPeerData?, + interlocutorLanguage: String? +) -> PeerPayload? { + switch EnginePeer(peer) { + case let .legacyGroup(group): + PeerPayload.legacyGroup( + extractPayload( + group: group, + cachedData: cachedData as? CachedGroupData, + lastMessageLanguageCode: interlocutorLanguage + ) + ) + case let .channel(channel): + PeerPayload.channel( + extractPayload( + channel: channel, + cachedData: cachedData as? CachedChannelData, + lastMessageLanguageCode: interlocutorLanguage + ) + ) + case let .user(user): + if let botInfo = user.botInfo { + PeerPayload.user( + extractPayload( + bot: user, + botInfo: botInfo, + lastMessageLanguageCode: interlocutorLanguage + ) + ) + } else { + nil + } + default: + nil + } +} + +private func extractPayload(group: TelegramGroup, cachedData: CachedGroupData?, lastMessageLanguageCode: String?) -> LegacyGroupPayload { + LegacyGroupPayload( + deactivated: group.flags.contains(.deactivated), + title: group.title, + participantsCount: group.participantCount, + date: group.creationDate, + migratedTo: group.migrationReference?.peerId.id._internalGetInt64Value(), + photo: extractChatPhoto(peer: group), + lastMessageLang: lastMessageLanguageCode, + about: cachedData?.about + ) +} + +private func extractPayload(channel: TelegramChannel, cachedData: CachedChannelData?, lastMessageLanguageCode: String?) -> ChannelPayload { + ChannelPayload( + verified: channel.isVerified, + scam: channel.isScam, + hasGeo: channel.flags.contains(.hasGeo), + fake: channel.isFake, + gigagroup: channel.flags.contains(.isGigagroup), + title: channel.title, + username: channel.username, + date: channel.creationDate, + restrictions: extractRestrictions(restrictionInfo: channel.restrictionInfo), + participantsCount: cachedData?.participantsSummary.memberCount, + photo: extractChatPhoto(peer: channel), + lastMessageLang: lastMessageLanguageCode, + about: cachedData?.about, + geoLocation: cachedData?.peerGeoLocation.map { + extractGeoLocation($0) + } + ) +} + +private func extractPayload( + bot: TelegramUser, + botInfo: BotUserInfo, + lastMessageLanguageCode: String? +) -> UserPayload { + let botFlags = botInfo.flags + let userFlags = bot.flags + + return UserPayload( + deleted: bot.isDeleted, + bot: true, + botChatHistory: botFlags.contains(.hasAccessToChatHistory), + botNochats: !botFlags.contains(.worksWithGroups), + verified: bot.isVerified, + restricted: bot.restrictionInfo != nil, + botInlineGeo: botFlags.contains(.requiresGeolocationForInlineRequests), + support: userFlags.contains(.isSupport), + scam: bot.isScam, + fake: bot.isFake, + botAttachMenu: botFlags.contains(.canBeAddedToAttachMenu), + premium: bot.isPremium, + botCanEdit: botFlags.contains(.canEdit), + firstName: bot.firstName, + lastName: bot.lastName, + username: bot.username, + phone: bot.phone, + photo: extractUserProfilePhoto(peer: bot), + restrictionReason: extractRestrictions(restrictionInfo: bot.restrictionInfo), + botInlinePlaceholder: botInfo.inlinePlaceholder, + langCode: lastMessageLanguageCode, + usernames: bot.usernames.map { + .init( + editable: $0.flags.contains(.isEditable), + active: $0.isActive, + username: $0.username + ) + } + ) +} + +private func extractChatPhoto( + peer: Peer +) -> ChatPhoto? { + guard let imageRepresentation = peer.profileImageRepresentations.first else { + return nil + } + guard let resource = imageRepresentation.resource as? CloudPeerPhotoSizeMediaResource else { + return nil + } + + return ChatPhoto(mediaResourceId: resource.id.stringRepresentation, datacenterId: resource.datacenterId, photoId: resource.photoId, volumeId: resource.volumeId, localId: resource.localId, sizeSpec: resource.sizeSpec.rawValue) +} + +private func extractUserProfilePhoto( + peer: Peer +) -> UserProfilePhoto? { + guard let imageRepresentation = peer.profileImageRepresentations.first else { + return nil + } + guard let resource = imageRepresentation.resource as? CloudPeerPhotoSizeMediaResource else { + return nil + } + + return UserProfilePhoto( + hasVideo: imageRepresentation.hasVideo, + personal: imageRepresentation.isPersonal, + photoId: resource.photoId, + dcId: resource.datacenterId + ) +} + +func extractParticipantsCount(peer: Peer, cachedData: CachedPeerData?) -> Int { + switch EnginePeer(peer) { + case .user: + return 2 + case .channel: + let channelData = cachedData as? CachedChannelData + return Int(channelData?.participantsSummary.memberCount ?? 0) + case let .legacyGroup(group): + return group.participantCount + case .secretChat: + return 0 + } +} + +private func extractRestrictions( + restrictionInfo: PeerAccessRestrictionInfo? +) -> [FeatNicegramHub.RestrictionRule] { + restrictionInfo?.rules.map { + FeatNicegramHub.RestrictionRule( + platform: $0.platform, + reason: $0.reason, + text: $0.text + ) + } ?? [] +} + +private func extractGeoLocation( + _ geo: PeerGeoLocation +) -> GeoLocation { + GeoLocation(latitude: geo.latitude, longitude: geo.longitude, address: geo.address) +} + +private func extractInviteLinks(_ links: ExportedInvitations?) -> [InviteLink]? { + links?.list?.compactMap { link in + switch link { + case let .link(link, title, isPermanent, requestApproval, isRevoked, adminId, date, startDate, expireDate, usageLimit, count, requestedCount): + InviteLink(link: link, title: title, isPermanent: isPermanent, requestApproval: requestApproval, isRevoked: isRevoked, adminId: adminId.id._internalGetInt64Value(), date: date, startDate: startDate, expireDate: expireDate, usageLimit: usageLimit, count: count, requestedCount: requestedCount) + case .publicJoinRequest: + nil + } + } +} diff --git a/Nicegram/NGStats/Sources/DTO.swift b/Nicegram/NGStats/Sources/DTO.swift deleted file mode 100644 index ff68c394315..00000000000 --- a/Nicegram/NGStats/Sources/DTO.swift +++ /dev/null @@ -1,137 +0,0 @@ -import TelegramCore -import Foundation - -struct ProfileInfoBody: Encodable { - let id: Int64 - let type: ProfileTypeDTO - let inviteLinks: [InviteLinkDTO] - let icon: String? - @Stringified var payload: AnyProfilePayload -} - -protocol ProfilePayload: Encodable {} -struct AnyProfilePayload: ProfilePayload { - let wrapped: ProfilePayload - - func encode(to encoder: Encoder) throws { - try wrapped.encode(to: encoder) - } -} - -struct GroupPayload: ProfilePayload { - let deactivated: Bool - let title: String - let participantsCount: Int - let date: Int32 - let migratedTo: Int64? - let photo: ProfileImageDTO? - let lastMessageLang: String? - let about: String? -} - -struct ChannelPayload: ProfilePayload { - let verified: Bool - let scam: Bool - let hasGeo: Bool - let fake: Bool - let gigagroup: Bool - let title: String - let username: String? - let date: Int32 - let restrictions: [RestrictionRuleDTO] - let participantsCount: Int32? - let photo: ProfileImageDTO? - let lastMessageLang: String? - let about: String? - let geoLocation: GeoLocationDTO? -} - -enum ProfileTypeDTO: String, Encodable { - case channel - case group -} - -struct ProfileImageDTO: Encodable { - let mediaResourceId: String - let datacenterId: Int - let photoId: Int64? - let volumeId: Int64? - let localId: Int32? - let sizeSpec: Int32 -} - -struct RestrictionRuleDTO: Encodable { - let platform: String - let reason: String - let text: String -} - -struct GeoLocationDTO: Encodable { - let latitude: Double - let longitude: Double - let address: String -} - -struct InviteLinkDTO: Encodable { - let link: String - let title: String? - let isPermanent: Bool - let requestApproval: Bool - let isRevoked: Bool - let adminId: Int64 - let date: Int32 - let startDate: Int32? - let expireDate: Int32? - let usageLimit: Int32? - let count: Int32? - let requestedCount: Int32? -} - -@propertyWrapper -struct Stringified: Encodable { - - var wrappedValue: T - - init(wrappedValue: T) { - self.wrappedValue = wrappedValue - } - - func encode(to encoder: Encoder) throws { - var container = encoder.singleValueContainer() - - let data = try JSONEncoder().encode(wrappedValue) - let string = String(data: data, encoding: .utf8) - try container.encode(string) - } -} - -// MARK: - Mapping - -extension ProfileImageDTO { - init(cloudPeerPhotoSizeMediaResource resource: CloudPeerPhotoSizeMediaResource) { - self.init(mediaResourceId: resource.id.stringRepresentation, datacenterId: resource.datacenterId, photoId: resource.photoId, volumeId: resource.volumeId, localId: resource.localId, sizeSpec: resource.sizeSpec.rawValue) - } -} - -extension RestrictionRuleDTO { - init(restrictionRule rule: RestrictionRule) { - self.init(platform: rule.platform, reason: rule.reason, text: rule.text) - } -} - -extension GeoLocationDTO { - init(peerGeoLocation geo: PeerGeoLocation) { - self.init(latitude: geo.latitude, longitude: geo.longitude, address: geo.address) - } -} - -extension InviteLinkDTO { - init?(exportedInvitation: ExportedInvitation) { - switch exportedInvitation { - case let .link(link, title, isPermanent, requestApproval, isRevoked, adminId, date, startDate, expireDate, usageLimit, count, requestedCount): - self.init(link: link, title: title, isPermanent: isPermanent, requestApproval: requestApproval, isRevoked: isRevoked, adminId: adminId.id._internalGetInt64Value(), date: date, startDate: startDate, expireDate: expireDate, usageLimit: usageLimit, count: count, requestedCount: requestedCount) - case .publicJoinRequest: - return nil - } - } -} diff --git a/Nicegram/NGStats/Sources/MetaStorage.swift b/Nicegram/NGStats/Sources/MetaStorage.swift deleted file mode 100644 index e720f5df01e..00000000000 --- a/Nicegram/NGStats/Sources/MetaStorage.swift +++ /dev/null @@ -1,41 +0,0 @@ -import NGCore -import Foundation - -struct ChatStatsMeta: Codable { - let id: Int64 - let sharedAt: Date -} - -class ChatStatsMetaStorage { - - // MARK: - Dependencies - - private let storage = FileStorage<[Int64: ChatStatsMeta]>(path: "chat-stats-meta") - - // MARK: - Lifecycle - - init() {} - - // MARK: - Public Functions - - func getSharedAt(peerId: Int64) -> Date? { - var dict = storage.read() ?? [:] - return dict[peerId]?.sharedAt - } - - func setSharedAt(_ sharedAt: Date, peerId: Int64) { - var dict = storage.read() ?? [:] - dict[peerId] = ChatStatsMeta(id: peerId, sharedAt: sharedAt) - storage.save(dict) - } - - public func removeItems(whereSharedAt: (Date) -> Bool) { - var dict = storage.read() ?? [:] - for (id, meta) in dict { - if whereSharedAt(meta.sharedAt) { - dict.removeValue(forKey: id) - } - } - storage.save(dict) - } -} diff --git a/Nicegram/NGStats/Sources/NGStats.swift b/Nicegram/NGStats/Sources/NGStats.swift deleted file mode 100644 index 3f5b169717e..00000000000 --- a/Nicegram/NGStats/Sources/NGStats.swift +++ /dev/null @@ -1,182 +0,0 @@ -import AccountContext -import NGApi -import NGData -import NGEnv -import NGUtils -import Postbox -import SwiftSignalKit -import TelegramCore -import Foundation - -private let thresholdGroupMemebrsCount: Int = 1000 - -private let apiClient = EsimApiClient( - baseUrl: URL(string: NGENV.esim_api_url)!, - apiKey: NGENV.esim_api_key, - mobileIdentifier: "" -) -private let throttlingService = ChatStatsThrottlingService() - -public func isShareChannelsInfoEnabled() -> Bool { - return NGSettings.shareChannelsInfo -} - -public func setShareChannelsInfo(enabled: Bool) { - NGSettings.shareChannelsInfo = enabled -} - -public func shareChannelInfo(peerId: PeerId, context: AccountContext) { - if !isShareChannelsInfoEnabled() { - return - } - - if throttlingService.shouldSkipShare(peerId: peerId) { - return - } - - _ = (context.account.viewTracker.peerView(peerId, updateData: true) - |> take(1)) - .start(next: { peerView in - if let peer = peerView.peers[peerView.peerId] { - shareChannelInfo(peer: peer, cachedData: peerView.cachedData, context: context) - } - }) -} - -private func shareChannelInfo(peer: Peer, cachedData: CachedPeerData?, context: AccountContext) { - if isGroup(peer: peer) { - let participantsCount = extractParticipantsCount(peer: peer, cachedData: cachedData) - if participantsCount < thresholdGroupMemebrsCount { - return - } - } else if !isChannel(peer: peer) { - return - } - - let avatarImageSignal = fetchAvatarImage(peer: peer, context: context) - let inviteLinksSignal = context.engine.peers.direct_peerExportedInvitations(peerId: peer.id, revoked: false) - let interlocutorLanguageSignal = wrapped_detectInterlocutorLanguage(forChatWith: peer.id, context: context) - - _ = (combineLatest(avatarImageSignal, inviteLinksSignal, interlocutorLanguageSignal) - |> take(1)).start(next: { avatarImageData, inviteLinks, interlocutorLanguage in - shareChannelInfo(peer: peer, cachedData: cachedData, avatarImageData: avatarImageData, inviteLinks: inviteLinks, interlocutorLanguage: interlocutorLanguage) - }) -} - -private func shareChannelInfo(peer: Peer, cachedData: CachedPeerData?, avatarImageData: Data?, inviteLinks: ExportedInvitations?, interlocutorLanguage: String?) { - let id = extractPeerId(peer: peer) - - let type: ProfileTypeDTO - let payload: ProfilePayload - switch EnginePeer(peer) { - case let .legacyGroup(group): - type = .group - payload = extractPayload( - group: group, - cachedData: cachedData as? CachedGroupData, - lastMessageLanguageCode: interlocutorLanguage - ) - case let .channel(channel): - switch channel.info { - case .broadcast: - type = .channel - case .group: - type = .group - } - payload = extractPayload( - channel: channel, - cachedData: cachedData as? CachedChannelData, - lastMessageLanguageCode: interlocutorLanguage - ) - default: - return - } - - let inviteLinks = inviteLinks?.list?.compactMap({ InviteLinkDTO(exportedInvitation: $0) }) ?? [] - - let profileImageBase64 = avatarImageData?.base64EncodedString() - - let body = ProfileInfoBody(id: id, type: type, inviteLinks: inviteLinks, icon: profileImageBase64, payload: AnyProfilePayload(wrapped: payload)) - - throttlingService.markAsShared(peerId: peer.id) - apiClient.send(.post(path: "telegram/chat", body: body), completion: nil) -} - -private func extractPayload(group: TelegramGroup, cachedData: CachedGroupData?, lastMessageLanguageCode: String?) -> ProfilePayload { - let isDeactivated = group.flags.contains(.deactivated) - let title = group.title - let participantsCount = group.participantCount - let date = group.creationDate - let migratedTo = group.migrationReference?.peerId.id._internalGetInt64Value() - let photo = extractProfileImageDTO(peer: group) - let about = cachedData?.about - - return GroupPayload(deactivated: isDeactivated, title: title, participantsCount: participantsCount, date: date, migratedTo: migratedTo, photo: photo, lastMessageLang: lastMessageLanguageCode, about: about) -} - -private func extractPayload(channel: TelegramChannel, cachedData: CachedChannelData?, lastMessageLanguageCode: String?) -> ProfilePayload { - let isVerified = channel.isVerified - let isScam = channel.isScam - let hasGeo = channel.flags.contains(.hasGeo) - let isFake = channel.isFake - let isGigagroup = channel.flags.contains(.isGigagroup) - let title = channel.title - let username = channel.username - let date = channel.creationDate - let restrictions = channel.restrictionInfo?.rules.map({ RestrictionRuleDTO(restrictionRule: $0) }) ?? [] - let participantsCount = cachedData?.participantsSummary.memberCount - let photo = extractProfileImageDTO(peer: channel) - let about = cachedData?.about - let geoLocation = cachedData?.peerGeoLocation.flatMap({ GeoLocationDTO(peerGeoLocation: $0) }) - - return ChannelPayload(verified: isVerified, scam: isScam, hasGeo: hasGeo, fake: isFake, gigagroup: isGigagroup, title: title, username: username, date: date, restrictions: restrictions, participantsCount: participantsCount, photo: photo, lastMessageLang: lastMessageLanguageCode, about: about, geoLocation: geoLocation) -} - -private func extractProfileImageDTO(peer: Peer) -> ProfileImageDTO? { - guard let imageRepresentation = peer.profileImageRepresentations.first else { - return nil - } - guard let resource = imageRepresentation.resource as? CloudPeerPhotoSizeMediaResource else { - return nil - } - return ProfileImageDTO(cloudPeerPhotoSizeMediaResource: resource) -} - -private func isChannel(peer: Peer) -> Bool { - if case let .channel(channel) = EnginePeer(peer), - case .broadcast = channel.info { - return true - } else { - return false - } -} - -public func isGroup(peer: Peer) -> Bool { - switch EnginePeer(peer) { - case let .channel(channel): - switch channel.info { - case .group: - return true - case .broadcast: - return false - } - case .legacyGroup: - return true - default: - return false - } -} - -func extractParticipantsCount(peer: Peer, cachedData: CachedPeerData?) -> Int { - switch EnginePeer(peer) { - case .user: - return 2 - case .channel: - let channelData = cachedData as? CachedChannelData - return Int(channelData?.participantsSummary.memberCount ?? 0) - case let .legacyGroup(group): - return group.participantCount - case .secretChat: - return 0 - } -} diff --git a/Nicegram/NGStats/Sources/StickersSharing.swift b/Nicegram/NGStats/Sources/StickersSharing.swift new file mode 100644 index 00000000000..faafca6a954 --- /dev/null +++ b/Nicegram/NGStats/Sources/StickersSharing.swift @@ -0,0 +1,60 @@ +import AccountContext +import FeatNicegramHub +import TelegramCore + +@available(iOS 13.0, *) +public class StickersDataProviderImpl { + + // MARK: - Dependencies + + private let context: AccountContext + + // MARK: - Lifecycle + + public init(context: AccountContext) { + self.context = context + } +} + +@available(iOS 13.0, *) +extension StickersDataProviderImpl: StickersDataProvider { + public func getStickersData() async -> StickersData? { + await withCheckedContinuation { continuation in + _ = context.account.postbox.transaction { transaction in + transaction + .getItemCollectionsInfos( + namespace: Namespaces.ItemCollection.CloudStickerPacks + ).compactMap { + $0.1 as? StickerPackCollectionInfo + }.map { pack in + let resource = pack.thumbnail?.resource as? CloudStickerPackThumbnailMediaResource + + let flags = pack.flags + + return StickerSet( + archived: false, + channelEmojiStatus: flags.contains(.isAvailableAsChannelStatus), + official: flags.contains(.isOfficial), + masks: flags.contains(.isMasks), + animated: flags.contains(.isAnimated), + videos: flags.contains(.isVideo), + emojis: flags.contains(.isEmoji), + id: pack.id.id, + title: pack.title, + shortName: pack.shortName, + thumbDcId: resource?.datacenterId, + thumbVersion: resource?.thumbVersion, + thumbDocumentId: pack.thumbnailFileId, + count: pack.count + ) + } + }.start(next: { stickerSets in + continuation.resume( + returning: StickersData( + stickerSets: stickerSets + ) + ) + }) + } + } +} diff --git a/Nicegram/NGStats/Sources/Throttling.swift b/Nicegram/NGStats/Sources/Throttling.swift deleted file mode 100644 index 47db7c6d4e5..00000000000 --- a/Nicegram/NGStats/Sources/Throttling.swift +++ /dev/null @@ -1,53 +0,0 @@ -import Foundation -import NGRemoteConfig -import Postbox - -class ChatStatsThrottlingService { - - // MARK: - Dependencies - - private let remoteConfig = RemoteConfigServiceImpl.shared - private let metaStorage = ChatStatsMetaStorage() - - // MARK: - Lifecycle - - init() { - metaStorage.removeItems { !self.shouldSkipShare(sharedAt: $0) } - } - - // MARK: - Public Functions - - func shouldSkipShare(peerId: PeerId) -> Bool { - let peerId = peerId.id._internalGetInt64Value() - - let sharedAt = metaStorage.getSharedAt(peerId: peerId) - return shouldSkipShare(sharedAt: sharedAt) - } - - func markAsShared(peerId: PeerId) { - let peerId = peerId.id._internalGetInt64Value() - - metaStorage.setSharedAt(Date(), peerId: peerId) - } - - // MARK: - Private Functions - - private func getShareChannelsConfig() -> ShareChannelsConfig { - let remoteValue = RemoteConfigServiceImpl.shared.get(ShareChannelsConfig.self, byKey: "shareChannelsConfig") - let defaultValue = ShareChannelsConfig(throttlingInterval: 86400) - return remoteValue ?? defaultValue - } - - private func shouldSkipShare(sharedAt: Date?) -> Bool { - let sharedAt = sharedAt ?? .distantPast - let currentDate = Date() - let throttlingInterval = getShareChannelsConfig().throttlingInterval - - return (sharedAt.addingTimeInterval(throttlingInterval) > currentDate) - } - -} - -private struct ShareChannelsConfig: Decodable { - let throttlingInterval: TimeInterval -} diff --git a/Nicegram/NGUI/BUILD b/Nicegram/NGUI/BUILD index 47093e25d0d..7cbfa09d22b 100644 --- a/Nicegram/NGUI/BUILD +++ b/Nicegram/NGUI/BUILD @@ -42,6 +42,8 @@ swift_library( "//Nicegram/NGStealthMode:NGStealthMode", "@swiftpkg_nicegram_assistant_ios//:NGAiChatUI", "@swiftpkg_nicegram_assistant_ios//:FeatImagesHubUI", + "@swiftpkg_nicegram_assistant_ios//:FeatNicegramHub", + "@swiftpkg_nicegram_assistant_ios//:FeatPinnedChats", "@swiftpkg_nicegram_assistant_ios//:FeatSpeechToText", ], visibility = [ diff --git a/Nicegram/NGUI/Sources/NicegramSettingsController.swift b/Nicegram/NGUI/Sources/NicegramSettingsController.swift index 071928c9a7e..0f32b58e427 100644 --- a/Nicegram/NGUI/Sources/NicegramSettingsController.swift +++ b/Nicegram/NGUI/Sources/NicegramSettingsController.swift @@ -11,6 +11,7 @@ import AccountContext import Display import FeatImagesHubUI +import FeatNicegramHub import FeatPinnedChats import Foundation import ItemListUI @@ -70,7 +71,7 @@ private enum NicegramSettingsControllerSection: Int32 { case Account case Other case QuickReplies - case ShareChannelsInfo + case ShareData case PinnedChats } @@ -120,8 +121,10 @@ private enum NicegramSettingsControllerEntry: ItemListNodeEntry { case secretMenu(String) - case shareChannelsInfoToggle(String, Bool) - case shareChannelsInfoNote(String) + case shareBotsData(String, Bool) + case shareChannelsData(String, Bool) + case shareStickersData(String, Bool) + case shareDataNote(String) // MARK: Section @@ -143,8 +146,8 @@ private enum NicegramSettingsControllerEntry: ItemListNodeEntry { return NicegramSettingsControllerSection.Account.rawValue case .secretMenu: return NicegramSettingsControllerSection.SecretMenu.rawValue - case .shareChannelsInfoToggle, .shareChannelsInfoNote: - return NicegramSettingsControllerSection.ShareChannelsInfo.rawValue + case .shareBotsData, .shareChannelsData, .shareStickersData, .shareDataNote: + return NicegramSettingsControllerSection.ShareData.rawValue case .pinnedChatsHeader, .pinnedChat: return NicegramSettingsControllerSection.PinnedChats.rawValue } @@ -223,10 +226,14 @@ private enum NicegramSettingsControllerEntry: ItemListNodeEntry { case let .easyToggle(index, _, _, _): return 5000 + Int32(index) - case .shareChannelsInfoToggle: + case .shareBotsData: return 6000 - case .shareChannelsInfoNote: + case .shareChannelsData: return 6001 + case .shareStickersData: + return 6002 + case .shareDataNote: + return 6010 } } @@ -373,14 +380,26 @@ private enum NicegramSettingsControllerEntry: ItemListNodeEntry { } else { return false } - case let .shareChannelsInfoToggle(lhsText, lhsValue): - if case let .shareChannelsInfoToggle(rhsText, rhsValue) = rhs, lhsText == rhsText, lhsValue == rhsValue { + case let .shareBotsData(lhsText, lhsValue): + if case let .shareBotsData(rhsText, rhsValue) = rhs, lhsText == rhsText, lhsValue == rhsValue { return true } else { return false } - case let .shareChannelsInfoNote(lhsText): - if case let .shareChannelsInfoNote(rhsText) = rhs, lhsText == rhsText { + case let .shareChannelsData(lhsText, lhsValue): + if case let .shareChannelsData(rhsText, rhsValue) = rhs, lhsText == rhsText, lhsValue == rhsValue { + return true + } else { + return false + } + case let .shareStickersData(lhsText, lhsValue): + if case let .shareStickersData(rhsText, rhsValue) = rhs, lhsText == rhsText, lhsValue == rhsValue { + return true + } else { + return false + } + case let .shareDataNote(lhsText): + if case let .shareDataNote(rhsText) = rhs, lhsText == rhsText { return true } else { return false @@ -530,11 +549,43 @@ private enum NicegramSettingsControllerEntry: ItemListNodeEntry { return ItemListActionItem(presentationData: presentationData, title: text, kind: .neutral, alignment: .natural, sectionId: section, style: .blocks) { arguments.pushController(secretMenuController(context: arguments.context)) } - case let .shareChannelsInfoToggle(text, value): + case let .shareBotsData(text, value): + return ItemListSwitchItem(presentationData: presentationData, title: text, value: value, enabled: true, sectionId: section, style: .blocks, updated: { value in + if #available(iOS 13.0, *) { + Task { + let updateSharingSettingsUseCase = NicegramHubContainer.shared.updateSharingSettingsUseCase() + + await updateSharingSettingsUseCase { + $0.with(\.shareBotsData, value) + } + } + } + }) + case let .shareChannelsData(text, value): + return ItemListSwitchItem(presentationData: presentationData, title: text, value: value, enabled: true, sectionId: section, style: .blocks, updated: { value in + if #available(iOS 13.0, *) { + Task { + let updateSharingSettingsUseCase = NicegramHubContainer.shared.updateSharingSettingsUseCase() + + await updateSharingSettingsUseCase { + $0.with(\.shareChannelsData, value) + } + } + } + }) + case let .shareStickersData(text, value): return ItemListSwitchItem(presentationData: presentationData, title: text, value: value, enabled: true, sectionId: section, style: .blocks, updated: { value in - setShareChannelsInfo(enabled: value) + if #available(iOS 13.0, *) { + Task { + let updateSharingSettingsUseCase = NicegramHubContainer.shared.updateSharingSettingsUseCase() + + await updateSharingSettingsUseCase { + $0.with(\.shareStickersData, value) + } + } + } }) - case let .shareChannelsInfoNote(text): + case let .shareDataNote(text): return ItemListTextItem(presentationData: presentationData, text: .plain(text), sectionId: section) case .pinnedChatsHeader: return ItemListSectionHeaderItem( @@ -570,7 +621,7 @@ private enum NicegramSettingsControllerEntry: ItemListNodeEntry { // MARK: Entries list -private func nicegramSettingsControllerEntries(presentationData: PresentationData, experimentalSettings: ExperimentalUISettings, showCalls: Bool, pinnedChats: [PinnedChat], context: AccountContext) -> [NicegramSettingsControllerEntry] { +private func nicegramSettingsControllerEntries(presentationData: PresentationData, experimentalSettings: ExperimentalUISettings, showCalls: Bool, pinnedChats: [PinnedChat], sharingSettings: SharingSettings?, context: AccountContext) -> [NicegramSettingsControllerEntry] { var entries: [NicegramSettingsControllerEntry] = [] if canOpenSecretMenu(context: context) { @@ -666,8 +717,29 @@ private func nicegramSettingsControllerEntries(presentationData: PresentationDat entries.append(.easyToggle(toggleIndex, .hideStories, l("NicegramSettings.HideStories"), NGSettings.hideStories)) toggleIndex += 1 - entries.append(.shareChannelsInfoToggle(l("NicegramSettings.ShareChannelsInfoToggle"), isShareChannelsInfoEnabled())) - entries.append(.shareChannelsInfoNote(l("NicegramSettings.ShareChannelsInfoToggle.Note"))) + if let sharingSettings { + entries.append( + .shareBotsData( + l("NicegramSettings.ShareBotsToggle"), + sharingSettings.shareBotsData + ) + ) + entries.append( + .shareChannelsData( + l("NicegramSettings.ShareChannelsToggle"), + sharingSettings.shareChannelsData + ) + ) + entries.append( + .shareStickersData( + l("NicegramSettings.ShareStickersToggle"), + sharingSettings.shareStickersData + ) + ) + entries.append( + .shareDataNote(l("NicegramSettings.ShareData.Note")) + ) + } return entries } @@ -715,8 +787,19 @@ public func nicegramSettingsController(context: AccountContext, accountsContexts } else { pinnedChatsSignal = .single([]) } + + let sharingSettingsSignal: Signal + if #available(iOS 13.0, *) { + sharingSettingsSignal = NicegramHubContainer.shared.getSharingSettingsUseCase() + .publisher() + .map { Optional($0) } + .toSignal() + .skipError() + } else { + sharingSettingsSignal = .single(nil) + } - let signal = combineLatest(context.sharedContext.presentationData, sharedDataSignal, showCallsTab, pinnedChatsSignal) |> map { presentationData, sharedData, showCalls, pinnedChats -> (ItemListControllerState, (ItemListNodeState, Any)) in + let signal = combineLatest(context.sharedContext.presentationData, sharedDataSignal, showCallsTab, pinnedChatsSignal, sharingSettingsSignal) |> map { presentationData, sharedData, showCalls, pinnedChats, sharingSettings -> (ItemListControllerState, (ItemListNodeState, Any)) in let experimentalSettings: ExperimentalUISettings = sharedData.entries[ApplicationSpecificSharedDataKeys.experimentalUISettings]?.get(ExperimentalUISettings.self) ?? ExperimentalUISettings.defaultSettings @@ -727,7 +810,7 @@ public func nicegramSettingsController(context: AccountContext, accountsContexts }) } - let entries = nicegramSettingsControllerEntries(presentationData: presentationData, experimentalSettings: experimentalSettings, showCalls: showCalls, pinnedChats: pinnedChats, context: context) + let entries = nicegramSettingsControllerEntries(presentationData: presentationData, experimentalSettings: experimentalSettings, showCalls: showCalls, pinnedChats: pinnedChats, sharingSettings: sharingSettings, context: context) let controllerState = ItemListControllerState(presentationData: ItemListPresentationData(presentationData), title: .text(l("AppName")), leftNavigationButton: leftNavigationButton, rightNavigationButton: nil, backNavigationButton: ItemListBackButton(title: presentationData.strings.Common_Back)) let listState = ItemListNodeState(presentationData: ItemListPresentationData(presentationData), entries: entries, style: .blocks) diff --git a/Package.resolved b/Package.resolved index df872f4878b..a44a6d54870 100644 --- a/Package.resolved +++ b/Package.resolved @@ -14,8 +14,8 @@ "kind" : "remoteSourceControl", "location" : "https://github.com/scenee/FloatingPanel", "state" : { - "revision" : "5b33d3d5ff1f50f4a2d64158ccfe8c07b5a3e649", - "version" : "2.8.1" + "revision" : "8f2be39bf49b4d5e22bbf7bdde69d5b76d0ecd2a", + "version" : "2.8.2" } }, { @@ -36,13 +36,22 @@ "version" : "1.2.0" } }, + { + "identity" : "navigation-stack-backport", + "kind" : "remoteSourceControl", + "location" : "https://github.com/denis15yo/navigation-stack-backport.git", + "state" : { + "branch" : "main", + "revision" : "66716ce9c31198931c2275a0b69de2fdaa687e74" + } + }, { "identity" : "nicegram-assistant-ios", "kind" : "remoteSourceControl", "location" : "git@bitbucket.org:mobyrix/nicegram-assistant-ios.git", "state" : { "branch" : "develop", - "revision" : "123eb001dcc9477f9087e40ff1b0e7f012182d45" + "revision" : "9d2bc90674d3fa38fdd2e6f8f069c67d296d2bde" } }, { @@ -59,8 +68,8 @@ "kind" : "remoteSourceControl", "location" : "https://github.com/SDWebImage/SDWebImage.git", "state" : { - "revision" : "b11493f76481dff17ac8f45274a6b698ba0d3af5", - "version" : "5.18.11" + "revision" : "73b9397cfbd902f606572964055464903b1d84c6", + "version" : "5.19.0" } }, { diff --git a/Telegram/SiriIntents/IntentMessages.swift b/Telegram/SiriIntents/IntentMessages.swift index 935c3a1db4d..37360288e34 100644 --- a/Telegram/SiriIntents/IntentMessages.swift +++ b/Telegram/SiriIntents/IntentMessages.swift @@ -53,7 +53,7 @@ func unreadMessages(account: Account) -> Signal<[INMessage], NoError> { } if !isMuted && hasUnread { - signals.append(account.postbox.aroundMessageHistoryViewForLocation(.peer(peerId: index.messageIndex.id.peerId, threadId: nil), anchor: .upperBound, ignoreMessagesInTimestampRange: nil, count: 10, fixedCombinedReadStates: fixedCombinedReadStates, topTaggedMessageIdNamespaces: Set(), tag: nil, appendMessagesFromTheSameGroup: false, namespaces: .not(Namespaces.Message.allScheduled), orderStatistics: .combinedLocation) + signals.append(account.postbox.aroundMessageHistoryViewForLocation(.peer(peerId: index.messageIndex.id.peerId, threadId: nil), anchor: .upperBound, ignoreMessagesInTimestampRange: nil, count: 10, fixedCombinedReadStates: fixedCombinedReadStates, topTaggedMessageIdNamespaces: Set(), tag: nil, appendMessagesFromTheSameGroup: false, namespaces: .not(Namespaces.Message.allNonRegular), orderStatistics: .combinedLocation) |> take(1) |> map { view -> [INMessage] in var messages: [INMessage] = [] diff --git a/Telegram/Telegram-iOS/ar.lproj/NiceLocalizable.strings b/Telegram/Telegram-iOS/ar.lproj/NiceLocalizable.strings index 7481159fd2f..dca106d3d3f 100644 --- a/Telegram/Telegram-iOS/ar.lproj/NiceLocalizable.strings +++ b/Telegram/Telegram-iOS/ar.lproj/NiceLocalizable.strings @@ -162,8 +162,6 @@ "NicegramSettings.Other.hideReactions" = "إخفاء ردود الفعل"; "NicegramSettings.RoundVideos.DownloadVideos" = "نزّل الفيديوهات إلى المعرض"; "Premium.OnetapTranslate.UseIgnoreLanguages" = "تجاهل اللغات"; -"NicegramSettings.ShareChannelsInfoToggle" = "شارك معلومات القناة"; -"NicegramSettings.ShareChannelsInfoToggle.Note" = "ساهم في موسوعة ويكيبيديا الأكثر شمولاً لقنوات ومجموعات تيليجرام عن طريق إرسال معلومات القناة تلقائياً والارتباط بقاعدة البيانات الخاصة بنا. نحن لا نربط ملفك الشخصي وقناتك على تيليجرام ولا نشارك بياناتك الشخصية."; "Premium.OnetapTranslate.UseIgnoreLanguages.Note" = "التطبيق يحدد لغات كل رسالة في جهازك. ويتطلب اداء عالي من الجهاز و بالتالي إستهلاك البطارية."; "Premium.rememberFolderOnExit" = "تذكر المجلد الحالي عند الخروج"; "NicegramSettings.HideStories" = "إخفاء الستوري"; @@ -221,3 +219,9 @@ "ChatContextMenu.Hide" = "إخفاء"; "ChatContextMenu.Unhide" = "إظهار"; "HiddenChatsTooltip" = "اضغط مع الاستمرار على شعار 'N' لكشف الدردشات السرية المخفية"; + +/*Data Sharing*/ +"NicegramSettings.ShareBotsToggle" = "مشاركة معلومات البوتات"; +"NicegramSettings.ShareChannelsToggle" = "مشاركة معلومات القناة"; +"NicegramSettings.ShareStickersToggle" = "مشاركة معلومات الملصقات"; +"NicegramSettings.ShareData.Note" = "ساهم في أكثر ويكيبيديا شمولية لقنوات ومجموعات تليغرام من خلال إرسال معلومات القناة والرابط تلقائيًا إلى قاعدة بياناتنا. نحن لا نربط ملفك الشخصي في تليغرام والقناة ولا نشارك بياناتك الشخصية."; diff --git a/Telegram/Telegram-iOS/de.lproj/NiceLocalizable.strings b/Telegram/Telegram-iOS/de.lproj/NiceLocalizable.strings index bca5c528f4e..07cac4b05a3 100644 --- a/Telegram/Telegram-iOS/de.lproj/NiceLocalizable.strings +++ b/Telegram/Telegram-iOS/de.lproj/NiceLocalizable.strings @@ -162,8 +162,6 @@ "NicegramSettings.Other.hideReactions" = "Reaktionen verbergen"; "NicegramSettings.RoundVideos.DownloadVideos" = "Videos in die Galerie herunterladen"; "Premium.OnetapTranslate.UseIgnoreLanguages" = "Sprachen ignorieren"; -"NicegramSettings.ShareChannelsInfoToggle" = "Kanalinformationen teilen"; -"NicegramSettings.ShareChannelsInfoToggle.Note" = "Unterstützen Sie die umfassendste Wikipedia von Telegrammkanälen und Gruppen, indem Sie automatisch Kanal-Informationen und Links zu unserer Datenbank senden. Wir verbinden Ihr Telegramm-Profil und Ihren Kanal nicht und teilen Ihre persönlichen Daten nicht."; "Premium.OnetapTranslate.UseIgnoreLanguages.Note" = "Die App erkennt die Sprache jeder Nachricht auf deinem Gerät. Das ist eine Aufgabe mit hoher Priorität, die sich auf die Akkulaufzeit auswirken kann."; "Premium.rememberFolderOnExit" = "Aktuellen Ordner beim Beenden merken"; "NicegramSettings.HideStories" = "Stories verbergen"; @@ -221,3 +219,9 @@ "ChatContextMenu.Hide" = "Verbergen"; "ChatContextMenu.Unhide" = "Einblenden"; "HiddenChatsTooltip" = "Drücken und halten Sie das 'N'-Logo, um versteckte Geheimchats zu enthüllen"; + +/*Data Sharing*/ +"NicegramSettings.ShareBotsToggle" = "Teilen Sie Bot-Informationen"; +"NicegramSettings.ShareChannelsToggle" = "Teilen Sie Kanal-Informationen"; +"NicegramSettings.ShareStickersToggle" = "Teilen Sie Sticker-Informationen"; +"NicegramSettings.ShareData.Note" = "Tragen Sie zur umfassendsten Wikipedia von Telegram-Kanälen und -Gruppen bei, indem Sie automatisch Kanalinformationen und Links zu unserer Datenbank übermitteln. Wir verbinden Ihr Telegram-Profil und Ihren Kanal nicht und teilen Ihre persönlichen Daten nicht."; diff --git a/Telegram/Telegram-iOS/en.lproj/Localizable.strings b/Telegram/Telegram-iOS/en.lproj/Localizable.strings index 934fe2e4221..f75375eb404 100644 --- a/Telegram/Telegram-iOS/en.lproj/Localizable.strings +++ b/Telegram/Telegram-iOS/en.lproj/Localizable.strings @@ -11218,6 +11218,7 @@ Sorry for the inconvenience."; "GroupBoost.AdditionalFeatures" = "Additional Features"; "GroupBoost.AdditionalFeaturesText" = "By gaining **boosts**, your group reaches higher levels and unlocks more features."; +"ChannelBoost.AdditionalFeaturesText" = "By gaining **boosts**, your channel reaches higher levels and unlocks more features."; "Stats.Boosts.Group.NoBoostersYet" = "No users currently boost your group"; "Stats.Boosts.Group.BoostersInfo" = "Your group is currently boosted by these members."; @@ -11315,3 +11316,281 @@ Sorry for the inconvenience."; "ChannelBoost.Header.Giveaway" = "giveaway"; "ChannelBoost.Header.Features" = "features"; + +"GroupBoost.Table.Group.VoiceToText" = "Voice-to-Text Conversion"; +"GroupBoost.Table.Group.EmojiPack" = "Custom Emojipack"; + +"ChannelBoost.Header.Boost" = "boost"; +"ChannelBoost.Header.Giveaway" = "giveaway"; +"ChannelBoost.Header.Features" = "features"; + +"Premium.FolderTags" = "Folder Tags"; +"Premium.FolderTagsInfo" = "Add colorful labels to chats for faster access with Telegram Premium."; +"Premium.FolderTagsStandaloneInfo" = "Add colorful labels to chats for faster access with Telegram Premium."; +"Premium.FolderTags.Proceed" = "About Telegram Premium"; + +"Premium.Business" = "Telegram Business"; +"Premium.BusinessInfo" = "Upgrade your account with business features such as location, opening hours and quick replies."; + +"Premium.Business.Location.Title" = "Location"; +"Premium.Business.Location.Text" = "Display the location of your business on your account."; + +"Premium.Business.Hours.Title" = "Opening Hours"; +"Premium.Business.Hours.Text" = "Show to your customers when you are open for business."; + +"Premium.Business.Replies.Title" = "Quick Replies"; +"Premium.Business.Replies.Text" = "Set up shortcuts with rich text and media to respond to messages faster."; + +"Premium.Business.Greetings.Title" = "Greeting Messages"; +"Premium.Business.Greetings.Text" = "Create greetings that will be automatically sent to new customers."; + +"Premium.Business.Away.Title" = "Away Messages"; +"Premium.Business.Away.Text" = "Define messages that are automatically sent when you are off."; + +"Premium.Business.Chatbots.Title" = "Chatbots"; +"Premium.Business.Chatbots.Text" = "Add any third party chatbots that will process customer interactions."; + +"Business.Title" = "Telegram Business"; +"Business.Description" = "Turn your account into a **business page** with these additional features."; +"Business.SubscribedDescription" = "You have now unlocked these additional business features."; + +"Business.Location" = "Location"; +"Business.OpeningHours" = "Opening Hours"; +"Business.QuickReplies" = "Quick Replies"; +"Business.GreetingMessages" = "Greeting Messages"; +"Business.AwayMessages" = "Away Messages"; +"Business.Chatbots" = "Chatbots"; + +"Business.LocationInfo" = "Display the location of your business on your account."; +"Business.OpeningHoursInfo" = "Show to your customers when you are open for business."; +"Business.QuickRepliesInfo" = "Set up shortcuts with rich text and media to respond to messages faster."; +"Business.GreetingMessagesInfo" = "Create greetings that will be automatically sent to new customers."; +"Business.AwayMessagesInfo" = "Define messages that are automatically sent when you are off."; +"Business.ChatbotsInfo" = "Add any third-party chatbots that will process customer interactions."; + +"Business.MoreFeaturesTitle" = "MORE BUSINESS FEATURES"; +"Business.MoreFeaturesInfo" = "Check this section later for new business features."; + +"Business.SetEmojiStatus" = "Set Emoji Status"; +"Business.SetEmojiStatusInfo" = "Display the current status of your business next to your name."; + +"Business.TagYourChats" = "Tag Your Chats"; +"Business.TagYourChatsInfo" = "Add colorful labels to chats for faster access in chat list."; + +"Business.AddPost" = "Add a Post"; +"Business.AddPostInfo" = "Publish photos and videos of your goods or services on your page."; + +"StoryList.SavedEmptyPosts.Title" = "No posts yet..."; +"StoryList.SavedEmptyPosts.Text" = "Publish photos and videos to display on your profile page."; +"StoryList.SavedAddAction" = "Add a Post"; + +"PeerInfo.Channel.Boost" = "Boost Channel"; + +"ChannelBoost.Title" = "Boost Channel"; +"ChannelBoost.Info" = "Subscribers of your channel can **boost** it so that it **levels up** and gets **exclusive features**."; + +"Conversation.BoostToUnrestrictText" = "Boost this group to send messages"; + +"Settings.Business" = "Telegram Business"; + +"Story.Editor.DiscardText" = "If you go back now, you will lose any changes you made."; + +"Premium.Business.Description" = "Turn your account to a business page with these additional features."; + +"Attachment.Reply" = "Reply"; + +"ChatList.ItemMoreMessagesFormat_1" = "+1 MORE"; +"ChatList.ItemMoreMessagesFormat_any" = "+%d MORE"; +"ChatList.ItemMenuEdit" = "Edit"; +"ChatList.ItemMenuDelete" = "Delete"; +"ChatList.PeerTypeNonContact" = "non-contact"; + +"ChatListFilter.TagLabelNoTag" = "NO TAG"; +"ChatListFilter.TagLabelPremiumExpired" = "PREMIUM EXPIRED"; +"ChatListFilter.TagSectionTitle" = "FOLDER COLOR"; +"ChatListFilter.TagSectionFooter" = "This color will be used for the folder's tag in the chat list"; +"ChatListFilter.TagPremiumRequiredTooltipText" = "Subscribe to **Telegram Premium** to select folder color."; +"ChatListFilterList.ShowTags" = "Show Folder Tags"; +"ChatListFilterList.ShowTagsFooter" = "Display folder names for each chat in the chat list."; + +"PeerInfo.BusinessHours.DayOpen24h" = "open 24 hours"; +"PeerInfo.BusinessHours.DayClosed" = "closed"; +"PeerInfo.BusinessHours.StatusOpen" = "Open"; +"PeerInfo.BusinessHours.StatusClosed" = "Closed"; +"PeerInfo.BusinessHours.StatusOpensInMinutes_1" = "Opens in one minute"; +"PeerInfo.BusinessHours.StatusOpensInMinutes_any" = "Opens in %d minutes"; +"PeerInfo.BusinessHours.StatusOpensInHours_1" = "Opens in one hour"; +"PeerInfo.BusinessHours.StatusOpensInHours_any" = "Opens in %d hours"; +"PeerInfo.BusinessHours.StatusOpensOnDate" = "Opens %@"; +"PeerInfo.BusinessHours.StatusOpensTodayAt" = "Opens today at %@"; +"PeerInfo.BusinessHours.StatusOpensTomorrowAt" = "Opens tomorrow at %@"; +"PeerInfo.BusinessHours.TimezoneSwitchMy" = "my time"; +"PeerInfo.BusinessHours.TimezoneSwitchBusiness" = "local time"; +"PeerInfo.BusinessHours.Label" = "business hours"; +"PeerInfo.Location.Label" = "location"; + +"QuickReplies.EmptyState.Title" = "No Quick Replies"; +"QuickReplies.EmptyState.Text" = "Set up shortcuts with rich text and media to respond to messages faster."; +"QuickReplies.EmptyState.AddButton" = "Add Quick Reply"; + +"Chat.CommandList.EditQuickReplies" = "Edit Quick Replies"; + +"Conversation.EditingQuickReplyPanelTitle" = "Edit Quick Reply"; + +"Chat.Placeholder.QuickReply" = "Add quick reply.."; +"Chat.Placeholder.GreetingMessage" = "Add greeting message..."; +"Chat.Placeholder.AwayMessage" = "Add away message..."; + +"Chat.QuickReplyMessageLimitReachedText_1" = "Limit of %d message reached"; +"Chat.QuickReplyMessageLimitReachedText_any" = "Limit of %d messages reached"; + +"Chat.EmptyState.QuickReply.Title" = "New Quick Reply"; +"Chat.EmptyState.QuickReply.Text1" = "· Enter a message below that will be sent in chats when you type \"**/%@\"**."; +"Chat.EmptyState.QuickReply.Text2" = "· You can access Quick Replies in any chat by typing \"/\" or using the Attachment menu."; +"EmptyState.GreetingMessage.Title" = "New Greeting Message"; +"EmptyState.GreetingMessage.Text" = "Create greetings that will be automatically sent to new customers"; +"EmptyState.AwayMessage.Title" = "New Away Message"; +"EmptyState.AwayMessage.Text" = "Add messages that are automatically sent when you are off."; + +"QuickReply.Title" = "Quick Replies"; +"QuickReply.SelectedTitle_1" = "%d Selected"; +"QuickReply.SelectedTitle_any" = "%d Selected"; + +"QuickReply.ShortcutPlaceholder" = "Shortcut"; +"QuickReply.CreateShortcutTitle" = "New Quick Reply"; +"QuickReply.CreateShortcutText" = "Add a shortcut for your quick reply."; +"QuickReply.EditShortcutTitle" = "Edit Shortcut"; +"QuickReply.EditShortcutText" = "Add a new name for your shortcut."; +"QuickReply.ShortcutExistsInlineError" = "Shortcut with that name already exists"; + +"QuickReply.ChatRemoveGeneric.Title" = "Remove Shortcut"; +"QuickReply.ChatRemoveGeneric.Text" = "You didn't create a quick reply message. Exiting will remove the shortcut."; +"QuickReply.ChatRemoveGreetingMessage.Title" = "Remove Greeting Message"; +"QuickReply.ChatRemoveGreetingMessage.Text" = "You didn't create a greeting message. Exiting will remove it."; +"QuickReply.ChatRemoveAwayMessage.Title" = "Remove Away Message"; +"QuickReply.ChatRemoveAwayMessage.Text" = "You didn't create an away message. Exiting will remove it."; +"QuickReply.ChatRemoveGeneric.DeleteAction" = "Delete"; +"QuickReply.TitleGreetingMessage" = "Greeting Message"; +"QuickReply.TitleAwayMessage" = "Away Message"; + +"QuickReply.InlineCreateAction" = "New Quick Reply"; +"QuickReply.DeleteConfirmationSingle" = "Delete Quick Reply"; +"QuickReply.DeleteConfirmationMultiple" = "Delete Quick Replies"; +"QuickReply.DeleteAction_1" = "Delete 1 Quick Reply"; +"QuickReply.DeleteAction_any" = "Delete %d Quick Replies"; + +"TimeZoneSelection.Title" = "Time Zone"; + +"BusinessMessageSetup.TitleGreetingMessage" = "Greeting Message"; +"BusinessMessageSetup.TextGreetingMessage" = "Greet customers when they message you the first time or after a period of no activity."; +"BusinessMessageSetup.TitleAwayMessage" = "Away Message"; +"BusinessMessageSetup.TextAwayMessage" = "Automatically reply with a message when you are away."; +"BusinessMessageSetup.ToggleGreetingMessage" = "Send Greeting Message"; +"BusinessMessageSetup.ToggleAwayMessage" = "Send Away Message"; +"BusinessMessageSetup.CreateGreetingMessage" = "Create a Greeting Message"; +"BusinessMessageSetup.CreateAwayMessage" = "Create an Away Message"; +"BusinessMessageSetup.GreetingMessageSectionHeader" = "GREETING MESSAGE"; +"BusinessMessageSetup.AwayMessageSectionHeader" = "AWAY MESSAGE"; + +"BusinessMessageSetup.ScheduleSectionHeader" = "SCHEDULE"; +"BusinessMessageSetup.ScheduleAlways" = "Always Send"; +"BusinessMessageSetup.ScheduleOutsideBusinessHours" = "Outside of Business Hours"; +"BusinessMessageSetup.ScheduleCustom" = "Custom Schedule"; +"BusinessMessageSetup.ScheduleStartTime" = "Start Time"; +"BusinessMessageSetup.ScheduleEndTime" = "End Time"; +"BusinessMessageSetup.ScheduleTimePlaceholder" = "Set"; + +"BusinessMessageSetup.SendWhenOffline" = "Only if Offline"; +"BusinessMessageSetup.SendWhenOfflineFooter" = "Don't send the away message if you've recently been online."; + +"BusinessMessageSetup.ErrorNoRecipients.Text" = "No recipients selected. Reset?"; +"BusinessMessageSetup.ErrorNoRecipients.ResetAction" = "Reset"; +"BusinessMessageSetup.ErrorScheduleEndTimeBeforeStartTime.Text" = "Custom schedule end time must be larger than start time."; +"BusinessMessageSetup.ErrorScheduleTimeMissing.Text" = "Custom schedule time is missing."; +"BusinessMessageSetup.ErrorScheduleStartTimeMissing.Text" = "Custom schedule start time is missing."; +"BusinessMessageSetup.ErrorScheduleEndTimeMissing.Text" = "Custom schedule end time is missing."; +"BusinessMessageSetup.ErrorScheduleTime.ResetAction" = "Reset"; + +"BusinessMessageSetup.RecipientsSectionHeader" = "RECIPIENTS"; +"BusinessMessageSetup.RecipientsOptionAllExcept" = "All 1-to-1 Chats Except..."; +"BusinessMessageSetup.RecipientsOptionOnly" = "Only Selected Chats"; + +"BusinessMessageSetup.Recipients.AddExclude" = "Exclude Chats..."; +"BusinessMessageSetup.Recipients.AddInclude" = "Include Chats..."; + +"BusinessMessageSetup.Recipients.CategoryExistingChats" = "Existing Chats"; +"BusinessMessageSetup.Recipients.CategoryNewChats" = "New Chats"; +"BusinessMessageSetup.Recipients.CategoryContacts" = "Contacts"; +"BusinessMessageSetup.Recipients.CategoryNonContacts" = "Non-Contacts"; +"BusinessMessageSetup.Recipients.IncludeSearchTitle" = "Include Chats"; +"BusinessMessageSetup.Recipients.ExcludeSearchTitle" = "Exclude Chats"; + +"BusinessMessageSetup.Recipients.IncludedSectionHeader" = "INCLUDED CHATS"; +"BusinessMessageSetup.Recipients.ExcludedSectionHeader" = "EXCLUDED CHATS"; +"BusinessMessageSetup.Recipients.GreetingMessageFooter" = "Choose chats or entire chat categories for sending a greeting message."; +"BusinessMessageSetup.Recipients.AwayMessageFooter" = "Choose chats or entire chat categories for sending an away message."; + +"BusinessMessageSetup.InactivitySectionHeader" = "PERIOD OF NO ACTIVITY"; +"BusinessMessageSetup.InactivitySectionFooter" = "Choose how many days should pass after your last interaction with a recipient to send them the greeting in response to their message."; + +"BusinessHoursSetup.Title" = "Business Hours"; +"BusinessHoursSetup.Text" = "Turn this on to show your opening hours schedule to your customers."; + +"BusinessHoursSetup.MainToggle" = "Show Business Hours"; + +"BusinessHoursSetup.DaySwitch" = "Open On This Day"; +"BusinessHoursSetup.DayIntervalStart" = "Opening time"; +"BusinessHoursSetup.DayIntervalEnd" = "Closing time"; +"BusinessHoursSetup.DayIntervalRemove" = "Remove"; +"BusinessHoursSetup.AddSectionFooter" = "Specify your working hours during the day."; +"BusinessHoursSetup.AddAction" = "Add a Set of Hours"; + +"BusinessHoursSetup.ErrorIntersectingHours.Text" = "Business hours are intersecting. Reset?"; +"BusinessHoursSetup.ErrorIntersectingHours.ResetAction" = "Reset"; + +"BusinessHoursSetup.ErrorIntersectingDays.Text" = "Business hours are intersecting. Reset?"; +"BusinessHoursSetup.ErrorIntersectingDays.ResetAction" = "Reset"; + +"BusinessHoursSetup.DayOpen24h" = "Open 24 Hours"; +"BusinessHoursSetup.DayClosed" = "Closed"; +"BusinessHoursSetup.DaysSectionTitle" = "BUSINESS HOURS"; + +"BusinessHoursSetup.TimeZone" = "Time Zone"; + +"BusinessLocationSetup.Title" = "Location"; +"BusinessLocationSetup.Text" = "Display the location of your business on your account."; + +"BusinessLocationSetup.AddressPlaceholder" = "Enter Address"; +"BusinessLocationSetup.SetLocationOnMap" = "Set Location on Map"; +"BusinessLocationSetup.DeleteLocation" = "Delete Location"; + +"BusinessLocationSetup.ErrorAddressEmpty.Text" = "Address can't be empty."; +"BusinessLocationSetup.ErrorAddressEmpty.ResetAction" = "Delete"; + +"BusinessLocationSetup.AlertUnsavedChanges.Text" = "You have unsaved changes."; +"BusinessLocationSetup.AlertUnsavedChanges.ResetAction" = "Revert"; + +"ChatbotSetup.Title" = "Chatbots"; +"ChatbotSetup.Text" = "Add a bot to your account to help you automatically process and respond to the messages you receive. [Learn More >]()"; +"ChatbotSetup.TextLink" = "https://telegram.org"; + +"ChatbotSetup.BotSearchPlaceholder" = "Bot Username"; +"ChatbotSetup.BotSectionFooter" = "Enter the username or URL of the Telegram bot that you want to automatically process your chats."; + +"ChatbotSetup.RecipientsSectionHeader" = "CHATS ACCESSIBLE FOR THE BOT"; + +"ChatbotSetup.Recipients.ExcludedSectionFooter" = "Select chats or entire chat categories which the bot **WILL NOT** have access to."; +"ChatbotSetup.Recipients.IncludedSectionFooter" = "Select chats or entire chat categories which the bot **WILL** have access to."; + +"ChatbotSetup.PermissionsSectionHeader" = "BOT PERMISSIONS"; +"ChatbotSetup.PermissionsSectionFooter" = "The bot will be able to view all new incoming messages, but not the messages that had been sent before you added the bot."; +"ChatbotSetup.Permission.ReplyToMessages" = "Reply to Messages"; + +"ChatbotSetup.BotAddAction" = "ADD"; +"ChatbotSetup.BotNotFoundStatus" = "Chatbot not found"; + +"Chat.QuickReply.ServiceHeader1" = "To edit or delete your quick reply, tap an hold on it."; +"Chat.QuickReply.ServiceHeader2" = "To use this quick reply in a chat, type / and select the shortcut from the list."; + +"Chat.QuickReplyMediaMessageLimitReachedText_1" = "There can be at most %d message in this chat."; +"Chat.QuickReplyMediaMessageLimitReachedText_any" = "There can be at most %d messages in this chat."; diff --git a/Telegram/Telegram-iOS/en.lproj/NiceLocalizable.strings b/Telegram/Telegram-iOS/en.lproj/NiceLocalizable.strings index 771b089e502..e74c084ce00 100644 --- a/Telegram/Telegram-iOS/en.lproj/NiceLocalizable.strings +++ b/Telegram/Telegram-iOS/en.lproj/NiceLocalizable.strings @@ -162,8 +162,6 @@ "NicegramSettings.Other.hideReactions" = "Hide Reactions"; "NicegramSettings.RoundVideos.DownloadVideos" = "Download videos to Gallery"; "Premium.OnetapTranslate.UseIgnoreLanguages" = "Ignore Languages"; -"NicegramSettings.ShareChannelsInfoToggle" = "Share channel information"; -"NicegramSettings.ShareChannelsInfoToggle.Note" = "Contribute to the most comprehensive wikipedia of Telegram channels and groups by automatically submitting channel information and link to our database. We do not connect your telegram profile and channel and do not share your personal data."; "Premium.OnetapTranslate.UseIgnoreLanguages.Note" = "App detects language of each message on your device. It's a high-performance task which can affect your battery life."; "Premium.rememberFolderOnExit" = "Remember current folder on exit"; "NicegramSettings.HideStories" = "Hide Stories"; @@ -221,3 +219,9 @@ "ChatContextMenu.Hide" = "Hide"; "ChatContextMenu.Unhide" = "Unhide"; "HiddenChatsTooltip" = "Press and hold the 'N' logo to reveal hidden secret chats"; + +/*Data Sharing*/ +"NicegramSettings.ShareBotsToggle" = "Share bots information"; +"NicegramSettings.ShareChannelsToggle" = "Share channel information"; +"NicegramSettings.ShareStickersToggle" = "Share stickers information"; +"NicegramSettings.ShareData.Note" = "Contribute to the most comprehensive wikipedia of Telegram channels and groups by automatically submitting channel information and link to our database. We do not connect your telegram profile and channel and do not share your personal data."; diff --git a/Telegram/Telegram-iOS/es.lproj/NiceLocalizable.strings b/Telegram/Telegram-iOS/es.lproj/NiceLocalizable.strings index a7e44c19dc1..cbf6e73784d 100644 --- a/Telegram/Telegram-iOS/es.lproj/NiceLocalizable.strings +++ b/Telegram/Telegram-iOS/es.lproj/NiceLocalizable.strings @@ -162,8 +162,6 @@ "NicegramSettings.Other.hideReactions" = "Ocultar reacciones"; "NicegramSettings.RoundVideos.DownloadVideos" = "Descargar vídeos en la galería"; "Premium.OnetapTranslate.UseIgnoreLanguages" = "Ignorar idiomas"; -"NicegramSettings.ShareChannelsInfoToggle" = "Comparta información del canal"; -"NicegramSettings.ShareChannelsInfoToggle.Note" = "Contribuya a la Wikipedia más completa de canales y grupos de Telegram enviando automáticamente la información del canal y el enlace a nuestra base de datos. No conectamos su perfil y canal de Telegram y no compartimos sus datos personales."; "Premium.OnetapTranslate.UseIgnoreLanguages.Note" = "La aplicación detecta el idioma de cada mensaje en su aparato. Es una tarea de alta predeterminación que puede afectar la duración de su batería."; "Premium.rememberFolderOnExit" = "Recordar la carpeta actual a la salida"; "NicegramSettings.HideStories" = "Ocultar Historias"; @@ -221,3 +219,9 @@ "ChatContextMenu.Hide" = "Esconder"; "ChatContextMenu.Unhide" = "Mostrar"; "HiddenChatsTooltip" = "Mantén presionado el logo 'N' para revelar chats secretos ocultos"; + +/*Data Sharing*/ +"NicegramSettings.ShareBotsToggle" = "Compartir información de bots"; +"NicegramSettings.ShareChannelsToggle" = "Compartir información de canales"; +"NicegramSettings.ShareStickersToggle" = "Compartir información de stickers"; +"NicegramSettings.ShareData.Note" = "Contribuye a la wikipedia más completa de canales y grupos de Telegram enviando automáticamente información y enlaces de canales a nuestra base de datos. No conectamos tu perfil de telegram con el canal y no compartimos tus datos personales."; diff --git a/Telegram/Telegram-iOS/fr.lproj/NiceLocalizable.strings b/Telegram/Telegram-iOS/fr.lproj/NiceLocalizable.strings index b247a1e6232..cc9d348f72d 100644 --- a/Telegram/Telegram-iOS/fr.lproj/NiceLocalizable.strings +++ b/Telegram/Telegram-iOS/fr.lproj/NiceLocalizable.strings @@ -162,8 +162,6 @@ "NicegramSettings.Other.hideReactions" = "Masquer les réactions"; "NicegramSettings.RoundVideos.DownloadVideos" = "Télécharger des vidéos dans la galerie"; "Premium.OnetapTranslate.UseIgnoreLanguages" = "Ignorer les langues"; -"NicegramSettings.ShareChannelsInfoToggle" = "Partager les informations du canal"; -"NicegramSettings.ShareChannelsInfoToggle.Note" = "Contribuez au Wikipédia le plus complet de chaînes et de groupes Telegram en soumettant automatiquement des informations sur les chaînes et un lien vers notre base de données. Nous ne connectons pas votre profil de télégramme et votre canal et ne partageons pas vos données personnelles."; "Premium.OnetapTranslate.UseIgnoreLanguages.Note" = "L'application détecte la langue de chaque message sur votre appareil. C'est une tâche de haute performance qui peut affecter l'autonomie de votre batterie."; "Premium.rememberFolderOnExit" = "Mémoriser le dossier actuel en quittant"; "NicegramSettings.HideStories" = "Cacher les histoires"; @@ -221,3 +219,9 @@ "ChatContextMenu.Hide" = "Masquer"; "ChatContextMenu.Unhide" = "Révéler"; "HiddenChatsTooltip" = "Appuyez et maintenez le logo 'N' pour révéler les chats secrets cachés"; + +/*Data Sharing*/ +"NicegramSettings.ShareBotsToggle" = "Partager les informations des bots"; +"NicegramSettings.ShareChannelsToggle" = "Partager les informations du canal"; +"NicegramSettings.ShareStickersToggle" = "Partager des informations sur les stickers"; +"NicegramSettings.ShareData.Note" = "Contribuez au Wikipédia le plus complet de chaînes et de groupes Telegram en soumettant automatiquement des informations sur les chaînes et un lien vers notre base de données. Nous ne connectons pas votre profil de télégramme et votre canal et ne partageons pas vos données personnelles."; diff --git a/Telegram/Telegram-iOS/it.lproj/NiceLocalizable.strings b/Telegram/Telegram-iOS/it.lproj/NiceLocalizable.strings index 9e172601316..ebf6aef7acc 100644 --- a/Telegram/Telegram-iOS/it.lproj/NiceLocalizable.strings +++ b/Telegram/Telegram-iOS/it.lproj/NiceLocalizable.strings @@ -162,8 +162,6 @@ "NicegramSettings.Other.hideReactions" = "Nascondi Reazioni"; "NicegramSettings.RoundVideos.DownloadVideos" = "Scarica i video nella Galleria"; "Premium.OnetapTranslate.UseIgnoreLanguages" = "Ignora Lingue"; -"NicegramSettings.ShareChannelsInfoToggle" = "Condividi informazioni sul canale"; -"NicegramSettings.ShareChannelsInfoToggle.Note" = "Contribuisci alla wikipedia più completa di canali e gruppi di Telegram inviando automaticamente le informazioni e i link dei canali al nostro database. Non colleghiamo il tuo profilo e canale Telegram e non condividiamo i tuoi dati personali."; "Premium.OnetapTranslate.UseIgnoreLanguages.Note" = "L'app rileva la lingua di ogni messaggio sul tuo dispositivo. È un'attività che consuma tante risorse e che può influire sulla durata della batteria."; "Premium.rememberFolderOnExit" = "Ricorda la cartella corrente all'uscita"; "NicegramSettings.HideStories" = "Nascondi Storie"; @@ -221,3 +219,9 @@ "ChatContextMenu.Hide" = "Nascondi"; "ChatContextMenu.Unhide" = "Rivelare"; "HiddenChatsTooltip" = "Premi e tieni premuto il logo 'N' per rivelare chat segrete nascoste"; + +/*Data Sharing*/ +"NicegramSettings.ShareBotsToggle" = "Condividi informazioni sui bot"; +"NicegramSettings.ShareChannelsToggle" = "Condividi informazioni sul canale"; +"NicegramSettings.ShareStickersToggle" = "Condividi informazioni sugli sticker"; +"NicegramSettings.ShareData.Note" = "Contribuisci all'enciclopedia più completa di canali e gruppi Telegram inviando automaticamente informazioni e link sul canale al nostro database. Non collegiamo il tuo profilo Telegram e il canale e non condividiamo i tuoi dati personali."; diff --git a/Telegram/Telegram-iOS/km.lproj/NiceLocalizable.strings b/Telegram/Telegram-iOS/km.lproj/NiceLocalizable.strings index 9afc7c4d916..ed58fa4dda2 100644 --- a/Telegram/Telegram-iOS/km.lproj/NiceLocalizable.strings +++ b/Telegram/Telegram-iOS/km.lproj/NiceLocalizable.strings @@ -44,7 +44,7 @@ "Common.Later" = "ពេលក្រោយ"; "Common.RestartRequired" = "តម្រូវ​ឲ្យ​ចាប់ផ្ដើម​ឡើង​វិញ!"; "NiceFeatures.Tabs.ShowNames" = "បង្ហាញឈ្មោះថេប"; -"Chat.ForwardAsCopy" = "បញ្ជូនបន្តជាចម្លង"; +"Chat.ForwardAsCopy" = "បញ្ជូនបន្តជាច្បាប់ចម្លង"; /*Folder*/ "Folder.DefaultName" = "ថត"; @@ -87,3 +87,13 @@ "IAP.Premium.Title" = "ព្រីមៀរ"; "IAP.Premium.Subtitle" = "លក្ខណៈពិសេសប្លែក ដែលអ្នកមិនអាចបដិសេធ!"; "IAP.Premium.Features" = "អ្នកបកប្រែសាររហ័ស\n\nចងចាំថតដែលបានជ្រើសនៅពេលចេញ"; + +/*Registration Date*/ +"NiceFeatures.RegDate.OlderThan" = "ចាស់ជាង"; +"NiceFeatures.RegDate.Approximately" = "ប្រហែល"; +"NiceFeatures.RegDate.NewerThan" = "ថ្មោងថ្មីជាង"; +"NicegramOnboarding.1.Title" = "អតិថូបករណ៍ №1 សម្រាប់កម្មវិធីសារតេឡេក្រាម"; +"NicegramOnboarding.1.Desc" = "អ្នកប្រើប្រាស់ Nicegram ច្រើនជាង 2 លាននាក់ និងអាចអាក់សេសអតិថូបករណ៍តេឡេក្រាមដ៏មានឥទ្ធិពល និងសុវត្ថិភាពបំផុតសម្រាប់អាជីវកម្ម។"; +"NicegramOnboarding.2.Title" = "បទពិសោធន៍ផ្ញើសារកម្រិតខ្ពស់"; +"NicegramOnboarding.3.Title" = "ប្រូហ្វាលអត្ថជនបន្ថែម"; +"NicegramOnboarding.4.Title" = "ផ្នែកបន្ថែមអាជីវកម្មតែមួយគត់"; diff --git a/Telegram/Telegram-iOS/ko.lproj/NiceLocalizable.strings b/Telegram/Telegram-iOS/ko.lproj/NiceLocalizable.strings index ffbe637fe54..f93eb79f46a 100644 --- a/Telegram/Telegram-iOS/ko.lproj/NiceLocalizable.strings +++ b/Telegram/Telegram-iOS/ko.lproj/NiceLocalizable.strings @@ -162,8 +162,6 @@ "NicegramSettings.Other.hideReactions" = "반응 숨기기"; "NicegramSettings.RoundVideos.DownloadVideos" = "사진앨범으로 동영상 다운로드"; "Premium.OnetapTranslate.UseIgnoreLanguages" = "언어 무시하기"; -"NicegramSettings.ShareChannelsInfoToggle" = "채널 정보 공유"; -"NicegramSettings.ShareChannelsInfoToggle.Note" = "저희 데이터 베이스에 자동으로 채널 정보와 링크를 제출함으로써 텔레그램(Telegram) 채널 및 그룹의 가장 포괄적인 위키피디아에 기여하세요. 저희는 귀하의 텔레그램 프로필과 채널을 연결하지 않으며 귀하의 개인 정보도 공유하지 않습니다."; "Premium.OnetapTranslate.UseIgnoreLanguages.Note" = "앱은 기기에서 각 메시지의 언어를 감지합니다. 배터리 수명에 영향을 줄 수 있는 고성능 작업입니다."; "Premium.rememberFolderOnExit" = "종료시 현재 폴더 기억하기"; "NicegramSettings.HideStories" = "스토리 숨기기"; @@ -221,3 +219,9 @@ "ChatContextMenu.Hide" = "숨기기"; "ChatContextMenu.Unhide" = "숨김 해제"; "HiddenChatsTooltip" = "숨겨진 비밀 채팅을 보려면 'N' 로고를 누르고 있으세요"; + +/*Data Sharing*/ +"NicegramSettings.ShareBotsToggle" = "봇 정보 공유하기"; +"NicegramSettings.ShareChannelsToggle" = "채널 정보 공유하기"; +"NicegramSettings.ShareStickersToggle" = "스티커 정보 공유하기"; +"NicegramSettings.ShareData.Note" = "자동으로 채널 정보와 링크를 제출함으로써 Telegram 채널과 그룹들의 가장 포괄적인 위키백과에 기여하세요. 우리는 여러분의 Telegram 프로필과 채널을 연결하거나 개인 데이터를 공유하지 않습니다."; diff --git a/Telegram/Telegram-iOS/pl.lproj/NiceLocalizable.strings b/Telegram/Telegram-iOS/pl.lproj/NiceLocalizable.strings index 1e5f394ed54..da62bd6b7bf 100644 --- a/Telegram/Telegram-iOS/pl.lproj/NiceLocalizable.strings +++ b/Telegram/Telegram-iOS/pl.lproj/NiceLocalizable.strings @@ -162,8 +162,6 @@ "NicegramSettings.Other.hideReactions" = "Ukryj reakcje"; "NicegramSettings.RoundVideos.DownloadVideos" = "Pobierz filmy do galerii"; "Premium.OnetapTranslate.UseIgnoreLanguages" = "Ignorowane języki"; -"NicegramSettings.ShareChannelsInfoToggle" = "Udostępnij informacje o kanale"; -"NicegramSettings.ShareChannelsInfoToggle.Note" = "Przyczyniaj się do najbardziej wszechstronnej bazy danych kanałów i grup Telegram, automatycznie przesyłając informacje o kanałach i linki do naszej bazy danych. Nie łączymy twojego profilu Telegram z kanałem i nie udostępniamy twoich danych osobowych."; "Premium.OnetapTranslate.UseIgnoreLanguages.Note" = "Aplikacja wykrywa język każdej wiadomości na urządzeniu. Jest to zadanie o wysokim stopniu zaawansowania, które może mieć wpływ na żywotność baterii."; "Premium.rememberFolderOnExit" = "Zapamiętaj bieżący folder przy wyjściu"; "NicegramSettings.HideStories" = "Ukryj Historie"; @@ -221,3 +219,9 @@ "ChatContextMenu.Hide" = "Ukryj"; "ChatContextMenu.Unhide" = "Odkryj"; "HiddenChatsTooltip" = "Naciśnij i przytrzymaj logo 'N', aby odkryć ukryte sekretne czaty"; + +/*Data Sharing*/ +"NicegramSettings.ShareBotsToggle" = "Udostępnij informacje o botach"; +"NicegramSettings.ShareChannelsToggle" = "Udostępnij informacje o kanale"; +"NicegramSettings.ShareStickersToggle" = "Udostępnij informacje o naklejkach"; +"NicegramSettings.ShareData.Note" = "Przyczyn się do najbardziej kompleksowej wikipedii kanałów i grup Telegrama, automatycznie przesyłając informacje o kanale i link do naszej bazy danych. Nie łączymy Twojego profilu na Telegramie z kanałem i nie udostępniamy Twoich danych osobowych."; diff --git a/Telegram/Telegram-iOS/pt.lproj/NiceLocalizable.strings b/Telegram/Telegram-iOS/pt.lproj/NiceLocalizable.strings index d527d5285b2..7ea9e485fef 100644 --- a/Telegram/Telegram-iOS/pt.lproj/NiceLocalizable.strings +++ b/Telegram/Telegram-iOS/pt.lproj/NiceLocalizable.strings @@ -162,8 +162,6 @@ "NicegramSettings.Other.hideReactions" = "Ocultar Reações"; "NicegramSettings.RoundVideos.DownloadVideos" = "Baixar vídeos para a Galeria"; "Premium.OnetapTranslate.UseIgnoreLanguages" = "Ignorar idiomas"; -"NicegramSettings.ShareChannelsInfoToggle" = "Compartilhar informações do canal"; -"NicegramSettings.ShareChannelsInfoToggle.Note" = "Contribua para a wikipedia mais abrangente de canais e grupos do Telegram enviando automaticamente informações do canal e link para o nosso banco de dados. Não conectamos o seu perfil e canal do Telegram e não compartilhamos os seus dados pessoais."; "Premium.OnetapTranslate.UseIgnoreLanguages.Note" = "O aplicativo detecta o idioma de cada mensagem em seu dispositivo. É uma tarefa de alto desempenho que pode afetar a vida útil da bateria."; "Premium.rememberFolderOnExit" = "Lembrar pasta atual antes de sair"; "NicegramSettings.HideStories" = "Ocultar Stories"; @@ -221,3 +219,9 @@ "ChatContextMenu.Hide" = "Esconder"; "ChatContextMenu.Unhide" = "Revelar"; "HiddenChatsTooltip" = "Pressione e segure o logo 'N' para revelar chats secretos ocultos"; + +/*Data Sharing*/ +"NicegramSettings.ShareBotsToggle" = "Partilhar informação de bots"; +"NicegramSettings.ShareChannelsToggle" = "Partilhar informação de canais"; +"NicegramSettings.ShareStickersToggle" = "Partilhar informação de stickers"; +"NicegramSettings.ShareData.Note" = "Contribua para a wikipedia mais abrangente de canais e grupos do Telegram, submetendo automaticamente informações e links de canais à nossa base de dados. Não associamos o seu perfil do Telegram ao canal e não partilhamos os seus dados pessoais."; diff --git a/Telegram/Telegram-iOS/ru.lproj/NiceLocalizable.strings b/Telegram/Telegram-iOS/ru.lproj/NiceLocalizable.strings index d0bf6285af4..aa1a46a937b 100644 --- a/Telegram/Telegram-iOS/ru.lproj/NiceLocalizable.strings +++ b/Telegram/Telegram-iOS/ru.lproj/NiceLocalizable.strings @@ -162,8 +162,6 @@ "NicegramSettings.Other.hideReactions" = "Скрыть реакции"; "NicegramSettings.RoundVideos.DownloadVideos" = "Скачивать видео в галерею"; "Premium.OnetapTranslate.UseIgnoreLanguages" = "Игнорировать языки"; -"NicegramSettings.ShareChannelsInfoToggle" = "Поделиться информацией канала"; -"NicegramSettings.ShareChannelsInfoToggle.Note" = "Внесите свой вклад в самую полную Википедию каналов и групп Telegram, автоматически отправляя информацию о каналах и ссылки в нашу базу данных. Мы не связываем ваш телеграм-профиль и канал, а также никому не передаем ваши личные данные."; "Premium.OnetapTranslate.UseIgnoreLanguages.Note" = "Приложение определяет язык для каждого сообщения на Вашем устройстве. Поскольку это высокопроизводительная задача, время работы аккумулятора может уменьшиться."; "Premium.rememberFolderOnExit" = "Запомнить текущую папку при выходе"; "NicegramSettings.HideStories" = "Скрыть истории"; @@ -221,3 +219,9 @@ "ChatContextMenu.Hide" = "Скрыть"; "ChatContextMenu.Unhide" = "Показать"; "HiddenChatsTooltip" = "Нажмите и удерживайте логотип 'N', чтобы показать скрытые секретные чаты"; + +/*Data Sharing*/ +"NicegramSettings.ShareBotsToggle" = "Поделиться информацией о ботах"; +"NicegramSettings.ShareChannelsToggle" = "Поделиться информацией о канале"; +"NicegramSettings.ShareStickersToggle" = "Поделиться информацией о стикерах"; +"NicegramSettings.ShareData.Note" = "Внесите свой вклад в самую полную википедию каналов и групп Telegram, автоматически отправляя информацию о канале и ссылку в нашу базу данных. Мы не связываем ваш профиль Telegram и канал и не делимся вашими личными данными."; diff --git a/Telegram/Telegram-iOS/tr.lproj/NiceLocalizable.strings b/Telegram/Telegram-iOS/tr.lproj/NiceLocalizable.strings index fff7b6c8eec..f8d40272807 100644 --- a/Telegram/Telegram-iOS/tr.lproj/NiceLocalizable.strings +++ b/Telegram/Telegram-iOS/tr.lproj/NiceLocalizable.strings @@ -162,8 +162,6 @@ "NicegramSettings.Other.hideReactions" = "Tepkileri Gizle"; "NicegramSettings.RoundVideos.DownloadVideos" = "Videoları galeriye indir"; "Premium.OnetapTranslate.UseIgnoreLanguages" = "Dilleri görmezden gel"; -"NicegramSettings.ShareChannelsInfoToggle" = "Kanal bilgilerini paylaş"; -"NicegramSettings.ShareChannelsInfoToggle.Note" = "Kanal bilgilerini ve bağlantısını veri tabanımıza göndererek en kapsamlı Telegram kanalı wikipedia'sına katkıda bulunun. Telegram profilinizi ve kanalınızı bağlamayız ve kişisel verilerinizi paylaşmayız."; "Premium.OnetapTranslate.UseIgnoreLanguages.Note" = "Uygulama, cihazınızdaki her mesajın dilini algılar. Bu, pil ömrünüzü etkileyebilecek yüksek öncelikli bir görevdir."; "Premium.rememberFolderOnExit" = "Çıkışta şu anki klasörü hatırla"; "NicegramSettings.HideStories" = "Hikayeleri Gizle"; @@ -221,3 +219,9 @@ "ChatContextMenu.Hide" = "Gizle"; "ChatContextMenu.Unhide" = "Göster"; "HiddenChatsTooltip" = "Gizli gizli sohbetleri görmek için 'N' logosuna basılı tutun"; + +/*Data Sharing*/ +"NicegramSettings.ShareBotsToggle" = "Bot bilgilerini paylaş"; +"NicegramSettings.ShareChannelsToggle" = "Kanal bilgilerini paylaş"; +"NicegramSettings.ShareStickersToggle" = "Sticker bilgilerini paylaş"; +"NicegramSettings.ShareData.Note" = "Kanal bilgileri ve bağlantısını otomatik olarak göndererek Telegram kanalları ve gruplarının en kapsamlı vikipedisine katkıda bulunun. Telegram profilinizle ve kanalınızla bağlantı kurmuyoruz ve kişisel verilerinizi paylaşmıyoruz."; diff --git a/Telegram/Telegram-iOS/uk.lproj/NiceLocalizable.strings b/Telegram/Telegram-iOS/uk.lproj/NiceLocalizable.strings index 11421761fb2..79d2240183f 100644 --- a/Telegram/Telegram-iOS/uk.lproj/NiceLocalizable.strings +++ b/Telegram/Telegram-iOS/uk.lproj/NiceLocalizable.strings @@ -162,8 +162,6 @@ "NicegramSettings.Other.hideReactions" = "Приховати реакції"; "NicegramSettings.RoundVideos.DownloadVideos" = "Завантажити відео у Галерею"; "Premium.OnetapTranslate.UseIgnoreLanguages" = "Ігнорувати мови"; -"NicegramSettings.ShareChannelsInfoToggle" = "Поділитися інформацією про канал"; -"NicegramSettings.ShareChannelsInfoToggle.Note" = "Приєднуйтесь до найбільш всеосяжної Вікіпедії каналів і груп Telegram, автоматично надсилаючи інформацію про канал і посилання на нашу базу даних. Ми не зв'язуємо ваш профіль у Telegram і канал і не поширюємо ваші персональні дані."; "Premium.OnetapTranslate.UseIgnoreLanguages.Note" = "Додаток виявляє мову кожного повідомлення на вашому пристрої. Це високопродуктивне завдання, яке може вплинути на життя батареї."; "Premium.rememberFolderOnExit" = "Запам’ятати поточну теку при виході"; diff --git a/Telegram/Telegram-iOS/vi.lproj/NiceLocalizable.strings b/Telegram/Telegram-iOS/vi.lproj/NiceLocalizable.strings index 449d34446dc..5a4506401c2 100644 --- a/Telegram/Telegram-iOS/vi.lproj/NiceLocalizable.strings +++ b/Telegram/Telegram-iOS/vi.lproj/NiceLocalizable.strings @@ -124,6 +124,40 @@ "NGLab.RegDate.Title" = "Registration Date"; "NGLab.RegDate.Notice" = "This is an approximate date"; "NGLab.RegDate.MenuItem" = "registered"; +"NGLab.RegDate.FetchError" = "."; +"NGLab.BadDeviceToken" = ""; + +/*Backup Settings*/ +"NiceFeatures.BackupIcloud" = ""; +"NiceFeatures.BackupSettings" = "Backup Settings & Folders"; +"NiceFeatures.BackupSettings.Notice" = "."; +"NiceFeatures.BackupSettings.Done" = ""; +"NiceFeatures.BackupSettings.Error" = ""; + +"NiceFeatures.RestoreSettings.Confirm" = ""; +"NiceFeatures.RestoreSettings.Done" = ""; +"NiceFeatures.RestoreSettings.Error" = ""; + +/*Preview Mode*/ +"Gmod.Restricted" = "."; +"Gmod.Unavailable" = ""; +"Gmod.Disable.Notice" = ""; +"Gmod" = ""; +"Gmod.Enable" = "?"; +"Gmod.Disable" = "?"; +"Gmod.Notice" = ""; + +"SendWithKb" = ""; +"NiceFeatures.ShowGmodIcon" = ""; +"Gmod.OpenChatQ" = "?"; +"Gmod.OpenChatNotice" = ""; +"Gmod.OpenChatBtn" = ""; +"Gmod.DisableBtn" = ""; + +"NicegramSettings.Other.hidePhoneInSettingsNotice" = "."; +"NicegramSettings.Other.showProfileId" = ""; +"NicegramSettings.Other.showRegDate" = ""; +"NicegramSettings.Unblock.Header" = ""; "DoubleBottom.Description" = "To enable this feature, you should have more than one account and switch on Passcode Lock (go to Settings → Privacy & Security → Passcode Lock)"; "DoubleBottom.Enabled.Title" = "Double Bottom is enabled"; "DoubleBottom.Enabled.Description" = "Please remember the passcode you’ve just set and restart the app for Double Bottom to work well"; diff --git a/Telegram/Telegram-iOS/zh-hans.lproj/NiceLocalizable.strings b/Telegram/Telegram-iOS/zh-hans.lproj/NiceLocalizable.strings index 02b08e3b734..bff18cd0737 100644 --- a/Telegram/Telegram-iOS/zh-hans.lproj/NiceLocalizable.strings +++ b/Telegram/Telegram-iOS/zh-hans.lproj/NiceLocalizable.strings @@ -162,8 +162,6 @@ "NicegramSettings.Other.hideReactions" = "隐藏反应"; "NicegramSettings.RoundVideos.DownloadVideos" = "将视频下载到图库"; "Premium.OnetapTranslate.UseIgnoreLanguages" = "忽略语言"; -"NicegramSettings.ShareChannelsInfoToggle" = "分享频道信息"; -"NicegramSettings.ShareChannelsInfoToggle.Note" = "通过自动提交频道信息和链接至我们的数据库,向最全面的 Telegram 频道和群组百科贡献力量。我们不会关联您的 Telegram 个人资料和频道,不会分享您的个人数据。"; "Premium.OnetapTranslate.UseIgnoreLanguages.Note" = "应用会检测您设备上每条消息使用的语言。这一功能需要很高的性能,可能会影响电池寿命。"; "Premium.rememberFolderOnExit" = "退出时记住当前分组"; "NicegramSettings.HideStories" = "隐藏故事"; @@ -221,3 +219,9 @@ "ChatContextMenu.Hide" = "隐藏"; "ChatContextMenu.Unhide" = "取消隐藏"; "HiddenChatsTooltip" = "长按 'N' 标志以显示隐藏的秘密聊天"; + +/*Data Sharing*/ +"NicegramSettings.ShareBotsToggle" = "分享机器人信息"; +"NicegramSettings.ShareChannelsToggle" = "分享频道信息"; +"NicegramSettings.ShareStickersToggle" = "分享贴纸信息"; +"NicegramSettings.ShareData.Note" = "通过自动提交频道信息和链接到我们的数据库,为最全面的Telegram频道和群组维基百科做出贡献。我们不会连接您的Telegram个人资料和频道,也不会分享您的个人数据。"; diff --git a/Telegram/Telegram-iOS/zh-hant.lproj/NiceLocalizable.strings b/Telegram/Telegram-iOS/zh-hant.lproj/NiceLocalizable.strings index f1137df846b..78c1b13372e 100644 --- a/Telegram/Telegram-iOS/zh-hant.lproj/NiceLocalizable.strings +++ b/Telegram/Telegram-iOS/zh-hant.lproj/NiceLocalizable.strings @@ -162,8 +162,6 @@ "NicegramSettings.Other.hideReactions" = "隱藏心情回應"; "NicegramSettings.RoundVideos.DownloadVideos" = "將影片下載至圖片庫"; "Premium.OnetapTranslate.UseIgnoreLanguages" = "忽略語言"; -"NicegramSettings.ShareChannelsInfoToggle" = "分享頻道資訊"; -"NicegramSettings.ShareChannelsInfoToggle.Note" = "將頻道資訊與連結自動提交給我們的資料庫,來為最全面的 Telegram 頻道與群組維基百科做出貢獻。我們不會連接您的 Telegram 個人檔案與頻道,也不會分享您的個人資料。"; "Premium.OnetapTranslate.UseIgnoreLanguages.Note" = "App 會檢測您設備上每則訊息的語言。這是一項高耗電的功能,會影響您的電池壽命。"; "Premium.rememberFolderOnExit" = "記下應用程式關閉時所在的資料夾"; "NicegramSettings.HideStories" = "隱藏故事"; @@ -221,3 +219,9 @@ "ChatContextMenu.Hide" = "隱藏"; "ChatContextMenu.Unhide" = "取消隱藏"; "HiddenChatsTooltip" = "長按 'N' 標誌以顯示隱藏的秘密聊天"; + +/*Data Sharing*/ +"NicegramSettings.ShareBotsToggle" = "分享機器人資訊"; +"NicegramSettings.ShareChannelsToggle" = "分享頻道資訊"; +"NicegramSettings.ShareStickersToggle" = "分享貼圖資訊"; +"NicegramSettings.ShareData.Note" = "通過自動提交頻道資訊和連結到我們的數據庫,貢獻給最全面的Telegram頻道和群組維基百科。我們不會連接您的Telegram個人檔案和頻道,也不會分享您的個人資料。"; diff --git a/submodules/AccountContext/Sources/AccountContext.swift b/submodules/AccountContext/Sources/AccountContext.swift index d1987030db6..db396754e82 100644 --- a/submodules/AccountContext/Sources/AccountContext.swift +++ b/submodules/AccountContext/Sources/AccountContext.swift @@ -395,13 +395,13 @@ public enum ChatSearchDomain: Equatable { public enum ChatLocation: Equatable { case peer(id: PeerId) case replyThread(message: ChatReplyThreadMessage) - case feed(id: Int32) + case customChatContents } public extension ChatLocation { var normalized: ChatLocation { switch self { - case .peer, .feed: + case .peer, .customChatContents: return self case let .replyThread(message): return .replyThread(message: message.normalized) @@ -851,6 +851,15 @@ public protocol TelegramRootControllerInterface: NavigationController { func openSettings() } +public protocol QuickReplySetupScreenInitialData: AnyObject { +} + +public protocol AutomaticBusinessMessageSetupScreenInitialData: AnyObject { +} + +public protocol ChatbotSetupScreenInitialData: AnyObject { +} + public protocol SharedAccountContext: AnyObject { var sharedContainerPath: String { get } var basePath: String { get } @@ -938,8 +947,16 @@ public protocol SharedAccountContext: AnyObject { func makeHashtagSearchController(context: AccountContext, peer: EnginePeer?, query: String, all: Bool) -> ViewController func makeMyStoriesController(context: AccountContext, isArchive: Bool) -> ViewController func makeArchiveSettingsController(context: AccountContext) -> ViewController + func makeFilterSettingsController(context: AccountContext, modal: Bool, scrollToTags: Bool, dismissed: (() -> Void)?) -> ViewController func makeBusinessSetupScreen(context: AccountContext) -> ViewController - func makeChatbotSetupScreen(context: AccountContext) -> ViewController + func makeChatbotSetupScreen(context: AccountContext, initialData: ChatbotSetupScreenInitialData) -> ViewController + func makeChatbotSetupScreenInitialData(context: AccountContext) -> Signal + func makeBusinessLocationSetupScreen(context: AccountContext, initialValue: TelegramBusinessLocation?, completion: @escaping (TelegramBusinessLocation?) -> Void) -> ViewController + func makeBusinessHoursSetupScreen(context: AccountContext, initialValue: TelegramBusinessHours?, completion: @escaping (TelegramBusinessHours?) -> Void) -> ViewController + func makeAutomaticBusinessMessageSetupScreen(context: AccountContext, initialData: AutomaticBusinessMessageSetupScreenInitialData, isAwayMode: Bool) -> ViewController + func makeAutomaticBusinessMessageSetupScreenInitialData(context: AccountContext) -> Signal + func makeQuickReplySetupScreen(context: AccountContext, initialData: QuickReplySetupScreenInitialData) -> ViewController + func makeQuickReplySetupScreenInitialData(context: AccountContext) -> Signal func navigateToChatController(_ params: NavigateToChatControllerParams) func navigateToForumChannel(context: AccountContext, peerId: EnginePeer.Id, navigationController: NavigationController) func navigateToForumThread(context: AccountContext, peerId: EnginePeer.Id, threadId: Int64, messageId: EngineMessage.Id?, navigationController: NavigationController, activateInput: ChatControllerActivateInput?, keepStack: NavigateToChatKeepStack) -> Signal diff --git a/submodules/AccountContext/Sources/ChatController.swift b/submodules/AccountContext/Sources/ChatController.swift index 8773f184871..f8941880428 100644 --- a/submodules/AccountContext/Sources/ChatController.swift +++ b/submodules/AccountContext/Sources/ChatController.swift @@ -744,6 +744,7 @@ public enum ChatControllerSubject: Equatable { case scheduledMessages case pinnedMessages(id: EngineMessage.Id?) case messageOptions(peerIds: [EnginePeer.Id], ids: [EngineMessage.Id], info: MessageOptionsInfo) + case customChatContents(contents: ChatCustomContentsProtocol) public static func ==(lhs: ChatControllerSubject, rhs: ChatControllerSubject) -> Bool { switch lhs { @@ -771,6 +772,12 @@ public enum ChatControllerSubject: Equatable { } else { return false } + case let .customChatContents(lhsValue): + if case let .customChatContents(rhsValue) = rhs, lhsValue === rhsValue { + return true + } else { + return false + } } } @@ -796,6 +803,25 @@ public enum ChatControllerPresentationMode: Equatable { case inline(NavigationController?) } +public enum ChatInputTextCommand: Equatable { + case command(PeerCommand) + case shortcut(ShortcutMessageList.Item) +} + +public struct ChatInputQueryCommandsResult: Equatable { + public var commands: [ChatInputTextCommand] + public var accountPeer: EnginePeer? + public var hasShortcuts: Bool + public var query: String + + public init(commands: [ChatInputTextCommand], accountPeer: EnginePeer?, hasShortcuts: Bool, query: String) { + self.commands = commands + self.accountPeer = accountPeer + self.hasShortcuts = hasShortcuts + self.query = query + } +} + public enum ChatPresentationInputQueryResult: Equatable { // MARK: Nicegram QuickReplies case quickReplies([String]) @@ -803,7 +829,7 @@ public enum ChatPresentationInputQueryResult: Equatable { case stickers([FoundStickerItem]) case hashtags([String]) case mentions([EnginePeer]) - case commands([PeerCommand]) + case commands(ChatInputQueryCommandsResult) case emojis([(String, TelegramMediaFile?, String)], NSRange) case contextRequestResult(EnginePeer?, ChatContextResultCollection?) @@ -1061,7 +1087,30 @@ public enum ChatHistoryListSource { } case `default` - case custom(messages: Signal<([Message], Int32, Bool), NoError>, messageId: MessageId, quote: Quote?, loadMore: (() -> Void)?) + case custom(messages: Signal<([Message], Int32, Bool), NoError>, messageId: MessageId?, quote: Quote?, loadMore: (() -> Void)?) + case customView(historyView: Signal<(MessageHistoryView, ViewUpdateType), NoError>) +} + +public enum ChatQuickReplyShortcutType { + case generic + case greeting + case away +} + +public enum ChatCustomContentsKind: Equatable { + case quickReplyMessageInput(shortcut: String, shortcutType: ChatQuickReplyShortcutType) +} + +public protocol ChatCustomContentsProtocol: AnyObject { + var kind: ChatCustomContentsKind { get } + var historyView: Signal<(MessageHistoryView, ViewUpdateType), NoError> { get } + var messageLimit: Int? { get } + + func enqueueMessages(messages: [EnqueueMessage]) + func deleteMessages(ids: [EngineMessage.Id]) + func editMessage(id: EngineMessage.Id, text: String, media: RequestEditMessageMedia, entities: TextEntitiesMessageAttribute?, webpagePreviewAttribute: WebpagePreviewMessageAttribute?, disableUrlPreview: Bool) + + func quickReplyUpdateShortcut(value: String) } public enum ChatHistoryListDisplayHeaders { @@ -1080,7 +1129,7 @@ public protocol ChatControllerInteractionProtocol: AnyObject { public enum ChatHistoryNodeHistoryState: Equatable { case loading - case loaded(isEmpty: Bool) + case loaded(isEmpty: Bool, hasReachedLimits: Bool) } public protocol ChatHistoryListNode: ListView { diff --git a/submodules/AccountContext/Sources/ContactMultiselectionController.swift b/submodules/AccountContext/Sources/ContactMultiselectionController.swift index 6d601745281..45cbfa89ff7 100644 --- a/submodules/AccountContext/Sources/ContactMultiselectionController.swift +++ b/submodules/AccountContext/Sources/ContactMultiselectionController.swift @@ -45,6 +45,7 @@ public enum ContactMultiselectionControllerMode { public var chatListFilters: [ChatListFilter]? public var displayAutoremoveTimeout: Bool public var displayPresence: Bool + public var onlyUsers: Bool public init( title: String, @@ -53,7 +54,8 @@ public enum ContactMultiselectionControllerMode { additionalCategories: ContactMultiselectionControllerAdditionalCategories?, chatListFilters: [ChatListFilter]?, displayAutoremoveTimeout: Bool = false, - displayPresence: Bool = false + displayPresence: Bool = false, + onlyUsers: Bool = false ) { self.title = title self.searchPlaceholder = searchPlaceholder @@ -62,6 +64,7 @@ public enum ContactMultiselectionControllerMode { self.chatListFilters = chatListFilters self.displayAutoremoveTimeout = displayAutoremoveTimeout self.displayPresence = displayPresence + self.onlyUsers = onlyUsers } } diff --git a/submodules/AccountContext/Sources/MediaManager.swift b/submodules/AccountContext/Sources/MediaManager.swift index 02cda6c109f..092fc4c52ba 100644 --- a/submodules/AccountContext/Sources/MediaManager.swift +++ b/submodules/AccountContext/Sources/MediaManager.swift @@ -36,8 +36,8 @@ public enum PeerMessagesPlaylistLocation: Equatable, SharedMediaPlaylistLocation return .peer(peerId) case let .replyThread(replyThreaMessage): return .peer(replyThreaMessage.peerId) - case let .feed(id): - return .feed(id) + case .customChatContents: + return .custom } case let .singleMessage(id): return .peer(id.peerId) diff --git a/submodules/AccountContext/Sources/PeerNameColors.swift b/submodules/AccountContext/Sources/PeerNameColors.swift index 0bca8e23461..1966a168ea3 100644 --- a/submodules/AccountContext/Sources/PeerNameColors.swift +++ b/submodules/AccountContext/Sources/PeerNameColors.swift @@ -80,6 +80,7 @@ public class PeerNameColors: Equatable { colors: defaultSingleColors, darkColors: [:], displayOrder: [5, 3, 1, 0, 2, 4, 6], + chatFolderTagDisplayOrder: [5, 3, 1, 0, 2, 4, 6], profileColors: [:], profileDarkColors: [:], profilePaletteColors: [:], @@ -97,6 +98,8 @@ public class PeerNameColors: Equatable { public let darkColors: [Int32: Colors] public let displayOrder: [Int32] + public let chatFolderTagDisplayOrder: [Int32] + public let profileColors: [Int32: Colors] public let profileDarkColors: [Int32: Colors] public let profilePaletteColors: [Int32: Colors] @@ -119,6 +122,16 @@ public class PeerNameColors: Equatable { } } + public func getChatFolderTag(_ color: PeerNameColor, dark: Bool = false) -> Colors { + if dark, let colors = self.darkColors[color.rawValue] { + return colors + } else if let colors = self.colors[color.rawValue] { + return colors + } else { + return PeerNameColors.defaultSingleColors[5]! + } + } + public func getProfile(_ color: PeerNameColor, dark: Bool = false, subject: Subject = .background) -> Colors { switch subject { case .background: @@ -152,6 +165,7 @@ public class PeerNameColors: Equatable { colors: [Int32: Colors], darkColors: [Int32: Colors], displayOrder: [Int32], + chatFolderTagDisplayOrder: [Int32], profileColors: [Int32: Colors], profileDarkColors: [Int32: Colors], profilePaletteColors: [Int32: Colors], @@ -166,6 +180,7 @@ public class PeerNameColors: Equatable { self.colors = colors self.darkColors = darkColors self.displayOrder = displayOrder + self.chatFolderTagDisplayOrder = chatFolderTagDisplayOrder self.profileColors = profileColors self.profileDarkColors = profileDarkColors self.profilePaletteColors = profilePaletteColors @@ -257,6 +272,7 @@ public class PeerNameColors: Equatable { colors: colors, darkColors: darkColors, displayOrder: displayOrder, + chatFolderTagDisplayOrder: PeerNameColors.defaultValue.chatFolderTagDisplayOrder, profileColors: profileColors, profileDarkColors: profileDarkColors, profilePaletteColors: profilePaletteColors, @@ -280,6 +296,9 @@ public class PeerNameColors: Equatable { if lhs.displayOrder != rhs.displayOrder { return false } + if lhs.chatFolderTagDisplayOrder != rhs.chatFolderTagDisplayOrder { + return false + } if lhs.profileColors != rhs.profileColors { return false } diff --git a/submodules/AccountContext/Sources/Premium.swift b/submodules/AccountContext/Sources/Premium.swift index a8a5e153852..c327bc55978 100644 --- a/submodules/AccountContext/Sources/Premium.swift +++ b/submodules/AccountContext/Sources/Premium.swift @@ -38,6 +38,7 @@ public enum PremiumIntroSource { case presence case readTime case messageTags + case folderTags } public enum PremiumGiftSource: Equatable { @@ -70,6 +71,14 @@ public enum PremiumDemoSubject { case messageTags case lastSeen case messagePrivacy + case folderTags + + case businessLocation + case businessHours + case businessGreetingMessage + case businessQuickReplies + case businessAwayMessage + case businessChatBots } public enum PremiumLimitSubject { diff --git a/submodules/AttachmentUI/Sources/AttachmentController.swift b/submodules/AttachmentUI/Sources/AttachmentController.swift index 0fe0add1fa6..f1ec3c9d1f7 100644 --- a/submodules/AttachmentUI/Sources/AttachmentController.swift +++ b/submodules/AttachmentUI/Sources/AttachmentController.swift @@ -19,6 +19,7 @@ public enum AttachmentButtonType: Equatable { case gallery case file case location + case quickReply case contact case poll case app(AttachMenuBot) @@ -27,54 +28,60 @@ public enum AttachmentButtonType: Equatable { public static func ==(lhs: AttachmentButtonType, rhs: AttachmentButtonType) -> Bool { switch lhs { - case .gallery: - if case .gallery = rhs { - return true - } else { - return false - } - case .file: - if case .file = rhs { - return true - } else { - return false - } - case .location: - if case .location = rhs { - return true - } else { - return false - } - case .contact: - if case .contact = rhs { - return true - } else { - return false - } - case .poll: - if case .poll = rhs { - return true - } else { - return false - } - case let .app(lhsBot): - if case let .app(rhsBot) = rhs, lhsBot.peer.id == rhsBot.peer.id { - return true - } else { - return false - } - case .gift: - if case .gift = rhs { - return true - } else { - return false - } - case .standalone: - if case .standalone = rhs { - return true - } else { - return false - } + case .gallery: + if case .gallery = rhs { + return true + } else { + return false + } + case .file: + if case .file = rhs { + return true + } else { + return false + } + case .location: + if case .location = rhs { + return true + } else { + return false + } + case .quickReply: + if case .quickReply = rhs { + return true + } else { + return false + } + case .contact: + if case .contact = rhs { + return true + } else { + return false + } + case .poll: + if case .poll = rhs { + return true + } else { + return false + } + case let .app(lhsBot): + if case let .app(rhsBot) = rhs, lhsBot.peer.id == rhsBot.peer.id { + return true + } else { + return false + } + case .gift: + if case .gift = rhs { + return true + } else { + return false + } + case .standalone: + if case .standalone = rhs { + return true + } else { + return false + } } } } diff --git a/submodules/AttachmentUI/Sources/AttachmentPanel.swift b/submodules/AttachmentUI/Sources/AttachmentPanel.swift index e8c6db3c67b..6e9f7ef17d6 100644 --- a/submodules/AttachmentUI/Sources/AttachmentPanel.swift +++ b/submodules/AttachmentUI/Sources/AttachmentPanel.swift @@ -217,6 +217,9 @@ private final class AttachButtonComponent: CombinedComponent { name = "" imageName = "" imageFile = nil + case .quickReply: + name = strings.Attachment_Reply + imageName = "Chat/Attach Menu/Reply" } let tintColor = component.isSelected ? component.theme.rootController.tabBar.selectedIconColor : component.theme.rootController.tabBar.iconColor @@ -824,6 +827,8 @@ final class AttachmentPanel: ASDisplayNode, UIScrollViewDelegate { }, sendContextResult: { _, _, _, _ in return false }, sendBotCommand: { _, _ in + }, sendShortcut: { _ in + }, openEditShortcuts: { }, sendBotStart: { _ in }, botSwitchChatWithPayload: { _, _ in }, beginMediaRecording: { _ in @@ -1184,6 +1189,8 @@ final class AttachmentPanel: ASDisplayNode, UIScrollViewDelegate { accessibilityTitle = bot.shortName case .standalone: accessibilityTitle = "" + case .quickReply: + accessibilityTitle = self.presentationData.strings.Attachment_Reply } buttonView.isAccessibilityElement = true buttonView.accessibilityLabel = accessibilityTitle diff --git a/submodules/AuthorizationUI/Sources/AuthorizationSequenceController.swift b/submodules/AuthorizationUI/Sources/AuthorizationSequenceController.swift index bdc89f11928..dd9f8e0154c 100644 --- a/submodules/AuthorizationUI/Sources/AuthorizationSequenceController.swift +++ b/submodules/AuthorizationUI/Sources/AuthorizationSequenceController.swift @@ -1305,18 +1305,9 @@ public final class AuthorizationSequenceController: NavigationController, ASAuth } public static func defaultCountryCode() -> Int32 { - var countryId: String? = nil - let networkInfo = CTTelephonyNetworkInfo() - if let carrier = networkInfo.serviceSubscriberCellularProviders?.values.first { - countryId = carrier.isoCountryCode - } - - if countryId == nil { - countryId = (Locale.current as NSLocale).object(forKey: .countryCode) as? String - } - + let countryId = (Locale.current as NSLocale).object(forKey: .countryCode) as? String + var countryCode: Int32 = 1 - if let countryId = countryId { let normalizedId = countryId.uppercased() for (code, idAndName) in countryCodeToIdAndName { diff --git a/submodules/BrowserUI/Sources/BrowserScreen.swift b/submodules/BrowserUI/Sources/BrowserScreen.swift index 95e06060e34..1bfe441e3b3 100644 --- a/submodules/BrowserUI/Sources/BrowserScreen.swift +++ b/submodules/BrowserUI/Sources/BrowserScreen.swift @@ -618,6 +618,7 @@ public class BrowserScreen: ViewController { bottom: layout.intrinsicInsets.bottom + layout.safeInsets.bottom, right: layout.safeInsets.right ), + additionalInsets: layout.additionalInsets, inputHeight: layout.inputHeight ?? 0.0, metrics: layout.metrics, deviceMetrics: layout.deviceMetrics, diff --git a/submodules/ChatListUI/BUILD b/submodules/ChatListUI/BUILD index 0efe2ce08ad..da10aaf47c4 100644 --- a/submodules/ChatListUI/BUILD +++ b/submodules/ChatListUI/BUILD @@ -118,6 +118,7 @@ swift_library( "//submodules/TelegramUI/Components/Settings/ArchiveInfoScreen", "//submodules/TelegramUI/Components/Settings/NewSessionInfoScreen", "//submodules/TelegramUI/Components/Settings/PeerNameColorItem", + "//submodules/Components/MultilineTextComponent", ], visibility = [ "//visibility:public", diff --git a/submodules/ChatListUI/Sources/ChatListController.swift b/submodules/ChatListUI/Sources/ChatListController.swift index 223096241e8..6133450f963 100644 --- a/submodules/ChatListUI/Sources/ChatListController.swift +++ b/submodules/ChatListUI/Sources/ChatListController.swift @@ -5782,9 +5782,10 @@ public class ChatListControllerImpl: TelegramBaseController, ChatListController private func openFilterSettings() { self.chatListDisplayNode.mainContainerNode.updateEnableAdjacentFilterLoading(false) if let navigationController = self.context.sharedContext.mainWindow?.viewController as? NavigationController { - navigationController.pushViewController(chatListFilterPresetListController(context: self.context, mode: .modal, dismissed: { [weak self] in + let controller = self.context.sharedContext.makeFilterSettingsController(context: self.context, modal: true, scrollToTags: false, dismissed: { [weak self] in self?.chatListDisplayNode.mainContainerNode.updateEnableAdjacentFilterLoading(true) - })) + }) + navigationController.pushViewController(controller) } } diff --git a/submodules/ChatListUI/Sources/ChatListControllerNode.swift b/submodules/ChatListUI/Sources/ChatListControllerNode.swift index 5bb05f7d6f1..9abab9038d3 100644 --- a/submodules/ChatListUI/Sources/ChatListControllerNode.swift +++ b/submodules/ChatListUI/Sources/ChatListControllerNode.swift @@ -1342,6 +1342,7 @@ final class ChatListControllerNode: ASDisplayNode, UIGestureRecognizerDelegate { statusBarHeight: layout.statusBarHeight ?? 0.0, sideInset: layout.safeInsets.left, isSearchActive: self.isSearchDisplayControllerActive, + isSearchEnabled: true, primaryContent: headerContent?.primaryContent, secondaryContent: headerContent?.secondaryContent, secondaryTransition: self.inlineStackContainerTransitionFraction, diff --git a/submodules/ChatListUI/Sources/ChatListFilterPresetController.swift b/submodules/ChatListUI/Sources/ChatListFilterPresetController.swift index 3e6973ec32d..662f8cfe9cc 100644 --- a/submodules/ChatListUI/Sources/ChatListFilterPresetController.swift +++ b/submodules/ChatListUI/Sources/ChatListFilterPresetController.swift @@ -44,6 +44,7 @@ private final class ChatListFilterPresetControllerArguments { let linkContextAction: (ExportedChatFolderLink?, ASDisplayNode, ContextGesture?) -> Void let peerContextAction: (EnginePeer, ASDisplayNode, ContextGesture?, CGPoint?) -> Void let updateTagColor: (PeerNameColor?) -> Void + let openTagColorPremium: () -> Void init( context: AccountContext, @@ -63,7 +64,8 @@ private final class ChatListFilterPresetControllerArguments { removeLink: @escaping (ExportedChatFolderLink) -> Void, linkContextAction: @escaping (ExportedChatFolderLink?, ASDisplayNode, ContextGesture?) -> Void, peerContextAction: @escaping (EnginePeer, ASDisplayNode, ContextGesture?, CGPoint?) -> Void, - updateTagColor: @escaping (PeerNameColor?) -> Void + updateTagColor: @escaping (PeerNameColor?) -> Void, + openTagColorPremium: @escaping () -> Void ) { self.context = context self.updateState = updateState @@ -83,6 +85,7 @@ private final class ChatListFilterPresetControllerArguments { self.linkContextAction = linkContextAction self.peerContextAction = peerContextAction self.updateTagColor = updateTagColor + self.openTagColorPremium = openTagColorPremium } } @@ -231,8 +234,8 @@ private enum ChatListFilterPresetEntry: ItemListNodeEntry { case inviteLinkCreate(hasLinks: Bool) case inviteLink(Int, ExportedChatFolderLink) case inviteLinkInfo(text: String) - case tagColorHeader(name: String, color: PeerNameColors.Colors) - case tagColor(colors: PeerNameColors, currentColor: PeerNameColor?) + case tagColorHeader(name: String, color: PeerNameColors.Colors?, isPremium: Bool) + case tagColor(colors: PeerNameColors, currentColor: PeerNameColor?, isPremium: Bool) case tagColorFooter var section: ItemListSectionId { @@ -458,26 +461,43 @@ private enum ChatListFilterPresetEntry: ItemListNodeEntry { return ItemListPeerActionItem(presentationData: presentationData, icon: PresentationResourcesItemList.downArrowImage(presentationData.theme), title: text, sectionId: self.section, editing: false, action: { arguments.expandSection(.exclude) }) - case let .tagColorHeader(name, color): - //TODO:localize - return ItemListSectionHeaderItem(presentationData: presentationData, text: "FOLDER COLOR", badge: name.uppercased(), badgeStyle: ItemListSectionHeaderItem.BadgeStyle( - background: color.main.withMultipliedAlpha(0.1), - foreground: color.main - ), sectionId: self.section) - case let .tagColor(colors, color): + case let .tagColorHeader(name, color, isPremium): + var badge: String? + var badgeStyle: ItemListSectionHeaderItem.BadgeStyle? + var accessoryText: ItemListSectionHeaderAccessoryText? + if isPremium { + if let color { + badge = name.uppercased() + badgeStyle = ItemListSectionHeaderItem.BadgeStyle( + background: color.main.withMultipliedAlpha(0.1), + foreground: color.main + ) + } else { + accessoryText = ItemListSectionHeaderAccessoryText(value: presentationData.strings.ChatListFilter_TagLabelNoTag, color: .generic) + } + } else if color != nil { + accessoryText = ItemListSectionHeaderAccessoryText(value: presentationData.strings.ChatListFilter_TagLabelPremiumExpired, color: .generic) + } + return ItemListSectionHeaderItem(presentationData: presentationData, text: presentationData.strings.ChatListFilter_TagSectionTitle, badge: badge, badgeStyle: badgeStyle, accessoryText: accessoryText, sectionId: self.section) + case let .tagColor(colors, color, isPremium): return PeerNameColorItem( theme: presentationData.theme, colors: colors, - isProfile: true, - currentColor: color, + mode: .folderTag, + displayEmptyColor: true, + currentColor: isPremium ? color : nil, + isLocked: !isPremium, updated: { color in - arguments.updateTagColor(color) + if isPremium { + arguments.updateTagColor(color) + } else { + arguments.openTagColorPremium() + } }, sectionId: self.section ) case .tagColorFooter: - //TODO:localize - return ItemListTextItem(presentationData: presentationData, text: .plain("This color will be used for the folder's tag in the chat list"), sectionId: self.section) + return ItemListTextItem(presentationData: presentationData, text: .plain(presentationData.strings.ChatListFilter_TagSectionFooter), sectionId: self.section) case .inviteLinkHeader: return ItemListSectionHeaderItem(presentationData: presentationData, text: presentationData.strings.ChatListFilter_SectionShare, badge: nil, sectionId: self.section) case let .inviteLinkCreate(hasLinks): @@ -502,6 +522,7 @@ private struct ChatListFilterPresetControllerState: Equatable { var name: String var changedName: Bool var color: PeerNameColor? + var colorUpdated: Bool = false var includeCategories: ChatListFilterPeerCategories var excludeMuted: Bool var excludeRead: Bool @@ -616,12 +637,20 @@ private func chatListFilterPresetControllerEntries(context: AccountContext, pres entries.append(.excludePeerInfo(presentationData.strings.ChatListFolder_ExcludeSectionInfo)) } - /*let tagColor = state.color ?? .blue - let resolvedColor = context.peerNameColors.getProfile(tagColor, dark: presentationData.theme.overallDarkAppearance, subject: .palette) + let tagColor: PeerNameColor? + if state.colorUpdated { + tagColor = state.color + } else { + tagColor = state.color ?? .blue + } + var resolvedColor: PeerNameColors.Colors? + if let tagColor { + resolvedColor = context.peerNameColors.getChatFolderTag(tagColor, dark: presentationData.theme.overallDarkAppearance) + } - entries.append(.tagColorHeader(name: state.name, color: resolvedColor)) - entries.append(.tagColor(colors: context.peerNameColors, currentColor: tagColor)) - entries.append(.tagColorFooter)*/ + entries.append(.tagColorHeader(name: state.name, color: resolvedColor, isPremium: isPremium)) + entries.append(.tagColor(colors: context.peerNameColors, currentColor: tagColor, isPremium: isPremium)) + entries.append(.tagColorFooter) var hasLinks = false if let inviteLinks, !inviteLinks.isEmpty { @@ -1018,7 +1047,8 @@ func chatListFilterPresetController(context: AccountContext, currentPreset initi } else { initialName = "" } - let initialState = ChatListFilterPresetControllerState(name: initialName, changedName: initialPreset != nil, color: initialPreset?.data?.color, includeCategories: initialPreset?.data?.categories ?? [], excludeMuted: initialPreset?.data?.excludeMuted ?? false, excludeRead: initialPreset?.data?.excludeRead ?? false, excludeArchived: initialPreset?.data?.excludeArchived ?? false, additionallyIncludePeers: initialPreset?.data?.includePeers.peers ?? [], additionallyExcludePeers: initialPreset?.data?.excludePeers ?? [], expandedSections: []) + var initialState = ChatListFilterPresetControllerState(name: initialName, changedName: initialPreset != nil, color: initialPreset?.data?.color, includeCategories: initialPreset?.data?.categories ?? [], excludeMuted: initialPreset?.data?.excludeMuted ?? false, excludeRead: initialPreset?.data?.excludeRead ?? false, excludeArchived: initialPreset?.data?.excludeArchived ?? false, additionallyIncludePeers: initialPreset?.data?.includePeers.peers ?? [], additionallyExcludePeers: initialPreset?.data?.excludePeers ?? [], expandedSections: []) + initialState.colorUpdated = true let updatedCurrentPreset: Signal if let initialPreset { @@ -1543,8 +1573,20 @@ func chatListFilterPresetController(context: AccountContext, currentPreset initi updateState { state in var state = state state.color = color + state.colorUpdated = true return state } + }, + openTagColorPremium: { + var replaceImpl: ((ViewController) -> Void)? + let controller = context.sharedContext.makePremiumDemoController(context: context, subject: .folderTags, action: { + let controller = context.sharedContext.makePremiumIntroController(context: context, source: .folderTags, forceDark: false, dismissed: nil) + replaceImpl?(controller) + }) + replaceImpl = { [weak controller] c in + controller?.replace(with: c) + } + pushControllerImpl?(controller) } ) diff --git a/submodules/ChatListUI/Sources/ChatListFilterPresetListController.swift b/submodules/ChatListUI/Sources/ChatListFilterPresetListController.swift index c41c81ded51..c398146a213 100644 --- a/submodules/ChatListUI/Sources/ChatListFilterPresetListController.swift +++ b/submodules/ChatListUI/Sources/ChatListFilterPresetListController.swift @@ -22,8 +22,9 @@ private final class ChatListFilterPresetListControllerArguments { let setItemWithRevealedOptions: (Int32?, Int32?) -> Void let removePreset: (Int32) -> Void let updateDisplayTags: (Bool) -> Void + let updateDisplayTagsLocked: () -> Void - init(context: AccountContext, addSuggestedPressed: @escaping (String, ChatListFilterData) -> Void, openPreset: @escaping (ChatListFilter) -> Void, addNew: @escaping () -> Void, setItemWithRevealedOptions: @escaping (Int32?, Int32?) -> Void, removePreset: @escaping (Int32) -> Void, updateDisplayTags: @escaping (Bool) -> Void) { + init(context: AccountContext, addSuggestedPressed: @escaping (String, ChatListFilterData) -> Void, openPreset: @escaping (ChatListFilter) -> Void, addNew: @escaping () -> Void, setItemWithRevealedOptions: @escaping (Int32?, Int32?) -> Void, removePreset: @escaping (Int32) -> Void, updateDisplayTags: @escaping (Bool) -> Void, updateDisplayTagsLocked: @escaping () -> Void) { self.context = context self.addSuggestedPressed = addSuggestedPressed self.openPreset = openPreset @@ -31,6 +32,7 @@ private final class ChatListFilterPresetListControllerArguments { self.setItemWithRevealedOptions = setItemWithRevealedOptions self.removePreset = removePreset self.updateDisplayTags = updateDisplayTags + self.updateDisplayTagsLocked = updateDisplayTagsLocked } } @@ -41,6 +43,19 @@ private enum ChatListFilterPresetListSection: Int32 { case tags } +public enum ChatListFilterPresetListEntryTag: ItemListItemTag { + case displayTags + + public func isEqual(to other: ItemListItemTag) -> Bool { + if let other = other as? ChatListFilterPresetListEntryTag, self == other { + return true + } else { + return false + } + } +} + + private func stringForUserCount(_ peers: [EnginePeer.Id: SelectivePrivacyPeer], strings: PresentationStrings) -> String { if peers.isEmpty { return strings.PrivacyLastSeenSettings_EmpryUsersPlaceholder @@ -80,10 +95,10 @@ private enum ChatListFilterPresetListEntry: ItemListNodeEntry { case suggestedPreset(index: PresetIndex, title: String, label: String, preset: ChatListFilterData) case suggestedAddCustom(String) case listHeader(String) - case preset(index: PresetIndex, title: String, label: String, preset: ChatListFilter, canBeReordered: Bool, canBeDeleted: Bool, isEditing: Bool, isAllChats: Bool, isDisabled: Bool) + case preset(index: PresetIndex, title: String, label: String, preset: ChatListFilter, canBeReordered: Bool, canBeDeleted: Bool, isEditing: Bool, isAllChats: Bool, isDisabled: Bool, displayTags: Bool) case addItem(text: String, isEditing: Bool) case listFooter(String) - case displayTags(Bool) + case displayTags(Bool?) case displayTagsFooter var section: ItemListSectionId { @@ -107,7 +122,7 @@ private enum ChatListFilterPresetListEntry: ItemListNodeEntry { return 100 case .addItem: return 101 - case let .preset(index, _, _, _, _, _, _, _, _): + case let .preset(index, _, _, _, _, _, _, _, _, _): return 102 + index.value case .listFooter: return 1001 @@ -136,7 +151,7 @@ private enum ChatListFilterPresetListEntry: ItemListNodeEntry { return .suggestedAddCustom case .listHeader: return .listHeader - case let .preset(_, _, _, preset, _, _, _, _, _): + case let .preset(_, _, _, preset, _, _, _, _, _, _): return .preset(preset.id) case .addItem: return .addItem @@ -170,8 +185,16 @@ private enum ChatListFilterPresetListEntry: ItemListNodeEntry { }) case let .listHeader(text): return ItemListSectionHeaderItem(presentationData: presentationData, text: text, multiline: true, sectionId: self.section) - case let .preset(_, title, label, preset, canBeReordered, canBeDeleted, isEditing, isAllChats, isDisabled): - return ChatListFilterPresetListItem(presentationData: presentationData, preset: preset, title: title, label: label, editing: ChatListFilterPresetListItemEditing(editable: true, editing: isEditing, revealed: false), canBeReordered: canBeReordered, canBeDeleted: canBeDeleted, isAllChats: isAllChats, isDisabled: isDisabled, sectionId: self.section, action: { + case let .preset(_, title, label, preset, canBeReordered, canBeDeleted, isEditing, isAllChats, isDisabled, displayTags): + var resolvedColor: UIColor? + if displayTags, case let .filter(_, _, _, data) = preset { + let tagColor = data.color + if let tagColor { + resolvedColor = arguments.context.peerNameColors.getChatFolderTag(tagColor, dark: presentationData.theme.overallDarkAppearance).main + } + } + + return ChatListFilterPresetListItem(presentationData: presentationData, preset: preset, title: title, label: label, tagColor: resolvedColor, editing: ChatListFilterPresetListItemEditing(editable: true, editing: isEditing, revealed: false), canBeReordered: canBeReordered, canBeDeleted: canBeDeleted, isAllChats: isAllChats, isDisabled: isDisabled, sectionId: self.section, action: { if isDisabled { arguments.addNew() } else { @@ -189,13 +212,17 @@ private enum ChatListFilterPresetListEntry: ItemListNodeEntry { case let .listFooter(text): return ItemListTextItem(presentationData: presentationData, text: .plain(text), sectionId: self.section) case let .displayTags(value): - //TODO:localize - return ItemListSwitchItem(presentationData: presentationData, title: "Show Folder Tags", value: value, sectionId: self.section, style: .blocks, updated: { value in - arguments.updateDisplayTags(value) - }) + return ItemListSwitchItem(presentationData: presentationData, title: presentationData.strings.ChatListFilterList_ShowTags, value: value == true, enableInteractiveChanges: value != nil, enabled: true, displayLocked: value == nil, sectionId: self.section, style: .blocks, updated: { updatedValue in + if value != nil { + arguments.updateDisplayTags(updatedValue) + } else { + arguments.updateDisplayTagsLocked() + } + }, activatedWhileDisabled: { + arguments.updateDisplayTagsLocked() + }, tag: ChatListFilterPresetListEntryTag.displayTags) case .displayTagsFooter: - //TODO:localize - return ItemListTextItem(presentationData: presentationData, text: .plain("Display folder names for each chat in the chat list."), sectionId: self.section) + return ItemListTextItem(presentationData: presentationData, text: .plain(presentationData.strings.ChatListFilterList_ShowTagsFooter), sectionId: self.section) } } } @@ -249,15 +276,20 @@ private func chatListFilterPresetListControllerEntries(presentationData: Present entries.append(.addItem(text: presentationData.strings.ChatListFilterList_CreateFolder, isEditing: state.isEditing)) + var effectiveDisplayTags: Bool? + if isPremium { + effectiveDisplayTags = displayTags + } + if !filters.isEmpty || suggestedFilters.isEmpty { var folderCount = 0 for (filter, chatCount) in filtersWithAppliedOrder(filters: filters, order: updatedFilterOrder) { if case .allChats = filter { - entries.append(.preset(index: PresetIndex(value: entries.count), title: "", label: "", preset: filter, canBeReordered: filters.count > 1, canBeDeleted: false, isEditing: state.isEditing, isAllChats: true, isDisabled: false)) + entries.append(.preset(index: PresetIndex(value: entries.count), title: "", label: "", preset: filter, canBeReordered: filters.count > 1, canBeDeleted: false, isEditing: state.isEditing, isAllChats: true, isDisabled: false, displayTags: effectiveDisplayTags == true)) } if case let .filter(_, title, _, _) = filter { folderCount += 1 - entries.append(.preset(index: PresetIndex(value: entries.count), title: title, label: chatCount == 0 ? "" : "\(chatCount)", preset: filter, canBeReordered: filters.count > 1, canBeDeleted: true, isEditing: state.isEditing, isAllChats: false, isDisabled: !isPremium && folderCount > limits.maxFoldersCount)) + entries.append(.preset(index: PresetIndex(value: entries.count), title: title, label: chatCount == 0 ? "" : "\(chatCount)", preset: filter, canBeReordered: filters.count > 1, canBeDeleted: true, isEditing: state.isEditing, isAllChats: false, isDisabled: !isPremium && folderCount > limits.maxFoldersCount, displayTags: effectiveDisplayTags == true)) } } @@ -274,8 +306,8 @@ private func chatListFilterPresetListControllerEntries(presentationData: Present } } - /*entries.append(.displayTags(displayTags)) - entries.append(.displayTagsFooter)*/ + entries.append(.displayTags(effectiveDisplayTags)) + entries.append(.displayTagsFooter) return entries } @@ -285,7 +317,7 @@ public enum ChatListFilterPresetListControllerMode { case modal } -public func chatListFilterPresetListController(context: AccountContext, mode: ChatListFilterPresetListControllerMode, dismissed: (() -> Void)? = nil) -> ViewController { +public func chatListFilterPresetListController(context: AccountContext, mode: ChatListFilterPresetListControllerMode, scrollToTags: Bool = false, dismissed: (() -> Void)? = nil) -> ViewController { let initialState = ChatListFilterPresetListControllerState() let statePromise = ValuePromise(initialState, ignoreRepeated: true) let stateValue = Atomic(value: initialState) @@ -308,6 +340,8 @@ public func chatListFilterPresetListController(context: AccountContext, mode: Ch let filtersWithCounts = Promise<[(ChatListFilter, Int)]>() filtersWithCounts.set(filtersWithCountsSignal) + let animateNextShowHideTagsTransition = Atomic(value: nil) + let arguments = ChatListFilterPresetListControllerArguments(context: context, addSuggestedPressed: { title, data in let _ = combineLatest( @@ -523,6 +557,16 @@ public func chatListFilterPresetListController(context: AccountContext, mode: Ch }) }, updateDisplayTags: { value in context.engine.peers.updateChatListFiltersDisplayTags(isEnabled: value) + }, updateDisplayTagsLocked: { + var replaceImpl: ((ViewController) -> Void)? + let controller = context.sharedContext.makePremiumDemoController(context: context, subject: .folderTags, action: { + let controller = context.sharedContext.makePremiumIntroController(context: context, source: .folderTags, forceDark: false, dismissed: nil) + replaceImpl?(controller) + }) + replaceImpl = { [weak controller] c in + controller?.replace(with: c) + } + pushControllerImpl?(controller) }) let featuredFilters = context.account.postbox.preferencesView(keys: [PreferencesKeys.chatListFiltersFeaturedState]) @@ -538,6 +582,8 @@ public func chatListFilterPresetListController(context: AccountContext, mode: Ch let preferences = context.account.postbox.preferencesView(keys: [ApplicationSpecificPreferencesKeys.chatListFilterSettings]) + let previousDisplayTags = Atomic(value: nil) + let limits = context.engine.data.get( TelegramEngine.EngineData.Item.Configuration.UserLimits(isPremium: false), TelegramEngine.EngineData.Item.Configuration.UserLimits(isPremium: true) @@ -626,8 +672,14 @@ public func chatListFilterPresetListController(context: AccountContext, mode: Ch rightNavigationButton = nil } + let previousDisplayTagsValue = previousDisplayTags.swap(displayTags) + if let previousDisplayTagsValue, previousDisplayTagsValue != displayTags { + let _ = animateNextShowHideTagsTransition.swap(displayTags) + } + let controllerState = ItemListControllerState(presentationData: ItemListPresentationData(presentationData), title: .text(presentationData.strings.ChatListFolderSettings_Title), leftNavigationButton: leftNavigationButton, rightNavigationButton: rightNavigationButton, backNavigationButton: ItemListBackButton(title: presentationData.strings.Common_Back), animateChanges: false) - let listState = ItemListNodeState(presentationData: ItemListPresentationData(presentationData), entries: chatListFilterPresetListControllerEntries(presentationData: presentationData, state: state, filters: filtersWithCountsValue, updatedFilterOrder: updatedFilterOrderValue, suggestedFilters: suggestedFilters, displayTags: displayTags, isPremium: isPremium, limits: limits, premiumLimits: premiumLimits), style: .blocks, animateChanges: true) + let entries = chatListFilterPresetListControllerEntries(presentationData: presentationData, state: state, filters: filtersWithCountsValue, updatedFilterOrder: updatedFilterOrderValue, suggestedFilters: suggestedFilters, displayTags: displayTags, isPremium: isPremium, limits: limits, premiumLimits: premiumLimits) + let listState = ItemListNodeState(presentationData: ItemListPresentationData(presentationData), entries: entries, style: .blocks, initialScrollToItem: scrollToTags ? ListViewScrollToItem(index: entries.count - 1, position: .center(.bottom), animated: true, curve: .Spring(duration: 0.4), directionHint: .Down) : nil, animateChanges: true) return (controllerState, (listState, arguments)) } @@ -659,7 +711,7 @@ public func chatListFilterPresetListController(context: AccountContext, mode: Ch } controller.setReorderEntry({ (fromIndex: Int, toIndex: Int, entries: [ChatListFilterPresetListEntry]) -> Signal in let fromEntry = entries[fromIndex] - guard case let .preset(_, _, _, fromPreset, _, _, _, _, _) = fromEntry else { + guard case let .preset(_, _, _, fromPreset, _, _, _, _, _, _) = fromEntry else { return .single(false) } var referenceFilter: ChatListFilter? @@ -667,7 +719,7 @@ public func chatListFilterPresetListController(context: AccountContext, mode: Ch var afterAll = false if toIndex < entries.count { switch entries[toIndex] { - case let .preset(_, _, _, preset, _, _, _, _, _): + case let .preset(_, _, _, preset, _, _, _, _, _, _): referenceFilter = preset default: if entries[toIndex] < fromEntry { @@ -744,6 +796,29 @@ public func chatListFilterPresetListController(context: AccountContext, mode: Ch } }) }) + controller.afterTransactionCompleted = { [weak controller] in + guard let toggleDirection = animateNextShowHideTagsTransition.swap(nil) else { + return + } + + guard let controller else { + return + } + var presetItemNodes: [ChatListFilterPresetListItemNode] = [] + controller.forEachItemNode { itemNode in + if let itemNode = itemNode as? ChatListFilterPresetListItemNode { + presetItemNodes.append(itemNode) + } + } + + var delay: Double = 0.0 + for itemNode in presetItemNodes.reversed() { + if toggleDirection { + itemNode.animateTagColorIn(delay: delay) + } + delay += 0.02 + } + } return controller } diff --git a/submodules/ChatListUI/Sources/ChatListFilterPresetListItem.swift b/submodules/ChatListUI/Sources/ChatListFilterPresetListItem.swift index 609c9ae333e..4af495c8a96 100644 --- a/submodules/ChatListUI/Sources/ChatListFilterPresetListItem.swift +++ b/submodules/ChatListUI/Sources/ChatListFilterPresetListItem.swift @@ -19,6 +19,7 @@ final class ChatListFilterPresetListItem: ListViewItem, ItemListItem { let preset: ChatListFilter let title: String let label: String + let tagColor: UIColor? let editing: ChatListFilterPresetListItemEditing let canBeReordered: Bool let canBeDeleted: Bool @@ -34,6 +35,7 @@ final class ChatListFilterPresetListItem: ListViewItem, ItemListItem { preset: ChatListFilter, title: String, label: String, + tagColor: UIColor?, editing: ChatListFilterPresetListItemEditing, canBeReordered: Bool, canBeDeleted: Bool, @@ -48,6 +50,7 @@ final class ChatListFilterPresetListItem: ListViewItem, ItemListItem { self.preset = preset self.title = title self.label = label + self.tagColor = tagColor self.editing = editing self.canBeReordered = canBeReordered self.canBeDeleted = canBeDeleted @@ -109,7 +112,7 @@ final class ChatListFilterPresetListItem: ListViewItem, ItemListItem { private let titleFont = Font.regular(17.0) -private final class ChatListFilterPresetListItemNode: ItemListRevealOptionsItemNode { +final class ChatListFilterPresetListItemNode: ItemListRevealOptionsItemNode { private let backgroundNode: ASDisplayNode private let topStripeNode: ASDisplayNode private let bottomStripeNode: ASDisplayNode @@ -125,6 +128,7 @@ private final class ChatListFilterPresetListItemNode: ItemListRevealOptionsItemN private let labelNode: TextNode private let arrowNode: ASImageNode private let sharedIconNode: ASImageNode + private var tagIconView: UIImageView? private let activateArea: AccessibilityAreaNode @@ -173,7 +177,7 @@ private final class ChatListFilterPresetListItemNode: ItemListRevealOptionsItemN self.sharedIconNode.displayWithoutProcessing = true self.sharedIconNode.displaysAsynchronously = false self.sharedIconNode.isLayerBacked = true - + self.activateArea = AccessibilityAreaNode() self.highlightedBackgroundNode = ASDisplayNode() @@ -214,6 +218,7 @@ private final class ChatListFilterPresetListItemNode: ItemListRevealOptionsItemN } else { updateArrowImage = PresentationResourcesItemList.disclosureArrowImage(item.presentationData.theme) } + updatedSharedIconImage = generateTintedImage(image: UIImage(bundleImageName: "Chat List/SharedFolderListIcon"), color: item.presentationData.theme.list.disclosureArrowColor) } @@ -406,7 +411,15 @@ private final class ChatListFilterPresetListItemNode: ItemListRevealOptionsItemN strongSelf.arrowNode.isHidden = item.isAllChats if let sharedIconImage = strongSelf.sharedIconNode.image { - strongSelf.sharedIconNode.frame = CGRect(origin: CGPoint(x: strongSelf.arrowNode.frame.minX + 2.0 - sharedIconImage.size.width, y: floorToScreenPixels((layout.contentSize.height - sharedIconImage.size.height) / 2.0) + 1.0), size: sharedIconImage.size) + var sharedIconFrame = CGRect(origin: CGPoint(x: strongSelf.arrowNode.frame.minX + 2.0 - sharedIconImage.size.width, y: floorToScreenPixels((layout.contentSize.height - sharedIconImage.size.height) / 2.0) + 1.0), size: sharedIconImage.size) + if item.tagColor != nil { + sharedIconFrame.origin.x -= 34.0 + } + if strongSelf.sharedIconNode.bounds.isEmpty { + strongSelf.sharedIconNode.frame = sharedIconFrame + } else { + transition.updateFrame(node: strongSelf.sharedIconNode, frame: sharedIconFrame) + } } var isShared = false if case let .filter(_, _, _, data) = item.preset, data.isShared { @@ -414,6 +427,33 @@ private final class ChatListFilterPresetListItemNode: ItemListRevealOptionsItemN } strongSelf.sharedIconNode.isHidden = !isShared + if let tagColor = item.tagColor { + let tagIconView: UIImageView + var tagIconTransition = transition + if let current = strongSelf.tagIconView { + tagIconView = current + } else { + tagIconTransition = .immediate + tagIconView = UIImageView(image: generateStretchableFilledCircleImage(diameter: 24.0, color: .white)?.withRenderingMode(.alwaysTemplate)) + strongSelf.tagIconView = tagIconView + strongSelf.containerNode.view.addSubview(tagIconView) + } + tagIconView.tintColor = tagColor + + let tagIconFrame = CGRect(origin: CGPoint(x: strongSelf.arrowNode.frame.minX - 2.0 - 24.0, y: floorToScreenPixels((layout.contentSize.height - 24.0) / 2.0)), size: CGSize(width: 24.0, height: 24.0)) + + tagIconTransition.updateAlpha(layer: tagIconView.layer, alpha: reorderControlSizeAndApply != nil ? 0.0 : 1.0) + tagIconTransition.updateFrame(view: tagIconView, frame: tagIconFrame) + } else { + if let tagIconView = strongSelf.tagIconView { + strongSelf.tagIconView = nil + transition.updateAlpha(layer: tagIconView.layer, alpha: 0.0, completion: { [weak tagIconView] _ in + tagIconView?.removeFromSuperview() + }) + transition.updateTransformScale(layer: tagIconView.layer, scale: 0.001) + } + } + strongSelf.activateArea.frame = CGRect(origin: CGPoint(x: leftInset + revealOffset + editingOffset, y: 0.0), size: CGSize(width: params.width - params.rightInset - 56.0 - (leftInset + revealOffset + editingOffset), height: layout.contentSize.height)) strongSelf.highlightedBackgroundNode.frame = CGRect(origin: CGPoint(x: 0.0, y: -UIScreenPixel), size: CGSize(width: params.width, height: layout.contentSize.height + UIScreenPixel + UIScreenPixel)) @@ -427,6 +467,13 @@ private final class ChatListFilterPresetListItemNode: ItemListRevealOptionsItemN } } + func animateTagColorIn(delay: Double) { + if let tagIconView = self.tagIconView { + tagIconView.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.12, delay: delay) + tagIconView.layer.animateSpring(from: 0.001 as NSNumber, to: 1.0 as NSNumber, keyPath: "transform.scale", duration: 0.4, delay: delay) + } + } + override func setHighlighted(_ highlighted: Bool, at point: CGPoint, animated: Bool) { super.setHighlighted(highlighted, at: point, animated: animated) @@ -507,7 +554,16 @@ private final class ChatListFilterPresetListItemNode: ItemListRevealOptionsItemN var sharedIconFrame = self.sharedIconNode.frame sharedIconFrame.origin.x = arrowFrame.minX + 2.0 - sharedIconFrame.width + if self.item?.tagColor != nil { + sharedIconFrame.origin.x -= 34.0 + } transition.updateFrame(node: self.sharedIconNode, frame: sharedIconFrame) + + if let tagIconView = self.tagIconView { + var tagIconFrame = tagIconView.frame + tagIconFrame.origin.x = arrowFrame.minX - 2.0 - tagIconFrame.width + transition.updateFrame(view: tagIconView, frame: tagIconFrame) + } } override func revealOptionsInteractivelyOpened() { diff --git a/submodules/ChatListUI/Sources/ChatListSearchListPaneNode.swift b/submodules/ChatListUI/Sources/ChatListSearchListPaneNode.swift index 2a6b2cc4f5c..0ac6bf755af 100644 --- a/submodules/ChatListUI/Sources/ChatListSearchListPaneNode.swift +++ b/submodules/ChatListUI/Sources/ChatListSearchListPaneNode.swift @@ -2324,6 +2324,7 @@ final class ChatListSearchListPaneNode: ASDisplayNode, ChatListSearchPaneNode { self.interaction.openStories?(id, sourceNode.avatarNode) } }, dismissNotice: { _ in + }, editPeer: { _ in }) chatListInteraction.isSearchMode = true @@ -3689,6 +3690,7 @@ public final class ChatListSearchShimmerNode: ASDisplayNode { }, openChatFolderUpdates: {}, hideChatFolderUpdates: { }, openStories: { _, _ in }, dismissNotice: { _ in + }, editPeer: { _ in }) var isInlineMode = false if case .topics = key { diff --git a/submodules/ChatListUI/Sources/ChatListShimmerNode.swift b/submodules/ChatListUI/Sources/ChatListShimmerNode.swift index 760ec065918..109edbc4db1 100644 --- a/submodules/ChatListUI/Sources/ChatListShimmerNode.swift +++ b/submodules/ChatListUI/Sources/ChatListShimmerNode.swift @@ -157,6 +157,7 @@ public final class ChatListShimmerNode: ASDisplayNode { }, messageSelected: { _, _, _, _ in}, groupSelected: { _ in }, addContact: { _ in }, setPeerIdWithRevealedOptions: { _, _ in }, setItemPinned: { _, _ in }, setPeerMuted: { _, _ in }, setPeerThreadMuted: { _, _, _ in }, deletePeer: { _, _ in }, deletePeerThread: { _, _ in }, setPeerThreadStopped: { _, _, _ in }, setPeerThreadPinned: { _, _, _ in }, setPeerThreadHidden: { _, _, _ in }, updatePeerGrouping: { _, _ in }, togglePeerMarkedUnread: { _, _ in}, toggleArchivedFolderHiddenByDefault: {}, toggleThreadsSelection: { _, _ in }, hidePsa: { _ in }, activateChatPreview: { _, _, _, gesture, _ in gesture?.cancel() }, present: { _ in }, openForumThread: { _, _ in }, openStorageManagement: {}, openPasswordSetup: {}, openPremiumIntro: {}, openPremiumGift: {}, openActiveSessions: {}, performActiveSessionAction: { _, _ in }, openChatFolderUpdates: {}, hideChatFolderUpdates: {}, openStories: { _, _ in }, dismissNotice: { _ in + }, editPeer: { _ in }) interaction.isInlineMode = isInlineMode diff --git a/submodules/ChatListUI/Sources/Node/ChatListItem.swift b/submodules/ChatListUI/Sources/Node/ChatListItem.swift index 6949eac27f8..5b2ffa3c124 100644 --- a/submodules/ChatListUI/Sources/Node/ChatListItem.swift +++ b/submodules/ChatListUI/Sources/Node/ChatListItem.swift @@ -30,6 +30,8 @@ import TextNodeWithEntities import ComponentFlow import EmojiStatusComponent import AvatarVideoNode +import AppBundle +import MultilineTextComponent public enum ChatListItemContent { public struct ThreadInfo: Equatable { @@ -93,6 +95,20 @@ public enum ChatListItemContent { } } + public struct CustomMessageListData: Equatable { + public var commandPrefix: String? + public var searchQuery: String? + public var messageCount: Int? + public var hideSeparator: Bool + + public init(commandPrefix: String?, searchQuery: String?, messageCount: Int?, hideSeparator: Bool) { + self.commandPrefix = commandPrefix + self.searchQuery = searchQuery + self.messageCount = messageCount + self.hideSeparator = hideSeparator + } + } + public struct PeerData { public var messages: [EngineMessage] public var peer: EngineRenderedPeer @@ -116,6 +132,7 @@ public enum ChatListItemContent { public var requiresPremiumForMessaging: Bool public var displayAsTopicList: Bool public var tags: [Tag] + public var customMessageListData: CustomMessageListData? public init( messages: [EngineMessage], @@ -139,7 +156,8 @@ public enum ChatListItemContent { storyState: StoryState?, requiresPremiumForMessaging: Bool, displayAsTopicList: Bool, - tags: [Tag] + tags: [Tag], + customMessageListData: CustomMessageListData? = nil ) { self.messages = messages self.peer = peer @@ -163,6 +181,7 @@ public enum ChatListItemContent { self.requiresPremiumForMessaging = requiresPremiumForMessaging self.displayAsTopicList = displayAsTopicList self.tags = tags + self.customMessageListData = customMessageListData } } @@ -212,15 +231,18 @@ private final class ChatListItemTagListComponent: Component { let context: AccountContext let tags: [ChatListItemContent.Tag] let theme: PresentationTheme + let sizeFactor: CGFloat init( context: AccountContext, tags: [ChatListItemContent.Tag], - theme: PresentationTheme + theme: PresentationTheme, + sizeFactor: CGFloat ) { self.context = context self.tags = tags self.theme = theme + self.sizeFactor = sizeFactor } static func ==(lhs: ChatListItemTagListComponent, rhs: ChatListItemTagListComponent) -> Bool { @@ -233,6 +255,9 @@ private final class ChatListItemTagListComponent: Component { if lhs.theme !== rhs.theme { return false } + if lhs.sizeFactor != rhs.sizeFactor { + return false + } return true } @@ -252,16 +277,18 @@ private final class ChatListItemTagListComponent: Component { preconditionFailure() } - func update(context: AccountContext, title: String, backgroundColor: UIColor, foregroundColor: UIColor) -> CGSize { + func update(context: AccountContext, title: String, backgroundColor: UIColor, foregroundColor: UIColor, sizeFactor: CGFloat) -> CGSize { let titleSize = self.title.update( transition: .immediate, - component: AnyComponent(Text(text: title.isEmpty ? " " : title, font: Font.semibold(11.0), color: foregroundColor)), + component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: title.isEmpty ? " " : title, font: Font.semibold(floor(11.0 * sizeFactor)), textColor: foregroundColor)) + )), environment: {}, containerSize: CGSize(width: 100.0, height: 100.0) ) - let backgroundSideInset: CGFloat = 4.0 - let backgroundVerticalInset: CGFloat = 2.0 + let backgroundSideInset: CGFloat = floorToScreenPixels(4.0 * sizeFactor) + let backgroundVerticalInset: CGFloat = floorToScreenPixels(2.0 * sizeFactor) let backgroundSize = CGSize(width: titleSize.width + backgroundSideInset * 2.0, height: titleSize.height + backgroundVerticalInset * 2.0) let backgroundFrame = CGRect(origin: CGPoint(), size: backgroundSize) @@ -293,7 +320,7 @@ private final class ChatListItemTagListComponent: Component { func update(component: ChatListItemTagListComponent, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { var validIds: [Int32] = [] - let spacing: CGFloat = 5.0 + let spacing: CGFloat = floorToScreenPixels(5.0 * component.sizeFactor) var nextX: CGFloat = 0.0 for tag in component.tags { if nextX != 0.0 { @@ -314,7 +341,7 @@ private final class ChatListItemTagListComponent: Component { itemId = tag.id let tagColor = PeerNameColor(rawValue: tag.colorId) - let resolvedColor = component.context.peerNameColors.getProfile(tagColor, dark: component.theme.overallDarkAppearance, subject: .palette) + let resolvedColor = component.context.peerNameColors.getChatFolderTag(tagColor, dark: component.theme.overallDarkAppearance) itemTitle = tag.title.uppercased() itemBackgroundColor = resolvedColor.main.withMultipliedAlpha(0.1) @@ -330,7 +357,7 @@ private final class ChatListItemTagListComponent: Component { self.addSubview(itemView) } - let itemSize = itemView.update(context: component.context, title: itemTitle, backgroundColor: itemBackgroundColor, foregroundColor: itemForegroundColor) + let itemSize = itemView.update(context: component.context, title: itemTitle, backgroundColor: itemBackgroundColor, foregroundColor: itemForegroundColor, sizeFactor: component.sizeFactor) let itemFrame = CGRect(origin: CGPoint(x: nextX, y: 0.0), size: itemSize) itemView.frame = itemFrame @@ -566,6 +593,7 @@ private enum RevealOptionKey: Int32 { case hidePsa case open case close + case edit } private func canArchivePeer(id: EnginePeer.Id, accountPeerId: EnginePeer.Id) -> Bool { @@ -806,8 +834,8 @@ private let playIconImage = UIImage(bundleImageName: "Chat List/MiniThumbnailPla private final class ChatListMediaPreviewNode: ASDisplayNode { private let context: AccountContext - private let message: EngineMessage - private let media: EngineMedia + let message: EngineMessage + let media: EngineMedia private let imageNode: TransformImageNode private let playIcon: ASImageNode @@ -868,10 +896,14 @@ private final class ChatListMediaPreviewNode: ASDisplayNode { if file.isInstantVideo { isRound = true } - if file.isAnimated { + if file.isSticker || file.isAnimatedSticker { self.playIcon.isHidden = true } else { - self.playIcon.isHidden = false + if file.isAnimated { + self.playIcon.isHidden = true + } else { + self.playIcon.isHidden = false + } } if let mediaDimensions = file.dimensions { dimensions = mediaDimensions.cgSize @@ -883,9 +915,18 @@ private final class ChatListMediaPreviewNode: ASDisplayNode { } } + let radius: CGFloat + if isRound { + radius = size.width / 2.0 + } else if size.width >= 30.0 { + radius = 8.0 + } else { + radius = 2.0 + } + let makeLayout = self.imageNode.asyncLayout() self.imageNode.frame = CGRect(origin: CGPoint(), size: size) - let apply = makeLayout(TransformImageArguments(corners: ImageCorners(radius: isRound ? size.width / 2.0 : 2.0), imageSize: dimensions.aspectFilled(size), boundingSize: size, intrinsicInsets: UIEdgeInsets())) + let apply = makeLayout(TransformImageArguments(corners: ImageCorners(radius: radius), imageSize: dimensions.aspectFilled(size), boundingSize: size, intrinsicInsets: UIEdgeInsets())) apply() } } @@ -1148,6 +1189,18 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { } } + private struct ContentImageSpec { + var message: EngineMessage + var media: EngineMedia + var size: CGSize + + init(message: EngineMessage, media: EngineMedia, size: CGSize) { + self.message = message + self.media = media + self.size = size + } + } + var item: ChatListItem? private let backgroundNode: ASDisplayNode @@ -1162,6 +1215,7 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { var avatarIconComponent: EmojiStatusComponent? var avatarVideoNode: AvatarVideoNode? var avatarTapRecognizer: UITapGestureRecognizer? + private var avatarMediaNode: ChatListMediaPreviewNode? private var inlineNavigationMarkLayer: SimpleLayer? @@ -1174,10 +1228,13 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { private var currentItemHeight: CGFloat? let forwardedIconNode: ASImageNode let textNode: TextNodeWithEntities + var trailingTextBadgeNode: TextNode? + var trailingTextBadgeBackground: UIImageView? var dustNode: InvisibleInkDustNode? let inputActivitiesNode: ChatListInputActivitiesNode let dateNode: TextNode var dateStatusIconNode: ASImageNode? + var dateDisclosureIconView: UIImageView? let separatorNode: ASDisplayNode let statusNode: ChatListStatusNode let badgeNode: ChatListBadgeNode @@ -1199,7 +1256,7 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { private var cachedDataDisposable = MetaDisposable() private var currentTextLeftCutout: CGFloat = 0.0 - private var currentMediaPreviewSpecs: [(message: EngineMessage, media: EngineMedia, size: CGSize)] = [] + private var currentMediaPreviewSpecs: [ContentImageSpec] = [] private var mediaPreviewNodes: [EngineMedia.Id: ChatListMediaPreviewNode] = [:] var selectableControlNode: ItemListSelectableControlNode? @@ -1444,6 +1501,7 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { self.textNode = TextNodeWithEntities() self.textNode.textNode.isUserInteractionEnabled = false self.textNode.textNode.displaysAsynchronously = true + self.textNode.textNode.anchorPoint = CGPoint() self.inputActivitiesNode = ChatListInputActivitiesNode() self.inputActivitiesNode.isUserInteractionEnabled = false @@ -1624,7 +1682,8 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { if let peer = peer { var overrideImage: AvatarNodeImageOverride? - if peer.id.isReplies { + if case let .peer(peerData) = item.content, peerData.customMessageListData != nil { + } else if peer.id.isReplies { overrideImage = .repliesIcon } else if peer.id.isAnonymousSavedMessages { overrideImage = .anonymousSavedMessagesIcon @@ -1646,7 +1705,19 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { isForumAvatar = true } } + + var avatarDiameter = min(60.0, floor(item.presentationData.fontSize.baseDisplaySize * 60.0 / 17.0)) + + if case let .peer(peerData) = item.content, let customMessageListData = peerData.customMessageListData, customMessageListData.commandPrefix != nil { + avatarDiameter = 40.0 + } + if avatarDiameter != 60.0 { + let avatarFontSize = floor(avatarDiameter * 26.0 / 60.0) + if self.avatarNode.font.pointSize != avatarFontSize { + self.avatarNode.font = avatarPlaceholderFont(size: avatarFontSize) + } + } self.avatarNode.setPeer(context: item.context, theme: item.presentationData.theme, peer: peer, overrideImage: overrideImage, emptyColor: item.presentationData.theme.list.mediaPlaceholderColor, clipStyle: isForumAvatar ? .roundedRect : .round, synchronousLoad: synchronousLoads, displayDimensions: CGSize(width: 60.0, height: 60.0)) if peer.isPremium && peer.id != item.context.account.peerId { @@ -1842,6 +1913,7 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { func asyncLayout() -> (_ item: ChatListItem, _ params: ListViewItemLayoutParams, _ first: Bool, _ last: Bool, _ firstWithHeader: Bool, _ nextIsPinned: Bool) -> (ListViewItemNodeLayout, (Bool, Bool) -> Void) { let dateLayout = TextNode.asyncLayout(self.dateNode) let textLayout = TextNodeWithEntities.asyncLayout(self.textNode) + let makeTrailingTextBadgeLayout = TextNode.asyncLayout(self.trailingTextBadgeNode) let titleLayout = TextNode.asyncLayout(self.titleNode) let authorLayout = self.authorNode.asyncLayout() let makeMeasureLayout = TextNode.asyncLayout(self.measureNode) @@ -2045,26 +2117,41 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { let editingOffset: CGFloat var reorderInset: CGFloat = 0.0 if item.editing { - let sizeAndApply = selectableControlLayout(item.presentationData.theme.list.itemCheckColors.strokeColor, item.presentationData.theme.list.itemCheckColors.fillColor, item.presentationData.theme.list.itemCheckColors.foregroundColor, item.selected, true) + let selectionControlStyle: ItemListSelectableControlNode.Style + if case let .peer(peerData) = item.content, let customMessageListData = peerData.customMessageListData, customMessageListData.commandPrefix != nil { + selectionControlStyle = .small + } else { + selectionControlStyle = .compact + } + + let sizeAndApply = selectableControlLayout(item.presentationData.theme.list.itemCheckColors.strokeColor, item.presentationData.theme.list.itemCheckColors.fillColor, item.presentationData.theme.list.itemCheckColors.foregroundColor, item.selected, selectionControlStyle) if promoInfo == nil && !isPeerGroup { selectableControlSizeAndApply = sizeAndApply } editingOffset = sizeAndApply.0 + var canReorder = false + if case let .chatList(index) = item.index, index.pinningIndex != nil, promoInfo == nil, !isPeerGroup { - let sizeAndApply = reorderControlLayout(item.presentationData.theme) - reorderControlSizeAndApply = sizeAndApply - reorderInset = sizeAndApply.0 + canReorder = true } else if case let .forum(pinnedIndex, _, _, _, _) = item.index, case .index = pinnedIndex { if case let .chat(itemPeer) = contentPeer, case let .channel(channel) = itemPeer.peer { let canPin = channel.flags.contains(.isCreator) || channel.hasPermission(.pinMessages) if canPin { - let sizeAndApply = reorderControlLayout(item.presentationData.theme) - reorderControlSizeAndApply = sizeAndApply - reorderInset = sizeAndApply.0 + canReorder = true } } } + + if case let .peer(peerData) = item.content, let customMessageListData = peerData.customMessageListData, customMessageListData.commandPrefix != nil { + canReorder = true + } + + if canReorder { + let sizeAndApply = reorderControlLayout(item.presentationData.theme) + reorderControlSizeAndApply = sizeAndApply + reorderInset = sizeAndApply.0 + } } else { editingOffset = 0.0 } @@ -2076,15 +2163,21 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { let enableChatListPhotos = true - let avatarDiameter = min(60.0, floor(item.presentationData.fontSize.baseDisplaySize * 60.0 / 17.0)) - + // if changed, adjust setupItem accordingly + var avatarDiameter = min(60.0, floor(item.presentationData.fontSize.baseDisplaySize * 60.0 / 17.0)) let avatarLeftInset: CGFloat - if item.interaction.isInlineMode { - avatarLeftInset = 12.0 - } else if !useChatListLayout { - avatarLeftInset = 50.0 + + if case let .peer(peerData) = item.content, let customMessageListData = peerData.customMessageListData, customMessageListData.commandPrefix != nil { + avatarDiameter = 40.0 + avatarLeftInset = 17.0 + avatarDiameter } else { - avatarLeftInset = 18.0 + avatarDiameter + if item.interaction.isInlineMode { + avatarLeftInset = 12.0 + } else if !useChatListLayout { + avatarLeftInset = 50.0 + } else { + avatarLeftInset = 18.0 + avatarDiameter + } } let badgeDiameter = floor(item.presentationData.fontSize.baseDisplaySize * 20.0 / 17.0) @@ -2138,7 +2231,7 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { hideAuthor = true } - let attributedText: NSAttributedString + var attributedText: NSAttributedString var hasDraft = false var inlineAuthorPrefix: String? @@ -2147,13 +2240,23 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { useInlineAuthorPrefix = true } if !itemTags.isEmpty { - if case let .chat(peer) = contentPeer, peer.peerId == item.context.account.peerId { - } else { - useInlineAuthorPrefix = true - } - forumTopicData = nil topForumTopicItems = [] + + if case let .chat(itemPeer, _, _, _, _, _, _) = contentData { + if let messagePeer = itemPeer.chatMainPeer { + switch messagePeer { + case let .channel(channel): + if case .group = channel.info { + useInlineAuthorPrefix = true + } + case .legacyGroup: + useInlineAuthorPrefix = true + default: + break + } + } + } } if useInlineAuthorPrefix { @@ -2176,7 +2279,9 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { let contentImageSpacing: CGFloat = 2.0 let forwardedIconSpacing: CGFloat = 6.0 let contentImageTrailingSpace: CGFloat = 5.0 - var contentImageSpecs: [(message: EngineMessage, media: EngineMedia, size: CGSize)] = [] + + var contentImageSpecs: [ContentImageSpec] = [] + var avatarContentImageSpec: ContentImageSpec? var forumThread: (id: Int64, title: String, iconId: Int64?, iconColor: Int32, isUnread: Bool)? var displayForwardedIcon = false @@ -2279,20 +2384,27 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { hasDraft = true authorAttributedString = NSAttributedString(string: item.presentationData.strings.DialogList_Draft, font: textFont, textColor: theme.messageDraftTextColor) - //TODO:localize switch mediaDraftContentType { case .audio: - attributedText = NSAttributedString(string: "Voice Message", font: textFont, textColor: theme.messageTextColor) + attributedText = NSAttributedString(string: item.presentationData.strings.Message_Audio, font: textFont, textColor: theme.messageTextColor) case .video: - attributedText = NSAttributedString(string: "Video Message", font: textFont, textColor: theme.messageTextColor) + attributedText = NSAttributedString(string: item.presentationData.strings.Message_VideoMessage, font: textFont, textColor: theme.messageTextColor) } } else if inlineAuthorPrefix == nil, let draftState = draftState { hasDraft = true - authorAttributedString = NSAttributedString(string: item.presentationData.strings.DialogList_Draft, font: textFont, textColor: theme.messageDraftTextColor) - let draftText = stringWithAppliedEntities(draftState.text, entities: draftState.entities, baseColor: theme.messageTextColor, linkColor: theme.messageTextColor, baseFont: textFont, linkFont: textFont, boldFont: textFont, italicFont: textFont, boldItalicFont: textFont, fixedFont: textFont, blockQuoteFont: textFont, message: nil) - attributedText = foldLineBreaks(draftText) + if !itemTags.isEmpty { + let tempAttributedText = foldLineBreaks(draftText) + let attributedTextWithDraft = NSMutableAttributedString() + attributedTextWithDraft.append(NSAttributedString(string: item.presentationData.strings.DialogList_Draft + ": ", font: textFont, textColor: theme.messageDraftTextColor)) + attributedTextWithDraft.append(tempAttributedText) + attributedText = attributedTextWithDraft + } else { + authorAttributedString = NSAttributedString(string: item.presentationData.strings.DialogList_Draft, font: textFont, textColor: theme.messageDraftTextColor) + + attributedText = foldLineBreaks(draftText) + } } else if let message = messages.first { var composedString: NSMutableAttributedString @@ -2472,6 +2584,19 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { attributedText = composedString + if case let .peer(peerData) = item.content, let customMessageListData = peerData.customMessageListData, let commandPrefix = customMessageListData.commandPrefix { + let mutableAttributedText = NSMutableAttributedString(attributedString: attributedText) + let boldTextFont = Font.semibold(floor(item.presentationData.fontSize.itemListBaseFontSize * 15.0 / 17.0)) + mutableAttributedText.insert(NSAttributedString(string: commandPrefix + " ", font: boldTextFont, textColor: theme.titleColor), at: 0) + if let searchQuery = customMessageListData.searchQuery { + let range = (mutableAttributedText.string as NSString).range(of: searchQuery) + if range.location == 0 { + mutableAttributedText.addAttribute(.foregroundColor, value: item.presentationData.theme.list.itemAccentColor, range: range) + } + } + attributedText = mutableAttributedText + } + if !ignoreForwardedIcon { if case .savedMessagesChats = item.chatListLocation { displayForwardedIcon = false @@ -2491,21 +2616,31 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { if displayMediaPreviews { let contentImageFillSize = CGSize(width: 8.0, height: contentImageSize.height) _ = contentImageFillSize + + var contentImageIsDisplayedAsAvatar = false + if case let .peer(peerData) = item.content, let customMessageListData = peerData.customMessageListData, customMessageListData.commandPrefix != nil { + contentImageIsDisplayedAsAvatar = true + } + for message in messages { if contentImageSpecs.count >= 3 { break } + inner: for media in message.media { if let image = media as? TelegramMediaImage { if let _ = largestImageRepresentation(image.representations) { let fitSize = contentImageSize - contentImageSpecs.append((message, .image(image), fitSize)) + contentImageSpecs.append(ContentImageSpec(message: message, media: .image(image), size: fitSize)) } break inner } else if let file = media as? TelegramMediaFile { if file.isVideo, !file.isVideoSticker, let _ = file.dimensions { let fitSize = contentImageSize - contentImageSpecs.append((message, .file(file), fitSize)) + contentImageSpecs.append(ContentImageSpec(message: message, media: .file(file), size: fitSize)) + } else if contentImageIsDisplayedAsAvatar && (file.isSticker || file.isVideoSticker) { + let fitSize = contentImageSize + contentImageSpecs.append(ContentImageSpec(message: message, media: .file(file), size: fitSize)) } break inner } else if let webpage = media as? TelegramMediaWebpage, case let .Loaded(content) = webpage.content { @@ -2513,36 +2648,41 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { if let image = content.image, let type = content.type, imageTypes.contains(type) { if let _ = largestImageRepresentation(image.representations) { let fitSize = contentImageSize - contentImageSpecs.append((message, .image(image), fitSize)) + contentImageSpecs.append(ContentImageSpec(message: message, media: .image(image), size: fitSize)) } break inner } else if let file = content.file { if file.isVideo, !file.isInstantVideo, let _ = file.dimensions { let fitSize = contentImageSize - contentImageSpecs.append((message, .file(file), fitSize)) + contentImageSpecs.append(ContentImageSpec(message: message, media: .file(file), size: fitSize)) } break inner } } else if let action = media as? TelegramMediaAction, case let .suggestedProfilePhoto(image) = action.action, let _ = image { let fitSize = contentImageSize - contentImageSpecs.append((message, .action(action), fitSize)) + contentImageSpecs.append(ContentImageSpec(message: message, media: .action(action), size: fitSize)) } else if let storyMedia = media as? TelegramMediaStory, let story = message.associatedStories[storyMedia.storyId], !story.data.isEmpty, case let .item(storyItem) = story.get(Stories.StoredItem.self) { if let image = storyItem.media as? TelegramMediaImage { if let _ = largestImageRepresentation(image.representations) { let fitSize = contentImageSize - contentImageSpecs.append((message, .image(image), fitSize)) + contentImageSpecs.append(ContentImageSpec(message: message, media: .image(image), size: fitSize)) } break inner } else if let file = storyItem.media as? TelegramMediaFile { if file.isVideo, !file.isInstantVideo, let _ = file.dimensions { let fitSize = contentImageSize - contentImageSpecs.append((message, .file(file), fitSize)) + contentImageSpecs.append(ContentImageSpec(message: message, media: .file(file), size: fitSize)) } break inner } } } } + + if contentImageIsDisplayedAsAvatar { + avatarContentImageSpec = contentImageSpecs.first + contentImageSpecs.removeAll() + } } } else { attributedText = NSAttributedString(string: messageText, font: textFont, textColor: theme.messageTextColor) @@ -2619,7 +2759,21 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { switch contentData { case let .chat(itemPeer, threadInfo, _, _, _, _, _): - if let threadInfo = threadInfo { + if case let .peer(peerData) = item.content, let customMessageListData = peerData.customMessageListData { + if customMessageListData.commandPrefix != nil { + titleAttributedString = nil + } else { + if let displayTitle = itemPeer.chatMainPeer?.displayTitle(strings: item.presentationData.strings, displayOrder: item.presentationData.nameDisplayOrder) { + let textColor: UIColor + if case let .chatList(index) = item.index, index.messageIndex.id.peerId.namespace == Namespaces.Peer.SecretChat { + textColor = theme.secretTitleColor + } else { + textColor = theme.titleColor + } + titleAttributedString = NSAttributedString(string: displayTitle, font: titleFont, textColor: textColor) + } + } + } else if let threadInfo = threadInfo { titleAttributedString = NSAttributedString(string: threadInfo.info.title, font: titleFont, textColor: theme.titleColor) } else if let message = messages.last, case let .user(author) = message.author, displayAsMessage { titleAttributedString = NSAttributedString(string: author.id == account.peerId ? item.presentationData.strings.DialogList_You : EnginePeer.user(author).displayTitle(strings: item.presentationData.strings, displayOrder: item.presentationData.nameDisplayOrder), font: titleFont, textColor: theme.titleColor) @@ -2658,7 +2812,13 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { case let .peer(peerData): topIndex = peerData.messages.first?.index } - if let topIndex { + if case let .peer(peerData) = item.content, let customMessageListData = peerData.customMessageListData { + if let messageCount = customMessageListData.messageCount, customMessageListData.commandPrefix == nil { + dateText = "\(messageCount)" + } else { + dateText = " " + } + } else if let topIndex { var t = Int(topIndex.timestamp) var timeinfo = tm() localtime_r(&t, &timeinfo) @@ -2827,7 +2987,9 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { switch item.content { case let .peer(peerData): if let peer = peerData.messages.last?.author { - if case .savedMessagesChats = item.chatListLocation, peer.id == item.context.account.peerId { + if case let .peer(peerData) = item.content, peerData.customMessageListData != nil { + currentCredibilityIconContent = nil + } else if case .savedMessagesChats = item.chatListLocation, peer.id == item.context.account.peerId { currentCredibilityIconContent = nil } else if peer.isScam { currentCredibilityIconContent = .text(color: item.presentationData.theme.chat.message.incoming.scamColor, string: item.presentationData.strings.Message_ScamAccount.uppercased()) @@ -2849,7 +3011,9 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { break } } else if case let .chat(itemPeer) = contentPeer, let peer = itemPeer.chatMainPeer { - if case .savedMessagesChats = item.chatListLocation, peer.id == item.context.account.peerId { + if case let .peer(peerData) = item.content, peerData.customMessageListData != nil { + currentCredibilityIconContent = nil + } else if case .savedMessagesChats = item.chatListLocation, peer.id == item.context.account.peerId { currentCredibilityIconContent = nil } else if peer.isScam { currentCredibilityIconContent = .text(color: item.presentationData.theme.chat.message.incoming.scamColor, string: item.presentationData.strings.Message_ScamAccount.uppercased()) @@ -2984,9 +3148,22 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { let (authorLayout, authorApply) = authorLayout(item.context, rawContentWidth - badgeSize, item.presentationData.theme, effectiveAuthorTitle, forumThreads) + var textBottomRightCutout: CGFloat = 0.0 + + let trailingTextBadgeInsets = UIEdgeInsets(top: 2.0 - UIScreenPixel, left: 5.0, bottom: 2.0 - UIScreenPixel, right: 5.0) + var trailingTextBadgeLayoutAndApply: (TextNodeLayout, () -> TextNode)? + if case let .peer(peerData) = item.content, let customMessageListData = peerData.customMessageListData, customMessageListData.commandPrefix != nil, let messageCount = customMessageListData.messageCount, messageCount > 1 { + let trailingText: String + trailingText = item.presentationData.strings.ChatList_ItemMoreMessagesFormat(Int32(messageCount - 1)) + let trailingAttributedText = NSAttributedString(string: trailingText, font: Font.regular(12.0), textColor: theme.messageTextColor) + let (layout, apply) = makeTrailingTextBadgeLayout(TextNodeLayoutArguments(attributedString: trailingAttributedText, backgroundColor: nil, maximumNumberOfLines: 1, truncationType: .end, constrainedSize: CGSize(width: rawContentWidth, height: CGFloat.greatestFiniteMagnitude), alignment: .natural, cutout: nil, insets: UIEdgeInsets())) + trailingTextBadgeLayoutAndApply = (layout, apply) + textBottomRightCutout += layout.size.width + 4.0 + trailingTextBadgeInsets.left + trailingTextBadgeInsets.right + } + var textCutout: TextNodeCutout? - if !textLeftCutout.isZero { - textCutout = TextNodeCutout(topLeft: CGSize(width: textLeftCutout, height: 10.0), topRight: nil, bottomRight: nil) + if !textLeftCutout.isZero || !textBottomRightCutout.isZero { + textCutout = TextNodeCutout(topLeft: textLeftCutout.isZero ? nil : CGSize(width: textLeftCutout, height: 10.0), topRight: nil, bottomRight: textBottomRightCutout.isZero ? nil : CGSize(width: textBottomRightCutout, height: 10.0)) } var textMaxWidth = rawContentWidth - badgeSize @@ -3128,6 +3305,16 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { ItemListRevealOption(key: RevealOptionKey.hidePsa.rawValue, title: item.presentationData.strings.ChatList_HideAction, icon: deleteIcon, color: item.presentationData.theme.list.itemDisclosureActions.inactive.fillColor, textColor: item.presentationData.theme.list.itemDisclosureActions.neutral1.foregroundColor) ] peerLeftRevealOptions = [] + } else if case let .peer(peerData) = item.content, let customMessageListData = peerData.customMessageListData { + peerLeftRevealOptions = [] + if customMessageListData.commandPrefix != nil { + peerRevealOptions = [ + ItemListRevealOption(key: RevealOptionKey.edit.rawValue, title: item.presentationData.strings.ChatList_ItemMenuEdit, icon: .none, color: item.presentationData.theme.list.itemDisclosureActions.neutral2.fillColor, textColor: item.presentationData.theme.list.itemDisclosureActions.neutral2.foregroundColor), + ItemListRevealOption(key: RevealOptionKey.delete.rawValue, title: item.presentationData.strings.ChatList_ItemMenuDelete, icon: .none, color: item.presentationData.theme.list.itemDisclosureActions.destructive.fillColor, textColor: item.presentationData.theme.list.itemDisclosureActions.destructive.foregroundColor) + ] + } else { + peerRevealOptions = [] + } } else if promoInfo == nil { peerRevealOptions = revealOptions(strings: item.presentationData.strings, theme: item.presentationData.theme, isPinned: isPinned, isMuted: !isAccountPeer ? (currentMutedIconImage != nil) : nil, location: item.chatListLocation, peerId: renderedPeer.peerId, accountPeerId: item.context.account.peerId, canDelete: true, isEditing: item.editing, filterData: item.filterData) if case let .chat(itemPeer) = contentPeer { @@ -3159,16 +3346,21 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { animateContent = true } - let (measureLayout, measureApply) = makeMeasureLayout(TextNodeLayoutArguments(attributedString: titleAttributedString, backgroundColor: nil, maximumNumberOfLines: 1, truncationType: .end, constrainedSize: CGSize(width: titleRectWidth, height: CGFloat.greatestFiniteMagnitude), alignment: .natural, cutout: nil, insets: UIEdgeInsets())) + let (measureLayout, measureApply) = makeMeasureLayout(TextNodeLayoutArguments(attributedString: NSAttributedString(string: " ", font: titleFont, textColor: .black), backgroundColor: nil, maximumNumberOfLines: 1, truncationType: .end, constrainedSize: CGSize(width: titleRectWidth, height: CGFloat.greatestFiniteMagnitude), alignment: .natural, cutout: nil, insets: UIEdgeInsets())) let titleSpacing: CGFloat = -1.0 let authorSpacing: CGFloat = -3.0 var itemHeight: CGFloat = 8.0 * 2.0 + 1.0 itemHeight -= 21.0 - itemHeight += titleLayout.size.height - itemHeight += measureLayout.size.height * 3.0 - itemHeight += titleSpacing - itemHeight += authorSpacing + if case let .peer(peerData) = item.content, let customMessageListData = peerData.customMessageListData, customMessageListData.commandPrefix != nil { + itemHeight += measureLayout.size.height * 2.0 + itemHeight += 20.0 + } else { + itemHeight += titleLayout.size.height + itemHeight += measureLayout.size.height * 3.0 + itemHeight += titleSpacing + itemHeight += authorSpacing + } let rawContentRect = CGRect(origin: CGPoint(x: 2.0, y: layoutOffset + floor(item.presentationData.fontSize.itemListBaseFontSize * 8.0 / 17.0)), size: CGSize(width: rawContentWidth, height: itemHeight - 12.0 - 9.0)) @@ -3541,6 +3733,19 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { let _ = measureApply() let _ = dateApply() + var currentTextSnapshotView: UIView? + if transition.isAnimated, let currentItem, currentItem.editing != item.editing, strongSelf.textNode.textNode.cachedLayout?.linesRects() != textLayout.linesRects() { + if let textSnapshotView = strongSelf.textNode.textNode.view.snapshotContentTree() { + textSnapshotView.layer.anchorPoint = CGPoint() + currentTextSnapshotView = textSnapshotView + strongSelf.textNode.textNode.view.superview?.insertSubview(textSnapshotView, aboveSubview: strongSelf.textNode.textNode.view) + textSnapshotView.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.2, removeOnCompletion: false, completion: { [weak textSnapshotView] _ in + textSnapshotView?.removeFromSuperview() + }) + strongSelf.textNode.textNode.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.18) + } + } + let _ = textApply(TextNodeWithEntities.Arguments( context: item.context, cache: item.interaction.animationCache, @@ -3559,7 +3764,32 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { let _ = mentionBadgeApply(animateBadges, true) let _ = onlineApply(animateContent && animateOnline) - transition.updateFrame(node: strongSelf.dateNode, frame: CGRect(origin: CGPoint(x: contentRect.origin.x + contentRect.size.width - dateLayout.size.width, y: contentRect.origin.y + 2.0), size: dateLayout.size)) + var dateFrame = CGRect(origin: CGPoint(x: contentRect.origin.x + contentRect.size.width - dateLayout.size.width, y: contentRect.origin.y + 2.0), size: dateLayout.size) + + if case let .peer(peerData) = item.content, let customMessageListData = peerData.customMessageListData, customMessageListData.messageCount != nil, customMessageListData.commandPrefix == nil { + dateFrame.origin.x -= 10.0 + + let dateDisclosureIconView: UIImageView + if let current = strongSelf.dateDisclosureIconView { + dateDisclosureIconView = current + } else { + dateDisclosureIconView = UIImageView(image: UIImage(bundleImageName: "Item List/DisclosureArrow")?.withRenderingMode(.alwaysTemplate)) + strongSelf.dateDisclosureIconView = dateDisclosureIconView + strongSelf.mainContentContainerNode.view.addSubview(dateDisclosureIconView) + } + dateDisclosureIconView.tintColor = item.presentationData.theme.list.disclosureArrowColor + let iconScale: CGFloat = 0.7 + if let image = dateDisclosureIconView.image { + let imageSize = CGSize(width: floor(image.size.width * iconScale), height: floor(image.size.height * iconScale)) + let iconFrame = CGRect(origin: CGPoint(x: contentRect.origin.x + contentRect.size.width - imageSize.width + 4.0, y: floorToScreenPixels(dateFrame.midY - imageSize.height * 0.5)), size: imageSize) + dateDisclosureIconView.frame = iconFrame + } + } else if let dateDisclosureIconView = strongSelf.dateDisclosureIconView { + strongSelf.dateDisclosureIconView = nil + dateDisclosureIconView.removeFromSuperview() + } + + transition.updateFrame(node: strongSelf.dateNode, frame: dateFrame) var statusOffset: CGFloat = 0.0 if let dateIconImage { @@ -3782,13 +4012,71 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { strongSelf.authorNode.assignParentNode(parentNode: nil) } + if let currentTextSnapshotView { + transition.updatePosition(layer: currentTextSnapshotView.layer, position: textNodeFrame.origin) + } + + if let trailingTextBadgeLayoutAndApply { + let badgeSize = CGSize(width: trailingTextBadgeLayoutAndApply.0.size.width + trailingTextBadgeInsets.left + trailingTextBadgeInsets.right, height: trailingTextBadgeLayoutAndApply.0.size.height + trailingTextBadgeInsets.top + trailingTextBadgeInsets.bottom - UIScreenPixel) + + var badgeFrame: CGRect + if textLayout.numberOfLines > 1 { + badgeFrame = CGRect(origin: CGPoint(x: textLayout.trailingLineWidth + 4.0, y: textNodeFrame.height - 3.0 - badgeSize.height), size: badgeSize) + } else { + let firstLineFrame = textLayout.linesRects().first ?? CGRect(origin: CGPoint(), size: textNodeFrame.size) + badgeFrame = CGRect(origin: CGPoint(x: 0.0, y: firstLineFrame.height + 5.0), size: badgeSize) + } + + if badgeFrame.origin.x + badgeFrame.width >= textNodeFrame.width - 2.0 - 10.0 { + badgeFrame.origin.x = textNodeFrame.width - 2.0 - badgeFrame.width + } + + let trailingTextBadgeBackground: UIImageView + if let current = strongSelf.trailingTextBadgeBackground { + trailingTextBadgeBackground = current + } else { + trailingTextBadgeBackground = UIImageView(image: tagBackgroundImage) + strongSelf.trailingTextBadgeBackground = trailingTextBadgeBackground + strongSelf.textNode.textNode.view.addSubview(trailingTextBadgeBackground) + } + trailingTextBadgeBackground.tintColor = theme.pinnedItemBackgroundColor.mixedWith(theme.unreadBadgeInactiveBackgroundColor, alpha: 0.1) + + trailingTextBadgeBackground.frame = badgeFrame + + let trailingTextBadgeFrame = CGRect(origin: CGPoint(x: badgeFrame.minX + trailingTextBadgeInsets.left, y: badgeFrame.minY + trailingTextBadgeInsets.top), size: trailingTextBadgeLayoutAndApply.0.size) + let trailingTextBadgeNode = trailingTextBadgeLayoutAndApply.1() + if strongSelf.trailingTextBadgeNode !== trailingTextBadgeNode { + strongSelf.trailingTextBadgeNode?.removeFromSupernode() + strongSelf.trailingTextBadgeNode = trailingTextBadgeNode + + strongSelf.textNode.textNode.addSubnode(trailingTextBadgeNode) + + trailingTextBadgeNode.layer.anchorPoint = CGPoint() + } + + trailingTextBadgeNode.frame = trailingTextBadgeFrame + } else { + if let trailingTextBadgeNode = strongSelf.trailingTextBadgeNode { + strongSelf.trailingTextBadgeNode = nil + trailingTextBadgeNode.removeFromSupernode() + } + if let trailingTextBadgeBackground = strongSelf.trailingTextBadgeBackground { + strongSelf.trailingTextBadgeBackground = nil + trailingTextBadgeBackground.removeFromSuperview() + } + } + if !itemTags.isEmpty { - let itemTagListFrame = CGRect(origin: CGPoint(x: contentRect.minX, y: contentRect.maxY - 12.0), size: CGSize(width: contentRect.width, height: 20.0)) + let sizeFactor = item.presentationData.fontSize.itemListBaseFontSize / 17.0 + let itemTagListFrame = CGRect(origin: CGPoint(x: contentRect.minX, y: contentRect.minY + measureLayout.size.height * 2.0 + floorToScreenPixels(2.0 * sizeFactor)), size: CGSize(width: contentRect.width, height: floorToScreenPixels(20.0 * sizeFactor))) + + var itemTagListTransition = transition let itemTagList: ComponentView if let current = strongSelf.itemTagList { itemTagList = current } else { + itemTagListTransition = .immediate itemTagList = ComponentView() strongSelf.itemTagList = itemTagList } @@ -3797,7 +4085,8 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { component: AnyComponent(ChatListItemTagListComponent( context: item.context, tags: itemTags, - theme: item.presentationData.theme + theme: item.presentationData.theme, + sizeFactor: sizeFactor )), environment: {}, containerSize: itemTagListFrame.size @@ -3807,7 +4096,8 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { itemTagListView.isUserInteractionEnabled = false strongSelf.mainContentContainerNode.view.addSubview(itemTagListView) } - itemTagListView.frame = itemTagListFrame + + itemTagListTransition.updateFrame(view: itemTagListView, frame: itemTagListFrame) } } else { if let itemTagList = strongSelf.itemTagList { @@ -3926,7 +4216,11 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { } var validMediaIds: [EngineMedia.Id] = [] - for (message, media, mediaSize) in contentImageSpecs { + for spec in contentImageSpecs { + let message = spec.message + let media = spec.media + let mediaSize = spec.size + var mediaId = media.id if mediaId == nil, case let .action(action) = media, case let .suggestedProfilePhoto(image) = action.action { mediaId = image?.id @@ -3965,6 +4259,36 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { strongSelf.currentMediaPreviewSpecs = contentImageSpecs strongSelf.currentTextLeftCutout = textLeftCutout + if let avatarContentImageSpec { + strongSelf.avatarNode.isHidden = true + + if let previous = strongSelf.avatarMediaNode, previous.media != avatarContentImageSpec.media { + strongSelf.avatarMediaNode = nil + previous.removeFromSupernode() + } + + var avatarMediaNodeTransition = transition + let avatarMediaNode: ChatListMediaPreviewNode + if let current = strongSelf.avatarMediaNode { + avatarMediaNode = current + } else { + avatarMediaNodeTransition = .immediate + avatarMediaNode = ChatListMediaPreviewNode(context: item.context, message: avatarContentImageSpec.message, media: avatarContentImageSpec.media) + strongSelf.avatarMediaNode = avatarMediaNode + strongSelf.contextContainer.addSubnode(avatarMediaNode) + } + + avatarMediaNodeTransition.updateFrame(node: avatarMediaNode, frame: avatarFrame) + avatarMediaNode.updateLayout(size: avatarFrame.size, synchronousLoads: synchronousLoads) + } else { + strongSelf.avatarNode.isHidden = false + + if let avatarMediaNode = strongSelf.avatarMediaNode { + strongSelf.avatarMediaNode = nil + avatarMediaNode.removeFromSupernode() + } + } + if !contentDelta.x.isZero || !contentDelta.y.isZero { let titlePosition = strongSelf.titleNode.position transition.animatePosition(node: strongSelf.titleNode, from: CGPoint(x: titlePosition.x - contentDelta.x, y: titlePosition.y - contentDelta.y)) @@ -4090,6 +4414,12 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { transition.updateAlpha(node: strongSelf.separatorNode, alpha: 1.0) } + if case let .peer(peerData) = item.content, let customMessageListData = peerData.customMessageListData { + if customMessageListData.hideSeparator { + strongSelf.separatorNode.isHidden = true + } + } + transition.updateFrame(node: strongSelf.backgroundNode, frame: CGRect(origin: CGPoint(x: 0.0, y: 0.0), size: CGSize(width: layout.contentSize.width, height: itemHeight))) let backgroundColor: UIColor let highlightedBackgroundColor: UIColor @@ -4105,7 +4435,11 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { highlightedBackgroundColor = theme.pinnedItemHighlightedBackgroundColor } } else { - backgroundColor = theme.itemBackgroundColor + if case let .peer(peerData) = item.content, peerData.customMessageListData != nil { + backgroundColor = .clear + } else { + backgroundColor = theme.itemBackgroundColor + } highlightedBackgroundColor = theme.itemHighlightedBackgroundColor } @@ -4340,6 +4674,9 @@ class ChatListItemNode: ItemListRevealOptionsItemNode { self.revealOptionsInteractivelyClosed() self.customAnimationInProgress = false } + case RevealOptionKey.edit.rawValue: + item.interaction.editPeer(item) + close = true default: break } diff --git a/submodules/ChatListUI/Sources/Node/ChatListNode.swift b/submodules/ChatListUI/Sources/Node/ChatListNode.swift index a1c0d487420..0ac1958dbfe 100644 --- a/submodules/ChatListUI/Sources/Node/ChatListNode.swift +++ b/submodules/ChatListUI/Sources/Node/ChatListNode.swift @@ -125,6 +125,7 @@ public final class ChatListNodeInteraction { let hideChatFolderUpdates: () -> Void let openStories: (ChatListNode.OpenStoriesSubject, ASDisplayNode?) -> Void let dismissNotice: (ChatListNotice) -> Void + let editPeer: (ChatListItem) -> Void public var searchTextHighightState: String? var highlightedChatLocation: ChatListHighlightedLocation? @@ -179,7 +180,8 @@ public final class ChatListNodeInteraction { openChatFolderUpdates: @escaping () -> Void, hideChatFolderUpdates: @escaping () -> Void, openStories: @escaping (ChatListNode.OpenStoriesSubject, ASDisplayNode?) -> Void, - dismissNotice: @escaping (ChatListNotice) -> Void + dismissNotice: @escaping (ChatListNotice) -> Void, + editPeer: @escaping (ChatListItem) -> Void ) { self.activateSearch = activateSearch // MARK: Nicegram PinnedChats @@ -222,6 +224,7 @@ public final class ChatListNodeInteraction { self.hideChatFolderUpdates = hideChatFolderUpdates self.openStories = openStories self.dismissNotice = dismissNotice + self.editPeer = editPeer } } @@ -370,7 +373,7 @@ public struct ChatListNodeState: Equatable { } } -private func mappedInsertEntries(context: AccountContext, nodeInteraction: ChatListNodeInteraction, location: ChatListControllerLocation, filterData: ChatListItemFilterData?, chatListFilters: [ChatListFilter]?, mode: ChatListNodeMode, isPeerEnabled: ((EnginePeer) -> Bool)?, entries: [ChatListNodeViewTransitionInsertEntry]) -> [ListViewInsertItem] { +private func mappedInsertEntries(context: AccountContext, nodeInteraction: ChatListNodeInteraction, location: ChatListControllerLocation, isPremium: Bool, filterData: ChatListItemFilterData?, chatListFilters: [ChatListFilter]?, mode: ChatListNodeMode, isPeerEnabled: ((EnginePeer) -> Bool)?, entries: [ChatListNodeViewTransitionInsertEntry]) -> [ListViewInsertItem] { return entries.map { entry -> ListViewInsertItem in switch entry.entry { case .HeaderEntry: @@ -456,7 +459,7 @@ private func mappedInsertEntries(context: AccountContext, nodeInteraction: ChatL }, requiresPremiumForMessaging: peerEntry.requiresPremiumForMessaging, displayAsTopicList: peerEntry.displayAsTopicList, - tags: chatListItemTags(accountPeerId: context.account.peerId, peer: peer.chatMainPeer, isUnread: combinedReadState?.isUnread ?? false, isMuted: isRemovedFromTotalUnreadCount, isContact: isContact, hasUnseenMentions: hasUnseenMentions, chatListFilters: chatListFilters) + tags: chatListItemTags(location: location, accountPeerId: context.account.peerId, isPremium: isPremium, peer: peer.chatMainPeer, isUnread: combinedReadState?.isUnread ?? false, isMuted: isRemovedFromTotalUnreadCount, isContact: isContact, hasUnseenMentions: hasUnseenMentions, chatListFilters: chatListFilters) )), editing: editing, hasActiveRevealControls: hasActiveRevealControls, @@ -774,7 +777,7 @@ private func mappedInsertEntries(context: AccountContext, nodeInteraction: ChatL } } -private func mappedUpdateEntries(context: AccountContext, nodeInteraction: ChatListNodeInteraction, location: ChatListControllerLocation, filterData: ChatListItemFilterData?, chatListFilters: [ChatListFilter]?, mode: ChatListNodeMode, isPeerEnabled: ((EnginePeer) -> Bool)?, entries: [ChatListNodeViewTransitionUpdateEntry]) -> [ListViewUpdateItem] { +private func mappedUpdateEntries(context: AccountContext, nodeInteraction: ChatListNodeInteraction, location: ChatListControllerLocation, isPremium: Bool, filterData: ChatListItemFilterData?, chatListFilters: [ChatListFilter]?, mode: ChatListNodeMode, isPeerEnabled: ((EnginePeer) -> Bool)?, entries: [ChatListNodeViewTransitionUpdateEntry]) -> [ListViewUpdateItem] { return entries.map { entry -> ListViewUpdateItem in switch entry.entry { case let .PeerEntry(peerEntry): @@ -837,7 +840,7 @@ private func mappedUpdateEntries(context: AccountContext, nodeInteraction: ChatL }, requiresPremiumForMessaging: peerEntry.requiresPremiumForMessaging, displayAsTopicList: peerEntry.displayAsTopicList, - tags: chatListItemTags(accountPeerId: context.account.peerId, peer: peer.chatMainPeer, isUnread: combinedReadState?.isUnread ?? false, isMuted: isRemovedFromTotalUnreadCount, isContact: isContact, hasUnseenMentions: hasUnseenMentions, chatListFilters: chatListFilters) + tags: chatListItemTags(location: location, accountPeerId: context.account.peerId, isPremium: isPremium, peer: peer.chatMainPeer, isUnread: combinedReadState?.isUnread ?? false, isMuted: isRemovedFromTotalUnreadCount, isContact: isContact, hasUnseenMentions: hasUnseenMentions, chatListFilters: chatListFilters) )), editing: editing, hasActiveRevealControls: hasActiveRevealControls, @@ -1132,8 +1135,8 @@ private func mappedUpdateEntries(context: AccountContext, nodeInteraction: ChatL } } -private func mappedChatListNodeViewListTransition(context: AccountContext, nodeInteraction: ChatListNodeInteraction, location: ChatListControllerLocation, filterData: ChatListItemFilterData?, chatListFilters: [ChatListFilter]?, mode: ChatListNodeMode, isPeerEnabled: ((EnginePeer) -> Bool)?, transition: ChatListNodeViewTransition) -> ChatListNodeListViewTransition { - return ChatListNodeListViewTransition(chatListView: transition.chatListView, deleteItems: transition.deleteItems, insertItems: mappedInsertEntries(context: context, nodeInteraction: nodeInteraction, location: location, filterData: filterData, chatListFilters: chatListFilters, mode: mode, isPeerEnabled: isPeerEnabled, entries: transition.insertEntries), updateItems: mappedUpdateEntries(context: context, nodeInteraction: nodeInteraction, location: location, filterData: filterData, chatListFilters: chatListFilters, mode: mode, isPeerEnabled: isPeerEnabled, entries: transition.updateEntries), options: transition.options, scrollToItem: transition.scrollToItem, stationaryItemRange: transition.stationaryItemRange, adjustScrollToFirstItem: transition.adjustScrollToFirstItem, animateCrossfade: transition.animateCrossfade) +private func mappedChatListNodeViewListTransition(context: AccountContext, nodeInteraction: ChatListNodeInteraction, location: ChatListControllerLocation, isPremium: Bool, filterData: ChatListItemFilterData?, chatListFilters: [ChatListFilter]?, mode: ChatListNodeMode, isPeerEnabled: ((EnginePeer) -> Bool)?, transition: ChatListNodeViewTransition) -> ChatListNodeListViewTransition { + return ChatListNodeListViewTransition(chatListView: transition.chatListView, deleteItems: transition.deleteItems, insertItems: mappedInsertEntries(context: context, nodeInteraction: nodeInteraction, location: location, isPremium: isPremium, filterData: filterData, chatListFilters: chatListFilters, mode: mode, isPeerEnabled: isPeerEnabled, entries: transition.insertEntries), updateItems: mappedUpdateEntries(context: context, nodeInteraction: nodeInteraction, location: location, isPremium: isPremium, filterData: filterData, chatListFilters: chatListFilters, mode: mode, isPeerEnabled: isPeerEnabled, entries: transition.updateEntries), options: transition.options, scrollToItem: transition.scrollToItem, stationaryItemRange: transition.stationaryItemRange, adjustScrollToFirstItem: transition.adjustScrollToFirstItem, animateCrossfade: transition.animateCrossfade) } private final class ChatListOpaqueTransactionState { @@ -1822,6 +1825,7 @@ public final class ChatListNode: ListView { default: break } + }, editPeer: { _ in }) nodeInteraction.isInlineMode = isInlineMode @@ -2172,8 +2176,17 @@ public final class ChatListNode: ListView { } let previousChatListFilters = Atomic<[ChatListFilter]?>(value: nil) - let chatListNodeViewTransition = combineLatest( - queue: viewProcessingQueue, + let previousAccountIsPremium = Atomic(value: nil) + + let accountIsPremium = context.engine.data.subscribe( + TelegramEngine.EngineData.Item.Peer.Peer(id: context.account.peerId) + ) + |> map { peer -> Bool in + return peer?.isPremium ?? false + } + |> distinctUntilChanged + + let chatListNodeViewTransition = combineLatest(queue: viewProcessingQueue, hideArchivedFolderByDefault, displayArchiveIntro, storageInfo, @@ -2188,11 +2201,12 @@ public final class ChatListNode: ListView { // self.statePromise.get(), contacts, - chatListFilters + chatListFilters, + accountIsPremium ) // MARK: Nicegram PinnedChats, nicegramItems added // MARK: Nicegram HiddenChats, hiddenChats added - |> mapToQueue { (hideArchivedFolderByDefault, displayArchiveIntro, storageInfo, suggestedChatListNotice, savedMessagesPeer, updateAndFilter, nicegramItems, hiddenChats, state, contacts, chatListFilters) -> Signal in + |> mapToQueue { (hideArchivedFolderByDefault, displayArchiveIntro, storageInfo, suggestedChatListNotice, savedMessagesPeer, updateAndFilter, nicegramItems, hiddenChats, state, contacts, chatListFilters, accountIsPremium) -> Signal in let (update, filter) = updateAndFilter let previousHideArchivedFolderByDefaultValue = previousHideArchivedFolderByDefault.swap(hideArchivedFolderByDefault) @@ -2675,9 +2689,12 @@ public final class ChatListNode: ListView { if chatListFilters != previousChatListFiltersValue { forceAllUpdated = true } + if accountIsPremium != previousAccountIsPremium.swap(accountIsPremium) { + forceAllUpdated = true + } return preparedChatListNodeViewTransition(from: previousView, to: processedView, reason: reason, previewing: previewing, disableAnimations: disableAnimations, account: context.account, scrollPosition: updatedScrollPosition, searchMode: searchMode, forceAllUpdated: forceAllUpdated) - |> map({ mappedChatListNodeViewListTransition(context: context, nodeInteraction: nodeInteraction, location: location, filterData: filterData, chatListFilters: chatListFilters, mode: mode, isPeerEnabled: isPeerEnabled, transition: $0) }) + |> map({ mappedChatListNodeViewListTransition(context: context, nodeInteraction: nodeInteraction, location: location, isPremium: accountIsPremium, filterData: filterData, chatListFilters: chatListFilters, mode: mode, isPeerEnabled: isPeerEnabled, transition: $0) }) |> runOn(prepareOnMainQueue ? Queue.mainQueue() : viewProcessingQueue) } @@ -3277,7 +3294,7 @@ public final class ChatListNode: ListView { } private func resetFilter() { - if let chatListFilter = self.chatListFilter { + if let chatListFilter = self.chatListFilter, chatListFilter.id != Int32.max { self.updatedFilterDisposable.set((self.context.engine.peers.updatedChatListFilters() |> map { filters -> ChatListFilter? in for filter in filters { @@ -4292,7 +4309,14 @@ func hideChatListContacts(context: AccountContext) { let _ = ApplicationSpecificNotice.setDisplayChatListContacts(accountManager: context.sharedContext.accountManager).startStandalone() } -func chatListItemTags(accountPeerId: EnginePeer.Id, peer: EnginePeer?, isUnread: Bool, isMuted: Bool, isContact: Bool, hasUnseenMentions: Bool, chatListFilters: [ChatListFilter]?) -> [ChatListItemContent.Tag] { +func chatListItemTags(location: ChatListControllerLocation, accountPeerId: EnginePeer.Id, isPremium: Bool, peer: EnginePeer?, isUnread: Bool, isMuted: Bool, isContact: Bool, hasUnseenMentions: Bool, chatListFilters: [ChatListFilter]?) -> [ChatListItemContent.Tag] { + if !isPremium { + return [] + } + if case .chatList = location { + } else { + return [] + } guard let chatListFilters, !chatListFilters.isEmpty else { return [] } @@ -4302,13 +4326,15 @@ func chatListItemTags(accountPeerId: EnginePeer.Id, peer: EnginePeer?, isUnread: var result: [ChatListItemContent.Tag] = [] for case let .filter(id, title, _, data) in chatListFilters { - let predicate = chatListFilterPredicate(filter: data, accountPeerId: accountPeerId) - if predicate.includes(peer: peer._asPeer(), groupId: .root, isRemovedFromTotalUnreadCount: isMuted, isUnread: isUnread, isContact: isContact, messageTagSummaryResult: hasUnseenMentions) { - result.append(ChatListItemContent.Tag( - id: id, - title: title, - colorId: data.color?.rawValue ?? PeerNameColor.blue.rawValue - )) + if data.color != nil { + let predicate = chatListFilterPredicate(filter: data, accountPeerId: accountPeerId) + if predicate.pinnedPeerIds.contains(peer.id) || predicate.includes(peer: peer._asPeer(), groupId: .root, isRemovedFromTotalUnreadCount: isMuted, isUnread: isUnread, isContact: isContact, messageTagSummaryResult: hasUnseenMentions) { + result.append(ChatListItemContent.Tag( + id: id, + title: title, + colorId: data.color?.rawValue ?? PeerNameColor.blue.rawValue + )) + } } } return result diff --git a/submodules/ChatListUI/Sources/Node/ChatListNodeLocation.swift b/submodules/ChatListUI/Sources/Node/ChatListNodeLocation.swift index c4526126451..7b67cae1b2f 100644 --- a/submodules/ChatListUI/Sources/Node/ChatListNodeLocation.swift +++ b/submodules/ChatListUI/Sources/Node/ChatListNodeLocation.swift @@ -67,7 +67,7 @@ public func chatListFilterPredicate(filter: ChatListFilterData, accountPeerId: E } if !filter.categories.contains(.contacts) && isContact { if let user = peer as? TelegramUser { - if user.botInfo == nil { + if user.botInfo == nil && !user.flags.contains(.isSupport) { return false } } else if let _ = peer as? TelegramSecretChat { @@ -88,7 +88,7 @@ public func chatListFilterPredicate(filter: ChatListFilterData, accountPeerId: E } if !filter.categories.contains(.bots) { if let user = peer as? TelegramUser { - if user.botInfo != nil { + if user.botInfo != nil || user.flags.contains(.isSupport) { return false } } diff --git a/submodules/ChatPresentationInterfaceState/Sources/ChatPanelInterfaceInteraction.swift b/submodules/ChatPresentationInterfaceState/Sources/ChatPanelInterfaceInteraction.swift index 8b8e94bd367..0b947f19102 100644 --- a/submodules/ChatPresentationInterfaceState/Sources/ChatPanelInterfaceInteraction.swift +++ b/submodules/ChatPresentationInterfaceState/Sources/ChatPanelInterfaceInteraction.swift @@ -111,6 +111,8 @@ public final class ChatPanelInterfaceInteraction { public let togglePeerNotifications: () -> Void public let sendContextResult: (ChatContextResultCollection, ChatContextResult, ASDisplayNode, CGRect) -> Bool public let sendBotCommand: (Peer, String) -> Void + public let sendShortcut: (Int32) -> Void + public let openEditShortcuts: () -> Void public let sendBotStart: (String?) -> Void public let botSwitchChatWithPayload: (PeerId, String) -> Void public let beginMediaRecording: (Bool) -> Void @@ -231,6 +233,8 @@ public final class ChatPanelInterfaceInteraction { togglePeerNotifications: @escaping () -> Void, sendContextResult: @escaping (ChatContextResultCollection, ChatContextResult, ASDisplayNode, CGRect) -> Bool, sendBotCommand: @escaping (Peer, String) -> Void, + sendShortcut: @escaping (Int32) -> Void, + openEditShortcuts: @escaping () -> Void, sendBotStart: @escaping (String?) -> Void, botSwitchChatWithPayload: @escaping (PeerId, String) -> Void, beginMediaRecording: @escaping (Bool) -> Void, @@ -350,6 +354,8 @@ public final class ChatPanelInterfaceInteraction { self.togglePeerNotifications = togglePeerNotifications self.sendContextResult = sendContextResult self.sendBotCommand = sendBotCommand + self.sendShortcut = sendShortcut + self.openEditShortcuts = openEditShortcuts self.sendBotStart = sendBotStart self.botSwitchChatWithPayload = botSwitchChatWithPayload self.beginMediaRecording = beginMediaRecording @@ -473,6 +479,8 @@ public final class ChatPanelInterfaceInteraction { }, sendContextResult: { _, _, _, _ in return false }, sendBotCommand: { _, _ in + }, sendShortcut: { _ in + }, openEditShortcuts: { }, sendBotStart: { _ in }, botSwitchChatWithPayload: { _, _ in }, beginMediaRecording: { _ in diff --git a/submodules/ChatPresentationInterfaceState/Sources/ChatPresentationInterfaceState.swift b/submodules/ChatPresentationInterfaceState/Sources/ChatPresentationInterfaceState.swift index 2bf79006efc..75406cdcf66 100644 --- a/submodules/ChatPresentationInterfaceState/Sources/ChatPresentationInterfaceState.swift +++ b/submodules/ChatPresentationInterfaceState/Sources/ChatPresentationInterfaceState.swift @@ -17,7 +17,7 @@ public extension ChatLocation { return peerId case let .replyThread(replyThreadMessage): return replyThreadMessage.peerId - case .feed: + case .customChatContents: return nil } } @@ -28,7 +28,7 @@ public extension ChatLocation { return nil case let .replyThread(replyThreadMessage): return replyThreadMessage.threadId - case .feed: + case .customChatContents: return nil } } @@ -1202,6 +1202,8 @@ public func canSendMessagesToChat(_ state: ChatPresentationInterfaceState) -> Bo } else { return false } + } else if case .customChatContents = state.chatLocation { + return true } else { return false } diff --git a/submodules/ChatSendMessageActionUI/Sources/ChatSendMessageActionSheetControllerNode.swift b/submodules/ChatSendMessageActionUI/Sources/ChatSendMessageActionSheetControllerNode.swift index 77d610c7434..a2afd7dc9b5 100644 --- a/submodules/ChatSendMessageActionUI/Sources/ChatSendMessageActionSheetControllerNode.swift +++ b/submodules/ChatSendMessageActionUI/Sources/ChatSendMessageActionSheetControllerNode.swift @@ -550,7 +550,7 @@ final class ChatSendMessageActionSheetControllerNode: ViewControllerTracingNode, } var clipDelta = delta - if inputHeight.isZero || layout.isNonExclusive { + if inputHeight < 70.0 || layout.isNonExclusive { clipDelta -= self.contentContainerNode.frame.height + 16.0 } @@ -673,7 +673,7 @@ final class ChatSendMessageActionSheetControllerNode: ViewControllerTracingNode, } var clipDelta = delta - if inputHeight.isZero || layout.isNonExclusive { + if inputHeight < 70.0 || layout.isNonExclusive { clipDelta -= self.contentContainerNode.frame.height + 16.0 } @@ -745,7 +745,7 @@ final class ChatSendMessageActionSheetControllerNode: ViewControllerTracingNode, } else { contentOrigin = CGPoint(x: layout.size.width - sideInset - contentSize.width - layout.safeInsets.right, y: layout.size.height - 6.0 - insets.bottom - contentSize.height) } - if inputHeight > 0.0 && !layout.isNonExclusive && self.animateInputField { + if inputHeight > 70.0 && !layout.isNonExclusive && self.animateInputField { contentOrigin.y += menuHeightWithInset } contentOrigin.y = min(contentOrigin.y + contentOffset, layout.size.height - 6.0 - layout.intrinsicInsets.bottom - contentSize.height) @@ -759,7 +759,7 @@ final class ChatSendMessageActionSheetControllerNode: ViewControllerTracingNode, } var sendButtonFrame = CGRect(origin: CGPoint(x: layout.size.width - initialSendButtonFrame.width + 1.0 - UIScreenPixel - layout.safeInsets.right, y: layout.size.height - insets.bottom - initialSendButtonFrame.height), size: initialSendButtonFrame.size) - if (inputHeight.isZero || layout.isNonExclusive) && self.animateInputField { + if (inputHeight < 70.0 || layout.isNonExclusive) && self.animateInputField { sendButtonFrame.origin.y -= menuHeightWithInset } sendButtonFrame.origin.y = min(sendButtonFrame.origin.y + contentOffset, layout.size.height - layout.intrinsicInsets.bottom - initialSendButtonFrame.height) @@ -772,7 +772,7 @@ final class ChatSendMessageActionSheetControllerNode: ViewControllerTracingNode, let messageHeightAddition: CGFloat = max(0.0, 35.0 - messageFrame.size.height) - if inputHeight.isZero || layout.isNonExclusive { + if inputHeight < 70.0 || layout.isNonExclusive { messageFrame.origin.y += menuHeightWithInset } diff --git a/submodules/ComponentFlow/Source/Base/CombinedComponent.swift b/submodules/ComponentFlow/Source/Base/CombinedComponent.swift index b4f27b849e8..04df0fd8cea 100644 --- a/submodules/ComponentFlow/Source/Base/CombinedComponent.swift +++ b/submodules/ComponentFlow/Source/Base/CombinedComponent.swift @@ -722,11 +722,11 @@ public extension CombinedComponent { updatedChild.view.layer.shadowRadius = 0.0 updatedChild.view.layer.shadowOpacity = 0.0 } - updatedChild.view.context(typeErasedComponent: updatedChild.component).erasedState._updated = { [weak viewContext] transition in + updatedChild.view.context(typeErasedComponent: updatedChild.component).erasedState._updated = { [weak viewContext] transition, isLocal in guard let viewContext = viewContext else { return } - viewContext.state.updated(transition: transition) + viewContext.state.updated(transition: transition, isLocal: isLocal) } if let transitionAppearWithGuide = updatedChild.transitionAppearWithGuide { diff --git a/submodules/ComponentFlow/Source/Base/Component.swift b/submodules/ComponentFlow/Source/Base/Component.swift index 77163495773..fabce5cf5cc 100644 --- a/submodules/ComponentFlow/Source/Base/Component.swift +++ b/submodules/ComponentFlow/Source/Base/Component.swift @@ -89,15 +89,15 @@ extension UIView { } open class ComponentState { - open var _updated: ((Transition) -> Void)? + open var _updated: ((Transition, Bool) -> Void)? var isUpdated: Bool = false public init() { } - public final func updated(transition: Transition = .immediate) { + public final func updated(transition: Transition = .immediate, isLocal: Bool = false) { self.isUpdated = true - self._updated?(transition) + self._updated?(transition, isLocal) } } diff --git a/submodules/ComponentFlow/Source/Components/RoundedRectangle.swift b/submodules/ComponentFlow/Source/Components/RoundedRectangle.swift index 32e0c7edfe2..275772854a1 100644 --- a/submodules/ComponentFlow/Source/Components/RoundedRectangle.swift +++ b/submodules/ComponentFlow/Source/Components/RoundedRectangle.swift @@ -8,16 +8,16 @@ public final class RoundedRectangle: Component { } public let colors: [UIColor] - public let cornerRadius: CGFloat + public let cornerRadius: CGFloat? public let gradientDirection: GradientDirection public let stroke: CGFloat? public let strokeColor: UIColor? - public convenience init(color: UIColor, cornerRadius: CGFloat, stroke: CGFloat? = nil, strokeColor: UIColor? = nil) { + public convenience init(color: UIColor, cornerRadius: CGFloat?, stroke: CGFloat? = nil, strokeColor: UIColor? = nil) { self.init(colors: [color], cornerRadius: cornerRadius, stroke: stroke, strokeColor: strokeColor) } - public init(colors: [UIColor], cornerRadius: CGFloat, gradientDirection: GradientDirection = .horizontal, stroke: CGFloat? = nil, strokeColor: UIColor? = nil) { + public init(colors: [UIColor], cornerRadius: CGFloat?, gradientDirection: GradientDirection = .horizontal, stroke: CGFloat? = nil, strokeColor: UIColor? = nil) { self.colors = colors self.cornerRadius = cornerRadius self.gradientDirection = gradientDirection @@ -49,8 +49,10 @@ public final class RoundedRectangle: Component { func update(component: RoundedRectangle, availableSize: CGSize, transition: Transition) -> CGSize { if self.component != component { + let cornerRadius = component.cornerRadius ?? min(availableSize.width, availableSize.height) * 0.5 + if component.colors.count == 1, let color = component.colors.first { - let imageSize = CGSize(width: max(component.stroke ?? 0.0, component.cornerRadius) * 2.0, height: max(component.stroke ?? 0.0, component.cornerRadius) * 2.0) + let imageSize = CGSize(width: max(component.stroke ?? 0.0, cornerRadius) * 2.0, height: max(component.stroke ?? 0.0, cornerRadius) * 2.0) UIGraphicsBeginImageContextWithOptions(imageSize, false, 0.0) if let context = UIGraphicsGetCurrentContext() { if let strokeColor = component.strokeColor { @@ -69,13 +71,13 @@ public final class RoundedRectangle: Component { context.fillEllipse(in: CGRect(origin: CGPoint(), size: imageSize).insetBy(dx: stroke, dy: stroke)) } } - self.image = UIGraphicsGetImageFromCurrentImageContext()?.stretchableImage(withLeftCapWidth: Int(component.cornerRadius), topCapHeight: Int(component.cornerRadius)) + self.image = UIGraphicsGetImageFromCurrentImageContext()?.stretchableImage(withLeftCapWidth: Int(cornerRadius), topCapHeight: Int(cornerRadius)) UIGraphicsEndImageContext() } else if component.colors.count > 1 { let imageSize = availableSize UIGraphicsBeginImageContextWithOptions(imageSize, false, 0.0) if let context = UIGraphicsGetCurrentContext() { - context.addPath(UIBezierPath(roundedRect: CGRect(origin: CGPoint(), size: imageSize), cornerRadius: component.cornerRadius).cgPath) + context.addPath(UIBezierPath(roundedRect: CGRect(origin: CGPoint(), size: imageSize), cornerRadius: cornerRadius).cgPath) context.clip() let colors = component.colors @@ -93,12 +95,12 @@ public final class RoundedRectangle: Component { if let stroke = component.stroke, stroke > 0.0 { context.resetClip() - context.addPath(UIBezierPath(roundedRect: CGRect(origin: CGPoint(), size: imageSize).insetBy(dx: stroke, dy: stroke), cornerRadius: component.cornerRadius).cgPath) + context.addPath(UIBezierPath(roundedRect: CGRect(origin: CGPoint(), size: imageSize).insetBy(dx: stroke, dy: stroke), cornerRadius: cornerRadius).cgPath) context.setBlendMode(.clear) context.fill(CGRect(origin: .zero, size: imageSize)) } } - self.image = UIGraphicsGetImageFromCurrentImageContext()?.stretchableImage(withLeftCapWidth: Int(component.cornerRadius), topCapHeight: Int(component.cornerRadius)) + self.image = UIGraphicsGetImageFromCurrentImageContext()?.stretchableImage(withLeftCapWidth: Int(cornerRadius), topCapHeight: Int(cornerRadius)) UIGraphicsEndImageContext() } } diff --git a/submodules/ComponentFlow/Source/Host/ComponentHostView.swift b/submodules/ComponentFlow/Source/Host/ComponentHostView.swift index 2c8a3d87b6c..cdc4a7c3abd 100644 --- a/submodules/ComponentFlow/Source/Host/ComponentHostView.swift +++ b/submodules/ComponentFlow/Source/Host/ComponentHostView.swift @@ -82,7 +82,7 @@ public final class ComponentHostView: UIView { self.currentComponent = component self.currentContainerSize = containerSize - componentState._updated = { [weak self] transition in + componentState._updated = { [weak self] transition, _ in guard let strongSelf = self else { return } @@ -208,11 +208,11 @@ public final class ComponentView { self.currentComponent = component self.currentContainerSize = containerSize - componentState._updated = { [weak self] transition in + componentState._updated = { [weak self] transition, isLocal in guard let strongSelf = self else { return } - if let parentState = strongSelf.parentState { + if !isLocal, let parentState = strongSelf.parentState { parentState.updated(transition: transition) } else { let _ = strongSelf._update(transition: transition, component: component, maybeEnvironment: { diff --git a/submodules/Components/ViewControllerComponent/Sources/ViewControllerComponent.swift b/submodules/Components/ViewControllerComponent/Sources/ViewControllerComponent.swift index aedc76242ed..d86b99280ac 100644 --- a/submodules/Components/ViewControllerComponent/Sources/ViewControllerComponent.swift +++ b/submodules/Components/ViewControllerComponent/Sources/ViewControllerComponent.swift @@ -46,6 +46,7 @@ open class ViewControllerComponentContainer: ViewController { public let statusBarHeight: CGFloat public let navigationHeight: CGFloat public let safeInsets: UIEdgeInsets + public let additionalInsets: UIEdgeInsets public let inputHeight: CGFloat public let metrics: LayoutMetrics public let deviceMetrics: DeviceMetrics @@ -60,6 +61,7 @@ open class ViewControllerComponentContainer: ViewController { statusBarHeight: CGFloat, navigationHeight: CGFloat, safeInsets: UIEdgeInsets, + additionalInsets: UIEdgeInsets, inputHeight: CGFloat, metrics: LayoutMetrics, deviceMetrics: DeviceMetrics, @@ -73,6 +75,7 @@ open class ViewControllerComponentContainer: ViewController { self.statusBarHeight = statusBarHeight self.navigationHeight = navigationHeight self.safeInsets = safeInsets + self.additionalInsets = additionalInsets self.inputHeight = inputHeight self.metrics = metrics self.deviceMetrics = deviceMetrics @@ -98,6 +101,9 @@ open class ViewControllerComponentContainer: ViewController { if lhs.safeInsets != rhs.safeInsets { return false } + if lhs.additionalInsets != rhs.additionalInsets { + return false + } if lhs.inputHeight != rhs.inputHeight { return false } @@ -167,6 +173,7 @@ open class ViewControllerComponentContainer: ViewController { statusBarHeight: layout.statusBarHeight ?? 0.0, navigationHeight: navigationHeight, safeInsets: UIEdgeInsets(top: layout.intrinsicInsets.top + layout.safeInsets.top, left: layout.safeInsets.left, bottom: layout.intrinsicInsets.bottom + layout.safeInsets.bottom, right: layout.safeInsets.right), + additionalInsets: layout.additionalInsets, inputHeight: layout.inputHeight ?? 0.0, metrics: layout.metrics, deviceMetrics: layout.deviceMetrics, diff --git a/submodules/ContactListUI/Sources/ContactListNode.swift b/submodules/ContactListUI/Sources/ContactListNode.swift index 3787faaee95..ad63bda0fef 100644 --- a/submodules/ContactListUI/Sources/ContactListNode.swift +++ b/submodules/ContactListUI/Sources/ContactListNode.swift @@ -702,9 +702,29 @@ private struct ContactsListNodeTransition { } public enum ContactListPresentation { + public struct Search { + public var signal: Signal + public var searchChatList: Bool + public var searchDeviceContacts: Bool + public var searchGroups: Bool + public var searchChannels: Bool + public var globalSearch: Bool + public var displaySavedMessages: Bool + + public init(signal: Signal, searchChatList: Bool, searchDeviceContacts: Bool, searchGroups: Bool, searchChannels: Bool, globalSearch: Bool, displaySavedMessages: Bool) { + self.signal = signal + self.searchChatList = searchChatList + self.searchDeviceContacts = searchDeviceContacts + self.searchGroups = searchGroups + self.searchChannels = searchChannels + self.globalSearch = globalSearch + self.displaySavedMessages = displaySavedMessages + } + } + case orderedByPresence(options: [ContactListAdditionalOption]) case natural(options: [ContactListAdditionalOption], includeChatList: Bool, topPeers: Bool) - case search(signal: Signal, searchChatList: Bool, searchDeviceContacts: Bool, searchGroups: Bool, searchChannels: Bool, globalSearch: Bool) + case search(Search) public var sortOrder: ContactsSortOrder? { switch self { @@ -1098,7 +1118,15 @@ public final class ContactListNode: ASDisplayNode { displayTopPeers = displayTopPeersValue } - if case let .search(query, searchChatList, searchDeviceContacts, searchGroups, searchChannels, globalSearch) = presentation { + if case let .search(search) = presentation { + let query = search.signal + let searchChatList = search.searchChatList + let searchDeviceContacts = search.searchDeviceContacts + let searchGroups = search.searchGroups + let searchChannels = search.searchChannels + let globalSearch = search.globalSearch + let displaySavedMessages = search.displaySavedMessages + return query |> mapToSignal { query in let foundLocalContacts: Signal<([FoundPeer], [EnginePeer.Id: EnginePeer.Presence]), NoError> @@ -1109,6 +1137,12 @@ public final class ContactListNode: ASDisplayNode { var resultPeers: [FoundPeer] = [] for peer in peers { + if !displaySavedMessages { + if peer.peerId == context.account.peerId { + continue + } + } + if searchGroups || searchChannels { let mainPeer = peer.chatMainPeer if let _ = mainPeer as? TelegramUser { diff --git a/submodules/ContactListUI/Sources/ContactsController.swift b/submodules/ContactListUI/Sources/ContactsController.swift index 5f95d9bf267..c79c7a5bb55 100644 --- a/submodules/ContactListUI/Sources/ContactsController.swift +++ b/submodules/ContactListUI/Sources/ContactsController.swift @@ -678,41 +678,18 @@ public class ContactsController: ViewController { return false } if value == .commit { - let presentationData = self.presentationData - let deleteContactsFromDevice: Signal if let contactDataManager = self.context.sharedContext.contactDataManager { - deleteContactsFromDevice = contactDataManager.deleteContactWithAppSpecificReference(peerId: peerIds.first!) + deleteContactsFromDevice = combineLatest(peerIds.map { contactDataManager.deleteContactWithAppSpecificReference(peerId: $0) } + ) + |> ignoreValues } else { deleteContactsFromDevice = .complete() } - var deleteSignal = self.context.engine.contacts.deleteContacts(peerIds: peerIds) + let deleteSignal = self.context.engine.contacts.deleteContacts(peerIds: peerIds) |> then(deleteContactsFromDevice) - let progressSignal = Signal { [weak self] subscriber in - guard let self else { - return EmptyDisposable - } - let statusController = OverlayStatusController(theme: presentationData.theme, type: .loading(cancelled: nil)) - self.present(statusController, in: .window(.root)) - return ActionDisposable { [weak statusController] in - Queue.mainQueue().async() { - statusController?.dismiss() - } - } - } - |> runOn(Queue.mainQueue()) - |> delay(0.15, queue: Queue.mainQueue()) - let progressDisposable = progressSignal.start() - - deleteSignal = deleteSignal - |> afterDisposed { - Queue.mainQueue().async { - progressDisposable.dispose() - } - } - for peerId in peerIds { deleteSendMessageIntents(peerId: peerId) } @@ -724,6 +701,9 @@ public class ContactsController: ViewController { } return state } + + let _ = deleteSignal.start() + return true } else if value == .undo { self.contactsNode.contactListNode.updatePendingRemovalPeerIds { state in diff --git a/submodules/ContactListUI/Sources/ContactsControllerNode.swift b/submodules/ContactListUI/Sources/ContactsControllerNode.swift index 6105a64ba90..32642969bfc 100644 --- a/submodules/ContactListUI/Sources/ContactsControllerNode.swift +++ b/submodules/ContactListUI/Sources/ContactsControllerNode.swift @@ -362,6 +362,7 @@ final class ContactsControllerNode: ASDisplayNode, UIGestureRecognizerDelegate { statusBarHeight: layout.statusBarHeight ?? 0.0, sideInset: layout.safeInsets.left, isSearchActive: self.isSearchDisplayControllerActive, + isSearchEnabled: true, primaryContent: primaryContent, secondaryContent: nil, secondaryTransition: 0.0, diff --git a/submodules/Display/Source/AlertController.swift b/submodules/Display/Source/AlertController.swift index 00bb5cf1238..8589a0baef0 100644 --- a/submodules/Display/Source/AlertController.swift +++ b/submodules/Display/Source/AlertController.swift @@ -83,7 +83,7 @@ open class AlertController: ViewController, StandalonePresentableController, Key } } } - private let contentNode: AlertContentNode + public let contentNode: AlertContentNode private let allowInputInset: Bool private weak var existingAlertController: AlertController? diff --git a/submodules/Display/Source/TextNode.swift b/submodules/Display/Source/TextNode.swift index d0ac26e952f..b469147b603 100644 --- a/submodules/Display/Source/TextNode.swift +++ b/submodules/Display/Source/TextNode.swift @@ -409,6 +409,8 @@ public final class TextNodeLayout: NSObject { switch self.resolvedAlignment { case .center: lineFrame = CGRect(origin: CGPoint(x: floor((size.width - line.frame.size.width) / 2.0), y: line.frame.minY), size: line.frame.size) + case .right: + lineFrame = CGRect(origin: CGPoint(x: size.width - line.frame.size.width, y: line.frame.minY), size: line.frame.size) default: lineFrame = displayLineFrame(frame: line.frame, isRTL: line.isRTL, boundingRect: CGRect(origin: CGPoint(), size: size), cutout: cutout) } @@ -521,6 +523,8 @@ public final class TextNodeLayout: NSObject { lineFrame.origin.x = floor((self.size.width - lineFrame.size.width) / 2.0) case .natural: lineFrame = displayLineFrame(frame: lineFrame, isRTL: line.isRTL, boundingRect: CGRect(origin: CGPoint(), size: self.size), cutout: self.cutout) + case .right: + lineFrame.origin.x = self.size.width - lineFrame.size.width default: break } @@ -589,6 +593,8 @@ public final class TextNodeLayout: NSObject { lineFrame.origin.x = floor((self.size.width - lineFrame.size.width) / 2.0) case .natural: lineFrame = displayLineFrame(frame: lineFrame, isRTL: line.isRTL, boundingRect: CGRect(origin: CGPoint(), size: self.size), cutout: self.cutout) + case .right: + lineFrame.origin.x = self.size.width - lineFrame.size.width default: break } @@ -666,6 +672,8 @@ public final class TextNodeLayout: NSObject { lineFrame.origin.x = floor((self.size.width - lineFrame.size.width) / 2.0) case .natural: lineFrame = displayLineFrame(frame: lineFrame, isRTL: line.isRTL, boundingRect: CGRect(origin: CGPoint(), size: self.size), cutout: self.cutout) + case .right: + lineFrame.origin.x = self.size.width - lineFrame.size.width default: break } @@ -846,6 +854,8 @@ public final class TextNodeLayout: NSObject { lineFrame.origin.x = floor((self.size.width - lineFrame.size.width) / 2.0) case .natural: lineFrame = displayLineFrame(frame: lineFrame, isRTL: line.isRTL, boundingRect: CGRect(origin: CGPoint(), size: self.size), cutout: self.cutout) + case .right: + lineFrame.origin.x = self.size.width - lineFrame.size.width default: break } @@ -1430,7 +1440,8 @@ open class TextNode: ASDisplayNode { let line = CTTypesetterCreateLine(typesetter, CFRange(location: currentLineStartIndex, length: lineCharacterCount)) var lineAscent: CGFloat = 0.0 var lineDescent: CGFloat = 0.0 - let lineWidth = CTLineGetTypographicBounds(line, &lineAscent, &lineDescent, nil) + var lineWidth = CTLineGetTypographicBounds(line, &lineAscent, &lineDescent, nil) + lineWidth = min(lineWidth, constrainedSegmentWidth - additionalSegmentRightInset) var isRTL = false let glyphRuns = CTLineGetGlyphRuns(line) as NSArray @@ -2009,7 +2020,7 @@ open class TextNode: ASDisplayNode { var descent: CGFloat = 0.0 CTLineGetTypographicBounds(coreTextLine, &ascent, &descent, nil) - addAttachment(attachment: attachment, line: coreTextLine, ascent: ascent, descent: descent, startIndex: range.location, endIndex: max(range.location, min(lineRange.location + lineRange.length - 1, range.location + range.length)), isAtEndOfTheLine: range.location + range.length >= lineRange.location + lineRange.length - 1) + addAttachment(attachment: attachment, line: coreTextLine, ascent: ascent, descent: descent, startIndex: range.location, endIndex: max(range.location, min(lineRange.location + lineRange.length, range.location + range.length)), isAtEndOfTheLine: range.location + range.length >= lineRange.location + lineRange.length - 1) } } } @@ -2124,7 +2135,7 @@ open class TextNode: ASDisplayNode { var descent: CGFloat = 0.0 CTLineGetTypographicBounds(coreTextLine, &ascent, &descent, nil) - addAttachment(attachment: attachment, line: coreTextLine, ascent: ascent, descent: descent, startIndex: range.location, endIndex: max(range.location, min(lineRange.location + lineRange.length - 1, range.location + range.length)), isAtEndOfTheLine: range.location + range.length >= lineRange.location + lineRange.length - 1) + addAttachment(attachment: attachment, line: coreTextLine, ascent: ascent, descent: descent, startIndex: range.location, endIndex: max(range.location, min(lineRange.location + lineRange.length, range.location + range.length)), isAtEndOfTheLine: range.location + range.length >= lineRange.location + lineRange.length - 1) } } @@ -2407,6 +2418,8 @@ open class TextNode: ASDisplayNode { } else { lineFrame.origin.x += offset.x } + } else if alignment == .right { + lineFrame.origin.x = offset.x + (bounds.size.width - lineFrame.width) } //context.setStrokeColor(UIColor.red.cgColor) diff --git a/submodules/Display/Source/WindowContent.swift b/submodules/Display/Source/WindowContent.swift index cdc5ee9df87..19416153361 100644 --- a/submodules/Display/Source/WindowContent.swift +++ b/submodules/Display/Source/WindowContent.swift @@ -243,7 +243,6 @@ public final class WindowKeyboardGestureRecognizerDelegate: NSObject, UIGestureR public class Window1 { public let hostView: WindowHostView public let badgeView: UIImageView - private let customProximityDimView: UIView private var deviceMetrics: DeviceMetrics @@ -331,10 +330,6 @@ public class Window1 { self.badgeView.image = UIImage(bundleImageName: "Components/AppBadge") self.badgeView.isHidden = true - self.customProximityDimView = UIView() - self.customProximityDimView.backgroundColor = .black - self.customProximityDimView.isHidden = true - self.systemUserInterfaceStyle = hostView.systemUserInterfaceStyle let boundsSize = self.hostView.eventView.bounds.size @@ -673,7 +668,6 @@ public class Window1 { self.windowPanRecognizer = recognizer self.hostView.containerView.addGestureRecognizer(recognizer) self.hostView.containerView.addSubview(self.badgeView) - self.hostView.containerView.addSubview(self.customProximityDimView) } public required init(coder aDecoder: NSCoder) { @@ -707,11 +701,18 @@ public class Window1 { self.updateBadgeVisibility() } + private var proximityDimController: CustomDimController? public func setProximityDimHidden(_ hidden: Bool) { - guard hidden != self.customProximityDimView.isHidden else { - return + if !hidden { + if self.proximityDimController == nil { + let proximityDimController = CustomDimController(navigationBarPresentationData: nil) + self.proximityDimController = proximityDimController + (self.viewController as? NavigationController)?.presentOverlay(controller: proximityDimController, inGlobal: true, blockInteraction: false) + } + } else if let proximityDimController = self.proximityDimController { + self.proximityDimController = nil + proximityDimController.dismiss() } - self.customProximityDimView.isHidden = hidden } private func updateBadgeVisibility() { @@ -1172,8 +1173,6 @@ public class Window1 { self.updateBadgeVisibility() self.badgeView.frame = CGRect(origin: CGPoint(x: floorToScreenPixels((self.windowLayout.size.width - image.size.width) / 2.0), y: 5.0), size: image.size) } - - self.customProximityDimView.frame = CGRect(origin: .zero, size: self.windowLayout.size) } } } @@ -1393,3 +1392,25 @@ public extension Window1 { } } // + +private class CustomDimController: ViewController { + class Node: ASDisplayNode { + override init() { + super.init() + + self.backgroundColor = .black + } + } + override init(navigationBarPresentationData: NavigationBarPresentationData?) { + super.init(navigationBarPresentationData: nil) + } + + required public init(coder aDecoder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + override func loadDisplayNode() { + let node = Node() + self.displayNode = node + } +} diff --git a/submodules/DrawingUI/Sources/DrawingEntitiesView.swift b/submodules/DrawingUI/Sources/DrawingEntitiesView.swift index 78d220a9116..34253dde775 100644 --- a/submodules/DrawingUI/Sources/DrawingEntitiesView.swift +++ b/submodules/DrawingUI/Sources/DrawingEntitiesView.swift @@ -60,6 +60,7 @@ public final class DrawingEntitiesView: UIView, TGPhotoDrawingEntitiesView { private let context: AccountContext private let size: CGSize private let hasBin: Bool + private let isStickerEditor: Bool weak var drawingView: DrawingView? public weak var selectionContainerView: DrawingSelectionContainerView? @@ -95,16 +96,20 @@ public final class DrawingEntitiesView: UIView, TGPhotoDrawingEntitiesView { private let yAxisView = UIView() private let angleLayer = SimpleShapeLayer() private let bin = ComponentView() - + + private let stickerOverlayLayer = SimpleShapeLayer() + private let stickerFrameLayer = SimpleShapeLayer() + public var onInteractionUpdated: (Bool) -> Void = { _ in } public var edgePreviewUpdated: (Bool) -> Void = { _ in } private let hapticFeedback = HapticFeedback() - public init(context: AccountContext, size: CGSize, hasBin: Bool = false) { + public init(context: AccountContext, size: CGSize, hasBin: Bool = false, isStickerEditor: Bool = false) { self.context = context self.size = size self.hasBin = hasBin + self.isStickerEditor = isStickerEditor super.init(frame: CGRect(origin: .zero, size: size)) @@ -140,6 +145,13 @@ public final class DrawingEntitiesView: UIView, TGPhotoDrawingEntitiesView { self.angleLayer.opacity = 0.0 self.angleLayer.lineDashPattern = [12, 12] as [NSNumber] + self.stickerOverlayLayer.fillColor = UIColor(rgb: 0x000000, alpha: 0.6).cgColor + + self.stickerFrameLayer.fillColor = UIColor.clear.cgColor + self.stickerFrameLayer.strokeColor = UIColor(rgb: 0xffffff, alpha: 0.55).cgColor + self.stickerFrameLayer.lineDashPattern = [24, 24] as [NSNumber] + self.stickerFrameLayer.lineCap = .round + self.addSubview(self.topEdgeView) self.addSubview(self.leftEdgeView) self.addSubview(self.rightEdgeView) @@ -148,12 +160,25 @@ public final class DrawingEntitiesView: UIView, TGPhotoDrawingEntitiesView { self.addSubview(self.xAxisView) self.addSubview(self.yAxisView) self.layer.addSublayer(self.angleLayer) + + if isStickerEditor { + self.layer.addSublayer(self.stickerOverlayLayer) + self.layer.addSublayer(self.stickerFrameLayer) + } } required init?(coder: NSCoder) { fatalError("init(coder:) has not been implemented") } + public override func addSubview(_ view: UIView) { + super.addSubview(view) + if self.stickerOverlayLayer.superlayer != nil, view is DrawingEntityView { + self.layer.addSublayer(self.stickerOverlayLayer) + self.layer.addSublayer(self.stickerFrameLayer) + } + } + public override func layoutSubviews() { super.layoutSubviews() @@ -189,6 +214,25 @@ public final class DrawingEntitiesView: UIView, TGPhotoDrawingEntitiesView { self.angleLayer.path = anglePath self.angleLayer.lineWidth = width self.angleLayer.bounds = CGRect(origin: .zero, size: CGSize(width: 3000.0, height: width)) + + let frameWidth = floor(self.bounds.width * 0.97) + let frameRect = CGRect(origin: CGPoint(x: floor((self.bounds.width - frameWidth) / 2.0), y: floor((self.bounds.height - frameWidth) / 2.0)), size: CGSize(width: frameWidth, height: frameWidth)) + + self.stickerOverlayLayer.frame = self.bounds + + let overlayOuterRect = UIBezierPath(rect: self.bounds) + let overlayInnerRect = UIBezierPath(cgPath: CGPath(roundedRect: frameRect, cornerWidth: frameWidth / 8.0, cornerHeight: frameWidth / 8.0, transform: nil)) + let overlayLineWidth: CGFloat = 2.0 * 2.2 + + overlayOuterRect.append(overlayInnerRect) + overlayOuterRect.usesEvenOddFillRule = true + + self.stickerOverlayLayer.path = overlayOuterRect.cgPath + self.stickerOverlayLayer.fillRule = .evenOdd + + self.stickerFrameLayer.frame = self.bounds + self.stickerFrameLayer.lineWidth = overlayLineWidth + self.stickerFrameLayer.path = CGPath(roundedRect: frameRect.insetBy(dx: -overlayLineWidth / 2.0, dy: -overlayLineWidth / 2.0), cornerWidth: frameWidth / 8.0 * 1.02, cornerHeight: frameWidth / 8.0 * 1.02, transform: nil) } public var entities: [DrawingEntity] { @@ -841,7 +885,7 @@ public final class DrawingEntitiesView: UIView, TGPhotoDrawingEntitiesView { } else if self.autoSelectEntities, gestureRecognizer.numberOfTouches == 1, let viewToSelect = self.entity(at: location) { self.selectEntity(viewToSelect.entity, animate: false) self.onInteractionUpdated(true) - } else if gestureRecognizer.numberOfTouches == 2, let mediaEntityView = self.subviews.first(where: { $0 is DrawingEntityMediaView }) as? DrawingEntityMediaView { + } else if gestureRecognizer.numberOfTouches == 2 || (self.isStickerEditor && self.autoSelectEntities), let mediaEntityView = self.subviews.first(where: { $0 is DrawingEntityMediaView }) as? DrawingEntityMediaView { mediaEntityView.handlePan(gestureRecognizer) } } diff --git a/submodules/DrawingUI/Sources/DrawingScreen.swift b/submodules/DrawingUI/Sources/DrawingScreen.swift index b996d290e13..8ce1998d791 100644 --- a/submodules/DrawingUI/Sources/DrawingScreen.swift +++ b/submodules/DrawingUI/Sources/DrawingScreen.swift @@ -2634,6 +2634,7 @@ public class DrawingScreen: ViewController, TGPhotoDrawingInterfaceController, U bottom: layout.intrinsicInsets.bottom + layout.safeInsets.bottom, right: layout.safeInsets.right ), + additionalInsets: layout.additionalInsets, inputHeight: layout.inputHeight ?? 0.0, metrics: layout.metrics, deviceMetrics: layout.deviceMetrics, diff --git a/submodules/DrawingUI/Sources/ImageObjectSeparation.swift b/submodules/DrawingUI/Sources/ImageObjectSeparation.swift deleted file mode 100644 index 764975d4f6b..00000000000 --- a/submodules/DrawingUI/Sources/ImageObjectSeparation.swift +++ /dev/null @@ -1,84 +0,0 @@ -import Foundation -import UIKit -import Vision -import CoreImage -import CoreImage.CIFilterBuiltins -import SwiftSignalKit -import VideoToolbox - -private let queue = Queue() - -public func cutoutStickerImage(from image: UIImage) -> Signal { - if #available(iOS 17.0, *) { - guard let cgImage = image.cgImage else { - return .single(nil) - } - return Signal { subscriber in - let ciContext = CIContext(options: nil) - let inputImage = CIImage(cgImage: cgImage) - let handler = VNImageRequestHandler(cgImage: cgImage, options: [:]) - let request = VNGenerateForegroundInstanceMaskRequest { [weak handler] request, error in - guard let handler, let result = request.results?.first as? VNInstanceMaskObservation else { - subscriber.putNext(nil) - subscriber.putCompletion() - return - } - let instances = instances(atPoint: nil, inObservation: result) - if let mask = try? result.generateScaledMaskForImage(forInstances: instances, from: handler) { - let filter = CIFilter.blendWithMask() - filter.inputImage = inputImage - filter.backgroundImage = CIImage(color: .clear) - filter.maskImage = CIImage(cvPixelBuffer: mask) - if let output = filter.outputImage, let cgImage = ciContext.createCGImage(output, from: inputImage.extent) { - let image = UIImage(cgImage: cgImage) - subscriber.putNext(image) - subscriber.putCompletion() - return - } - } - subscriber.putNext(nil) - subscriber.putCompletion() - } - try? handler.perform([request]) - return ActionDisposable { - request.cancel() - } - } - |> runOn(queue) - } else { - return .single(nil) - } -} - -@available(iOS 17.0, *) -private func instances(atPoint maybePoint: CGPoint?, inObservation observation: VNInstanceMaskObservation) -> IndexSet { - guard let point = maybePoint else { - return observation.allInstances - } - - let instanceMap = observation.instanceMask - let coords = VNImagePointForNormalizedPoint(point, CVPixelBufferGetWidth(instanceMap) - 1, CVPixelBufferGetHeight(instanceMap) - 1) - - CVPixelBufferLockBaseAddress(instanceMap, .readOnly) - guard let pixels = CVPixelBufferGetBaseAddress(instanceMap) else { - fatalError() - } - let bytesPerRow = CVPixelBufferGetBytesPerRow(instanceMap) - let instanceLabel = pixels.load(fromByteOffset: Int(coords.y) * bytesPerRow + Int(coords.x), as: UInt8.self) - CVPixelBufferUnlockBaseAddress(instanceMap, .readOnly) - - return instanceLabel == 0 ? observation.allInstances : [Int(instanceLabel)] -} - -private extension UIImage { - convenience init?(pixelBuffer: CVPixelBuffer) { - var cgImage: CGImage? - VTCreateCGImageFromCVPixelBuffer(pixelBuffer, options: nil, imageOut: &cgImage) - - guard let cgImage = cgImage else { - return nil - } - - self.init(cgImage: cgImage) - } -} diff --git a/submodules/DrawingUI/Sources/StickerPickerScreen.swift b/submodules/DrawingUI/Sources/StickerPickerScreen.swift index f56914ff3f7..270874cdbc9 100644 --- a/submodules/DrawingUI/Sources/StickerPickerScreen.swift +++ b/submodules/DrawingUI/Sources/StickerPickerScreen.swift @@ -192,6 +192,7 @@ private final class StickerSelectionComponent: Component { insertText: { _ in }, backwardsDeleteText: {}, + openStickerEditor: {}, presentController: { [weak self] c, a in if let self, let controller = self.component?.getController() { controller.present(c, in: .window(.root), with: a) diff --git a/submodules/Emoji/Sources/EmojiUtils.swift b/submodules/Emoji/Sources/EmojiUtils.swift index 690a82aebc8..4b5033e7b85 100644 --- a/submodules/Emoji/Sources/EmojiUtils.swift +++ b/submodules/Emoji/Sources/EmojiUtils.swift @@ -2,7 +2,7 @@ import Foundation import CoreText import AVFoundation -extension Character { +public extension Character { var isSimpleEmoji: Bool { guard let firstScalar = unicodeScalars.first else { return false } if #available(iOS 10.2, macOS 10.12.2, *) { diff --git a/submodules/GalleryData/Sources/GalleryData.swift b/submodules/GalleryData/Sources/GalleryData.swift index 916c791d542..26087254b04 100644 --- a/submodules/GalleryData/Sources/GalleryData.swift +++ b/submodules/GalleryData/Sources/GalleryData.swift @@ -236,6 +236,11 @@ public func chatMessageGalleryControllerData(context: AccountContext, chatLocati }*/ } + var source = source + if standalone { + source = .standaloneMessage(message) + } + if internalDocumentItemSupportsMimeType(file.mimeType, fileName: file.fileName ?? "file") { let gallery = GalleryController(context: context, source: source ?? .peerMessagesAtId(messageId: message.id, chatLocation: chatLocation ?? .peer(id: message.id.peerId), customTag: chatFilterTag, chatLocationContextHolder: chatLocationContextHolder ?? Atomic(value: nil)), invertItemOrder: reverseMessageGalleryOrder, streamSingleVideo: stream, fromPlayingVideo: autoplayingVideo, landscape: landscape, timecode: timecode, synchronousLoad: synchronousLoad, replaceRootController: { [weak navigationController] controller, ready in navigationController?.replaceTopController(controller, animated: false, ready: ready) diff --git a/submodules/GalleryUI/Sources/ChatItemGalleryFooterContentNode.swift b/submodules/GalleryUI/Sources/ChatItemGalleryFooterContentNode.swift index 1ad9b62ab5e..29ca5ae77b8 100644 --- a/submodules/GalleryUI/Sources/ChatItemGalleryFooterContentNode.swift +++ b/submodules/GalleryUI/Sources/ChatItemGalleryFooterContentNode.swift @@ -819,8 +819,13 @@ final class ChatItemGalleryFooterContentNode: GalleryFooterContentNode, UIScroll func setMessage(_ message: Message, displayInfo: Bool = true, translateToLanguage: String? = nil, peerIsCopyProtected: Bool = false) { self.currentMessage = message + var displayInfo = displayInfo + if Namespaces.Message.allNonRegular.contains(message.id.namespace) { + displayInfo = false + } + var canDelete: Bool - var canShare = !message.containsSecretMedia + var canShare = !message.containsSecretMedia && !Namespaces.Message.allNonRegular.contains(message.id.namespace) var canFullscreen = false diff --git a/submodules/GalleryUI/Sources/GalleryController.swift b/submodules/GalleryUI/Sources/GalleryController.swift index 295df4a0122..2ddb27fafed 100644 --- a/submodules/GalleryUI/Sources/GalleryController.swift +++ b/submodules/GalleryUI/Sources/GalleryController.swift @@ -251,13 +251,18 @@ public func galleryItemForEntry( if let result = addLocallyGeneratedEntities(text, enabledTypes: [.timecode], entities: entities, mediaDuration: file.duration.flatMap(Double.init)) { entities = result } + + var originData = GalleryItemOriginData(title: message.effectiveAuthor.flatMap(EnginePeer.init)?.displayTitle(strings: presentationData.strings, displayOrder: presentationData.nameDisplayOrder), timestamp: message.timestamp) + if Namespaces.Message.allNonRegular.contains(message.id.namespace) { + originData = GalleryItemOriginData(title: nil, timestamp: nil) + } let caption = galleryCaptionStringWithAppliedEntities(context: context, text: text, entities: entities, message: message) return UniversalVideoGalleryItem( context: context, presentationData: presentationData, content: content, - originData: GalleryItemOriginData(title: message.effectiveAuthor.flatMap(EnginePeer.init)?.displayTitle(strings: presentationData.strings, displayOrder: presentationData.nameDisplayOrder), timestamp: message.timestamp), + originData: originData, indexData: location.flatMap { GalleryItemIndexData(position: Int32($0.index), totalCount: Int32($0.count)) }, contentInfo: .message(message), caption: caption, @@ -351,11 +356,17 @@ public func galleryItemForEntry( } description = galleryCaptionStringWithAppliedEntities(context: context, text: descriptionText, entities: entities, message: message) } + + var originData = GalleryItemOriginData(title: message.effectiveAuthor.flatMap(EnginePeer.init)?.displayTitle(strings: presentationData.strings, displayOrder: presentationData.nameDisplayOrder), timestamp: message.timestamp) + if Namespaces.Message.allNonRegular.contains(message.id.namespace) { + originData = GalleryItemOriginData(title: nil, timestamp: nil) + } + return UniversalVideoGalleryItem( context: context, presentationData: presentationData, content: content, - originData: GalleryItemOriginData(title: message.effectiveAuthor.flatMap(EnginePeer.init)?.displayTitle(strings: presentationData.strings, displayOrder: presentationData.nameDisplayOrder), timestamp: message.timestamp), + originData: originData, indexData: location.flatMap { GalleryItemIndexData(position: Int32($0.index), totalCount: Int32($0.count)) }, contentInfo: .message(message), caption: NSAttributedString(string: ""), @@ -611,7 +622,7 @@ public class GalleryController: ViewController, StandalonePresentableController, case let .replyThread(message): peerIdValue = message.peerId threadIdValue = message.threadId - case .feed: + case .customChatContents: break } if peerIdValue == context.account.peerId, let customTag { @@ -662,8 +673,10 @@ public class GalleryController: ViewController, StandalonePresentableController, let namespaces: MessageIdNamespaces if Namespaces.Message.allScheduled.contains(message.id.namespace) { namespaces = .just(Namespaces.Message.allScheduled) + } else if Namespaces.Message.allQuickReply.contains(message.id.namespace) { + namespaces = .just(Namespaces.Message.allQuickReply) } else { - namespaces = .not(Namespaces.Message.allScheduled) + namespaces = .not(Namespaces.Message.allNonRegular) } let inputTag: HistoryViewInputTag if let customTag { @@ -1397,8 +1410,10 @@ public class GalleryController: ViewController, StandalonePresentableController, let namespaces: MessageIdNamespaces if Namespaces.Message.allScheduled.contains(message.id.namespace) { namespaces = .just(Namespaces.Message.allScheduled) + } else if Namespaces.Message.allQuickReply.contains(message.id.namespace) { + namespaces = .just(Namespaces.Message.allQuickReply) } else { - namespaces = .not(Namespaces.Message.allScheduled) + namespaces = .not(Namespaces.Message.allNonRegular) } let signal = strongSelf.context.account.postbox.aroundMessageHistoryViewForLocation(strongSelf.context.chatLocationInput(for: chatLocation, contextHolder: chatLocationContextHolder), anchor: .index(reloadAroundIndex), ignoreMessagesInTimestampRange: nil, count: 50, clipHoles: false, fixedCombinedReadStates: nil, topTaggedMessageIdNamespaces: [], tag: tag, appendMessagesFromTheSameGroup: false, namespaces: namespaces, orderStatistics: [.combinedLocation]) |> mapToSignal { (view, _, _) -> Signal in diff --git a/submodules/GalleryUI/Sources/Items/UniversalVideoGalleryItem.swift b/submodules/GalleryUI/Sources/Items/UniversalVideoGalleryItem.swift index 688f3ea1b6e..11d90442d43 100644 --- a/submodules/GalleryUI/Sources/Items/UniversalVideoGalleryItem.swift +++ b/submodules/GalleryUI/Sources/Items/UniversalVideoGalleryItem.swift @@ -1184,7 +1184,7 @@ final class UniversalVideoGalleryItemNode: ZoomableContentGalleryItemNode { var hintSeekable = false if let contentInfo = item.contentInfo, case let .message(message) = contentInfo { - if Namespaces.Message.allScheduled.contains(message.id.namespace) { + if Namespaces.Message.allNonRegular.contains(message.id.namespace) { disablePictureInPicture = true } else { let throttledSignal = videoNode.status diff --git a/submodules/HashtagSearchUI/Sources/HashtagSearchController.swift b/submodules/HashtagSearchUI/Sources/HashtagSearchController.swift index a9a40ac489e..f2ea14db130 100644 --- a/submodules/HashtagSearchUI/Sources/HashtagSearchController.swift +++ b/submodules/HashtagSearchUI/Sources/HashtagSearchController.swift @@ -102,6 +102,7 @@ public final class HashtagSearchController: TelegramBaseController { }, hideChatFolderUpdates: { }, openStories: { _, _ in }, dismissNotice: { _ in + }, editPeer: { _ in }) let previousSearchItems = Atomic<[ChatListSearchEntry]?>(value: nil) diff --git a/submodules/ItemListAddressItem/Sources/ItemListAddressItem.swift b/submodules/ItemListAddressItem/Sources/ItemListAddressItem.swift index ede2fdcd3a5..950b799e93e 100644 --- a/submodules/ItemListAddressItem/Sources/ItemListAddressItem.swift +++ b/submodules/ItemListAddressItem/Sources/ItemListAddressItem.swift @@ -20,12 +20,12 @@ public final class ItemListAddressItem: ListViewItem, ItemListItem { let style: ItemListStyle let displayDecorations: Bool let action: (() -> Void)? - let longTapAction: (() -> Void)? + let longTapAction: ((ASDisplayNode, String) -> Void)? let linkItemAction: ((TextLinkItemActionType, TextLinkItem) -> Void)? public let tag: Any? - public init(theme: PresentationTheme, label: String, text: String, imageSignal: Signal<(TransformImageArguments) -> DrawingContext?, NoError>?, selected: Bool? = nil, sectionId: ItemListSectionId, style: ItemListStyle, displayDecorations: Bool = true, action: (() -> Void)?, longTapAction: (() -> Void)? = nil, linkItemAction: ((TextLinkItemActionType, TextLinkItem) -> Void)? = nil, tag: Any? = nil) { + public init(theme: PresentationTheme, label: String, text: String, imageSignal: Signal<(TransformImageArguments) -> DrawingContext?, NoError>?, selected: Bool? = nil, sectionId: ItemListSectionId, style: ItemListStyle, displayDecorations: Bool = true, action: (() -> Void)?, longTapAction: ((ASDisplayNode, String) -> Void)? = nil, linkItemAction: ((TextLinkItemActionType, TextLinkItem) -> Void)? = nil, tag: Any? = nil) { self.theme = theme self.label = label self.text = text @@ -188,7 +188,7 @@ public class ItemListAddressItemNode: ListViewItemNode { var leftOffset: CGFloat = 0.0 var selectionNodeWidthAndApply: (CGFloat, (CGSize, Bool) -> ItemListSelectableControlNode)? if let selected = item.selected { - let (selectionWidth, selectionApply) = selectionNodeLayout(item.theme.list.itemCheckColors.strokeColor, item.theme.list.itemCheckColors.fillColor, item.theme.list.itemCheckColors.foregroundColor, selected, false) + let (selectionWidth, selectionApply) = selectionNodeLayout(item.theme.list.itemCheckColors.strokeColor, item.theme.list.itemCheckColors.fillColor, item.theme.list.itemCheckColors.foregroundColor, selected, .regular) selectionNodeWidthAndApply = (selectionWidth, selectionApply) leftOffset += selectionWidth - 8.0 } @@ -202,11 +202,11 @@ public class ItemListAddressItemNode: ListViewItemNode { let (textLayout, textApply) = makeTextLayout(TextNodeLayoutArguments(attributedString: string, backgroundColor: nil, maximumNumberOfLines: 0, truncationType: .end, constrainedSize: CGSize(width: params.width - leftOffset - leftInset - rightInset - 98.0, height: CGFloat.greatestFiniteMagnitude), alignment: .natural, cutout: nil, insets: UIEdgeInsets())) let padding: CGFloat = !item.label.isEmpty ? 39.0 : 20.0 - let imageSide = min(90.0, max(46.0, textLayout.size.height + padding - 18.0)) + let imageSide = min(90.0, max(66.0, textLayout.size.height + padding - 18.0)) let imageSize = CGSize(width: imageSide, height: imageSide) let imageApply = makeImageLayout(TransformImageArguments(corners: ImageCorners(radius: 4.0), imageSize: imageSize, boundingSize: imageSize, intrinsicInsets: UIEdgeInsets())) - let contentSize = CGSize(width: params.width, height: max(textLayout.size.height + padding, imageSize.height + 18.0)) + let contentSize = CGSize(width: params.width, height: max(textLayout.size.height + padding, imageSize.height + 14.0)) let nodeLayout = ListViewItemNodeLayout(contentSize: contentSize, insets: insets) return (nodeLayout, { [weak self] animation in @@ -268,7 +268,7 @@ public class ItemListAddressItemNode: ListViewItemNode { strongSelf.labelNode.frame = CGRect(origin: CGPoint(x: leftOffset + leftInset, y: 11.0), size: labelLayout.size) strongSelf.textNode.frame = CGRect(origin: CGPoint(x: leftOffset + leftInset, y: item.label.isEmpty ? 11.0 : 31.0), size: textLayout.size) - let imageFrame = CGRect(origin: CGPoint(x: params.width - imageSize.width - rightInset, y: floorToScreenPixels((contentSize.height - imageSize.height) / 2.0)), size: imageSize) + let imageFrame = CGRect(origin: CGPoint(x: params.width - imageSize.width - rightInset, y: 7.0), size: imageSize) strongSelf.imageNode.frame = imageFrame if let icon = strongSelf.iconNode.image { @@ -396,7 +396,10 @@ public class ItemListAddressItemNode: ListViewItemNode { } override public func longTapped() { - self.item?.longTapAction?() + guard let item = self.item else { + return + } + item.longTapAction?(self, item.text) } public var tag: Any? { diff --git a/submodules/ItemListPeerActionItem/Sources/ItemListPeerActionItem.swift b/submodules/ItemListPeerActionItem/Sources/ItemListPeerActionItem.swift index 591f0177488..d51d900bf3a 100644 --- a/submodules/ItemListPeerActionItem/Sources/ItemListPeerActionItem.swift +++ b/submodules/ItemListPeerActionItem/Sources/ItemListPeerActionItem.swift @@ -55,6 +55,9 @@ public class ItemListPeerActionItem: ListViewItem, ItemListItem { if self.alwaysPlain { neighbors.top = .sameSection(alwaysPlain: false) } + if self.alwaysPlain { + neighbors.bottom = .sameSection(alwaysPlain: false) + } let (layout, apply) = node.asyncLayout()(self, params, neighbors) node.contentSize = layout.contentSize @@ -83,6 +86,9 @@ public class ItemListPeerActionItem: ListViewItem, ItemListItem { if self.alwaysPlain { neighbors.top = .sameSection(alwaysPlain: false) } + if self.alwaysPlain { + neighbors.bottom = .sameSection(alwaysPlain: false) + } let (layout, apply) = makeLayout(self, params, neighbors) Queue.mainQueue().async { completion(layout, { _ in diff --git a/submodules/ItemListStickerPackItem/Sources/ItemListStickerPackItem.swift b/submodules/ItemListStickerPackItem/Sources/ItemListStickerPackItem.swift index dd7996cd039..43710530f40 100644 --- a/submodules/ItemListStickerPackItem/Sources/ItemListStickerPackItem.swift +++ b/submodules/ItemListStickerPackItem/Sources/ItemListStickerPackItem.swift @@ -440,7 +440,7 @@ class ItemListStickerPackItemNode: ItemListRevealOptionsItemNode { if case let .check(checked) = item.control { selected = checked } - let sizeAndApply = selectableControlLayout(item.presentationData.theme.list.itemCheckColors.strokeColor, item.presentationData.theme.list.itemCheckColors.fillColor, item.presentationData.theme.list.itemCheckColors.foregroundColor, selected, true) + let sizeAndApply = selectableControlLayout(item.presentationData.theme.list.itemCheckColors.strokeColor, item.presentationData.theme.list.itemCheckColors.fillColor, item.presentationData.theme.list.itemCheckColors.foregroundColor, selected, .compact) selectableControlSizeAndApply = sizeAndApply editingOffset = sizeAndApply.0 } else { diff --git a/submodules/ItemListUI/Sources/ItemListController.swift b/submodules/ItemListUI/Sources/ItemListController.swift index ea6a8cb6336..e0bfc8910e1 100644 --- a/submodules/ItemListUI/Sources/ItemListController.swift +++ b/submodules/ItemListUI/Sources/ItemListController.swift @@ -251,6 +251,13 @@ open class ItemListController: ViewController, KeyShortcutResponder, Presentable public var willDisappear: ((Bool) -> Void)? public var didDisappear: ((Bool) -> Void)? + public var afterTransactionCompleted: (() -> Void)? { + didSet { + if self.isNodeLoaded { + (self.displayNode as! ItemListControllerNode).afterTransactionCompleted = self.afterTransactionCompleted + } + } + } public init(presentationData: ItemListPresentationData, updatedPresentationData: Signal, state: Signal<(ItemListControllerState, (ItemListNodeState, ItemGenerationArguments)), NoError>, tabBarItem: Signal?) { self.state = state @@ -486,6 +493,7 @@ open class ItemListController: ViewController, KeyShortcutResponder, Presentable displayNode.searchActivated = self.searchActivated displayNode.reorderEntry = self.reorderEntry displayNode.reorderCompleted = self.reorderCompleted + displayNode.afterTransactionCompleted = self.afterTransactionCompleted displayNode.listNode.experimentalSnapScrollToItem = self.experimentalSnapScrollToItem displayNode.listNode.didScrollWithOffset = self.didScrollWithOffset displayNode.requestLayout = { [weak self] transition in diff --git a/submodules/ItemListUI/Sources/ItemListControllerNode.swift b/submodules/ItemListUI/Sources/ItemListControllerNode.swift index 32654583a98..995c5a2d1f3 100644 --- a/submodules/ItemListUI/Sources/ItemListControllerNode.swift +++ b/submodules/ItemListUI/Sources/ItemListControllerNode.swift @@ -285,6 +285,7 @@ open class ItemListControllerNode: ASDisplayNode { public var searchActivated: ((Bool) -> Void)? public var reorderEntry: ((Int, Int, [ItemListNodeAnyEntry]) -> Signal)? public var reorderCompleted: (([ItemListNodeAnyEntry]) -> Void)? + public var afterTransactionCompleted: (() -> Void)? public var requestLayout: ((ContainedViewLayoutTransition) -> Void)? public var enableInteractiveDismiss = false { @@ -920,6 +921,8 @@ open class ItemListControllerNode: ASDisplayNode { } } } + + strongSelf.afterTransactionCompleted?() } }) var updateEmptyStateItem = false diff --git a/submodules/ItemListUI/Sources/ItemListSelectableControlNode.swift b/submodules/ItemListUI/Sources/ItemListSelectableControlNode.swift index 87cd88ca508..ceac041c539 100644 --- a/submodules/ItemListUI/Sources/ItemListSelectableControlNode.swift +++ b/submodules/ItemListUI/Sources/ItemListSelectableControlNode.swift @@ -5,6 +5,12 @@ import Display import CheckNode public final class ItemListSelectableControlNode: ASDisplayNode { + public enum Style { + case regular + case compact + case small + } + private let checkNode: CheckNode public init(strokeColor: UIColor, fillColor: UIColor, foregroundColor: UIColor) { @@ -16,8 +22,8 @@ public final class ItemListSelectableControlNode: ASDisplayNode { self.addSubnode(self.checkNode) } - public static func asyncLayout(_ node: ItemListSelectableControlNode?) -> (_ strokeColor: UIColor, _ fillColor: UIColor, _ foregroundColor: UIColor, _ selected: Bool, _ compact: Bool) -> (CGFloat, (CGSize, Bool) -> ItemListSelectableControlNode) { - return { strokeColor, fillColor, foregroundColor, selected, compact in + public static func asyncLayout(_ node: ItemListSelectableControlNode?) -> (_ strokeColor: UIColor, _ fillColor: UIColor, _ foregroundColor: UIColor, _ selected: Bool, _ style: Style) -> (CGFloat, (CGSize, Bool) -> ItemListSelectableControlNode) { + return { strokeColor, fillColor, foregroundColor, selected, style in let resultNode: ItemListSelectableControlNode if let node = node { resultNode = node @@ -25,9 +31,28 @@ public final class ItemListSelectableControlNode: ASDisplayNode { resultNode = ItemListSelectableControlNode(strokeColor: strokeColor, fillColor: fillColor, foregroundColor: foregroundColor) } - return (compact ? 38.0 : 45.0, { size, animated in - let checkSize = CGSize(width: 26.0, height: 26.0) - resultNode.checkNode.frame = CGRect(origin: CGPoint(x: compact ? 11.0 : 13.0, y: floorToScreenPixels((size.height - checkSize.height) / 2.0)), size: checkSize) + let offsetSize: CGFloat + switch style { + case .regular: + offsetSize = 45.0 + case .compact: + offsetSize = 38.0 + case .small: + offsetSize = 44.0 + } + + return (offsetSize, { size, animated in + let checkSize: CGSize + let checkOffset: CGFloat + switch style { + case .regular, .compact: + checkSize = CGSize(width: 26.0, height: 26.0) + checkOffset = style == .compact ? 11.0 : 13.0 + case .small: + checkSize = CGSize(width: 22.0, height: 22.0) + checkOffset = 16.0 + } + resultNode.checkNode.frame = CGRect(origin: CGPoint(x: checkOffset, y: floorToScreenPixels((size.height - checkSize.height) / 2.0)), size: checkSize) resultNode.checkNode.setSelected(selected, animated: animated) return resultNode }) diff --git a/submodules/ItemListUI/Sources/Items/ItemListTextWithLabelItem.swift b/submodules/ItemListUI/Sources/Items/ItemListTextWithLabelItem.swift index 6dd6f49b087..1f4a6c2f4a5 100644 --- a/submodules/ItemListUI/Sources/Items/ItemListTextWithLabelItem.swift +++ b/submodules/ItemListUI/Sources/Items/ItemListTextWithLabelItem.swift @@ -189,7 +189,7 @@ public class ItemListTextWithLabelItemNode: ListViewItemNode { var leftOffset: CGFloat = 0.0 var selectionNodeWidthAndApply: (CGFloat, (CGSize, Bool) -> ItemListSelectableControlNode)? if let selected = item.selected { - let (selectionWidth, selectionApply) = selectionNodeLayout(item.presentationData.theme.list.itemCheckColors.strokeColor, item.presentationData.theme.list.itemCheckColors.fillColor, item.presentationData.theme.list.itemCheckColors.foregroundColor, selected, false) + let (selectionWidth, selectionApply) = selectionNodeLayout(item.presentationData.theme.list.itemCheckColors.strokeColor, item.presentationData.theme.list.itemCheckColors.fillColor, item.presentationData.theme.list.itemCheckColors.foregroundColor, selected, .regular) selectionNodeWidthAndApply = (selectionWidth, selectionApply) leftOffset += selectionWidth - 8.0 } diff --git a/submodules/LegacyComponents/Sources/TGMediaAssetsController.m b/submodules/LegacyComponents/Sources/TGMediaAssetsController.m index 90541a5d095..994ddbf39e1 100644 --- a/submodules/LegacyComponents/Sources/TGMediaAssetsController.m +++ b/submodules/LegacyComponents/Sources/TGMediaAssetsController.m @@ -567,6 +567,9 @@ - (instancetype)initWithContext:(id)context intent:(TGM updateGroupingButtonVisibility(); } file:__FILE_NAME__ line:__LINE__]]; + if (_adjustmentsChangedDisposable) { + [_adjustmentsChangedDisposable dispose]; + } _adjustmentsChangedDisposable = [[SMetaDisposable alloc] init]; [_adjustmentsChangedDisposable setDisposable:[_editingContext.adjustmentsUpdatedSignal startStrictWithNext:^(__unused NSNumber *next) { @@ -583,6 +586,7 @@ - (void)dealloc self.delegate = nil; [_selectionChangedDisposable dispose]; [_tooltipDismissDisposable dispose]; + [_adjustmentsChangedDisposable dispose]; } - (void)loadView diff --git a/submodules/LegacyMediaPickerUI/Sources/LegacyAttachmentMenu.swift b/submodules/LegacyMediaPickerUI/Sources/LegacyAttachmentMenu.swift index e6101f8cc4d..6f668dfad33 100644 --- a/submodules/LegacyMediaPickerUI/Sources/LegacyAttachmentMenu.swift +++ b/submodules/LegacyMediaPickerUI/Sources/LegacyAttachmentMenu.swift @@ -222,7 +222,7 @@ public func legacyMediaEditor(context: AccountContext, peer: Peer, threadTitle: }) } -public func legacyAttachmentMenu(context: AccountContext, peer: Peer, threadTitle: String?, chatLocation: ChatLocation, editMediaOptions: LegacyAttachmentMenuMediaEditing?, saveEditedPhotos: Bool, allowGrouping: Bool, hasSchedule: Bool, canSendPolls: Bool, updatedPresentationData: (initial: PresentationData, signal: Signal), parentController: LegacyController, recentlyUsedInlineBots: [Peer], initialCaption: NSAttributedString, openGallery: @escaping () -> Void, openCamera: @escaping (TGAttachmentCameraView?, TGMenuSheetController?) -> Void, openFileGallery: @escaping () -> Void, openWebSearch: @escaping () -> Void, openMap: @escaping () -> Void, openContacts: @escaping () -> Void, openPoll: @escaping () -> Void, presentSelectionLimitExceeded: @escaping () -> Void, presentCantSendMultipleFiles: @escaping () -> Void, presentJpegConversionAlert: @escaping (@escaping (Bool) -> Void) -> Void, presentSchedulePicker: @escaping (Bool, @escaping (Int32) -> Void) -> Void, presentTimerPicker: @escaping (@escaping (Int32) -> Void) -> Void, sendMessagesWithSignals: @escaping ([Any]?, Bool, Int32, ((String) -> UIView?)?, @escaping () -> Void) -> Void, selectRecentlyUsedInlineBot: @escaping (Peer) -> Void, getCaptionPanelView: @escaping () -> TGCaptionPanelView?, present: @escaping (ViewController, Any?) -> Void) -> TGMenuSheetController { +public func legacyAttachmentMenu(context: AccountContext, peer: Peer?, threadTitle: String?, chatLocation: ChatLocation, editMediaOptions: LegacyAttachmentMenuMediaEditing?, saveEditedPhotos: Bool, allowGrouping: Bool, hasSchedule: Bool, canSendPolls: Bool, updatedPresentationData: (initial: PresentationData, signal: Signal), parentController: LegacyController, recentlyUsedInlineBots: [Peer], initialCaption: NSAttributedString, openGallery: @escaping () -> Void, openCamera: @escaping (TGAttachmentCameraView?, TGMenuSheetController?) -> Void, openFileGallery: @escaping () -> Void, openWebSearch: @escaping () -> Void, openMap: @escaping () -> Void, openContacts: @escaping () -> Void, openPoll: @escaping () -> Void, presentSelectionLimitExceeded: @escaping () -> Void, presentCantSendMultipleFiles: @escaping () -> Void, presentJpegConversionAlert: @escaping (@escaping (Bool) -> Void) -> Void, presentSchedulePicker: @escaping (Bool, @escaping (Int32) -> Void) -> Void, presentTimerPicker: @escaping (@escaping (Int32) -> Void) -> Void, sendMessagesWithSignals: @escaping ([Any]?, Bool, Int32, ((String) -> UIView?)?, @escaping () -> Void) -> Void, selectRecentlyUsedInlineBot: @escaping (Peer) -> Void, getCaptionPanelView: @escaping () -> TGCaptionPanelView?, present: @escaping (ViewController, Any?) -> Void) -> TGMenuSheetController { let defaultVideoPreset = defaultVideoPresetForContext(context) UserDefaults.standard.set(defaultVideoPreset.rawValue as NSNumber, forKey: "TG_preferredVideoPreset_v0") @@ -230,18 +230,20 @@ public func legacyAttachmentMenu(context: AccountContext, peer: Peer, threadTitl let recipientName: String if let threadTitle { recipientName = threadTitle - } else { + } else if let peer { if peer.id == context.account.peerId { recipientName = presentationData.strings.DialogList_SavedMessages } else { recipientName = EnginePeer(peer).displayTitle(strings: presentationData.strings, displayOrder: presentationData.nameDisplayOrder) } + } else { + recipientName = "" } let actionSheetTheme = ActionSheetControllerTheme(presentationData: presentationData) let fontSize = floor(actionSheetTheme.baseFontSize * 20.0 / 17.0) - let isSecretChat = peer.id.namespace == Namespaces.Peer.SecretChat + let isSecretChat = peer?.id.namespace == Namespaces.Peer.SecretChat let controller = TGMenuSheetController(context: parentController.context, dark: false)! controller.dismissesByOutsideTap = true @@ -314,14 +316,14 @@ public func legacyAttachmentMenu(context: AccountContext, peer: Peer, threadTitl carouselItem.selectionLimitExceeded = { presentSelectionLimitExceeded() } - if peer.id != context.account.peerId { + if let peer, peer.id != context.account.peerId { if peer is TelegramUser { carouselItem.hasTimer = hasSchedule } carouselItem.hasSilentPosting = true } carouselItem.hasSchedule = hasSchedule - carouselItem.reminder = peer.id == context.account.peerId + carouselItem.reminder = peer?.id == context.account.peerId carouselItem.presentScheduleController = { media, done in presentSchedulePicker(media, { time in done?(time) @@ -449,7 +451,12 @@ public func legacyAttachmentMenu(context: AccountContext, peer: Peer, threadTitl navigationController.setNavigationBarHidden(true, animated: false) legacyController.bind(controller: navigationController) - let recipientName = EnginePeer(peer).displayTitle(strings: presentationData.strings, displayOrder: presentationData.nameDisplayOrder) + let recipientName: String + if let peer { + recipientName = EnginePeer(peer).displayTitle(strings: presentationData.strings, displayOrder: presentationData.nameDisplayOrder) + } else { + recipientName = "" + } legacyController.enableSizeClassSignal = true @@ -489,12 +496,14 @@ public func legacyAttachmentMenu(context: AccountContext, peer: Peer, threadTitl itemViews.append(locationItem) var peerSupportsPolls = false - if peer is TelegramGroup || peer is TelegramChannel { - peerSupportsPolls = true - } else if let user = peer as? TelegramUser, let _ = user.botInfo { - peerSupportsPolls = true + if let peer { + if peer is TelegramGroup || peer is TelegramChannel { + peerSupportsPolls = true + } else if let user = peer as? TelegramUser, let _ = user.botInfo { + peerSupportsPolls = true + } } - if peerSupportsPolls && canSendMessagesToPeer(peer) && canSendPolls { + if let peer, peerSupportsPolls, canSendMessagesToPeer(peer) && canSendPolls { let pollItem = TGMenuSheetButtonItemView(title: presentationData.strings.AttachmentMenu_Poll, type: TGMenuSheetButtonTypeDefault, fontSize: fontSize, action: { [weak controller] in controller?.dismiss(animated: true) openPoll() diff --git a/submodules/ListMessageItem/Sources/ListMessageFileItemNode.swift b/submodules/ListMessageItem/Sources/ListMessageFileItemNode.swift index c91bb0375ab..723d732467f 100644 --- a/submodules/ListMessageItem/Sources/ListMessageFileItemNode.swift +++ b/submodules/ListMessageItem/Sources/ListMessageFileItemNode.swift @@ -595,7 +595,7 @@ public final class ListMessageFileItemNode: ListMessageNode { var leftOffset: CGFloat = 0.0 var selectionNodeWidthAndApply: (CGFloat, (CGSize, Bool) -> ItemListSelectableControlNode)? if case let .selectable(selected) = item.selection { - let (selectionWidth, selectionApply) = selectionNodeLayout(item.presentationData.theme.theme.list.itemCheckColors.strokeColor, item.presentationData.theme.theme.list.itemCheckColors.fillColor, item.presentationData.theme.theme.list.itemCheckColors.foregroundColor, selected, false) + let (selectionWidth, selectionApply) = selectionNodeLayout(item.presentationData.theme.theme.list.itemCheckColors.strokeColor, item.presentationData.theme.theme.list.itemCheckColors.fillColor, item.presentationData.theme.theme.list.itemCheckColors.foregroundColor, selected, .regular) selectionNodeWidthAndApply = (selectionWidth, selectionApply) leftOffset += selectionWidth } diff --git a/submodules/ListMessageItem/Sources/ListMessageSnippetItemNode.swift b/submodules/ListMessageItem/Sources/ListMessageSnippetItemNode.swift index a640ceb8d12..a585475f196 100644 --- a/submodules/ListMessageItem/Sources/ListMessageSnippetItemNode.swift +++ b/submodules/ListMessageItem/Sources/ListMessageSnippetItemNode.swift @@ -259,7 +259,7 @@ public final class ListMessageSnippetItemNode: ListMessageNode { var leftOffset: CGFloat = 0.0 var selectionNodeWidthAndApply: (CGFloat, (CGSize, Bool) -> ItemListSelectableControlNode)? if case let .selectable(selected) = item.selection { - let (selectionWidth, selectionApply) = selectionNodeLayout(item.presentationData.theme.theme.list.itemCheckColors.strokeColor, item.presentationData.theme.theme.list.itemCheckColors.fillColor, item.presentationData.theme.theme.list.itemCheckColors.foregroundColor, selected, false) + let (selectionWidth, selectionApply) = selectionNodeLayout(item.presentationData.theme.theme.list.itemCheckColors.strokeColor, item.presentationData.theme.theme.list.itemCheckColors.fillColor, item.presentationData.theme.theme.list.itemCheckColors.foregroundColor, selected, .regular) selectionNodeWidthAndApply = (selectionWidth, selectionApply) leftOffset += selectionWidth } diff --git a/submodules/LocationUI/Sources/LocationPickerController.swift b/submodules/LocationUI/Sources/LocationPickerController.swift index c557339e3ac..6d73920913a 100644 --- a/submodules/LocationUI/Sources/LocationPickerController.swift +++ b/submodules/LocationUI/Sources/LocationPickerController.swift @@ -187,7 +187,7 @@ public final class LocationPickerController: ViewController, AttachmentContainab if ["home", "work"].contains(venueType) { completion(TelegramMediaMap(latitude: venue.latitude, longitude: venue.longitude, heading: nil, accuracyRadius: nil, geoPlace: nil, venue: nil, liveBroadcastingTimeout: nil, liveProximityNotificationRadius: nil), nil, nil, nil, nil) } else { - completion(venue, queryId, resultId, nil, nil) + completion(venue, queryId, resultId, venue.venue?.address, nil) } strongSelf.dismiss() }, toggleMapModeSelection: { [weak self] in diff --git a/submodules/LocationUI/Sources/LocationPickerControllerNode.swift b/submodules/LocationUI/Sources/LocationPickerControllerNode.swift index 7b2a3931806..069066261f0 100644 --- a/submodules/LocationUI/Sources/LocationPickerControllerNode.swift +++ b/submodules/LocationUI/Sources/LocationPickerControllerNode.swift @@ -889,11 +889,15 @@ final class LocationPickerControllerNode: ViewControllerTracingNode, CLLocationM } }) - if case let .share(_, selfPeer, _) = self.mode { + switch self.mode { + case let .share(_, selfPeer, _): if let selfPeer { self.headerNode.mapNode.userLocationAnnotation = LocationPinAnnotation(context: context, theme: self.presentationData.theme, peer: selfPeer) } self.headerNode.mapNode.hasPickerAnnotation = true + case .pick: + self.headerNode.mapNode.userLocationAnnotation = LocationPinAnnotation(context: context, theme: self.presentationData.theme, location: TelegramMediaMap(coordinate: CLLocationCoordinate2DMake(0, 0)), queryId: nil, resultId: nil, forcedSelection: true) + self.headerNode.mapNode.hasPickerAnnotation = true } self.listNode.updateFloatingHeaderOffset = { [weak self] offset, listTransition in diff --git a/submodules/MediaPickerUI/Sources/LegacyMediaPickerGallery.swift b/submodules/MediaPickerUI/Sources/LegacyMediaPickerGallery.swift index b1f9141e160..d048ffb3eac 100644 --- a/submodules/MediaPickerUI/Sources/LegacyMediaPickerGallery.swift +++ b/submodules/MediaPickerUI/Sources/LegacyMediaPickerGallery.swift @@ -228,7 +228,7 @@ func presentLegacyMediaPickerGallery(context: AccountContext, peer: EnginePeer?, }) } } - if !isScheduledMessages { + if !isScheduledMessages && peer != nil { model.interfaceView.doneLongPressed = { [weak selectionContext, weak editingContext, weak legacyController, weak model] item in if let legacyController = legacyController, let item = item as? TGMediaPickerGalleryItem, let model = model, let selectionContext = selectionContext { var effectiveHasSchedule = hasSchedule @@ -281,8 +281,8 @@ func presentLegacyMediaPickerGallery(context: AccountContext, peer: EnginePeer?, // let _ = (sendWhenOnlineAvailable - |> take(1) - |> deliverOnMainQueue).start(next: { sendWhenOnlineAvailable in + |> take(1) + |> deliverOnMainQueue).start(next: { sendWhenOnlineAvailable in let legacySheetController = LegacyController(presentation: .custom, theme: presentationData.theme, initialLayout: nil) // MARK: Nicegram RoundedVideos, canSendAsRoundedVideo added let sheetController = TGMediaPickerSendActionSheetController(context: legacyController.context, isDark: true, sendButtonFrame: model.interfaceView.doneButtonFrame, canSendAsRoundedVideo: canSendAsRoundedVideo, canSendSilently: hasSilentPosting, canSendWhenOnline: sendWhenOnlineAvailable && effectiveHasSchedule, canSchedule: effectiveHasSchedule, reminder: reminder, hasTimer: hasTimer) diff --git a/submodules/MediaPickerUI/Sources/MediaPickerScreen.swift b/submodules/MediaPickerUI/Sources/MediaPickerScreen.swift index 5194733f640..535be68ba9c 100644 --- a/submodules/MediaPickerUI/Sources/MediaPickerScreen.swift +++ b/submodules/MediaPickerUI/Sources/MediaPickerScreen.swift @@ -159,6 +159,7 @@ public final class MediaPickerScreen: ViewController, AttachmentContainable { case wallpaper case story case addImage + case createSticker } case assets(PHAssetCollection?, AssetsMode) @@ -277,7 +278,7 @@ public final class MediaPickerScreen: ViewController, AttachmentContainable { self.presentationData = controller.presentationData var assetType: PHAssetMediaType? - if case let .assets(_, mode) = controller.subject, [.wallpaper, .addImage].contains(mode) { + if case let .assets(_, mode) = controller.subject, [.wallpaper, .addImage, .createSticker].contains(mode) { assetType = .image } let mediaAssetsContext = MediaAssetsContext(assetType: assetType) @@ -436,7 +437,7 @@ public final class MediaPickerScreen: ViewController, AttachmentContainable { self.gridNode.scrollView.alwaysBounceVertical = true self.gridNode.scrollView.showsVerticalScrollIndicator = false - if case let .assets(_, mode) = controller.subject, [.wallpaper, .story, .addImage].contains(mode) { + if case let .assets(_, mode) = controller.subject, [.wallpaper, .story, .addImage, .createSticker].contains(mode) { } else { let selectionGesture = MediaPickerGridSelectionGesture() @@ -1570,7 +1571,7 @@ public final class MediaPickerScreen: ViewController, AttachmentContainable { self.titleView.title = collection.localizedTitle ?? presentationData.strings.Attachment_Gallery } else { switch mode { - case .default: + case .default, .createSticker: self.titleView.title = presentationData.strings.MediaPicker_Recents self.titleView.isEnabled = true case .story: @@ -2336,15 +2337,15 @@ public final class MediaPickerScreen: ViewController, AttachmentContainable { return self.controllerNode.defaultTransitionView() } - fileprivate func transitionView(for identifier: String, snapshot: Bool, hideSource: Bool = false) -> UIView? { + public func transitionView(for identifier: String, snapshot: Bool, hideSource: Bool = false) -> UIView? { return self.controllerNode.transitionView(for: identifier, snapshot: snapshot, hideSource: hideSource) } - fileprivate func transitionImage(for identifier: String) -> UIImage? { + public func transitionImage(for identifier: String) -> UIImage? { return self.controllerNode.transitionImage(for: identifier) } - func updateHiddenMediaId(_ id: String?) { + public func updateHiddenMediaId(_ id: String?) { self.controllerNode.hiddenMediaId.set(.single(id)) } diff --git a/submodules/PassportUI/Sources/SecureIdPlaintextFormControllerNode.swift b/submodules/PassportUI/Sources/SecureIdPlaintextFormControllerNode.swift index a13b1d0ca6f..5e98f572940 100644 --- a/submodules/PassportUI/Sources/SecureIdPlaintextFormControllerNode.swift +++ b/submodules/PassportUI/Sources/SecureIdPlaintextFormControllerNode.swift @@ -5,7 +5,6 @@ import Display import TelegramCore import Postbox import SwiftSignalKit -import CoreTelephony import TelegramPresentationData import AccountContext import AlertUI @@ -382,17 +381,9 @@ extension SecureIdPlaintextFormInnerState { init(type: SecureIdPlaintextFormType, immediatelyAvailableValue: SecureIdValue?) { switch type { case .phone: - var countryId: String? = nil - if let carrier = CTTelephonyNetworkInfo().serviceSubscriberCellularProviders?.values.first { - countryId = carrier.isoCountryCode - } - - if countryId == nil { - countryId = (Locale.current as NSLocale).object(forKey: .countryCode) as? String - } + let countryId = (Locale.current as NSLocale).object(forKey: .countryCode) as? String var countryCodeAndId: (Int32, String) = (1, "US") - if let countryId = countryId { let normalizedId = countryId.uppercased() for (code, idAndName) in countryCodeToIdAndName { diff --git a/submodules/PeerInfoUI/Sources/DeviceContactInfoController.swift b/submodules/PeerInfoUI/Sources/DeviceContactInfoController.swift index f55b840c697..c4daf6d5e22 100644 --- a/submodules/PeerInfoUI/Sources/DeviceContactInfoController.swift +++ b/submodules/PeerInfoUI/Sources/DeviceContactInfoController.swift @@ -510,7 +510,7 @@ private enum DeviceContactInfoEntry: ItemListNodeEntry { } else { arguments.openAddress(value) } - }, longTapAction: { + }, longTapAction: { _, _ in if selected == nil { arguments.displayCopyContextMenu(.info(index), string) } diff --git a/submodules/Postbox/Sources/AdditionalMessageHistoryViewData.swift b/submodules/Postbox/Sources/AdditionalMessageHistoryViewData.swift index 9cf5cb60e61..88ccb841c34 100644 --- a/submodules/Postbox/Sources/AdditionalMessageHistoryViewData.swift +++ b/submodules/Postbox/Sources/AdditionalMessageHistoryViewData.swift @@ -1,6 +1,6 @@ import Foundation -public enum AdditionalMessageHistoryViewData { +public enum AdditionalMessageHistoryViewData: Equatable { case cachedPeerData(PeerId) case cachedPeerDataMessages(PeerId) case peerChatState(PeerId) diff --git a/submodules/Postbox/Sources/ChatLocation.swift b/submodules/Postbox/Sources/ChatLocation.swift index 45a6eaacd08..b4e4cfdb54f 100644 --- a/submodules/Postbox/Sources/ChatLocation.swift +++ b/submodules/Postbox/Sources/ChatLocation.swift @@ -4,7 +4,7 @@ import SwiftSignalKit public enum ChatLocationInput { case peer(peerId: PeerId, threadId: Int64?) case thread(peerId: PeerId, threadId: Int64, data: Signal) - case feed(id: Int32, data: Signal) + case customChatContents } public extension ChatLocationInput { @@ -14,7 +14,7 @@ public extension ChatLocationInput { return peerId case let .thread(peerId, _, _): return peerId - case .feed: + case .customChatContents: return nil } } @@ -25,7 +25,7 @@ public extension ChatLocationInput { return threadId case let .thread(_, threadId, _): return threadId - case .feed: + case .customChatContents: return nil } } diff --git a/submodules/Postbox/Sources/Message.swift b/submodules/Postbox/Sources/Message.swift index 4a85acefda2..255cf48937e 100644 --- a/submodules/Postbox/Sources/Message.swift +++ b/submodules/Postbox/Sources/Message.swift @@ -639,6 +639,10 @@ public extension MessageAttribute { public struct MessageGroupInfo: Equatable { public let stableId: UInt32 + + public init(stableId: UInt32) { + self.stableId = stableId + } } public final class Message { @@ -732,6 +736,14 @@ public final class Message { self.associatedStories = associatedStories } + public func withUpdatedStableId(stableId: UInt32) -> Message { + return Message(stableId: stableId, stableVersion: self.stableVersion, id: self.id, globallyUniqueId: self.globallyUniqueId, groupingKey: self.groupingKey, groupInfo: self.groupInfo, threadId: self.threadId, timestamp: self.timestamp, flags: self.flags, tags: self.tags, globalTags: self.globalTags, localTags: self.localTags, customTags: self.customTags, forwardInfo: self.forwardInfo, author: self.author, text: self.text, attributes: self.attributes, media: self.media, peers: self.peers, associatedMessages: self.associatedMessages, associatedMessageIds: self.associatedMessageIds, associatedMedia: self.associatedMedia, associatedThreadInfo: self.associatedThreadInfo, associatedStories: self.associatedStories) + } + + public func withUpdatedId(id: MessageId) -> Message { + return Message(stableId: self.stableId, stableVersion: self.stableVersion, id: id, globallyUniqueId: self.globallyUniqueId, groupingKey: self.groupingKey, groupInfo: self.groupInfo, threadId: self.threadId, timestamp: self.timestamp, flags: self.flags, tags: self.tags, globalTags: self.globalTags, localTags: self.localTags, customTags: self.customTags, forwardInfo: self.forwardInfo, author: self.author, text: self.text, attributes: self.attributes, media: self.media, peers: self.peers, associatedMessages: self.associatedMessages, associatedMessageIds: self.associatedMessageIds, associatedMedia: self.associatedMedia, associatedThreadInfo: self.associatedThreadInfo, associatedStories: self.associatedStories) + } + public func withUpdatedStableVersion(stableVersion: UInt32) -> Message { return Message(stableId: self.stableId, stableVersion: stableVersion, id: self.id, globallyUniqueId: self.globallyUniqueId, groupingKey: self.groupingKey, groupInfo: self.groupInfo, threadId: self.threadId, timestamp: self.timestamp, flags: self.flags, tags: self.tags, globalTags: self.globalTags, localTags: self.localTags, customTags: self.customTags, forwardInfo: self.forwardInfo, author: self.author, text: self.text, attributes: self.attributes, media: self.media, peers: self.peers, associatedMessages: self.associatedMessages, associatedMessageIds: self.associatedMessageIds, associatedMedia: self.associatedMedia, associatedThreadInfo: self.associatedThreadInfo, associatedStories: self.associatedStories) } @@ -1017,7 +1029,7 @@ final class InternalStoreMessage { } } -public enum MessageIdNamespaces { +public enum MessageIdNamespaces: Equatable { case all case just(Set) case not(Set) @@ -1033,3 +1045,23 @@ public enum MessageIdNamespaces { } } } + +public struct PeerAndThreadId: Hashable { + public var peerId: PeerId + public var threadId: Int64? + + public init(peerId: PeerId, threadId: Int64?) { + self.peerId = peerId + self.threadId = threadId + } +} + +public struct MessageAndThreadId: Hashable { + public var messageId: MessageId + public var threadId: Int64? + + public init(messageId: MessageId, threadId: Int64?) { + self.messageId = messageId + self.threadId = threadId + } +} diff --git a/submodules/Postbox/Sources/MessageHistoryView.swift b/submodules/Postbox/Sources/MessageHistoryView.swift index 22cf028d3dc..cca6c8d97d1 100644 --- a/submodules/Postbox/Sources/MessageHistoryView.swift +++ b/submodules/Postbox/Sources/MessageHistoryView.swift @@ -237,7 +237,7 @@ public enum MessageHistoryViewRelativeHoleDirection: Equatable, Hashable, Custom } } -public struct MessageHistoryViewOrderStatistics: OptionSet { +public struct MessageHistoryViewOrderStatistics: OptionSet, Equatable { public var rawValue: Int32 public init(rawValue: Int32) { @@ -290,7 +290,7 @@ public enum MessageHistoryViewInput: Equatable { case external(MessageHistoryViewExternalInput) } -public enum MessageHistoryViewReadState { +public enum MessageHistoryViewReadState: Equatable { case peer([PeerId: CombinedPeerReadState]) } @@ -302,7 +302,7 @@ public enum HistoryViewInputAnchor: Equatable { case unread } -final class MutableMessageHistoryView { +final class MutableMessageHistoryView: MutablePostboxView { private(set) var peerIds: MessageHistoryViewInput private let ignoreMessagesInTimestampRange: ClosedRange? let tag: HistoryViewInputTag? @@ -310,6 +310,7 @@ final class MutableMessageHistoryView { let namespaces: MessageIdNamespaces private let orderStatistics: MessageHistoryViewOrderStatistics private let clipHoles: Bool + private let trackHoles: Bool private let anchor: HistoryViewInputAnchor fileprivate var combinedReadStates: MessageHistoryViewReadState? @@ -333,6 +334,7 @@ final class MutableMessageHistoryView { postbox: PostboxImpl, orderStatistics: MessageHistoryViewOrderStatistics, clipHoles: Bool, + trackHoles: Bool, peerIds: MessageHistoryViewInput, ignoreMessagesInTimestampRange: ClosedRange?, anchor inputAnchor: HistoryViewInputAnchor, @@ -343,13 +345,13 @@ final class MutableMessageHistoryView { namespaces: MessageIdNamespaces, count: Int, topTaggedMessages: [MessageId.Namespace: MessageHistoryTopTaggedMessage?], - additionalDatas: [AdditionalMessageHistoryViewDataEntry], - getMessageCountInRange: (MessageIndex, MessageIndex) -> Int32 + additionalDatas: [AdditionalMessageHistoryViewDataEntry] ) { self.anchor = inputAnchor self.orderStatistics = orderStatistics self.clipHoles = clipHoles + self.trackHoles = trackHoles self.peerIds = peerIds self.ignoreMessagesInTimestampRange = ignoreMessagesInTimestampRange self.combinedReadStates = combinedReadStates @@ -1040,6 +1042,10 @@ final class MutableMessageHistoryView { } func firstHole() -> (MessageHistoryViewHole, MessageHistoryViewRelativeHoleDirection, Int, Int64?)? { + if !self.trackHoles { + return nil + } + switch self.sampledState { case let .loading(loadingSample): switch loadingSample { @@ -1065,9 +1071,13 @@ final class MutableMessageHistoryView { } } } + + func immutableView() -> PostboxView { + return MessageHistoryView(self) + } } -public final class MessageHistoryView { +public final class MessageHistoryView: PostboxView { public let tag: HistoryViewInputTag? public let namespaces: MessageIdNamespaces public let anchorIndex: MessageHistoryAnchorIndex @@ -1101,7 +1111,7 @@ public final class MessageHistoryView { self.topTaggedMessages = [] self.additionalData = [] self.isLoading = isLoading - self.isLoadingEarlier = true + self.isLoadingEarlier = false self.isAddedToChatList = false self.peerStoryStats = [:] } diff --git a/submodules/Postbox/Sources/MessageOfInterestHolesView.swift b/submodules/Postbox/Sources/MessageOfInterestHolesView.swift index 978b7b52c32..48ab8b21d8f 100644 --- a/submodules/Postbox/Sources/MessageOfInterestHolesView.swift +++ b/submodules/Postbox/Sources/MessageOfInterestHolesView.swift @@ -60,7 +60,7 @@ final class MutableMessageOfInterestHolesView: MutablePostboxView { } } self.anchor = anchor - self.wrappedView = MutableMessageHistoryView(postbox: postbox, orderStatistics: [], clipHoles: true, peerIds: peerIds, ignoreMessagesInTimestampRange: nil, anchor: self.anchor, combinedReadStates: nil, transientReadStates: nil, tag: nil, appendMessagesFromTheSameGroup: false, namespaces: .all, count: self.count, topTaggedMessages: [:], additionalDatas: [], getMessageCountInRange: { _, _ in return 0}) + self.wrappedView = MutableMessageHistoryView(postbox: postbox, orderStatistics: [], clipHoles: true, trackHoles: false, peerIds: peerIds, ignoreMessagesInTimestampRange: nil, anchor: self.anchor, combinedReadStates: nil, transientReadStates: nil, tag: nil, appendMessagesFromTheSameGroup: false, namespaces: .all, count: self.count, topTaggedMessages: [:], additionalDatas: []) let _ = self.updateFromView() } @@ -134,7 +134,7 @@ final class MutableMessageOfInterestHolesView: MutablePostboxView { case let .peer(id, threadId): peerIds = postbox.peerIdsForLocation(.peer(peerId: id, threadId: threadId), ignoreRelatedChats: false) } - self.wrappedView = MutableMessageHistoryView(postbox: postbox, orderStatistics: [], clipHoles: true, peerIds: peerIds, ignoreMessagesInTimestampRange: nil, anchor: self.anchor, combinedReadStates: nil, transientReadStates: nil, tag: nil, appendMessagesFromTheSameGroup: false, namespaces: .all, count: self.count, topTaggedMessages: [:], additionalDatas: [], getMessageCountInRange: { _, _ in return 0}) + self.wrappedView = MutableMessageHistoryView(postbox: postbox, orderStatistics: [], clipHoles: true, trackHoles: false, peerIds: peerIds, ignoreMessagesInTimestampRange: nil, anchor: self.anchor, combinedReadStates: nil, transientReadStates: nil, tag: nil, appendMessagesFromTheSameGroup: false, namespaces: .all, count: self.count, topTaggedMessages: [:], additionalDatas: []) return self.updateFromView() } else if self.wrappedView.replay(postbox: postbox, transaction: transaction) { var reloadView = false @@ -167,7 +167,7 @@ final class MutableMessageOfInterestHolesView: MutablePostboxView { case let .peer(id, threadId): peerIds = postbox.peerIdsForLocation(.peer(peerId: id, threadId: threadId), ignoreRelatedChats: false) } - self.wrappedView = MutableMessageHistoryView(postbox: postbox, orderStatistics: [], clipHoles: true, peerIds: peerIds, ignoreMessagesInTimestampRange: nil, anchor: self.anchor, combinedReadStates: nil, transientReadStates: nil, tag: nil, appendMessagesFromTheSameGroup: false, namespaces: .all, count: self.count, topTaggedMessages: [:], additionalDatas: [], getMessageCountInRange: { _, _ in return 0}) + self.wrappedView = MutableMessageHistoryView(postbox: postbox, orderStatistics: [], clipHoles: true, trackHoles: false, peerIds: peerIds, ignoreMessagesInTimestampRange: nil, anchor: self.anchor, combinedReadStates: nil, transientReadStates: nil, tag: nil, appendMessagesFromTheSameGroup: false, namespaces: .all, count: self.count, topTaggedMessages: [:], additionalDatas: []) } return self.updateFromView() diff --git a/submodules/Postbox/Sources/Postbox.swift b/submodules/Postbox/Sources/Postbox.swift index 170b35af202..2b706d4cc7e 100644 --- a/submodules/Postbox/Sources/Postbox.swift +++ b/submodules/Postbox/Sources/Postbox.swift @@ -3019,9 +3019,11 @@ final class PostboxImpl { let isInTransaction: Atomic - private func internalTransaction(_ f: (Transaction) -> T) -> (result: T, updatedTransactionStateVersion: Int64?, updatedMasterClientId: Int64?) { + private func internalTransaction(_ f: (Transaction) -> T, file: String = #file, line: Int = #line) -> (result: T, updatedTransactionStateVersion: Int64?, updatedMasterClientId: Int64?) { let _ = self.isInTransaction.swap(true) + let startTime = CFAbsoluteTimeGetCurrent() + self.valueBox.begin() let transaction = Transaction(queue: self.queue, postbox: self) self.afterBegin(transaction: transaction) @@ -3030,6 +3032,12 @@ final class PostboxImpl { transaction.disposed = true self.valueBox.commit() + let endTime = CFAbsoluteTimeGetCurrent() + let transactionDuration = endTime - startTime + if transactionDuration > 0.1 { + postboxLog("Postbox transaction took \(transactionDuration * 1000.0) ms, from: \(file), on:\(line)") + } + let _ = self.isInTransaction.swap(false) if let currentUpdatedState = self.currentUpdatedState { @@ -3069,13 +3077,14 @@ final class PostboxImpl { } } - public func transaction(userInteractive: Bool = false, ignoreDisabled: Bool = false, _ f: @escaping(Transaction) -> T) -> Signal { + + public func transaction(userInteractive: Bool = false, ignoreDisabled: Bool = false, _ f: @escaping(Transaction) -> T, file: String = #file, line: Int = #line) -> Signal { return Signal { subscriber in let f: () -> Void = { self.beginInternalTransaction(ignoreDisabled: ignoreDisabled, { let (result, updatedTransactionState, updatedMasterClientId) = self.internalTransaction({ transaction in return f(transaction) - }) + }, file: file, line: line) if updatedTransactionState != nil || updatedMasterClientId != nil { //self.pipeNotifier.notify() @@ -3104,7 +3113,7 @@ final class PostboxImpl { switch chatLocation { case let .peer(peerId, threadId): return .single((.peer(peerId: peerId, threadId: threadId), false)) - case .thread(_, _, let data), .feed(_, let data): + case .thread(_, _, let data): return Signal { subscriber in var isHoleFill = false return (data @@ -3114,6 +3123,9 @@ final class PostboxImpl { return (.external(value), wasHoleFill) }).start(next: subscriber.putNext, error: subscriber.putError, completed: subscriber.putCompletion) } + case .customChatContents: + assert(false) + return .never() } } @@ -3358,13 +3370,7 @@ final class PostboxImpl { readStates = transientReadStates } - let mutableView = MutableMessageHistoryView(postbox: self, orderStatistics: orderStatistics, clipHoles: clipHoles, peerIds: peerIds, ignoreMessagesInTimestampRange: ignoreMessagesInTimestampRange, anchor: anchor, combinedReadStates: readStates, transientReadStates: transientReadStates, tag: tag, appendMessagesFromTheSameGroup: appendMessagesFromTheSameGroup, namespaces: namespaces, count: count, topTaggedMessages: topTaggedMessages, additionalDatas: additionalDataEntries, getMessageCountInRange: { lowerBound, upperBound in - if case let .tag(tagMask) = tag { - return Int32(self.messageHistoryTable.getMessageCountInRange(peerId: lowerBound.id.peerId, namespace: lowerBound.id.namespace, tag: tagMask, lowerBound: lowerBound, upperBound: upperBound)) - } else { - return 0 - } - }) + let mutableView = MutableMessageHistoryView(postbox: self, orderStatistics: orderStatistics, clipHoles: clipHoles, trackHoles: true, peerIds: peerIds, ignoreMessagesInTimestampRange: ignoreMessagesInTimestampRange, anchor: anchor, combinedReadStates: readStates, transientReadStates: transientReadStates, tag: tag, appendMessagesFromTheSameGroup: appendMessagesFromTheSameGroup, namespaces: namespaces, count: count, topTaggedMessages: topTaggedMessages, additionalDatas: additionalDataEntries) let initialUpdateType: ViewUpdateType = .Initial @@ -4421,12 +4427,12 @@ public class Postbox { } } - public func transaction(userInteractive: Bool = false, ignoreDisabled: Bool = false, _ f: @escaping(Transaction) -> T) -> Signal { + public func transaction(userInteractive: Bool = false, ignoreDisabled: Bool = false, _ f: @escaping(Transaction) -> T, file: String = #file, line: Int = #line) -> Signal { return Signal { subscriber in let disposable = MetaDisposable() self.impl.with { impl in - disposable.set(impl.transaction(userInteractive: userInteractive, ignoreDisabled: ignoreDisabled, f).start(next: subscriber.putNext, error: subscriber.putError, completed: subscriber.putCompletion)) + disposable.set(impl.transaction(userInteractive: userInteractive, ignoreDisabled: ignoreDisabled, f, file: file, line: line).start(next: subscriber.putNext, error: subscriber.putError, completed: subscriber.putCompletion)) } return disposable diff --git a/submodules/Postbox/Sources/PostboxView.swift b/submodules/Postbox/Sources/PostboxView.swift index 7cf89152ae5..eb629a5d168 100644 --- a/submodules/Postbox/Sources/PostboxView.swift +++ b/submodules/Postbox/Sources/PostboxView.swift @@ -1,9 +1,9 @@ import Foundation -public protocol PostboxView { +public protocol PostboxView: AnyObject { } -protocol MutablePostboxView { +protocol MutablePostboxView: AnyObject { func replay(postbox: PostboxImpl, transaction: PostboxTransaction) -> Bool func refreshDueToExternalTransaction(postbox: PostboxImpl) -> Bool func immutableView() -> PostboxView @@ -16,14 +16,71 @@ final class CombinedMutableView { self.views = views } - func replay(postbox: PostboxImpl, transaction: PostboxTransaction) -> Bool { - var updated = false + func replay(postbox: PostboxImpl, transaction: PostboxTransaction) -> (updated: Bool, updateTrackedHoles: Bool) { + var anyUpdated = false + var updateTrackedHoles = false for (_, view) in self.views { - if view.replay(postbox: postbox, transaction: transaction) { - updated = true + if let mutableView = view as? MutableMessageHistoryView { + var innerUpdated = false + + let previousPeerIds = mutableView.peerIds + + if mutableView.replay(postbox: postbox, transaction: transaction) { + innerUpdated = true + } + + var updateType: ViewUpdateType = .Generic + switch mutableView.peerIds { + case let .single(peerId, threadId): + for key in transaction.currentPeerHoleOperations.keys { + if key.peerId == peerId && key.threadId == threadId { + updateType = .FillHole + break + } + } + case .associated: + var ids = Set() + switch mutableView.peerIds { + case .single, .external: + assertionFailure() + case let .associated(mainPeerId, associatedId): + ids.insert(mainPeerId) + if let associatedId = associatedId { + ids.insert(associatedId.peerId) + } + } + + if !ids.isEmpty { + for key in transaction.currentPeerHoleOperations.keys { + if ids.contains(key.peerId) { + updateType = .FillHole + break + } + } + } + case .external: + break + } + + mutableView.updatePeerIds(transaction: transaction) + if mutableView.peerIds != previousPeerIds { + updateType = .UpdateVisible + + let _ = mutableView.refreshDueToExternalTransaction(postbox: postbox) + innerUpdated = true + } + + if innerUpdated { + anyUpdated = true + updateTrackedHoles = true + let _ = updateType + //pipe.putNext((MessageHistoryView(mutableView), updateType)) + } + } else if view.replay(postbox: postbox, transaction: transaction) { + anyUpdated = true } } - return updated + return (anyUpdated, updateTrackedHoles) } func refreshDueToExternalTransaction(postbox: PostboxImpl) -> Bool { diff --git a/submodules/Postbox/Sources/ViewTracker.swift b/submodules/Postbox/Sources/ViewTracker.swift index 12d96f1118d..ff3f3252c8f 100644 --- a/submodules/Postbox/Sources/ViewTracker.swift +++ b/submodules/Postbox/Sources/ViewTracker.swift @@ -1,7 +1,7 @@ import Foundation import SwiftSignalKit -public enum ViewUpdateType : Equatable { +public enum ViewUpdateType: Equatable { case Initial case InitialUnread(MessageIndex) case Generic @@ -351,10 +351,6 @@ final class ViewTracker { self.updateTrackedChatListHoles() - if updateTrackedHoles { - self.updateTrackedHoles() - } - if self.unsentMessageView.replay(transaction.unsentMessageOperations) { self.unsentViewUpdated() } @@ -414,9 +410,14 @@ final class ViewTracker { } for (mutableView, pipe) in self.combinedViews.copyItems() { - if mutableView.replay(postbox: postbox, transaction: transaction) { + let result = mutableView.replay(postbox: postbox, transaction: transaction) + + if result.updated { pipe.putNext(mutableView.immutableView()) } + if result.updateTrackedHoles { + updateTrackedHoles = true + } } for (view, pipe) in self.failedMessageIdsViews.copyItems() { @@ -425,6 +426,10 @@ final class ViewTracker { } } + if updateTrackedHoles { + self.updateTrackedHoles() + } + self.updateTrackedForumTopicListHoles() } @@ -486,6 +491,26 @@ final class ViewTracker { firstHolesAndTags.insert(MessageHistoryHolesViewEntry(hole: hole, direction: direction, space: space, count: count, userId: userId)) } } + for (view, _) in self.combinedViews.copyItems() { + for (_, subview) in view.views { + if let subview = subview as? MutableMessageHistoryView { + if let (hole, direction, count, userId) = subview.firstHole() { + let space: MessageHistoryHoleOperationSpace + if let tag = subview.tag { + switch tag { + case let .tag(value): + space = .tag(value) + case let .customTag(value, regularTag): + space = .customTag(value, regularTag) + } + } else { + space = .everywhere + } + firstHolesAndTags.insert(MessageHistoryHolesViewEntry(hole: hole, direction: direction, space: space, count: count, userId: userId)) + } + } + } + } if self.messageHistoryHolesView.update(firstHolesAndTags) { for subscriber in self.messageHistoryHolesViewSubscribers.copyItems() { diff --git a/submodules/Postbox/Sources/Views.swift b/submodules/Postbox/Sources/Views.swift index 9fd7e8b9e47..c3b0e95f187 100644 --- a/submodules/Postbox/Sources/Views.swift +++ b/submodules/Postbox/Sources/Views.swift @@ -1,6 +1,52 @@ import Foundation public enum PostboxViewKey: Hashable { + public struct HistoryView: Equatable { + public var peerId: PeerId + public var threadId: Int64? + public var clipHoles: Bool + public var trackHoles: Bool + public var orderStatistics: MessageHistoryViewOrderStatistics + public var ignoreMessagesInTimestampRange: ClosedRange? + public var anchor: HistoryViewInputAnchor + public var combinedReadStates: MessageHistoryViewReadState? + public var transientReadStates: MessageHistoryViewReadState? + public var tag: HistoryViewInputTag? + public var appendMessagesFromTheSameGroup: Bool + public var namespaces: MessageIdNamespaces + public var count: Int + + public init( + peerId: PeerId, + threadId: Int64?, + clipHoles: Bool, + trackHoles: Bool, + orderStatistics: MessageHistoryViewOrderStatistics = [], + ignoreMessagesInTimestampRange: ClosedRange? = nil, + anchor: HistoryViewInputAnchor, + combinedReadStates: MessageHistoryViewReadState? = nil, + transientReadStates: MessageHistoryViewReadState? = nil, + tag: HistoryViewInputTag? = nil, + appendMessagesFromTheSameGroup: Bool, + namespaces: MessageIdNamespaces, + count: Int + ) { + self.peerId = peerId + self.threadId = threadId + self.clipHoles = clipHoles + self.trackHoles = trackHoles + self.orderStatistics = orderStatistics + self.ignoreMessagesInTimestampRange = ignoreMessagesInTimestampRange + self.anchor = anchor + self.combinedReadStates = combinedReadStates + self.transientReadStates = transientReadStates + self.tag = tag + self.appendMessagesFromTheSameGroup = appendMessagesFromTheSameGroup + self.namespaces = namespaces + self.count = count + } + } + case itemCollectionInfos(namespaces: [ItemCollectionId.Namespace]) case itemCollectionIds(namespaces: [ItemCollectionId.Namespace]) case itemCollectionInfo(id: ItemCollectionId) @@ -50,6 +96,7 @@ public enum PostboxViewKey: Hashable { case savedMessagesIndex(peerId: PeerId) case savedMessagesStats(peerId: PeerId) case chatInterfaceState(peerId: PeerId) + case historyView(HistoryView) public func hash(into hasher: inout Hasher) { switch self { @@ -168,6 +215,10 @@ public enum PostboxViewKey: Hashable { hasher.combine(peerId) case let .chatInterfaceState(peerId): hasher.combine(peerId) + case let .historyView(historyView): + hasher.combine(20) + hasher.combine(historyView.peerId) + hasher.combine(historyView.threadId) } } @@ -467,6 +518,12 @@ public enum PostboxViewKey: Hashable { } else { return false } + case let .historyView(historyView): + if case .historyView(historyView) = rhs { + return true + } else { + return false + } } } } @@ -571,5 +628,23 @@ func postboxViewForKey(postbox: PostboxImpl, key: PostboxViewKey) -> MutablePost return MutableMessageHistorySavedMessagesStatsView(postbox: postbox, peerId: peerId) case let .chatInterfaceState(peerId): return MutableChatInterfaceStateView(postbox: postbox, peerId: peerId) + case let .historyView(historyView): + return MutableMessageHistoryView( + postbox: postbox, + orderStatistics: historyView.orderStatistics, + clipHoles: historyView.clipHoles, + trackHoles: historyView.trackHoles, + peerIds: .single(peerId: historyView.peerId, threadId: historyView.threadId), + ignoreMessagesInTimestampRange: historyView.ignoreMessagesInTimestampRange, + anchor: historyView.anchor, + combinedReadStates: historyView.combinedReadStates, + transientReadStates: historyView.transientReadStates, + tag: historyView.tag, + appendMessagesFromTheSameGroup: historyView.appendMessagesFromTheSameGroup, + namespaces: historyView.namespaces, + count: historyView.count, + topTaggedMessages: [:], + additionalDatas: [] + ) } } diff --git a/submodules/PremiumUI/BUILD b/submodules/PremiumUI/BUILD index fe7b51fe7b9..078e57cf19a 100644 --- a/submodules/PremiumUI/BUILD +++ b/submodules/PremiumUI/BUILD @@ -118,6 +118,10 @@ swift_library( "//submodules/AlertUI", "//submodules/TelegramUI/Components/Chat/MergedAvatarsNode", "//submodules/TelegramUI/Components/LottieComponent", + "//submodules/TelegramUI/Components/ListSectionComponent", + "//submodules/TelegramUI/Components/ListActionItemComponent", + "//submodules/TelegramUI/Components/EmojiStatusSelectionComponent", + "//submodules/TelegramUI/Components/EntityKeyboard", ], visibility = [ "//visibility:public", diff --git a/submodules/PremiumUI/Resources/badge b/submodules/PremiumUI/Resources/badge new file mode 100644 index 0000000000000000000000000000000000000000..933f6153e38fe23a63bdbd3b9f36f01174e4b0bf GIT binary patch literal 13043 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 BenS8N literal 0 HcmV?d00001 diff --git a/submodules/PremiumUI/Resources/badge.scn b/submodules/PremiumUI/Resources/badge.scn deleted file mode 100644 index 6c59e8ffb0fb648b8d6228965a70c7ffb59f6dc7..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 27949 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 diff --git a/submodules/PremiumUI/Resources/boost b/submodules/PremiumUI/Resources/boost new file mode 100644 index 0000000000000000000000000000000000000000..3c0b0748ef52213bc7df669208aa37e2401cb0f8 GIT binary patch literal 17568 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 WE#KUAj{8E3Uim)?ALnBZssI2&VmeCz literal 0 HcmV?d00001 diff --git a/submodules/PremiumUI/Resources/boost.scn b/submodules/PremiumUI/Resources/boost.scn deleted file mode 100644 index d301679181d53d8a454871d9379f440cde6d6be5..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 43534 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 diff --git a/submodules/PremiumUI/Resources/business.png b/submodules/PremiumUI/Resources/business.png new file mode 100644 index 0000000000000000000000000000000000000000..ed3acfa1f7adbc22ea8ba02f67a802f8a0070839 GIT binary patch literal 2160 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%0D}(kX?2aEMNmYNmuoO>c=XRjl}d! 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 literal 0 HcmV?d00001 diff --git a/submodules/PremiumUI/Resources/coin_edge.png b/submodules/PremiumUI/Resources/coin_edge.png new file mode 100644 index 0000000000000000000000000000000000000000..f0bf2761f183db46f90c4a82363772e2de528d34 GIT binary patch literal 332 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 Hf!qHD+!%aU literal 0 HcmV?d00001 diff --git a/submodules/PremiumUI/Resources/diagonal_shine.png b/submodules/PremiumUI/Resources/diagonal_shine.png new file mode 100644 index 0000000000000000000000000000000000000000..1194211fb7d5dd881f3fe6bc891947e93545e653 GIT binary patch literal 11771 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 literal 0 HcmV?d00001 diff --git a/submodules/PremiumUI/Resources/emoji b/submodules/PremiumUI/Resources/emoji new file mode 100644 index 0000000000000000000000000000000000000000..6eddddf5e4d42d424d2501a44e6a79a1c7b5b341 GIT binary patch literal 20580 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

BzWK>df(W%&U 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 literal 0 HcmV?d00001 diff --git a/submodules/PremiumUI/Resources/emoji.scn b/submodules/PremiumUI/Resources/emoji.scn deleted file mode 100644 index f4f25b3ba7ae752d155bc6a2aae0753f75c3d88e..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 58392 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⪼JtI0mwqo@`D01F5RDuR(FAT`E_p#?$_L?Q`7L;*2kq$q;ih^UAZ;UbC@ zVdj)Mv%6q`h%jg7oS8G5-JM-ZD4Lm_L|R08M~c!T5Gf+a{or#yy+7iW9+y(2Hd3mT zCZ$WAq^?qTDND+ehDyVvG17Qxg7m&LS(+k!B7G{&lh#O@q)SXPB_V~Sh!m6J(mH8_ zR4Q$jwn{%ruJoI9PP!mnma3$y(sk)i>85mB`dhju-Ir^~wdC4zUAdmzL{62{i{2DHVi{TO|LkKY> zFaSevIV^$(+zBh;Zny{Tg9qRtcmy7UC*ZH}G&~D`hv(r%co|lq6x0MYM=eon)DESh zH1sUWKwVHK>VbNpKBymh5xs;m2_bZj9UEt~e9-z&&wq z{5F2JMlSUeXm#Rw}{#mn&uT#Q#^8;5ZTj^eF&7ycRlg7@P6 z_#i%vkK&8?GOof`@pb$szKL(+J4z2FTN$JbR`Qf#%IiwLGD?}Kysu1B3Y8hkLS?bC zM3EGrAVpDBWsRaKy22Dw5lUEzC^6-Va!0vOYLHswA@VS3KpK)(S%SW`i?qYouE!rr>is7&(%5VTy>uMmHM^1Mm?@xR4=KQ)v7?9Ku#bx@K#`S zpfK>?z>MI7TPmfM=e9^qGf75v}|prHd~vc&D9oZk_NPp z_N}&3E7mS(7qv^;WvxoPs$J8rYt{4tTAS9T^=Jdyh(1A^(2g{NcA@=fHXTTFX&xO) zhtUyq9Gy=W&_(ncx|GTkQcMXA&=6fti|9&POjpx2)Tb468{J7O>2A7*?xP3jA$pV^ zrzhztdWQZ+&(RC?re0gGtJl*X)*sa$(;Mk2dOJN;Pt%{(GxRQcH@&-_rGKt3(j^_} zNDt`C^dfzw&UI6_bXzagx9L0dUHZ@ZFZy16zkX2vT|cj1)GzB*`c?h9{-=J^Xl^`h zbTl%I&PI2mr_smgYYaAq7?8IG`;>jgX0TaoHe1XD3$qA|u{c}DHn38* znQdi1GM6P;Is1uiXFFLXt7bRZU+gZs&ujA9ye_ZLALR{s3UADt@h5pJ-iEj5seB+G z&hz;wUckrk@q8km#6RQH`3ydbFXRCp;>&pvU&)L4YQBa?xx*8DJ>SST@h$uZUdAi= zUcR3nR5+1PAqHaAE&W~Eyltu9ukm1Xs^a;#iykTuxKw%4Wzs)B20rgk&?NjuHXvU}Nm?7sF3_KSA5J~Quuhn*A78Rx8X#kuO-aBe&Iocr;I;`QPU;wkZF@mBE;@r-!S zc;EQ+_|o{Y`1f%ZcjJfRC*o)0)$zX)brX*ynk1SfS|pxIv`=(M^iJd^UP-)__$V{wTZc)^_W=P2D!`Gj4{P<-Xz$cPF_k-C}pOyVliQ?yhq; zxm(>bx7^+4?sO~N(_T&QF|Ui4>1BDny}n+5FWbxUUiR|5SH0ohNbgOrz#HRz;eF*T z@D_VZJ>X%FctLNOSLChqzVp_2)MK9M{ooz&u6g&8Pb8m94oK!FXC`5iCbuMaCQl}> zC##cp{nmavzk{FVr~A+Oo&9cpcfY6K$M5I&_p|*R|7CxO|BC;HKh~e*f8u}c&-W!i z;IHy^zr^3*mzIn2jpb$KyUKqq-&4N7BE6zpMX!o}6}c6!{hO*aYyNA45&uutf2|n# EKf175+W-In diff --git a/submodules/PremiumUI/Resources/gift b/submodules/PremiumUI/Resources/gift new file mode 100644 index 0000000000000000000000000000000000000000..87a9c2a5e8490c3eb9efd5fa2d842144210e5f77 GIT binary patch literal 21826 zcmV)&K#ad1iwFo=14zHYfPdh##mxEYGO1RQ%sDpCHkM;zOZ4ln_xKo#r?Ghr6& z0SjSwSOoilj&KYlp%spWFTrteJe&Y0!k6JBI2lfXQ{gM19c&M$!DVncY=kS|O87c_ z1FnK^!qxB{xDjrJhu~rO5&Re)fuF#m@KbmU9)~C3N%$E&1y93E@JskJkifej7XAVs z!p98A;4xwu`HWtS0!D8}A4XqBA)|=Vk5SC%<d{U<_i6WQ<~rXG~%&V!Xy!$#|2o zlkqO&0OKIz9OFFWOU8A^eZ~XELyUp3F+WU*MPMDVPFMn#f+;W!mX3A7x?1Gq*5zGxsx3GCyNpWL{zZ#Jt0N#C*c?V6|oiutHc;Rst)9)t!~g%479n z6|hQK6|AwW@vJGVsjOF7^H|GRZ?e|1HnGmIZnC~%{lxl(&1Li09&B%RTXr;C%UfaZRYLb?d2Wh-QxYmd(3C^`TQV$ zFh7#to}bKD@-_S{zL7thKZn1Vzns5`|1N(o|2Y2y|2+Q^|2F?e{saDRxEt<{`{6;j z7?nXaduS`)^oU1yR8)Sg)?90^s*lcC-b-Jt>(gWbOWcJmH$z)pW6-6c zGHd&yGSW;2lew_MtgY7d)Yj_y>5N15W|OfRu~uj?)tSq51wC^LEmTcZ=>yOu8OCyZ zr6kh|SBcaB0In!~H~=IgTzMPN8`lb#S>by*wj)wW{S60An>%!Ox57Q)s_wR9D}2KW z_nrpvs9QUNP9Om!f+UcPI*Hd LhK^jPZHoI^4{9Jkp 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 diff --git a/submodules/PremiumUI/Resources/lighterTexture.jpg b/submodules/PremiumUI/Resources/lighterTexture.jpg new file mode 100644 index 0000000000000000000000000000000000000000..029374174778cc17066f932b779d55737278174a GIT binary patch literal 9199 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 O_x}NidutjcIRF4z@jyiY literal 0 HcmV?d00001 diff --git a/submodules/PremiumUI/Resources/lightspeed.scn b/submodules/PremiumUI/Resources/lightspeed.scn deleted file mode 100644 index 660d751f6d9ffe7f61bc427f19ae6329c105598c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 14629 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 ZOp*o3%!vU$1OO2B{+s9S`%Uin{{U@Q3qJq= diff --git a/submodules/PremiumUI/Resources/star b/submodules/PremiumUI/Resources/star index 3c0c2710e319802eb4a6441e3a7a242d6ffdd4db..3b27e0220e6c6fa2b28a4e5fd665c0c47182bc7e 100644 GIT binary patch literal 59131 zcmV*1KzP3&iwFo~#Lr~_19Nm?axQaYZUF3k2V4_N*Y`|%vI)fA6|r{&yOQiGVnb0u z1*8}t6a|7wP_b_8y|L+r>Q&esujSf%?;X9iYwv!u2_d_Rw|tNHd7k%u4ZpBw zXV1=@ojK>s`Jdg)MyDnvTC84PrvL;ZAO;d31u`H9s*s9(jAl!sDJ8U`!5ov2i1&gk zT1}}T6;07SjWJeBK>$ZCiqdL>T6AbkEiffBP-+4oAR|a1B>-hwYd9ry~=g0JCw5fF(*gs6_Fi>Rxp znQ4djqC`mL}G(Mv_P~>v`Vy7lqK3D+Aq2&x+JMP$H73ByJK9NkvH&NgYX$Bwf;9GE_2BGE1^ZvP7~~ z@{@#-{31CmIU~6uxgmKjc_H~Ah0;RO!qU=G52?4biBu=;DD5QeEbS`oCXJOQNQX;D zNhe4rN~cR_OEaabq-&*Hr01o-Ngqg`OJB=WGD22JR!mk+R!in3Ya|Po4V4X(jgw81 zWyn^_*2)+eEBjS;NOnbbRd!eQo9u(^qg*Ce$xF-2$ScXK$(zbs$Xm%ne4>1ce5QPse4%`?e2sjw{3kghzbStqe<}Z{5Ge{PiYUq`$}4;n4HZok%@uKq z@rntG>5AEk)rw7upA`EQ2NWk1XB0V#dy3x`FO+hnLa9|2R#sM4QPxrVC|fGqD?2G; zl(9;qGEv!6*;hG0IZHWLnW4;7u2im7ZdGP0k#fKCq4JXos1z!-s;sJs-CIdsXh?}h=N2(qAcM>)Fm1dzC<9= zjp$A!6Dfp=un<;aC^3VWOUxry5s!$+#8cu0@s@Z;d{n!u3#p5$E2@*#DeB&8vwDbn zrh1lowtAj=p*mB&LcLbKPQ62YOnqE^LVa3&PJK~*O?_MaR3p>KH7bopqtz7A6xCGK zG}DlpPMVsU&YG^8Fin)EmZpa$R?|~cUDHcr(hSrL(%3X3G#fNqHH>DDX0PUe=CJ0R z=8ER3=7uIm^HB3b^G@^LP2{F@b8{=;R?w}uTN$^CZq?lCxYcuO;zqi)cI)I8?AG0_ zhg-Z`Z@0c~{oDq)4RjmiHrQ>58|^mSZG_toZp+xr!_S)@@+grDHZtvYbxP5f{;Qde#R6_qBXr`zf@C-F~u6CE!o$QVzmV2S6f)LA*NTn*_4(N+qGg$ zlEGr>nwCoSOT;p2aaBQ9gW2kAs+%z-HaIca*e%x|M`=CszYIfiYLc-9uG1QV>u6z0 zGMPi;%!XuRXM@$)-I&rR(QHad#-fEt=1-zRvf>Q; z02+cupfP9yn&JUe0$)%y#EP2^vKmvnSW=C~*e;eBV~Vj`yxGtP_axpkBpEB{rxdV% zXfX|Q8@*<4^aRiY5>TLdn4=3*@TG=0NN%v*@x;`TH%ghH1@QZ_IHW^RpnXESs^#?1 zAG89kK^rV)ThI;!fc9AM4j>S81f8(X1%Y7D1#|`7KnMuMQ&}Yt&M7yD1VNEzJWfV) zVoH3V8E>%?S`FN=bT^q3ai81ikW>>^5&IZwW0JUHmv(kD;o-zr-95%&!CI4Gh&A;M zk2R)R6FQoW9q~0`(PQvH#+%IP%@eKd4VGSA6B1)%jVWPx-c5?-6y6e=B7aHk{Y)`woX&O5waNAdb1$PA2MkKb@bv_}qO^rN z1hr4I8lsboeu;KnHki{F<3mZnFT6vLtL%im@f0v{Eii#p&>NV61z15E=mYwKejpw6 z2LosoJ)Yi6f3m?wHrU$+(`=Bo!4Wn%$p&ZH;36BmL65$QyTCElgYj6m#-lwX-_f># zp)f`U19i%iAllNgg^3`)0Hjme69dYc{_0D6pV9KuBDOPn9Y=GXam zJ`J(=HMgY+M^}$=T0kz!31)%WU=ElI=7ITO0ayqY zfeerd7K0^VDOd)UgB4&USOr#tHDE1R2iAiPU?cbuYyz9X7O)lk1h#?g;AgM{?8NP4 zfo#A479g+-`~r4^Jzy``2Yvuh1+-_A&)dHvN|0{tr z_AU2!HkUdF&Vvj1$R%(YT*2FG;5sh70d9g@;I^IPj=he%_*f-y-(K>Yz3zv0I>=s6 zReQHTv6nd74T^B~dYfcJyfHF5(QHldOEf#&V3f9)AubNn#@OcRxrXkaXiSR522{d% zqQBXpQ(X<_M1$ij&qS`WAY<<|Bldr7Qm~yGlKeUM$z#$QlM^kR6ZT6p+lzuN*lk5v ztY)J%Cc)njV>Owhw1s&l&y%Z&poPw+zII=Y<#kHcBEeve=Qmf_Vv4hN%spypnZj8J zo^1-`U4M$zG3j)-m~cr-e0ywRcs8>42

T9Z@KoghvEP2i2m7UG0k&YdRClz@~|({C=e z5T_yNgA1V`;qX#wPk7TY5#P)Io>hemEK10G2SiS+q@5Z z|Hkde?aCd@)o=~mv0RF~k-L+7hI@^BoBNXw&!>}5AD;m}Y9GCi6~Z#M_Zvt;BZ#r)# zZx(MpubH=uw~Du#_YQ9}?>*i&-VWYw-d+gbIKn%{JI}iaVH`Jj-}4^v5ud~NW*;v^$S+lG~_O5KZ>@(S=h<78lM(mF`6mcZtvxpNB zry|ZooQt>|aW~@ohgE$=IjmnX> zMVX>pQK6_(R6|gUP2o^9Dw-5y6)!8sD<&u=DHbZ;Qmj#|RjgOMuQ;VRulOYrMRFqj zBEuu)k^LgmBgaL~i(C=8A@aS*&mylz-i-V{@^=V^5k>_?b&2X4C5e(n$)mbQMMtGX zf##`g3 z$M25c8~;)Kf%t>*N8-=LUx>dHe>wi!_#5%J;=hakQ`uSBO_{CCfxwr1Wr4CtS)wdc zmMbfiRmy6mPFbU@Q;t;5Qof>`t9(s4Px*%OO=Yukv2v+$xpJj)mGUj+TIG7>M&&-` zdF4;aKNEryq9JsqIAL_c+=Mj=dlEiRxRUT7;pc=uRAH)cl~fg>QmCR--BmHFSXD1o z9|*5es#L1}s)4E`Rf=kesz5bDRiiSiMyn>MW~%0?mZ;uRZBo74uchDSe%t#U==X8I iL;a5Q5AWZtf6xAL{S*6VI@Jq7xUOeC>V`@5ANoJP))TM* diff --git a/submodules/PremiumUI/Resources/tag b/submodules/PremiumUI/Resources/tag new file mode 100644 index 0000000000000000000000000000000000000000..5c89a52aa1540a9bb3ef9836374b622c26cb7380 GIT binary patch literal 20564 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 diff --git a/submodules/PremiumUI/Resources/texture.jpg b/submodules/PremiumUI/Resources/texture.jpg index 52057beb40db9b83eca2fb2344b7184277c0e258..c4ff553a06516538629103a79bce558f7421b177 100644 GIT binary patch literal 9195 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 Th97hAaA8WFqA>iH*n9s34@UlC literal 2902 zcmai02~bl<7=9rTD25#BC>}r+#tdG77|)uQ1_Xk^45Fdo3`0XA zg@`uQfGxD&IDLJ6BchQpgPL&ylo7m6+u@Ut3bSc1uM|V{_ID|sgvHt4IZQoKqfmK`C zGpAU_A0b3=T!)IM0?_F3+_4R|zOXZbn8LDL75_Iu+1^egxW~5tnvoU z-(%}L-gb4YDc6m3y|dB89-s%K6cSlnVv_q(p-@;}k@EUZopW6f>~bm1wf>YmyRq$< zl9S~|I^%e~Q_9R?EUH>tQsTWMrD-HN$=26gTLZvoYlymOb+2~cEcz@u4|aij2904Z z_(jrP9J89i$#Tis6)xvDnowT;?(N;7s9Y4p-je!fY7y)iAi8Uf(hagm5L2hrPniqI zC9=d*`+Sx{we(dh@>tERR@mKB84*wR%Ci-WZXh?Uv&~PrUb#2I#d%HsT%a~iP%x)y zNyrio>@-#eFzlIO6c8M-=$%AW&8p0#T_dWuY54 zW2pWQeK$#OP3ybo3`C|ExpSZ{uX|S~uQ1^1`_T8)W`A#gPp`^hA>;~4TI#H(Yv{T3 z#ns&`_PX+;>(u0XdoEU&KXReFZ`APN(JIL&IxHEUUJ3is;#Eb+|5*9(B_mMgQ~PWs z_s4`_j+!{{ug@;ttcvsa$GS8W_G4g?eA9nGbqe%}BNUW8z7L)&G&wGLBm+E6n*ZGi3g?@&rEHO0eQ= zIa?Ju#}2Zl<*~uvU{X<5b!;_P79C-QqWdG30rJZ?sxRNDDtfx7d(Pz$fGx=JzEbC0 zGk=WszTBQE<;#Isg%uyWSKn^uwGY`JX+i#fDJUDoWpmBJns|D|0@aqNhCXaSn-|jwQ z!;{q3$iX$Suam<%lrx)Ol+c=!T{)HxbQG8uQ&gUbVBF8HYgM5mI3%WM z+lLoUrEQ<8tA7g2aWg?vJ@@`0&A7?g?HW;Yn>7K$Bn<|$Td&pqa`oGfaWyhQeih3+ zsNplF^kz1S zV`TPdD8_3oO>-$?BlP}&|77$KRs|Eyb#?9uo)vZ>Rw&oBTv&nD#Kv@=?Gk!BMsM?` zuE07Ii@vJzaNAEP#9C?h*Q?z((X5_|N%zO-9xGUad85B#2cxgUK#5JIcCHl2TcxKgKI2M%#&8Q~DdWJtj-cn19ME!}$ zIllt2n0>ATO-GC$ hNoB=`bH-Q%+?Go0lcqy(LJ%|TGoUud;V}Ed{{ZAX(?b9N diff --git a/submodules/PremiumUI/Sources/BadgeBusinessView.swift b/submodules/PremiumUI/Sources/BadgeBusinessView.swift new file mode 100644 index 00000000000..1127c153662 --- /dev/null +++ b/submodules/PremiumUI/Sources/BadgeBusinessView.swift @@ -0,0 +1,61 @@ +import Foundation +import UIKit +import SceneKit +import Display +import AppBundle + +private let sceneVersion: Int = 1 + +final class BadgeBusinessView: UIView, PhoneDemoDecorationView { + private let sceneView: SCNView + + private var leftParticles: SCNNode? + private var rightParticles: SCNNode? + + override init(frame: CGRect) { + self.sceneView = SCNView(frame: CGRect(origin: .zero, size: frame.size)) + self.sceneView.backgroundColor = .clear + if let scene = loadCompressedScene(name: "business", version: sceneVersion) { + self.sceneView.scene = scene + } + self.sceneView.isUserInteractionEnabled = false + self.sceneView.preferredFramesPerSecond = 60 + + super.init(frame: frame) + + self.alpha = 0.0 + + self.addSubview(self.sceneView) + + self.leftParticles = self.sceneView.scene?.rootNode.childNode(withName: "leftParticles", recursively: false) + self.rightParticles = self.sceneView.scene?.rootNode.childNode(withName: "rightParticles", recursively: false) + } + + required init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + func setVisible(_ visible: Bool) { + if visible, let leftParticles = self.leftParticles, let rightParticles = self.rightParticles, leftParticles.parent == nil { + self.sceneView.scene?.rootNode.addChildNode(leftParticles) + self.sceneView.scene?.rootNode.addChildNode(rightParticles) + } + + let transition = ContainedViewLayoutTransition.animated(duration: 0.3, curve: .linear) + transition.updateAlpha(layer: self.layer, alpha: visible ? 0.5 : 0.0, completion: { [weak self] finished in + if let strongSelf = self, finished && !visible && strongSelf.leftParticles?.parent != nil { + strongSelf.leftParticles?.removeFromParentNode() + strongSelf.rightParticles?.removeFromParentNode() + } + }) + } + + func resetAnimation() { + } + + override func layoutSubviews() { + super.layoutSubviews() + + self.sceneView.frame = CGRect(origin: .zero, size: frame.size) + } +} diff --git a/submodules/PremiumUI/Sources/BadgeStarsView.swift b/submodules/PremiumUI/Sources/BadgeStarsView.swift index f8f80a116d3..f1374916e56 100644 --- a/submodules/PremiumUI/Sources/BadgeStarsView.swift +++ b/submodules/PremiumUI/Sources/BadgeStarsView.swift @@ -13,8 +13,8 @@ final class BadgeStarsView: UIView, PhoneDemoDecorationView { override init(frame: CGRect) { self.sceneView = SCNView(frame: CGRect(origin: .zero, size: frame.size)) self.sceneView.backgroundColor = .clear - if let url = getAppBundle().url(forResource: "badge", withExtension: "scn") { - self.sceneView.scene = try? SCNScene(url: url, options: nil) + if let scene = loadCompressedScene(name: "badge", version: 1) { + self.sceneView.scene = scene } self.sceneView.isUserInteractionEnabled = false self.sceneView.preferredFramesPerSecond = 60 @@ -67,8 +67,8 @@ final class EmojiStarsView: UIView, PhoneDemoDecorationView { override init(frame: CGRect) { self.sceneView = SCNView(frame: CGRect(origin: .zero, size: frame.size)) self.sceneView.backgroundColor = .clear - if let url = getAppBundle().url(forResource: "emoji", withExtension: "scn") { - self.sceneView.scene = try? SCNScene(url: url, options: nil) + if let scene = loadCompressedScene(name: "emoji", version: 1) { + self.sceneView.scene = scene } self.sceneView.isUserInteractionEnabled = false self.sceneView.preferredFramesPerSecond = 60 @@ -121,8 +121,8 @@ final class TagStarsView: UIView, PhoneDemoDecorationView { override init(frame: CGRect) { self.sceneView = SCNView(frame: CGRect(origin: .zero, size: frame.size)) self.sceneView.backgroundColor = .clear - if let url = getAppBundle().url(forResource: "tag", withExtension: "scn") { - self.sceneView.scene = try? SCNScene(url: url, options: nil) + if let scene = loadCompressedScene(name: "tag", version: 1) { + self.sceneView.scene = scene } self.sceneView.isUserInteractionEnabled = false self.sceneView.preferredFramesPerSecond = 60 diff --git a/submodules/PremiumUI/Sources/BoostHeaderBackgroundComponent.swift b/submodules/PremiumUI/Sources/BoostHeaderBackgroundComponent.swift index b82af37cfd7..fc1216ae68e 100644 --- a/submodules/PremiumUI/Sources/BoostHeaderBackgroundComponent.swift +++ b/submodules/PremiumUI/Sources/BoostHeaderBackgroundComponent.swift @@ -12,7 +12,7 @@ import TelegramCore import MultilineTextComponent import TelegramPresentationData -private let sceneVersion: Int = 3 +private let sceneVersion: Int = 1 public final class BoostHeaderBackgroundComponent: Component { let isVisible: Bool @@ -58,7 +58,7 @@ public final class BoostHeaderBackgroundComponent: Component { private func setup() { - guard let url = getAppBundle().url(forResource: "boost", withExtension: "scn"), let scene = try? SCNScene(url: url, options: nil) else { + guard let scene = loadCompressedScene(name: "boost", version: sceneVersion) else { return } diff --git a/submodules/PremiumUI/Sources/BusinessPageComponent.swift b/submodules/PremiumUI/Sources/BusinessPageComponent.swift new file mode 100644 index 00000000000..4a1be46cf05 --- /dev/null +++ b/submodules/PremiumUI/Sources/BusinessPageComponent.swift @@ -0,0 +1,652 @@ +import Foundation +import UIKit +import Display +import ComponentFlow +import SwiftSignalKit +import TelegramCore +import AccountContext +import MultilineTextComponent +import BlurredBackgroundComponent +import Markdown +import TelegramPresentationData +import BundleIconComponent + +private final class HeaderComponent: Component { + let context: AccountContext + let theme: PresentationTheme + let strings: PresentationStrings + + init(context: AccountContext, theme: PresentationTheme, strings: PresentationStrings) { + self.context = context + self.theme = theme + self.strings = strings + } + + static func ==(lhs: HeaderComponent, rhs: HeaderComponent) -> Bool { + if lhs.context !== rhs.context { + return false + } + if lhs.theme !== rhs.theme { + return false + } + return true + } + + final class View: UIView { + private let coin = ComponentView() + private let text = ComponentView() + + private var component: HeaderComponent? + private weak var state: EmptyComponentState? + + override init(frame: CGRect) { + super.init(frame: frame) + } + + required init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + func update(component: HeaderComponent, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + self.component = component + self.state = state + + let containerSize = CGSize(width: min(414.0, availableSize.width), height: 220.0) + + let coinSize = self.coin.update( + transition: .immediate, + component: AnyComponent(PremiumCoinComponent(isIntro: true, isVisible: true, hasIdleAnimations: true)), + environment: {}, + containerSize: containerSize + ) + if let view = self.coin.view { + if view.superview == nil { + self.addSubview(view) + } + view.frame = CGRect(origin: CGPoint(x: floor((availableSize.width - coinSize.width) / 2.0), y: -84.0), size: coinSize) + } + + let textSize = self.text.update( + transition: .immediate, + component: AnyComponent( + MultilineTextComponent( + text: .plain(NSAttributedString(string: component.strings.Premium_Business_Description, font: Font.regular(15.0), textColor: component.theme.list.itemPrimaryTextColor)), + horizontalAlignment: .center, + maximumNumberOfLines: 0, + lineSpacing: 0.2 + ) + ), + environment: {}, + containerSize: CGSize(width: availableSize.width - 32.0, height: 1000.0) + ) + if let view = self.text.view { + if view.superview == nil { + self.addSubview(view) + } + view.frame = CGRect(origin: CGPoint(x: floor((availableSize.width - textSize.width) / 2.0), y: 139.0), size: textSize) + } + + return CGSize(width: availableSize.width, height: 210.0) + } + } + + func makeView() -> View { + return View(frame: CGRect()) + } + + func update(view: View, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + return view.update(component: self, availableSize: availableSize, state: state, environment: environment, transition: transition) + } +} + +private final class ParagraphComponent: CombinedComponent { + let title: String + let titleColor: UIColor + let text: String + let textColor: UIColor + let iconName: String + let iconColor: UIColor + + public init( + title: String, + titleColor: UIColor, + text: String, + textColor: UIColor, + iconName: String, + iconColor: UIColor + ) { + self.title = title + self.titleColor = titleColor + self.text = text + self.textColor = textColor + self.iconName = iconName + self.iconColor = iconColor + } + + static func ==(lhs: ParagraphComponent, rhs: ParagraphComponent) -> Bool { + if lhs.title != rhs.title { + return false + } + if lhs.titleColor != rhs.titleColor { + return false + } + if lhs.text != rhs.text { + return false + } + if lhs.textColor != rhs.textColor { + return false + } + if lhs.iconName != rhs.iconName { + return false + } + if lhs.iconColor != rhs.iconColor { + return false + } + return true + } + + static var body: Body { + let title = Child(MultilineTextComponent.self) + let text = Child(MultilineTextComponent.self) + let icon = Child(BundleIconComponent.self) + + return { context in + let component = context.component + + let leftInset: CGFloat = 64.0 + let rightInset: CGFloat = 32.0 + let textSideInset: CGFloat = leftInset + 8.0 + let spacing: CGFloat = 5.0 + + let textTopInset: CGFloat = 9.0 + + let title = title.update( + component: MultilineTextComponent( + text: .plain(NSAttributedString( + string: component.title, + font: Font.semibold(15.0), + textColor: component.titleColor, + paragraphAlignment: .natural + )), + horizontalAlignment: .center, + maximumNumberOfLines: 1 + ), + availableSize: CGSize(width: context.availableSize.width - leftInset - rightInset, height: CGFloat.greatestFiniteMagnitude), + transition: .immediate + ) + + let textFont = Font.regular(15.0) + let boldTextFont = Font.semibold(15.0) + let textColor = component.textColor + let markdownAttributes = MarkdownAttributes( + body: MarkdownAttributeSet(font: textFont, textColor: textColor), + bold: MarkdownAttributeSet(font: boldTextFont, textColor: textColor), + link: MarkdownAttributeSet(font: textFont, textColor: textColor), + linkAttribute: { _ in + return nil + } + ) + + let text = text.update( + component: MultilineTextComponent( + text: .markdown(text: component.text, attributes: markdownAttributes), + horizontalAlignment: .natural, + maximumNumberOfLines: 0, + lineSpacing: 0.2 + ), + availableSize: CGSize(width: context.availableSize.width - leftInset - rightInset, height: context.availableSize.height), + transition: .immediate + ) + + let icon = icon.update( + component: BundleIconComponent( + name: component.iconName, + tintColor: component.iconColor + ), + availableSize: CGSize(width: context.availableSize.width, height: context.availableSize.height), + transition: .immediate + ) + + context.add(title + .position(CGPoint(x: textSideInset + title.size.width / 2.0, y: textTopInset + title.size.height / 2.0)) + ) + + context.add(text + .position(CGPoint(x: textSideInset + text.size.width / 2.0, y: textTopInset + title.size.height + spacing + text.size.height / 2.0)) + ) + + context.add(icon + .position(CGPoint(x: 47.0, y: textTopInset + 18.0)) + ) + + return CGSize(width: context.availableSize.width, height: textTopInset + title.size.height + text.size.height + 25.0) + } + } +} + +private final class BusinessListComponent: CombinedComponent { + typealias EnvironmentType = (Empty, ScrollChildEnvironment) + + let context: AccountContext + let theme: PresentationTheme + let topInset: CGFloat + let bottomInset: CGFloat + + init(context: AccountContext, theme: PresentationTheme, topInset: CGFloat, bottomInset: CGFloat) { + self.context = context + self.theme = theme + self.topInset = topInset + self.bottomInset = bottomInset + } + + static func ==(lhs: BusinessListComponent, rhs: BusinessListComponent) -> Bool { + if lhs.context !== rhs.context { + return false + } + if lhs.theme !== rhs.theme { + return false + } + if lhs.topInset != rhs.topInset { + return false + } + if lhs.bottomInset != rhs.bottomInset { + return false + } + return true + } + + final class State: ComponentState { + private let context: AccountContext + + private var disposable: Disposable? + var limits: EngineConfiguration.UserLimits = .defaultValue + var premiumLimits: EngineConfiguration.UserLimits = .defaultValue + + var accountPeer: EnginePeer? + + init(context: AccountContext) { + self.context = context + + super.init() + + self.disposable = (context.engine.data.get( + TelegramEngine.EngineData.Item.Configuration.UserLimits(isPremium: false), + TelegramEngine.EngineData.Item.Configuration.UserLimits(isPremium: true), + TelegramEngine.EngineData.Item.Peer.Peer(id: context.account.peerId) + ) + |> deliverOnMainQueue).start(next: { [weak self] limits, premiumLimits, accountPeer in + if let strongSelf = self { + strongSelf.limits = limits + strongSelf.premiumLimits = premiumLimits + strongSelf.accountPeer = accountPeer + strongSelf.updated(transition: .immediate) + } + }) + } + + deinit { + self.disposable?.dispose() + } + } + + func makeState() -> State { + return State(context: self.context) + } + + static var body: Body { + let list = Child(List.self) + + return { context in + let theme = context.component.theme + let strings = context.component.context.sharedContext.currentPresentationData.with { $0 }.strings + + let colors = [ + UIColor(rgb: 0x007aff), + UIColor(rgb: 0x798aff), + UIColor(rgb: 0xac64f3), + UIColor(rgb: 0xc456ae), + UIColor(rgb: 0xe95d44), + UIColor(rgb: 0xf2822a), + UIColor(rgb: 0xe79519), + UIColor(rgb: 0xe7ad19) + ] + + let titleColor = theme.list.itemPrimaryTextColor + let textColor = theme.list.itemSecondaryTextColor + + var items: [AnyComponentWithIdentity] = [] + + items.append( + AnyComponentWithIdentity( + id: "header", + component: AnyComponent(HeaderComponent( + context: context.component.context, + theme: theme, + strings: strings + )) + ) + ) + + items.append( + AnyComponentWithIdentity( + id: "location", + component: AnyComponent(ParagraphComponent( + title: strings.Premium_Business_Location_Title, + titleColor: titleColor, + text: strings.Premium_Business_Location_Text, + textColor: textColor, + iconName: "Premium/Business/Location", + iconColor: colors[0] + )) + ) + ) + + items.append( + AnyComponentWithIdentity( + id: "hours", + component: AnyComponent(ParagraphComponent( + title: strings.Premium_Business_Hours_Title, + titleColor: titleColor, + text: strings.Premium_Business_Hours_Text, + textColor: textColor, + iconName: "Premium/Business/Hours", + iconColor: colors[1] + )) + ) + ) + + items.append( + AnyComponentWithIdentity( + id: "replies", + component: AnyComponent(ParagraphComponent( + title: strings.Premium_Business_Replies_Title, + titleColor: titleColor, + text: strings.Premium_Business_Replies_Text, + textColor: textColor, + iconName: "Premium/Business/Replies", + iconColor: colors[2] + )) + ) + ) + + items.append( + AnyComponentWithIdentity( + id: "greetings", + component: AnyComponent(ParagraphComponent( + title: strings.Premium_Business_Greetings_Title, + titleColor: titleColor, + text: strings.Premium_Business_Greetings_Text, + textColor: textColor, + iconName: "Premium/Business/Greetings", + iconColor: colors[3] + )) + ) + ) + + items.append( + AnyComponentWithIdentity( + id: "away", + component: AnyComponent(ParagraphComponent( + title: strings.Premium_Business_Away_Title, + titleColor: titleColor, + text: strings.Premium_Business_Away_Text, + textColor: textColor, + iconName: "Premium/Business/Away", + iconColor: colors[4] + )) + ) + ) + + items.append( + AnyComponentWithIdentity( + id: "chatbots", + component: AnyComponent(ParagraphComponent( + title: strings.Premium_Business_Chatbots_Title, + titleColor: titleColor, + text: strings.Premium_Business_Chatbots_Text, + textColor: textColor, + iconName: "Premium/Business/Chatbots", + iconColor: colors[5] + )) + ) + ) + + let list = list.update( + component: List(items), + availableSize: CGSize(width: context.availableSize.width, height: 10000.0), + transition: context.transition + ) + + let contentHeight = context.component.topInset - 56.0 + list.size.height + context.component.bottomInset + context.add(list + .position(CGPoint(x: list.size.width / 2.0, y: context.component.topInset + list.size.height / 2.0)) + ) + + return CGSize(width: context.availableSize.width, height: contentHeight) + } + } +} + +final class BusinessPageComponent: CombinedComponent { + typealias EnvironmentType = DemoPageEnvironment + + let context: AccountContext + let theme: PresentationTheme + let neighbors: PageNeighbors + let bottomInset: CGFloat + let updatedBottomAlpha: (CGFloat) -> Void + let updatedDismissOffset: (CGFloat) -> Void + let updatedIsDisplaying: (Bool) -> Void + + init(context: AccountContext, theme: PresentationTheme, neighbors: PageNeighbors, bottomInset: CGFloat, updatedBottomAlpha: @escaping (CGFloat) -> Void, updatedDismissOffset: @escaping (CGFloat) -> Void, updatedIsDisplaying: @escaping (Bool) -> Void) { + self.context = context + self.theme = theme + self.neighbors = neighbors + self.bottomInset = bottomInset + self.updatedBottomAlpha = updatedBottomAlpha + self.updatedDismissOffset = updatedDismissOffset + self.updatedIsDisplaying = updatedIsDisplaying + } + + static func ==(lhs: BusinessPageComponent, rhs: BusinessPageComponent) -> Bool { + if lhs.context !== rhs.context { + return false + } + if lhs.theme !== rhs.theme { + return false + } + if lhs.neighbors != rhs.neighbors { + return false + } + if lhs.bottomInset != rhs.bottomInset { + return false + } + return true + } + + final class State: ComponentState { + let updateBottomAlpha: (CGFloat) -> Void + let updateDismissOffset: (CGFloat) -> Void + let updatedIsDisplaying: (Bool) -> Void + + var resetScroll: ActionSlot? + + var topContentOffset: CGFloat = 0.0 + var bottomContentOffset: CGFloat = 100.0 { + didSet { + self.updateAlpha() + } + } + + var position: CGFloat? { + didSet { + self.updateAlpha() + } + } + + var isDisplaying = false { + didSet { + if oldValue != self.isDisplaying { + self.updatedIsDisplaying(self.isDisplaying) + + if !self.isDisplaying { + self.resetScroll?.invoke(Void()) + } + } + } + } + + var neighbors = PageNeighbors(leftIsList: false, rightIsList: false) + + init(updateBottomAlpha: @escaping (CGFloat) -> Void, updateDismissOffset: @escaping (CGFloat) -> Void, updateIsDisplaying: @escaping (Bool) -> Void) { + self.updateBottomAlpha = updateBottomAlpha + self.updateDismissOffset = updateDismissOffset + self.updatedIsDisplaying = updateIsDisplaying + + super.init() + } + + func updateAlpha() { + var dismissToLeft = false + if let position = self.position, position > 0.0 { + dismissToLeft = true + } + var dismissPosition = min(1.0, abs(self.position ?? 0.0) / 1.3333) + var position = min(1.0, abs(self.position ?? 0.0)) + if position > 0.001, (dismissToLeft && self.neighbors.leftIsList) || (!dismissToLeft && self.neighbors.rightIsList) { + dismissPosition = 0.0 + position = 1.0 + } + self.updateDismissOffset(dismissPosition) + + let verticalPosition = 1.0 - min(30.0, self.bottomContentOffset) / 30.0 + + let backgroundAlpha: CGFloat = max(position, verticalPosition) + self.updateBottomAlpha(backgroundAlpha) + } + } + + func makeState() -> State { + return State(updateBottomAlpha: self.updatedBottomAlpha, updateDismissOffset: self.updatedDismissOffset, updateIsDisplaying: self.updatedIsDisplaying) + } + + static var body: Body { + let background = Child(Rectangle.self) + let scroll = Child(ScrollComponent.self) + let topPanel = Child(BlurredBackgroundComponent.self) + let topSeparator = Child(Rectangle.self) + let title = Child(MultilineTextComponent.self) + + let resetScroll = ActionSlot() + + return { context in + let state = context.state + + let environment = context.environment[DemoPageEnvironment.self].value + state.neighbors = context.component.neighbors + state.resetScroll = resetScroll + state.position = environment.position + state.isDisplaying = environment.isDisplaying + + let theme = context.component.theme + let strings = context.component.context.sharedContext.currentPresentationData.with { $0 }.strings + + let topInset: CGFloat = 56.0 + + let scroll = scroll.update( + component: ScrollComponent( + content: AnyComponent( + BusinessListComponent( + context: context.component.context, + theme: theme, + topInset: topInset, + bottomInset: context.component.bottomInset + 110.0 + ) + ), + contentInsets: UIEdgeInsets(top: topInset, left: 0.0, bottom: 0.0, right: 0.0), + contentOffsetUpdated: { [weak state] topContentOffset, bottomContentOffset in + state?.topContentOffset = topContentOffset + state?.bottomContentOffset = bottomContentOffset + Queue.mainQueue().justDispatch { + state?.updated(transition: .immediate) + } + }, + contentOffsetWillCommit: { _ in }, + resetScroll: resetScroll + ), + availableSize: context.availableSize, + transition: context.transition + ) + + let background = background.update( + component: Rectangle(color: theme.overallDarkAppearance ? theme.list.blocksBackgroundColor : theme.list.plainBackgroundColor), + availableSize: scroll.size, + transition: context.transition + ) + context.add(background + .position(CGPoint(x: context.availableSize.width / 2.0, y: background.size.height / 2.0)) + ) + + context.add(scroll + .position(CGPoint(x: context.availableSize.width / 2.0, y: scroll.size.height / 2.0)) + ) + + let topPanel = topPanel.update( + component: BlurredBackgroundComponent( + color: theme.rootController.navigationBar.blurredBackgroundColor + ), + availableSize: CGSize(width: context.availableSize.width, height: topInset), + transition: context.transition + ) + + let topSeparator = topSeparator.update( + component: Rectangle( + color: theme.rootController.navigationBar.separatorColor + ), + availableSize: CGSize(width: context.availableSize.width, height: UIScreenPixel), + transition: context.transition + ) + + let title = title.update( + component: MultilineTextComponent( + text: .plain(NSAttributedString(string: strings.Premium_Business, font: Font.semibold(20.0), textColor: theme.rootController.navigationBar.primaryTextColor)), + horizontalAlignment: .center, + truncationType: .end, + maximumNumberOfLines: 1 + ), + availableSize: context.availableSize, + transition: context.transition + ) + + let topPanelAlpha: CGFloat + if state.topContentOffset > 78.0 { + topPanelAlpha = min(30.0, state.topContentOffset - 78.0) / 30.0 + } else { + topPanelAlpha = 0.0 + } + context.add(topPanel + .position(CGPoint(x: context.availableSize.width / 2.0, y: topPanel.size.height / 2.0)) + .opacity(topPanelAlpha) + ) + context.add(topSeparator + .position(CGPoint(x: context.availableSize.width / 2.0, y: topPanel.size.height)) + .opacity(topPanelAlpha) + ) + + let titleTopOriginY = topPanel.size.height / 2.0 + let titleBottomOriginY: CGFloat = 176.0 + let titleOriginDelta = titleTopOriginY - titleBottomOriginY + + let fraction = min(1.0, state.topContentOffset / abs(titleOriginDelta)) + let titleOriginY: CGFloat = titleBottomOriginY + fraction * titleOriginDelta + let titleScale = 1.0 - max(0.0, fraction * 0.2) + + context.add(title + .position(CGPoint(x: context.availableSize.width / 2.0, y: titleOriginY)) + .scale(titleScale) + ) + + return scroll.size + } + } +} diff --git a/submodules/PremiumUI/Sources/EmojiHeaderComponent.swift b/submodules/PremiumUI/Sources/EmojiHeaderComponent.swift index 4f020cfca58..765be52fb3f 100644 --- a/submodules/PremiumUI/Sources/EmojiHeaderComponent.swift +++ b/submodules/PremiumUI/Sources/EmojiHeaderComponent.swift @@ -13,8 +13,6 @@ import AnimationCache import MultiAnimationRenderer import EmojiStatusComponent -private let sceneVersion: Int = 3 - class EmojiHeaderComponent: Component { let context: AccountContext let animationCache: AnimationCache diff --git a/submodules/PremiumUI/Sources/FasterStarsView.swift b/submodules/PremiumUI/Sources/FasterStarsView.swift index a96b632a1f9..967fc2213df 100644 --- a/submodules/PremiumUI/Sources/FasterStarsView.swift +++ b/submodules/PremiumUI/Sources/FasterStarsView.swift @@ -5,6 +5,8 @@ import Display import AppBundle import LegacyComponents +private let sceneVersion: Int = 1 + final class FasterStarsView: UIView, PhoneDemoDecorationView { private let sceneView: SCNView @@ -13,8 +15,8 @@ final class FasterStarsView: UIView, PhoneDemoDecorationView { override init(frame: CGRect) { self.sceneView = SCNView(frame: CGRect(origin: .zero, size: frame.size)) self.sceneView.backgroundColor = .clear - if let url = getAppBundle().url(forResource: "lightspeed", withExtension: "scn") { - self.sceneView.scene = try? SCNScene(url: url, options: nil) + if let scene = loadCompressedScene(name: "lightspeed", version: sceneVersion) { + self.sceneView.scene = scene } self.sceneView.isUserInteractionEnabled = false self.sceneView.preferredFramesPerSecond = 60 diff --git a/submodules/PremiumUI/Sources/GiftAvatarComponent.swift b/submodules/PremiumUI/Sources/GiftAvatarComponent.swift index 6a1b8796498..771982bd653 100644 --- a/submodules/PremiumUI/Sources/GiftAvatarComponent.swift +++ b/submodules/PremiumUI/Sources/GiftAvatarComponent.swift @@ -14,7 +14,7 @@ import MergedAvatarsNode import MultilineTextComponent import TelegramPresentationData -private let sceneVersion: Int = 3 +private let sceneVersion: Int = 1 final class GiftAvatarComponent: Component { let context: AccountContext @@ -106,7 +106,7 @@ final class GiftAvatarComponent: Component { } private func setup() { - guard let url = getAppBundle().url(forResource: "gift", withExtension: "scn"), let scene = try? SCNScene(url: url, options: nil) else { + guard let scene = loadCompressedScene(name: "gift", version: sceneVersion) else { return } diff --git a/submodules/PremiumUI/Sources/GiftOptionItem.swift b/submodules/PremiumUI/Sources/GiftOptionItem.swift index 0ff43e8df6e..882912171be 100644 --- a/submodules/PremiumUI/Sources/GiftOptionItem.swift +++ b/submodules/PremiumUI/Sources/GiftOptionItem.swift @@ -296,7 +296,7 @@ class GiftOptionItemNode: ItemListRevealOptionsItemNode { var editingOffset: CGFloat = 0.0 if let isSelected = item.isSelected { - let sizeAndApply = selectableControlLayout(item.presentationData.theme.list.itemCheckColors.strokeColor, item.presentationData.theme.list.itemCheckColors.fillColor, item.presentationData.theme.list.itemCheckColors.foregroundColor, isSelected, false) + let sizeAndApply = selectableControlLayout(item.presentationData.theme.list.itemCheckColors.strokeColor, item.presentationData.theme.list.itemCheckColors.fillColor, item.presentationData.theme.list.itemCheckColors.foregroundColor, isSelected, .regular) selectableControlSizeAndApply = sizeAndApply editingOffset = sizeAndApply.0 } @@ -309,6 +309,10 @@ class GiftOptionItemNode: ItemListRevealOptionsItemNode { textConstrainedWidth -= 54.0 subtitleConstrainedWidth -= 30.0 } + if let _ = item.titleBadge { + textConstrainedWidth -= 32.0 + subtitleConstrainedWidth -= 32.0 + } let (titleLayout, titleApply) = makeTitleLayout(TextNodeLayoutArguments(attributedString: titleAttributedString, maximumNumberOfLines: 1, truncationType: .end, constrainedSize: CGSize(width: textConstrainedWidth, height: .greatestFiniteMagnitude))) let (statusLayout, statusApply) = makeStatusLayout(TextNodeLayoutArguments(attributedString: statusAttributedString, backgroundColor: nil, maximumNumberOfLines: 1, truncationType: .end, constrainedSize: CGSize(width: subtitleConstrainedWidth, height: CGFloat.greatestFiniteMagnitude), alignment: .natural, cutout: nil, insets: UIEdgeInsets())) diff --git a/submodules/PremiumUI/Sources/PhoneDemoComponent.swift b/submodules/PremiumUI/Sources/PhoneDemoComponent.swift index 5a918550c98..d65a427ff33 100644 --- a/submodules/PremiumUI/Sources/PhoneDemoComponent.swift +++ b/submodules/PremiumUI/Sources/PhoneDemoComponent.swift @@ -371,6 +371,7 @@ final class PhoneDemoComponent: Component { case emoji case hello case tag + case business } enum Model { @@ -547,6 +548,13 @@ final class PhoneDemoComponent: Component { self.decorationView = starsView self.decorationContainerView.addSubview(starsView) } + case .business: + if let _ = self.decorationView as? BadgeBusinessView { + } else { + let starsView = BadgeBusinessView(frame: self.decorationContainerView.bounds) + self.decorationView = starsView + self.decorationContainerView.addSubview(starsView) + } } self.phoneView.setup(context: component.context, videoFile: component.videoFile, position: component.position) diff --git a/submodules/PremiumUI/Sources/PremiumBoostLevelsScreen.swift b/submodules/PremiumUI/Sources/PremiumBoostLevelsScreen.swift index 1c97beee065..96c2cbfb8f1 100644 --- a/submodules/PremiumUI/Sources/PremiumBoostLevelsScreen.swift +++ b/submodules/PremiumUI/Sources/PremiumBoostLevelsScreen.swift @@ -698,7 +698,7 @@ private final class SheetContent: CombinedComponent { isCurrent = mode == .current } case .features: - textString = strings.GroupBoost_AdditionalFeaturesText + textString = isGroup ? strings.GroupBoost_AdditionalFeaturesText : strings.ChannelBoost_AdditionalFeaturesText } let defaultTitle = strings.ChannelBoost_Level("\(level)").string diff --git a/submodules/PremiumUI/Sources/PremiumCoinComponent.swift b/submodules/PremiumUI/Sources/PremiumCoinComponent.swift new file mode 100644 index 00000000000..24a3c5ec509 --- /dev/null +++ b/submodules/PremiumUI/Sources/PremiumCoinComponent.swift @@ -0,0 +1,509 @@ +import Foundation +import UIKit +import Display +import ComponentFlow +import SwiftSignalKit +import SceneKit +import GZip +import AppBundle +import LegacyComponents + +private let sceneVersion: Int = 2 + +private func deg2rad(_ number: Float) -> Float { + return number * .pi / 180 +} + +private func rad2deg(_ number: Float) -> Float { + return number * 180.0 / .pi +} + +class PremiumCoinComponent: Component { + let isIntro: Bool + let isVisible: Bool + let hasIdleAnimations: Bool + + init(isIntro: Bool, isVisible: Bool, hasIdleAnimations: Bool) { + self.isIntro = isIntro + self.isVisible = isVisible + self.hasIdleAnimations = hasIdleAnimations + } + + static func ==(lhs: PremiumCoinComponent, rhs: PremiumCoinComponent) -> Bool { + return lhs.isIntro == rhs.isIntro && lhs.isVisible == rhs.isVisible && lhs.hasIdleAnimations == rhs.hasIdleAnimations + } + + final class View: UIView, SCNSceneRendererDelegate, ComponentTaggedView { + final class Tag { + } + + func matches(tag: Any) -> Bool { + if let _ = tag as? Tag { + return true + } + return false + } + + private var _ready = Promise() + var ready: Signal { + return self._ready.get() + } + + weak var animateFrom: UIView? + weak var containerView: UIView? + + private let sceneView: SCNView + + private var previousInteractionTimestamp: Double = 0.0 + private var timer: SwiftSignalKit.Timer? + private var hasIdleAnimations = false + + private let isIntro: Bool + + init(frame: CGRect, isIntro: Bool) { + self.isIntro = isIntro + + self.sceneView = SCNView(frame: CGRect(origin: .zero, size: CGSize(width: 64.0, height: 64.0))) + self.sceneView.backgroundColor = .clear + self.sceneView.transform = CGAffineTransform(scaleX: 0.5, y: 0.5) + self.sceneView.isUserInteractionEnabled = false + self.sceneView.preferredFramesPerSecond = 60 + self.sceneView.isJitteringEnabled = true + + super.init(frame: frame) + + self.addSubview(self.sceneView) + + self.setup() + + let panGestureRecoginzer = UIPanGestureRecognizer(target: self, action: #selector(self.handlePan(_:))) + self.addGestureRecognizer(panGestureRecoginzer) + + let tapGestureRecoginzer = UITapGestureRecognizer(target: self, action: #selector(self.handleTap(_:))) + self.addGestureRecognizer(tapGestureRecoginzer) + + self.disablesInteractiveModalDismiss = true + self.disablesInteractiveTransitionGestureRecognizer = true + } + + required init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + deinit { + self.timer?.invalidate() + } + + private let hapticFeedback = HapticFeedback() + + private var delayTapsTill: Double? + @objc private func handleTap(_ gesture: UITapGestureRecognizer) { + guard let scene = self.sceneView.scene, let node = scene.rootNode.childNode(withName: "star", recursively: false) else { + return + } + + let currentTime = CACurrentMediaTime() + self.previousInteractionTimestamp = currentTime + if let delayTapsTill = self.delayTapsTill, currentTime < delayTapsTill { + return + } + + var left: Bool? + var top: Bool? + if let view = gesture.view { + let point = gesture.location(in: view) + let horizontalDistanceFromCenter = abs(point.x - view.frame.size.width / 2.0) + if horizontalDistanceFromCenter > 60.0 { + return + } + let verticalDistanceFromCenter = abs(point.y - view.frame.size.height / 2.0) + if horizontalDistanceFromCenter > 20.0 { + left = point.x < view.frame.width / 2.0 + } + if verticalDistanceFromCenter > 20.0 { + top = point.y < view.frame.height / 2.0 + } + } + + if node.animationKeys.contains("tapRotate"), let left = left { + self.playAppearanceAnimation(velocity: nil, mirror: left, explode: true) + + self.hapticFeedback.impact(.medium) + return + } + + let initial = node.eulerAngles + var yaw: CGFloat = 0.0 + var pitch: CGFloat = 0.0 + if let left = left { + yaw = left ? -0.6 : 0.6 + } + if let top = top { + pitch = top ? -0.3 : 0.3 + } + let target = SCNVector3(pitch, yaw, 0.0) + + let animation = CABasicAnimation(keyPath: "eulerAngles") + animation.fromValue = NSValue(scnVector3: initial) + animation.toValue = NSValue(scnVector3: target) + animation.duration = 0.25 + animation.timingFunction = CAMediaTimingFunction(name: .easeOut) + animation.fillMode = .forwards + node.addAnimation(animation, forKey: "tapRotate") + + node.eulerAngles = target + + Queue.mainQueue().after(0.25) { + node.eulerAngles = initial + let springAnimation = CASpringAnimation(keyPath: "eulerAngles") + springAnimation.fromValue = NSValue(scnVector3: target) + springAnimation.toValue = NSValue(scnVector3: SCNVector3(x: 0.0, y: 0.0, z: 0.0)) + springAnimation.mass = 1.0 + springAnimation.stiffness = 21.0 + springAnimation.damping = 5.8 + springAnimation.duration = springAnimation.settlingDuration * 0.8 + node.addAnimation(springAnimation, forKey: "tapRotate") + } + + self.hapticFeedback.tap() + } + + private var previousYaw: Float = 0.0 + @objc private func handlePan(_ gesture: UIPanGestureRecognizer) { + guard let scene = self.sceneView.scene, let node = scene.rootNode.childNode(withName: "star", recursively: false) else { + return + } + + self.previousInteractionTimestamp = CACurrentMediaTime() + + let keys = [ + "rotate", + "tapRotate" + ] + + for key in keys { + node.removeAnimation(forKey: key) + } + + switch gesture.state { + case .began: + self.previousYaw = 0.0 + case .changed: + let translation = gesture.translation(in: gesture.view) + let yawPan = deg2rad(Float(translation.x)) + + func rubberBandingOffset(offset: CGFloat, bandingStart: CGFloat) -> CGFloat { + let bandedOffset = offset - bandingStart + let range: CGFloat = 60.0 + let coefficient: CGFloat = 0.4 + return bandingStart + (1.0 - (1.0 / ((bandedOffset * coefficient / range) + 1.0))) * range + } + + var pitchTranslation = rubberBandingOffset(offset: abs(translation.y), bandingStart: 0.0) + if translation.y < 0.0 { + pitchTranslation *= -1.0 + } + let pitchPan = deg2rad(Float(pitchTranslation)) + + self.previousYaw = yawPan + node.eulerAngles = SCNVector3(pitchPan, yawPan, 0.0) + case .ended: + let velocity = gesture.velocity(in: gesture.view) + + var smallAngle = false + if (self.previousYaw < .pi / 2 && self.previousYaw > -.pi / 2) && abs(velocity.x) < 200 { + smallAngle = true + } + + self.playAppearanceAnimation(velocity: velocity.x, smallAngle: smallAngle, explode: !smallAngle && abs(velocity.x) > 600) + node.eulerAngles = SCNVector3(0.0, 0.0, 0.0) + default: + break + } + } + + private func setup() { + guard let scene = loadCompressedScene(name: "coin", version: sceneVersion) else { + return + } + + self.sceneView.scene = scene + self.sceneView.delegate = self + + let _ = self.sceneView.snapshot() + } + + private var didSetReady = false + func renderer(_ renderer: SCNSceneRenderer, didRenderScene scene: SCNScene, atTime time: TimeInterval) { + if !self.didSetReady { + self.didSetReady = true + + Queue.mainQueue().justDispatch { + self._ready.set(.single(true)) + self.onReady() + } + } + } + + private func maybeAnimateIn() { + guard let scene = self.sceneView.scene, let _ = scene.rootNode.childNode(withName: "star", recursively: false), let animateFrom = self.animateFrom, var containerView = self.containerView else { + return + } + + containerView = containerView.subviews[2].subviews[1] + + let initialPosition = self.sceneView.center + let targetPosition = self.sceneView.superview!.convert(self.sceneView.center, to: containerView) + let sourcePosition = animateFrom.superview!.convert(animateFrom.center, to: containerView).offsetBy(dx: 0.0, dy: -20.0) + + containerView.addSubview(self.sceneView) + self.sceneView.center = targetPosition + + animateFrom.alpha = 0.0 + self.sceneView.layer.animateScale(from: 0.05, to: 0.5, duration: 1.0, timingFunction: kCAMediaTimingFunctionSpring) + self.sceneView.layer.animatePosition(from: sourcePosition, to: targetPosition, duration: 1.0, timingFunction: kCAMediaTimingFunctionSpring, completion: { _ in + self.addSubview(self.sceneView) + self.sceneView.center = initialPosition + }) + + Queue.mainQueue().after(0.4, { + animateFrom.alpha = 1.0 + }) + + self.animateFrom = nil + self.containerView = nil + } + + private func onReady() { + self.setupScaleAnimation() + self.setupGradientAnimation() + self.setupShineAnimation() + + self.maybeAnimateIn() + self.playAppearanceAnimation(explode: true) + + self.previousInteractionTimestamp = CACurrentMediaTime() + self.timer = SwiftSignalKit.Timer(timeout: 1.0, repeat: true, completion: { [weak self] in + if let strongSelf = self, strongSelf.hasIdleAnimations { + let currentTimestamp = CACurrentMediaTime() + if currentTimestamp > strongSelf.previousInteractionTimestamp + 5.0 { + strongSelf.playAppearanceAnimation() + } + } + }, queue: Queue.mainQueue()) + self.timer?.start() + } + + private func setupScaleAnimation() { + guard let scene = self.sceneView.scene, let node = scene.rootNode.childNode(withName: "star", recursively: false) else { + return + } + + let fromScale: Float = 0.9 + let toScale: Float = 1.0 + + let animation = CABasicAnimation(keyPath: "scale") + animation.duration = 2.0 + animation.fromValue = NSValue(scnVector3: SCNVector3(x: fromScale, y: fromScale, z: fromScale)) + animation.toValue = NSValue(scnVector3: SCNVector3(x: toScale, y: toScale, z: toScale)) + animation.timingFunction = CAMediaTimingFunction(name: .easeInEaseOut) + animation.autoreverses = true + animation.repeatCount = .infinity + + node.addAnimation(animation, forKey: "scale") + } + + private func setupGradientAnimation() { + guard let scene = self.sceneView.scene, let node = scene.rootNode.childNode(withName: "star", recursively: false) else { + return + } + + for node in node.childNodes { + guard let initial = node.geometry?.materials.first?.diffuse.contentsTransform else { + return + } + + let animation = CABasicAnimation(keyPath: "contentsTransform") + animation.duration = 4.5 + animation.fromValue = NSValue(scnMatrix4: initial) + animation.toValue = NSValue(scnMatrix4: SCNMatrix4Translate(initial, -0.35, 0.35, 0)) + animation.timingFunction = CAMediaTimingFunction(name: .linear) + animation.autoreverses = true + animation.repeatCount = .infinity + + node.geometry?.materials.first?.diffuse.addAnimation(animation, forKey: "gradient") + } + } + + private func setupShineAnimation() { + guard let scene = self.sceneView.scene, let node = scene.rootNode.childNode(withName: "star", recursively: false) else { + return + } + + for node in node.childNodes { + guard let initial = node.geometry?.materials.first?.emission.contentsTransform else { + return + } + + if #available(iOS 17.0, *), let material = node.geometry?.materials.first { + material.metalness.intensity = 0.3 + } + + let animation = CABasicAnimation(keyPath: "contentsTransform") + animation.fillMode = .forwards + animation.fromValue = NSValue(scnMatrix4: initial) + animation.toValue = NSValue(scnMatrix4: SCNMatrix4Translate(initial, -1.6, 0.0, 0.0)) + animation.timingFunction = CAMediaTimingFunction(name: .easeOut) + animation.beginTime = 1.1 + animation.duration = 0.9 + + let group = CAAnimationGroup() + group.animations = [animation] + group.beginTime = 1.0 + group.duration = 4.0 + group.repeatCount = .infinity + + node.geometry?.materials.first?.emission.addAnimation(group, forKey: "shimmer") + } + } + + private func playAppearanceAnimation(velocity: CGFloat? = nil, smallAngle: Bool = false, mirror: Bool = false, explode: Bool = false) { + guard let scene = self.sceneView.scene, let node = scene.rootNode.childNode(withName: "star", recursively: false) else { + return + } + + let currentTime = CACurrentMediaTime() + self.previousInteractionTimestamp = currentTime + self.delayTapsTill = currentTime + 0.85 + + if explode, let node = scene.rootNode.childNode(withName: "swirl", recursively: false), let particlesLeft = scene.rootNode.childNode(withName: "particles_left", recursively: false), let particlesRight = scene.rootNode.childNode(withName: "particles_right", recursively: false), let particlesBottomLeft = scene.rootNode.childNode(withName: "particles_left_bottom", recursively: false), let particlesBottomRight = scene.rootNode.childNode(withName: "particles_right_bottom", recursively: false) { + if let leftParticleSystem = particlesLeft.particleSystems?.first, let rightParticleSystem = particlesRight.particleSystems?.first, let leftBottomParticleSystem = particlesBottomLeft.particleSystems?.first, let rightBottomParticleSystem = particlesBottomRight.particleSystems?.first { + leftParticleSystem.speedFactor = 2.0 + leftParticleSystem.particleVelocity = 1.6 + leftParticleSystem.birthRate = 60.0 + leftParticleSystem.particleLifeSpan = 4.0 + + rightParticleSystem.speedFactor = 2.0 + rightParticleSystem.particleVelocity = 1.6 + rightParticleSystem.birthRate = 60.0 + rightParticleSystem.particleLifeSpan = 4.0 + + leftBottomParticleSystem.particleVelocity = 1.6 + leftBottomParticleSystem.birthRate = 24.0 + leftBottomParticleSystem.particleLifeSpan = 7.0 + + rightBottomParticleSystem.particleVelocity = 1.6 + rightBottomParticleSystem.birthRate = 24.0 + rightBottomParticleSystem.particleLifeSpan = 7.0 + + node.physicsField?.isActive = true + Queue.mainQueue().after(1.0) { + node.physicsField?.isActive = false + + leftParticleSystem.birthRate = 15.0 + leftParticleSystem.particleVelocity = 1.0 + leftParticleSystem.particleLifeSpan = 3.0 + + rightParticleSystem.birthRate = 15.0 + rightParticleSystem.particleVelocity = 1.0 + rightParticleSystem.particleLifeSpan = 3.0 + + leftBottomParticleSystem.particleVelocity = 1.0 + leftBottomParticleSystem.birthRate = 10.0 + leftBottomParticleSystem.particleLifeSpan = 5.0 + + rightBottomParticleSystem.particleVelocity = 1.0 + rightBottomParticleSystem.birthRate = 10.0 + rightBottomParticleSystem.particleLifeSpan = 5.0 + + let leftAnimation = POPBasicAnimation() + leftAnimation.property = (POPAnimatableProperty.property(withName: "speedFactor", initializer: { property in + property?.readBlock = { particleSystem, values in + values?.pointee = (particleSystem as! SCNParticleSystem).speedFactor + } + property?.writeBlock = { particleSystem, values in + (particleSystem as! SCNParticleSystem).speedFactor = values!.pointee + } + property?.threshold = 0.01 + }) as! POPAnimatableProperty) + leftAnimation.fromValue = 1.2 as NSNumber + leftAnimation.toValue = 0.85 as NSNumber + leftAnimation.timingFunction = CAMediaTimingFunction(name: CAMediaTimingFunctionName.linear) + leftAnimation.duration = 0.5 + leftParticleSystem.pop_add(leftAnimation, forKey: "speedFactor") + + let rightAnimation = POPBasicAnimation() + rightAnimation.property = (POPAnimatableProperty.property(withName: "speedFactor", initializer: { property in + property?.readBlock = { particleSystem, values in + values?.pointee = (particleSystem as! SCNParticleSystem).speedFactor + } + property?.writeBlock = { particleSystem, values in + (particleSystem as! SCNParticleSystem).speedFactor = values!.pointee + } + property?.threshold = 0.01 + }) as! POPAnimatableProperty) + rightAnimation.fromValue = 1.2 as NSNumber + rightAnimation.toValue = 0.85 as NSNumber + rightAnimation.timingFunction = CAMediaTimingFunction(name: CAMediaTimingFunctionName.linear) + rightAnimation.duration = 0.5 + rightParticleSystem.pop_add(rightAnimation, forKey: "speedFactor") + } + } + } + + var from = node.presentation.eulerAngles + if abs(from.y - .pi * 2.0) < 0.001 { + from.y = 0.0 + } + node.removeAnimation(forKey: "tapRotate") + + var toValue: Float = smallAngle ? 0.0 : .pi * 2.0 + if let velocity = velocity, !smallAngle && abs(velocity) > 200 && velocity < 0.0 { + toValue *= -1 + } + if mirror { + toValue *= -1 + } + let to = SCNVector3(x: 0.0, y: toValue, z: 0.0) + let distance = rad2deg(to.y - from.y) + + guard !distance.isZero else { + return + } + + let springAnimation = CASpringAnimation(keyPath: "eulerAngles") + springAnimation.fromValue = NSValue(scnVector3: from) + springAnimation.toValue = NSValue(scnVector3: to) + springAnimation.mass = 1.0 + springAnimation.stiffness = 21.0 + springAnimation.damping = 5.8 + springAnimation.duration = springAnimation.settlingDuration * 0.75 + springAnimation.initialVelocity = velocity.flatMap { abs($0 / CGFloat(distance)) } ?? 1.7 + springAnimation.completion = { [weak node] finished in + if finished { + node?.eulerAngles = SCNVector3(x: 0.0, y: 0.0, z: 0.0) + } + } + node.addAnimation(springAnimation, forKey: "rotate") + } + + func update(component: PremiumCoinComponent, availableSize: CGSize, transition: Transition) -> CGSize { + self.sceneView.bounds = CGRect(origin: .zero, size: CGSize(width: availableSize.width * 2.0, height: availableSize.height * 2.0)) + if self.sceneView.superview == self { + self.sceneView.center = CGPoint(x: availableSize.width / 2.0, y: availableSize.height / 2.0) + } + + self.hasIdleAnimations = component.hasIdleAnimations + + return availableSize + } + } + + func makeView() -> View { + return View(frame: CGRect(), isIntro: self.isIntro) + } + + func update(view: View, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + return view.update(component: self, availableSize: availableSize, transition: transition) + } +} diff --git a/submodules/PremiumUI/Sources/PremiumDemoScreen.swift b/submodules/PremiumUI/Sources/PremiumDemoScreen.swift index 470f4d2f62c..5957a9920f1 100644 --- a/submodules/PremiumUI/Sources/PremiumDemoScreen.swift +++ b/submodules/PremiumUI/Sources/PremiumDemoScreen.swift @@ -1004,7 +1004,7 @@ private final class DemoSheetContent: CombinedComponent { position: .top, model: .island, videoFile: configuration.videos["last_seen"], - decoration: .tag + decoration: .badgeStars )), title: strings.Premium_LastSeen, text: strings.Premium_LastSeenInfo, @@ -1023,7 +1023,7 @@ private final class DemoSheetContent: CombinedComponent { position: .top, model: .island, videoFile: configuration.videos["message_privacy"], - decoration: .tag + decoration: .swirlStars )), title: strings.Premium_MessagePrivacy, text: strings.Premium_MessagePrivacyInfo, @@ -1033,6 +1033,26 @@ private final class DemoSheetContent: CombinedComponent { ) ) + availableItems[.folderTags] = DemoPagerComponent.Item( + AnyComponentWithIdentity( + id: PremiumDemoScreen.Subject.folderTags, + component: AnyComponent( + PageComponent( + content: AnyComponent(PhoneDemoComponent( + context: component.context, + position: .top, + model: .island, + videoFile: configuration.videos["folder_tags"], + decoration: .tag + )), + title: strings.Premium_FolderTags, + text: strings.Premium_FolderTagsStandaloneInfo, + textColor: textColor + ) + ) + ) + ) + let index: Int = 0 var items: [DemoPagerComponent.Item] = [] if let item = availableItems.first(where: { $0.value.content.id == component.subject as AnyHashable }) { @@ -1136,6 +1156,8 @@ private final class DemoSheetContent: CombinedComponent { buttonText = strings.Premium_LastSeen_Proceed case .messagePrivacy: buttonText = strings.Premium_MessagePrivacy_Proceed + case .folderTags: + buttonText = strings.Premium_FolderTags_Proceed default: buttonText = strings.Common_OK } @@ -1177,7 +1199,9 @@ private final class DemoSheetContent: CombinedComponent { text = strings.Premium_LastSeenInfo case .messagePrivacy: text = strings.Premium_MessagePrivacyInfo - case .doubleLimits, .stories: + case .folderTags: + text = strings.Premium_FolderTagsStandaloneInfo + default: text = "" } @@ -1391,6 +1415,15 @@ public class PremiumDemoScreen: ViewControllerComponentContainer { case messageTags case lastSeen case messagePrivacy + case business + case folderTags + + case businessLocation + case businessHours + case businessGreetingMessage + case businessQuickReplies + case businessAwayMessage + case businessChatBots } public enum Source: Equatable { diff --git a/submodules/PremiumUI/Sources/PremiumGiftScreen.swift b/submodules/PremiumUI/Sources/PremiumGiftScreen.swift index ee7bc25662d..2e81b49ed3f 100644 --- a/submodules/PremiumUI/Sources/PremiumGiftScreen.swift +++ b/submodules/PremiumUI/Sources/PremiumGiftScreen.swift @@ -424,6 +424,7 @@ private final class PremiumGiftScreenContentComponent: CombinedComponent { UIColor(rgb: 0xef6922), UIColor(rgb: 0xe95a2c), UIColor(rgb: 0xe74e33), + UIColor(rgb: 0xe74e33), //replace UIColor(rgb: 0xe54937), UIColor(rgb: 0xe3433c), UIColor(rgb: 0xdb374b), @@ -434,6 +435,7 @@ private final class PremiumGiftScreenContentComponent: CombinedComponent { UIColor(rgb: 0x9b4fed), UIColor(rgb: 0x8958ff), UIColor(rgb: 0x676bff), + UIColor(rgb: 0x676bff), //replace UIColor(rgb: 0x6172ff), UIColor(rgb: 0x5b79ff), UIColor(rgb: 0x4492ff), @@ -530,6 +532,10 @@ private final class PremiumGiftScreenContentComponent: CombinedComponent { demoSubject = .lastSeen case .messagePrivacy: demoSubject = .messagePrivacy + case .business: + demoSubject = .business + default: + demoSubject = .doubleLimits } let buttonText: String diff --git a/submodules/PremiumUI/Sources/PremiumIntroScreen.swift b/submodules/PremiumUI/Sources/PremiumIntroScreen.swift index 6fb944509b2..b6c6202a8ed 100644 --- a/submodules/PremiumUI/Sources/PremiumIntroScreen.swift +++ b/submodules/PremiumUI/Sources/PremiumIntroScreen.swift @@ -31,6 +31,11 @@ import MultiAnimationRenderer import TelegramNotices import UndoUI import TelegramStringFormatting +import ListSectionComponent +import ListActionItemComponent +import EmojiStatusSelectionComponent +import EmojiStatusComponent +import EntityKeyboard public enum PremiumSource: Equatable { public static func == (lhs: PremiumSource, rhs: PremiumSource) -> Bool { @@ -281,6 +286,12 @@ public enum PremiumSource: Equatable { } else { return false } + case .folderTags: + if case .folderTags = rhs { + return true + } else { + return false + } } } @@ -325,6 +336,7 @@ public enum PremiumSource: Equatable { case presence case readTime case messageTags + case folderTags var identifier: String? { switch self { @@ -412,6 +424,8 @@ public enum PremiumSource: Equatable { return "read_time" case .messageTags: return "saved_tags" + case .folderTags: + return "folder_tags" } } } @@ -437,6 +451,15 @@ public enum PremiumPerk: CaseIterable { case messageTags case lastSeen case messagePrivacy + case business + case folderTags + + case businessLocation + case businessHours + case businessGreetingMessage + case businessQuickReplies + case businessAwayMessage + case businessChatBots public static var allCases: [PremiumPerk] { return [ @@ -459,12 +482,29 @@ public enum PremiumPerk: CaseIterable { .wallpapers, .messageTags, .lastSeen, - .messagePrivacy + .messagePrivacy, + .folderTags, + .business ] } - init?(identifier: String) { - for perk in PremiumPerk.allCases { + public static var allBusinessCases: [PremiumPerk] { + return [ + .businessLocation, + .businessHours, + .businessGreetingMessage, + .businessQuickReplies, + .businessAwayMessage, + .businessChatBots, +// .emojiStatus, +// .folderTags, +// .stories, + ] + } + + + init?(identifier: String, business: Bool) { + for perk in business ? PremiumPerk.allBusinessCases : PremiumPerk.allCases { if perk.identifier == identifier { self = perk return @@ -515,6 +555,22 @@ public enum PremiumPerk: CaseIterable { return "last_seen" case .messagePrivacy: return "message_privacy" + case .folderTags: + return "folder_tags" + case .business: + return "business" + case .businessLocation: + return "business_location" + case .businessHours: + return "business_hours" + case .businessQuickReplies: + return "quick_replies" + case .businessGreetingMessage: + return "greeting_message" + case .businessAwayMessage: + return "away_message" + case .businessChatBots: + return "business_bots" } } @@ -560,6 +616,23 @@ public enum PremiumPerk: CaseIterable { return strings.Premium_LastSeen case .messagePrivacy: return strings.Premium_MessagePrivacy + case .folderTags: + return strings.Premium_FolderTags + case .business: + return strings.Premium_Business + + case .businessLocation: + return strings.Business_Location + case .businessHours: + return strings.Business_OpeningHours + case .businessQuickReplies: + return strings.Business_QuickReplies + case .businessGreetingMessage: + return strings.Business_GreetingMessages + case .businessAwayMessage: + return strings.Business_AwayMessages + case .businessChatBots: + return strings.Business_Chatbots } } @@ -605,6 +678,23 @@ public enum PremiumPerk: CaseIterable { return strings.Premium_LastSeenInfo case .messagePrivacy: return strings.Premium_MessagePrivacyInfo + case .folderTags: + return strings.Premium_FolderTagsInfo + case .business: + return strings.Premium_BusinessInfo + + case .businessLocation: + return strings.Business_LocationInfo + case .businessHours: + return strings.Business_OpeningHoursInfo + case .businessQuickReplies: + return strings.Business_QuickRepliesInfo + case .businessGreetingMessage: + return strings.Business_GreetingMessagesInfo + case .businessAwayMessage: + return strings.Business_AwayMessagesInfo + case .businessChatBots: + return strings.Business_ChatbotsInfo } } @@ -650,6 +740,23 @@ public enum PremiumPerk: CaseIterable { return "Premium/Perk/LastSeen" case .messagePrivacy: return "Premium/Perk/MessagePrivacy" + case .folderTags: + return "Premium/Perk/MessageTags" + case .business: + return "Premium/Perk/Business" + + case .businessLocation: + return "Premium/BusinessPerk/Location" + case .businessHours: + return "Premium/BusinessPerk/Hours" + case .businessQuickReplies: + return "Premium/BusinessPerk/Replies" + case .businessGreetingMessage: + return "Premium/BusinessPerk/Greetings" + case .businessAwayMessage: + return "Premium/BusinessPerk/Away" + case .businessChatBots: + return "Premium/BusinessPerk/Chatbots" } } } @@ -676,21 +783,34 @@ struct PremiumIntroConfiguration { .appIcons, .uniqueReactions, .animatedUserpics, - .premiumStickers + .premiumStickers, + .business + ], businessPerks: [ + .businessGreetingMessage, + .businessAwayMessage, + .businessQuickReplies, + .businessChatBots, + .businessHours, + .businessLocation +// .emojiStatus, +// .folderTags, +// .stories ]) } let perks: [PremiumPerk] + let businessPerks: [PremiumPerk] - fileprivate init(perks: [PremiumPerk]) { + fileprivate init(perks: [PremiumPerk], businessPerks: [PremiumPerk]) { self.perks = perks + self.businessPerks = businessPerks } public static func with(appConfiguration: AppConfiguration) -> PremiumIntroConfiguration { if let data = appConfiguration.data, let values = data["premium_promo_order"] as? [String] { var perks: [PremiumPerk] = [] for value in values { - if let perk = PremiumPerk(identifier: value) { + if let perk = PremiumPerk(identifier: value, business: false) { if !perks.contains(perk) { perks.append(perk) } else { @@ -715,8 +835,30 @@ struct PremiumIntroConfiguration { if !perks.contains(.messageTags) { perks.append(.messageTags) } + if !perks.contains(.business) { + perks.append(.business) + } #endif - return PremiumIntroConfiguration(perks: perks) + + + var businessPerks: [PremiumPerk] = [] + if let values = data["business_promo_order"] as? [String] { + for value in values { + if let perk = PremiumPerk(identifier: value, business: true) { + if !businessPerks.contains(perk) { + businessPerks.append(perk) + } else { + businessPerks = [] + break + } + } + } + } + if businessPerks.count < 4 { + businessPerks = PremiumIntroConfiguration.defaultValue.businessPerks + } + + return PremiumIntroConfiguration(perks: perks, businessPerks: businessPerks) } else { return .defaultValue } @@ -752,337 +894,71 @@ private struct PremiumProduct: Equatable { } } -final class PremiumOptionComponent: CombinedComponent { - let title: String - let subtitle: String - let labelPrice: String - let discount: String - let multiple: Bool - let selected: Bool - let primaryTextColor: UIColor - let secondaryTextColor: UIColor - let accentColor: UIColor - let checkForegroundColor: UIColor - let checkBorderColor: UIColor +final class PerkIconComponent: CombinedComponent { + let backgroundColor: UIColor + let foregroundColor: UIColor + let iconName: String init( - title: String, - subtitle: String, - labelPrice: String, - discount: String, - multiple: Bool = false, - selected: Bool, - primaryTextColor: UIColor, - secondaryTextColor: UIColor, - accentColor: UIColor, - checkForegroundColor: UIColor, - checkBorderColor: UIColor + backgroundColor: UIColor, + foregroundColor: UIColor, + iconName: String ) { - self.title = title - self.subtitle = subtitle - self.labelPrice = labelPrice - self.discount = discount - self.multiple = multiple - self.selected = selected - self.primaryTextColor = primaryTextColor - self.secondaryTextColor = secondaryTextColor - self.accentColor = accentColor - self.checkForegroundColor = checkForegroundColor - self.checkBorderColor = checkBorderColor + self.backgroundColor = backgroundColor + self.foregroundColor = foregroundColor + self.iconName = iconName } - static func ==(lhs: PremiumOptionComponent, rhs: PremiumOptionComponent) -> Bool { - if lhs.title != rhs.title { - return false - } - if lhs.subtitle != rhs.subtitle { - return false - } - if lhs.labelPrice != rhs.labelPrice { - return false - } - if lhs.discount != rhs.discount { - return false - } - if lhs.multiple != rhs.multiple { - return false - } - if lhs.selected != rhs.selected { - return false - } - if lhs.primaryTextColor != rhs.primaryTextColor { - return false - } - if lhs.secondaryTextColor != rhs.secondaryTextColor { - return false - } - if lhs.accentColor != rhs.accentColor { + static func ==(lhs: PerkIconComponent, rhs: PerkIconComponent) -> Bool { + if lhs.backgroundColor != rhs.backgroundColor { return false } - if lhs.checkForegroundColor != rhs.checkForegroundColor { + if lhs.foregroundColor != rhs.foregroundColor { return false } - if lhs.checkBorderColor != rhs.checkBorderColor { + if lhs.iconName != rhs.iconName { return false } return true } static var body: Body { - let check = Child(CheckComponent.self) - let title = Child(MultilineTextComponent.self) - let subtitle = Child(MultilineTextComponent.self) - let discountBackground = Child(RoundedRectangle.self) - let discount = Child(MultilineTextComponent.self) - let label = Child(MultilineTextComponent.self) - + let background = Child(RoundedRectangle.self) + let icon = Child(BundleIconComponent.self) + return { context in let component = context.component + + let iconSize = CGSize(width: 30.0, height: 30.0) - var insets = UIEdgeInsets(top: 11.0, left: 46.0, bottom: 13.0, right: 16.0) - - let label = label.update( - component: MultilineTextComponent( - text: .plain( - NSAttributedString( - string: component.labelPrice, - font: Font.regular(17), - textColor: component.secondaryTextColor - ) - ), - maximumNumberOfLines: 1 - ), - availableSize: context.availableSize, - transition: context.transition - ) - - let title = title.update( - component: MultilineTextComponent( - text: .plain( - NSAttributedString( - string: component.title, - font: Font.regular(17), - textColor: component.primaryTextColor - ) - ), - maximumNumberOfLines: 1 + let background = background.update( + component: RoundedRectangle( + color: component.backgroundColor, + cornerRadius: 7.0 ), - availableSize: CGSize(width: context.availableSize.width - insets.left - insets.right - label.size.width, height: context.availableSize.height), + availableSize: iconSize, transition: context.transition ) - - let discountSize: CGSize - if !component.discount.isEmpty { - let discount = discount.update( - component: MultilineTextComponent( - text: .plain( - NSAttributedString( - string: component.discount, - font: Font.with(size: 14.0, design: .round, weight: .semibold, traits: []), - textColor: .white - ) - ), - maximumNumberOfLines: 1 - ), - availableSize: context.availableSize, - transition: context.transition - ) - - discountSize = CGSize(width: discount.size.width + 6.0, height: 18.0) - let discountBackground = discountBackground.update( - component: RoundedRectangle( - color: component.accentColor, - cornerRadius: 5.0 - ), - availableSize: discountSize, - transition: context.transition - ) - - let discountPosition = CGPoint(x: insets.left + title.size.width + 6.0 + discountSize.width / 2.0, y: insets.top + title.size.height / 2.0) - - context.add(discountBackground - .position(discountPosition) - ) - context.add(discount - .position(discountPosition) - ) - } else { - discountSize = CGSize(width: 0.0, height: 18.0) - } - - var spacing: CGFloat = 0.0 - var subtitleSize = CGSize() - if !component.subtitle.isEmpty { - spacing = 2.0 - - let subtitleFont = Font.regular(13) - let subtitleColor = component.secondaryTextColor - - let subtitleString = parseMarkdownIntoAttributedString( - component.subtitle, - attributes: MarkdownAttributes( - body: MarkdownAttributeSet(font: subtitleFont, textColor: subtitleColor), - bold: MarkdownAttributeSet(font: subtitleFont, textColor: subtitleColor, additionalAttributes: [NSAttributedString.Key.strikethroughStyle.rawValue: NSUnderlineStyle.single.rawValue as NSNumber]), - link: MarkdownAttributeSet(font: subtitleFont, textColor: subtitleColor), - linkAttribute: { _ in return nil } - ) - ) - - let subtitle = subtitle.update( - component: MultilineTextComponent( - text: .plain(subtitleString), - maximumNumberOfLines: 1 - ), - availableSize: CGSize(width: context.availableSize.width - insets.left - insets.right, height: context.availableSize.height), - transition: context.transition - ) - context.add(subtitle - .position(CGPoint(x: insets.left + subtitle.size.width / 2.0, y: insets.top + title.size.height + spacing + subtitle.size.height / 2.0)) - ) - subtitleSize = subtitle.size - - insets.top -= 2.0 - insets.bottom -= 2.0 - } - - let check = check.update( - component: CheckComponent( - theme: CheckComponent.Theme( - backgroundColor: component.accentColor, - strokeColor: component.checkForegroundColor, - borderColor: component.checkBorderColor, - overlayBorder: false, - hasInset: false, - hasShadow: false - ), - selected: component.selected + let icon = icon.update( + component: BundleIconComponent( + name: component.iconName, + tintColor: .white ), - availableSize: context.availableSize, + availableSize: iconSize, transition: context.transition ) - - context.add(title - .position(CGPoint(x: insets.left + title.size.width / 2.0, y: insets.top + title.size.height / 2.0)) - ) - - let size = CGSize(width: context.availableSize.width, height: insets.top + title.size.height + spacing + subtitleSize.height + insets.bottom) - let distance = context.availableSize.width - insets.left - insets.right - label.size.width - subtitleSize.width - - let labelY: CGFloat - if distance > 8.0 { - labelY = size.height / 2.0 - } else { - labelY = insets.top + title.size.height / 2.0 - } - - context.add(label - .position(CGPoint(x: context.availableSize.width - insets.right - label.size.width / 2.0, y: labelY)) - ) - - context.add(check - .position(CGPoint(x: 4.0 + check.size.width / 2.0, y: size.height / 2.0)) + let iconPosition = CGPoint(x: background.size.width / 2.0, y: background.size.height / 2.0) + context.add(background + .position(iconPosition) ) - - return size - } - } -} - -private final class CheckComponent: Component { - struct Theme: Equatable { - public let backgroundColor: UIColor - public let strokeColor: UIColor - public let borderColor: UIColor - public let overlayBorder: Bool - public let hasInset: Bool - public let hasShadow: Bool - public let filledBorder: Bool - public let borderWidth: CGFloat? - - public init(backgroundColor: UIColor, strokeColor: UIColor, borderColor: UIColor, overlayBorder: Bool, hasInset: Bool, hasShadow: Bool, filledBorder: Bool = false, borderWidth: CGFloat? = nil) { - self.backgroundColor = backgroundColor - self.strokeColor = strokeColor - self.borderColor = borderColor - self.overlayBorder = overlayBorder - self.hasInset = hasInset - self.hasShadow = hasShadow - self.filledBorder = filledBorder - self.borderWidth = borderWidth - } - - var checkNodeTheme: CheckNodeTheme { - return CheckNodeTheme( - backgroundColor: self.backgroundColor, - strokeColor: self.strokeColor, - borderColor: self.borderColor, - overlayBorder: self.overlayBorder, - hasInset: self.hasInset, - hasShadow: self.hasShadow, - filledBorder: self.filledBorder, - borderWidth: self.borderWidth + context.add(icon + .position(iconPosition) ) + return iconSize } } - - let theme: Theme - let selected: Bool - - init( - theme: Theme, - selected: Bool - ) { - self.theme = theme - self.selected = selected - } - - static func ==(lhs: CheckComponent, rhs: CheckComponent) -> Bool { - if lhs.theme != rhs.theme { - return false - } - if lhs.selected != rhs.selected { - return false - } - return true - } - - final class View: UIView { - private var currentValue: CGFloat? - private var animator: DisplayLinkAnimator? - - private var checkLayer: CheckLayer { - return self.layer as! CheckLayer - } - - override class var layerClass: AnyClass { - return CheckLayer.self - } - - init() { - super.init(frame: CGRect()) - } - - required init?(coder aDecoder: NSCoder) { - preconditionFailure() - } - - - func update(component: CheckComponent, availableSize: CGSize, transition: Transition) -> CGSize { - self.checkLayer.setSelected(component.selected, animated: true) - self.checkLayer.theme = component.theme.checkNodeTheme - - return CGSize(width: 22.0, height: 22.0) - } - } - - func makeView() -> View { - return View() - } - - func update(view: View, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { - return view.update(component: self, availableSize: availableSize, transition: transition) - } } final class SectionGroupComponent: Component { @@ -1483,6 +1359,7 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { typealias EnvironmentType = (ViewControllerComponentContainer.Environment, ScrollChildEnvironment) let context: AccountContext + let mode: PremiumIntroScreen.Mode let source: PremiumSource let forceDark: Bool let isPremium: Bool? @@ -1493,6 +1370,7 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { let validPurchases: [InAppPurchaseManager.ReceiptPurchase] let promoConfiguration: PremiumPromoConfiguration? let present: (ViewController) -> Void + let push: (ViewController) -> Void let selectProduct: (String) -> Void let buy: () -> Void let updateIsFocused: (Bool) -> Void @@ -1501,6 +1379,7 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { init( context: AccountContext, + mode: PremiumIntroScreen.Mode, source: PremiumSource, forceDark: Bool, isPremium: Bool?, @@ -1511,6 +1390,7 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { validPurchases: [InAppPurchaseManager.ReceiptPurchase], promoConfiguration: PremiumPromoConfiguration?, present: @escaping (ViewController) -> Void, + push: @escaping (ViewController) -> Void, selectProduct: @escaping (String) -> Void, buy: @escaping () -> Void, updateIsFocused: @escaping (Bool) -> Void, @@ -1518,6 +1398,7 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { shareLink: @escaping (String) -> Void ) { self.context = context + self.mode = mode self.source = source self.forceDark = forceDark self.isPremium = isPremium @@ -1528,6 +1409,7 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { self.validPurchases = validPurchases self.promoConfiguration = promoConfiguration self.present = present + self.push = push self.selectProduct = selectProduct self.buy = buy self.updateIsFocused = updateIsFocused @@ -1572,6 +1454,7 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { final class State: ComponentState { private let context: AccountContext + private let present: (ViewController) -> Void var products: [PremiumProduct]? var selectedProductId: String? @@ -1580,6 +1463,7 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { var newPerks: [String] = [] var isPremium: Bool? + var peer: EnginePeer? private var disposable: Disposable? private(set) var configuration = PremiumIntroConfiguration.defaultValue @@ -1608,20 +1492,29 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { } } - init(context: AccountContext, source: PremiumSource) { + init( + context: AccountContext, + source: PremiumSource, + present: @escaping (ViewController) -> Void + ) { self.context = context + self.present = present super.init() self.disposable = (context.engine.data.subscribe( - TelegramEngine.EngineData.Item.Configuration.App() + TelegramEngine.EngineData.Item.Configuration.App(), + TelegramEngine.EngineData.Item.Peer.Peer(id: context.account.peerId) ) - |> deliverOnMainQueue).start(next: { [weak self] appConfiguration in + |> deliverOnMainQueue).start(next: { [weak self] appConfiguration, accountPeer in if let strongSelf = self { + let isFirstTime = strongSelf.peer == nil + strongSelf.configuration = PremiumIntroConfiguration.with(appConfiguration: appConfiguration) + strongSelf.peer = accountPeer strongSelf.updated(transition: .immediate) - if let identifier = source.identifier { + if let identifier = source.identifier, isFirstTime { var jsonString: String = "{" jsonString += "\"source\": \"\(identifier)\"," @@ -1671,8 +1564,9 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { ApplicationSpecificNotice.dismissedPremiumColorsBadge(accountManager: context.sharedContext.accountManager), ApplicationSpecificNotice.dismissedMessageTagsBadge(accountManager: context.sharedContext.accountManager), ApplicationSpecificNotice.dismissedLastSeenBadge(accountManager: context.sharedContext.accountManager), - ApplicationSpecificNotice.dismissedMessagePrivacyBadge(accountManager: context.sharedContext.accountManager) - ).startStrict(next: { [weak self] dismissedPremiumAppIconsBadge, dismissedPremiumWallpapersBadge, dismissedPremiumColorsBadge, dismissedMessageTagsBadge, dismissedLastSeenBadge, dismissedMessagePrivacyBadge in + ApplicationSpecificNotice.dismissedMessagePrivacyBadge(accountManager: context.sharedContext.accountManager), + ApplicationSpecificNotice.dismissedBusinessBadge(accountManager: context.sharedContext.accountManager) + ).startStrict(next: { [weak self] dismissedPremiumAppIconsBadge, dismissedPremiumWallpapersBadge, dismissedPremiumColorsBadge, dismissedMessageTagsBadge, dismissedLastSeenBadge, dismissedMessagePrivacyBadge, dismissedBusinessBadge in guard let self else { return } @@ -1692,6 +1586,9 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { if !dismissedMessagePrivacyBadge { newPerks.append(PremiumPerk.messagePrivacy.identifier) } + if !dismissedBusinessBadge { + newPerks.append(PremiumPerk.business.identifier) + } self.newPerks = newPerks self.updated() }) @@ -1703,10 +1600,59 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { self.stickersDisposable?.dispose() self.newPerksDisposable?.dispose() } + + private var updatedPeerStatus: PeerEmojiStatus? + + private weak var emojiStatusSelectionController: ViewController? + private var previousEmojiSetupTimestamp: Double? + func openEmojiSetup(sourceView: UIView, currentFileId: Int64?, color: UIColor?) { + let currentTimestamp = CACurrentMediaTime() + if let previousTimestamp = self.previousEmojiSetupTimestamp, currentTimestamp < previousTimestamp + 1.0 { + return + } + self.previousEmojiSetupTimestamp = currentTimestamp + + self.emojiStatusSelectionController?.dismiss() + var selectedItems = Set() + if let currentFileId { + selectedItems.insert(MediaId(namespace: Namespaces.Media.CloudFile, id: currentFileId)) + } + + let controller = EmojiStatusSelectionController( + context: self.context, + mode: .statusSelection, + sourceView: sourceView, + emojiContent: EmojiPagerContentComponent.emojiInputData( + context: self.context, + animationCache: self.context.animationCache, + animationRenderer: self.context.animationRenderer, + isStandalone: false, + subject: .status, + hasTrending: false, + topReactionItems: [], + areUnicodeEmojiEnabled: false, + areCustomEmojiEnabled: true, + chatPeerId: self.context.account.peerId, + selectedItems: selectedItems, + topStatusTitle: nil, + backgroundIconColor: color + ), + currentSelection: currentFileId, + color: color, + destinationItemView: { [weak sourceView] in + guard let sourceView else { + return nil + } + return sourceView + } + ) + self.emojiStatusSelectionController = controller + self.present(controller) + } } func makeState() -> State { - return State(context: self.context, source: self.source) + return State(context: self.context, source: self.source, present: self.present) } static var body: Body { @@ -1716,8 +1662,9 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { let completedText = Child(MultilineTextComponent.self) let linkButton = Child(Button.self) let optionsSection = Child(SectionGroupComponent.self) - let perksTitle = Child(MultilineTextComponent.self) - let perksSection = Child(SectionGroupComponent.self) + let businessSection = Child(ListSectionComponent.self) + let moreBusinessSection = Child(ListSectionComponent.self) + let perksSection = Child(ListSectionComponent.self) let infoBackground = Child(RoundedRectangle.self) let infoTitle = Child(MultilineTextComponent.self) let infoText = Child(MultilineTextComponent.self) @@ -1736,6 +1683,7 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { let theme = environment.theme let strings = environment.strings + let presentationData = context.component.context.sharedContext.currentPresentationData.with { $0 } let availableWidth = context.availableSize.width let sideInsets = sideInset * 2.0 + environment.safeInsets.left + environment.safeInsets.right @@ -1776,9 +1724,7 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { let textColor = theme.list.itemPrimaryTextColor let accentColor = theme.list.itemAccentColor - let titleColor = theme.list.itemPrimaryTextColor let subtitleColor = theme.list.itemSecondaryTextColor - let arrowColor = theme.list.disclosureArrowColor let textFont = Font.regular(15.0) let boldTextFont = Font.semibold(15.0) @@ -1809,13 +1755,21 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { textString = strings.Premium_PersonalDescription } } else if context.component.isPremium == true { - if !context.component.justBought, let products = state.products, let current = products.first(where: { $0.isCurrent }), current.months == 1 { - textString = strings.Premium_UpgradeDescription + if case .business = context.component.mode { + textString = strings.Business_SubscribedDescription } else { - textString = strings.Premium_SubscribedDescription + if !context.component.justBought, let products = state.products, let current = products.first(where: { $0.isCurrent }), current.months == 1 { + textString = strings.Premium_UpgradeDescription + } else { + textString = strings.Premium_SubscribedDescription + } } } else { - textString = strings.Premium_Description + if case .business = context.component.mode { + textString = strings.Business_Description + } else { + textString = strings.Premium_Description + } } let markdownAttributes = MarkdownAttributes(body: MarkdownAttributeSet(font: textFont, textColor: textColor), bold: MarkdownAttributeSet(font: boldTextFont, textColor: textColor), link: MarkdownAttributeSet(font: textFont, textColor: accentColor), linkAttribute: { contents in @@ -1866,6 +1820,7 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { UIColor(rgb: 0xef6922), UIColor(rgb: 0xe95a2c), UIColor(rgb: 0xe74e33), + UIColor(rgb: 0xe74e33), //replace UIColor(rgb: 0xe54937), UIColor(rgb: 0xe3433c), UIColor(rgb: 0xdb374b), @@ -1876,6 +1831,7 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { UIColor(rgb: 0x9b4fed), UIColor(rgb: 0x8958ff), UIColor(rgb: 0x676bff), + UIColor(rgb: 0x676bff), //replace UIColor(rgb: 0x6172ff), UIColor(rgb: 0x5b79ff), UIColor(rgb: 0x4492ff), @@ -1887,6 +1843,7 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { let accountContext = context.component.context let present = context.component.present + let push = context.component.push let selectProduct = context.component.selectProduct let buy = context.component.buy let updateIsFocused = context.component.updateIsFocused @@ -2009,50 +1966,54 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { let forceDark = context.component.forceDark let layoutPerks = { size.height += 8.0 - let perksTitle = perksTitle.update( - component: MultilineTextComponent( - text: .plain( - NSAttributedString(string: strings.Premium_WhatsIncluded.uppercased(), font: Font.regular(14.0), textColor: environment.theme.list.freeTextColor) - ), - horizontalAlignment: .natural, - maximumNumberOfLines: 0, - lineSpacing: 0.2 - ), - environment: {}, - availableSize: CGSize(width: availableWidth - sideInsets, height: .greatestFiniteMagnitude), - transition: context.transition - ) - context.add(perksTitle - .position(CGPoint(x: sideInset + environment.safeInsets.left + textSideInset + perksTitle.size.width / 2.0, y: size.height + perksTitle.size.height / 2.0)) - ) - size.height += perksTitle.size.height - size.height += 3.0 - + var i = 0 - var perksItems: [SectionGroupComponent.Item] = [] - for perk in state.configuration.perks { - let iconBackgroundColors = gradientColors[i] - perksItems.append(SectionGroupComponent.Item( - AnyComponentWithIdentity( - id: perk.identifier, - component: AnyComponent( - PerkComponent( - iconName: perk.iconName, - iconBackgroundColors: [ - iconBackgroundColors - ], - title: perk.title(strings: strings), - titleColor: titleColor, - subtitle: perk.subtitle(strings: strings), - subtitleColor: subtitleColor, - arrowColor: arrowColor, - accentColor: accentColor, - badge: state.newPerks.contains(perk.identifier) ? strings.Premium_New : nil - ) - ) - ), - accessibilityLabel: "\(perk.title(strings: strings)). \(perk.subtitle(strings: strings))", - action: { [weak state] in + var perksItems: [AnyComponentWithIdentity] = [] + for perk in state.configuration.perks { + if case .business = context.component.mode, case .business = perk { + continue + } + + let isNew = state.newPerks.contains(perk.identifier) + let titleComponent = AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: perk.title(strings: strings), + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemPrimaryTextColor + )), + maximumNumberOfLines: 0 + )) + + let titleCombinedComponent: AnyComponent + if isNew { + titleCombinedComponent = AnyComponent(HStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: titleComponent), + AnyComponentWithIdentity(id: AnyHashable(1), component: AnyComponent(BadgeComponent(color: gradientColors[i], text: strings.Premium_New))) + ], spacing: 5.0)) + } else { + titleCombinedComponent = AnyComponent(HStack([AnyComponentWithIdentity(id: AnyHashable(0), component: titleComponent)], spacing: 0.0)) + } + + perksItems.append(AnyComponentWithIdentity(id: perksItems.count, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: titleCombinedComponent), + AnyComponentWithIdentity(id: AnyHashable(1), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: perk.subtitle(strings: strings), + font: Font.regular(floor(presentationData.listsFontSize.baseDisplaySize * 13.0 / 17.0)), + textColor: environment.theme.list.itemSecondaryTextColor + )), + maximumNumberOfLines: 0, + lineSpacing: 0.18 + ))) + ], alignment: .left, spacing: 2.0)), + leftIcon: AnyComponentWithIdentity(id: 0, component: AnyComponent(PerkIconComponent( + backgroundColor: gradientColors[i], + foregroundColor: .white, + iconName: perk.iconName + ))), + action: { [weak state] _ in var demoSubject: PremiumDemoScreen.Subject switch perk { case .doubleLimits: @@ -2077,7 +2038,6 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { demoSubject = .animatedUserpics case .appIcons: demoSubject = .appIcons -// let _ = ApplicationSpecificNotice.setDismissedPremiumAppIconsBadge(accountManager: accountContext.sharedContext.accountManager).startStandalone() case .animatedEmoji: demoSubject = .animatedEmoji case .emojiStatus: @@ -2101,8 +2061,13 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { case .messagePrivacy: demoSubject = .messagePrivacy let _ = ApplicationSpecificNotice.setDismissedMessagePrivacyBadge(accountManager: accountContext.sharedContext.accountManager).startStandalone() + case .business: + demoSubject = .business + let _ = ApplicationSpecificNotice.setDismissedBusinessBadge(accountManager: accountContext.sharedContext.accountManager).startStandalone() + default: + demoSubject = .doubleLimits } - + let isPremium = state?.isPremium == true var dismissImpl: (() -> Void)? let controller = PremiumLimitsListScreen(context: accountContext, subject: demoSubject, source: .intro(state?.price), order: state?.configuration.perks, buttonText: isPremium ? strings.Common_OK : (state?.isAnnual == true ? strings.Premium_SubscribeForAnnual(state?.price ?? "—").string : strings.Premium_SubscribeFor(state?.price ?? "–").string), isPremium: isPremium, forceDark: forceDark) @@ -2120,19 +2085,26 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { controller?.dismiss(animated: true, completion: nil) } updateIsFocused(true) - + addAppLogEvent(postbox: accountContext.account.postbox, type: "premium.promo_screen_tap", data: ["item": perk.identifier]) } - )) + )))) i += 1 } let perksSection = perksSection.update( - component: SectionGroupComponent( - items: perksItems, - backgroundColor: environment.theme.list.itemBlocksBackgroundColor, - selectionColor: environment.theme.list.itemHighlightedBackgroundColor, - separatorColor: environment.theme.list.itemBlocksSeparatorColor + component: ListSectionComponent( + theme: environment.theme, + header: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: strings.Premium_WhatsIncluded.uppercased(), + font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), + textColor: environment.theme.list.freeTextColor + )), + maximumNumberOfLines: 0 + )), + footer: nil, + items: perksItems ), environment: {}, availableSize: CGSize(width: availableWidth - sideInsets, height: .greatestFiniteMagnitude), @@ -2142,6 +2114,7 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { .position(CGPoint(x: availableWidth / 2.0, y: size.height + perksSection.size.height / 2.0)) .clipsToBounds(true) .cornerRadius(10.0) + .disappear(.default(alpha: true)) ) size.height += perksSection.size.height @@ -2156,6 +2129,314 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { } } + let layoutBusinessPerks = { + size.height += 8.0 + + let gradientColors: [UIColor] = [ + UIColor(rgb: 0xef6922), + UIColor(rgb: 0xe54937), + UIColor(rgb: 0xdb374b), + UIColor(rgb: 0xbc4395), + UIColor(rgb: 0x9b4fed), + UIColor(rgb: 0x8958ff) + ] + + var i = 0 + var perksItems: [AnyComponentWithIdentity] = [] + for perk in state.configuration.businessPerks { + perksItems.append(AnyComponentWithIdentity(id: perksItems.count, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: perk.title(strings: strings), + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemPrimaryTextColor + )), + maximumNumberOfLines: 0 + ))), + AnyComponentWithIdentity(id: AnyHashable(1), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: perk.subtitle(strings: strings), + font: Font.regular(floor(presentationData.listsFontSize.baseDisplaySize * 13.0 / 17.0)), + textColor: environment.theme.list.itemSecondaryTextColor + )), + maximumNumberOfLines: 0, + lineSpacing: 0.18 + ))) + ], alignment: .left, spacing: 2.0)), + leftIcon: AnyComponentWithIdentity(id: 0, component: AnyComponent(PerkIconComponent( + backgroundColor: gradientColors[i], + foregroundColor: .white, + iconName: perk.iconName + ))), + action: { [weak state] _ in + let isPremium = state?.isPremium == true + if isPremium { + switch perk { + case .businessLocation: + let _ = (accountContext.engine.data.get( + TelegramEngine.EngineData.Item.Peer.BusinessLocation(id: accountContext.account.peerId) + ) + |> deliverOnMainQueue).start(next: { [weak accountContext] businessLocation in + guard let accountContext else { + return + } + push(accountContext.sharedContext.makeBusinessLocationSetupScreen(context: accountContext, initialValue: businessLocation, completion: { _ in })) + }) + case .businessHours: + let _ = (accountContext.engine.data.get( + TelegramEngine.EngineData.Item.Peer.BusinessHours(id: accountContext.account.peerId) + ) + |> deliverOnMainQueue).start(next: { [weak accountContext] businessHours in + guard let accountContext else { + return + } + push(accountContext.sharedContext.makeBusinessHoursSetupScreen(context: accountContext, initialValue: businessHours, completion: { _ in })) + }) + case .businessQuickReplies: + let _ = (accountContext.sharedContext.makeQuickReplySetupScreenInitialData(context: accountContext) + |> take(1) + |> deliverOnMainQueue).start(next: { [weak accountContext] initialData in + guard let accountContext else { + return + } + push(accountContext.sharedContext.makeQuickReplySetupScreen(context: accountContext, initialData: initialData)) + }) + case .businessGreetingMessage: + let _ = (accountContext.sharedContext.makeAutomaticBusinessMessageSetupScreenInitialData(context: accountContext) + |> take(1) + |> deliverOnMainQueue).start(next: { [weak accountContext] initialData in + guard let accountContext else { + return + } + push(accountContext.sharedContext.makeAutomaticBusinessMessageSetupScreen(context: accountContext, initialData: initialData, isAwayMode: false)) + }) + case .businessAwayMessage: + let _ = (accountContext.sharedContext.makeAutomaticBusinessMessageSetupScreenInitialData(context: accountContext) + |> take(1) + |> deliverOnMainQueue).start(next: { [weak accountContext] initialData in + guard let accountContext else { + return + } + push(accountContext.sharedContext.makeAutomaticBusinessMessageSetupScreen(context: accountContext, initialData: initialData, isAwayMode: true)) + }) + case .businessChatBots: + let _ = (accountContext.sharedContext.makeChatbotSetupScreenInitialData(context: accountContext) + |> take(1) + |> deliverOnMainQueue).start(next: { [weak accountContext] initialData in + guard let accountContext else { + return + } + push(accountContext.sharedContext.makeChatbotSetupScreen(context: accountContext, initialData: initialData)) + }) + default: + fatalError() + } + } else { + var demoSubject: PremiumDemoScreen.Subject + switch perk { + case .businessLocation: + demoSubject = .businessLocation + case .businessHours: + demoSubject = .businessHours + case .businessQuickReplies: + demoSubject = .businessQuickReplies + case .businessGreetingMessage: + demoSubject = .businessGreetingMessage + case .businessAwayMessage: + demoSubject = .businessAwayMessage + case .businessChatBots: + demoSubject = .businessChatBots + default: + fatalError() + } + var dismissImpl: (() -> Void)? + let controller = PremiumLimitsListScreen(context: accountContext, subject: demoSubject, source: .intro(state?.price), order: state?.configuration.businessPerks, buttonText: isPremium ? strings.Common_OK : (state?.isAnnual == true ? strings.Premium_SubscribeForAnnual(state?.price ?? "—").string : strings.Premium_SubscribeFor(state?.price ?? "–").string), isPremium: isPremium, forceDark: forceDark) + controller.action = { [weak state] in + dismissImpl?() + if state?.isPremium == false { + buy() + } + } + controller.disposed = { + updateIsFocused(false) + } + present(controller) + dismissImpl = { [weak controller] in + controller?.dismiss(animated: true, completion: nil) + } + updateIsFocused(true) + } + } + )))) + i += 1 + } + + let businessSection = businessSection.update( + component: ListSectionComponent( + theme: environment.theme, + header: nil, + footer: nil, + items: perksItems + ), + environment: {}, + availableSize: CGSize(width: availableWidth - sideInsets, height: .greatestFiniteMagnitude), + transition: context.transition + ) + context.add(businessSection + .position(CGPoint(x: availableWidth / 2.0, y: size.height + businessSection.size.height / 2.0)) + .clipsToBounds(true) + .cornerRadius(10.0) + ) + size.height += businessSection.size.height + size.height += 23.0 + } + + let layoutMoreBusinessPerks = { + size.height += 8.0 + + let status = state.peer?.emojiStatus + + let accentColor = environment.theme.list.itemAccentColor + var perksItems: [AnyComponentWithIdentity] = [] + perksItems.append(AnyComponentWithIdentity(id: perksItems.count, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: strings.Business_SetEmojiStatus, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemPrimaryTextColor + )), + maximumNumberOfLines: 0 + ))), + AnyComponentWithIdentity(id: AnyHashable(1), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: strings.Business_SetEmojiStatusInfo, + font: Font.regular(floor(presentationData.listsFontSize.baseDisplaySize * 13.0 / 17.0)), + textColor: environment.theme.list.itemSecondaryTextColor + )), + maximumNumberOfLines: 0, + lineSpacing: 0.18 + ))) + ], alignment: .left, spacing: 2.0)), + leftIcon: AnyComponentWithIdentity(id: 0, component: AnyComponent(PerkIconComponent( + backgroundColor: UIColor(rgb: 0x676bff), + foregroundColor: .white, + iconName: "Premium/BusinessPerk/Status" + ))), + icon: ListActionItemComponent.Icon(component: AnyComponentWithIdentity(id: 0, component: AnyComponent(EmojiActionIconComponent( + context: context.component.context, + color: accentColor, + fileId: status?.fileId, + file: nil + )))), + accessory: nil, + action: { [weak state] view in + guard let view = view as? ListActionItemComponent.View, let iconView = view.iconView else { + return + } + state?.openEmojiSetup(sourceView: iconView, currentFileId: nil, color: accentColor) + } + )))) + + perksItems.append(AnyComponentWithIdentity(id: perksItems.count, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: strings.Business_TagYourChats, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemPrimaryTextColor + )), + maximumNumberOfLines: 0 + ))), + AnyComponentWithIdentity(id: AnyHashable(1), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: strings.Business_TagYourChatsInfo, + font: Font.regular(floor(presentationData.listsFontSize.baseDisplaySize * 13.0 / 17.0)), + textColor: environment.theme.list.itemSecondaryTextColor + )), + maximumNumberOfLines: 0, + lineSpacing: 0.18 + ))) + ], alignment: .left, spacing: 2.0)), + leftIcon: AnyComponentWithIdentity(id: 0, component: AnyComponent(PerkIconComponent( + backgroundColor: UIColor(rgb: 0x4492ff), + foregroundColor: .white, + iconName: "Premium/BusinessPerk/Tag" + ))), + action: { _ in + push(accountContext.sharedContext.makeFilterSettingsController(context: accountContext, modal: false, scrollToTags: true, dismissed: nil)) + } + )))) + + perksItems.append(AnyComponentWithIdentity(id: perksItems.count, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: strings.Business_AddPost, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemPrimaryTextColor + )), + maximumNumberOfLines: 0 + ))), + AnyComponentWithIdentity(id: AnyHashable(1), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: strings.Business_AddPostInfo, + font: Font.regular(floor(presentationData.listsFontSize.baseDisplaySize * 13.0 / 17.0)), + textColor: environment.theme.list.itemSecondaryTextColor + )), + maximumNumberOfLines: 0, + lineSpacing: 0.18 + ))) + ], alignment: .left, spacing: 2.0)), + leftIcon: AnyComponentWithIdentity(id: 0, component: AnyComponent(PerkIconComponent( + backgroundColor: UIColor(rgb: 0x41a6a5), + foregroundColor: .white, + iconName: "Premium/Perk/Stories" + ))), + action: { _ in + push(accountContext.sharedContext.makeMyStoriesController(context: accountContext, isArchive: false)) + } + )))) + + let moreBusinessSection = moreBusinessSection.update( + component: ListSectionComponent( + theme: environment.theme, + header: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: strings.Business_MoreFeaturesTitle.uppercased(), + font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), + textColor: environment.theme.list.freeTextColor + )), + maximumNumberOfLines: 0 + )), + footer: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: environment.strings.Business_MoreFeaturesInfo, + font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), + textColor: environment.theme.list.freeTextColor + )), + maximumNumberOfLines: 0 + )), + items: perksItems + ), + environment: {}, + availableSize: CGSize(width: availableWidth - sideInsets, height: .greatestFiniteMagnitude), + transition: context.transition + ) + context.add(moreBusinessSection + .position(CGPoint(x: availableWidth / 2.0, y: size.height + moreBusinessSection.size.height / 2.0)) + .clipsToBounds(true) + .cornerRadius(10.0) + ) + size.height += moreBusinessSection.size.height + size.height += 23.0 + } + let copyLink = context.component.copyLink if case .emojiStatus = context.component.source { layoutPerks() @@ -2186,154 +2467,165 @@ private final class PremiumIntroScreenContentComponent: CombinedComponent { layoutPerks() } else { layoutOptions() - layoutPerks() - let textPadding: CGFloat = 13.0 + if case .business = context.component.mode { + layoutBusinessPerks() + if context.component.isPremium == true { + layoutMoreBusinessPerks() + } + } else { + layoutPerks() - let infoTitle = infoTitle.update( - component: MultilineTextComponent( - text: .plain( - NSAttributedString(string: strings.Premium_AboutTitle.uppercased(), font: Font.regular(14.0), textColor: environment.theme.list.freeTextColor) + let textPadding: CGFloat = 13.0 + + let infoTitle = infoTitle.update( + component: MultilineTextComponent( + text: .plain( + NSAttributedString(string: strings.Premium_AboutTitle.uppercased(), font: Font.regular(14.0), textColor: environment.theme.list.freeTextColor) + ), + horizontalAlignment: .natural, + maximumNumberOfLines: 0, + lineSpacing: 0.2 ), - horizontalAlignment: .natural, - maximumNumberOfLines: 0, - lineSpacing: 0.2 - ), - environment: {}, - availableSize: CGSize(width: availableWidth - sideInsets, height: .greatestFiniteMagnitude), - transition: context.transition - ) - context.add(infoTitle - .position(CGPoint(x: sideInset + environment.safeInsets.left + textSideInset + infoTitle.size.width / 2.0, y: size.height + infoTitle.size.height / 2.0)) - ) - size.height += infoTitle.size.height - size.height += 3.0 - - let infoText = infoText.update( - component: MultilineTextComponent( - text: .markdown( - text: strings.Premium_AboutText, - attributes: markdownAttributes + environment: {}, + availableSize: CGSize(width: availableWidth - sideInsets, height: .greatestFiniteMagnitude), + transition: context.transition + ) + context.add(infoTitle + .position(CGPoint(x: sideInset + environment.safeInsets.left + textSideInset + infoTitle.size.width / 2.0, y: size.height + infoTitle.size.height / 2.0)) + ) + size.height += infoTitle.size.height + size.height += 3.0 + + let infoText = infoText.update( + component: MultilineTextComponent( + text: .markdown( + text: strings.Premium_AboutText, + attributes: markdownAttributes + ), + horizontalAlignment: .natural, + maximumNumberOfLines: 0, + lineSpacing: 0.2 ), - horizontalAlignment: .natural, - maximumNumberOfLines: 0, - lineSpacing: 0.2 - ), - environment: {}, - availableSize: CGSize(width: availableWidth - sideInsets - textSideInset * 2.0, height: .greatestFiniteMagnitude), - transition: context.transition - ) - - let infoBackground = infoBackground.update( - component: RoundedRectangle( - color: environment.theme.list.itemBlocksBackgroundColor, - cornerRadius: 10.0 - ), - environment: {}, - availableSize: CGSize(width: availableWidth - sideInsets, height: infoText.size.height + textPadding * 2.0), - transition: context.transition - ) - context.add(infoBackground - .position(CGPoint(x: size.width / 2.0, y: size.height + infoBackground.size.height / 2.0)) - ) - context.add(infoText - .position(CGPoint(x: sideInset + environment.safeInsets.left + textSideInset + infoText.size.width / 2.0, y: size.height + textPadding + infoText.size.height / 2.0)) - ) - size.height += infoBackground.size.height - size.height += 6.0 - - let termsFont = Font.regular(13.0) - let boldTermsFont = Font.semibold(13.0) - let italicTermsFont = Font.italic(13.0) - let boldItalicTermsFont = Font.semiboldItalic(13.0) - let monospaceTermsFont = Font.monospace(13.0) - let termsTextColor = environment.theme.list.freeTextColor - let termsMarkdownAttributes = MarkdownAttributes(body: MarkdownAttributeSet(font: termsFont, textColor: termsTextColor), bold: MarkdownAttributeSet(font: termsFont, textColor: termsTextColor), link: MarkdownAttributeSet(font: termsFont, textColor: environment.theme.list.itemAccentColor), linkAttribute: { contents in - return (TelegramTextAttributes.URL, contents) - }) - - var isGiftView = false - if case let .gift(fromId, _, _, _) = context.component.source { - if fromId == context.component.context.account.peerId { - isGiftView = true - } - } - - let termsString: MultilineTextComponent.TextContent - if isGiftView { - termsString = .plain(NSAttributedString()) - } else if let promoConfiguration = context.component.promoConfiguration { - let attributedString = stringWithAppliedEntities(promoConfiguration.status, entities: promoConfiguration.statusEntities, baseColor: termsTextColor, linkColor: environment.theme.list.itemAccentColor, baseFont: termsFont, linkFont: termsFont, boldFont: boldTermsFont, italicFont: italicTermsFont, boldItalicFont: boldItalicTermsFont, fixedFont: monospaceTermsFont, blockQuoteFont: termsFont, message: nil) - termsString = .plain(attributedString) - } else { - termsString = .markdown( - text: strings.Premium_Terms, - attributes: termsMarkdownAttributes + environment: {}, + availableSize: CGSize(width: availableWidth - sideInsets - textSideInset * 2.0, height: .greatestFiniteMagnitude), + transition: context.transition ) - } - - let controller = environment.controller - let termsTapActionImpl: ([NSAttributedString.Key: Any]) -> Void = { attributes in - if let url = attributes[NSAttributedString.Key(rawValue: TelegramTextAttributes.URL)] as? String, - let controller = controller() as? PremiumIntroScreen, let navigationController = controller.navigationController as? NavigationController { - if url.hasPrefix("https://apps.apple.com/account/subscriptions") { - controller.context.sharedContext.applicationBindings.openSubscriptions() - } else if url.hasPrefix("https://") || url.hasPrefix("tg://") { - controller.context.sharedContext.openExternalUrl(context: controller.context, urlContext: .generic, url: url, forceExternal: !url.hasPrefix("tg://"), presentationData: controller.context.sharedContext.currentPresentationData.with({$0}), navigationController: nil, dismissInput: {}) - } else { - let context = controller.context - let signal: Signal? - switch url { - case "terms": - signal = cachedTermsPage(context: context) - case "privacy": - signal = cachedPrivacyPage(context: context) - default: - signal = nil - } - if let signal = signal { - let _ = (signal - |> deliverOnMainQueue).start(next: { resolvedUrl in - context.sharedContext.openResolvedUrl(resolvedUrl, context: context, urlContext: .generic, navigationController: navigationController, forceExternal: false, openPeer: { peer, navigation in - }, sendFile: nil, sendSticker: nil, requestMessageActionUrlAuth: nil, joinVoiceChat: nil, present: { [weak controller] c, arguments in - controller?.push(c) - }, dismissInput: {}, contentContext: nil, progress: nil, completion: nil) - }) - } + + let infoBackground = infoBackground.update( + component: RoundedRectangle( + color: environment.theme.list.itemBlocksBackgroundColor, + cornerRadius: 10.0 + ), + environment: {}, + availableSize: CGSize(width: availableWidth - sideInsets, height: infoText.size.height + textPadding * 2.0), + transition: context.transition + ) + context.add(infoBackground + .position(CGPoint(x: size.width / 2.0, y: size.height + infoBackground.size.height / 2.0)) + ) + context.add(infoText + .position(CGPoint(x: sideInset + environment.safeInsets.left + textSideInset + infoText.size.width / 2.0, y: size.height + textPadding + infoText.size.height / 2.0)) + ) + size.height += infoBackground.size.height + size.height += 6.0 + + let termsFont = Font.regular(13.0) + let boldTermsFont = Font.semibold(13.0) + let italicTermsFont = Font.italic(13.0) + let boldItalicTermsFont = Font.semiboldItalic(13.0) + let monospaceTermsFont = Font.monospace(13.0) + let termsTextColor = environment.theme.list.freeTextColor + let termsMarkdownAttributes = MarkdownAttributes(body: MarkdownAttributeSet(font: termsFont, textColor: termsTextColor), bold: MarkdownAttributeSet(font: termsFont, textColor: termsTextColor), link: MarkdownAttributeSet(font: termsFont, textColor: environment.theme.list.itemAccentColor), linkAttribute: { contents in + return (TelegramTextAttributes.URL, contents) + }) + + var isGiftView = false + if case let .gift(fromId, _, _, _) = context.component.source { + if fromId == context.component.context.account.peerId { + isGiftView = true } } - } - - let termsText = termsText.update( - component: MultilineTextComponent( - text: termsString, - horizontalAlignment: .natural, - maximumNumberOfLines: 0, - lineSpacing: 0.0, - highlightColor: environment.theme.list.itemAccentColor.withAlphaComponent(0.2), - highlightAction: { attributes in - if let _ = attributes[NSAttributedString.Key(rawValue: TelegramTextAttributes.URL)] { - return NSAttributedString.Key(rawValue: TelegramTextAttributes.URL) + + let termsString: MultilineTextComponent.TextContent + if isGiftView { + termsString = .plain(NSAttributedString()) + } else if let promoConfiguration = context.component.promoConfiguration { + let attributedString = stringWithAppliedEntities(promoConfiguration.status, entities: promoConfiguration.statusEntities, baseColor: termsTextColor, linkColor: environment.theme.list.itemAccentColor, baseFont: termsFont, linkFont: termsFont, boldFont: boldTermsFont, italicFont: italicTermsFont, boldItalicFont: boldItalicTermsFont, fixedFont: monospaceTermsFont, blockQuoteFont: termsFont, message: nil) + termsString = .plain(attributedString) + } else { + termsString = .markdown( + text: strings.Premium_Terms, + attributes: termsMarkdownAttributes + ) + } + + let controller = environment.controller + let termsTapActionImpl: ([NSAttributedString.Key: Any]) -> Void = { attributes in + if let url = attributes[NSAttributedString.Key(rawValue: TelegramTextAttributes.URL)] as? String, + let controller = controller() as? PremiumIntroScreen, let navigationController = controller.navigationController as? NavigationController { + if url.hasPrefix("https://apps.apple.com/account/subscriptions") { + controller.context.sharedContext.applicationBindings.openSubscriptions() + } else if url.hasPrefix("https://") || url.hasPrefix("tg://") { + controller.context.sharedContext.openExternalUrl(context: controller.context, urlContext: .generic, url: url, forceExternal: !url.hasPrefix("tg://"), presentationData: controller.context.sharedContext.currentPresentationData.with({$0}), navigationController: nil, dismissInput: {}) } else { - return nil + let context = controller.context + let signal: Signal? + switch url { + case "terms": + signal = cachedTermsPage(context: context) + case "privacy": + signal = cachedPrivacyPage(context: context) + default: + signal = nil + } + if let signal = signal { + let _ = (signal + |> deliverOnMainQueue).start(next: { resolvedUrl in + context.sharedContext.openResolvedUrl(resolvedUrl, context: context, urlContext: .generic, navigationController: navigationController, forceExternal: false, openPeer: { peer, navigation in + }, sendFile: nil, sendSticker: nil, requestMessageActionUrlAuth: nil, joinVoiceChat: nil, present: { [weak controller] c, arguments in + controller?.push(c) + }, dismissInput: {}, contentContext: nil, progress: nil, completion: nil) + }) + } } - }, - tapAction: { attributes, _ in - termsTapActionImpl(attributes) } - ), - environment: {}, - availableSize: CGSize(width: availableWidth - sideInsets - textSideInset * 2.0, height: .greatestFiniteMagnitude), - transition: context.transition - ) - context.add(termsText - .position(CGPoint(x: sideInset + environment.safeInsets.left + textSideInset + termsText.size.width / 2.0, y: size.height + termsText.size.height / 2.0)) - ) - size.height += termsText.size.height - size.height += 10.0 + } + + let termsText = termsText.update( + component: MultilineTextComponent( + text: termsString, + horizontalAlignment: .natural, + maximumNumberOfLines: 0, + lineSpacing: 0.0, + highlightColor: environment.theme.list.itemAccentColor.withAlphaComponent(0.2), + highlightAction: { attributes in + if let _ = attributes[NSAttributedString.Key(rawValue: TelegramTextAttributes.URL)] { + return NSAttributedString.Key(rawValue: TelegramTextAttributes.URL) + } else { + return nil + } + }, + tapAction: { attributes, _ in + termsTapActionImpl(attributes) + } + ), + environment: {}, + availableSize: CGSize(width: availableWidth - sideInsets - textSideInset * 2.0, height: .greatestFiniteMagnitude), + transition: context.transition + ) + context.add(termsText + .position(CGPoint(x: sideInset + environment.safeInsets.left + textSideInset + termsText.size.width / 2.0, y: size.height + termsText.size.height / 2.0)) + ) + size.height += termsText.size.height + size.height += 10.0 + } } size.height += scrollEnvironment.insets.bottom + if case .business = context.component.mode, state.isPremium == false { + size.height += 123.0 + } if context.component.source != .settings { size.height += 44.0 @@ -2348,6 +2640,7 @@ private final class PremiumIntroScreenComponent: CombinedComponent { typealias EnvironmentType = ViewControllerComponentContainer.Environment let context: AccountContext + let mode: PremiumIntroScreen.Mode let source: PremiumSource let forceDark: Bool let forceHasPremium: Bool @@ -2358,8 +2651,9 @@ private final class PremiumIntroScreenComponent: CombinedComponent { let copyLink: (String) -> Void let shareLink: (String) -> Void - init(context: AccountContext, source: PremiumSource, forceDark: Bool, forceHasPremium: Bool, updateInProgress: @escaping (Bool) -> Void, present: @escaping (ViewController) -> Void, push: @escaping (ViewController) -> Void, completion: @escaping () -> Void, copyLink: @escaping (String) -> Void, shareLink: @escaping (String) -> Void) { + init(context: AccountContext, mode: PremiumIntroScreen.Mode, source: PremiumSource, forceDark: Bool, forceHasPremium: Bool, updateInProgress: @escaping (Bool) -> Void, present: @escaping (ViewController) -> Void, push: @escaping (ViewController) -> Void, completion: @escaping () -> Void, copyLink: @escaping (String) -> Void, shareLink: @escaping (String) -> Void) { self.context = context + self.mode = mode self.source = source self.forceDark = forceDark self.forceHasPremium = forceHasPremium @@ -2375,6 +2669,9 @@ private final class PremiumIntroScreenComponent: CombinedComponent { if lhs.context !== rhs.context { return false } + if lhs.mode != rhs.mode { + return false + } if lhs.source != rhs.source { return false } @@ -2757,6 +3054,7 @@ private final class PremiumIntroScreenComponent: CombinedComponent { let scrollContent = Child(ScrollComponent.self) let star = Child(PremiumStarComponent.self) let emoji = Child(EmojiHeaderComponent.self) + let coin = Child(PremiumCoinComponent.self) let topPanel = Child(BlurredBackgroundComponent.self) let topSeparator = Child(Rectangle.self) let title = Child(MultilineTextComponent.self) @@ -2782,7 +3080,17 @@ private final class PremiumIntroScreenComponent: CombinedComponent { } let header: _UpdatedChildComponent - if case let .emojiStatus(_, fileId, _, _) = context.component.source { + if case .business = context.component.mode { + header = coin.update( + component: PremiumCoinComponent( + isIntro: isIntro, + isVisible: starIsVisible, + hasIdleAnimations: state.hasIdleAnimations + ), + availableSize: CGSize(width: min(414.0, context.availableSize.width), height: 220.0), + transition: context.transition + ) + } else if case let .emojiStatus(_, fileId, _, _) = context.component.source { header = emoji.update( component: EmojiHeaderComponent( context: context.component.context, @@ -2826,7 +3134,9 @@ private final class PremiumIntroScreenComponent: CombinedComponent { ) let titleString: String - if case .emojiStatus = context.component.source { + if case .business = context.component.mode { + titleString = environment.strings.Business_Title + } else if case .emojiStatus = context.component.source { titleString = environment.strings.Premium_Title } else if case .giftTerms = context.component.source { titleString = environment.strings.Premium_Title @@ -2858,10 +3168,14 @@ private final class PremiumIntroScreenComponent: CombinedComponent { let secondaryTitleText: String var isAnonymous = false if var otherPeerName = state.otherPeerName { - if case let .emojiStatus(_, _, file, maybeEmojiPack) = context.component.source, let emojiPack = maybeEmojiPack, case let .result(info, _, _) = emojiPack { + if case let .emojiStatus(peerId, _, file, maybeEmojiPack) = context.component.source, let emojiPack = maybeEmojiPack, case let .result(info, _, _) = emojiPack { loadedEmojiPack = maybeEmojiPack highlightableLinks = true + if peerId.isGroupOrChannel, otherPeerName.count > 20 { + otherPeerName = otherPeerName.prefix(20).trimmingCharacters(in: .whitespacesAndNewlines) + "\u{2026}" + } + var packReference: StickerPackReference? if let file = file { for attribute in file.attributes { @@ -2978,6 +3292,7 @@ private final class PremiumIntroScreenComponent: CombinedComponent { component: ScrollComponent( content: AnyComponent(PremiumIntroScreenContentComponent( context: context.component.context, + mode: context.component.mode, source: context.component.source, forceDark: context.component.forceDark, isPremium: state.isPremium, @@ -2988,6 +3303,7 @@ private final class PremiumIntroScreenComponent: CombinedComponent { validPurchases: state.validPurchases, promoConfiguration: state.promoConfiguration, present: context.component.present, + push: context.component.push, selectProduct: { [weak state] productId in state?.selectProduct(productId) }, @@ -3204,7 +3520,13 @@ private final class PremiumIntroScreenComponent: CombinedComponent { } public final class PremiumIntroScreen: ViewControllerComponentContainer { + public enum Mode { + case premium + case business + } + fileprivate let context: AccountContext + fileprivate let mode: Mode private var didSetReady = false private let _ready = Promise() @@ -3216,8 +3538,9 @@ public final class PremiumIntroScreen: ViewControllerComponentContainer { public weak var containerView: UIView? public var animationColor: UIColor? - public init(context: AccountContext, modal: Bool = true, source: PremiumSource, forceDark: Bool = false, forceHasPremium: Bool = false) { + public init(context: AccountContext, mode: Mode = .premium, source: PremiumSource, modal: Bool = true, forceDark: Bool = false, forceHasPremium: Bool = false) { self.context = context + self.mode = mode var updateInProgressImpl: ((Bool) -> Void)? var pushImpl: ((ViewController) -> Void)? @@ -3227,6 +3550,7 @@ public final class PremiumIntroScreen: ViewControllerComponentContainer { var shareLinkImpl: ((String) -> Void)? super.init(context: context, component: PremiumIntroScreenComponent( context: context, + mode: mode, source: source, forceDark: forceDark, forceHasPremium: forceHasPremium, @@ -3324,6 +3648,10 @@ public final class PremiumIntroScreen: ViewControllerComponentContainer { } navigationController.pushViewController(peerSelectionController) } + + if case .business = mode { + context.account.viewTracker.keepQuickRepliesApproximatelyUpdated() + } } required public init(coder aDecoder: NSCoder) { @@ -3362,7 +3690,10 @@ public final class PremiumIntroScreen: ViewControllerComponentContainer { super.containerLayoutUpdated(layout, transition: transition) if !self.didSetReady { - if let view = self.node.hostView.findTaggedView(tag: PremiumStarComponent.View.Tag()) as? PremiumStarComponent.View { + if let view = self.node.hostView.findTaggedView(tag: PremiumCoinComponent.View.Tag()) as? PremiumCoinComponent.View { + self.didSetReady = true + self._ready.set(view.ready) + } else if let view = self.node.hostView.findTaggedView(tag: PremiumStarComponent.View.Tag()) as? PremiumStarComponent.View { self.didSetReady = true self._ready.set(view.ready) @@ -3393,3 +3724,144 @@ public final class PremiumIntroScreen: ViewControllerComponentContainer { } } } + + + + +private final class EmojiActionIconComponent: Component { + let context: AccountContext + let color: UIColor + let fileId: Int64? + let file: TelegramMediaFile? + + init( + context: AccountContext, + color: UIColor, + fileId: Int64?, + file: TelegramMediaFile? + ) { + self.context = context + self.color = color + self.fileId = fileId + self.file = file + } + + static func ==(lhs: EmojiActionIconComponent, rhs: EmojiActionIconComponent) -> Bool { + if lhs.context !== rhs.context { + return false + } + if lhs.color != rhs.color { + return false + } + if lhs.fileId != rhs.fileId { + return false + } + if lhs.file != rhs.file { + return false + } + return true + } + + final class View: UIView { + private let icon = ComponentView() + + func update(component: EmojiActionIconComponent, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + let size = CGSize(width: 24.0, height: 24.0) + + let _ = self.icon.update( + transition: .immediate, + component: AnyComponent(EmojiStatusComponent( + context: component.context, + animationCache: component.context.animationCache, + animationRenderer: component.context.animationRenderer, + content: component.fileId.flatMap { .animation( + content: .customEmoji(fileId: $0), + size: CGSize(width: size.width * 2.0, height: size.height * 2.0), + placeholderColor: .lightGray, + themeColor: component.color, + loopMode: .forever + ) } ?? .premium(color: component.color), + isVisibleForAnimations: false, + action: nil + )), + environment: {}, + containerSize: size + ) + let iconFrame = CGRect(origin: CGPoint(), size: size) + if let iconView = self.icon.view { + if iconView.superview == nil { + self.addSubview(iconView) + } + iconView.frame = iconFrame + } + + return size + } + } + + func makeView() -> View { + return View(frame: CGRect()) + } + + func update(view: View, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + return view.update(component: self, availableSize: availableSize, state: state, environment: environment, transition: transition) + } +} + +private final class BadgeComponent: CombinedComponent { + let color: UIColor + let text: String + + init( + color: UIColor, + text: String + ) { + self.color = color + self.text = text + } + + static func ==(lhs: BadgeComponent, rhs: BadgeComponent) -> Bool { + if lhs.color != rhs.color { + return false + } + if lhs.text != rhs.text { + return false + } + return true + } + + static var body: Body { + let badgeBackground = Child(RoundedRectangle.self) + let badgeText = Child(MultilineTextComponent.self) + + return { context in + let component = context.component + + let badgeText = badgeText.update( + component: MultilineTextComponent(text: .plain(NSAttributedString(string: component.text, font: Font.semibold(11.0), textColor: .white))), + availableSize: context.availableSize, + transition: context.transition + ) + + let badgeSize = CGSize(width: badgeText.size.width + 7.0, height: 16.0) + let badgeBackground = badgeBackground.update( + component: RoundedRectangle( + color: component.color, + cornerRadius: 5.0 + ), + availableSize: badgeSize, + transition: context.transition + ) + + context.add(badgeBackground + .position(CGPoint(x: badgeSize.width / 2.0, y: badgeSize.height / 2.0)) + ) + + context.add(badgeText + .position(CGPoint(x: badgeSize.width / 2.0, y: badgeSize.height / 2.0)) + ) + + return badgeSize + } + } +} diff --git a/submodules/PremiumUI/Sources/PremiumLimitsListScreen.swift b/submodules/PremiumUI/Sources/PremiumLimitsListScreen.swift index 0ca02854bfe..dc6accd6486 100644 --- a/submodules/PremiumUI/Sources/PremiumLimitsListScreen.swift +++ b/submodules/PremiumUI/Sources/PremiumLimitsListScreen.swift @@ -29,6 +29,7 @@ public class PremiumLimitsListScreen: ViewController { let backgroundView: ComponentHostView let pagerView: ComponentHostView let closeView: ComponentHostView + let closeDarkIconView = UIImageView() fileprivate let footerNode: FooterNode @@ -227,6 +228,8 @@ public class PremiumLimitsListScreen: ViewController { if scrollView.contentSize.width > scrollView.contentSize.height || scrollView.contentSize.height > 1500.0 { return false } + } else if otherGestureRecognizer.view is PremiumCoinComponent.View { + return false } return true } @@ -380,22 +383,46 @@ public class PremiumLimitsListScreen: ViewController { isStandalone = true } - if let stickers = self.stickers, let appIcons = self.appIcons, let configuration = self.promoConfiguration { + let theme = self.presentationData.theme + let strings = self.presentationData.strings + + let videos: [String: TelegramMediaFile] = self.promoConfiguration?.videos ?? [:] + let stickers = self.stickers ?? [] + let appIcons = self.appIcons ?? [] + + let isReady: Bool + switch controller.subject { + case .premiumStickers: + isReady = !stickers.isEmpty + case .appIcons: + isReady = !appIcons.isEmpty + case .stories: + isReady = true + case .doubleLimits: + isReady = true + case .business: + isReady = true + default: + isReady = !videos.isEmpty + } + + if isReady { let context = controller.context - let theme = self.presentationData.theme - let strings = self.presentationData.strings - + let textColor = theme.actionSheet.primaryTextColor var availableItems: [PremiumPerk: DemoPagerComponent.Item] = [:] var storiesIndex: Int? var limitsIndex: Int? + var businessIndex: Int? var storiesNeighbors = PageNeighbors(leftIsList: false, rightIsList: false) var limitsNeighbors = PageNeighbors(leftIsList: false, rightIsList: false) + let businessNeighbors = PageNeighbors(leftIsList: false, rightIsList: false) if let order = controller.order { storiesIndex = order.firstIndex(where: { $0 == .stories }) limitsIndex = order.firstIndex(where: { $0 == .doubleLimits }) + businessIndex = order.firstIndex(where: { $0 == .business }) if let limitsIndex, let storiesIndex { if limitsIndex == storiesIndex + 1 { storiesNeighbors.rightIsList = true @@ -477,7 +504,7 @@ public class PremiumLimitsListScreen: ViewController { content: AnyComponent(PhoneDemoComponent( context: context, position: .bottom, - videoFile: configuration.videos["more_upload"], + videoFile: videos["more_upload"], decoration: .dataRain )), title: strings.Premium_UploadSize, @@ -495,7 +522,7 @@ public class PremiumLimitsListScreen: ViewController { content: AnyComponent(PhoneDemoComponent( context: context, position: .top, - videoFile: configuration.videos["faster_download"], + videoFile: videos["faster_download"], decoration: .fasterStars )), title: strings.Premium_FasterSpeed, @@ -513,7 +540,7 @@ public class PremiumLimitsListScreen: ViewController { content: AnyComponent(PhoneDemoComponent( context: context, position: .top, - videoFile: configuration.videos["voice_to_text"], + videoFile: videos["voice_to_text"], decoration: .badgeStars )), title: strings.Premium_VoiceToText, @@ -531,7 +558,7 @@ public class PremiumLimitsListScreen: ViewController { content: AnyComponent(PhoneDemoComponent( context: context, position: .bottom, - videoFile: configuration.videos["no_ads"], + videoFile: videos["no_ads"], decoration: .swirlStars )), title: strings.Premium_NoAds, @@ -549,7 +576,7 @@ public class PremiumLimitsListScreen: ViewController { content: AnyComponent(PhoneDemoComponent( context: context, position: .top, - videoFile: configuration.videos["infinite_reactions"], + videoFile: videos["infinite_reactions"], decoration: .swirlStars )), title: strings.Premium_InfiniteReactions, @@ -588,7 +615,7 @@ public class PremiumLimitsListScreen: ViewController { content: AnyComponent(PhoneDemoComponent( context: context, position: .top, - videoFile: configuration.videos["emoji_status"], + videoFile: videos["emoji_status"], decoration: .badgeStars )), title: strings.Premium_EmojiStatus, @@ -606,7 +633,7 @@ public class PremiumLimitsListScreen: ViewController { content: AnyComponent(PhoneDemoComponent( context: context, position: .top, - videoFile: configuration.videos["advanced_chat_management"], + videoFile: videos["advanced_chat_management"], decoration: .swirlStars )), title: strings.Premium_ChatManagement, @@ -624,7 +651,7 @@ public class PremiumLimitsListScreen: ViewController { content: AnyComponent(PhoneDemoComponent( context: context, position: .top, - videoFile: configuration.videos["profile_badge"], + videoFile: videos["profile_badge"], decoration: .badgeStars )), title: strings.Premium_Badge, @@ -642,7 +669,7 @@ public class PremiumLimitsListScreen: ViewController { content: AnyComponent(PhoneDemoComponent( context: context, position: .top, - videoFile: configuration.videos["animated_userpics"], + videoFile: videos["animated_userpics"], decoration: .swirlStars )), title: strings.Premium_Avatar, @@ -676,7 +703,7 @@ public class PremiumLimitsListScreen: ViewController { content: AnyComponent(PhoneDemoComponent( context: context, position: .bottom, - videoFile: configuration.videos["animated_emoji"], + videoFile: videos["animated_emoji"], decoration: .emoji )), title: strings.Premium_AnimatedEmoji, @@ -695,7 +722,7 @@ public class PremiumLimitsListScreen: ViewController { context: context, position: .top, model: .island, - videoFile: configuration.videos["translations"], + videoFile: videos["translations"], decoration: .hello )), title: strings.Premium_Translation, @@ -713,7 +740,7 @@ public class PremiumLimitsListScreen: ViewController { content: AnyComponent(PhoneDemoComponent( context: context, position: .top, - videoFile: configuration.videos["peer_colors"], + videoFile: videos["peer_colors"], decoration: .badgeStars )), title: strings.Premium_Colors, @@ -732,7 +759,7 @@ public class PremiumLimitsListScreen: ViewController { context: context, position: .top, model: .island, - videoFile: configuration.videos["wallpapers"], + videoFile: videos["wallpapers"], decoration: .swirlStars )), title: strings.Premium_Wallpapers, @@ -751,7 +778,7 @@ public class PremiumLimitsListScreen: ViewController { context: context, position: .top, model: .island, - videoFile: configuration.videos["saved_tags"], + videoFile: videos["saved_tags"], decoration: .tag )), title: strings.Premium_MessageTags, @@ -770,7 +797,7 @@ public class PremiumLimitsListScreen: ViewController { context: context, position: .top, model: .island, - videoFile: configuration.videos["last_seen"], + videoFile: videos["last_seen"], decoration: .badgeStars )), title: strings.Premium_LastSeen, @@ -789,7 +816,7 @@ public class PremiumLimitsListScreen: ViewController { context: context, position: .top, model: .island, - videoFile: configuration.videos["message_privacy"], + videoFile: videos["message_privacy"], decoration: .swirlStars )), title: strings.Premium_MessagePrivacy, @@ -799,7 +826,159 @@ public class PremiumLimitsListScreen: ViewController { ) ) ) - + availableItems[.business] = DemoPagerComponent.Item( + AnyComponentWithIdentity( + id: PremiumDemoScreen.Subject.business, + component: AnyComponent( + BusinessPageComponent( + context: context, + theme: self.presentationData.theme, + neighbors: businessNeighbors, + bottomInset: self.footerNode.frame.height, + updatedBottomAlpha: { [weak self] alpha in + if let strongSelf = self { + strongSelf.footerNode.updateCoverAlpha(alpha, transition: .immediate) + } + }, + updatedDismissOffset: { [weak self] offset in + if let strongSelf = self { + strongSelf.updateDismissOffset(offset) + } + }, + updatedIsDisplaying: { [weak self] isDisplaying in + if let self, self.isExpanded && !isDisplaying { + if let businessIndex, let indexPosition = self.indexPosition, abs(CGFloat(businessIndex) - indexPosition) < 0.1 { + } else { + self.update(isExpanded: false, transition: .animated(duration: 0.2, curve: .easeInOut)) + } + } + } + ) + ) + ) + ) + + + availableItems[.businessLocation] = DemoPagerComponent.Item( + AnyComponentWithIdentity( + id: PremiumDemoScreen.Subject.businessLocation, + component: AnyComponent( + PageComponent( + content: AnyComponent(PhoneDemoComponent( + context: context, + position: .top, + model: .island, + videoFile: videos["business_location"], + decoration: .business + )), + title: strings.Business_Location, + text: strings.Business_LocationInfo, + textColor: textColor + ) + ) + ) + ) + + availableItems[.businessHours] = DemoPagerComponent.Item( + AnyComponentWithIdentity( + id: PremiumDemoScreen.Subject.businessHours, + component: AnyComponent( + PageComponent( + content: AnyComponent(PhoneDemoComponent( + context: context, + position: .top, + model: .island, + videoFile: videos["business_hours"], + decoration: .business + )), + title: strings.Business_OpeningHours, + text: strings.Business_OpeningHoursInfo, + textColor: textColor + ) + ) + ) + ) + + availableItems[.businessQuickReplies] = DemoPagerComponent.Item( + AnyComponentWithIdentity( + id: PremiumDemoScreen.Subject.businessQuickReplies, + component: AnyComponent( + PageComponent( + content: AnyComponent(PhoneDemoComponent( + context: context, + position: .top, + model: .island, + videoFile: videos["quick_replies"], + decoration: .business + )), + title: strings.Business_QuickReplies, + text: strings.Business_QuickRepliesInfo, + textColor: textColor + ) + ) + ) + ) + + availableItems[.businessGreetingMessage] = DemoPagerComponent.Item( + AnyComponentWithIdentity( + id: PremiumDemoScreen.Subject.businessGreetingMessage, + component: AnyComponent( + PageComponent( + content: AnyComponent(PhoneDemoComponent( + context: context, + position: .top, + model: .island, + videoFile: videos["greeting_message"], + decoration: .business + )), + title: strings.Business_GreetingMessages, + text: strings.Business_GreetingMessagesInfo, + textColor: textColor + ) + ) + ) + ) + + availableItems[.businessAwayMessage] = DemoPagerComponent.Item( + AnyComponentWithIdentity( + id: PremiumDemoScreen.Subject.businessAwayMessage, + component: AnyComponent( + PageComponent( + content: AnyComponent(PhoneDemoComponent( + context: context, + position: .top, + model: .island, + videoFile: videos["away_message"], + decoration: .business + )), + title: strings.Business_AwayMessages, + text: strings.Business_AwayMessagesInfo, + textColor: textColor + ) + ) + ) + ) + + availableItems[.businessChatBots] = DemoPagerComponent.Item( + AnyComponentWithIdentity( + id: PremiumDemoScreen.Subject.businessChatBots, + component: AnyComponent( + PageComponent( + content: AnyComponent(PhoneDemoComponent( + context: context, + position: .top, + model: .island, + videoFile: videos["business_bots"], + decoration: .business + )), + title: strings.Business_Chatbots, + text: strings.Business_ChatbotsInfo, + textColor: textColor + ) + ) + ) + ) + if let order = controller.order { var items: [DemoPagerComponent.Item] = order.compactMap { availableItems[$0] } let initialIndex: Int @@ -822,8 +1001,42 @@ public class PremiumLimitsListScreen: ViewController { nextAction: nextAction, updated: { [weak self] position, count in if let self { - self.indexPosition = position * CGFloat(count) + let indexPosition = position * CGFloat(count - 1) + self.indexPosition = indexPosition self.footerNode.updatePosition(position, count: count) + + var distance: CGFloat? + if let storiesIndex { + let value = indexPosition - CGFloat(storiesIndex) + if abs(value) < 1.0 { + distance = value + } + } + if let limitsIndex { + let value = indexPosition - CGFloat(limitsIndex) + if abs(value) < 1.0 { + distance = value + } + } + if let businessIndex { + let value = indexPosition - CGFloat(businessIndex) + if abs(value) < 1.0 { + distance = value + } + } + var distanceToPage: CGFloat = 1.0 + if let distance { + if distance >= 0.0 && distance < 0.1 { + distanceToPage = distance / 0.1 + } else if distance < 0.0 { + if distance >= -1.0 && distance < -0.9 { + distanceToPage = ((distance * -1.0) - 0.9) / 0.1 + } else { + distanceToPage = 0.0 + } + } + } + self.closeDarkIconView.alpha = 1.0 - max(0.0, min(1.0, distanceToPage)) } } ) @@ -852,7 +1065,7 @@ public class PremiumLimitsListScreen: ViewController { id: "background", component: AnyComponent( BlurredBackgroundComponent( - color: UIColor(rgb: 0x888888, alpha: 0.3) + color: UIColor(rgb: 0xbbbbbb, alpha: 0.22) ) ) ), @@ -874,6 +1087,12 @@ public class PremiumLimitsListScreen: ViewController { self.closeView.clipsToBounds = true self.closeView.layer.cornerRadius = 15.0 self.closeView.frame = CGRect(origin: CGPoint(x: contentSize.width - closeSize.width * 1.5, y: 28.0 - closeSize.height / 2.0), size: closeSize) + + if self.closeDarkIconView.image == nil { + self.closeDarkIconView.image = generateCloseButtonImage(backgroundColor: .clear, foregroundColor: theme.list.itemSecondaryTextColor)! + self.closeDarkIconView.frame = CGRect(origin: .zero, size: closeSize) + self.closeView.addSubview(self.closeDarkIconView) + } } private var cachedCloseImage: UIImage? diff --git a/submodules/PremiumUI/Sources/PremiumOptionComponent.swift b/submodules/PremiumUI/Sources/PremiumOptionComponent.swift new file mode 100644 index 00000000000..23ec27e787c --- /dev/null +++ b/submodules/PremiumUI/Sources/PremiumOptionComponent.swift @@ -0,0 +1,340 @@ +import Foundation +import UIKit +import Display +import ComponentFlow +import MultilineTextComponent +import CheckNode +import Markdown + +final class PremiumOptionComponent: CombinedComponent { + let title: String + let subtitle: String + let labelPrice: String + let discount: String + let multiple: Bool + let selected: Bool + let primaryTextColor: UIColor + let secondaryTextColor: UIColor + let accentColor: UIColor + let checkForegroundColor: UIColor + let checkBorderColor: UIColor + + init( + title: String, + subtitle: String, + labelPrice: String, + discount: String, + multiple: Bool = false, + selected: Bool, + primaryTextColor: UIColor, + secondaryTextColor: UIColor, + accentColor: UIColor, + checkForegroundColor: UIColor, + checkBorderColor: UIColor + ) { + self.title = title + self.subtitle = subtitle + self.labelPrice = labelPrice + self.discount = discount + self.multiple = multiple + self.selected = selected + self.primaryTextColor = primaryTextColor + self.secondaryTextColor = secondaryTextColor + self.accentColor = accentColor + self.checkForegroundColor = checkForegroundColor + self.checkBorderColor = checkBorderColor + } + + static func ==(lhs: PremiumOptionComponent, rhs: PremiumOptionComponent) -> Bool { + if lhs.title != rhs.title { + return false + } + if lhs.subtitle != rhs.subtitle { + return false + } + if lhs.labelPrice != rhs.labelPrice { + return false + } + if lhs.discount != rhs.discount { + return false + } + if lhs.multiple != rhs.multiple { + return false + } + if lhs.selected != rhs.selected { + return false + } + if lhs.primaryTextColor != rhs.primaryTextColor { + return false + } + if lhs.secondaryTextColor != rhs.secondaryTextColor { + return false + } + if lhs.accentColor != rhs.accentColor { + return false + } + if lhs.checkForegroundColor != rhs.checkForegroundColor { + return false + } + if lhs.checkBorderColor != rhs.checkBorderColor { + return false + } + return true + } + + static var body: Body { + let check = Child(CheckComponent.self) + let title = Child(MultilineTextComponent.self) + let subtitle = Child(MultilineTextComponent.self) + let discountBackground = Child(RoundedRectangle.self) + let discount = Child(MultilineTextComponent.self) + let label = Child(MultilineTextComponent.self) + + return { context in + let component = context.component + + var insets = UIEdgeInsets(top: 11.0, left: 46.0, bottom: 13.0, right: 16.0) + + let label = label.update( + component: MultilineTextComponent( + text: .plain( + NSAttributedString( + string: component.labelPrice, + font: Font.regular(17), + textColor: component.secondaryTextColor + ) + ), + maximumNumberOfLines: 1 + ), + availableSize: context.availableSize, + transition: context.transition + ) + + let title = title.update( + component: MultilineTextComponent( + text: .plain( + NSAttributedString( + string: component.title, + font: Font.regular(17), + textColor: component.primaryTextColor + ) + ), + maximumNumberOfLines: 1 + ), + availableSize: CGSize(width: context.availableSize.width - insets.left - insets.right - label.size.width, height: context.availableSize.height), + transition: context.transition + ) + + let discountSize: CGSize + if !component.discount.isEmpty { + let discount = discount.update( + component: MultilineTextComponent( + text: .plain( + NSAttributedString( + string: component.discount, + font: Font.with(size: 14.0, design: .round, weight: .semibold, traits: []), + textColor: .white + ) + ), + maximumNumberOfLines: 1 + ), + availableSize: context.availableSize, + transition: context.transition + ) + + discountSize = CGSize(width: discount.size.width + 6.0, height: 18.0) + + let discountBackground = discountBackground.update( + component: RoundedRectangle( + color: component.accentColor, + cornerRadius: 5.0 + ), + availableSize: discountSize, + transition: context.transition + ) + + let discountPosition = CGPoint(x: insets.left + title.size.width + 6.0 + discountSize.width / 2.0, y: insets.top + title.size.height / 2.0) + + context.add(discountBackground + .position(discountPosition) + ) + context.add(discount + .position(discountPosition) + ) + } else { + discountSize = CGSize(width: 0.0, height: 18.0) + } + + var spacing: CGFloat = 0.0 + var subtitleSize = CGSize() + if !component.subtitle.isEmpty { + spacing = 2.0 + + let subtitleFont = Font.regular(13) + let subtitleColor = component.secondaryTextColor + + let subtitleString = parseMarkdownIntoAttributedString( + component.subtitle, + attributes: MarkdownAttributes( + body: MarkdownAttributeSet(font: subtitleFont, textColor: subtitleColor), + bold: MarkdownAttributeSet(font: subtitleFont, textColor: subtitleColor, additionalAttributes: [NSAttributedString.Key.strikethroughStyle.rawValue: NSUnderlineStyle.single.rawValue as NSNumber]), + link: MarkdownAttributeSet(font: subtitleFont, textColor: subtitleColor), + linkAttribute: { _ in return nil } + ) + ) + + let subtitle = subtitle.update( + component: MultilineTextComponent( + text: .plain(subtitleString), + maximumNumberOfLines: 1 + ), + availableSize: CGSize(width: context.availableSize.width - insets.left - insets.right, height: context.availableSize.height), + transition: context.transition + ) + context.add(subtitle + .position(CGPoint(x: insets.left + subtitle.size.width / 2.0, y: insets.top + title.size.height + spacing + subtitle.size.height / 2.0)) + ) + subtitleSize = subtitle.size + + insets.top -= 2.0 + insets.bottom -= 2.0 + } + + let check = check.update( + component: CheckComponent( + theme: CheckComponent.Theme( + backgroundColor: component.accentColor, + strokeColor: component.checkForegroundColor, + borderColor: component.checkBorderColor, + overlayBorder: false, + hasInset: false, + hasShadow: false + ), + selected: component.selected + ), + availableSize: context.availableSize, + transition: context.transition + ) + + context.add(title + .position(CGPoint(x: insets.left + title.size.width / 2.0, y: insets.top + title.size.height / 2.0)) + ) + + let size = CGSize(width: context.availableSize.width, height: insets.top + title.size.height + spacing + subtitleSize.height + insets.bottom) + + let distance = context.availableSize.width - insets.left - insets.right - label.size.width - subtitleSize.width + + let labelY: CGFloat + if distance > 8.0 { + labelY = size.height / 2.0 + } else { + labelY = insets.top + title.size.height / 2.0 + } + + context.add(label + .position(CGPoint(x: context.availableSize.width - insets.right - label.size.width / 2.0, y: labelY)) + ) + + context.add(check + .position(CGPoint(x: 4.0 + check.size.width / 2.0, y: size.height / 2.0)) + ) + + return size + } + } +} + +private final class CheckComponent: Component { + struct Theme: Equatable { + public let backgroundColor: UIColor + public let strokeColor: UIColor + public let borderColor: UIColor + public let overlayBorder: Bool + public let hasInset: Bool + public let hasShadow: Bool + public let filledBorder: Bool + public let borderWidth: CGFloat? + + public init(backgroundColor: UIColor, strokeColor: UIColor, borderColor: UIColor, overlayBorder: Bool, hasInset: Bool, hasShadow: Bool, filledBorder: Bool = false, borderWidth: CGFloat? = nil) { + self.backgroundColor = backgroundColor + self.strokeColor = strokeColor + self.borderColor = borderColor + self.overlayBorder = overlayBorder + self.hasInset = hasInset + self.hasShadow = hasShadow + self.filledBorder = filledBorder + self.borderWidth = borderWidth + } + + var checkNodeTheme: CheckNodeTheme { + return CheckNodeTheme( + backgroundColor: self.backgroundColor, + strokeColor: self.strokeColor, + borderColor: self.borderColor, + overlayBorder: self.overlayBorder, + hasInset: self.hasInset, + hasShadow: self.hasShadow, + filledBorder: self.filledBorder, + borderWidth: self.borderWidth + ) + } + } + + let theme: Theme + let selected: Bool + + init( + theme: Theme, + selected: Bool + ) { + self.theme = theme + self.selected = selected + } + + static func ==(lhs: CheckComponent, rhs: CheckComponent) -> Bool { + if lhs.theme != rhs.theme { + return false + } + if lhs.selected != rhs.selected { + return false + } + return true + } + + final class View: UIView { + private var currentValue: CGFloat? + private var animator: DisplayLinkAnimator? + + private var checkLayer: CheckLayer { + return self.layer as! CheckLayer + } + + override class var layerClass: AnyClass { + return CheckLayer.self + } + + init() { + super.init(frame: CGRect()) + } + + required init?(coder aDecoder: NSCoder) { + preconditionFailure() + } + + + func update(component: CheckComponent, availableSize: CGSize, transition: Transition) -> CGSize { + self.checkLayer.setSelected(component.selected, animated: true) + self.checkLayer.theme = component.theme.checkNodeTheme + + return CGSize(width: 22.0, height: 22.0) + } + } + + func makeView() -> View { + return View() + } + + func update(view: View, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + return view.update(component: self, availableSize: availableSize, transition: transition) + } +} diff --git a/submodules/PremiumUI/Sources/PremiumStarComponent.swift b/submodules/PremiumUI/Sources/PremiumStarComponent.swift index 4dc409d4802..549f8e605a9 100644 --- a/submodules/PremiumUI/Sources/PremiumStarComponent.swift +++ b/submodules/PremiumUI/Sources/PremiumStarComponent.swift @@ -8,7 +8,7 @@ import GZip import AppBundle import LegacyComponents -private let sceneVersion: Int = 6 +private let sceneVersion: Int = 7 private func deg2rad(_ number: Float) -> Float { return number * .pi / 180 @@ -45,7 +45,31 @@ private func generateDiffuseTexture() -> UIImage { })! } -class PremiumStarComponent: Component { +func loadCompressedScene(name: String, version: Int) -> SCNScene? { + let resourceUrl: URL + if let url = getAppBundle().url(forResource: name, withExtension: "scn") { + resourceUrl = url + } else { + let fileName = "\(name)_\(version).scn" + let tmpUrl = URL(fileURLWithPath: NSTemporaryDirectory() + fileName) + if !FileManager.default.fileExists(atPath: tmpUrl.path) { + guard let url = getAppBundle().url(forResource: name, withExtension: ""), + let compressedData = try? Data(contentsOf: url), + let decompressedData = TGGUnzipData(compressedData, 8 * 1024 * 1024) else { + return nil + } + try? decompressedData.write(to: tmpUrl) + } + resourceUrl = tmpUrl + } + + guard let scene = try? SCNScene(url: resourceUrl, options: nil) else { + return nil + } + return scene +} + +final class PremiumStarComponent: Component { let isIntro: Bool let isVisible: Bool let hasIdleAnimations: Bool @@ -251,24 +275,7 @@ class PremiumStarComponent: Component { } private func setup() { - let resourceUrl: URL - if let url = getAppBundle().url(forResource: "star", withExtension: "scn") { - resourceUrl = url - } else { - let fileName = "star_\(sceneVersion).scn" - let tmpUrl = URL(fileURLWithPath: NSTemporaryDirectory() + fileName) - if !FileManager.default.fileExists(atPath: tmpUrl.path) { - guard let url = getAppBundle().url(forResource: "star", withExtension: ""), - let compressedData = try? Data(contentsOf: url), - let decompressedData = TGGUnzipData(compressedData, 8 * 1024 * 1024) else { - return - } - try? decompressedData.write(to: tmpUrl) - } - resourceUrl = tmpUrl - } - - guard let scene = try? SCNScene(url: resourceUrl, options: nil) else { + guard let scene = loadCompressedScene(name: "star", version: sceneVersion) else { return } diff --git a/submodules/PremiumUI/Sources/ReplaceBoostScreen.swift b/submodules/PremiumUI/Sources/ReplaceBoostScreen.swift index 973e4d3fb49..3f596f9ca95 100644 --- a/submodules/PremiumUI/Sources/ReplaceBoostScreen.swift +++ b/submodules/PremiumUI/Sources/ReplaceBoostScreen.swift @@ -477,6 +477,7 @@ public class ReplaceBoostScreen: ViewController { statusBarHeight: 0.0, navigationHeight: navigationHeight, safeInsets: UIEdgeInsets(top: layout.intrinsicInsets.top + layout.safeInsets.top, left: layout.safeInsets.left, bottom: layout.intrinsicInsets.bottom + layout.safeInsets.bottom, right: layout.safeInsets.right), + additionalInsets: layout.additionalInsets, inputHeight: layout.inputHeight ?? 0.0, metrics: layout.metrics, deviceMetrics: layout.deviceMetrics, diff --git a/submodules/PremiumUI/Sources/SwirlStarsView.swift b/submodules/PremiumUI/Sources/SwirlStarsView.swift index d7ba345bf7a..8c3ce911b09 100644 --- a/submodules/PremiumUI/Sources/SwirlStarsView.swift +++ b/submodules/PremiumUI/Sources/SwirlStarsView.swift @@ -5,6 +5,8 @@ import Display import AppBundle import SwiftSignalKit +private let sceneVersion: Int = 1 + final class SwirlStarsView: UIView, PhoneDemoDecorationView { private let sceneView: SCNView @@ -13,8 +15,8 @@ final class SwirlStarsView: UIView, PhoneDemoDecorationView { override init(frame: CGRect) { self.sceneView = SCNView(frame: CGRect(origin: .zero, size: frame.size)) self.sceneView.backgroundColor = .clear - if let url = getAppBundle().url(forResource: "swirl", withExtension: "scn") { - self.sceneView.scene = try? SCNScene(url: url, options: nil) + if let scene = loadCompressedScene(name: "swirl", version: sceneVersion) { + self.sceneView.scene = scene } self.sceneView.isUserInteractionEnabled = false self.sceneView.preferredFramesPerSecond = 60 diff --git a/submodules/SearchPeerMembers/Sources/SearchPeerMembers.swift b/submodules/SearchPeerMembers/Sources/SearchPeerMembers.swift index 8cef0161a84..5581cdb9d37 100644 --- a/submodules/SearchPeerMembers/Sources/SearchPeerMembers.swift +++ b/submodules/SearchPeerMembers/Sources/SearchPeerMembers.swift @@ -81,7 +81,7 @@ public func searchPeerMembers(context: AccountContext, peerId: EnginePeer.Id, ch return ActionDisposable { disposable.dispose() } - case .feed: + case .customChatContents: subscriber.putNext(([], true)) return ActionDisposable { diff --git a/submodules/SettingsUI/Sources/ChangePhoneNumberController.swift b/submodules/SettingsUI/Sources/ChangePhoneNumberController.swift index d2ad53873f0..17a5ce1a2a2 100644 --- a/submodules/SettingsUI/Sources/ChangePhoneNumberController.swift +++ b/submodules/SettingsUI/Sources/ChangePhoneNumberController.swift @@ -162,8 +162,26 @@ public func ChangePhoneNumberController(context: AccountContext) -> ViewControll } Queue.mainQueue().justDispatch { - controller.updateData(countryCode: AuthorizationSequenceController.defaultCountryCode(), countryName: nil, number: "") - controller.updateCountryCode() + let _ = (context.engine.data.get(TelegramEngine.EngineData.Item.Peer.Peer(id: context.account.peerId)) + |> deliverOnMainQueue).start(next: { accountPeer in + guard let accountPeer, case let .user(user) = accountPeer else { + return + } + + let initialCountryCode: Int32 + if let phone = user.phone { + if let (_, countryCode) = lookupCountryIdByNumber(phone, configuration: context.currentCountriesConfiguration.with { $0 }), let codeValue = Int32(countryCode.code) { + initialCountryCode = codeValue + } else { + initialCountryCode = AuthorizationSequenceController.defaultCountryCode() + } + } else { + initialCountryCode = AuthorizationSequenceController.defaultCountryCode() + } + controller.updateData(countryCode: initialCountryCode, countryName: nil, number: "") + controller.updateCountryCode() + }) + } return controller diff --git a/submodules/SettingsUI/Sources/ChangePhoneNumberControllerNode.swift b/submodules/SettingsUI/Sources/ChangePhoneNumberControllerNode.swift index 856b8fa1c47..e7152020851 100644 --- a/submodules/SettingsUI/Sources/ChangePhoneNumberControllerNode.swift +++ b/submodules/SettingsUI/Sources/ChangePhoneNumberControllerNode.swift @@ -3,7 +3,6 @@ import UIKit import AsyncDisplayKit import Display import TelegramCore -import CoreTelephony import TelegramPresentationData import PhoneInputNode import CountrySelectionUI @@ -211,18 +210,9 @@ final class ChangePhoneNumberControllerNode: ASDisplayNode { } } - var countryId: String? = nil - let networkInfo = CTTelephonyNetworkInfo() - if let carrier = networkInfo.serviceSubscriberCellularProviders?.values.first { - countryId = carrier.isoCountryCode - } - - if countryId == nil { - countryId = (Locale.current as NSLocale).object(forKey: .countryCode) as? String - } - + let countryId = (Locale.current as NSLocale).object(forKey: .countryCode) as? String + var countryCodeAndId: (Int32, String) = (1, "US") - if let countryId = countryId { let normalizedId = countryId.uppercased() for (code, idAndName) in countryCodeToIdAndName { diff --git a/submodules/SettingsUI/Sources/DeleteAccountPhoneItem.swift b/submodules/SettingsUI/Sources/DeleteAccountPhoneItem.swift index a64fdfdd586..0e0fe77e80e 100644 --- a/submodules/SettingsUI/Sources/DeleteAccountPhoneItem.swift +++ b/submodules/SettingsUI/Sources/DeleteAccountPhoneItem.swift @@ -10,7 +10,6 @@ import ItemListUI import PresentationDataUtils import PhoneInputNode import CountrySelectionUI -import CoreTelephony private func generateCountryButtonBackground(color: UIColor, strokeColor: UIColor) -> UIImage? { return generateImage(CGSize(width: 56, height: 44.0 + 6.0), rotatedContext: { size, context in @@ -234,18 +233,9 @@ class DeleteAccountPhoneItemNode: ListViewItemNode, ItemListItemNode { } } - var countryId: String? = nil - let networkInfo = CTTelephonyNetworkInfo() - if let carrier = networkInfo.serviceSubscriberCellularProviders?.values.first { - countryId = carrier.isoCountryCode - } - - if countryId == nil { - countryId = (Locale.current as NSLocale).object(forKey: .countryCode) as? String - } - + let countryId = (Locale.current as NSLocale).object(forKey: .countryCode) as? String + var countryCodeAndId: (Int32, String) = (1, "US") - if let countryId = countryId { let normalizedId = countryId.uppercased() for (code, idAndName) in countryCodeToIdAndName { diff --git a/submodules/SettingsUI/Sources/Search/SettingsSearchableItems.swift b/submodules/SettingsUI/Sources/Search/SettingsSearchableItems.swift index f6dc8b4ecef..1d0507719d0 100644 --- a/submodules/SettingsUI/Sources/Search/SettingsSearchableItems.swift +++ b/submodules/SettingsUI/Sources/Search/SettingsSearchableItems.swift @@ -279,13 +279,13 @@ private func premiumSearchableItems(context: AccountContext) -> [SettingsSearcha var result: [SettingsSearchableItem] = [] result.append(SettingsSearchableItem(id: .premium(0), title: strings.Settings_Premium, alternate: synonyms(strings.SettingsSearch_Synonyms_Premium), icon: icon, breadcrumbs: [], present: { context, _, present in - present(.push, PremiumIntroScreen(context: context, modal: false, source: .settings)) + present(.push, PremiumIntroScreen(context: context, source: .settings, modal: false)) })) let presentDemo: (PremiumDemoScreen.Subject, (SettingsSearchableItemPresentation, ViewController?) -> Void) -> Void = { subject, present in var replaceImpl: ((ViewController) -> Void)? let controller = PremiumDemoScreen(context: context, subject: subject, action: { - let controller = PremiumIntroScreen(context: context, modal: false, source: .settings) + let controller = PremiumIntroScreen(context: context, source: .settings, modal: false) replaceImpl?(controller) }) replaceImpl = { [weak controller] c in diff --git a/submodules/SettingsUI/Sources/Text Size/TextSizeSelectionController.swift b/submodules/SettingsUI/Sources/Text Size/TextSizeSelectionController.swift index 17af5284bfa..9bf1d44c491 100644 --- a/submodules/SettingsUI/Sources/Text Size/TextSizeSelectionController.swift +++ b/submodules/SettingsUI/Sources/Text Size/TextSizeSelectionController.swift @@ -227,6 +227,7 @@ private final class TextSizeSelectionControllerNode: ASDisplayNode, UIScrollView }, openChatFolderUpdates: {}, hideChatFolderUpdates: { }, openStories: { _, _ in }, dismissNotice: { _ in + }, editPeer: { _ in }) let chatListPresentationData = ChatListPresentationData(theme: self.presentationData.theme, fontSize: self.presentationData.listsFontSize, strings: self.presentationData.strings, dateTimeFormat: self.presentationData.dateTimeFormat, nameSortOrder: self.presentationData.nameSortOrder, nameDisplayOrder: self.presentationData.nameDisplayOrder, disableAnimations: true) diff --git a/submodules/SettingsUI/Sources/Themes/ThemePreviewControllerNode.swift b/submodules/SettingsUI/Sources/Themes/ThemePreviewControllerNode.swift index d096a99ee1d..f9201ecc1c4 100644 --- a/submodules/SettingsUI/Sources/Themes/ThemePreviewControllerNode.swift +++ b/submodules/SettingsUI/Sources/Themes/ThemePreviewControllerNode.swift @@ -376,6 +376,7 @@ final class ThemePreviewControllerNode: ASDisplayNode, UIScrollViewDelegate { }, openChatFolderUpdates: {}, hideChatFolderUpdates: { }, openStories: { _, _ in }, dismissNotice: { _ in + }, editPeer: { _ in }) func makeChatListItem( diff --git a/submodules/ShareController/BUILD b/submodules/ShareController/BUILD index cbb874ee479..f94e1882f2d 100644 --- a/submodules/ShareController/BUILD +++ b/submodules/ShareController/BUILD @@ -44,7 +44,6 @@ swift_library( "//submodules/TelegramUI/Components/AnimationCache", "//submodules/TelegramUI/Components/MultiAnimationRenderer", "//submodules/UndoUI", - "//submodules/PremiumUI", ], visibility = [ "//visibility:public", diff --git a/submodules/ShareController/Sources/ShareController.swift b/submodules/ShareController/Sources/ShareController.swift index 6206a025a45..21d03d474b3 100644 --- a/submodules/ShareController/Sources/ShareController.swift +++ b/submodules/ShareController/Sources/ShareController.swift @@ -23,7 +23,6 @@ import TelegramIntents import AnimationCache import MultiAnimationRenderer import ObjectiveC -import PremiumUI import UndoUI private var ObjCKey_DeinitWatcher: Int? @@ -1273,7 +1272,8 @@ public final class ShareController: ViewController { } if case .undo = action { self.controllerNode.cancel?() - let premiumController = PremiumIntroScreen(context: context.context, source: .settings) + + let premiumController = context.context.sharedContext.makePremiumIntroController(context: context.context, source: .settings, forceDark: false, dismissed: nil) parentNavigationController.pushViewController(premiumController) } return false diff --git a/submodules/StatisticsUI/Sources/ChannelStatsController.swift b/submodules/StatisticsUI/Sources/ChannelStatsController.swift index 1e28116b269..e9ecb15ee60 100644 --- a/submodules/StatisticsUI/Sources/ChannelStatsController.swift +++ b/submodules/StatisticsUI/Sources/ChannelStatsController.swift @@ -1311,9 +1311,13 @@ public func channelStatsController(context: AccountContext, updatedPresentationD var headerItem: BoostHeaderItem? var leftNavigationButton: ItemListNavigationButton? var boostsOnly = false - if isGroup, section == .boosts { + if section == .boosts { title = .text("") - headerItem = BoostHeaderItem(context: context, theme: presentationData.theme, strings: presentationData.strings, status: boostData, title: presentationData.strings.GroupBoost_Title, text: presentationData.strings.GroupBoost_Info, openBoost: { + + let headerTitle = isGroup ? presentationData.strings.GroupBoost_Title : presentationData.strings.ChannelBoost_Title + let headerText = isGroup ? presentationData.strings.GroupBoost_Info : presentationData.strings.ChannelBoost_Info + + headerItem = BoostHeaderItem(context: context, theme: presentationData.theme, strings: presentationData.strings, status: boostData, title: headerTitle, text: headerText, openBoost: { openBoostImpl?(false) }, createGiveaway: { arguments.openGifts() diff --git a/submodules/StickerPeekUI/BUILD b/submodules/StickerPeekUI/BUILD index 2772c1830cd..ec677548331 100644 --- a/submodules/StickerPeekUI/BUILD +++ b/submodules/StickerPeekUI/BUILD @@ -28,7 +28,6 @@ swift_library( "//submodules/ShimmerEffect:ShimmerEffect", "//submodules/ContextUI:ContextUI", "//submodules/SolidRoundedButtonNode:SolidRoundedButtonNode", - "//submodules/PremiumUI:PremiumUI", ], visibility = [ "//visibility:public", diff --git a/submodules/TelegramApi/Sources/Api0.swift b/submodules/TelegramApi/Sources/Api0.swift index 4188aaf0c25..1a147326a55 100644 --- a/submodules/TelegramApi/Sources/Api0.swift +++ b/submodules/TelegramApi/Sources/Api0.swift @@ -12,6 +12,7 @@ public enum Api { public enum phone {} public enum photos {} public enum premium {} + public enum smsjobs {} public enum stats {} public enum stickers {} public enum storage {} @@ -34,6 +35,7 @@ public enum Api { public enum phone {} public enum photos {} public enum premium {} + public enum smsjobs {} public enum stats {} public enum stickers {} public enum stories {} @@ -78,6 +80,7 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[706514033] = { return Api.Boost.parse_boost($0) } dict[-1778593322] = { return Api.BotApp.parse_botApp($0) } dict[1571189943] = { return Api.BotApp.parse_botAppNotModified($0) } + dict[-1989921868] = { return Api.BotBusinessConnection.parse_botBusinessConnection($0) } dict[-1032140601] = { return Api.BotCommand.parse_botCommand($0) } dict[-1180016534] = { return Api.BotCommandScope.parse_botCommandScopeChatAdmins($0) } dict[1877059713] = { return Api.BotCommandScope.parse_botCommandScopeChats($0) } @@ -99,6 +102,15 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[-944407322] = { return Api.BotMenuButton.parse_botMenuButton($0) } dict[1113113093] = { return Api.BotMenuButton.parse_botMenuButtonCommands($0) } dict[1966318984] = { return Api.BotMenuButton.parse_botMenuButtonDefault($0) } + dict[-283809188] = { return Api.BusinessAwayMessage.parse_businessAwayMessage($0) } + dict[-910564679] = { return Api.BusinessAwayMessageSchedule.parse_businessAwayMessageScheduleAlways($0) } + dict[-867328308] = { return Api.BusinessAwayMessageSchedule.parse_businessAwayMessageScheduleCustom($0) } + dict[-1007487743] = { return Api.BusinessAwayMessageSchedule.parse_businessAwayMessageScheduleOutsideWorkHours($0) } + dict[-451302485] = { return Api.BusinessGreetingMessage.parse_businessGreetingMessage($0) } + dict[-1403249929] = { return Api.BusinessLocation.parse_businessLocation($0) } + dict[554733559] = { return Api.BusinessRecipients.parse_businessRecipients($0) } + dict[302717625] = { return Api.BusinessWeeklyOpen.parse_businessWeeklyOpen($0) } + dict[-1936543592] = { return Api.BusinessWorkHours.parse_businessWorkHours($0) } dict[1462101002] = { return Api.CdnConfig.parse_cdnConfig($0) } dict[-914167110] = { return Api.CdnPublicKey.parse_cdnPublicKey($0) } dict[531458253] = { return Api.ChannelAdminLogEvent.parse_channelAdminLogEvent($0) } @@ -196,6 +208,7 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[1713193015] = { return Api.ChatReactions.parse_chatReactionsSome($0) } dict[-1390068360] = { return Api.CodeSettings.parse_codeSettings($0) } dict[-870702050] = { return Api.Config.parse_config($0) } + dict[-404121113] = { return Api.ConnectedBot.parse_connectedBot($0) } dict[341499403] = { return Api.Contact.parse_contact($0) } dict[383348795] = { return Api.ContactStatus.parse_contactStatus($0) } dict[2104790276] = { return Api.DataJSON.parse_dataJSON($0) } @@ -203,8 +216,8 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[1135897376] = { return Api.DefaultHistoryTTL.parse_defaultHistoryTTL($0) } dict[-712374074] = { return Api.Dialog.parse_dialog($0) } dict[1908216652] = { return Api.Dialog.parse_dialogFolder($0) } - dict[1949890536] = { return Api.DialogFilter.parse_dialogFilter($0) } - dict[-699792216] = { return Api.DialogFilter.parse_dialogFilterChatlist($0) } + dict[1605718587] = { return Api.DialogFilter.parse_dialogFilter($0) } + dict[-1612542300] = { return Api.DialogFilter.parse_dialogFilterChatlist($0) } dict[909284270] = { return Api.DialogFilter.parse_dialogFilterDefault($0) } dict[2004110666] = { return Api.DialogFilterSuggested.parse_dialogFilterSuggested($0) } dict[-445792507] = { return Api.DialogPeer.parse_dialogPeer($0) } @@ -295,6 +308,9 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[-459324] = { return Api.InputBotInlineResult.parse_inputBotInlineResultDocument($0) } dict[1336154098] = { return Api.InputBotInlineResult.parse_inputBotInlineResultGame($0) } dict[-1462213465] = { return Api.InputBotInlineResult.parse_inputBotInlineResultPhoto($0) } + dict[-2094959136] = { return Api.InputBusinessAwayMessage.parse_inputBusinessAwayMessage($0) } + dict[26528571] = { return Api.InputBusinessGreetingMessage.parse_inputBusinessGreetingMessage($0) } + dict[1871393450] = { return Api.InputBusinessRecipients.parse_inputBusinessRecipients($0) } dict[-212145112] = { return Api.InputChannel.parse_inputChannel($0) } dict[-292807034] = { return Api.InputChannel.parse_inputChannelEmpty($0) } dict[1536380829] = { return Api.InputChannel.parse_inputChannelFromMessage($0) } @@ -396,6 +412,8 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[-380694650] = { return Api.InputPrivacyRule.parse_inputPrivacyValueDisallowChatParticipants($0) } dict[195371015] = { return Api.InputPrivacyRule.parse_inputPrivacyValueDisallowContacts($0) } dict[-1877932953] = { return Api.InputPrivacyRule.parse_inputPrivacyValueDisallowUsers($0) } + dict[609840449] = { return Api.InputQuickReplyShortcut.parse_inputQuickReplyShortcut($0) } + dict[18418929] = { return Api.InputQuickReplyShortcut.parse_inputQuickReplyShortcutId($0) } dict[583071445] = { return Api.InputReplyTo.parse_inputReplyToMessage($0) } dict[1484862010] = { return Api.InputReplyTo.parse_inputReplyToStory($0) } dict[1399317950] = { return Api.InputSecureFile.parse_inputSecureFile($0) } @@ -473,7 +491,7 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[340088945] = { return Api.MediaArea.parse_mediaAreaSuggestedReaction($0) } dict[-1098720356] = { return Api.MediaArea.parse_mediaAreaVenue($0) } dict[64088654] = { return Api.MediaAreaCoordinates.parse_mediaAreaCoordinates($0) } - dict[508332649] = { return Api.Message.parse_message($0) } + dict[-1502839044] = { return Api.Message.parse_message($0) } dict[-1868117372] = { return Api.Message.parse_messageEmpty($0) } dict[721967202] = { return Api.Message.parse_messageService($0) } dict[-872240531] = { return Api.MessageAction.parse_messageActionBoostApply($0) } @@ -705,6 +723,7 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[-463335103] = { return Api.PrivacyRule.parse_privacyValueDisallowUsers($0) } dict[32685898] = { return Api.PublicForward.parse_publicForwardMessage($0) } dict[-302797360] = { return Api.PublicForward.parse_publicForwardStory($0) } + dict[110563371] = { return Api.QuickReply.parse_quickReply($0) } dict[-1992950669] = { return Api.Reaction.parse_reactionCustomEmoji($0) } dict[455247544] = { return Api.Reaction.parse_reactionEmoji($0) } dict[2046153753] = { return Api.Reaction.parse_reactionEmpty($0) } @@ -812,6 +831,7 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[-651419003] = { return Api.SendMessageAction.parse_speakingInGroupCallAction($0) } dict[-1239335713] = { return Api.ShippingOption.parse_shippingOption($0) } dict[-2010155333] = { return Api.SimpleWebViewResult.parse_simpleWebViewResultUrl($0) } + dict[-425595208] = { return Api.SmsJob.parse_smsJob($0) } dict[-313293833] = { return Api.SponsoredMessage.parse_sponsoredMessage($0) } dict[1035529315] = { return Api.SponsoredWebPage.parse_sponsoredWebPage($0) } dict[-884757282] = { return Api.StatsAbsValueAndPrev.parse_statsAbsValueAndPrev($0) } @@ -846,6 +866,7 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[1964978502] = { return Api.TextWithEntities.parse_textWithEntities($0) } dict[-1609668650] = { return Api.Theme.parse_theme($0) } dict[-94849324] = { return Api.ThemeSettings.parse_themeSettings($0) } + dict[-7173643] = { return Api.Timezone.parse_timezone($0) } dict[-305282981] = { return Api.TopPeer.parse_topPeer($0) } dict[344356834] = { return Api.TopPeerCategory.parse_topPeerCategoryBotsInline($0) } dict[-1419371685] = { return Api.TopPeerCategory.parse_topPeerCategoryBotsPM($0) } @@ -858,15 +879,19 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[-75283823] = { return Api.TopPeerCategoryPeers.parse_topPeerCategoryPeers($0) } dict[397910539] = { return Api.Update.parse_updateAttachMenuBots($0) } dict[-335171433] = { return Api.Update.parse_updateAutoSaveSettings($0) } + dict[-1964652166] = { return Api.Update.parse_updateBotBusinessConnect($0) } dict[-1177566067] = { return Api.Update.parse_updateBotCallbackQuery($0) } dict[-1873947492] = { return Api.Update.parse_updateBotChatBoost($0) } dict[299870598] = { return Api.Update.parse_updateBotChatInviteRequester($0) } dict[1299263278] = { return Api.Update.parse_updateBotCommands($0) } + dict[-1590796039] = { return Api.Update.parse_updateBotDeleteBusinessMessage($0) } + dict[1420915171] = { return Api.Update.parse_updateBotEditBusinessMessage($0) } dict[1232025500] = { return Api.Update.parse_updateBotInlineQuery($0) } dict[317794823] = { return Api.Update.parse_updateBotInlineSend($0) } dict[347625491] = { return Api.Update.parse_updateBotMenuButton($0) } dict[-1407069234] = { return Api.Update.parse_updateBotMessageReaction($0) } dict[164329305] = { return Api.Update.parse_updateBotMessageReactions($0) } + dict[-2142069794] = { return Api.Update.parse_updateBotNewBusinessMessage($0) } dict[-1934976362] = { return Api.Update.parse_updateBotPrecheckoutQuery($0) } dict[-1246823043] = { return Api.Update.parse_updateBotShippingQuery($0) } dict[-997782967] = { return Api.Update.parse_updateBotStopped($0) } @@ -897,6 +922,8 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[-1906403213] = { return Api.Update.parse_updateDcOptions($0) } dict[-1020437742] = { return Api.Update.parse_updateDeleteChannelMessages($0) } dict[-1576161051] = { return Api.Update.parse_updateDeleteMessages($0) } + dict[1407644140] = { return Api.Update.parse_updateDeleteQuickReply($0) } + dict[1450174413] = { return Api.Update.parse_updateDeleteQuickReplyMessages($0) } dict[-1870238482] = { return Api.Update.parse_updateDeleteScheduledMessages($0) } dict[654302845] = { return Api.Update.parse_updateDialogFilter($0) } dict[-1512627963] = { return Api.Update.parse_updateDialogFilterOrder($0) } @@ -930,6 +957,7 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[1656358105] = { return Api.Update.parse_updateNewChannelMessage($0) } dict[314359194] = { return Api.Update.parse_updateNewEncryptedMessage($0) } dict[522914557] = { return Api.Update.parse_updateNewMessage($0) } + dict[-180508905] = { return Api.Update.parse_updateNewQuickReply($0) } dict[967122427] = { return Api.Update.parse_updateNewScheduledMessage($0) } dict[1753886890] = { return Api.Update.parse_updateNewStickerSet($0) } dict[-1094555409] = { return Api.Update.parse_updateNotifySettings($0) } @@ -947,6 +975,8 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[1751942566] = { return Api.Update.parse_updatePinnedSavedDialogs($0) } dict[-298113238] = { return Api.Update.parse_updatePrivacy($0) } dict[861169551] = { return Api.Update.parse_updatePtsChanged($0) } + dict[-112784718] = { return Api.Update.parse_updateQuickReplies($0) } + dict[1040518415] = { return Api.Update.parse_updateQuickReplyMessage($0) } dict[-693004986] = { return Api.Update.parse_updateReadChannelDiscussionInbox($0) } dict[1767677564] = { return Api.Update.parse_updateReadChannelDiscussionOutbox($0) } dict[-1842450928] = { return Api.Update.parse_updateReadChannelInbox($0) } @@ -966,6 +996,7 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[1960361625] = { return Api.Update.parse_updateSavedRingtones($0) } dict[2103604867] = { return Api.Update.parse_updateSentStoryReaction($0) } dict[-337352679] = { return Api.Update.parse_updateServiceNotification($0) } + dict[-245208620] = { return Api.Update.parse_updateSmsJob($0) } dict[834816008] = { return Api.Update.parse_updateStickerSets($0) } dict[196268545] = { return Api.Update.parse_updateStickerSetsOrder($0) } dict[738741697] = { return Api.Update.parse_updateStoriesStealthMode($0) } @@ -993,7 +1024,7 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[-1831650802] = { return Api.UrlAuthResult.parse_urlAuthResultRequest($0) } dict[559694904] = { return Api.User.parse_user($0) } dict[-742634630] = { return Api.User.parse_userEmpty($0) } - dict[-1179571092] = { return Api.UserFull.parse_userFull($0) } + dict[587153029] = { return Api.UserFull.parse_userFull($0) } dict[-2100168954] = { return Api.UserProfilePhoto.parse_userProfilePhoto($0) } dict[1326562017] = { return Api.UserProfilePhoto.parse_userProfilePhotoEmpty($0) } dict[164646985] = { return Api.UserStatus.parse_userStatusEmpty($0) } @@ -1024,6 +1055,7 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[1275039392] = { return Api.account.Authorizations.parse_authorizations($0) } dict[1674235686] = { return Api.account.AutoDownloadSettings.parse_autoDownloadSettings($0) } dict[1279133341] = { return Api.account.AutoSaveSettings.parse_autoSaveSettings($0) } + dict[400029819] = { return Api.account.ConnectedBots.parse_connectedBots($0) } dict[1474462241] = { return Api.account.ContentSettings.parse_contentSettings($0) } dict[731303195] = { return Api.account.EmailVerified.parse_emailVerified($0) } dict[-507835039] = { return Api.account.EmailVerified.parse_emailVerifiedLogin($0) } @@ -1120,6 +1152,8 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[2013922064] = { return Api.help.TermsOfService.parse_termsOfService($0) } dict[686618977] = { return Api.help.TermsOfServiceUpdate.parse_termsOfServiceUpdate($0) } dict[-483352705] = { return Api.help.TermsOfServiceUpdate.parse_termsOfServiceUpdateEmpty($0) } + dict[2071260529] = { return Api.help.TimezonesList.parse_timezonesList($0) } + dict[-1761146676] = { return Api.help.TimezonesList.parse_timezonesListNotModified($0) } dict[32192344] = { return Api.help.UserInfo.parse_userInfo($0) } dict[-206688531] = { return Api.help.UserInfo.parse_userInfoEmpty($0) } dict[-275956116] = { return Api.messages.AffectedFoundMessages.parse_affectedFoundMessages($0) } @@ -1141,6 +1175,7 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[-1571952873] = { return Api.messages.CheckedHistoryImportPeer.parse_checkedHistoryImportPeer($0) } dict[740433629] = { return Api.messages.DhConfig.parse_dhConfig($0) } dict[-1058912715] = { return Api.messages.DhConfig.parse_dhConfigNotModified($0) } + dict[718878489] = { return Api.messages.DialogFilters.parse_dialogFilters($0) } dict[364538944] = { return Api.messages.Dialogs.parse_dialogs($0) } dict[-253500010] = { return Api.messages.Dialogs.parse_dialogsNotModified($0) } dict[1910543603] = { return Api.messages.Dialogs.parse_dialogsSlice($0) } @@ -1170,6 +1205,8 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[978610270] = { return Api.messages.Messages.parse_messagesSlice($0) } dict[863093588] = { return Api.messages.PeerDialogs.parse_peerDialogs($0) } dict[1753266509] = { return Api.messages.PeerSettings.parse_peerSettings($0) } + dict[-963811691] = { return Api.messages.QuickReplies.parse_quickReplies($0) } + dict[1603398491] = { return Api.messages.QuickReplies.parse_quickRepliesNotModified($0) } dict[-352454890] = { return Api.messages.Reactions.parse_reactions($0) } dict[-1334846497] = { return Api.messages.Reactions.parse_reactionsNotModified($0) } dict[-1999405994] = { return Api.messages.RecentStickers.parse_recentStickers($0) } @@ -1222,6 +1259,8 @@ fileprivate let parsers: [Int32 : (BufferReader) -> Any?] = { dict[-2030542532] = { return Api.premium.BoostsList.parse_boostsList($0) } dict[1230586490] = { return Api.premium.BoostsStatus.parse_boostsStatus($0) } dict[-1696454430] = { return Api.premium.MyBoosts.parse_myBoosts($0) } + dict[-594852657] = { return Api.smsjobs.EligibilityToJoin.parse_eligibleToJoin($0) } + dict[720277905] = { return Api.smsjobs.Status.parse_status($0) } dict[963421692] = { return Api.stats.BroadcastStats.parse_broadcastStats($0) } dict[-276825834] = { return Api.stats.MegagroupStats.parse_megagroupStats($0) } dict[2145983508] = { return Api.stats.MessageStats.parse_messageStats($0) } @@ -1354,6 +1393,8 @@ public extension Api { _1.serialize(buffer, boxed) case let _1 as Api.BotApp: _1.serialize(buffer, boxed) + case let _1 as Api.BotBusinessConnection: + _1.serialize(buffer, boxed) case let _1 as Api.BotCommand: _1.serialize(buffer, boxed) case let _1 as Api.BotCommandScope: @@ -1366,6 +1407,20 @@ public extension Api { _1.serialize(buffer, boxed) case let _1 as Api.BotMenuButton: _1.serialize(buffer, boxed) + case let _1 as Api.BusinessAwayMessage: + _1.serialize(buffer, boxed) + case let _1 as Api.BusinessAwayMessageSchedule: + _1.serialize(buffer, boxed) + case let _1 as Api.BusinessGreetingMessage: + _1.serialize(buffer, boxed) + case let _1 as Api.BusinessLocation: + _1.serialize(buffer, boxed) + case let _1 as Api.BusinessRecipients: + _1.serialize(buffer, boxed) + case let _1 as Api.BusinessWeeklyOpen: + _1.serialize(buffer, boxed) + case let _1 as Api.BusinessWorkHours: + _1.serialize(buffer, boxed) case let _1 as Api.CdnConfig: _1.serialize(buffer, boxed) case let _1 as Api.CdnPublicKey: @@ -1412,6 +1467,8 @@ public extension Api { _1.serialize(buffer, boxed) case let _1 as Api.Config: _1.serialize(buffer, boxed) + case let _1 as Api.ConnectedBot: + _1.serialize(buffer, boxed) case let _1 as Api.Contact: _1.serialize(buffer, boxed) case let _1 as Api.ContactStatus: @@ -1514,6 +1571,12 @@ public extension Api { _1.serialize(buffer, boxed) case let _1 as Api.InputBotInlineResult: _1.serialize(buffer, boxed) + case let _1 as Api.InputBusinessAwayMessage: + _1.serialize(buffer, boxed) + case let _1 as Api.InputBusinessGreetingMessage: + _1.serialize(buffer, boxed) + case let _1 as Api.InputBusinessRecipients: + _1.serialize(buffer, boxed) case let _1 as Api.InputChannel: _1.serialize(buffer, boxed) case let _1 as Api.InputChatPhoto: @@ -1568,6 +1631,8 @@ public extension Api { _1.serialize(buffer, boxed) case let _1 as Api.InputPrivacyRule: _1.serialize(buffer, boxed) + case let _1 as Api.InputQuickReplyShortcut: + _1.serialize(buffer, boxed) case let _1 as Api.InputReplyTo: _1.serialize(buffer, boxed) case let _1 as Api.InputSecureFile: @@ -1738,6 +1803,8 @@ public extension Api { _1.serialize(buffer, boxed) case let _1 as Api.PublicForward: _1.serialize(buffer, boxed) + case let _1 as Api.QuickReply: + _1.serialize(buffer, boxed) case let _1 as Api.Reaction: _1.serialize(buffer, boxed) case let _1 as Api.ReactionCount: @@ -1798,6 +1865,8 @@ public extension Api { _1.serialize(buffer, boxed) case let _1 as Api.SimpleWebViewResult: _1.serialize(buffer, boxed) + case let _1 as Api.SmsJob: + _1.serialize(buffer, boxed) case let _1 as Api.SponsoredMessage: _1.serialize(buffer, boxed) case let _1 as Api.SponsoredWebPage: @@ -1844,6 +1913,8 @@ public extension Api { _1.serialize(buffer, boxed) case let _1 as Api.ThemeSettings: _1.serialize(buffer, boxed) + case let _1 as Api.Timezone: + _1.serialize(buffer, boxed) case let _1 as Api.TopPeer: _1.serialize(buffer, boxed) case let _1 as Api.TopPeerCategory: @@ -1892,6 +1963,8 @@ public extension Api { _1.serialize(buffer, boxed) case let _1 as Api.account.AutoSaveSettings: _1.serialize(buffer, boxed) + case let _1 as Api.account.ConnectedBots: + _1.serialize(buffer, boxed) case let _1 as Api.account.ContentSettings: _1.serialize(buffer, boxed) case let _1 as Api.account.EmailVerified: @@ -2006,6 +2079,8 @@ public extension Api { _1.serialize(buffer, boxed) case let _1 as Api.help.TermsOfServiceUpdate: _1.serialize(buffer, boxed) + case let _1 as Api.help.TimezonesList: + _1.serialize(buffer, boxed) case let _1 as Api.help.UserInfo: _1.serialize(buffer, boxed) case let _1 as Api.messages.AffectedFoundMessages: @@ -2038,6 +2113,8 @@ public extension Api { _1.serialize(buffer, boxed) case let _1 as Api.messages.DhConfig: _1.serialize(buffer, boxed) + case let _1 as Api.messages.DialogFilters: + _1.serialize(buffer, boxed) case let _1 as Api.messages.Dialogs: _1.serialize(buffer, boxed) case let _1 as Api.messages.DiscussionMessage: @@ -2076,6 +2153,8 @@ public extension Api { _1.serialize(buffer, boxed) case let _1 as Api.messages.PeerSettings: _1.serialize(buffer, boxed) + case let _1 as Api.messages.QuickReplies: + _1.serialize(buffer, boxed) case let _1 as Api.messages.Reactions: _1.serialize(buffer, boxed) case let _1 as Api.messages.RecentStickers: @@ -2152,6 +2231,10 @@ public extension Api { _1.serialize(buffer, boxed) case let _1 as Api.premium.MyBoosts: _1.serialize(buffer, boxed) + case let _1 as Api.smsjobs.EligibilityToJoin: + _1.serialize(buffer, boxed) + case let _1 as Api.smsjobs.Status: + _1.serialize(buffer, boxed) case let _1 as Api.stats.BroadcastStats: _1.serialize(buffer, boxed) case let _1 as Api.stats.MegagroupStats: diff --git a/submodules/TelegramApi/Sources/Api1.swift b/submodules/TelegramApi/Sources/Api1.swift index f223f6a5e62..106159d9de9 100644 --- a/submodules/TelegramApi/Sources/Api1.swift +++ b/submodules/TelegramApi/Sources/Api1.swift @@ -1040,6 +1040,58 @@ public extension Api { } } +public extension Api { + enum BotBusinessConnection: TypeConstructorDescription { + case botBusinessConnection(flags: Int32, connectionId: String, userId: Int64, dcId: Int32, date: Int32) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .botBusinessConnection(let flags, let connectionId, let userId, let dcId, let date): + if boxed { + buffer.appendInt32(-1989921868) + } + serializeInt32(flags, buffer: buffer, boxed: false) + serializeString(connectionId, buffer: buffer, boxed: false) + serializeInt64(userId, buffer: buffer, boxed: false) + serializeInt32(dcId, buffer: buffer, boxed: false) + serializeInt32(date, buffer: buffer, boxed: false) + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .botBusinessConnection(let flags, let connectionId, let userId, let dcId, let date): + return ("botBusinessConnection", [("flags", flags as Any), ("connectionId", connectionId as Any), ("userId", userId as Any), ("dcId", dcId as Any), ("date", date as Any)]) + } + } + + public static func parse_botBusinessConnection(_ reader: BufferReader) -> BotBusinessConnection? { + var _1: Int32? + _1 = reader.readInt32() + var _2: String? + _2 = parseString(reader) + var _3: Int64? + _3 = reader.readInt64() + var _4: Int32? + _4 = reader.readInt32() + var _5: Int32? + _5 = reader.readInt32() + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + let _c4 = _4 != nil + let _c5 = _5 != nil + if _c1 && _c2 && _c3 && _c4 && _c5 { + return Api.BotBusinessConnection.botBusinessConnection(flags: _1!, connectionId: _2!, userId: _3!, dcId: _4!, date: _5!) + } + else { + return nil + } + } + + } +} public extension Api { enum BotCommand: TypeConstructorDescription { case botCommand(command: String, description: String) diff --git a/submodules/TelegramApi/Sources/Api10.swift b/submodules/TelegramApi/Sources/Api10.swift index 7249151a229..4fe4c26f44a 100644 --- a/submodules/TelegramApi/Sources/Api10.swift +++ b/submodules/TelegramApi/Sources/Api10.swift @@ -1,3 +1,59 @@ +public extension Api { + enum InputQuickReplyShortcut: TypeConstructorDescription { + case inputQuickReplyShortcut(shortcut: String) + case inputQuickReplyShortcutId(shortcutId: Int32) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .inputQuickReplyShortcut(let shortcut): + if boxed { + buffer.appendInt32(609840449) + } + serializeString(shortcut, buffer: buffer, boxed: false) + break + case .inputQuickReplyShortcutId(let shortcutId): + if boxed { + buffer.appendInt32(18418929) + } + serializeInt32(shortcutId, buffer: buffer, boxed: false) + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .inputQuickReplyShortcut(let shortcut): + return ("inputQuickReplyShortcut", [("shortcut", shortcut as Any)]) + case .inputQuickReplyShortcutId(let shortcutId): + return ("inputQuickReplyShortcutId", [("shortcutId", shortcutId as Any)]) + } + } + + public static func parse_inputQuickReplyShortcut(_ reader: BufferReader) -> InputQuickReplyShortcut? { + var _1: String? + _1 = parseString(reader) + let _c1 = _1 != nil + if _c1 { + return Api.InputQuickReplyShortcut.inputQuickReplyShortcut(shortcut: _1!) + } + else { + return nil + } + } + public static func parse_inputQuickReplyShortcutId(_ reader: BufferReader) -> InputQuickReplyShortcut? { + var _1: Int32? + _1 = reader.readInt32() + let _c1 = _1 != nil + if _c1 { + return Api.InputQuickReplyShortcut.inputQuickReplyShortcutId(shortcutId: _1!) + } + else { + return nil + } + } + + } +} public extension Api { indirect enum InputReplyTo: TypeConstructorDescription { case inputReplyToMessage(flags: Int32, replyToMsgId: Int32, topMsgId: Int32?, replyToPeerId: Api.InputPeer?, quoteText: String?, quoteEntities: [Api.MessageEntity]?, quoteOffset: Int32?) @@ -1014,83 +1070,3 @@ public extension Api { } } -public extension Api { - enum InputWallPaper: TypeConstructorDescription { - case inputWallPaper(id: Int64, accessHash: Int64) - case inputWallPaperNoFile(id: Int64) - case inputWallPaperSlug(slug: String) - - public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { - switch self { - case .inputWallPaper(let id, let accessHash): - if boxed { - buffer.appendInt32(-433014407) - } - serializeInt64(id, buffer: buffer, boxed: false) - serializeInt64(accessHash, buffer: buffer, boxed: false) - break - case .inputWallPaperNoFile(let id): - if boxed { - buffer.appendInt32(-1770371538) - } - serializeInt64(id, buffer: buffer, boxed: false) - break - case .inputWallPaperSlug(let slug): - if boxed { - buffer.appendInt32(1913199744) - } - serializeString(slug, buffer: buffer, boxed: false) - break - } - } - - public func descriptionFields() -> (String, [(String, Any)]) { - switch self { - case .inputWallPaper(let id, let accessHash): - return ("inputWallPaper", [("id", id as Any), ("accessHash", accessHash as Any)]) - case .inputWallPaperNoFile(let id): - return ("inputWallPaperNoFile", [("id", id as Any)]) - case .inputWallPaperSlug(let slug): - return ("inputWallPaperSlug", [("slug", slug as Any)]) - } - } - - public static func parse_inputWallPaper(_ reader: BufferReader) -> InputWallPaper? { - var _1: Int64? - _1 = reader.readInt64() - var _2: Int64? - _2 = reader.readInt64() - let _c1 = _1 != nil - let _c2 = _2 != nil - if _c1 && _c2 { - return Api.InputWallPaper.inputWallPaper(id: _1!, accessHash: _2!) - } - else { - return nil - } - } - public static func parse_inputWallPaperNoFile(_ reader: BufferReader) -> InputWallPaper? { - var _1: Int64? - _1 = reader.readInt64() - let _c1 = _1 != nil - if _c1 { - return Api.InputWallPaper.inputWallPaperNoFile(id: _1!) - } - else { - return nil - } - } - public static func parse_inputWallPaperSlug(_ reader: BufferReader) -> InputWallPaper? { - var _1: String? - _1 = parseString(reader) - let _c1 = _1 != nil - if _c1 { - return Api.InputWallPaper.inputWallPaperSlug(slug: _1!) - } - else { - return nil - } - } - - } -} diff --git a/submodules/TelegramApi/Sources/Api11.swift b/submodules/TelegramApi/Sources/Api11.swift index 666ea7ff21c..89037ef849d 100644 --- a/submodules/TelegramApi/Sources/Api11.swift +++ b/submodules/TelegramApi/Sources/Api11.swift @@ -1,3 +1,83 @@ +public extension Api { + enum InputWallPaper: TypeConstructorDescription { + case inputWallPaper(id: Int64, accessHash: Int64) + case inputWallPaperNoFile(id: Int64) + case inputWallPaperSlug(slug: String) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .inputWallPaper(let id, let accessHash): + if boxed { + buffer.appendInt32(-433014407) + } + serializeInt64(id, buffer: buffer, boxed: false) + serializeInt64(accessHash, buffer: buffer, boxed: false) + break + case .inputWallPaperNoFile(let id): + if boxed { + buffer.appendInt32(-1770371538) + } + serializeInt64(id, buffer: buffer, boxed: false) + break + case .inputWallPaperSlug(let slug): + if boxed { + buffer.appendInt32(1913199744) + } + serializeString(slug, buffer: buffer, boxed: false) + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .inputWallPaper(let id, let accessHash): + return ("inputWallPaper", [("id", id as Any), ("accessHash", accessHash as Any)]) + case .inputWallPaperNoFile(let id): + return ("inputWallPaperNoFile", [("id", id as Any)]) + case .inputWallPaperSlug(let slug): + return ("inputWallPaperSlug", [("slug", slug as Any)]) + } + } + + public static func parse_inputWallPaper(_ reader: BufferReader) -> InputWallPaper? { + var _1: Int64? + _1 = reader.readInt64() + var _2: Int64? + _2 = reader.readInt64() + let _c1 = _1 != nil + let _c2 = _2 != nil + if _c1 && _c2 { + return Api.InputWallPaper.inputWallPaper(id: _1!, accessHash: _2!) + } + else { + return nil + } + } + public static func parse_inputWallPaperNoFile(_ reader: BufferReader) -> InputWallPaper? { + var _1: Int64? + _1 = reader.readInt64() + let _c1 = _1 != nil + if _c1 { + return Api.InputWallPaper.inputWallPaperNoFile(id: _1!) + } + else { + return nil + } + } + public static func parse_inputWallPaperSlug(_ reader: BufferReader) -> InputWallPaper? { + var _1: String? + _1 = parseString(reader) + let _c1 = _1 != nil + if _c1 { + return Api.InputWallPaper.inputWallPaperSlug(slug: _1!) + } + else { + return nil + } + } + + } +} public extension Api { enum InputWebDocument: TypeConstructorDescription { case inputWebDocument(url: String, size: Int32, mimeType: String, attributes: [Api.DocumentAttribute]) @@ -904,165 +984,3 @@ public extension Api { } } -public extension Api { - enum LabeledPrice: TypeConstructorDescription { - case labeledPrice(label: String, amount: Int64) - - public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { - switch self { - case .labeledPrice(let label, let amount): - if boxed { - buffer.appendInt32(-886477832) - } - serializeString(label, buffer: buffer, boxed: false) - serializeInt64(amount, buffer: buffer, boxed: false) - break - } - } - - public func descriptionFields() -> (String, [(String, Any)]) { - switch self { - case .labeledPrice(let label, let amount): - return ("labeledPrice", [("label", label as Any), ("amount", amount as Any)]) - } - } - - public static func parse_labeledPrice(_ reader: BufferReader) -> LabeledPrice? { - var _1: String? - _1 = parseString(reader) - var _2: Int64? - _2 = reader.readInt64() - let _c1 = _1 != nil - let _c2 = _2 != nil - if _c1 && _c2 { - return Api.LabeledPrice.labeledPrice(label: _1!, amount: _2!) - } - else { - return nil - } - } - - } -} -public extension Api { - enum LangPackDifference: TypeConstructorDescription { - case langPackDifference(langCode: String, fromVersion: Int32, version: Int32, strings: [Api.LangPackString]) - - public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { - switch self { - case .langPackDifference(let langCode, let fromVersion, let version, let strings): - if boxed { - buffer.appendInt32(-209337866) - } - serializeString(langCode, buffer: buffer, boxed: false) - serializeInt32(fromVersion, buffer: buffer, boxed: false) - serializeInt32(version, buffer: buffer, boxed: false) - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(strings.count)) - for item in strings { - item.serialize(buffer, true) - } - break - } - } - - public func descriptionFields() -> (String, [(String, Any)]) { - switch self { - case .langPackDifference(let langCode, let fromVersion, let version, let strings): - return ("langPackDifference", [("langCode", langCode as Any), ("fromVersion", fromVersion as Any), ("version", version as Any), ("strings", strings as Any)]) - } - } - - public static func parse_langPackDifference(_ reader: BufferReader) -> LangPackDifference? { - var _1: String? - _1 = parseString(reader) - var _2: Int32? - _2 = reader.readInt32() - var _3: Int32? - _3 = reader.readInt32() - var _4: [Api.LangPackString]? - if let _ = reader.readInt32() { - _4 = Api.parseVector(reader, elementSignature: 0, elementType: Api.LangPackString.self) - } - let _c1 = _1 != nil - let _c2 = _2 != nil - let _c3 = _3 != nil - let _c4 = _4 != nil - if _c1 && _c2 && _c3 && _c4 { - return Api.LangPackDifference.langPackDifference(langCode: _1!, fromVersion: _2!, version: _3!, strings: _4!) - } - else { - return nil - } - } - - } -} -public extension Api { - enum LangPackLanguage: TypeConstructorDescription { - case langPackLanguage(flags: Int32, name: String, nativeName: String, langCode: String, baseLangCode: String?, pluralCode: String, stringsCount: Int32, translatedCount: Int32, translationsUrl: String) - - public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { - switch self { - case .langPackLanguage(let flags, let name, let nativeName, let langCode, let baseLangCode, let pluralCode, let stringsCount, let translatedCount, let translationsUrl): - if boxed { - buffer.appendInt32(-288727837) - } - serializeInt32(flags, buffer: buffer, boxed: false) - serializeString(name, buffer: buffer, boxed: false) - serializeString(nativeName, buffer: buffer, boxed: false) - serializeString(langCode, buffer: buffer, boxed: false) - if Int(flags) & Int(1 << 1) != 0 {serializeString(baseLangCode!, buffer: buffer, boxed: false)} - serializeString(pluralCode, buffer: buffer, boxed: false) - serializeInt32(stringsCount, buffer: buffer, boxed: false) - serializeInt32(translatedCount, buffer: buffer, boxed: false) - serializeString(translationsUrl, buffer: buffer, boxed: false) - break - } - } - - public func descriptionFields() -> (String, [(String, Any)]) { - switch self { - case .langPackLanguage(let flags, let name, let nativeName, let langCode, let baseLangCode, let pluralCode, let stringsCount, let translatedCount, let translationsUrl): - return ("langPackLanguage", [("flags", flags as Any), ("name", name as Any), ("nativeName", nativeName as Any), ("langCode", langCode as Any), ("baseLangCode", baseLangCode as Any), ("pluralCode", pluralCode as Any), ("stringsCount", stringsCount as Any), ("translatedCount", translatedCount as Any), ("translationsUrl", translationsUrl as Any)]) - } - } - - public static func parse_langPackLanguage(_ reader: BufferReader) -> LangPackLanguage? { - var _1: Int32? - _1 = reader.readInt32() - var _2: String? - _2 = parseString(reader) - var _3: String? - _3 = parseString(reader) - var _4: String? - _4 = parseString(reader) - var _5: String? - if Int(_1!) & Int(1 << 1) != 0 {_5 = parseString(reader) } - var _6: String? - _6 = parseString(reader) - var _7: Int32? - _7 = reader.readInt32() - var _8: Int32? - _8 = reader.readInt32() - var _9: String? - _9 = parseString(reader) - 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 = _6 != nil - let _c7 = _7 != nil - let _c8 = _8 != nil - let _c9 = _9 != nil - if _c1 && _c2 && _c3 && _c4 && _c5 && _c6 && _c7 && _c8 && _c9 { - return Api.LangPackLanguage.langPackLanguage(flags: _1!, name: _2!, nativeName: _3!, langCode: _4!, baseLangCode: _5, pluralCode: _6!, stringsCount: _7!, translatedCount: _8!, translationsUrl: _9!) - } - else { - return nil - } - } - - } -} diff --git a/submodules/TelegramApi/Sources/Api12.swift b/submodules/TelegramApi/Sources/Api12.swift index f63228f098c..1edeff5668b 100644 --- a/submodules/TelegramApi/Sources/Api12.swift +++ b/submodules/TelegramApi/Sources/Api12.swift @@ -1,3 +1,165 @@ +public extension Api { + enum LabeledPrice: TypeConstructorDescription { + case labeledPrice(label: String, amount: Int64) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .labeledPrice(let label, let amount): + if boxed { + buffer.appendInt32(-886477832) + } + serializeString(label, buffer: buffer, boxed: false) + serializeInt64(amount, buffer: buffer, boxed: false) + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .labeledPrice(let label, let amount): + return ("labeledPrice", [("label", label as Any), ("amount", amount as Any)]) + } + } + + public static func parse_labeledPrice(_ reader: BufferReader) -> LabeledPrice? { + var _1: String? + _1 = parseString(reader) + var _2: Int64? + _2 = reader.readInt64() + let _c1 = _1 != nil + let _c2 = _2 != nil + if _c1 && _c2 { + return Api.LabeledPrice.labeledPrice(label: _1!, amount: _2!) + } + else { + return nil + } + } + + } +} +public extension Api { + enum LangPackDifference: TypeConstructorDescription { + case langPackDifference(langCode: String, fromVersion: Int32, version: Int32, strings: [Api.LangPackString]) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .langPackDifference(let langCode, let fromVersion, let version, let strings): + if boxed { + buffer.appendInt32(-209337866) + } + serializeString(langCode, buffer: buffer, boxed: false) + serializeInt32(fromVersion, buffer: buffer, boxed: false) + serializeInt32(version, buffer: buffer, boxed: false) + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(strings.count)) + for item in strings { + item.serialize(buffer, true) + } + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .langPackDifference(let langCode, let fromVersion, let version, let strings): + return ("langPackDifference", [("langCode", langCode as Any), ("fromVersion", fromVersion as Any), ("version", version as Any), ("strings", strings as Any)]) + } + } + + public static func parse_langPackDifference(_ reader: BufferReader) -> LangPackDifference? { + var _1: String? + _1 = parseString(reader) + var _2: Int32? + _2 = reader.readInt32() + var _3: Int32? + _3 = reader.readInt32() + var _4: [Api.LangPackString]? + if let _ = reader.readInt32() { + _4 = Api.parseVector(reader, elementSignature: 0, elementType: Api.LangPackString.self) + } + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + let _c4 = _4 != nil + if _c1 && _c2 && _c3 && _c4 { + return Api.LangPackDifference.langPackDifference(langCode: _1!, fromVersion: _2!, version: _3!, strings: _4!) + } + else { + return nil + } + } + + } +} +public extension Api { + enum LangPackLanguage: TypeConstructorDescription { + case langPackLanguage(flags: Int32, name: String, nativeName: String, langCode: String, baseLangCode: String?, pluralCode: String, stringsCount: Int32, translatedCount: Int32, translationsUrl: String) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .langPackLanguage(let flags, let name, let nativeName, let langCode, let baseLangCode, let pluralCode, let stringsCount, let translatedCount, let translationsUrl): + if boxed { + buffer.appendInt32(-288727837) + } + serializeInt32(flags, buffer: buffer, boxed: false) + serializeString(name, buffer: buffer, boxed: false) + serializeString(nativeName, buffer: buffer, boxed: false) + serializeString(langCode, buffer: buffer, boxed: false) + if Int(flags) & Int(1 << 1) != 0 {serializeString(baseLangCode!, buffer: buffer, boxed: false)} + serializeString(pluralCode, buffer: buffer, boxed: false) + serializeInt32(stringsCount, buffer: buffer, boxed: false) + serializeInt32(translatedCount, buffer: buffer, boxed: false) + serializeString(translationsUrl, buffer: buffer, boxed: false) + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .langPackLanguage(let flags, let name, let nativeName, let langCode, let baseLangCode, let pluralCode, let stringsCount, let translatedCount, let translationsUrl): + return ("langPackLanguage", [("flags", flags as Any), ("name", name as Any), ("nativeName", nativeName as Any), ("langCode", langCode as Any), ("baseLangCode", baseLangCode as Any), ("pluralCode", pluralCode as Any), ("stringsCount", stringsCount as Any), ("translatedCount", translatedCount as Any), ("translationsUrl", translationsUrl as Any)]) + } + } + + public static func parse_langPackLanguage(_ reader: BufferReader) -> LangPackLanguage? { + var _1: Int32? + _1 = reader.readInt32() + var _2: String? + _2 = parseString(reader) + var _3: String? + _3 = parseString(reader) + var _4: String? + _4 = parseString(reader) + var _5: String? + if Int(_1!) & Int(1 << 1) != 0 {_5 = parseString(reader) } + var _6: String? + _6 = parseString(reader) + var _7: Int32? + _7 = reader.readInt32() + var _8: Int32? + _8 = reader.readInt32() + var _9: String? + _9 = parseString(reader) + 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 = _6 != nil + let _c7 = _7 != nil + let _c8 = _8 != nil + let _c9 = _9 != nil + if _c1 && _c2 && _c3 && _c4 && _c5 && _c6 && _c7 && _c8 && _c9 { + return Api.LangPackLanguage.langPackLanguage(flags: _1!, name: _2!, nativeName: _3!, langCode: _4!, baseLangCode: _5, pluralCode: _6!, stringsCount: _7!, translatedCount: _8!, translationsUrl: _9!) + } + else { + return nil + } + } + + } +} public extension Api { enum LangPackString: TypeConstructorDescription { case langPackString(key: String, value: String) @@ -424,15 +586,15 @@ public extension Api { } public extension Api { indirect enum Message: TypeConstructorDescription { - case message(flags: Int32, id: Int32, fromId: Api.Peer?, fromBoostsApplied: Int32?, peerId: Api.Peer, savedPeerId: Api.Peer?, fwdFrom: Api.MessageFwdHeader?, viaBotId: Int64?, replyTo: Api.MessageReplyHeader?, date: Int32, message: String, media: Api.MessageMedia?, replyMarkup: Api.ReplyMarkup?, entities: [Api.MessageEntity]?, views: Int32?, forwards: Int32?, replies: Api.MessageReplies?, editDate: Int32?, postAuthor: String?, groupedId: Int64?, reactions: Api.MessageReactions?, restrictionReason: [Api.RestrictionReason]?, ttlPeriod: Int32?) + case message(flags: Int32, id: Int32, fromId: Api.Peer?, fromBoostsApplied: Int32?, peerId: Api.Peer, savedPeerId: Api.Peer?, fwdFrom: Api.MessageFwdHeader?, viaBotId: Int64?, replyTo: Api.MessageReplyHeader?, date: Int32, message: String, media: Api.MessageMedia?, replyMarkup: Api.ReplyMarkup?, entities: [Api.MessageEntity]?, views: Int32?, forwards: Int32?, replies: Api.MessageReplies?, editDate: Int32?, postAuthor: String?, groupedId: Int64?, reactions: Api.MessageReactions?, restrictionReason: [Api.RestrictionReason]?, ttlPeriod: Int32?, quickReplyShortcutId: Int32?) case messageEmpty(flags: Int32, id: Int32, peerId: Api.Peer?) case messageService(flags: Int32, id: Int32, fromId: Api.Peer?, peerId: Api.Peer, replyTo: Api.MessageReplyHeader?, date: Int32, action: Api.MessageAction, ttlPeriod: Int32?) public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { switch self { - case .message(let flags, let id, let fromId, let fromBoostsApplied, let peerId, let savedPeerId, let fwdFrom, let viaBotId, let replyTo, let date, let message, let media, let replyMarkup, let entities, let views, let forwards, let replies, let editDate, let postAuthor, let groupedId, let reactions, let restrictionReason, let ttlPeriod): + case .message(let flags, let id, let fromId, let fromBoostsApplied, let peerId, let savedPeerId, let fwdFrom, let viaBotId, let replyTo, let date, let message, let media, let replyMarkup, let entities, let views, let forwards, let replies, let editDate, let postAuthor, let groupedId, let reactions, let restrictionReason, let ttlPeriod, let quickReplyShortcutId): if boxed { - buffer.appendInt32(508332649) + buffer.appendInt32(-1502839044) } serializeInt32(flags, buffer: buffer, boxed: false) serializeInt32(id, buffer: buffer, boxed: false) @@ -465,6 +627,7 @@ public extension Api { item.serialize(buffer, true) }} if Int(flags) & Int(1 << 25) != 0 {serializeInt32(ttlPeriod!, buffer: buffer, boxed: false)} + if Int(flags) & Int(1 << 30) != 0 {serializeInt32(quickReplyShortcutId!, buffer: buffer, boxed: false)} break case .messageEmpty(let flags, let id, let peerId): if boxed { @@ -492,8 +655,8 @@ public extension Api { public func descriptionFields() -> (String, [(String, Any)]) { switch self { - case .message(let flags, let id, let fromId, let fromBoostsApplied, let peerId, let savedPeerId, let fwdFrom, let viaBotId, let replyTo, let date, let message, let media, let replyMarkup, let entities, let views, let forwards, let replies, let editDate, let postAuthor, let groupedId, let reactions, let restrictionReason, let ttlPeriod): - return ("message", [("flags", flags as Any), ("id", id as Any), ("fromId", fromId as Any), ("fromBoostsApplied", fromBoostsApplied as Any), ("peerId", peerId as Any), ("savedPeerId", savedPeerId as Any), ("fwdFrom", fwdFrom as Any), ("viaBotId", viaBotId as Any), ("replyTo", replyTo as Any), ("date", date as Any), ("message", message as Any), ("media", media as Any), ("replyMarkup", replyMarkup as Any), ("entities", entities as Any), ("views", views as Any), ("forwards", forwards as Any), ("replies", replies as Any), ("editDate", editDate as Any), ("postAuthor", postAuthor as Any), ("groupedId", groupedId as Any), ("reactions", reactions as Any), ("restrictionReason", restrictionReason as Any), ("ttlPeriod", ttlPeriod as Any)]) + case .message(let flags, let id, let fromId, let fromBoostsApplied, let peerId, let savedPeerId, let fwdFrom, let viaBotId, let replyTo, let date, let message, let media, let replyMarkup, let entities, let views, let forwards, let replies, let editDate, let postAuthor, let groupedId, let reactions, let restrictionReason, let ttlPeriod, let quickReplyShortcutId): + return ("message", [("flags", flags as Any), ("id", id as Any), ("fromId", fromId as Any), ("fromBoostsApplied", fromBoostsApplied as Any), ("peerId", peerId as Any), ("savedPeerId", savedPeerId as Any), ("fwdFrom", fwdFrom as Any), ("viaBotId", viaBotId as Any), ("replyTo", replyTo as Any), ("date", date as Any), ("message", message as Any), ("media", media as Any), ("replyMarkup", replyMarkup as Any), ("entities", entities as Any), ("views", views as Any), ("forwards", forwards as Any), ("replies", replies as Any), ("editDate", editDate as Any), ("postAuthor", postAuthor as Any), ("groupedId", groupedId as Any), ("reactions", reactions as Any), ("restrictionReason", restrictionReason as Any), ("ttlPeriod", ttlPeriod as Any), ("quickReplyShortcutId", quickReplyShortcutId as Any)]) case .messageEmpty(let flags, let id, let peerId): return ("messageEmpty", [("flags", flags as Any), ("id", id as Any), ("peerId", peerId as Any)]) case .messageService(let flags, let id, let fromId, let peerId, let replyTo, let date, let action, let ttlPeriod): @@ -570,6 +733,8 @@ public extension Api { } } var _23: Int32? if Int(_1!) & Int(1 << 25) != 0 {_23 = reader.readInt32() } + var _24: Int32? + if Int(_1!) & Int(1 << 30) != 0 {_24 = reader.readInt32() } let _c1 = _1 != nil let _c2 = _2 != nil let _c3 = (Int(_1!) & Int(1 << 8) == 0) || _3 != nil @@ -593,8 +758,9 @@ public extension Api { let _c21 = (Int(_1!) & Int(1 << 20) == 0) || _21 != nil let _c22 = (Int(_1!) & Int(1 << 22) == 0) || _22 != nil let _c23 = (Int(_1!) & Int(1 << 25) == 0) || _23 != nil - if _c1 && _c2 && _c3 && _c4 && _c5 && _c6 && _c7 && _c8 && _c9 && _c10 && _c11 && _c12 && _c13 && _c14 && _c15 && _c16 && _c17 && _c18 && _c19 && _c20 && _c21 && _c22 && _c23 { - return Api.Message.message(flags: _1!, id: _2!, fromId: _3, fromBoostsApplied: _4, peerId: _5!, savedPeerId: _6, fwdFrom: _7, viaBotId: _8, replyTo: _9, date: _10!, message: _11!, media: _12, replyMarkup: _13, entities: _14, views: _15, forwards: _16, replies: _17, editDate: _18, postAuthor: _19, groupedId: _20, reactions: _21, restrictionReason: _22, ttlPeriod: _23) + let _c24 = (Int(_1!) & Int(1 << 30) == 0) || _24 != nil + if _c1 && _c2 && _c3 && _c4 && _c5 && _c6 && _c7 && _c8 && _c9 && _c10 && _c11 && _c12 && _c13 && _c14 && _c15 && _c16 && _c17 && _c18 && _c19 && _c20 && _c21 && _c22 && _c23 && _c24 { + return Api.Message.message(flags: _1!, id: _2!, fromId: _3, fromBoostsApplied: _4, peerId: _5!, savedPeerId: _6, fwdFrom: _7, viaBotId: _8, replyTo: _9, date: _10!, message: _11!, media: _12, replyMarkup: _13, entities: _14, views: _15, forwards: _16, replies: _17, editDate: _18, postAuthor: _19, groupedId: _20, reactions: _21, restrictionReason: _22, ttlPeriod: _23, quickReplyShortcutId: _24) } else { return nil diff --git a/submodules/TelegramApi/Sources/Api18.swift b/submodules/TelegramApi/Sources/Api18.swift index cce8c834d79..144d26302ee 100644 --- a/submodules/TelegramApi/Sources/Api18.swift +++ b/submodules/TelegramApi/Sources/Api18.swift @@ -244,6 +244,54 @@ public extension Api { } } +public extension Api { + enum QuickReply: TypeConstructorDescription { + case quickReply(shortcutId: Int32, shortcut: String, topMessage: Int32, count: Int32) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .quickReply(let shortcutId, let shortcut, let topMessage, let count): + if boxed { + buffer.appendInt32(110563371) + } + serializeInt32(shortcutId, buffer: buffer, boxed: false) + serializeString(shortcut, buffer: buffer, boxed: false) + serializeInt32(topMessage, buffer: buffer, boxed: false) + serializeInt32(count, buffer: buffer, boxed: false) + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .quickReply(let shortcutId, let shortcut, let topMessage, let count): + return ("quickReply", [("shortcutId", shortcutId as Any), ("shortcut", shortcut as Any), ("topMessage", topMessage as Any), ("count", count as Any)]) + } + } + + public static func parse_quickReply(_ reader: BufferReader) -> QuickReply? { + var _1: Int32? + _1 = reader.readInt32() + var _2: String? + _2 = parseString(reader) + var _3: Int32? + _3 = reader.readInt32() + var _4: Int32? + _4 = reader.readInt32() + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + let _c4 = _4 != nil + if _c1 && _c2 && _c3 && _c4 { + return Api.QuickReply.quickReply(shortcutId: _1!, shortcut: _2!, topMessage: _3!, count: _4!) + } + else { + return nil + } + } + + } +} public extension Api { enum Reaction: TypeConstructorDescription { case reactionCustomEmoji(documentId: Int64) diff --git a/submodules/TelegramApi/Sources/Api2.swift b/submodules/TelegramApi/Sources/Api2.swift index 5f696bd03ae..9436ea52397 100644 --- a/submodules/TelegramApi/Sources/Api2.swift +++ b/submodules/TelegramApi/Sources/Api2.swift @@ -588,6 +588,350 @@ public extension Api { } } +public extension Api { + enum BusinessAwayMessage: TypeConstructorDescription { + case businessAwayMessage(flags: Int32, shortcutId: Int32, schedule: Api.BusinessAwayMessageSchedule, recipients: Api.BusinessRecipients) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .businessAwayMessage(let flags, let shortcutId, let schedule, let recipients): + if boxed { + buffer.appendInt32(-283809188) + } + serializeInt32(flags, buffer: buffer, boxed: false) + serializeInt32(shortcutId, buffer: buffer, boxed: false) + schedule.serialize(buffer, true) + recipients.serialize(buffer, true) + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .businessAwayMessage(let flags, let shortcutId, let schedule, let recipients): + return ("businessAwayMessage", [("flags", flags as Any), ("shortcutId", shortcutId as Any), ("schedule", schedule as Any), ("recipients", recipients as Any)]) + } + } + + public static func parse_businessAwayMessage(_ reader: BufferReader) -> BusinessAwayMessage? { + var _1: Int32? + _1 = reader.readInt32() + var _2: Int32? + _2 = reader.readInt32() + var _3: Api.BusinessAwayMessageSchedule? + if let signature = reader.readInt32() { + _3 = Api.parse(reader, signature: signature) as? Api.BusinessAwayMessageSchedule + } + var _4: Api.BusinessRecipients? + if let signature = reader.readInt32() { + _4 = Api.parse(reader, signature: signature) as? Api.BusinessRecipients + } + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + let _c4 = _4 != nil + if _c1 && _c2 && _c3 && _c4 { + return Api.BusinessAwayMessage.businessAwayMessage(flags: _1!, shortcutId: _2!, schedule: _3!, recipients: _4!) + } + else { + return nil + } + } + + } +} +public extension Api { + enum BusinessAwayMessageSchedule: TypeConstructorDescription { + case businessAwayMessageScheduleAlways + case businessAwayMessageScheduleCustom(startDate: Int32, endDate: Int32) + case businessAwayMessageScheduleOutsideWorkHours + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .businessAwayMessageScheduleAlways: + if boxed { + buffer.appendInt32(-910564679) + } + + break + case .businessAwayMessageScheduleCustom(let startDate, let endDate): + if boxed { + buffer.appendInt32(-867328308) + } + serializeInt32(startDate, buffer: buffer, boxed: false) + serializeInt32(endDate, buffer: buffer, boxed: false) + break + case .businessAwayMessageScheduleOutsideWorkHours: + if boxed { + buffer.appendInt32(-1007487743) + } + + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .businessAwayMessageScheduleAlways: + return ("businessAwayMessageScheduleAlways", []) + case .businessAwayMessageScheduleCustom(let startDate, let endDate): + return ("businessAwayMessageScheduleCustom", [("startDate", startDate as Any), ("endDate", endDate as Any)]) + case .businessAwayMessageScheduleOutsideWorkHours: + return ("businessAwayMessageScheduleOutsideWorkHours", []) + } + } + + public static func parse_businessAwayMessageScheduleAlways(_ reader: BufferReader) -> BusinessAwayMessageSchedule? { + return Api.BusinessAwayMessageSchedule.businessAwayMessageScheduleAlways + } + public static func parse_businessAwayMessageScheduleCustom(_ reader: BufferReader) -> BusinessAwayMessageSchedule? { + 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.BusinessAwayMessageSchedule.businessAwayMessageScheduleCustom(startDate: _1!, endDate: _2!) + } + else { + return nil + } + } + public static func parse_businessAwayMessageScheduleOutsideWorkHours(_ reader: BufferReader) -> BusinessAwayMessageSchedule? { + return Api.BusinessAwayMessageSchedule.businessAwayMessageScheduleOutsideWorkHours + } + + } +} +public extension Api { + enum BusinessGreetingMessage: TypeConstructorDescription { + case businessGreetingMessage(shortcutId: Int32, recipients: Api.BusinessRecipients, noActivityDays: Int32) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .businessGreetingMessage(let shortcutId, let recipients, let noActivityDays): + if boxed { + buffer.appendInt32(-451302485) + } + serializeInt32(shortcutId, buffer: buffer, boxed: false) + recipients.serialize(buffer, true) + serializeInt32(noActivityDays, buffer: buffer, boxed: false) + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .businessGreetingMessage(let shortcutId, let recipients, let noActivityDays): + return ("businessGreetingMessage", [("shortcutId", shortcutId as Any), ("recipients", recipients as Any), ("noActivityDays", noActivityDays as Any)]) + } + } + + public static func parse_businessGreetingMessage(_ reader: BufferReader) -> BusinessGreetingMessage? { + var _1: Int32? + _1 = reader.readInt32() + var _2: Api.BusinessRecipients? + if let signature = reader.readInt32() { + _2 = Api.parse(reader, signature: signature) as? Api.BusinessRecipients + } + var _3: Int32? + _3 = reader.readInt32() + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + if _c1 && _c2 && _c3 { + return Api.BusinessGreetingMessage.businessGreetingMessage(shortcutId: _1!, recipients: _2!, noActivityDays: _3!) + } + else { + return nil + } + } + + } +} +public extension Api { + enum BusinessLocation: TypeConstructorDescription { + case businessLocation(flags: Int32, geoPoint: Api.GeoPoint?, address: String) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .businessLocation(let flags, let geoPoint, let address): + if boxed { + buffer.appendInt32(-1403249929) + } + serializeInt32(flags, buffer: buffer, boxed: false) + if Int(flags) & Int(1 << 0) != 0 {geoPoint!.serialize(buffer, true)} + serializeString(address, buffer: buffer, boxed: false) + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .businessLocation(let flags, let geoPoint, let address): + return ("businessLocation", [("flags", flags as Any), ("geoPoint", geoPoint as Any), ("address", address as Any)]) + } + } + + public static func parse_businessLocation(_ reader: BufferReader) -> BusinessLocation? { + var _1: Int32? + _1 = reader.readInt32() + var _2: Api.GeoPoint? + if Int(_1!) & Int(1 << 0) != 0 {if let signature = reader.readInt32() { + _2 = Api.parse(reader, signature: signature) as? Api.GeoPoint + } } + var _3: String? + _3 = parseString(reader) + let _c1 = _1 != nil + let _c2 = (Int(_1!) & Int(1 << 0) == 0) || _2 != nil + let _c3 = _3 != nil + if _c1 && _c2 && _c3 { + return Api.BusinessLocation.businessLocation(flags: _1!, geoPoint: _2, address: _3!) + } + else { + return nil + } + } + + } +} +public extension Api { + enum BusinessRecipients: TypeConstructorDescription { + case businessRecipients(flags: Int32, users: [Int64]?) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .businessRecipients(let flags, let users): + if boxed { + buffer.appendInt32(554733559) + } + serializeInt32(flags, buffer: buffer, boxed: false) + if Int(flags) & Int(1 << 4) != 0 {buffer.appendInt32(481674261) + buffer.appendInt32(Int32(users!.count)) + for item in users! { + serializeInt64(item, buffer: buffer, boxed: false) + }} + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .businessRecipients(let flags, let users): + return ("businessRecipients", [("flags", flags as Any), ("users", users as Any)]) + } + } + + public static func parse_businessRecipients(_ reader: BufferReader) -> BusinessRecipients? { + var _1: Int32? + _1 = reader.readInt32() + var _2: [Int64]? + if Int(_1!) & Int(1 << 4) != 0 {if let _ = reader.readInt32() { + _2 = Api.parseVector(reader, elementSignature: 570911930, elementType: Int64.self) + } } + let _c1 = _1 != nil + let _c2 = (Int(_1!) & Int(1 << 4) == 0) || _2 != nil + if _c1 && _c2 { + return Api.BusinessRecipients.businessRecipients(flags: _1!, users: _2) + } + else { + return nil + } + } + + } +} +public extension Api { + enum BusinessWeeklyOpen: TypeConstructorDescription { + case businessWeeklyOpen(startMinute: Int32, endMinute: Int32) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .businessWeeklyOpen(let startMinute, let endMinute): + if boxed { + buffer.appendInt32(302717625) + } + serializeInt32(startMinute, buffer: buffer, boxed: false) + serializeInt32(endMinute, buffer: buffer, boxed: false) + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .businessWeeklyOpen(let startMinute, let endMinute): + return ("businessWeeklyOpen", [("startMinute", startMinute as Any), ("endMinute", endMinute as Any)]) + } + } + + public static func parse_businessWeeklyOpen(_ reader: BufferReader) -> BusinessWeeklyOpen? { + 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.BusinessWeeklyOpen.businessWeeklyOpen(startMinute: _1!, endMinute: _2!) + } + else { + return nil + } + } + + } +} +public extension Api { + enum BusinessWorkHours: TypeConstructorDescription { + case businessWorkHours(flags: Int32, timezoneId: String, weeklyOpen: [Api.BusinessWeeklyOpen]) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .businessWorkHours(let flags, let timezoneId, let weeklyOpen): + if boxed { + buffer.appendInt32(-1936543592) + } + serializeInt32(flags, buffer: buffer, boxed: false) + serializeString(timezoneId, buffer: buffer, boxed: false) + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(weeklyOpen.count)) + for item in weeklyOpen { + item.serialize(buffer, true) + } + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .businessWorkHours(let flags, let timezoneId, let weeklyOpen): + return ("businessWorkHours", [("flags", flags as Any), ("timezoneId", timezoneId as Any), ("weeklyOpen", weeklyOpen as Any)]) + } + } + + public static func parse_businessWorkHours(_ reader: BufferReader) -> BusinessWorkHours? { + var _1: Int32? + _1 = reader.readInt32() + var _2: String? + _2 = parseString(reader) + var _3: [Api.BusinessWeeklyOpen]? + if let _ = reader.readInt32() { + _3 = Api.parseVector(reader, elementSignature: 0, elementType: Api.BusinessWeeklyOpen.self) + } + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + if _c1 && _c2 && _c3 { + return Api.BusinessWorkHours.businessWorkHours(flags: _1!, timezoneId: _2!, weeklyOpen: _3!) + } + else { + return nil + } + } + + } +} public extension Api { enum CdnConfig: TypeConstructorDescription { case cdnConfig(publicKeys: [Api.CdnPublicKey]) diff --git a/submodules/TelegramApi/Sources/Api21.swift b/submodules/TelegramApi/Sources/Api21.swift index 174f9c3cb72..c09e5c1ee60 100644 --- a/submodules/TelegramApi/Sources/Api21.swift +++ b/submodules/TelegramApi/Sources/Api21.swift @@ -84,6 +84,50 @@ public extension Api { } } +public extension Api { + enum SmsJob: TypeConstructorDescription { + case smsJob(jobId: String, phoneNumber: String, text: String) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .smsJob(let jobId, let phoneNumber, let text): + if boxed { + buffer.appendInt32(-425595208) + } + serializeString(jobId, buffer: buffer, boxed: false) + serializeString(phoneNumber, buffer: buffer, boxed: false) + serializeString(text, buffer: buffer, boxed: false) + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .smsJob(let jobId, let phoneNumber, let text): + return ("smsJob", [("jobId", jobId as Any), ("phoneNumber", phoneNumber as Any), ("text", text as Any)]) + } + } + + public static func parse_smsJob(_ reader: BufferReader) -> SmsJob? { + var _1: String? + _1 = parseString(reader) + var _2: String? + _2 = parseString(reader) + var _3: String? + _3 = parseString(reader) + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + if _c1 && _c2 && _c3 { + return Api.SmsJob.smsJob(jobId: _1!, phoneNumber: _2!, text: _3!) + } + else { + return nil + } + } + + } +} public extension Api { indirect enum SponsoredMessage: TypeConstructorDescription { case sponsoredMessage(flags: Int32, randomId: Buffer, fromId: Api.Peer?, chatInvite: Api.ChatInvite?, chatInviteHash: String?, channelPost: Int32?, startParam: String?, webpage: Api.SponsoredWebPage?, app: Api.BotApp?, message: String, entities: [Api.MessageEntity]?, buttonText: String?, sponsorInfo: String?, additionalInfo: String?) diff --git a/submodules/TelegramApi/Sources/Api22.swift b/submodules/TelegramApi/Sources/Api22.swift index 3d1854bcb31..e7175c51f54 100644 --- a/submodules/TelegramApi/Sources/Api22.swift +++ b/submodules/TelegramApi/Sources/Api22.swift @@ -254,6 +254,50 @@ public extension Api { } } +public extension Api { + enum Timezone: TypeConstructorDescription { + case timezone(id: String, name: String, utcOffset: Int32) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .timezone(let id, let name, let utcOffset): + if boxed { + buffer.appendInt32(-7173643) + } + serializeString(id, buffer: buffer, boxed: false) + serializeString(name, buffer: buffer, boxed: false) + serializeInt32(utcOffset, buffer: buffer, boxed: false) + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .timezone(let id, let name, let utcOffset): + return ("timezone", [("id", id as Any), ("name", name as Any), ("utcOffset", utcOffset as Any)]) + } + } + + public static func parse_timezone(_ reader: BufferReader) -> Timezone? { + var _1: String? + _1 = parseString(reader) + var _2: String? + _2 = parseString(reader) + var _3: Int32? + _3 = reader.readInt32() + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + if _c1 && _c2 && _c3 { + return Api.Timezone.timezone(id: _1!, name: _2!, utcOffset: _3!) + } + else { + return nil + } + } + + } +} public extension Api { enum TopPeer: TypeConstructorDescription { case topPeer(peer: Api.Peer, rating: Double) @@ -464,15 +508,19 @@ public extension Api { indirect enum Update: TypeConstructorDescription { case updateAttachMenuBots case updateAutoSaveSettings + case updateBotBusinessConnect(connection: Api.BotBusinessConnection, qts: Int32) case updateBotCallbackQuery(flags: Int32, queryId: Int64, userId: Int64, peer: Api.Peer, msgId: Int32, chatInstance: Int64, data: Buffer?, gameShortName: String?) case updateBotChatBoost(peer: Api.Peer, boost: Api.Boost, qts: Int32) case updateBotChatInviteRequester(peer: Api.Peer, date: Int32, userId: Int64, about: String, invite: Api.ExportedChatInvite, qts: Int32) case updateBotCommands(peer: Api.Peer, botId: Int64, commands: [Api.BotCommand]) + case updateBotDeleteBusinessMessage(connectionId: String, messages: [Int32], qts: Int32) + case updateBotEditBusinessMessage(connectionId: String, message: Api.Message, qts: Int32) case updateBotInlineQuery(flags: Int32, queryId: Int64, userId: Int64, query: String, geo: Api.GeoPoint?, peerType: Api.InlineQueryPeerType?, offset: String) case updateBotInlineSend(flags: Int32, userId: Int64, query: String, geo: Api.GeoPoint?, id: String, msgId: Api.InputBotInlineMessageID?) case updateBotMenuButton(botId: Int64, button: Api.BotMenuButton) case updateBotMessageReaction(peer: Api.Peer, msgId: Int32, date: Int32, actor: Api.Peer, oldReactions: [Api.Reaction], newReactions: [Api.Reaction], qts: Int32) case updateBotMessageReactions(peer: Api.Peer, msgId: Int32, date: Int32, reactions: [Api.ReactionCount], qts: Int32) + case updateBotNewBusinessMessage(connectionId: String, message: Api.Message, qts: Int32) case updateBotPrecheckoutQuery(flags: Int32, queryId: Int64, userId: Int64, payload: Buffer, info: Api.PaymentRequestedInfo?, shippingOptionId: String?, currency: String, totalAmount: Int64) case updateBotShippingQuery(queryId: Int64, userId: Int64, payload: Buffer, shippingAddress: Api.PostAddress) case updateBotStopped(userId: Int64, date: Int32, stopped: Api.Bool, qts: Int32) @@ -503,6 +551,8 @@ public extension Api { case updateDcOptions(dcOptions: [Api.DcOption]) case updateDeleteChannelMessages(channelId: Int64, messages: [Int32], pts: Int32, ptsCount: Int32) case updateDeleteMessages(messages: [Int32], pts: Int32, ptsCount: Int32) + case updateDeleteQuickReply(shortcutId: Int32) + case updateDeleteQuickReplyMessages(shortcutId: Int32, messages: [Int32]) case updateDeleteScheduledMessages(peer: Api.Peer, messages: [Int32]) case updateDialogFilter(flags: Int32, id: Int32, filter: Api.DialogFilter?) case updateDialogFilterOrder(order: [Int32]) @@ -536,6 +586,7 @@ public extension Api { case updateNewChannelMessage(message: Api.Message, pts: Int32, ptsCount: Int32) case updateNewEncryptedMessage(message: Api.EncryptedMessage, qts: Int32) case updateNewMessage(message: Api.Message, pts: Int32, ptsCount: Int32) + case updateNewQuickReply(quickReply: Api.QuickReply) case updateNewScheduledMessage(message: Api.Message) case updateNewStickerSet(stickerset: Api.messages.StickerSet) case updateNotifySettings(peer: Api.NotifyPeer, notifySettings: Api.PeerNotifySettings) @@ -553,6 +604,8 @@ public extension Api { case updatePinnedSavedDialogs(flags: Int32, order: [Api.DialogPeer]?) case updatePrivacy(key: Api.PrivacyKey, rules: [Api.PrivacyRule]) case updatePtsChanged + case updateQuickReplies(quickReplies: [Api.QuickReply]) + case updateQuickReplyMessage(message: Api.Message) case updateReadChannelDiscussionInbox(flags: Int32, channelId: Int64, topMsgId: Int32, readMaxId: Int32, broadcastId: Int64?, broadcastPost: Int32?) case updateReadChannelDiscussionOutbox(channelId: Int64, topMsgId: Int32, readMaxId: Int32) case updateReadChannelInbox(flags: Int32, folderId: Int32?, channelId: Int64, maxId: Int32, stillUnreadCount: Int32, pts: Int32) @@ -572,6 +625,7 @@ public extension Api { case updateSavedRingtones case updateSentStoryReaction(peer: Api.Peer, storyId: Int32, reaction: Api.Reaction) case updateServiceNotification(flags: Int32, inboxDate: Int32?, type: String, message: String, media: Api.MessageMedia, entities: [Api.MessageEntity]) + case updateSmsJob(jobId: String) case updateStickerSets(flags: Int32) case updateStickerSetsOrder(flags: Int32, order: [Int64]) case updateStoriesStealthMode(stealthMode: Api.StoriesStealthMode) @@ -601,6 +655,13 @@ public extension Api { buffer.appendInt32(-335171433) } + break + case .updateBotBusinessConnect(let connection, let qts): + if boxed { + buffer.appendInt32(-1964652166) + } + connection.serialize(buffer, true) + serializeInt32(qts, buffer: buffer, boxed: false) break case .updateBotCallbackQuery(let flags, let queryId, let userId, let peer, let msgId, let chatInstance, let data, let gameShortName): if boxed { @@ -646,6 +707,26 @@ public extension Api { item.serialize(buffer, true) } break + case .updateBotDeleteBusinessMessage(let connectionId, let messages, let qts): + if boxed { + buffer.appendInt32(-1590796039) + } + serializeString(connectionId, buffer: buffer, boxed: false) + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(messages.count)) + for item in messages { + serializeInt32(item, buffer: buffer, boxed: false) + } + serializeInt32(qts, buffer: buffer, boxed: false) + break + case .updateBotEditBusinessMessage(let connectionId, let message, let qts): + if boxed { + buffer.appendInt32(1420915171) + } + serializeString(connectionId, buffer: buffer, boxed: false) + message.serialize(buffer, true) + serializeInt32(qts, buffer: buffer, boxed: false) + break case .updateBotInlineQuery(let flags, let queryId, let userId, let query, let geo, let peerType, let offset): if boxed { buffer.appendInt32(1232025500) @@ -710,6 +791,14 @@ public extension Api { } serializeInt32(qts, buffer: buffer, boxed: false) break + case .updateBotNewBusinessMessage(let connectionId, let message, let qts): + if boxed { + buffer.appendInt32(-2142069794) + } + serializeString(connectionId, buffer: buffer, boxed: false) + message.serialize(buffer, true) + serializeInt32(qts, buffer: buffer, boxed: false) + break case .updateBotPrecheckoutQuery(let flags, let queryId, let userId, let payload, let info, let shippingOptionId, let currency, let totalAmount): if boxed { buffer.appendInt32(-1934976362) @@ -981,6 +1070,23 @@ public extension Api { serializeInt32(pts, buffer: buffer, boxed: false) serializeInt32(ptsCount, buffer: buffer, boxed: false) break + case .updateDeleteQuickReply(let shortcutId): + if boxed { + buffer.appendInt32(1407644140) + } + serializeInt32(shortcutId, buffer: buffer, boxed: false) + break + case .updateDeleteQuickReplyMessages(let shortcutId, let messages): + if boxed { + buffer.appendInt32(1450174413) + } + serializeInt32(shortcutId, buffer: buffer, boxed: false) + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(messages.count)) + for item in messages { + serializeInt32(item, buffer: buffer, boxed: false) + } + break case .updateDeleteScheduledMessages(let peer, let messages): if boxed { buffer.appendInt32(-1870238482) @@ -1251,6 +1357,12 @@ public extension Api { serializeInt32(pts, buffer: buffer, boxed: false) serializeInt32(ptsCount, buffer: buffer, boxed: false) break + case .updateNewQuickReply(let quickReply): + if boxed { + buffer.appendInt32(-180508905) + } + quickReply.serialize(buffer, true) + break case .updateNewScheduledMessage(let message): if boxed { buffer.appendInt32(967122427) @@ -1402,6 +1514,22 @@ public extension Api { buffer.appendInt32(861169551) } + break + case .updateQuickReplies(let quickReplies): + if boxed { + buffer.appendInt32(-112784718) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(quickReplies.count)) + for item in quickReplies { + item.serialize(buffer, true) + } + break + case .updateQuickReplyMessage(let message): + if boxed { + buffer.appendInt32(1040518415) + } + message.serialize(buffer, true) break case .updateReadChannelDiscussionInbox(let flags, let channelId, let topMsgId, let readMaxId, let broadcastId, let broadcastPost): if boxed { @@ -1560,6 +1688,12 @@ public extension Api { item.serialize(buffer, true) } break + case .updateSmsJob(let jobId): + if boxed { + buffer.appendInt32(-245208620) + } + serializeString(jobId, buffer: buffer, boxed: false) + break case .updateStickerSets(let flags): if boxed { buffer.appendInt32(834816008) @@ -1683,6 +1817,8 @@ public extension Api { return ("updateAttachMenuBots", []) case .updateAutoSaveSettings: return ("updateAutoSaveSettings", []) + case .updateBotBusinessConnect(let connection, let qts): + return ("updateBotBusinessConnect", [("connection", connection as Any), ("qts", qts as Any)]) case .updateBotCallbackQuery(let flags, let queryId, let userId, let peer, let msgId, let chatInstance, let data, let gameShortName): return ("updateBotCallbackQuery", [("flags", flags as Any), ("queryId", queryId as Any), ("userId", userId as Any), ("peer", peer as Any), ("msgId", msgId as Any), ("chatInstance", chatInstance as Any), ("data", data as Any), ("gameShortName", gameShortName as Any)]) case .updateBotChatBoost(let peer, let boost, let qts): @@ -1691,6 +1827,10 @@ public extension Api { return ("updateBotChatInviteRequester", [("peer", peer as Any), ("date", date as Any), ("userId", userId as Any), ("about", about as Any), ("invite", invite as Any), ("qts", qts as Any)]) case .updateBotCommands(let peer, let botId, let commands): return ("updateBotCommands", [("peer", peer as Any), ("botId", botId as Any), ("commands", commands as Any)]) + case .updateBotDeleteBusinessMessage(let connectionId, let messages, let qts): + return ("updateBotDeleteBusinessMessage", [("connectionId", connectionId as Any), ("messages", messages as Any), ("qts", qts as Any)]) + case .updateBotEditBusinessMessage(let connectionId, let message, let qts): + return ("updateBotEditBusinessMessage", [("connectionId", connectionId as Any), ("message", message as Any), ("qts", qts as Any)]) case .updateBotInlineQuery(let flags, let queryId, let userId, let query, let geo, let peerType, let offset): return ("updateBotInlineQuery", [("flags", flags as Any), ("queryId", queryId as Any), ("userId", userId as Any), ("query", query as Any), ("geo", geo as Any), ("peerType", peerType as Any), ("offset", offset as Any)]) case .updateBotInlineSend(let flags, let userId, let query, let geo, let id, let msgId): @@ -1701,6 +1841,8 @@ public extension Api { return ("updateBotMessageReaction", [("peer", peer as Any), ("msgId", msgId as Any), ("date", date as Any), ("actor", actor as Any), ("oldReactions", oldReactions as Any), ("newReactions", newReactions as Any), ("qts", qts as Any)]) case .updateBotMessageReactions(let peer, let msgId, let date, let reactions, let qts): return ("updateBotMessageReactions", [("peer", peer as Any), ("msgId", msgId as Any), ("date", date as Any), ("reactions", reactions as Any), ("qts", qts as Any)]) + case .updateBotNewBusinessMessage(let connectionId, let message, let qts): + return ("updateBotNewBusinessMessage", [("connectionId", connectionId as Any), ("message", message as Any), ("qts", qts as Any)]) case .updateBotPrecheckoutQuery(let flags, let queryId, let userId, let payload, let info, let shippingOptionId, let currency, let totalAmount): return ("updateBotPrecheckoutQuery", [("flags", flags as Any), ("queryId", queryId as Any), ("userId", userId as Any), ("payload", payload as Any), ("info", info as Any), ("shippingOptionId", shippingOptionId as Any), ("currency", currency as Any), ("totalAmount", totalAmount as Any)]) case .updateBotShippingQuery(let queryId, let userId, let payload, let shippingAddress): @@ -1761,6 +1903,10 @@ public extension Api { return ("updateDeleteChannelMessages", [("channelId", channelId as Any), ("messages", messages as Any), ("pts", pts as Any), ("ptsCount", ptsCount as Any)]) case .updateDeleteMessages(let messages, let pts, let ptsCount): return ("updateDeleteMessages", [("messages", messages as Any), ("pts", pts as Any), ("ptsCount", ptsCount as Any)]) + case .updateDeleteQuickReply(let shortcutId): + return ("updateDeleteQuickReply", [("shortcutId", shortcutId as Any)]) + case .updateDeleteQuickReplyMessages(let shortcutId, let messages): + return ("updateDeleteQuickReplyMessages", [("shortcutId", shortcutId as Any), ("messages", messages as Any)]) case .updateDeleteScheduledMessages(let peer, let messages): return ("updateDeleteScheduledMessages", [("peer", peer as Any), ("messages", messages as Any)]) case .updateDialogFilter(let flags, let id, let filter): @@ -1827,6 +1973,8 @@ public extension Api { return ("updateNewEncryptedMessage", [("message", message as Any), ("qts", qts as Any)]) case .updateNewMessage(let message, let pts, let ptsCount): return ("updateNewMessage", [("message", message as Any), ("pts", pts as Any), ("ptsCount", ptsCount as Any)]) + case .updateNewQuickReply(let quickReply): + return ("updateNewQuickReply", [("quickReply", quickReply as Any)]) case .updateNewScheduledMessage(let message): return ("updateNewScheduledMessage", [("message", message as Any)]) case .updateNewStickerSet(let stickerset): @@ -1861,6 +2009,10 @@ public extension Api { return ("updatePrivacy", [("key", key as Any), ("rules", rules as Any)]) case .updatePtsChanged: return ("updatePtsChanged", []) + case .updateQuickReplies(let quickReplies): + return ("updateQuickReplies", [("quickReplies", quickReplies as Any)]) + case .updateQuickReplyMessage(let message): + return ("updateQuickReplyMessage", [("message", message as Any)]) case .updateReadChannelDiscussionInbox(let flags, let channelId, let topMsgId, let readMaxId, let broadcastId, let broadcastPost): return ("updateReadChannelDiscussionInbox", [("flags", flags as Any), ("channelId", channelId as Any), ("topMsgId", topMsgId as Any), ("readMaxId", readMaxId as Any), ("broadcastId", broadcastId as Any), ("broadcastPost", broadcastPost as Any)]) case .updateReadChannelDiscussionOutbox(let channelId, let topMsgId, let readMaxId): @@ -1899,6 +2051,8 @@ public extension Api { return ("updateSentStoryReaction", [("peer", peer as Any), ("storyId", storyId as Any), ("reaction", reaction as Any)]) case .updateServiceNotification(let flags, let inboxDate, let type, let message, let media, let entities): return ("updateServiceNotification", [("flags", flags as Any), ("inboxDate", inboxDate as Any), ("type", type as Any), ("message", message as Any), ("media", media as Any), ("entities", entities as Any)]) + case .updateSmsJob(let jobId): + return ("updateSmsJob", [("jobId", jobId as Any)]) case .updateStickerSets(let flags): return ("updateStickerSets", [("flags", flags as Any)]) case .updateStickerSetsOrder(let flags, let order): @@ -1938,6 +2092,22 @@ public extension Api { public static func parse_updateAutoSaveSettings(_ reader: BufferReader) -> Update? { return Api.Update.updateAutoSaveSettings } + public static func parse_updateBotBusinessConnect(_ reader: BufferReader) -> Update? { + var _1: Api.BotBusinessConnection? + if let signature = reader.readInt32() { + _1 = Api.parse(reader, signature: signature) as? Api.BotBusinessConnection + } + var _2: Int32? + _2 = reader.readInt32() + let _c1 = _1 != nil + let _c2 = _2 != nil + if _c1 && _c2 { + return Api.Update.updateBotBusinessConnect(connection: _1!, qts: _2!) + } + else { + return nil + } + } public static func parse_updateBotCallbackQuery(_ reader: BufferReader) -> Update? { var _1: Int32? _1 = reader.readInt32() @@ -2044,6 +2214,44 @@ public extension Api { return nil } } + public static func parse_updateBotDeleteBusinessMessage(_ reader: BufferReader) -> Update? { + var _1: String? + _1 = parseString(reader) + var _2: [Int32]? + if let _ = reader.readInt32() { + _2 = Api.parseVector(reader, elementSignature: -1471112230, elementType: Int32.self) + } + var _3: Int32? + _3 = reader.readInt32() + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + if _c1 && _c2 && _c3 { + return Api.Update.updateBotDeleteBusinessMessage(connectionId: _1!, messages: _2!, qts: _3!) + } + else { + return nil + } + } + public static func parse_updateBotEditBusinessMessage(_ reader: BufferReader) -> Update? { + var _1: String? + _1 = parseString(reader) + var _2: Api.Message? + if let signature = reader.readInt32() { + _2 = Api.parse(reader, signature: signature) as? Api.Message + } + var _3: Int32? + _3 = reader.readInt32() + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + if _c1 && _c2 && _c3 { + return Api.Update.updateBotEditBusinessMessage(connectionId: _1!, message: _2!, qts: _3!) + } + else { + return nil + } + } public static func parse_updateBotInlineQuery(_ reader: BufferReader) -> Update? { var _1: Int32? _1 = reader.readInt32() @@ -2187,6 +2395,25 @@ public extension Api { return nil } } + public static func parse_updateBotNewBusinessMessage(_ reader: BufferReader) -> Update? { + var _1: String? + _1 = parseString(reader) + var _2: Api.Message? + if let signature = reader.readInt32() { + _2 = Api.parse(reader, signature: signature) as? Api.Message + } + var _3: Int32? + _3 = reader.readInt32() + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + if _c1 && _c2 && _c3 { + return Api.Update.updateBotNewBusinessMessage(connectionId: _1!, message: _2!, qts: _3!) + } + else { + return nil + } + } public static func parse_updateBotPrecheckoutQuery(_ reader: BufferReader) -> Update? { var _1: Int32? _1 = reader.readInt32() @@ -2766,6 +2993,33 @@ public extension Api { return nil } } + public static func parse_updateDeleteQuickReply(_ reader: BufferReader) -> Update? { + var _1: Int32? + _1 = reader.readInt32() + let _c1 = _1 != nil + if _c1 { + return Api.Update.updateDeleteQuickReply(shortcutId: _1!) + } + else { + return nil + } + } + public static func parse_updateDeleteQuickReplyMessages(_ reader: BufferReader) -> Update? { + var _1: Int32? + _1 = reader.readInt32() + var _2: [Int32]? + if let _ = reader.readInt32() { + _2 = Api.parseVector(reader, elementSignature: -1471112230, elementType: Int32.self) + } + let _c1 = _1 != nil + let _c2 = _2 != nil + if _c1 && _c2 { + return Api.Update.updateDeleteQuickReplyMessages(shortcutId: _1!, messages: _2!) + } + else { + return nil + } + } public static func parse_updateDeleteScheduledMessages(_ reader: BufferReader) -> Update? { var _1: Api.Peer? if let signature = reader.readInt32() { @@ -3321,6 +3575,19 @@ public extension Api { return nil } } + public static func parse_updateNewQuickReply(_ reader: BufferReader) -> Update? { + var _1: Api.QuickReply? + if let signature = reader.readInt32() { + _1 = Api.parse(reader, signature: signature) as? Api.QuickReply + } + let _c1 = _1 != nil + if _c1 { + return Api.Update.updateNewQuickReply(quickReply: _1!) + } + else { + return nil + } + } public static func parse_updateNewScheduledMessage(_ reader: BufferReader) -> Update? { var _1: Api.Message? if let signature = reader.readInt32() { @@ -3608,6 +3875,32 @@ public extension Api { public static func parse_updatePtsChanged(_ reader: BufferReader) -> Update? { return Api.Update.updatePtsChanged } + public static func parse_updateQuickReplies(_ reader: BufferReader) -> Update? { + var _1: [Api.QuickReply]? + if let _ = reader.readInt32() { + _1 = Api.parseVector(reader, elementSignature: 0, elementType: Api.QuickReply.self) + } + let _c1 = _1 != nil + if _c1 { + return Api.Update.updateQuickReplies(quickReplies: _1!) + } + else { + return nil + } + } + public static func parse_updateQuickReplyMessage(_ reader: BufferReader) -> Update? { + var _1: Api.Message? + if let signature = reader.readInt32() { + _1 = Api.parse(reader, signature: signature) as? Api.Message + } + let _c1 = _1 != nil + if _c1 { + return Api.Update.updateQuickReplyMessage(message: _1!) + } + else { + return nil + } + } public static func parse_updateReadChannelDiscussionInbox(_ reader: BufferReader) -> Update? { var _1: Int32? _1 = reader.readInt32() @@ -3876,6 +4169,17 @@ public extension Api { return nil } } + public static func parse_updateSmsJob(_ reader: BufferReader) -> Update? { + var _1: String? + _1 = parseString(reader) + let _c1 = _1 != nil + if _c1 { + return Api.Update.updateSmsJob(jobId: _1!) + } + else { + return nil + } + } public static func parse_updateStickerSets(_ reader: BufferReader) -> Update? { var _1: Int32? _1 = reader.readInt32() diff --git a/submodules/TelegramApi/Sources/Api23.swift b/submodules/TelegramApi/Sources/Api23.swift index 2d684d02d37..12f4890d7c7 100644 --- a/submodules/TelegramApi/Sources/Api23.swift +++ b/submodules/TelegramApi/Sources/Api23.swift @@ -602,15 +602,16 @@ public extension Api { } public extension Api { enum UserFull: TypeConstructorDescription { - case userFull(flags: Int32, id: Int64, about: String?, settings: Api.PeerSettings, personalPhoto: Api.Photo?, profilePhoto: Api.Photo?, fallbackPhoto: Api.Photo?, notifySettings: Api.PeerNotifySettings, botInfo: Api.BotInfo?, pinnedMsgId: Int32?, commonChatsCount: Int32, folderId: Int32?, ttlPeriod: Int32?, themeEmoticon: String?, privateForwardName: String?, botGroupAdminRights: Api.ChatAdminRights?, botBroadcastAdminRights: Api.ChatAdminRights?, premiumGifts: [Api.PremiumGiftOption]?, wallpaper: Api.WallPaper?, stories: Api.PeerStories?) + case userFull(flags: Int32, flags2: Int32, id: Int64, about: String?, settings: Api.PeerSettings, personalPhoto: Api.Photo?, profilePhoto: Api.Photo?, fallbackPhoto: Api.Photo?, notifySettings: Api.PeerNotifySettings, botInfo: Api.BotInfo?, pinnedMsgId: Int32?, commonChatsCount: Int32, folderId: Int32?, ttlPeriod: Int32?, themeEmoticon: String?, privateForwardName: String?, botGroupAdminRights: Api.ChatAdminRights?, botBroadcastAdminRights: Api.ChatAdminRights?, premiumGifts: [Api.PremiumGiftOption]?, wallpaper: Api.WallPaper?, stories: Api.PeerStories?, businessWorkHours: Api.BusinessWorkHours?, businessLocation: Api.BusinessLocation?, businessGreetingMessage: Api.BusinessGreetingMessage?, businessAwayMessage: Api.BusinessAwayMessage?) public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { switch self { - case .userFull(let flags, let id, let about, let settings, let personalPhoto, let profilePhoto, let fallbackPhoto, let notifySettings, let botInfo, let pinnedMsgId, let commonChatsCount, let folderId, let ttlPeriod, let themeEmoticon, let privateForwardName, let botGroupAdminRights, let botBroadcastAdminRights, let premiumGifts, let wallpaper, let stories): + case .userFull(let flags, let flags2, let id, let about, let settings, let personalPhoto, let profilePhoto, let fallbackPhoto, let notifySettings, let botInfo, let pinnedMsgId, let commonChatsCount, let folderId, let ttlPeriod, let themeEmoticon, let privateForwardName, let botGroupAdminRights, let botBroadcastAdminRights, let premiumGifts, let wallpaper, let stories, let businessWorkHours, let businessLocation, let businessGreetingMessage, let businessAwayMessage): if boxed { - buffer.appendInt32(-1179571092) + buffer.appendInt32(587153029) } serializeInt32(flags, buffer: buffer, boxed: false) + serializeInt32(flags2, buffer: buffer, boxed: false) serializeInt64(id, buffer: buffer, boxed: false) if Int(flags) & Int(1 << 1) != 0 {serializeString(about!, buffer: buffer, boxed: false)} settings.serialize(buffer, true) @@ -634,102 +635,129 @@ public extension Api { }} if Int(flags) & Int(1 << 24) != 0 {wallpaper!.serialize(buffer, true)} if Int(flags) & Int(1 << 25) != 0 {stories!.serialize(buffer, true)} + if Int(flags2) & Int(1 << 0) != 0 {businessWorkHours!.serialize(buffer, true)} + if Int(flags2) & Int(1 << 1) != 0 {businessLocation!.serialize(buffer, true)} + if Int(flags2) & Int(1 << 2) != 0 {businessGreetingMessage!.serialize(buffer, true)} + if Int(flags2) & Int(1 << 3) != 0 {businessAwayMessage!.serialize(buffer, true)} break } } public func descriptionFields() -> (String, [(String, Any)]) { switch self { - case .userFull(let flags, let id, let about, let settings, let personalPhoto, let profilePhoto, let fallbackPhoto, let notifySettings, let botInfo, let pinnedMsgId, let commonChatsCount, let folderId, let ttlPeriod, let themeEmoticon, let privateForwardName, let botGroupAdminRights, let botBroadcastAdminRights, let premiumGifts, let wallpaper, let stories): - return ("userFull", [("flags", flags as Any), ("id", id as Any), ("about", about as Any), ("settings", settings as Any), ("personalPhoto", personalPhoto as Any), ("profilePhoto", profilePhoto as Any), ("fallbackPhoto", fallbackPhoto as Any), ("notifySettings", notifySettings as Any), ("botInfo", botInfo as Any), ("pinnedMsgId", pinnedMsgId as Any), ("commonChatsCount", commonChatsCount as Any), ("folderId", folderId as Any), ("ttlPeriod", ttlPeriod as Any), ("themeEmoticon", themeEmoticon as Any), ("privateForwardName", privateForwardName as Any), ("botGroupAdminRights", botGroupAdminRights as Any), ("botBroadcastAdminRights", botBroadcastAdminRights as Any), ("premiumGifts", premiumGifts as Any), ("wallpaper", wallpaper as Any), ("stories", stories as Any)]) + case .userFull(let flags, let flags2, let id, let about, let settings, let personalPhoto, let profilePhoto, let fallbackPhoto, let notifySettings, let botInfo, let pinnedMsgId, let commonChatsCount, let folderId, let ttlPeriod, let themeEmoticon, let privateForwardName, let botGroupAdminRights, let botBroadcastAdminRights, let premiumGifts, let wallpaper, let stories, let businessWorkHours, let businessLocation, let businessGreetingMessage, let businessAwayMessage): + return ("userFull", [("flags", flags as Any), ("flags2", flags2 as Any), ("id", id as Any), ("about", about as Any), ("settings", settings as Any), ("personalPhoto", personalPhoto as Any), ("profilePhoto", profilePhoto as Any), ("fallbackPhoto", fallbackPhoto as Any), ("notifySettings", notifySettings as Any), ("botInfo", botInfo as Any), ("pinnedMsgId", pinnedMsgId as Any), ("commonChatsCount", commonChatsCount as Any), ("folderId", folderId as Any), ("ttlPeriod", ttlPeriod as Any), ("themeEmoticon", themeEmoticon as Any), ("privateForwardName", privateForwardName as Any), ("botGroupAdminRights", botGroupAdminRights as Any), ("botBroadcastAdminRights", botBroadcastAdminRights as Any), ("premiumGifts", premiumGifts as Any), ("wallpaper", wallpaper as Any), ("stories", stories as Any), ("businessWorkHours", businessWorkHours as Any), ("businessLocation", businessLocation as Any), ("businessGreetingMessage", businessGreetingMessage as Any), ("businessAwayMessage", businessAwayMessage as Any)]) } } public static func parse_userFull(_ reader: BufferReader) -> UserFull? { var _1: Int32? _1 = reader.readInt32() - var _2: Int64? - _2 = reader.readInt64() - var _3: String? - if Int(_1!) & Int(1 << 1) != 0 {_3 = parseString(reader) } - var _4: Api.PeerSettings? + var _2: Int32? + _2 = reader.readInt32() + var _3: Int64? + _3 = reader.readInt64() + var _4: String? + if Int(_1!) & Int(1 << 1) != 0 {_4 = parseString(reader) } + var _5: Api.PeerSettings? if let signature = reader.readInt32() { - _4 = Api.parse(reader, signature: signature) as? Api.PeerSettings + _5 = Api.parse(reader, signature: signature) as? Api.PeerSettings } - var _5: Api.Photo? - if Int(_1!) & Int(1 << 21) != 0 {if let signature = reader.readInt32() { - _5 = Api.parse(reader, signature: signature) as? Api.Photo - } } var _6: Api.Photo? - if Int(_1!) & Int(1 << 2) != 0 {if let signature = reader.readInt32() { + if Int(_1!) & Int(1 << 21) != 0 {if let signature = reader.readInt32() { _6 = Api.parse(reader, signature: signature) as? Api.Photo } } var _7: Api.Photo? - if Int(_1!) & Int(1 << 22) != 0 {if let signature = reader.readInt32() { + if Int(_1!) & Int(1 << 2) != 0 {if let signature = reader.readInt32() { _7 = Api.parse(reader, signature: signature) as? Api.Photo } } - var _8: Api.PeerNotifySettings? + var _8: Api.Photo? + if Int(_1!) & Int(1 << 22) != 0 {if let signature = reader.readInt32() { + _8 = Api.parse(reader, signature: signature) as? Api.Photo + } } + var _9: Api.PeerNotifySettings? if let signature = reader.readInt32() { - _8 = Api.parse(reader, signature: signature) as? Api.PeerNotifySettings + _9 = Api.parse(reader, signature: signature) as? Api.PeerNotifySettings } - var _9: Api.BotInfo? + var _10: Api.BotInfo? if Int(_1!) & Int(1 << 3) != 0 {if let signature = reader.readInt32() { - _9 = Api.parse(reader, signature: signature) as? Api.BotInfo + _10 = Api.parse(reader, signature: signature) as? Api.BotInfo } } - var _10: Int32? - if Int(_1!) & Int(1 << 6) != 0 {_10 = reader.readInt32() } var _11: Int32? - _11 = reader.readInt32() + if Int(_1!) & Int(1 << 6) != 0 {_11 = reader.readInt32() } var _12: Int32? - if Int(_1!) & Int(1 << 11) != 0 {_12 = reader.readInt32() } + _12 = reader.readInt32() var _13: Int32? - if Int(_1!) & Int(1 << 14) != 0 {_13 = reader.readInt32() } - var _14: String? - if Int(_1!) & Int(1 << 15) != 0 {_14 = parseString(reader) } + if Int(_1!) & Int(1 << 11) != 0 {_13 = reader.readInt32() } + var _14: Int32? + if Int(_1!) & Int(1 << 14) != 0 {_14 = reader.readInt32() } var _15: String? - if Int(_1!) & Int(1 << 16) != 0 {_15 = parseString(reader) } - var _16: Api.ChatAdminRights? + if Int(_1!) & Int(1 << 15) != 0 {_15 = parseString(reader) } + var _16: String? + if Int(_1!) & Int(1 << 16) != 0 {_16 = parseString(reader) } + var _17: Api.ChatAdminRights? if Int(_1!) & Int(1 << 17) != 0 {if let signature = reader.readInt32() { - _16 = Api.parse(reader, signature: signature) as? Api.ChatAdminRights + _17 = Api.parse(reader, signature: signature) as? Api.ChatAdminRights } } - var _17: Api.ChatAdminRights? + var _18: Api.ChatAdminRights? if Int(_1!) & Int(1 << 18) != 0 {if let signature = reader.readInt32() { - _17 = Api.parse(reader, signature: signature) as? Api.ChatAdminRights + _18 = Api.parse(reader, signature: signature) as? Api.ChatAdminRights } } - var _18: [Api.PremiumGiftOption]? + var _19: [Api.PremiumGiftOption]? if Int(_1!) & Int(1 << 19) != 0 {if let _ = reader.readInt32() { - _18 = Api.parseVector(reader, elementSignature: 0, elementType: Api.PremiumGiftOption.self) + _19 = Api.parseVector(reader, elementSignature: 0, elementType: Api.PremiumGiftOption.self) } } - var _19: Api.WallPaper? + var _20: Api.WallPaper? if Int(_1!) & Int(1 << 24) != 0 {if let signature = reader.readInt32() { - _19 = Api.parse(reader, signature: signature) as? Api.WallPaper + _20 = Api.parse(reader, signature: signature) as? Api.WallPaper } } - var _20: Api.PeerStories? + var _21: Api.PeerStories? if Int(_1!) & Int(1 << 25) != 0 {if let signature = reader.readInt32() { - _20 = Api.parse(reader, signature: signature) as? Api.PeerStories + _21 = Api.parse(reader, signature: signature) as? Api.PeerStories + } } + var _22: Api.BusinessWorkHours? + if Int(_2!) & Int(1 << 0) != 0 {if let signature = reader.readInt32() { + _22 = Api.parse(reader, signature: signature) as? Api.BusinessWorkHours + } } + var _23: Api.BusinessLocation? + if Int(_2!) & Int(1 << 1) != 0 {if let signature = reader.readInt32() { + _23 = Api.parse(reader, signature: signature) as? Api.BusinessLocation + } } + var _24: Api.BusinessGreetingMessage? + if Int(_2!) & Int(1 << 2) != 0 {if let signature = reader.readInt32() { + _24 = Api.parse(reader, signature: signature) as? Api.BusinessGreetingMessage + } } + var _25: Api.BusinessAwayMessage? + if Int(_2!) & Int(1 << 3) != 0 {if let signature = reader.readInt32() { + _25 = Api.parse(reader, signature: signature) as? Api.BusinessAwayMessage } } let _c1 = _1 != nil let _c2 = _2 != nil - let _c3 = (Int(_1!) & Int(1 << 1) == 0) || _3 != nil - let _c4 = _4 != nil - let _c5 = (Int(_1!) & Int(1 << 21) == 0) || _5 != nil - let _c6 = (Int(_1!) & Int(1 << 2) == 0) || _6 != nil - let _c7 = (Int(_1!) & Int(1 << 22) == 0) || _7 != nil - let _c8 = _8 != nil - let _c9 = (Int(_1!) & Int(1 << 3) == 0) || _9 != nil - let _c10 = (Int(_1!) & Int(1 << 6) == 0) || _10 != nil - let _c11 = _11 != nil - let _c12 = (Int(_1!) & Int(1 << 11) == 0) || _12 != nil - let _c13 = (Int(_1!) & Int(1 << 14) == 0) || _13 != nil - let _c14 = (Int(_1!) & Int(1 << 15) == 0) || _14 != nil - let _c15 = (Int(_1!) & Int(1 << 16) == 0) || _15 != nil - let _c16 = (Int(_1!) & Int(1 << 17) == 0) || _16 != nil - let _c17 = (Int(_1!) & Int(1 << 18) == 0) || _17 != nil - let _c18 = (Int(_1!) & Int(1 << 19) == 0) || _18 != nil - let _c19 = (Int(_1!) & Int(1 << 24) == 0) || _19 != nil - let _c20 = (Int(_1!) & Int(1 << 25) == 0) || _20 != nil - if _c1 && _c2 && _c3 && _c4 && _c5 && _c6 && _c7 && _c8 && _c9 && _c10 && _c11 && _c12 && _c13 && _c14 && _c15 && _c16 && _c17 && _c18 && _c19 && _c20 { - return Api.UserFull.userFull(flags: _1!, id: _2!, about: _3, settings: _4!, personalPhoto: _5, profilePhoto: _6, fallbackPhoto: _7, notifySettings: _8!, botInfo: _9, pinnedMsgId: _10, commonChatsCount: _11!, folderId: _12, ttlPeriod: _13, themeEmoticon: _14, privateForwardName: _15, botGroupAdminRights: _16, botBroadcastAdminRights: _17, premiumGifts: _18, wallpaper: _19, stories: _20) + let _c3 = _3 != nil + let _c4 = (Int(_1!) & Int(1 << 1) == 0) || _4 != nil + let _c5 = _5 != nil + let _c6 = (Int(_1!) & Int(1 << 21) == 0) || _6 != nil + let _c7 = (Int(_1!) & Int(1 << 2) == 0) || _7 != nil + let _c8 = (Int(_1!) & Int(1 << 22) == 0) || _8 != nil + let _c9 = _9 != nil + let _c10 = (Int(_1!) & Int(1 << 3) == 0) || _10 != nil + let _c11 = (Int(_1!) & Int(1 << 6) == 0) || _11 != nil + let _c12 = _12 != nil + let _c13 = (Int(_1!) & Int(1 << 11) == 0) || _13 != nil + let _c14 = (Int(_1!) & Int(1 << 14) == 0) || _14 != nil + let _c15 = (Int(_1!) & Int(1 << 15) == 0) || _15 != nil + let _c16 = (Int(_1!) & Int(1 << 16) == 0) || _16 != nil + let _c17 = (Int(_1!) & Int(1 << 17) == 0) || _17 != nil + let _c18 = (Int(_1!) & Int(1 << 18) == 0) || _18 != nil + let _c19 = (Int(_1!) & Int(1 << 19) == 0) || _19 != nil + let _c20 = (Int(_1!) & Int(1 << 24) == 0) || _20 != nil + let _c21 = (Int(_1!) & Int(1 << 25) == 0) || _21 != nil + let _c22 = (Int(_2!) & Int(1 << 0) == 0) || _22 != nil + let _c23 = (Int(_2!) & Int(1 << 1) == 0) || _23 != nil + let _c24 = (Int(_2!) & Int(1 << 2) == 0) || _24 != nil + let _c25 = (Int(_2!) & Int(1 << 3) == 0) || _25 != nil + if _c1 && _c2 && _c3 && _c4 && _c5 && _c6 && _c7 && _c8 && _c9 && _c10 && _c11 && _c12 && _c13 && _c14 && _c15 && _c16 && _c17 && _c18 && _c19 && _c20 && _c21 && _c22 && _c23 && _c24 && _c25 { + return Api.UserFull.userFull(flags: _1!, flags2: _2!, id: _3!, about: _4, settings: _5!, personalPhoto: _6, profilePhoto: _7, fallbackPhoto: _8, notifySettings: _9!, botInfo: _10, pinnedMsgId: _11, commonChatsCount: _12!, folderId: _13, ttlPeriod: _14, themeEmoticon: _15, privateForwardName: _16, botGroupAdminRights: _17, botBroadcastAdminRights: _18, premiumGifts: _19, wallpaper: _20, stories: _21, businessWorkHours: _22, businessLocation: _23, businessGreetingMessage: _24, businessAwayMessage: _25) } else { return nil diff --git a/submodules/TelegramApi/Sources/Api24.swift b/submodules/TelegramApi/Sources/Api24.swift index 55a953bc119..1cac2119e83 100644 --- a/submodules/TelegramApi/Sources/Api24.swift +++ b/submodules/TelegramApi/Sources/Api24.swift @@ -424,6 +424,58 @@ public extension Api.account { } } +public extension Api.account { + enum ConnectedBots: TypeConstructorDescription { + case connectedBots(connectedBots: [Api.ConnectedBot], users: [Api.User]) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .connectedBots(let connectedBots, let users): + if boxed { + buffer.appendInt32(400029819) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(connectedBots.count)) + for item in connectedBots { + item.serialize(buffer, true) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(users.count)) + for item in users { + item.serialize(buffer, true) + } + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .connectedBots(let connectedBots, let users): + return ("connectedBots", [("connectedBots", connectedBots as Any), ("users", users as Any)]) + } + } + + public static func parse_connectedBots(_ reader: BufferReader) -> ConnectedBots? { + var _1: [Api.ConnectedBot]? + if let _ = reader.readInt32() { + _1 = Api.parseVector(reader, elementSignature: 0, elementType: Api.ConnectedBot.self) + } + var _2: [Api.User]? + if let _ = reader.readInt32() { + _2 = Api.parseVector(reader, elementSignature: 0, elementType: Api.User.self) + } + let _c1 = _1 != nil + let _c2 = _2 != nil + if _c1 && _c2 { + return Api.account.ConnectedBots.connectedBots(connectedBots: _1!, users: _2!) + } + else { + return nil + } + } + + } +} public extension Api.account { enum ContentSettings: TypeConstructorDescription { case contentSettings(flags: Int32) @@ -1238,55 +1290,3 @@ public extension Api.account { } } -public extension Api.account { - enum WebAuthorizations: TypeConstructorDescription { - case webAuthorizations(authorizations: [Api.WebAuthorization], users: [Api.User]) - - public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { - switch self { - case .webAuthorizations(let authorizations, let users): - if boxed { - buffer.appendInt32(-313079300) - } - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(authorizations.count)) - for item in authorizations { - item.serialize(buffer, true) - } - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(users.count)) - for item in users { - item.serialize(buffer, true) - } - break - } - } - - public func descriptionFields() -> (String, [(String, Any)]) { - switch self { - case .webAuthorizations(let authorizations, let users): - return ("webAuthorizations", [("authorizations", authorizations as Any), ("users", users as Any)]) - } - } - - public static func parse_webAuthorizations(_ reader: BufferReader) -> WebAuthorizations? { - var _1: [Api.WebAuthorization]? - if let _ = reader.readInt32() { - _1 = Api.parseVector(reader, elementSignature: 0, elementType: Api.WebAuthorization.self) - } - var _2: [Api.User]? - if let _ = reader.readInt32() { - _2 = Api.parseVector(reader, elementSignature: 0, elementType: Api.User.self) - } - let _c1 = _1 != nil - let _c2 = _2 != nil - if _c1 && _c2 { - return Api.account.WebAuthorizations.webAuthorizations(authorizations: _1!, users: _2!) - } - else { - return nil - } - } - - } -} diff --git a/submodules/TelegramApi/Sources/Api25.swift b/submodules/TelegramApi/Sources/Api25.swift index 95534e002c8..a204a5cc8ef 100644 --- a/submodules/TelegramApi/Sources/Api25.swift +++ b/submodules/TelegramApi/Sources/Api25.swift @@ -1,3 +1,55 @@ +public extension Api.account { + enum WebAuthorizations: TypeConstructorDescription { + case webAuthorizations(authorizations: [Api.WebAuthorization], users: [Api.User]) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .webAuthorizations(let authorizations, let users): + if boxed { + buffer.appendInt32(-313079300) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(authorizations.count)) + for item in authorizations { + item.serialize(buffer, true) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(users.count)) + for item in users { + item.serialize(buffer, true) + } + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .webAuthorizations(let authorizations, let users): + return ("webAuthorizations", [("authorizations", authorizations as Any), ("users", users as Any)]) + } + } + + public static func parse_webAuthorizations(_ reader: BufferReader) -> WebAuthorizations? { + var _1: [Api.WebAuthorization]? + if let _ = reader.readInt32() { + _1 = Api.parseVector(reader, elementSignature: 0, elementType: Api.WebAuthorization.self) + } + var _2: [Api.User]? + if let _ = reader.readInt32() { + _2 = Api.parseVector(reader, elementSignature: 0, elementType: Api.User.self) + } + let _c1 = _1 != nil + let _c2 = _2 != nil + if _c1 && _c2 { + return Api.account.WebAuthorizations.webAuthorizations(authorizations: _1!, users: _2!) + } + else { + return nil + } + } + + } +} public extension Api.auth { enum Authorization: TypeConstructorDescription { case authorization(flags: Int32, otherwiseReloginDays: Int32?, tmpSessions: Int32?, futureAuthToken: Buffer?, user: Api.User) diff --git a/submodules/TelegramApi/Sources/Api27.swift b/submodules/TelegramApi/Sources/Api27.swift index 90373cb39cb..5069e6504ec 100644 --- a/submodules/TelegramApi/Sources/Api27.swift +++ b/submodules/TelegramApi/Sources/Api27.swift @@ -354,6 +354,64 @@ public extension Api.help { } } +public extension Api.help { + enum TimezonesList: TypeConstructorDescription { + case timezonesList(timezones: [Api.Timezone], hash: Int32) + case timezonesListNotModified + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .timezonesList(let timezones, let hash): + if boxed { + buffer.appendInt32(2071260529) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(timezones.count)) + for item in timezones { + item.serialize(buffer, true) + } + serializeInt32(hash, buffer: buffer, boxed: false) + break + case .timezonesListNotModified: + if boxed { + buffer.appendInt32(-1761146676) + } + + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .timezonesList(let timezones, let hash): + return ("timezonesList", [("timezones", timezones as Any), ("hash", hash as Any)]) + case .timezonesListNotModified: + return ("timezonesListNotModified", []) + } + } + + public static func parse_timezonesList(_ reader: BufferReader) -> TimezonesList? { + var _1: [Api.Timezone]? + if let _ = reader.readInt32() { + _1 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Timezone.self) + } + var _2: Int32? + _2 = reader.readInt32() + let _c1 = _1 != nil + let _c2 = _2 != nil + if _c1 && _c2 { + return Api.help.TimezonesList.timezonesList(timezones: _1!, hash: _2!) + } + else { + return nil + } + } + public static func parse_timezonesListNotModified(_ reader: BufferReader) -> TimezonesList? { + return Api.help.TimezonesList.timezonesListNotModified + } + + } +} public extension Api.help { enum UserInfo: TypeConstructorDescription { case userInfo(message: String, entities: [Api.MessageEntity], author: String, date: Int32) @@ -1233,67 +1291,19 @@ public extension Api.messages { } } public extension Api.messages { - enum Dialogs: TypeConstructorDescription { - case dialogs(dialogs: [Api.Dialog], messages: [Api.Message], chats: [Api.Chat], users: [Api.User]) - case dialogsNotModified(count: Int32) - case dialogsSlice(count: Int32, dialogs: [Api.Dialog], messages: [Api.Message], chats: [Api.Chat], users: [Api.User]) + enum DialogFilters: TypeConstructorDescription { + case dialogFilters(flags: Int32, filters: [Api.DialogFilter]) public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { switch self { - case .dialogs(let dialogs, let messages, let chats, let users): + case .dialogFilters(let flags, let filters): if boxed { - buffer.appendInt32(364538944) - } - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(dialogs.count)) - for item in dialogs { - item.serialize(buffer, true) - } - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(messages.count)) - for item in messages { - item.serialize(buffer, true) - } - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(chats.count)) - for item in chats { - item.serialize(buffer, true) - } - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(users.count)) - for item in users { - item.serialize(buffer, true) - } - break - case .dialogsNotModified(let count): - if boxed { - buffer.appendInt32(-253500010) - } - serializeInt32(count, buffer: buffer, boxed: false) - break - case .dialogsSlice(let count, let dialogs, let messages, let chats, let users): - if boxed { - buffer.appendInt32(1910543603) - } - serializeInt32(count, buffer: buffer, boxed: false) - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(dialogs.count)) - for item in dialogs { - item.serialize(buffer, true) - } - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(messages.count)) - for item in messages { - item.serialize(buffer, true) - } - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(chats.count)) - for item in chats { - item.serialize(buffer, true) + buffer.appendInt32(718878489) } + serializeInt32(flags, buffer: buffer, boxed: false) buffer.appendInt32(481674261) - buffer.appendInt32(Int32(users.count)) - for item in users { + buffer.appendInt32(Int32(filters.count)) + for item in filters { item.serialize(buffer, true) } break @@ -1302,80 +1312,22 @@ public extension Api.messages { public func descriptionFields() -> (String, [(String, Any)]) { switch self { - case .dialogs(let dialogs, let messages, let chats, let users): - return ("dialogs", [("dialogs", dialogs as Any), ("messages", messages as Any), ("chats", chats as Any), ("users", users as Any)]) - case .dialogsNotModified(let count): - return ("dialogsNotModified", [("count", count as Any)]) - case .dialogsSlice(let count, let dialogs, let messages, let chats, let users): - return ("dialogsSlice", [("count", count as Any), ("dialogs", dialogs as Any), ("messages", messages as Any), ("chats", chats as Any), ("users", users as Any)]) + case .dialogFilters(let flags, let filters): + return ("dialogFilters", [("flags", flags as Any), ("filters", filters as Any)]) } } - public static func parse_dialogs(_ reader: BufferReader) -> Dialogs? { - var _1: [Api.Dialog]? - if let _ = reader.readInt32() { - _1 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Dialog.self) - } - var _2: [Api.Message]? - if let _ = reader.readInt32() { - _2 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Message.self) - } - var _3: [Api.Chat]? - if let _ = reader.readInt32() { - _3 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Chat.self) - } - var _4: [Api.User]? - if let _ = reader.readInt32() { - _4 = Api.parseVector(reader, elementSignature: 0, elementType: Api.User.self) - } - let _c1 = _1 != nil - let _c2 = _2 != nil - let _c3 = _3 != nil - let _c4 = _4 != nil - if _c1 && _c2 && _c3 && _c4 { - return Api.messages.Dialogs.dialogs(dialogs: _1!, messages: _2!, chats: _3!, users: _4!) - } - else { - return nil - } - } - public static func parse_dialogsNotModified(_ reader: BufferReader) -> Dialogs? { - var _1: Int32? - _1 = reader.readInt32() - let _c1 = _1 != nil - if _c1 { - return Api.messages.Dialogs.dialogsNotModified(count: _1!) - } - else { - return nil - } - } - public static func parse_dialogsSlice(_ reader: BufferReader) -> Dialogs? { + public static func parse_dialogFilters(_ reader: BufferReader) -> DialogFilters? { var _1: Int32? _1 = reader.readInt32() - var _2: [Api.Dialog]? + var _2: [Api.DialogFilter]? if let _ = reader.readInt32() { - _2 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Dialog.self) - } - var _3: [Api.Message]? - if let _ = reader.readInt32() { - _3 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Message.self) - } - var _4: [Api.Chat]? - if let _ = reader.readInt32() { - _4 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Chat.self) - } - var _5: [Api.User]? - if let _ = reader.readInt32() { - _5 = Api.parseVector(reader, elementSignature: 0, elementType: Api.User.self) + _2 = Api.parseVector(reader, elementSignature: 0, elementType: Api.DialogFilter.self) } let _c1 = _1 != nil let _c2 = _2 != nil - let _c3 = _3 != nil - let _c4 = _4 != nil - let _c5 = _5 != nil - if _c1 && _c2 && _c3 && _c4 && _c5 { - return Api.messages.Dialogs.dialogsSlice(count: _1!, dialogs: _2!, messages: _3!, chats: _4!, users: _5!) + if _c1 && _c2 { + return Api.messages.DialogFilters.dialogFilters(flags: _1!, filters: _2!) } else { return nil diff --git a/submodules/TelegramApi/Sources/Api28.swift b/submodules/TelegramApi/Sources/Api28.swift index 639e18a7b26..cb218918b14 100644 --- a/submodules/TelegramApi/Sources/Api28.swift +++ b/submodules/TelegramApi/Sources/Api28.swift @@ -1,3 +1,155 @@ +public extension Api.messages { + enum Dialogs: TypeConstructorDescription { + case dialogs(dialogs: [Api.Dialog], messages: [Api.Message], chats: [Api.Chat], users: [Api.User]) + case dialogsNotModified(count: Int32) + case dialogsSlice(count: Int32, dialogs: [Api.Dialog], messages: [Api.Message], chats: [Api.Chat], users: [Api.User]) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .dialogs(let dialogs, let messages, let chats, let users): + if boxed { + buffer.appendInt32(364538944) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(dialogs.count)) + for item in dialogs { + item.serialize(buffer, true) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(messages.count)) + for item in messages { + item.serialize(buffer, true) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(chats.count)) + for item in chats { + item.serialize(buffer, true) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(users.count)) + for item in users { + item.serialize(buffer, true) + } + break + case .dialogsNotModified(let count): + if boxed { + buffer.appendInt32(-253500010) + } + serializeInt32(count, buffer: buffer, boxed: false) + break + case .dialogsSlice(let count, let dialogs, let messages, let chats, let users): + if boxed { + buffer.appendInt32(1910543603) + } + serializeInt32(count, buffer: buffer, boxed: false) + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(dialogs.count)) + for item in dialogs { + item.serialize(buffer, true) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(messages.count)) + for item in messages { + item.serialize(buffer, true) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(chats.count)) + for item in chats { + item.serialize(buffer, true) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(users.count)) + for item in users { + item.serialize(buffer, true) + } + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .dialogs(let dialogs, let messages, let chats, let users): + return ("dialogs", [("dialogs", dialogs as Any), ("messages", messages as Any), ("chats", chats as Any), ("users", users as Any)]) + case .dialogsNotModified(let count): + return ("dialogsNotModified", [("count", count as Any)]) + case .dialogsSlice(let count, let dialogs, let messages, let chats, let users): + return ("dialogsSlice", [("count", count as Any), ("dialogs", dialogs as Any), ("messages", messages as Any), ("chats", chats as Any), ("users", users as Any)]) + } + } + + public static func parse_dialogs(_ reader: BufferReader) -> Dialogs? { + var _1: [Api.Dialog]? + if let _ = reader.readInt32() { + _1 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Dialog.self) + } + var _2: [Api.Message]? + if let _ = reader.readInt32() { + _2 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Message.self) + } + var _3: [Api.Chat]? + if let _ = reader.readInt32() { + _3 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Chat.self) + } + var _4: [Api.User]? + if let _ = reader.readInt32() { + _4 = Api.parseVector(reader, elementSignature: 0, elementType: Api.User.self) + } + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + let _c4 = _4 != nil + if _c1 && _c2 && _c3 && _c4 { + return Api.messages.Dialogs.dialogs(dialogs: _1!, messages: _2!, chats: _3!, users: _4!) + } + else { + return nil + } + } + public static func parse_dialogsNotModified(_ reader: BufferReader) -> Dialogs? { + var _1: Int32? + _1 = reader.readInt32() + let _c1 = _1 != nil + if _c1 { + return Api.messages.Dialogs.dialogsNotModified(count: _1!) + } + else { + return nil + } + } + public static func parse_dialogsSlice(_ reader: BufferReader) -> Dialogs? { + var _1: Int32? + _1 = reader.readInt32() + var _2: [Api.Dialog]? + if let _ = reader.readInt32() { + _2 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Dialog.self) + } + var _3: [Api.Message]? + if let _ = reader.readInt32() { + _3 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Message.self) + } + var _4: [Api.Chat]? + if let _ = reader.readInt32() { + _4 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Chat.self) + } + var _5: [Api.User]? + if let _ = reader.readInt32() { + _5 = Api.parseVector(reader, elementSignature: 0, elementType: Api.User.self) + } + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + let _c4 = _4 != nil + let _c5 = _5 != nil + if _c1 && _c2 && _c3 && _c4 && _c5 { + return Api.messages.Dialogs.dialogsSlice(count: _1!, dialogs: _2!, messages: _3!, chats: _4!, users: _5!) + } + else { + return nil + } + } + + } +} public extension Api.messages { enum DiscussionMessage: TypeConstructorDescription { case discussionMessage(flags: Int32, messages: [Api.Message], maxId: Int32?, readInboxMaxId: Int32?, readOutboxMaxId: Int32?, unreadCount: Int32, chats: [Api.Chat], users: [Api.User]) @@ -1289,94 +1441,40 @@ public extension Api.messages { } } public extension Api.messages { - enum Reactions: TypeConstructorDescription { - case reactions(hash: Int64, reactions: [Api.Reaction]) - case reactionsNotModified + enum QuickReplies: TypeConstructorDescription { + case quickReplies(quickReplies: [Api.QuickReply], messages: [Api.Message], chats: [Api.Chat], users: [Api.User]) + case quickRepliesNotModified public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { switch self { - case .reactions(let hash, let reactions): + case .quickReplies(let quickReplies, let messages, let chats, let users): if boxed { - buffer.appendInt32(-352454890) + buffer.appendInt32(-963811691) } - serializeInt64(hash, buffer: buffer, boxed: false) buffer.appendInt32(481674261) - buffer.appendInt32(Int32(reactions.count)) - for item in reactions { + buffer.appendInt32(Int32(quickReplies.count)) + for item in quickReplies { item.serialize(buffer, true) } - break - case .reactionsNotModified: - if boxed { - buffer.appendInt32(-1334846497) - } - - break - } - } - - public func descriptionFields() -> (String, [(String, Any)]) { - switch self { - case .reactions(let hash, let reactions): - return ("reactions", [("hash", hash as Any), ("reactions", reactions as Any)]) - case .reactionsNotModified: - return ("reactionsNotModified", []) - } - } - - public static func parse_reactions(_ reader: BufferReader) -> Reactions? { - var _1: Int64? - _1 = reader.readInt64() - var _2: [Api.Reaction]? - if let _ = reader.readInt32() { - _2 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Reaction.self) - } - let _c1 = _1 != nil - let _c2 = _2 != nil - if _c1 && _c2 { - return Api.messages.Reactions.reactions(hash: _1!, reactions: _2!) - } - else { - return nil - } - } - public static func parse_reactionsNotModified(_ reader: BufferReader) -> Reactions? { - return Api.messages.Reactions.reactionsNotModified - } - - } -} -public extension Api.messages { - enum RecentStickers: TypeConstructorDescription { - case recentStickers(hash: Int64, packs: [Api.StickerPack], stickers: [Api.Document], dates: [Int32]) - case recentStickersNotModified - - public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { - switch self { - case .recentStickers(let hash, let packs, let stickers, let dates): - if boxed { - buffer.appendInt32(-1999405994) - } - serializeInt64(hash, buffer: buffer, boxed: false) buffer.appendInt32(481674261) - buffer.appendInt32(Int32(packs.count)) - for item in packs { + buffer.appendInt32(Int32(messages.count)) + for item in messages { item.serialize(buffer, true) } buffer.appendInt32(481674261) - buffer.appendInt32(Int32(stickers.count)) - for item in stickers { + buffer.appendInt32(Int32(chats.count)) + for item in chats { item.serialize(buffer, true) } buffer.appendInt32(481674261) - buffer.appendInt32(Int32(dates.count)) - for item in dates { - serializeInt32(item, buffer: buffer, boxed: false) + buffer.appendInt32(Int32(users.count)) + for item in users { + item.serialize(buffer, true) } break - case .recentStickersNotModified: + case .quickRepliesNotModified: if boxed { - buffer.appendInt32(186120336) + buffer.appendInt32(1603398491) } break @@ -1385,193 +1483,101 @@ public extension Api.messages { public func descriptionFields() -> (String, [(String, Any)]) { switch self { - case .recentStickers(let hash, let packs, let stickers, let dates): - return ("recentStickers", [("hash", hash as Any), ("packs", packs as Any), ("stickers", stickers as Any), ("dates", dates as Any)]) - case .recentStickersNotModified: - return ("recentStickersNotModified", []) + case .quickReplies(let quickReplies, let messages, let chats, let users): + return ("quickReplies", [("quickReplies", quickReplies as Any), ("messages", messages as Any), ("chats", chats as Any), ("users", users as Any)]) + case .quickRepliesNotModified: + return ("quickRepliesNotModified", []) } } - public static func parse_recentStickers(_ reader: BufferReader) -> RecentStickers? { - var _1: Int64? - _1 = reader.readInt64() - var _2: [Api.StickerPack]? + public static func parse_quickReplies(_ reader: BufferReader) -> QuickReplies? { + var _1: [Api.QuickReply]? if let _ = reader.readInt32() { - _2 = Api.parseVector(reader, elementSignature: 0, elementType: Api.StickerPack.self) + _1 = Api.parseVector(reader, elementSignature: 0, elementType: Api.QuickReply.self) } - var _3: [Api.Document]? + var _2: [Api.Message]? if let _ = reader.readInt32() { - _3 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Document.self) + _2 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Message.self) } - var _4: [Int32]? + var _3: [Api.Chat]? if let _ = reader.readInt32() { - _4 = Api.parseVector(reader, elementSignature: -1471112230, elementType: Int32.self) + _3 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Chat.self) + } + var _4: [Api.User]? + if let _ = reader.readInt32() { + _4 = Api.parseVector(reader, elementSignature: 0, elementType: Api.User.self) } let _c1 = _1 != nil let _c2 = _2 != nil let _c3 = _3 != nil let _c4 = _4 != nil if _c1 && _c2 && _c3 && _c4 { - return Api.messages.RecentStickers.recentStickers(hash: _1!, packs: _2!, stickers: _3!, dates: _4!) + return Api.messages.QuickReplies.quickReplies(quickReplies: _1!, messages: _2!, chats: _3!, users: _4!) } else { return nil } } - public static func parse_recentStickersNotModified(_ reader: BufferReader) -> RecentStickers? { - return Api.messages.RecentStickers.recentStickersNotModified + public static func parse_quickRepliesNotModified(_ reader: BufferReader) -> QuickReplies? { + return Api.messages.QuickReplies.quickRepliesNotModified } } } public extension Api.messages { - enum SavedDialogs: TypeConstructorDescription { - case savedDialogs(dialogs: [Api.SavedDialog], messages: [Api.Message], chats: [Api.Chat], users: [Api.User]) - case savedDialogsNotModified(count: Int32) - case savedDialogsSlice(count: Int32, dialogs: [Api.SavedDialog], messages: [Api.Message], chats: [Api.Chat], users: [Api.User]) + enum Reactions: TypeConstructorDescription { + case reactions(hash: Int64, reactions: [Api.Reaction]) + case reactionsNotModified public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { switch self { - case .savedDialogs(let dialogs, let messages, let chats, let users): + case .reactions(let hash, let reactions): if boxed { - buffer.appendInt32(-130358751) - } - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(dialogs.count)) - for item in dialogs { - item.serialize(buffer, true) - } - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(messages.count)) - for item in messages { - item.serialize(buffer, true) - } - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(chats.count)) - for item in chats { - item.serialize(buffer, true) + buffer.appendInt32(-352454890) } + serializeInt64(hash, buffer: buffer, boxed: false) buffer.appendInt32(481674261) - buffer.appendInt32(Int32(users.count)) - for item in users { + buffer.appendInt32(Int32(reactions.count)) + for item in reactions { item.serialize(buffer, true) } break - case .savedDialogsNotModified(let count): - if boxed { - buffer.appendInt32(-1071681560) - } - serializeInt32(count, buffer: buffer, boxed: false) - break - case .savedDialogsSlice(let count, let dialogs, let messages, let chats, let users): + case .reactionsNotModified: if boxed { - buffer.appendInt32(1153080793) - } - serializeInt32(count, buffer: buffer, boxed: false) - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(dialogs.count)) - for item in dialogs { - item.serialize(buffer, true) - } - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(messages.count)) - for item in messages { - item.serialize(buffer, true) - } - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(chats.count)) - for item in chats { - item.serialize(buffer, true) - } - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(users.count)) - for item in users { - item.serialize(buffer, true) + buffer.appendInt32(-1334846497) } + break } } public func descriptionFields() -> (String, [(String, Any)]) { switch self { - case .savedDialogs(let dialogs, let messages, let chats, let users): - return ("savedDialogs", [("dialogs", dialogs as Any), ("messages", messages as Any), ("chats", chats as Any), ("users", users as Any)]) - case .savedDialogsNotModified(let count): - return ("savedDialogsNotModified", [("count", count as Any)]) - case .savedDialogsSlice(let count, let dialogs, let messages, let chats, let users): - return ("savedDialogsSlice", [("count", count as Any), ("dialogs", dialogs as Any), ("messages", messages as Any), ("chats", chats as Any), ("users", users as Any)]) + case .reactions(let hash, let reactions): + return ("reactions", [("hash", hash as Any), ("reactions", reactions as Any)]) + case .reactionsNotModified: + return ("reactionsNotModified", []) } } - public static func parse_savedDialogs(_ reader: BufferReader) -> SavedDialogs? { - var _1: [Api.SavedDialog]? - if let _ = reader.readInt32() { - _1 = Api.parseVector(reader, elementSignature: 0, elementType: Api.SavedDialog.self) - } - var _2: [Api.Message]? - if let _ = reader.readInt32() { - _2 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Message.self) - } - var _3: [Api.Chat]? - if let _ = reader.readInt32() { - _3 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Chat.self) - } - var _4: [Api.User]? + public static func parse_reactions(_ reader: BufferReader) -> Reactions? { + var _1: Int64? + _1 = reader.readInt64() + var _2: [Api.Reaction]? if let _ = reader.readInt32() { - _4 = Api.parseVector(reader, elementSignature: 0, elementType: Api.User.self) + _2 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Reaction.self) } let _c1 = _1 != nil let _c2 = _2 != nil - let _c3 = _3 != nil - let _c4 = _4 != nil - if _c1 && _c2 && _c3 && _c4 { - return Api.messages.SavedDialogs.savedDialogs(dialogs: _1!, messages: _2!, chats: _3!, users: _4!) - } - else { - return nil - } - } - public static func parse_savedDialogsNotModified(_ reader: BufferReader) -> SavedDialogs? { - var _1: Int32? - _1 = reader.readInt32() - let _c1 = _1 != nil - if _c1 { - return Api.messages.SavedDialogs.savedDialogsNotModified(count: _1!) + if _c1 && _c2 { + return Api.messages.Reactions.reactions(hash: _1!, reactions: _2!) } else { return nil } } - public static func parse_savedDialogsSlice(_ reader: BufferReader) -> SavedDialogs? { - var _1: Int32? - _1 = reader.readInt32() - var _2: [Api.SavedDialog]? - if let _ = reader.readInt32() { - _2 = Api.parseVector(reader, elementSignature: 0, elementType: Api.SavedDialog.self) - } - var _3: [Api.Message]? - if let _ = reader.readInt32() { - _3 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Message.self) - } - var _4: [Api.Chat]? - if let _ = reader.readInt32() { - _4 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Chat.self) - } - var _5: [Api.User]? - if let _ = reader.readInt32() { - _5 = Api.parseVector(reader, elementSignature: 0, elementType: Api.User.self) - } - let _c1 = _1 != nil - let _c2 = _2 != nil - let _c3 = _3 != nil - let _c4 = _4 != nil - let _c5 = _5 != nil - if _c1 && _c2 && _c3 && _c4 && _c5 { - return Api.messages.SavedDialogs.savedDialogsSlice(count: _1!, dialogs: _2!, messages: _3!, chats: _4!, users: _5!) - } - else { - return nil - } + public static func parse_reactionsNotModified(_ reader: BufferReader) -> Reactions? { + return Api.messages.Reactions.reactionsNotModified } } diff --git a/submodules/TelegramApi/Sources/Api29.swift b/submodules/TelegramApi/Sources/Api29.swift index e19625e5582..fa6a3987160 100644 --- a/submodules/TelegramApi/Sources/Api29.swift +++ b/submodules/TelegramApi/Sources/Api29.swift @@ -1,3 +1,233 @@ +public extension Api.messages { + enum RecentStickers: TypeConstructorDescription { + case recentStickers(hash: Int64, packs: [Api.StickerPack], stickers: [Api.Document], dates: [Int32]) + case recentStickersNotModified + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .recentStickers(let hash, let packs, let stickers, let dates): + if boxed { + buffer.appendInt32(-1999405994) + } + serializeInt64(hash, buffer: buffer, boxed: false) + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(packs.count)) + for item in packs { + item.serialize(buffer, true) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(stickers.count)) + for item in stickers { + item.serialize(buffer, true) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(dates.count)) + for item in dates { + serializeInt32(item, buffer: buffer, boxed: false) + } + break + case .recentStickersNotModified: + if boxed { + buffer.appendInt32(186120336) + } + + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .recentStickers(let hash, let packs, let stickers, let dates): + return ("recentStickers", [("hash", hash as Any), ("packs", packs as Any), ("stickers", stickers as Any), ("dates", dates as Any)]) + case .recentStickersNotModified: + return ("recentStickersNotModified", []) + } + } + + public static func parse_recentStickers(_ reader: BufferReader) -> RecentStickers? { + var _1: Int64? + _1 = reader.readInt64() + var _2: [Api.StickerPack]? + if let _ = reader.readInt32() { + _2 = Api.parseVector(reader, elementSignature: 0, elementType: Api.StickerPack.self) + } + var _3: [Api.Document]? + if let _ = reader.readInt32() { + _3 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Document.self) + } + var _4: [Int32]? + if let _ = reader.readInt32() { + _4 = Api.parseVector(reader, elementSignature: -1471112230, elementType: Int32.self) + } + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + let _c4 = _4 != nil + if _c1 && _c2 && _c3 && _c4 { + return Api.messages.RecentStickers.recentStickers(hash: _1!, packs: _2!, stickers: _3!, dates: _4!) + } + else { + return nil + } + } + public static func parse_recentStickersNotModified(_ reader: BufferReader) -> RecentStickers? { + return Api.messages.RecentStickers.recentStickersNotModified + } + + } +} +public extension Api.messages { + enum SavedDialogs: TypeConstructorDescription { + case savedDialogs(dialogs: [Api.SavedDialog], messages: [Api.Message], chats: [Api.Chat], users: [Api.User]) + case savedDialogsNotModified(count: Int32) + case savedDialogsSlice(count: Int32, dialogs: [Api.SavedDialog], messages: [Api.Message], chats: [Api.Chat], users: [Api.User]) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .savedDialogs(let dialogs, let messages, let chats, let users): + if boxed { + buffer.appendInt32(-130358751) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(dialogs.count)) + for item in dialogs { + item.serialize(buffer, true) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(messages.count)) + for item in messages { + item.serialize(buffer, true) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(chats.count)) + for item in chats { + item.serialize(buffer, true) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(users.count)) + for item in users { + item.serialize(buffer, true) + } + break + case .savedDialogsNotModified(let count): + if boxed { + buffer.appendInt32(-1071681560) + } + serializeInt32(count, buffer: buffer, boxed: false) + break + case .savedDialogsSlice(let count, let dialogs, let messages, let chats, let users): + if boxed { + buffer.appendInt32(1153080793) + } + serializeInt32(count, buffer: buffer, boxed: false) + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(dialogs.count)) + for item in dialogs { + item.serialize(buffer, true) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(messages.count)) + for item in messages { + item.serialize(buffer, true) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(chats.count)) + for item in chats { + item.serialize(buffer, true) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(users.count)) + for item in users { + item.serialize(buffer, true) + } + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .savedDialogs(let dialogs, let messages, let chats, let users): + return ("savedDialogs", [("dialogs", dialogs as Any), ("messages", messages as Any), ("chats", chats as Any), ("users", users as Any)]) + case .savedDialogsNotModified(let count): + return ("savedDialogsNotModified", [("count", count as Any)]) + case .savedDialogsSlice(let count, let dialogs, let messages, let chats, let users): + return ("savedDialogsSlice", [("count", count as Any), ("dialogs", dialogs as Any), ("messages", messages as Any), ("chats", chats as Any), ("users", users as Any)]) + } + } + + public static func parse_savedDialogs(_ reader: BufferReader) -> SavedDialogs? { + var _1: [Api.SavedDialog]? + if let _ = reader.readInt32() { + _1 = Api.parseVector(reader, elementSignature: 0, elementType: Api.SavedDialog.self) + } + var _2: [Api.Message]? + if let _ = reader.readInt32() { + _2 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Message.self) + } + var _3: [Api.Chat]? + if let _ = reader.readInt32() { + _3 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Chat.self) + } + var _4: [Api.User]? + if let _ = reader.readInt32() { + _4 = Api.parseVector(reader, elementSignature: 0, elementType: Api.User.self) + } + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + let _c4 = _4 != nil + if _c1 && _c2 && _c3 && _c4 { + return Api.messages.SavedDialogs.savedDialogs(dialogs: _1!, messages: _2!, chats: _3!, users: _4!) + } + else { + return nil + } + } + public static func parse_savedDialogsNotModified(_ reader: BufferReader) -> SavedDialogs? { + var _1: Int32? + _1 = reader.readInt32() + let _c1 = _1 != nil + if _c1 { + return Api.messages.SavedDialogs.savedDialogsNotModified(count: _1!) + } + else { + return nil + } + } + public static func parse_savedDialogsSlice(_ reader: BufferReader) -> SavedDialogs? { + var _1: Int32? + _1 = reader.readInt32() + var _2: [Api.SavedDialog]? + if let _ = reader.readInt32() { + _2 = Api.parseVector(reader, elementSignature: 0, elementType: Api.SavedDialog.self) + } + var _3: [Api.Message]? + if let _ = reader.readInt32() { + _3 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Message.self) + } + var _4: [Api.Chat]? + if let _ = reader.readInt32() { + _4 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Chat.self) + } + var _5: [Api.User]? + if let _ = reader.readInt32() { + _5 = Api.parseVector(reader, elementSignature: 0, elementType: Api.User.self) + } + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + let _c4 = _4 != nil + let _c5 = _5 != nil + if _c1 && _c2 && _c3 && _c4 && _c5 { + return Api.messages.SavedDialogs.savedDialogsSlice(count: _1!, dialogs: _2!, messages: _3!, chats: _4!, users: _5!) + } + else { + return nil + } + } + + } +} public extension Api.messages { enum SavedGifs: TypeConstructorDescription { case savedGifs(hash: Int64, gifs: [Api.Document]) @@ -1234,259 +1464,3 @@ public extension Api.payments { } } -public extension Api.payments { - enum PaymentReceipt: TypeConstructorDescription { - case paymentReceipt(flags: Int32, date: Int32, botId: Int64, providerId: Int64, title: String, description: String, photo: Api.WebDocument?, invoice: Api.Invoice, info: Api.PaymentRequestedInfo?, shipping: Api.ShippingOption?, tipAmount: Int64?, currency: String, totalAmount: Int64, credentialsTitle: String, users: [Api.User]) - - public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { - switch self { - case .paymentReceipt(let flags, let date, let botId, let providerId, let title, let description, let photo, let invoice, let info, let shipping, let tipAmount, let currency, let totalAmount, let credentialsTitle, let users): - if boxed { - buffer.appendInt32(1891958275) - } - serializeInt32(flags, buffer: buffer, boxed: false) - serializeInt32(date, buffer: buffer, boxed: false) - serializeInt64(botId, buffer: buffer, boxed: false) - serializeInt64(providerId, buffer: buffer, boxed: false) - serializeString(title, buffer: buffer, boxed: false) - serializeString(description, buffer: buffer, boxed: false) - if Int(flags) & Int(1 << 2) != 0 {photo!.serialize(buffer, true)} - invoice.serialize(buffer, true) - if Int(flags) & Int(1 << 0) != 0 {info!.serialize(buffer, true)} - if Int(flags) & Int(1 << 1) != 0 {shipping!.serialize(buffer, true)} - if Int(flags) & Int(1 << 3) != 0 {serializeInt64(tipAmount!, buffer: buffer, boxed: false)} - serializeString(currency, buffer: buffer, boxed: false) - serializeInt64(totalAmount, buffer: buffer, boxed: false) - serializeString(credentialsTitle, buffer: buffer, boxed: false) - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(users.count)) - for item in users { - item.serialize(buffer, true) - } - break - } - } - - public func descriptionFields() -> (String, [(String, Any)]) { - switch self { - case .paymentReceipt(let flags, let date, let botId, let providerId, let title, let description, let photo, let invoice, let info, let shipping, let tipAmount, let currency, let totalAmount, let credentialsTitle, let users): - return ("paymentReceipt", [("flags", flags as Any), ("date", date as Any), ("botId", botId as Any), ("providerId", providerId as Any), ("title", title as Any), ("description", description as Any), ("photo", photo as Any), ("invoice", invoice as Any), ("info", info as Any), ("shipping", shipping as Any), ("tipAmount", tipAmount as Any), ("currency", currency as Any), ("totalAmount", totalAmount as Any), ("credentialsTitle", credentialsTitle as Any), ("users", users as Any)]) - } - } - - public static func parse_paymentReceipt(_ reader: BufferReader) -> PaymentReceipt? { - var _1: Int32? - _1 = reader.readInt32() - var _2: Int32? - _2 = reader.readInt32() - var _3: Int64? - _3 = reader.readInt64() - var _4: Int64? - _4 = reader.readInt64() - var _5: String? - _5 = parseString(reader) - var _6: String? - _6 = parseString(reader) - var _7: Api.WebDocument? - if Int(_1!) & Int(1 << 2) != 0 {if let signature = reader.readInt32() { - _7 = Api.parse(reader, signature: signature) as? Api.WebDocument - } } - var _8: Api.Invoice? - if let signature = reader.readInt32() { - _8 = Api.parse(reader, signature: signature) as? Api.Invoice - } - var _9: Api.PaymentRequestedInfo? - if Int(_1!) & Int(1 << 0) != 0 {if let signature = reader.readInt32() { - _9 = Api.parse(reader, signature: signature) as? Api.PaymentRequestedInfo - } } - var _10: Api.ShippingOption? - if Int(_1!) & Int(1 << 1) != 0 {if let signature = reader.readInt32() { - _10 = Api.parse(reader, signature: signature) as? Api.ShippingOption - } } - var _11: Int64? - if Int(_1!) & Int(1 << 3) != 0 {_11 = reader.readInt64() } - var _12: String? - _12 = parseString(reader) - var _13: Int64? - _13 = reader.readInt64() - var _14: String? - _14 = parseString(reader) - var _15: [Api.User]? - if let _ = reader.readInt32() { - _15 = Api.parseVector(reader, elementSignature: 0, elementType: Api.User.self) - } - let _c1 = _1 != nil - let _c2 = _2 != nil - let _c3 = _3 != nil - let _c4 = _4 != nil - let _c5 = _5 != nil - let _c6 = _6 != nil - let _c7 = (Int(_1!) & Int(1 << 2) == 0) || _7 != nil - let _c8 = _8 != nil - let _c9 = (Int(_1!) & Int(1 << 0) == 0) || _9 != nil - let _c10 = (Int(_1!) & Int(1 << 1) == 0) || _10 != nil - let _c11 = (Int(_1!) & Int(1 << 3) == 0) || _11 != nil - let _c12 = _12 != nil - let _c13 = _13 != nil - let _c14 = _14 != nil - let _c15 = _15 != nil - if _c1 && _c2 && _c3 && _c4 && _c5 && _c6 && _c7 && _c8 && _c9 && _c10 && _c11 && _c12 && _c13 && _c14 && _c15 { - return Api.payments.PaymentReceipt.paymentReceipt(flags: _1!, date: _2!, botId: _3!, providerId: _4!, title: _5!, description: _6!, photo: _7, invoice: _8!, info: _9, shipping: _10, tipAmount: _11, currency: _12!, totalAmount: _13!, credentialsTitle: _14!, users: _15!) - } - else { - return nil - } - } - - } -} -public extension Api.payments { - indirect enum PaymentResult: TypeConstructorDescription { - case paymentResult(updates: Api.Updates) - case paymentVerificationNeeded(url: String) - - public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { - switch self { - case .paymentResult(let updates): - if boxed { - buffer.appendInt32(1314881805) - } - updates.serialize(buffer, true) - break - case .paymentVerificationNeeded(let url): - if boxed { - buffer.appendInt32(-666824391) - } - serializeString(url, buffer: buffer, boxed: false) - break - } - } - - public func descriptionFields() -> (String, [(String, Any)]) { - switch self { - case .paymentResult(let updates): - return ("paymentResult", [("updates", updates as Any)]) - case .paymentVerificationNeeded(let url): - return ("paymentVerificationNeeded", [("url", url as Any)]) - } - } - - public static func parse_paymentResult(_ reader: BufferReader) -> PaymentResult? { - var _1: Api.Updates? - if let signature = reader.readInt32() { - _1 = Api.parse(reader, signature: signature) as? Api.Updates - } - let _c1 = _1 != nil - if _c1 { - return Api.payments.PaymentResult.paymentResult(updates: _1!) - } - else { - return nil - } - } - public static func parse_paymentVerificationNeeded(_ reader: BufferReader) -> PaymentResult? { - var _1: String? - _1 = parseString(reader) - let _c1 = _1 != nil - if _c1 { - return Api.payments.PaymentResult.paymentVerificationNeeded(url: _1!) - } - else { - return nil - } - } - - } -} -public extension Api.payments { - enum SavedInfo: TypeConstructorDescription { - case savedInfo(flags: Int32, savedInfo: Api.PaymentRequestedInfo?) - - public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { - switch self { - case .savedInfo(let flags, let savedInfo): - if boxed { - buffer.appendInt32(-74456004) - } - serializeInt32(flags, buffer: buffer, boxed: false) - if Int(flags) & Int(1 << 0) != 0 {savedInfo!.serialize(buffer, true)} - break - } - } - - public func descriptionFields() -> (String, [(String, Any)]) { - switch self { - case .savedInfo(let flags, let savedInfo): - return ("savedInfo", [("flags", flags as Any), ("savedInfo", savedInfo as Any)]) - } - } - - public static func parse_savedInfo(_ reader: BufferReader) -> SavedInfo? { - var _1: Int32? - _1 = reader.readInt32() - var _2: Api.PaymentRequestedInfo? - if Int(_1!) & Int(1 << 0) != 0 {if let signature = reader.readInt32() { - _2 = Api.parse(reader, signature: signature) as? Api.PaymentRequestedInfo - } } - let _c1 = _1 != nil - let _c2 = (Int(_1!) & Int(1 << 0) == 0) || _2 != nil - if _c1 && _c2 { - return Api.payments.SavedInfo.savedInfo(flags: _1!, savedInfo: _2) - } - else { - return nil - } - } - - } -} -public extension Api.payments { - enum ValidatedRequestedInfo: TypeConstructorDescription { - case validatedRequestedInfo(flags: Int32, id: String?, shippingOptions: [Api.ShippingOption]?) - - public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { - switch self { - case .validatedRequestedInfo(let flags, let id, let shippingOptions): - if boxed { - buffer.appendInt32(-784000893) - } - serializeInt32(flags, buffer: buffer, boxed: false) - if Int(flags) & Int(1 << 0) != 0 {serializeString(id!, buffer: buffer, boxed: false)} - if Int(flags) & Int(1 << 1) != 0 {buffer.appendInt32(481674261) - buffer.appendInt32(Int32(shippingOptions!.count)) - for item in shippingOptions! { - item.serialize(buffer, true) - }} - break - } - } - - public func descriptionFields() -> (String, [(String, Any)]) { - switch self { - case .validatedRequestedInfo(let flags, let id, let shippingOptions): - return ("validatedRequestedInfo", [("flags", flags as Any), ("id", id as Any), ("shippingOptions", shippingOptions as Any)]) - } - } - - public static func parse_validatedRequestedInfo(_ reader: BufferReader) -> ValidatedRequestedInfo? { - var _1: Int32? - _1 = reader.readInt32() - var _2: String? - if Int(_1!) & Int(1 << 0) != 0 {_2 = parseString(reader) } - var _3: [Api.ShippingOption]? - if Int(_1!) & Int(1 << 1) != 0 {if let _ = reader.readInt32() { - _3 = Api.parseVector(reader, elementSignature: 0, elementType: Api.ShippingOption.self) - } } - let _c1 = _1 != nil - let _c2 = (Int(_1!) & Int(1 << 0) == 0) || _2 != nil - let _c3 = (Int(_1!) & Int(1 << 1) == 0) || _3 != nil - if _c1 && _c2 && _c3 { - return Api.payments.ValidatedRequestedInfo.validatedRequestedInfo(flags: _1!, id: _2, shippingOptions: _3) - } - else { - return nil - } - } - - } -} diff --git a/submodules/TelegramApi/Sources/Api30.swift b/submodules/TelegramApi/Sources/Api30.swift index 5c39686c7dd..3e5ca3e260f 100644 --- a/submodules/TelegramApi/Sources/Api30.swift +++ b/submodules/TelegramApi/Sources/Api30.swift @@ -1,3 +1,259 @@ +public extension Api.payments { + enum PaymentReceipt: TypeConstructorDescription { + case paymentReceipt(flags: Int32, date: Int32, botId: Int64, providerId: Int64, title: String, description: String, photo: Api.WebDocument?, invoice: Api.Invoice, info: Api.PaymentRequestedInfo?, shipping: Api.ShippingOption?, tipAmount: Int64?, currency: String, totalAmount: Int64, credentialsTitle: String, users: [Api.User]) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .paymentReceipt(let flags, let date, let botId, let providerId, let title, let description, let photo, let invoice, let info, let shipping, let tipAmount, let currency, let totalAmount, let credentialsTitle, let users): + if boxed { + buffer.appendInt32(1891958275) + } + serializeInt32(flags, buffer: buffer, boxed: false) + serializeInt32(date, buffer: buffer, boxed: false) + serializeInt64(botId, buffer: buffer, boxed: false) + serializeInt64(providerId, buffer: buffer, boxed: false) + serializeString(title, buffer: buffer, boxed: false) + serializeString(description, buffer: buffer, boxed: false) + if Int(flags) & Int(1 << 2) != 0 {photo!.serialize(buffer, true)} + invoice.serialize(buffer, true) + if Int(flags) & Int(1 << 0) != 0 {info!.serialize(buffer, true)} + if Int(flags) & Int(1 << 1) != 0 {shipping!.serialize(buffer, true)} + if Int(flags) & Int(1 << 3) != 0 {serializeInt64(tipAmount!, buffer: buffer, boxed: false)} + serializeString(currency, buffer: buffer, boxed: false) + serializeInt64(totalAmount, buffer: buffer, boxed: false) + serializeString(credentialsTitle, buffer: buffer, boxed: false) + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(users.count)) + for item in users { + item.serialize(buffer, true) + } + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .paymentReceipt(let flags, let date, let botId, let providerId, let title, let description, let photo, let invoice, let info, let shipping, let tipAmount, let currency, let totalAmount, let credentialsTitle, let users): + return ("paymentReceipt", [("flags", flags as Any), ("date", date as Any), ("botId", botId as Any), ("providerId", providerId as Any), ("title", title as Any), ("description", description as Any), ("photo", photo as Any), ("invoice", invoice as Any), ("info", info as Any), ("shipping", shipping as Any), ("tipAmount", tipAmount as Any), ("currency", currency as Any), ("totalAmount", totalAmount as Any), ("credentialsTitle", credentialsTitle as Any), ("users", users as Any)]) + } + } + + public static func parse_paymentReceipt(_ reader: BufferReader) -> PaymentReceipt? { + var _1: Int32? + _1 = reader.readInt32() + var _2: Int32? + _2 = reader.readInt32() + var _3: Int64? + _3 = reader.readInt64() + var _4: Int64? + _4 = reader.readInt64() + var _5: String? + _5 = parseString(reader) + var _6: String? + _6 = parseString(reader) + var _7: Api.WebDocument? + if Int(_1!) & Int(1 << 2) != 0 {if let signature = reader.readInt32() { + _7 = Api.parse(reader, signature: signature) as? Api.WebDocument + } } + var _8: Api.Invoice? + if let signature = reader.readInt32() { + _8 = Api.parse(reader, signature: signature) as? Api.Invoice + } + var _9: Api.PaymentRequestedInfo? + if Int(_1!) & Int(1 << 0) != 0 {if let signature = reader.readInt32() { + _9 = Api.parse(reader, signature: signature) as? Api.PaymentRequestedInfo + } } + var _10: Api.ShippingOption? + if Int(_1!) & Int(1 << 1) != 0 {if let signature = reader.readInt32() { + _10 = Api.parse(reader, signature: signature) as? Api.ShippingOption + } } + var _11: Int64? + if Int(_1!) & Int(1 << 3) != 0 {_11 = reader.readInt64() } + var _12: String? + _12 = parseString(reader) + var _13: Int64? + _13 = reader.readInt64() + var _14: String? + _14 = parseString(reader) + var _15: [Api.User]? + if let _ = reader.readInt32() { + _15 = Api.parseVector(reader, elementSignature: 0, elementType: Api.User.self) + } + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + let _c4 = _4 != nil + let _c5 = _5 != nil + let _c6 = _6 != nil + let _c7 = (Int(_1!) & Int(1 << 2) == 0) || _7 != nil + let _c8 = _8 != nil + let _c9 = (Int(_1!) & Int(1 << 0) == 0) || _9 != nil + let _c10 = (Int(_1!) & Int(1 << 1) == 0) || _10 != nil + let _c11 = (Int(_1!) & Int(1 << 3) == 0) || _11 != nil + let _c12 = _12 != nil + let _c13 = _13 != nil + let _c14 = _14 != nil + let _c15 = _15 != nil + if _c1 && _c2 && _c3 && _c4 && _c5 && _c6 && _c7 && _c8 && _c9 && _c10 && _c11 && _c12 && _c13 && _c14 && _c15 { + return Api.payments.PaymentReceipt.paymentReceipt(flags: _1!, date: _2!, botId: _3!, providerId: _4!, title: _5!, description: _6!, photo: _7, invoice: _8!, info: _9, shipping: _10, tipAmount: _11, currency: _12!, totalAmount: _13!, credentialsTitle: _14!, users: _15!) + } + else { + return nil + } + } + + } +} +public extension Api.payments { + indirect enum PaymentResult: TypeConstructorDescription { + case paymentResult(updates: Api.Updates) + case paymentVerificationNeeded(url: String) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .paymentResult(let updates): + if boxed { + buffer.appendInt32(1314881805) + } + updates.serialize(buffer, true) + break + case .paymentVerificationNeeded(let url): + if boxed { + buffer.appendInt32(-666824391) + } + serializeString(url, buffer: buffer, boxed: false) + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .paymentResult(let updates): + return ("paymentResult", [("updates", updates as Any)]) + case .paymentVerificationNeeded(let url): + return ("paymentVerificationNeeded", [("url", url as Any)]) + } + } + + public static func parse_paymentResult(_ reader: BufferReader) -> PaymentResult? { + var _1: Api.Updates? + if let signature = reader.readInt32() { + _1 = Api.parse(reader, signature: signature) as? Api.Updates + } + let _c1 = _1 != nil + if _c1 { + return Api.payments.PaymentResult.paymentResult(updates: _1!) + } + else { + return nil + } + } + public static func parse_paymentVerificationNeeded(_ reader: BufferReader) -> PaymentResult? { + var _1: String? + _1 = parseString(reader) + let _c1 = _1 != nil + if _c1 { + return Api.payments.PaymentResult.paymentVerificationNeeded(url: _1!) + } + else { + return nil + } + } + + } +} +public extension Api.payments { + enum SavedInfo: TypeConstructorDescription { + case savedInfo(flags: Int32, savedInfo: Api.PaymentRequestedInfo?) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .savedInfo(let flags, let savedInfo): + if boxed { + buffer.appendInt32(-74456004) + } + serializeInt32(flags, buffer: buffer, boxed: false) + if Int(flags) & Int(1 << 0) != 0 {savedInfo!.serialize(buffer, true)} + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .savedInfo(let flags, let savedInfo): + return ("savedInfo", [("flags", flags as Any), ("savedInfo", savedInfo as Any)]) + } + } + + public static func parse_savedInfo(_ reader: BufferReader) -> SavedInfo? { + var _1: Int32? + _1 = reader.readInt32() + var _2: Api.PaymentRequestedInfo? + if Int(_1!) & Int(1 << 0) != 0 {if let signature = reader.readInt32() { + _2 = Api.parse(reader, signature: signature) as? Api.PaymentRequestedInfo + } } + let _c1 = _1 != nil + let _c2 = (Int(_1!) & Int(1 << 0) == 0) || _2 != nil + if _c1 && _c2 { + return Api.payments.SavedInfo.savedInfo(flags: _1!, savedInfo: _2) + } + else { + return nil + } + } + + } +} +public extension Api.payments { + enum ValidatedRequestedInfo: TypeConstructorDescription { + case validatedRequestedInfo(flags: Int32, id: String?, shippingOptions: [Api.ShippingOption]?) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .validatedRequestedInfo(let flags, let id, let shippingOptions): + if boxed { + buffer.appendInt32(-784000893) + } + serializeInt32(flags, buffer: buffer, boxed: false) + if Int(flags) & Int(1 << 0) != 0 {serializeString(id!, buffer: buffer, boxed: false)} + if Int(flags) & Int(1 << 1) != 0 {buffer.appendInt32(481674261) + buffer.appendInt32(Int32(shippingOptions!.count)) + for item in shippingOptions! { + item.serialize(buffer, true) + }} + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .validatedRequestedInfo(let flags, let id, let shippingOptions): + return ("validatedRequestedInfo", [("flags", flags as Any), ("id", id as Any), ("shippingOptions", shippingOptions as Any)]) + } + } + + public static func parse_validatedRequestedInfo(_ reader: BufferReader) -> ValidatedRequestedInfo? { + var _1: Int32? + _1 = reader.readInt32() + var _2: String? + if Int(_1!) & Int(1 << 0) != 0 {_2 = parseString(reader) } + var _3: [Api.ShippingOption]? + if Int(_1!) & Int(1 << 1) != 0 {if let _ = reader.readInt32() { + _3 = Api.parseVector(reader, elementSignature: 0, elementType: Api.ShippingOption.self) + } } + let _c1 = _1 != nil + let _c2 = (Int(_1!) & Int(1 << 0) == 0) || _2 != nil + let _c3 = (Int(_1!) & Int(1 << 1) == 0) || _3 != nil + if _c1 && _c2 && _c3 { + return Api.payments.ValidatedRequestedInfo.validatedRequestedInfo(flags: _1!, id: _2, shippingOptions: _3) + } + else { + return nil + } + } + + } +} public extension Api.phone { enum ExportedGroupCallInvite: TypeConstructorDescription { case exportedGroupCallInvite(link: String) @@ -724,6 +980,110 @@ public extension Api.premium { } } +public extension Api.smsjobs { + enum EligibilityToJoin: TypeConstructorDescription { + case eligibleToJoin(termsUrl: String, monthlySentSms: Int32) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .eligibleToJoin(let termsUrl, let monthlySentSms): + if boxed { + buffer.appendInt32(-594852657) + } + serializeString(termsUrl, buffer: buffer, boxed: false) + serializeInt32(monthlySentSms, buffer: buffer, boxed: false) + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .eligibleToJoin(let termsUrl, let monthlySentSms): + return ("eligibleToJoin", [("termsUrl", termsUrl as Any), ("monthlySentSms", monthlySentSms as Any)]) + } + } + + public static func parse_eligibleToJoin(_ reader: BufferReader) -> EligibilityToJoin? { + var _1: String? + _1 = parseString(reader) + var _2: Int32? + _2 = reader.readInt32() + let _c1 = _1 != nil + let _c2 = _2 != nil + if _c1 && _c2 { + return Api.smsjobs.EligibilityToJoin.eligibleToJoin(termsUrl: _1!, monthlySentSms: _2!) + } + else { + return nil + } + } + + } +} +public extension Api.smsjobs { + enum Status: TypeConstructorDescription { + case status(flags: Int32, recentSent: Int32, recentSince: Int32, recentRemains: Int32, totalSent: Int32, totalSince: Int32, lastGiftSlug: String?, termsUrl: String) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .status(let flags, let recentSent, let recentSince, let recentRemains, let totalSent, let totalSince, let lastGiftSlug, let termsUrl): + if boxed { + buffer.appendInt32(720277905) + } + serializeInt32(flags, buffer: buffer, boxed: false) + serializeInt32(recentSent, buffer: buffer, boxed: false) + serializeInt32(recentSince, buffer: buffer, boxed: false) + serializeInt32(recentRemains, buffer: buffer, boxed: false) + serializeInt32(totalSent, buffer: buffer, boxed: false) + serializeInt32(totalSince, buffer: buffer, boxed: false) + if Int(flags) & Int(1 << 1) != 0 {serializeString(lastGiftSlug!, buffer: buffer, boxed: false)} + serializeString(termsUrl, buffer: buffer, boxed: false) + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .status(let flags, let recentSent, let recentSince, let recentRemains, let totalSent, let totalSince, let lastGiftSlug, let termsUrl): + return ("status", [("flags", flags as Any), ("recentSent", recentSent as Any), ("recentSince", recentSince as Any), ("recentRemains", recentRemains as Any), ("totalSent", totalSent as Any), ("totalSince", totalSince as Any), ("lastGiftSlug", lastGiftSlug as Any), ("termsUrl", termsUrl as Any)]) + } + } + + public static func parse_status(_ reader: BufferReader) -> Status? { + var _1: Int32? + _1 = reader.readInt32() + var _2: Int32? + _2 = reader.readInt32() + var _3: Int32? + _3 = reader.readInt32() + var _4: Int32? + _4 = reader.readInt32() + var _5: Int32? + _5 = reader.readInt32() + var _6: Int32? + _6 = reader.readInt32() + var _7: String? + if Int(_1!) & Int(1 << 1) != 0 {_7 = parseString(reader) } + var _8: String? + _8 = parseString(reader) + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + let _c4 = _4 != nil + let _c5 = _5 != nil + let _c6 = _6 != nil + let _c7 = (Int(_1!) & Int(1 << 1) == 0) || _7 != nil + let _c8 = _8 != nil + if _c1 && _c2 && _c3 && _c4 && _c5 && _c6 && _c7 && _c8 { + return Api.smsjobs.Status.status(flags: _1!, recentSent: _2!, recentSince: _3!, recentRemains: _4!, totalSent: _5!, totalSince: _6!, lastGiftSlug: _7, termsUrl: _8!) + } + else { + return nil + } + } + + } +} public extension Api.stats { enum BroadcastStats: TypeConstructorDescription { case broadcastStats(period: Api.StatsDateRangeDays, followers: Api.StatsAbsValueAndPrev, viewsPerPost: Api.StatsAbsValueAndPrev, sharesPerPost: Api.StatsAbsValueAndPrev, reactionsPerPost: Api.StatsAbsValueAndPrev, viewsPerStory: Api.StatsAbsValueAndPrev, sharesPerStory: Api.StatsAbsValueAndPrev, reactionsPerStory: Api.StatsAbsValueAndPrev, enabledNotifications: Api.StatsPercentValue, growthGraph: Api.StatsGraph, followersGraph: Api.StatsGraph, muteGraph: Api.StatsGraph, topHoursGraph: Api.StatsGraph, interactionsGraph: Api.StatsGraph, ivInteractionsGraph: Api.StatsGraph, viewsBySourceGraph: Api.StatsGraph, newFollowersBySourceGraph: Api.StatsGraph, languagesGraph: Api.StatsGraph, reactionsByEmotionGraph: Api.StatsGraph, storyInteractionsGraph: Api.StatsGraph, storyReactionsByEmotionGraph: Api.StatsGraph, recentPostsInteractions: [Api.PostInteractionCounters]) @@ -1376,171 +1736,3 @@ public extension Api.storage { } } -public extension Api.stories { - enum AllStories: TypeConstructorDescription { - case allStories(flags: Int32, count: Int32, state: String, peerStories: [Api.PeerStories], chats: [Api.Chat], users: [Api.User], stealthMode: Api.StoriesStealthMode) - case allStoriesNotModified(flags: Int32, state: String, stealthMode: Api.StoriesStealthMode) - - public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { - switch self { - case .allStories(let flags, let count, let state, let peerStories, let chats, let users, let stealthMode): - if boxed { - buffer.appendInt32(1862033025) - } - serializeInt32(flags, buffer: buffer, boxed: false) - serializeInt32(count, buffer: buffer, boxed: false) - serializeString(state, buffer: buffer, boxed: false) - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(peerStories.count)) - for item in peerStories { - item.serialize(buffer, true) - } - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(chats.count)) - for item in chats { - item.serialize(buffer, true) - } - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(users.count)) - for item in users { - item.serialize(buffer, true) - } - stealthMode.serialize(buffer, true) - break - case .allStoriesNotModified(let flags, let state, let stealthMode): - if boxed { - buffer.appendInt32(291044926) - } - serializeInt32(flags, buffer: buffer, boxed: false) - serializeString(state, buffer: buffer, boxed: false) - stealthMode.serialize(buffer, true) - break - } - } - - public func descriptionFields() -> (String, [(String, Any)]) { - switch self { - case .allStories(let flags, let count, let state, let peerStories, let chats, let users, let stealthMode): - return ("allStories", [("flags", flags as Any), ("count", count as Any), ("state", state as Any), ("peerStories", peerStories as Any), ("chats", chats as Any), ("users", users as Any), ("stealthMode", stealthMode as Any)]) - case .allStoriesNotModified(let flags, let state, let stealthMode): - return ("allStoriesNotModified", [("flags", flags as Any), ("state", state as Any), ("stealthMode", stealthMode as Any)]) - } - } - - public static func parse_allStories(_ reader: BufferReader) -> AllStories? { - var _1: Int32? - _1 = reader.readInt32() - var _2: Int32? - _2 = reader.readInt32() - var _3: String? - _3 = parseString(reader) - var _4: [Api.PeerStories]? - if let _ = reader.readInt32() { - _4 = Api.parseVector(reader, elementSignature: 0, elementType: Api.PeerStories.self) - } - var _5: [Api.Chat]? - if let _ = reader.readInt32() { - _5 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Chat.self) - } - var _6: [Api.User]? - if let _ = reader.readInt32() { - _6 = Api.parseVector(reader, elementSignature: 0, elementType: Api.User.self) - } - var _7: Api.StoriesStealthMode? - if let signature = reader.readInt32() { - _7 = Api.parse(reader, signature: signature) as? Api.StoriesStealthMode - } - let _c1 = _1 != nil - let _c2 = _2 != nil - let _c3 = _3 != nil - let _c4 = _4 != nil - let _c5 = _5 != nil - let _c6 = _6 != nil - let _c7 = _7 != nil - if _c1 && _c2 && _c3 && _c4 && _c5 && _c6 && _c7 { - return Api.stories.AllStories.allStories(flags: _1!, count: _2!, state: _3!, peerStories: _4!, chats: _5!, users: _6!, stealthMode: _7!) - } - else { - return nil - } - } - public static func parse_allStoriesNotModified(_ reader: BufferReader) -> AllStories? { - var _1: Int32? - _1 = reader.readInt32() - var _2: String? - _2 = parseString(reader) - var _3: Api.StoriesStealthMode? - if let signature = reader.readInt32() { - _3 = Api.parse(reader, signature: signature) as? Api.StoriesStealthMode - } - let _c1 = _1 != nil - let _c2 = _2 != nil - let _c3 = _3 != nil - if _c1 && _c2 && _c3 { - return Api.stories.AllStories.allStoriesNotModified(flags: _1!, state: _2!, stealthMode: _3!) - } - else { - return nil - } - } - - } -} -public extension Api.stories { - enum PeerStories: TypeConstructorDescription { - case peerStories(stories: Api.PeerStories, chats: [Api.Chat], users: [Api.User]) - - public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { - switch self { - case .peerStories(let stories, let chats, let users): - if boxed { - buffer.appendInt32(-890861720) - } - stories.serialize(buffer, true) - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(chats.count)) - for item in chats { - item.serialize(buffer, true) - } - buffer.appendInt32(481674261) - buffer.appendInt32(Int32(users.count)) - for item in users { - item.serialize(buffer, true) - } - break - } - } - - public func descriptionFields() -> (String, [(String, Any)]) { - switch self { - case .peerStories(let stories, let chats, let users): - return ("peerStories", [("stories", stories as Any), ("chats", chats as Any), ("users", users as Any)]) - } - } - - public static func parse_peerStories(_ reader: BufferReader) -> PeerStories? { - var _1: Api.PeerStories? - if let signature = reader.readInt32() { - _1 = Api.parse(reader, signature: signature) as? Api.PeerStories - } - var _2: [Api.Chat]? - if let _ = reader.readInt32() { - _2 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Chat.self) - } - var _3: [Api.User]? - if let _ = reader.readInt32() { - _3 = Api.parseVector(reader, elementSignature: 0, elementType: Api.User.self) - } - let _c1 = _1 != nil - let _c2 = _2 != nil - let _c3 = _3 != nil - if _c1 && _c2 && _c3 { - return Api.stories.PeerStories.peerStories(stories: _1!, chats: _2!, users: _3!) - } - else { - return nil - } - } - - } -} diff --git a/submodules/TelegramApi/Sources/Api31.swift b/submodules/TelegramApi/Sources/Api31.swift index 52e6eb8e1fe..2b161fbd976 100644 --- a/submodules/TelegramApi/Sources/Api31.swift +++ b/submodules/TelegramApi/Sources/Api31.swift @@ -1,3 +1,171 @@ +public extension Api.stories { + enum AllStories: TypeConstructorDescription { + case allStories(flags: Int32, count: Int32, state: String, peerStories: [Api.PeerStories], chats: [Api.Chat], users: [Api.User], stealthMode: Api.StoriesStealthMode) + case allStoriesNotModified(flags: Int32, state: String, stealthMode: Api.StoriesStealthMode) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .allStories(let flags, let count, let state, let peerStories, let chats, let users, let stealthMode): + if boxed { + buffer.appendInt32(1862033025) + } + serializeInt32(flags, buffer: buffer, boxed: false) + serializeInt32(count, buffer: buffer, boxed: false) + serializeString(state, buffer: buffer, boxed: false) + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(peerStories.count)) + for item in peerStories { + item.serialize(buffer, true) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(chats.count)) + for item in chats { + item.serialize(buffer, true) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(users.count)) + for item in users { + item.serialize(buffer, true) + } + stealthMode.serialize(buffer, true) + break + case .allStoriesNotModified(let flags, let state, let stealthMode): + if boxed { + buffer.appendInt32(291044926) + } + serializeInt32(flags, buffer: buffer, boxed: false) + serializeString(state, buffer: buffer, boxed: false) + stealthMode.serialize(buffer, true) + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .allStories(let flags, let count, let state, let peerStories, let chats, let users, let stealthMode): + return ("allStories", [("flags", flags as Any), ("count", count as Any), ("state", state as Any), ("peerStories", peerStories as Any), ("chats", chats as Any), ("users", users as Any), ("stealthMode", stealthMode as Any)]) + case .allStoriesNotModified(let flags, let state, let stealthMode): + return ("allStoriesNotModified", [("flags", flags as Any), ("state", state as Any), ("stealthMode", stealthMode as Any)]) + } + } + + public static func parse_allStories(_ reader: BufferReader) -> AllStories? { + var _1: Int32? + _1 = reader.readInt32() + var _2: Int32? + _2 = reader.readInt32() + var _3: String? + _3 = parseString(reader) + var _4: [Api.PeerStories]? + if let _ = reader.readInt32() { + _4 = Api.parseVector(reader, elementSignature: 0, elementType: Api.PeerStories.self) + } + var _5: [Api.Chat]? + if let _ = reader.readInt32() { + _5 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Chat.self) + } + var _6: [Api.User]? + if let _ = reader.readInt32() { + _6 = Api.parseVector(reader, elementSignature: 0, elementType: Api.User.self) + } + var _7: Api.StoriesStealthMode? + if let signature = reader.readInt32() { + _7 = Api.parse(reader, signature: signature) as? Api.StoriesStealthMode + } + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + let _c4 = _4 != nil + let _c5 = _5 != nil + let _c6 = _6 != nil + let _c7 = _7 != nil + if _c1 && _c2 && _c3 && _c4 && _c5 && _c6 && _c7 { + return Api.stories.AllStories.allStories(flags: _1!, count: _2!, state: _3!, peerStories: _4!, chats: _5!, users: _6!, stealthMode: _7!) + } + else { + return nil + } + } + public static func parse_allStoriesNotModified(_ reader: BufferReader) -> AllStories? { + var _1: Int32? + _1 = reader.readInt32() + var _2: String? + _2 = parseString(reader) + var _3: Api.StoriesStealthMode? + if let signature = reader.readInt32() { + _3 = Api.parse(reader, signature: signature) as? Api.StoriesStealthMode + } + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + if _c1 && _c2 && _c3 { + return Api.stories.AllStories.allStoriesNotModified(flags: _1!, state: _2!, stealthMode: _3!) + } + else { + return nil + } + } + + } +} +public extension Api.stories { + enum PeerStories: TypeConstructorDescription { + case peerStories(stories: Api.PeerStories, chats: [Api.Chat], users: [Api.User]) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .peerStories(let stories, let chats, let users): + if boxed { + buffer.appendInt32(-890861720) + } + stories.serialize(buffer, true) + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(chats.count)) + for item in chats { + item.serialize(buffer, true) + } + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(users.count)) + for item in users { + item.serialize(buffer, true) + } + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .peerStories(let stories, let chats, let users): + return ("peerStories", [("stories", stories as Any), ("chats", chats as Any), ("users", users as Any)]) + } + } + + public static func parse_peerStories(_ reader: BufferReader) -> PeerStories? { + var _1: Api.PeerStories? + if let signature = reader.readInt32() { + _1 = Api.parse(reader, signature: signature) as? Api.PeerStories + } + var _2: [Api.Chat]? + if let _ = reader.readInt32() { + _2 = Api.parseVector(reader, elementSignature: 0, elementType: Api.Chat.self) + } + var _3: [Api.User]? + if let _ = reader.readInt32() { + _3 = Api.parseVector(reader, elementSignature: 0, elementType: Api.User.self) + } + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + if _c1 && _c2 && _c3 { + return Api.stories.PeerStories.peerStories(stories: _1!, chats: _2!, users: _3!) + } + else { + return nil + } + } + + } +} public extension Api.stories { enum Stories: TypeConstructorDescription { case stories(count: Int32, stories: [Api.StoryItem], chats: [Api.Chat], users: [Api.User]) diff --git a/submodules/TelegramApi/Sources/Api32.swift b/submodules/TelegramApi/Sources/Api32.swift index d280ba8d816..5b224f22fcb 100644 --- a/submodules/TelegramApi/Sources/Api32.swift +++ b/submodules/TelegramApi/Sources/Api32.swift @@ -328,6 +328,21 @@ public extension Api.functions.account { }) } } +public extension Api.functions.account { + static func getBotBusinessConnection(connectionId: String) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(1990746736) + serializeString(connectionId, buffer: buffer, boxed: false) + return (FunctionDescription(name: "account.getBotBusinessConnection", parameters: [("connectionId", String(describing: connectionId))]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.Updates? in + let reader = BufferReader(buffer) + var result: Api.Updates? + if let signature = reader.readInt32() { + result = Api.parse(reader, signature: signature) as? Api.Updates + } + return result + }) + } +} public extension Api.functions.account { static func getChannelDefaultEmojiStatuses(hash: Int64) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { let buffer = Buffer() @@ -373,6 +388,21 @@ public extension Api.functions.account { }) } } +public extension Api.functions.account { + static func getConnectedBots() -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(1319421967) + + return (FunctionDescription(name: "account.getConnectedBots", parameters: []), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.account.ConnectedBots? in + let reader = BufferReader(buffer) + var result: Api.account.ConnectedBots? + if let signature = reader.readInt32() { + result = Api.parse(reader, signature: signature) as? Api.account.ConnectedBots + } + return result + }) + } +} public extension Api.functions.account { static func getContactSignUpNotification() -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { let buffer = Buffer() @@ -1261,6 +1291,71 @@ public extension Api.functions.account { }) } } +public extension Api.functions.account { + static func updateBusinessAwayMessage(flags: Int32, message: Api.InputBusinessAwayMessage?) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(-1570078811) + serializeInt32(flags, buffer: buffer, boxed: false) + if Int(flags) & Int(1 << 0) != 0 {message!.serialize(buffer, true)} + return (FunctionDescription(name: "account.updateBusinessAwayMessage", parameters: [("flags", String(describing: flags)), ("message", String(describing: message))]), 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 extension Api.functions.account { + static func updateBusinessGreetingMessage(flags: Int32, message: Api.InputBusinessGreetingMessage?) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(1724755908) + serializeInt32(flags, buffer: buffer, boxed: false) + if Int(flags) & Int(1 << 0) != 0 {message!.serialize(buffer, true)} + return (FunctionDescription(name: "account.updateBusinessGreetingMessage", parameters: [("flags", String(describing: flags)), ("message", String(describing: message))]), 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 extension Api.functions.account { + static func updateBusinessLocation(flags: Int32, geoPoint: Api.InputGeoPoint?, address: String?) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(-1637149926) + serializeInt32(flags, buffer: buffer, boxed: false) + if Int(flags) & Int(1 << 1) != 0 {geoPoint!.serialize(buffer, true)} + if Int(flags) & Int(1 << 0) != 0 {serializeString(address!, buffer: buffer, boxed: false)} + return (FunctionDescription(name: "account.updateBusinessLocation", parameters: [("flags", String(describing: flags)), ("geoPoint", String(describing: geoPoint)), ("address", String(describing: address))]), 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 extension Api.functions.account { + static func updateBusinessWorkHours(flags: Int32, businessWorkHours: Api.BusinessWorkHours?) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(1258348646) + serializeInt32(flags, buffer: buffer, boxed: false) + if Int(flags) & Int(1 << 0) != 0 {businessWorkHours!.serialize(buffer, true)} + return (FunctionDescription(name: "account.updateBusinessWorkHours", parameters: [("flags", String(describing: flags)), ("businessWorkHours", String(describing: businessWorkHours))]), 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 extension Api.functions.account { static func updateColor(flags: Int32, color: Int32?, backgroundEmojiId: Int64?) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { let buffer = Buffer() @@ -1278,6 +1373,23 @@ public extension Api.functions.account { }) } } +public extension Api.functions.account { + static func updateConnectedBot(flags: Int32, bot: Api.InputUser, recipients: Api.InputBusinessRecipients) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(-1674751363) + serializeInt32(flags, buffer: buffer, boxed: false) + bot.serialize(buffer, true) + recipients.serialize(buffer, true) + return (FunctionDescription(name: "account.updateConnectedBot", parameters: [("flags", String(describing: flags)), ("bot", String(describing: bot)), ("recipients", String(describing: recipients))]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.Updates? in + let reader = BufferReader(buffer) + var result: Api.Updates? + if let signature = reader.readInt32() { + result = Api.parse(reader, signature: signature) as? Api.Updates + } + return result + }) + } +} public extension Api.functions.account { static func updateDeviceLocked(period: Int32) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { let buffer = Buffer() @@ -4130,6 +4242,21 @@ public extension Api.functions.help { }) } } +public extension Api.functions.help { + static func getTimezonesList(hash: Int32) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(1236468288) + serializeInt32(hash, buffer: buffer, boxed: false) + return (FunctionDescription(name: "help.getTimezonesList", parameters: [("hash", String(describing: hash))]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.help.TimezonesList? in + let reader = BufferReader(buffer) + var result: Api.help.TimezonesList? + if let signature = reader.readInt32() { + result = Api.parse(reader, signature: signature) as? Api.help.TimezonesList + } + return result + }) + } +} public extension Api.functions.help { static func getUserInfo(userId: Api.InputUser) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { let buffer = Buffer() @@ -4393,6 +4520,21 @@ public extension Api.functions.messages { }) } } +public extension Api.functions.messages { + static func checkQuickReplyShortcut(shortcut: String) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(-237962285) + serializeString(shortcut, buffer: buffer, boxed: false) + return (FunctionDescription(name: "messages.checkQuickReplyShortcut", parameters: [("shortcut", String(describing: shortcut))]), 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 extension Api.functions.messages { static func clearAllDrafts() -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { let buffer = Buffer() @@ -4562,6 +4704,41 @@ public extension Api.functions.messages { }) } } +public extension Api.functions.messages { + static func deleteQuickReplyMessages(shortcutId: Int32, id: [Int32]) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(-519706352) + serializeInt32(shortcutId, buffer: buffer, boxed: false) + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(id.count)) + for item in id { + serializeInt32(item, buffer: buffer, boxed: false) + } + return (FunctionDescription(name: "messages.deleteQuickReplyMessages", parameters: [("shortcutId", String(describing: shortcutId)), ("id", String(describing: id))]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.Updates? in + let reader = BufferReader(buffer) + var result: Api.Updates? + if let signature = reader.readInt32() { + result = Api.parse(reader, signature: signature) as? Api.Updates + } + return result + }) + } +} +public extension Api.functions.messages { + static func deleteQuickReplyShortcut(shortcutId: Int32) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(1019234112) + serializeInt32(shortcutId, buffer: buffer, boxed: false) + return (FunctionDescription(name: "messages.deleteQuickReplyShortcut", parameters: [("shortcutId", String(describing: shortcutId))]), 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 extension Api.functions.messages { static func deleteRevokedExportedChatInvites(peer: Api.InputPeer, adminId: Api.InputUser) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { let buffer = Buffer() @@ -4760,9 +4937,9 @@ public extension Api.functions.messages { } } public extension Api.functions.messages { - static func editMessage(flags: Int32, peer: Api.InputPeer, id: Int32, message: String?, media: Api.InputMedia?, replyMarkup: Api.ReplyMarkup?, entities: [Api.MessageEntity]?, scheduleDate: Int32?) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + static func editMessage(flags: Int32, peer: Api.InputPeer, id: Int32, message: String?, media: Api.InputMedia?, replyMarkup: Api.ReplyMarkup?, entities: [Api.MessageEntity]?, scheduleDate: Int32?, quickReplyShortcutId: Int32?) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { let buffer = Buffer() - buffer.appendInt32(1224152952) + buffer.appendInt32(-539934715) serializeInt32(flags, buffer: buffer, boxed: false) peer.serialize(buffer, true) serializeInt32(id, buffer: buffer, boxed: false) @@ -4775,7 +4952,8 @@ public extension Api.functions.messages { item.serialize(buffer, true) }} if Int(flags) & Int(1 << 15) != 0 {serializeInt32(scheduleDate!, buffer: buffer, boxed: false)} - return (FunctionDescription(name: "messages.editMessage", parameters: [("flags", String(describing: flags)), ("peer", String(describing: peer)), ("id", String(describing: id)), ("message", String(describing: message)), ("media", String(describing: media)), ("replyMarkup", String(describing: replyMarkup)), ("entities", String(describing: entities)), ("scheduleDate", String(describing: scheduleDate))]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.Updates? in + if Int(flags) & Int(1 << 17) != 0 {serializeInt32(quickReplyShortcutId!, buffer: buffer, boxed: false)} + return (FunctionDescription(name: "messages.editMessage", parameters: [("flags", String(describing: flags)), ("peer", String(describing: peer)), ("id", String(describing: id)), ("message", String(describing: message)), ("media", String(describing: media)), ("replyMarkup", String(describing: replyMarkup)), ("entities", String(describing: entities)), ("scheduleDate", String(describing: scheduleDate)), ("quickReplyShortcutId", String(describing: quickReplyShortcutId))]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.Updates? in let reader = BufferReader(buffer) var result: Api.Updates? if let signature = reader.readInt32() { @@ -4785,6 +4963,22 @@ public extension Api.functions.messages { }) } } +public extension Api.functions.messages { + static func editQuickReplyShortcut(shortcutId: Int32, shortcut: String) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(1543519471) + serializeInt32(shortcutId, buffer: buffer, boxed: false) + serializeString(shortcut, buffer: buffer, boxed: false) + return (FunctionDescription(name: "messages.editQuickReplyShortcut", parameters: [("shortcutId", String(describing: shortcutId)), ("shortcut", String(describing: shortcut))]), 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 extension Api.functions.messages { static func exportChatInvite(flags: Int32, peer: Api.InputPeer, expireDate: Int32?, usageLimit: Int32?, title: String?) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { let buffer = Buffer() @@ -4821,9 +5015,9 @@ public extension Api.functions.messages { } } public extension Api.functions.messages { - static func forwardMessages(flags: Int32, fromPeer: Api.InputPeer, id: [Int32], randomId: [Int64], toPeer: Api.InputPeer, topMsgId: Int32?, scheduleDate: Int32?, sendAs: Api.InputPeer?) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + static func forwardMessages(flags: Int32, fromPeer: Api.InputPeer, id: [Int32], randomId: [Int64], toPeer: Api.InputPeer, topMsgId: Int32?, scheduleDate: Int32?, sendAs: Api.InputPeer?, quickReplyShortcut: Api.InputQuickReplyShortcut?) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { let buffer = Buffer() - buffer.appendInt32(-966673468) + buffer.appendInt32(-721186296) serializeInt32(flags, buffer: buffer, boxed: false) fromPeer.serialize(buffer, true) buffer.appendInt32(481674261) @@ -4840,7 +5034,8 @@ public extension Api.functions.messages { if Int(flags) & Int(1 << 9) != 0 {serializeInt32(topMsgId!, buffer: buffer, boxed: false)} if Int(flags) & Int(1 << 10) != 0 {serializeInt32(scheduleDate!, buffer: buffer, boxed: false)} if Int(flags) & Int(1 << 13) != 0 {sendAs!.serialize(buffer, true)} - return (FunctionDescription(name: "messages.forwardMessages", parameters: [("flags", String(describing: flags)), ("fromPeer", String(describing: fromPeer)), ("id", String(describing: id)), ("randomId", String(describing: randomId)), ("toPeer", String(describing: toPeer)), ("topMsgId", String(describing: topMsgId)), ("scheduleDate", String(describing: scheduleDate)), ("sendAs", String(describing: sendAs))]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.Updates? in + if Int(flags) & Int(1 << 17) != 0 {quickReplyShortcut!.serialize(buffer, true)} + return (FunctionDescription(name: "messages.forwardMessages", parameters: [("flags", String(describing: flags)), ("fromPeer", String(describing: fromPeer)), ("id", String(describing: id)), ("randomId", String(describing: randomId)), ("toPeer", String(describing: toPeer)), ("topMsgId", String(describing: topMsgId)), ("scheduleDate", String(describing: scheduleDate)), ("sendAs", String(describing: sendAs)), ("quickReplyShortcut", String(describing: quickReplyShortcut))]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.Updates? in let reader = BufferReader(buffer) var result: Api.Updates? if let signature = reader.readInt32() { @@ -5130,15 +5325,15 @@ public extension Api.functions.messages { } } public extension Api.functions.messages { - static func getDialogFilters() -> (FunctionDescription, Buffer, DeserializeFunctionResponse<[Api.DialogFilter]>) { + static func getDialogFilters() -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { let buffer = Buffer() - buffer.appendInt32(-241247891) + buffer.appendInt32(-271283063) - return (FunctionDescription(name: "messages.getDialogFilters", parameters: []), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> [Api.DialogFilter]? in + return (FunctionDescription(name: "messages.getDialogFilters", parameters: []), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.messages.DialogFilters? in let reader = BufferReader(buffer) - var result: [Api.DialogFilter]? - if let _ = reader.readInt32() { - result = Api.parseVector(reader, elementSignature: 0, elementType: Api.DialogFilter.self) + var result: Api.messages.DialogFilters? + if let signature = reader.readInt32() { + result = Api.parse(reader, signature: signature) as? Api.messages.DialogFilters } return result }) @@ -5804,6 +5999,43 @@ public extension Api.functions.messages { }) } } +public extension Api.functions.messages { + static func getQuickReplies(hash: Int64) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(-729550168) + serializeInt64(hash, buffer: buffer, boxed: false) + return (FunctionDescription(name: "messages.getQuickReplies", parameters: [("hash", String(describing: hash))]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.messages.QuickReplies? in + let reader = BufferReader(buffer) + var result: Api.messages.QuickReplies? + if let signature = reader.readInt32() { + result = Api.parse(reader, signature: signature) as? Api.messages.QuickReplies + } + return result + }) + } +} +public extension Api.functions.messages { + static func getQuickReplyMessages(flags: Int32, shortcutId: Int32, id: [Int32]?, hash: Int64) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(-1801153085) + serializeInt32(flags, buffer: buffer, boxed: false) + serializeInt32(shortcutId, buffer: buffer, boxed: false) + if Int(flags) & Int(1 << 0) != 0 {buffer.appendInt32(481674261) + buffer.appendInt32(Int32(id!.count)) + for item in id! { + serializeInt32(item, buffer: buffer, boxed: false) + }} + serializeInt64(hash, buffer: buffer, boxed: false) + return (FunctionDescription(name: "messages.getQuickReplyMessages", parameters: [("flags", String(describing: flags)), ("shortcutId", String(describing: shortcutId)), ("id", String(describing: id)), ("hash", String(describing: hash))]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.messages.Messages? in + let reader = BufferReader(buffer) + var result: Api.messages.Messages? + if let signature = reader.readInt32() { + result = Api.parse(reader, signature: signature) as? Api.messages.Messages + } + return result + }) + } +} public extension Api.functions.messages { static func getRecentLocations(peer: Api.InputPeer, limit: Int32, hash: Int64) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { let buffer = Buffer() @@ -6566,6 +6798,25 @@ public extension Api.functions.messages { }) } } +public extension Api.functions.messages { + static func reorderQuickReplies(order: [Int32]) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(1613961479) + buffer.appendInt32(481674261) + buffer.appendInt32(Int32(order.count)) + for item in order { + serializeInt32(item, buffer: buffer, boxed: false) + } + return (FunctionDescription(name: "messages.reorderQuickReplies", parameters: [("order", String(describing: order))]), 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 extension Api.functions.messages { static func reorderStickerSets(flags: Int32, order: [Int64]) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { let buffer = Buffer() @@ -7029,9 +7280,9 @@ public extension Api.functions.messages { } } public extension Api.functions.messages { - static func sendInlineBotResult(flags: Int32, peer: Api.InputPeer, replyTo: Api.InputReplyTo?, randomId: Int64, queryId: Int64, id: String, scheduleDate: Int32?, sendAs: Api.InputPeer?) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + static func sendInlineBotResult(flags: Int32, peer: Api.InputPeer, replyTo: Api.InputReplyTo?, randomId: Int64, queryId: Int64, id: String, scheduleDate: Int32?, sendAs: Api.InputPeer?, quickReplyShortcut: Api.InputQuickReplyShortcut?) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { let buffer = Buffer() - buffer.appendInt32(-138647366) + buffer.appendInt32(1052698730) serializeInt32(flags, buffer: buffer, boxed: false) peer.serialize(buffer, true) if Int(flags) & Int(1 << 0) != 0 {replyTo!.serialize(buffer, true)} @@ -7040,7 +7291,8 @@ public extension Api.functions.messages { serializeString(id, buffer: buffer, boxed: false) if Int(flags) & Int(1 << 10) != 0 {serializeInt32(scheduleDate!, buffer: buffer, boxed: false)} if Int(flags) & Int(1 << 13) != 0 {sendAs!.serialize(buffer, true)} - return (FunctionDescription(name: "messages.sendInlineBotResult", parameters: [("flags", String(describing: flags)), ("peer", String(describing: peer)), ("replyTo", String(describing: replyTo)), ("randomId", String(describing: randomId)), ("queryId", String(describing: queryId)), ("id", String(describing: id)), ("scheduleDate", String(describing: scheduleDate)), ("sendAs", String(describing: sendAs))]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.Updates? in + if Int(flags) & Int(1 << 17) != 0 {quickReplyShortcut!.serialize(buffer, true)} + return (FunctionDescription(name: "messages.sendInlineBotResult", parameters: [("flags", String(describing: flags)), ("peer", String(describing: peer)), ("replyTo", String(describing: replyTo)), ("randomId", String(describing: randomId)), ("queryId", String(describing: queryId)), ("id", String(describing: id)), ("scheduleDate", String(describing: scheduleDate)), ("sendAs", String(describing: sendAs)), ("quickReplyShortcut", String(describing: quickReplyShortcut))]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.Updates? in let reader = BufferReader(buffer) var result: Api.Updates? if let signature = reader.readInt32() { @@ -7051,9 +7303,9 @@ public extension Api.functions.messages { } } public extension Api.functions.messages { - static func sendMedia(flags: Int32, peer: Api.InputPeer, replyTo: Api.InputReplyTo?, media: Api.InputMedia, message: String, randomId: Int64, replyMarkup: Api.ReplyMarkup?, entities: [Api.MessageEntity]?, scheduleDate: Int32?, sendAs: Api.InputPeer?) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + static func sendMedia(flags: Int32, peer: Api.InputPeer, replyTo: Api.InputReplyTo?, media: Api.InputMedia, message: String, randomId: Int64, replyMarkup: Api.ReplyMarkup?, entities: [Api.MessageEntity]?, scheduleDate: Int32?, sendAs: Api.InputPeer?, quickReplyShortcut: Api.InputQuickReplyShortcut?) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { let buffer = Buffer() - buffer.appendInt32(1926021693) + buffer.appendInt32(2077646913) serializeInt32(flags, buffer: buffer, boxed: false) peer.serialize(buffer, true) if Int(flags) & Int(1 << 0) != 0 {replyTo!.serialize(buffer, true)} @@ -7068,7 +7320,8 @@ public extension Api.functions.messages { }} if Int(flags) & Int(1 << 10) != 0 {serializeInt32(scheduleDate!, buffer: buffer, boxed: false)} if Int(flags) & Int(1 << 13) != 0 {sendAs!.serialize(buffer, true)} - return (FunctionDescription(name: "messages.sendMedia", parameters: [("flags", String(describing: flags)), ("peer", String(describing: peer)), ("replyTo", String(describing: replyTo)), ("media", String(describing: media)), ("message", String(describing: message)), ("randomId", String(describing: randomId)), ("replyMarkup", String(describing: replyMarkup)), ("entities", String(describing: entities)), ("scheduleDate", String(describing: scheduleDate)), ("sendAs", String(describing: sendAs))]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.Updates? in + if Int(flags) & Int(1 << 17) != 0 {quickReplyShortcut!.serialize(buffer, true)} + return (FunctionDescription(name: "messages.sendMedia", parameters: [("flags", String(describing: flags)), ("peer", String(describing: peer)), ("replyTo", String(describing: replyTo)), ("media", String(describing: media)), ("message", String(describing: message)), ("randomId", String(describing: randomId)), ("replyMarkup", String(describing: replyMarkup)), ("entities", String(describing: entities)), ("scheduleDate", String(describing: scheduleDate)), ("sendAs", String(describing: sendAs)), ("quickReplyShortcut", String(describing: quickReplyShortcut))]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.Updates? in let reader = BufferReader(buffer) var result: Api.Updates? if let signature = reader.readInt32() { @@ -7079,9 +7332,9 @@ public extension Api.functions.messages { } } public extension Api.functions.messages { - static func sendMessage(flags: Int32, peer: Api.InputPeer, replyTo: Api.InputReplyTo?, message: String, randomId: Int64, replyMarkup: Api.ReplyMarkup?, entities: [Api.MessageEntity]?, scheduleDate: Int32?, sendAs: Api.InputPeer?) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + static func sendMessage(flags: Int32, peer: Api.InputPeer, replyTo: Api.InputReplyTo?, message: String, randomId: Int64, replyMarkup: Api.ReplyMarkup?, entities: [Api.MessageEntity]?, scheduleDate: Int32?, sendAs: Api.InputPeer?, quickReplyShortcut: Api.InputQuickReplyShortcut?) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { let buffer = Buffer() - buffer.appendInt32(671943023) + buffer.appendInt32(-537394132) serializeInt32(flags, buffer: buffer, boxed: false) peer.serialize(buffer, true) if Int(flags) & Int(1 << 0) != 0 {replyTo!.serialize(buffer, true)} @@ -7095,7 +7348,8 @@ public extension Api.functions.messages { }} if Int(flags) & Int(1 << 10) != 0 {serializeInt32(scheduleDate!, buffer: buffer, boxed: false)} if Int(flags) & Int(1 << 13) != 0 {sendAs!.serialize(buffer, true)} - return (FunctionDescription(name: "messages.sendMessage", parameters: [("flags", String(describing: flags)), ("peer", String(describing: peer)), ("replyTo", String(describing: replyTo)), ("message", String(describing: message)), ("randomId", String(describing: randomId)), ("replyMarkup", String(describing: replyMarkup)), ("entities", String(describing: entities)), ("scheduleDate", String(describing: scheduleDate)), ("sendAs", String(describing: sendAs))]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.Updates? in + if Int(flags) & Int(1 << 17) != 0 {quickReplyShortcut!.serialize(buffer, true)} + return (FunctionDescription(name: "messages.sendMessage", parameters: [("flags", String(describing: flags)), ("peer", String(describing: peer)), ("replyTo", String(describing: replyTo)), ("message", String(describing: message)), ("randomId", String(describing: randomId)), ("replyMarkup", String(describing: replyMarkup)), ("entities", String(describing: entities)), ("scheduleDate", String(describing: scheduleDate)), ("sendAs", String(describing: sendAs)), ("quickReplyShortcut", String(describing: quickReplyShortcut))]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.Updates? in let reader = BufferReader(buffer) var result: Api.Updates? if let signature = reader.readInt32() { @@ -7106,9 +7360,9 @@ public extension Api.functions.messages { } } public extension Api.functions.messages { - static func sendMultiMedia(flags: Int32, peer: Api.InputPeer, replyTo: Api.InputReplyTo?, multiMedia: [Api.InputSingleMedia], scheduleDate: Int32?, sendAs: Api.InputPeer?) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + static func sendMultiMedia(flags: Int32, peer: Api.InputPeer, replyTo: Api.InputReplyTo?, multiMedia: [Api.InputSingleMedia], scheduleDate: Int32?, sendAs: Api.InputPeer?, quickReplyShortcut: Api.InputQuickReplyShortcut?) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { let buffer = Buffer() - buffer.appendInt32(1164872071) + buffer.appendInt32(211175177) serializeInt32(flags, buffer: buffer, boxed: false) peer.serialize(buffer, true) if Int(flags) & Int(1 << 0) != 0 {replyTo!.serialize(buffer, true)} @@ -7119,7 +7373,24 @@ public extension Api.functions.messages { } if Int(flags) & Int(1 << 10) != 0 {serializeInt32(scheduleDate!, buffer: buffer, boxed: false)} if Int(flags) & Int(1 << 13) != 0 {sendAs!.serialize(buffer, true)} - return (FunctionDescription(name: "messages.sendMultiMedia", parameters: [("flags", String(describing: flags)), ("peer", String(describing: peer)), ("replyTo", String(describing: replyTo)), ("multiMedia", String(describing: multiMedia)), ("scheduleDate", String(describing: scheduleDate)), ("sendAs", String(describing: sendAs))]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.Updates? in + if Int(flags) & Int(1 << 17) != 0 {quickReplyShortcut!.serialize(buffer, true)} + return (FunctionDescription(name: "messages.sendMultiMedia", parameters: [("flags", String(describing: flags)), ("peer", String(describing: peer)), ("replyTo", String(describing: replyTo)), ("multiMedia", String(describing: multiMedia)), ("scheduleDate", String(describing: scheduleDate)), ("sendAs", String(describing: sendAs)), ("quickReplyShortcut", String(describing: quickReplyShortcut))]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.Updates? in + let reader = BufferReader(buffer) + var result: Api.Updates? + if let signature = reader.readInt32() { + result = Api.parse(reader, signature: signature) as? Api.Updates + } + return result + }) + } +} +public extension Api.functions.messages { + static func sendQuickReplyMessages(peer: Api.InputPeer, shortcutId: Int32) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(857029332) + peer.serialize(buffer, true) + serializeInt32(shortcutId, buffer: buffer, boxed: false) + return (FunctionDescription(name: "messages.sendQuickReplyMessages", parameters: [("peer", String(describing: peer)), ("shortcutId", String(describing: shortcutId))]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.Updates? in let reader = BufferReader(buffer) var result: Api.Updates? if let signature = reader.readInt32() { @@ -7545,6 +7816,21 @@ public extension Api.functions.messages { }) } } +public extension Api.functions.messages { + static func toggleDialogFilterTags(enabled: Api.Bool) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(-47326647) + enabled.serialize(buffer, true) + return (FunctionDescription(name: "messages.toggleDialogFilterTags", parameters: [("enabled", String(describing: enabled))]), 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 extension Api.functions.messages { static func toggleDialogPin(flags: Int32, peer: Api.InputDialogPeer) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { let buffer = Buffer() @@ -8795,6 +9081,113 @@ public extension Api.functions.premium { }) } } +public extension Api.functions.smsjobs { + static func finishJob(flags: Int32, jobId: String, error: String?) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(1327415076) + serializeInt32(flags, buffer: buffer, boxed: false) + serializeString(jobId, buffer: buffer, boxed: false) + if Int(flags) & Int(1 << 0) != 0 {serializeString(error!, buffer: buffer, boxed: false)} + return (FunctionDescription(name: "smsjobs.finishJob", parameters: [("flags", String(describing: flags)), ("jobId", String(describing: jobId)), ("error", String(describing: error))]), 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 extension Api.functions.smsjobs { + static func getSmsJob(jobId: String) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(2005766191) + serializeString(jobId, buffer: buffer, boxed: false) + return (FunctionDescription(name: "smsjobs.getSmsJob", parameters: [("jobId", String(describing: jobId))]), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.SmsJob? in + let reader = BufferReader(buffer) + var result: Api.SmsJob? + if let signature = reader.readInt32() { + result = Api.parse(reader, signature: signature) as? Api.SmsJob + } + return result + }) + } +} +public extension Api.functions.smsjobs { + static func getStatus() -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(279353576) + + return (FunctionDescription(name: "smsjobs.getStatus", parameters: []), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.smsjobs.Status? in + let reader = BufferReader(buffer) + var result: Api.smsjobs.Status? + if let signature = reader.readInt32() { + result = Api.parse(reader, signature: signature) as? Api.smsjobs.Status + } + return result + }) + } +} +public extension Api.functions.smsjobs { + static func isEligibleToJoin() -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(249313744) + + return (FunctionDescription(name: "smsjobs.isEligibleToJoin", parameters: []), buffer, DeserializeFunctionResponse { (buffer: Buffer) -> Api.smsjobs.EligibilityToJoin? in + let reader = BufferReader(buffer) + var result: Api.smsjobs.EligibilityToJoin? + if let signature = reader.readInt32() { + result = Api.parse(reader, signature: signature) as? Api.smsjobs.EligibilityToJoin + } + return result + }) + } +} +public extension Api.functions.smsjobs { + static func join() -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(-1488007635) + + return (FunctionDescription(name: "smsjobs.join", parameters: []), 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 extension Api.functions.smsjobs { + static func leave() -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(-1734824589) + + return (FunctionDescription(name: "smsjobs.leave", parameters: []), 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 extension Api.functions.smsjobs { + static func updateSettings(flags: Int32) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { + let buffer = Buffer() + buffer.appendInt32(155164863) + serializeInt32(flags, buffer: buffer, boxed: false) + return (FunctionDescription(name: "smsjobs.updateSettings", parameters: [("flags", String(describing: flags))]), 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 extension Api.functions.stats { static func getBroadcastStats(flags: Int32, channel: Api.InputChannel) -> (FunctionDescription, Buffer, DeserializeFunctionResponse) { let buffer = Buffer() diff --git a/submodules/TelegramApi/Sources/Api4.swift b/submodules/TelegramApi/Sources/Api4.swift index 9a71f34213d..6e5cd0aab4b 100644 --- a/submodules/TelegramApi/Sources/Api4.swift +++ b/submodules/TelegramApi/Sources/Api4.swift @@ -662,6 +662,52 @@ public extension Api { } } +public extension Api { + enum ConnectedBot: TypeConstructorDescription { + case connectedBot(flags: Int32, botId: Int64, recipients: Api.BusinessRecipients) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .connectedBot(let flags, let botId, let recipients): + if boxed { + buffer.appendInt32(-404121113) + } + serializeInt32(flags, buffer: buffer, boxed: false) + serializeInt64(botId, buffer: buffer, boxed: false) + recipients.serialize(buffer, true) + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .connectedBot(let flags, let botId, let recipients): + return ("connectedBot", [("flags", flags as Any), ("botId", botId as Any), ("recipients", recipients as Any)]) + } + } + + public static func parse_connectedBot(_ reader: BufferReader) -> ConnectedBot? { + var _1: Int32? + _1 = reader.readInt32() + var _2: Int64? + _2 = reader.readInt64() + var _3: Api.BusinessRecipients? + if let signature = reader.readInt32() { + _3 = Api.parse(reader, signature: signature) as? Api.BusinessRecipients + } + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + if _c1 && _c2 && _c3 { + return Api.ConnectedBot.connectedBot(flags: _1!, botId: _2!, recipients: _3!) + } + else { + return nil + } + } + + } +} public extension Api { enum Contact: TypeConstructorDescription { case contact(userId: Int64, mutual: Api.Bool) @@ -1014,20 +1060,21 @@ public extension Api { } public extension Api { enum DialogFilter: TypeConstructorDescription { - case dialogFilter(flags: Int32, id: Int32, title: String, emoticon: String?, pinnedPeers: [Api.InputPeer], includePeers: [Api.InputPeer], excludePeers: [Api.InputPeer]) - case dialogFilterChatlist(flags: Int32, id: Int32, title: String, emoticon: String?, pinnedPeers: [Api.InputPeer], includePeers: [Api.InputPeer]) + case dialogFilter(flags: Int32, id: Int32, title: String, emoticon: String?, color: Int32?, pinnedPeers: [Api.InputPeer], includePeers: [Api.InputPeer], excludePeers: [Api.InputPeer]) + case dialogFilterChatlist(flags: Int32, id: Int32, title: String, emoticon: String?, color: Int32?, pinnedPeers: [Api.InputPeer], includePeers: [Api.InputPeer]) case dialogFilterDefault public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { switch self { - case .dialogFilter(let flags, let id, let title, let emoticon, let pinnedPeers, let includePeers, let excludePeers): + case .dialogFilter(let flags, let id, let title, let emoticon, let color, let pinnedPeers, let includePeers, let excludePeers): if boxed { - buffer.appendInt32(1949890536) + buffer.appendInt32(1605718587) } serializeInt32(flags, buffer: buffer, boxed: false) serializeInt32(id, buffer: buffer, boxed: false) serializeString(title, buffer: buffer, boxed: false) if Int(flags) & Int(1 << 25) != 0 {serializeString(emoticon!, buffer: buffer, boxed: false)} + if Int(flags) & Int(1 << 27) != 0 {serializeInt32(color!, buffer: buffer, boxed: false)} buffer.appendInt32(481674261) buffer.appendInt32(Int32(pinnedPeers.count)) for item in pinnedPeers { @@ -1044,14 +1091,15 @@ public extension Api { item.serialize(buffer, true) } break - case .dialogFilterChatlist(let flags, let id, let title, let emoticon, let pinnedPeers, let includePeers): + case .dialogFilterChatlist(let flags, let id, let title, let emoticon, let color, let pinnedPeers, let includePeers): if boxed { - buffer.appendInt32(-699792216) + buffer.appendInt32(-1612542300) } serializeInt32(flags, buffer: buffer, boxed: false) serializeInt32(id, buffer: buffer, boxed: false) serializeString(title, buffer: buffer, boxed: false) if Int(flags) & Int(1 << 25) != 0 {serializeString(emoticon!, buffer: buffer, boxed: false)} + if Int(flags) & Int(1 << 27) != 0 {serializeInt32(color!, buffer: buffer, boxed: false)} buffer.appendInt32(481674261) buffer.appendInt32(Int32(pinnedPeers.count)) for item in pinnedPeers { @@ -1074,10 +1122,10 @@ public extension Api { public func descriptionFields() -> (String, [(String, Any)]) { switch self { - case .dialogFilter(let flags, let id, let title, let emoticon, let pinnedPeers, let includePeers, let excludePeers): - return ("dialogFilter", [("flags", flags as Any), ("id", id as Any), ("title", title as Any), ("emoticon", emoticon as Any), ("pinnedPeers", pinnedPeers as Any), ("includePeers", includePeers as Any), ("excludePeers", excludePeers as Any)]) - case .dialogFilterChatlist(let flags, let id, let title, let emoticon, let pinnedPeers, let includePeers): - return ("dialogFilterChatlist", [("flags", flags as Any), ("id", id as Any), ("title", title as Any), ("emoticon", emoticon as Any), ("pinnedPeers", pinnedPeers as Any), ("includePeers", includePeers as Any)]) + case .dialogFilter(let flags, let id, let title, let emoticon, let color, let pinnedPeers, let includePeers, let excludePeers): + return ("dialogFilter", [("flags", flags as Any), ("id", id as Any), ("title", title as Any), ("emoticon", emoticon as Any), ("color", color as Any), ("pinnedPeers", pinnedPeers as Any), ("includePeers", includePeers as Any), ("excludePeers", excludePeers as Any)]) + case .dialogFilterChatlist(let flags, let id, let title, let emoticon, let color, let pinnedPeers, let includePeers): + return ("dialogFilterChatlist", [("flags", flags as Any), ("id", id as Any), ("title", title as Any), ("emoticon", emoticon as Any), ("color", color as Any), ("pinnedPeers", pinnedPeers as Any), ("includePeers", includePeers as Any)]) case .dialogFilterDefault: return ("dialogFilterDefault", []) } @@ -1092,10 +1140,8 @@ public extension Api { _3 = parseString(reader) var _4: String? if Int(_1!) & Int(1 << 25) != 0 {_4 = parseString(reader) } - var _5: [Api.InputPeer]? - if let _ = reader.readInt32() { - _5 = Api.parseVector(reader, elementSignature: 0, elementType: Api.InputPeer.self) - } + var _5: Int32? + if Int(_1!) & Int(1 << 27) != 0 {_5 = reader.readInt32() } var _6: [Api.InputPeer]? if let _ = reader.readInt32() { _6 = Api.parseVector(reader, elementSignature: 0, elementType: Api.InputPeer.self) @@ -1104,15 +1150,20 @@ public extension Api { if let _ = reader.readInt32() { _7 = Api.parseVector(reader, elementSignature: 0, elementType: Api.InputPeer.self) } + var _8: [Api.InputPeer]? + if let _ = reader.readInt32() { + _8 = Api.parseVector(reader, elementSignature: 0, elementType: Api.InputPeer.self) + } let _c1 = _1 != nil let _c2 = _2 != nil let _c3 = _3 != nil let _c4 = (Int(_1!) & Int(1 << 25) == 0) || _4 != nil - let _c5 = _5 != nil + let _c5 = (Int(_1!) & Int(1 << 27) == 0) || _5 != nil let _c6 = _6 != nil let _c7 = _7 != nil - if _c1 && _c2 && _c3 && _c4 && _c5 && _c6 && _c7 { - return Api.DialogFilter.dialogFilter(flags: _1!, id: _2!, title: _3!, emoticon: _4, pinnedPeers: _5!, includePeers: _6!, excludePeers: _7!) + let _c8 = _8 != nil + if _c1 && _c2 && _c3 && _c4 && _c5 && _c6 && _c7 && _c8 { + return Api.DialogFilter.dialogFilter(flags: _1!, id: _2!, title: _3!, emoticon: _4, color: _5, pinnedPeers: _6!, includePeers: _7!, excludePeers: _8!) } else { return nil @@ -1127,22 +1178,25 @@ public extension Api { _3 = parseString(reader) var _4: String? if Int(_1!) & Int(1 << 25) != 0 {_4 = parseString(reader) } - var _5: [Api.InputPeer]? - if let _ = reader.readInt32() { - _5 = Api.parseVector(reader, elementSignature: 0, elementType: Api.InputPeer.self) - } + var _5: Int32? + if Int(_1!) & Int(1 << 27) != 0 {_5 = reader.readInt32() } var _6: [Api.InputPeer]? if let _ = reader.readInt32() { _6 = Api.parseVector(reader, elementSignature: 0, elementType: Api.InputPeer.self) } + var _7: [Api.InputPeer]? + if let _ = reader.readInt32() { + _7 = Api.parseVector(reader, elementSignature: 0, elementType: Api.InputPeer.self) + } let _c1 = _1 != nil let _c2 = _2 != nil let _c3 = _3 != nil let _c4 = (Int(_1!) & Int(1 << 25) == 0) || _4 != nil - let _c5 = _5 != nil + let _c5 = (Int(_1!) & Int(1 << 27) == 0) || _5 != nil let _c6 = _6 != nil - if _c1 && _c2 && _c3 && _c4 && _c5 && _c6 { - return Api.DialogFilter.dialogFilterChatlist(flags: _1!, id: _2!, title: _3!, emoticon: _4, pinnedPeers: _5!, includePeers: _6!) + let _c7 = _7 != nil + if _c1 && _c2 && _c3 && _c4 && _c5 && _c6 && _c7 { + return Api.DialogFilter.dialogFilterChatlist(flags: _1!, id: _2!, title: _3!, emoticon: _4, color: _5, pinnedPeers: _6!, includePeers: _7!) } else { return nil diff --git a/submodules/TelegramApi/Sources/Api7.swift b/submodules/TelegramApi/Sources/Api7.swift index 80bc038f382..cbdf0ebf38a 100644 --- a/submodules/TelegramApi/Sources/Api7.swift +++ b/submodules/TelegramApi/Sources/Api7.swift @@ -262,6 +262,150 @@ public extension Api { } } +public extension Api { + enum InputBusinessAwayMessage: TypeConstructorDescription { + case inputBusinessAwayMessage(flags: Int32, shortcutId: Int32, schedule: Api.BusinessAwayMessageSchedule, recipients: Api.InputBusinessRecipients) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .inputBusinessAwayMessage(let flags, let shortcutId, let schedule, let recipients): + if boxed { + buffer.appendInt32(-2094959136) + } + serializeInt32(flags, buffer: buffer, boxed: false) + serializeInt32(shortcutId, buffer: buffer, boxed: false) + schedule.serialize(buffer, true) + recipients.serialize(buffer, true) + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .inputBusinessAwayMessage(let flags, let shortcutId, let schedule, let recipients): + return ("inputBusinessAwayMessage", [("flags", flags as Any), ("shortcutId", shortcutId as Any), ("schedule", schedule as Any), ("recipients", recipients as Any)]) + } + } + + public static func parse_inputBusinessAwayMessage(_ reader: BufferReader) -> InputBusinessAwayMessage? { + var _1: Int32? + _1 = reader.readInt32() + var _2: Int32? + _2 = reader.readInt32() + var _3: Api.BusinessAwayMessageSchedule? + if let signature = reader.readInt32() { + _3 = Api.parse(reader, signature: signature) as? Api.BusinessAwayMessageSchedule + } + var _4: Api.InputBusinessRecipients? + if let signature = reader.readInt32() { + _4 = Api.parse(reader, signature: signature) as? Api.InputBusinessRecipients + } + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + let _c4 = _4 != nil + if _c1 && _c2 && _c3 && _c4 { + return Api.InputBusinessAwayMessage.inputBusinessAwayMessage(flags: _1!, shortcutId: _2!, schedule: _3!, recipients: _4!) + } + else { + return nil + } + } + + } +} +public extension Api { + enum InputBusinessGreetingMessage: TypeConstructorDescription { + case inputBusinessGreetingMessage(shortcutId: Int32, recipients: Api.InputBusinessRecipients, noActivityDays: Int32) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .inputBusinessGreetingMessage(let shortcutId, let recipients, let noActivityDays): + if boxed { + buffer.appendInt32(26528571) + } + serializeInt32(shortcutId, buffer: buffer, boxed: false) + recipients.serialize(buffer, true) + serializeInt32(noActivityDays, buffer: buffer, boxed: false) + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .inputBusinessGreetingMessage(let shortcutId, let recipients, let noActivityDays): + return ("inputBusinessGreetingMessage", [("shortcutId", shortcutId as Any), ("recipients", recipients as Any), ("noActivityDays", noActivityDays as Any)]) + } + } + + public static func parse_inputBusinessGreetingMessage(_ reader: BufferReader) -> InputBusinessGreetingMessage? { + var _1: Int32? + _1 = reader.readInt32() + var _2: Api.InputBusinessRecipients? + if let signature = reader.readInt32() { + _2 = Api.parse(reader, signature: signature) as? Api.InputBusinessRecipients + } + var _3: Int32? + _3 = reader.readInt32() + let _c1 = _1 != nil + let _c2 = _2 != nil + let _c3 = _3 != nil + if _c1 && _c2 && _c3 { + return Api.InputBusinessGreetingMessage.inputBusinessGreetingMessage(shortcutId: _1!, recipients: _2!, noActivityDays: _3!) + } + else { + return nil + } + } + + } +} +public extension Api { + enum InputBusinessRecipients: TypeConstructorDescription { + case inputBusinessRecipients(flags: Int32, users: [Api.InputUser]?) + + public func serialize(_ buffer: Buffer, _ boxed: Swift.Bool) { + switch self { + case .inputBusinessRecipients(let flags, let users): + if boxed { + buffer.appendInt32(1871393450) + } + serializeInt32(flags, buffer: buffer, boxed: false) + if Int(flags) & Int(1 << 4) != 0 {buffer.appendInt32(481674261) + buffer.appendInt32(Int32(users!.count)) + for item in users! { + item.serialize(buffer, true) + }} + break + } + } + + public func descriptionFields() -> (String, [(String, Any)]) { + switch self { + case .inputBusinessRecipients(let flags, let users): + return ("inputBusinessRecipients", [("flags", flags as Any), ("users", users as Any)]) + } + } + + public static func parse_inputBusinessRecipients(_ reader: BufferReader) -> InputBusinessRecipients? { + var _1: Int32? + _1 = reader.readInt32() + var _2: [Api.InputUser]? + if Int(_1!) & Int(1 << 4) != 0 {if let _ = reader.readInt32() { + _2 = Api.parseVector(reader, elementSignature: 0, elementType: Api.InputUser.self) + } } + let _c1 = _1 != nil + let _c2 = (Int(_1!) & Int(1 << 4) == 0) || _2 != nil + if _c1 && _c2 { + return Api.InputBusinessRecipients.inputBusinessRecipients(flags: _1!, users: _2) + } + else { + return nil + } + } + + } +} public extension Api { indirect enum InputChannel: TypeConstructorDescription { case inputChannel(channelId: Int64, accessHash: Int64) diff --git a/submodules/TelegramCore/Sources/Account/AccountIntermediateState.swift b/submodules/TelegramCore/Sources/Account/AccountIntermediateState.swift index 0d1df5f992f..79c3c4658a1 100644 --- a/submodules/TelegramCore/Sources/Account/AccountIntermediateState.swift +++ b/submodules/TelegramCore/Sources/Account/AccountIntermediateState.swift @@ -62,6 +62,7 @@ enum AccountStateGlobalNotificationSettingsSubject { enum AccountStateMutationOperation { case AddMessages([StoreMessage], AddMessagesLocation) case AddScheduledMessages([StoreMessage]) + case AddQuickReplyMessages([StoreMessage]) case DeleteMessagesWithGlobalIds([Int32]) case DeleteMessages([MessageId]) case EditMessage(MessageId, StoreMessage) @@ -332,6 +333,10 @@ struct AccountMutableState { self.addOperation(.AddScheduledMessages(messages)) } + mutating func addQuickReplyMessages(_ messages: [StoreMessage]) { + self.addOperation(.AddQuickReplyMessages(messages)) + } + mutating func addDisplayAlert(_ text: String, isDropAuth: Bool) { self.displayAlerts.append((text: text, isDropAuth: isDropAuth)) } @@ -709,6 +714,19 @@ struct AccountMutableState { } } } + case let .AddQuickReplyMessages(messages): + for message in messages { + if case let .Id(id) = message.id { + self.storedMessages.insert(id) + inner: for attribute in message.attributes { + if let attribute = attribute as? ReplyMessageAttribute { + self.referencedReplyMessageIds.add(sourceId: id, targetId: attribute.messageId) + } else if let attribute = attribute as? ReplyStoryAttribute { + self.referencedStoryIds.insert(attribute.storyId) + } + } + } + } case let .UpdateState(state): self.state = state case let .UpdateChannelState(peerId, pts): diff --git a/submodules/TelegramCore/Sources/Account/AccountManager.swift b/submodules/TelegramCore/Sources/Account/AccountManager.swift index b302e346b5b..e7808d6f7da 100644 --- a/submodules/TelegramCore/Sources/Account/AccountManager.swift +++ b/submodules/TelegramCore/Sources/Account/AccountManager.swift @@ -293,6 +293,7 @@ private var declaredEncodables: Void = { declareEncodable(WebpagePreviewMessageAttribute.self, f: { WebpagePreviewMessageAttribute(decoder: $0) }) declareEncodable(DerivedDataMessageAttribute.self, f: { DerivedDataMessageAttribute(decoder: $0) }) declareEncodable(TelegramApplicationIcons.self, f: { TelegramApplicationIcons(decoder: $0) }) + declareEncodable(OutgoingQuickReplyMessageAttribute.self, f: { OutgoingQuickReplyMessageAttribute(decoder: $0) }) return }() diff --git a/submodules/TelegramCore/Sources/ApiUtils/StoreMessage_Telegram.swift b/submodules/TelegramCore/Sources/ApiUtils/StoreMessage_Telegram.swift index 76754ca04cb..5ecd19cfc1b 100644 --- a/submodules/TelegramCore/Sources/ApiUtils/StoreMessage_Telegram.swift +++ b/submodules/TelegramCore/Sources/ApiUtils/StoreMessage_Telegram.swift @@ -126,7 +126,7 @@ public func tagsForStoreMessage(incoming: Bool, attributes: [MessageAttribute], func apiMessagePeerId(_ messsage: Api.Message) -> PeerId? { switch messsage { - case let .message(_, _, _, _, messagePeerId, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _): + case let .message(_, _, _, _, messagePeerId, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _): let chatPeerId = messagePeerId return chatPeerId.peerId case let .messageEmpty(_, _, peerId): @@ -142,7 +142,7 @@ func apiMessagePeerId(_ messsage: Api.Message) -> PeerId? { func apiMessagePeerIds(_ message: Api.Message) -> [PeerId] { switch message { - case let .message(_, _, fromId, _, chatPeerId, savedPeerId, fwdHeader, viaBotId, replyTo, _, _, media, _, entities, _, _, _, _, _, _, _, _, _): + case let .message(_, _, fromId, _, chatPeerId, savedPeerId, fwdHeader, viaBotId, replyTo, _, _, media, _, entities, _, _, _, _, _, _, _, _, _, _): let peerId: PeerId = chatPeerId.peerId var result = [peerId] @@ -263,7 +263,7 @@ func apiMessagePeerIds(_ message: Api.Message) -> [PeerId] { func apiMessageAssociatedMessageIds(_ message: Api.Message) -> (replyIds: ReferencedReplyMessageIds, generalIds: [MessageId])? { switch message { - case let .message(_, id, _, _, chatPeerId, _, _, _, replyTo, _, _, _, _, _, _, _, _, _, _, _, _, _, _): + case let .message(_, id, _, _, chatPeerId, _, _, _, replyTo, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _): if let replyTo = replyTo { let peerId: PeerId = chatPeerId.peerId @@ -597,8 +597,13 @@ func messageTextEntitiesFromApiEntities(_ entities: [Api.MessageEntity]) -> [Mes extension StoreMessage { convenience init?(apiMessage: Api.Message, accountPeerId: PeerId, peerIsForum: Bool, namespace: MessageId.Namespace = Namespaces.Message.Cloud) { switch apiMessage { - case let .message(flags, id, fromId, boosts, chatPeerId, savedPeerId, fwdFrom, viaBotId, replyTo, date, message, media, replyMarkup, entities, views, forwards, replies, editDate, postAuthor, groupingId, reactions, restrictionReason, ttlPeriod): + case let .message(flags, id, fromId, boosts, chatPeerId, savedPeerId, fwdFrom, viaBotId, replyTo, date, message, media, replyMarkup, entities, views, forwards, replies, editDate, postAuthor, groupingId, reactions, restrictionReason, ttlPeriod, quickReplyShortcutId): let resolvedFromId = fromId?.peerId ?? chatPeerId.peerId + + var namespace = namespace + if quickReplyShortcutId != nil { + namespace = Namespaces.Message.QuickReplyCloud + } let peerId: PeerId var authorId: PeerId? @@ -663,7 +668,7 @@ extension StoreMessage { threadId = Int64(threadIdValue.id) } } - attributes.append(ReplyMessageAttribute(messageId: MessageId(peerId: replyPeerId, namespace: Namespaces.Message.Cloud, id: replyToMsgId), threadMessageId: threadMessageId, quote: quote, isQuote: isQuote)) + attributes.append(ReplyMessageAttribute(messageId: MessageId(peerId: replyPeerId, namespace: namespace, id: replyToMsgId), threadMessageId: threadMessageId, quote: quote, isQuote: isQuote)) } if let replyHeader = replyHeader { attributes.append(QuotedReplyMessageAttribute(apiHeader: replyHeader, quote: quote, isQuote: isQuote)) @@ -729,6 +734,10 @@ extension StoreMessage { if peerId == accountPeerId, let savedPeerId = savedPeerId { threadId = savedPeerId.peerId.toInt64() } + + if let quickReplyShortcutId { + threadId = Int64(quickReplyShortcutId) + } let messageText = message var medias: [Media] = [] @@ -791,7 +800,7 @@ extension StoreMessage { attributes.append(InlineBotMessageAttribute(peerId: PeerId(namespace: Namespaces.Peer.CloudUser, id: PeerId.Id._internalFromInt64Value(viaBotId)), title: nil)) } - if namespace != Namespaces.Message.ScheduledCloud { + if namespace != Namespaces.Message.ScheduledCloud && namespace != Namespaces.Message.QuickReplyCloud { if let views = views { attributes.append(ViewCountMessageAttribute(count: Int(views))) } diff --git a/submodules/TelegramCore/Sources/PendingMessages/EnqueueMessage.swift b/submodules/TelegramCore/Sources/PendingMessages/EnqueueMessage.swift index 73a643cec61..300da992bd4 100644 --- a/submodules/TelegramCore/Sources/PendingMessages/EnqueueMessage.swift +++ b/submodules/TelegramCore/Sources/PendingMessages/EnqueueMessage.swift @@ -138,6 +138,15 @@ public enum EnqueueMessage { } } + public func withUpdatedThreadId(_ threadId: Int64?) -> EnqueueMessage { + switch self { + case let .message(text, attributes, inlineStickers, mediaReference, _, replyToMessageId, replyToStoryId, localGroupingKey, correlationId, bubbleUpEmojiOrStickersets): + return .message(text: text, attributes: attributes, inlineStickers: inlineStickers, mediaReference: mediaReference, threadId: threadId, replyToMessageId: replyToMessageId, replyToStoryId: replyToStoryId, localGroupingKey: localGroupingKey, correlationId: correlationId, bubbleUpEmojiOrStickersets: bubbleUpEmojiOrStickersets) + case let .forward(source, _, grouping, attributes, correlationId, asCopy): + return .forward(source: source, threadId: threadId, grouping: grouping, attributes: attributes, correlationId: correlationId, asCopy: asCopy) + } + } + public var groupingKey: Int64? { if case let .message(_, _, _, _, _, _, _, localGroupingKey, _, _) = self { return localGroupingKey @@ -225,6 +234,8 @@ private func filterMessageAttributesForOutgoingMessage(_ attributes: [MessageAtt return true case _ as OutgoingScheduleInfoMessageAttribute: return true + case _ as OutgoingQuickReplyMessageAttribute: + return true case _ as EmbeddedMediaStickersMessageAttribute: return true case _ as EmojiSearchQueryMessageAttribute: @@ -254,6 +265,8 @@ private func filterMessageAttributesForForwardedMessage(_ attributes: [MessageAt return true case _ as OutgoingScheduleInfoMessageAttribute: return true + case _ as OutgoingQuickReplyMessageAttribute: + return true case _ as ForwardOptionsMessageAttribute: return true case _ as SendAsMessageAttribute: @@ -679,6 +692,9 @@ func enqueueMessages(transaction: Transaction, account: Account, peerId: PeerId, messageNamespace = Namespaces.Message.ScheduledLocal effectiveTimestamp = attribute.scheduleTime } + } else if attribute is OutgoingQuickReplyMessageAttribute { + messageNamespace = Namespaces.Message.QuickReplyLocal + effectiveTimestamp = 0 } else if let attribute = attribute as? SendAsMessageAttribute { if let peer = transaction.getPeer(attribute.peerId) { sendAsPeer = peer @@ -704,11 +720,14 @@ func enqueueMessages(transaction: Transaction, account: Account, peerId: PeerId, if messageNamespace != Namespaces.Message.ScheduledLocal { attributes.removeAll(where: { $0 is OutgoingScheduleInfoMessageAttribute }) } + if messageNamespace != Namespaces.Message.QuickReplyLocal { + attributes.removeAll(where: { $0 is OutgoingQuickReplyMessageAttribute }) + } if let peer = peer as? TelegramChannel { switch peer.info { case let .broadcast(info): - if messageNamespace != Namespaces.Message.ScheduledLocal { + if messageNamespace != Namespaces.Message.ScheduledLocal && messageNamespace != Namespaces.Message.QuickReplyLocal { attributes.append(ViewCountMessageAttribute(count: 1)) } if info.flags.contains(.messagesShouldHaveSignatures) { @@ -917,6 +936,9 @@ func enqueueMessages(transaction: Transaction, account: Account, peerId: PeerId, messageNamespace = Namespaces.Message.ScheduledLocal effectiveTimestamp = attribute.scheduleTime } + } else if attribute is OutgoingQuickReplyMessageAttribute { + messageNamespace = Namespaces.Message.QuickReplyLocal + effectiveTimestamp = 0 } else if let attribute = attribute as? ReplyMessageAttribute { if let threadMessageId = attribute.threadMessageId { threadId = Int64(threadMessageId.id) @@ -946,6 +968,9 @@ func enqueueMessages(transaction: Transaction, account: Account, peerId: PeerId, if messageNamespace != Namespaces.Message.ScheduledLocal { attributes.removeAll(where: { $0 is OutgoingScheduleInfoMessageAttribute }) } + if messageNamespace != Namespaces.Message.QuickReplyLocal { + attributes.removeAll(where: { $0 is OutgoingQuickReplyMessageAttribute }) + } let (tags, globalTags) = tagsForStoreMessage(incoming: false, attributes: attributes, media: sourceMessage.media, textEntities: entitiesAttribute?.entities, isPinned: false) diff --git a/submodules/TelegramCore/Sources/PendingMessages/RequestEditMessage.swift b/submodules/TelegramCore/Sources/PendingMessages/RequestEditMessage.swift index 2532a673a46..87447920860 100644 --- a/submodules/TelegramCore/Sources/PendingMessages/RequestEditMessage.swift +++ b/submodules/TelegramCore/Sources/PendingMessages/RequestEditMessage.swift @@ -174,7 +174,13 @@ private func requestEditMessageInternal(accountPeerId: PeerId, postbox: Postbox, } } - return network.request(Api.functions.messages.editMessage(flags: flags, peer: inputPeer, id: messageId.id, message: text, media: inputMedia, replyMarkup: nil, entities: apiEntities, scheduleDate: effectiveScheduleTime)) + var quickReplyShortcutId: Int32? + if messageId.namespace == Namespaces.Message.QuickReplyCloud { + quickReplyShortcutId = Int32(clamping: message.threadId ?? 0) + flags |= Int32(1 << 17) + } + + return network.request(Api.functions.messages.editMessage(flags: flags, peer: inputPeer, id: messageId.id, message: text, media: inputMedia, replyMarkup: nil, entities: apiEntities, scheduleDate: effectiveScheduleTime, quickReplyShortcutId: quickReplyShortcutId)) |> map { result -> Api.Updates? in return result } @@ -304,7 +310,7 @@ func _internal_requestEditLiveLocation(postbox: Postbox, network: Network, state inputMedia = .inputMediaGeoLive(flags: 1 << 0, geoPoint: .inputGeoPoint(flags: 0, lat: media.latitude, long: media.longitude, accuracyRadius: nil), heading: nil, period: nil, proximityNotificationRadius: nil) } - return network.request(Api.functions.messages.editMessage(flags: 1 << 14, peer: inputPeer, id: messageId.id, message: nil, media: inputMedia, replyMarkup: nil, entities: nil, scheduleDate: nil)) + return network.request(Api.functions.messages.editMessage(flags: 1 << 14, peer: inputPeer, id: messageId.id, message: nil, media: inputMedia, replyMarkup: nil, entities: nil, scheduleDate: nil, quickReplyShortcutId: nil)) |> map(Optional.init) |> `catch` { _ -> Signal in return .single(nil) diff --git a/submodules/TelegramCore/Sources/PendingMessages/StandaloneSendMessage.swift b/submodules/TelegramCore/Sources/PendingMessages/StandaloneSendMessage.swift index 5496447f8c9..bfeb8299480 100644 --- a/submodules/TelegramCore/Sources/PendingMessages/StandaloneSendMessage.swift +++ b/submodules/TelegramCore/Sources/PendingMessages/StandaloneSendMessage.swift @@ -410,7 +410,7 @@ private func sendUploadedMessageContent( } } - sendMessageRequest = network.requestWithAdditionalInfo(Api.functions.messages.sendMessage(flags: flags, peer: inputPeer, replyTo: replyTo, message: text, randomId: uniqueId, replyMarkup: nil, entities: messageEntities, scheduleDate: scheduleTime, sendAs: sendAsInputPeer), info: .acknowledgement, tag: dependencyTag) + sendMessageRequest = network.requestWithAdditionalInfo(Api.functions.messages.sendMessage(flags: flags, peer: inputPeer, replyTo: replyTo, message: text, randomId: uniqueId, replyMarkup: nil, entities: messageEntities, scheduleDate: scheduleTime, sendAs: sendAsInputPeer, quickReplyShortcut: nil), info: .acknowledgement, tag: dependencyTag) case let .media(inputMedia, text): if bubbleUpEmojiOrStickersets { flags |= Int32(1 << 15) @@ -432,7 +432,7 @@ private func sendUploadedMessageContent( } } - sendMessageRequest = network.request(Api.functions.messages.sendMedia(flags: flags, peer: inputPeer, replyTo: replyTo, media: inputMedia, message: text, randomId: uniqueId, replyMarkup: nil, entities: messageEntities, scheduleDate: scheduleTime, sendAs: sendAsInputPeer), tag: dependencyTag) + sendMessageRequest = network.request(Api.functions.messages.sendMedia(flags: flags, peer: inputPeer, replyTo: replyTo, media: inputMedia, message: text, randomId: uniqueId, replyMarkup: nil, entities: messageEntities, scheduleDate: scheduleTime, sendAs: sendAsInputPeer, quickReplyShortcut: nil), tag: dependencyTag) |> map(NetworkRequestResult.result) case let .forward(sourceInfo): var topMsgId: Int32? @@ -442,7 +442,7 @@ private func sendUploadedMessageContent( } if let forwardSourceInfoAttribute = forwardSourceInfoAttribute, let sourcePeer = transaction.getPeer(forwardSourceInfoAttribute.messageId.peerId), let sourceInputPeer = apiInputPeer(sourcePeer) { - sendMessageRequest = network.request(Api.functions.messages.forwardMessages(flags: flags, fromPeer: sourceInputPeer, id: [sourceInfo.messageId.id], randomId: [uniqueId], toPeer: inputPeer, topMsgId: topMsgId, scheduleDate: scheduleTime, sendAs: sendAsInputPeer), tag: dependencyTag) + sendMessageRequest = network.request(Api.functions.messages.forwardMessages(flags: flags, fromPeer: sourceInputPeer, id: [sourceInfo.messageId.id], randomId: [uniqueId], toPeer: inputPeer, topMsgId: topMsgId, scheduleDate: scheduleTime, sendAs: sendAsInputPeer, quickReplyShortcut: nil), tag: dependencyTag) |> map(NetworkRequestResult.result) } else { sendMessageRequest = .fail(MTRpcError(errorCode: 400, errorDescription: "internal")) @@ -468,7 +468,7 @@ private func sendUploadedMessageContent( } } - sendMessageRequest = network.request(Api.functions.messages.sendInlineBotResult(flags: flags, peer: inputPeer, replyTo: replyTo, randomId: uniqueId, queryId: chatContextResult.queryId, id: chatContextResult.id, scheduleDate: scheduleTime, sendAs: sendAsInputPeer)) + sendMessageRequest = network.request(Api.functions.messages.sendInlineBotResult(flags: flags, peer: inputPeer, replyTo: replyTo, randomId: uniqueId, queryId: chatContextResult.queryId, id: chatContextResult.id, scheduleDate: scheduleTime, sendAs: sendAsInputPeer, quickReplyShortcut: nil)) |> map(NetworkRequestResult.result) case .messageScreenshot: let replyTo: Api.InputReplyTo @@ -633,7 +633,7 @@ private func sendMessageContent(account: Account, peerId: PeerId, attributes: [M } } - sendMessageRequest = account.network.request(Api.functions.messages.sendMessage(flags: flags, peer: inputPeer, replyTo: replyTo, message: text, randomId: uniqueId, replyMarkup: nil, entities: messageEntities, scheduleDate: scheduleTime, sendAs: sendAsInputPeer)) + sendMessageRequest = account.network.request(Api.functions.messages.sendMessage(flags: flags, peer: inputPeer, replyTo: replyTo, message: text, randomId: uniqueId, replyMarkup: nil, entities: messageEntities, scheduleDate: scheduleTime, sendAs: sendAsInputPeer, quickReplyShortcut: nil)) |> `catch` { _ -> Signal in return .complete() } @@ -651,7 +651,7 @@ private func sendMessageContent(account: Account, peerId: PeerId, attributes: [M } } - sendMessageRequest = account.network.request(Api.functions.messages.sendMedia(flags: flags, peer: inputPeer, replyTo: replyTo, media: inputMedia, message: text, randomId: uniqueId, replyMarkup: nil, entities: messageEntities, scheduleDate: scheduleTime, sendAs: sendAsInputPeer)) + sendMessageRequest = account.network.request(Api.functions.messages.sendMedia(flags: flags, peer: inputPeer, replyTo: replyTo, media: inputMedia, message: text, randomId: uniqueId, replyMarkup: nil, entities: messageEntities, scheduleDate: scheduleTime, sendAs: sendAsInputPeer, quickReplyShortcut: nil)) |> `catch` { _ -> Signal in return .complete() } diff --git a/submodules/TelegramCore/Sources/State/AccountStateManagementUtils.swift b/submodules/TelegramCore/Sources/State/AccountStateManagementUtils.swift index a68c802ace2..7dfad231f59 100644 --- a/submodules/TelegramCore/Sources/State/AccountStateManagementUtils.swift +++ b/submodules/TelegramCore/Sources/State/AccountStateManagementUtils.swift @@ -1658,12 +1658,26 @@ private func finalStateWithUpdatesAndServerTime(accountPeerId: PeerId, postbox: if let message = StoreMessage(apiMessage: apiMessage, accountPeerId: accountPeerId, peerIsForum: peerIsForum, namespace: Namespaces.Message.ScheduledCloud) { updatedState.addScheduledMessages([message]) } + case let .updateQuickReplyMessage(apiMessage): + var peerIsForum = false + if let peerId = apiMessage.peerId { + peerIsForum = updatedState.isPeerForum(peerId: peerId) + } + if let message = StoreMessage(apiMessage: apiMessage, accountPeerId: accountPeerId, peerIsForum: peerIsForum, namespace: Namespaces.Message.QuickReplyCloud) { + updatedState.addQuickReplyMessages([message]) + } case let .updateDeleteScheduledMessages(peer, messages): var messageIds: [MessageId] = [] for message in messages { messageIds.append(MessageId(peerId: peer.peerId, namespace: Namespaces.Message.ScheduledCloud, id: message)) } updatedState.deleteMessages(messageIds) + case let .updateDeleteQuickReplyMessages(_, messages): + var messageIds: [MessageId] = [] + for message in messages { + messageIds.append(MessageId(peerId: accountPeerId, namespace: Namespaces.Message.QuickReplyCloud, id: message)) + } + updatedState.deleteMessages(messageIds) case let .updateTheme(theme): updatedState.updateTheme(TelegramTheme(apiTheme: theme)) case let .updateMessageID(id, randomId): @@ -3250,6 +3264,7 @@ private func optimizedOperations(_ operations: [AccountStateMutationOperation]) var currentAddMessages: OptimizeAddMessagesState? var currentAddScheduledMessages: OptimizeAddMessagesState? + var currentAddQuickReplyMessages: OptimizeAddMessagesState? for operation in operations { switch operation { case .DeleteMessages, .DeleteMessagesWithGlobalIds, .EditMessage, .UpdateMessagePoll, .UpdateMessageReactions, .UpdateMedia, .MergeApiChats, .MergeApiUsers, .MergePeerPresences, .UpdatePeer, .ReadInbox, .ReadOutbox, .ReadGroupFeedInbox, .ResetReadState, .ResetIncomingReadState, .UpdatePeerChatUnreadMark, .ResetMessageTagSummary, .UpdateNotificationSettings, .UpdateGlobalNotificationSettings, .UpdateSecretChat, .AddSecretMessages, .ReadSecretOutbox, .AddPeerInputActivity, .UpdateCachedPeerData, .UpdatePinnedItemIds, .UpdatePinnedSavedItemIds, .UpdatePinnedTopic, .UpdatePinnedTopicOrder, .ReadMessageContents, .UpdateMessageImpressionCount, .UpdateMessageForwardsCount, .UpdateInstalledStickerPacks, .UpdateRecentGifs, .UpdateChatInputState, .UpdateCall, .AddCallSignalingData, .UpdateLangPack, .UpdateMinAvailableMessage, .UpdateIsContact, .UpdatePeerChatInclusion, .UpdatePeersNearby, .UpdateTheme, .SyncChatListFilters, .UpdateChatListFilter, .UpdateChatListFilterOrder, .UpdateReadThread, .UpdateMessagesPinned, .UpdateGroupCallParticipants, .UpdateGroupCall, .UpdateAutoremoveTimeout, .UpdateAttachMenuBots, .UpdateAudioTranscription, .UpdateConfig, .UpdateExtendedMedia, .ResetForumTopic, .UpdateStory, .UpdateReadStories, .UpdateStoryStealthMode, .UpdateStorySentReaction, .UpdateNewAuthorization, .UpdateWallpaper: @@ -3259,6 +3274,9 @@ private func optimizedOperations(_ operations: [AccountStateMutationOperation]) if let currentAddScheduledMessages = currentAddScheduledMessages, !currentAddScheduledMessages.messages.isEmpty { result.append(.AddScheduledMessages(currentAddScheduledMessages.messages)) } + if let currentAddQuickReplyMessages = currentAddQuickReplyMessages, !currentAddQuickReplyMessages.messages.isEmpty { + result.append(.AddQuickReplyMessages(currentAddQuickReplyMessages.messages)) + } currentAddMessages = nil result.append(operation) case let .UpdateState(state): @@ -3284,6 +3302,12 @@ private func optimizedOperations(_ operations: [AccountStateMutationOperation]) } else { currentAddScheduledMessages = OptimizeAddMessagesState(messages: messages, location: .Random) } + case let .AddQuickReplyMessages(messages): + if let currentAddQuickReplyMessages = currentAddQuickReplyMessages { + currentAddQuickReplyMessages.messages.append(contentsOf: messages) + } else { + currentAddQuickReplyMessages = OptimizeAddMessagesState(messages: messages, location: .Random) + } } } if let currentAddMessages = currentAddMessages, !currentAddMessages.messages.isEmpty { @@ -3294,6 +3318,10 @@ private func optimizedOperations(_ operations: [AccountStateMutationOperation]) result.append(.AddScheduledMessages(currentAddScheduledMessages.messages)) } + if let currentAddQuickReplyMessages = currentAddQuickReplyMessages, !currentAddQuickReplyMessages.messages.isEmpty { + result.append(.AddQuickReplyMessages(currentAddQuickReplyMessages.messages)) + } + if let updatedState = updatedState { result.append(.UpdateState(updatedState)) } @@ -3745,6 +3773,16 @@ func replayFinalState( let _ = transaction.addMessages(messages, location: .Random) } } + case let .AddQuickReplyMessages(messages): + for message in messages { + if case let .Id(id) = message.id, let _ = transaction.getMessage(id) { + transaction.updateMessage(id) { _ -> PostboxUpdateMessage in + return .update(message) + } + } else { + let _ = transaction.addMessages(messages, location: .Random) + } + } case let .DeleteMessagesWithGlobalIds(ids): var resourceIds: [MediaResourceId] = [] transaction.deleteMessagesWithGlobalIds(ids, forEachMedia: { media in diff --git a/submodules/TelegramCore/Sources/State/AccountViewTracker.swift b/submodules/TelegramCore/Sources/State/AccountViewTracker.swift index d6aff342b5e..b82d15f6666 100644 --- a/submodules/TelegramCore/Sources/State/AccountViewTracker.swift +++ b/submodules/TelegramCore/Sources/State/AccountViewTracker.swift @@ -61,16 +61,30 @@ private func pollMessages(entries: [MessageHistoryEntry]) -> (Set, [M return (messageIds, messages) } -private func fetchWebpage(account: Account, messageId: MessageId) -> Signal { +private func fetchWebpage(account: Account, messageId: MessageId, threadId: Int64?) -> Signal { let accountPeerId = account.peerId return account.postbox.loadedPeerWithId(messageId.peerId) |> take(1) |> mapToSignal { peer in if let inputPeer = apiInputPeer(peer) { - let isScheduledMessage = Namespaces.Message.allScheduled.contains(messageId.namespace) + let targetMessageNamespace: MessageId.Namespace + if Namespaces.Message.allScheduled.contains(messageId.namespace) { + targetMessageNamespace = Namespaces.Message.ScheduledCloud + } else if Namespaces.Message.allQuickReply.contains(messageId.namespace) { + targetMessageNamespace = Namespaces.Message.QuickReplyCloud + } else { + targetMessageNamespace = Namespaces.Message.Cloud + } + let messages: Signal - if isScheduledMessage { + if Namespaces.Message.allScheduled.contains(messageId.namespace) { messages = account.network.request(Api.functions.messages.getScheduledMessages(peer: inputPeer, id: [messageId.id])) + } else if Namespaces.Message.allQuickReply.contains(messageId.namespace) { + if let threadId { + messages = account.network.request(Api.functions.messages.getQuickReplyMessages(flags: 1 << 0, shortcutId: Int32(clamping: threadId), id: [messageId.id], hash: 0)) + } else { + messages = .never() + } } else { switch inputPeer { case let .inputPeerChannel(channelId, accessHash): @@ -109,7 +123,7 @@ private func fetchWebpage(account: Account, messageId: MessageId) -> Signal() - private var updatedUnsupportedMediaMessageIdsAndTimestamps: [MessageId: Int32] = [:] + private var updatedUnsupportedMediaMessageIdsAndTimestamps: [MessageAndThreadId: Int32] = [:] private var refreshSecretChatMediaMessageIdsAndTimestamps: [MessageId: Int32] = [:] private var refreshStoriesForMessageIdsAndTimestamps: [MessageId: Int32] = [:] private var nextUpdatedUnsupportedMediaDisposableId: Int32 = 0 @@ -334,6 +348,9 @@ public final class AccountViewTracker { var resetPeerHoleManagement: ((PeerId) -> Void)? + private var quickRepliesUpdateDisposable: Disposable? + private var quickRepliesUpdateTimestamp: Double = 0.0 + init(account: Account) { self.account = account self.accountPeerId = account.peerId @@ -359,6 +376,7 @@ public final class AccountViewTracker { self.updatedViewCountDisposables.dispose() self.updatedReactionsDisposables.dispose() self.externallyUpdatedPeerIdDisposable.dispose() + self.quickRepliesUpdateDisposable?.dispose() } func reset() { @@ -367,7 +385,7 @@ public final class AccountViewTracker { } } - private func updatePendingWebpages(viewId: Int32, messageIds: Set, localWebpages: [MessageId: (MediaId, String)]) { + private func updatePendingWebpages(viewId: Int32, threadId: Int64?, messageIds: Set, localWebpages: [MessageId: (MediaId, String)]) { self.queue.async { var addedMessageIds: [MessageId] = [] var removedMessageIds: [MessageId] = [] @@ -440,7 +458,7 @@ public final class AccountViewTracker { } }) } else if messageId.namespace == Namespaces.Message.Cloud { - self.webpageDisposables[messageId] = fetchWebpage(account: account, messageId: messageId).start(completed: { [weak self] in + self.webpageDisposables[messageId] = fetchWebpage(account: account, messageId: messageId, threadId: threadId).start(completed: { [weak self] in if let strongSelf = self { strongSelf.queue.async { strongSelf.webpageDisposables.removeValue(forKey: messageId) @@ -1009,9 +1027,9 @@ public final class AccountViewTracker { } } - public func updateUnsupportedMediaForMessageIds(messageIds: Set) { + public func updateUnsupportedMediaForMessageIds(messageIds: Set) { self.queue.async { - var addedMessageIds: [MessageId] = [] + var addedMessageIds: [MessageAndThreadId] = [] let timestamp = Int32(CFAbsoluteTimeGetCurrent()) for messageId in messageIds { let messageTimestamp = self.updatedUnsupportedMediaMessageIdsAndTimestamps[messageId] @@ -1021,14 +1039,14 @@ public final class AccountViewTracker { } } if !addedMessageIds.isEmpty { - for (peerId, messageIds) in messagesIdsGroupedByPeerId(Set(addedMessageIds)) { + for (peerIdAndThreadId, messageIds) in messagesIdsGroupedByPeerId(Set(addedMessageIds)) { let disposableId = self.nextUpdatedUnsupportedMediaDisposableId self.nextUpdatedUnsupportedMediaDisposableId += 1 if let account = self.account { let accountPeerId = account.peerId let signal = account.postbox.transaction { transaction -> Peer? in - if let peer = transaction.getPeer(peerId) { + if let peer = transaction.getPeer(peerIdAndThreadId.peerId) { return peer } else { return nil @@ -1043,9 +1061,15 @@ public final class AccountViewTracker { if let inputPeer = apiInputPeer(peer) { fetchSignal = account.network.request(Api.functions.messages.getScheduledMessages(peer: inputPeer, id: messageIds.map { $0.id })) } - } else if peerId.namespace == Namespaces.Peer.CloudUser || peerId.namespace == Namespaces.Peer.CloudGroup { + } else if let messageId = messageIds.first, messageId.namespace == Namespaces.Message.QuickReplyCloud { + if let threadId = peerIdAndThreadId.threadId { + fetchSignal = account.network.request(Api.functions.messages.getQuickReplyMessages(flags: 1 << 0, shortcutId: Int32(clamping: threadId), id: messageIds.map { $0.id }, hash: 0)) + } else { + fetchSignal = .never() + } + } else if peerIdAndThreadId.peerId.namespace == Namespaces.Peer.CloudUser || peerIdAndThreadId.peerId.namespace == Namespaces.Peer.CloudGroup { fetchSignal = account.network.request(Api.functions.messages.getMessages(id: messageIds.map { Api.InputMessage.inputMessageID(id: $0.id) })) - } else if peerId.namespace == Namespaces.Peer.CloudChannel { + } else if peerIdAndThreadId.peerId.namespace == Namespaces.Peer.CloudChannel { if let inputChannel = apiInputChannel(peer) { fetchSignal = account.network.request(Api.functions.channels.getMessages(channel: inputChannel, id: messageIds.map { Api.InputMessage.inputMessageID(id: $0.id) })) } @@ -1444,7 +1468,7 @@ public final class AccountViewTracker { if i < slice.count { let value = result[i] transaction.updatePeerCachedData(peerIds: Set([slice[i].0]), update: { _, cachedData in - var cachedData = cachedData as? CachedUserData ?? CachedUserData(about: nil, botInfo: nil, editableBotInfo: nil, peerStatusSettings: nil, pinnedMessageId: nil, isBlocked: false, commonGroupCount: 0, voiceCallsAvailable: true, videoCallsAvailable: true, callsPrivate: true, canPinMessages: true, hasScheduledMessages: true, autoremoveTimeout: .unknown, themeEmoticon: nil, photo: .unknown, personalPhoto: .unknown, fallbackPhoto: .unknown, premiumGiftOptions: [], voiceMessagesAvailable: true, wallpaper: nil, flags: []) + var cachedData = cachedData as? CachedUserData ?? CachedUserData(about: nil, botInfo: nil, editableBotInfo: nil, peerStatusSettings: nil, pinnedMessageId: nil, isBlocked: false, commonGroupCount: 0, voiceCallsAvailable: true, videoCallsAvailable: true, callsPrivate: true, canPinMessages: true, hasScheduledMessages: true, autoremoveTimeout: .unknown, themeEmoticon: nil, photo: .unknown, personalPhoto: .unknown, fallbackPhoto: .unknown, premiumGiftOptions: [], voiceMessagesAvailable: true, wallpaper: nil, flags: [], businessHours: nil, businessLocation: nil, greetingMessage: nil, awayMessage: nil, connectedBot: nil) var flags = cachedData.flags if case .boolTrue = value { flags.insert(.premiumRequired) @@ -1815,7 +1839,7 @@ public final class AccountViewTracker { if let strongSelf = self { strongSelf.queue.async { let (messageIds, localWebpages) = pendingWebpages(entries: next.0.entries) - strongSelf.updatePendingWebpages(viewId: viewId, messageIds: messageIds, localWebpages: localWebpages) + strongSelf.updatePendingWebpages(viewId: viewId, threadId: chatLocation.threadId, messageIds: messageIds, localWebpages: localWebpages) let (pollMessageIds, pollMessageDict) = pollMessages(entries: next.0.entries) strongSelf.updatePolls(viewId: viewId, messageIds: pollMessageIds, messages: pollMessageDict) if case let .peer(peerId, _) = chatLocation, peerId.namespace == Namespaces.Peer.CloudChannel { @@ -1828,7 +1852,7 @@ public final class AccountViewTracker { }, disposed: { [weak self] viewId in if let strongSelf = self { strongSelf.queue.async { - strongSelf.updatePendingWebpages(viewId: viewId, messageIds: [], localWebpages: [:]) + strongSelf.updatePendingWebpages(viewId: viewId, threadId: chatLocation.threadId, messageIds: [], localWebpages: [:]) strongSelf.updatePolls(viewId: viewId, messageIds: [], messages: [:]) switch chatLocation { case let .peer(peerId, _): @@ -1839,7 +1863,7 @@ public final class AccountViewTracker { if peerId.namespace == Namespaces.Peer.CloudChannel { strongSelf.historyViewStateValidationContexts.updateView(id: viewId, view: nil, location: chatLocation) } - case .feed: + case .customChatContents: break } } @@ -1852,7 +1876,7 @@ public final class AccountViewTracker { peerId = peerIdValue case let .thread(peerIdValue, _, _): peerId = peerIdValue - case .feed: + case .customChatContents: peerId = nil } if let peerId = peerId, peerId.namespace == Namespaces.Peer.CloudChannel { @@ -1945,14 +1969,14 @@ public final class AccountViewTracker { if let strongSelf = self { strongSelf.queue.async { let (messageIds, localWebpages) = pendingWebpages(entries: next.0.entries) - strongSelf.updatePendingWebpages(viewId: viewId, messageIds: messageIds, localWebpages: localWebpages) + strongSelf.updatePendingWebpages(viewId: viewId, threadId: chatLocation.threadId, messageIds: messageIds, localWebpages: localWebpages) strongSelf.historyViewStateValidationContexts.updateView(id: viewId, view: next.0, location: chatLocation) } } }, disposed: { [weak self] viewId in if let strongSelf = self { strongSelf.queue.async { - strongSelf.updatePendingWebpages(viewId: viewId, messageIds: [], localWebpages: [:]) + strongSelf.updatePendingWebpages(viewId: viewId, threadId: chatLocation.threadId, messageIds: [], localWebpages: [:]) strongSelf.historyViewStateValidationContexts.updateView(id: viewId, view: nil, location: nil) } } @@ -1962,6 +1986,65 @@ public final class AccountViewTracker { } } + public func quickReplyMessagesViewForLocation(quickReplyId: Int32, additionalData: [AdditionalMessageHistoryViewData] = []) -> Signal<(MessageHistoryView, ViewUpdateType, InitialMessageHistoryData?), NoError> { + guard let account = self.account else { + return .never() + } + let chatLocation: ChatLocationInput = .peer(peerId: account.peerId, threadId: Int64(quickReplyId)) + let signal = account.postbox.aroundMessageHistoryViewForLocation(chatLocation, anchor: .upperBound, ignoreMessagesInTimestampRange: nil, count: 200, fixedCombinedReadStates: nil, topTaggedMessageIdNamespaces: [], tag: nil, appendMessagesFromTheSameGroup: false, namespaces: .just(Namespaces.Message.allQuickReply), orderStatistics: [], additionalData: additionalData) + return withState(signal, { [weak self] () -> Int32 in + if let strongSelf = self { + return OSAtomicIncrement32(&strongSelf.nextViewId) + } else { + return -1 + } + }, next: { [weak self] next, viewId in + if let strongSelf = self { + strongSelf.queue.async { + let (messageIds, localWebpages) = pendingWebpages(entries: next.0.entries) + strongSelf.updatePendingWebpages(viewId: viewId, threadId: Int64(quickReplyId), messageIds: messageIds, localWebpages: localWebpages) + strongSelf.historyViewStateValidationContexts.updateView(id: viewId, view: next.0, location: chatLocation) + } + } + }, disposed: { [weak self] viewId in + if let strongSelf = self { + strongSelf.queue.async { + strongSelf.updatePendingWebpages(viewId: viewId, threadId: Int64(quickReplyId), messageIds: [], localWebpages: [:]) + strongSelf.historyViewStateValidationContexts.updateView(id: viewId, view: nil, location: nil) + } + } + }) + } + + public func pendingQuickReplyMessagesViewForLocation(shortcut: String) -> Signal<(MessageHistoryView, ViewUpdateType, InitialMessageHistoryData?), NoError> { + guard let account = self.account else { + return .never() + } + let chatLocation: ChatLocationInput = .peer(peerId: account.peerId, threadId: nil) + let signal = account.postbox.aroundMessageHistoryViewForLocation(chatLocation, anchor: .upperBound, ignoreMessagesInTimestampRange: nil, count: 200, fixedCombinedReadStates: nil, topTaggedMessageIdNamespaces: [], tag: nil, appendMessagesFromTheSameGroup: false, namespaces: .just([Namespaces.Message.QuickReplyLocal]), orderStatistics: [], additionalData: []) + |> map { view, update, initialData in + var entries: [MessageHistoryEntry] = [] + for entry in view.entries { + var matches = false + inner: for attribute in entry.message.attributes { + if let attribute = attribute as? OutgoingQuickReplyMessageAttribute { + if attribute.shortcut == shortcut { + matches = true + } + break inner + } + } + if matches { + entries.append(entry) + } + } + let mappedView = MessageHistoryView(tag: nil, namespaces: .just([Namespaces.Message.QuickReplyLocal]), entries: entries, holeEarlier: false, holeLater: false, isLoading: false) + + return (mappedView, update, initialData) + } + return signal + } + public func aroundMessageOfInterestHistoryViewForLocation(_ chatLocation: ChatLocationInput, ignoreMessagesInTimestampRange: ClosedRange? = nil, count: Int, tag: HistoryViewInputTag? = nil, appendMessagesFromTheSameGroup: Bool = false, orderStatistics: MessageHistoryViewOrderStatistics = [], additionalData: [AdditionalMessageHistoryViewData] = [], useRootInterfaceStateForThread: Bool = false) -> Signal<(MessageHistoryView, ViewUpdateType, InitialMessageHistoryData?), NoError> { if let account = self.account { let signal: Signal<(MessageHistoryView, ViewUpdateType, InitialMessageHistoryData?), NoError> @@ -1994,7 +2077,7 @@ public final class AccountViewTracker { topTaggedMessageIdNamespaces: [], tag: tag, appendMessagesFromTheSameGroup: false, - namespaces: .not(Namespaces.Message.allScheduled), + namespaces: .not(Namespaces.Message.allNonRegular), orderStatistics: orderStatistics, additionalData: wrappedHistoryViewAdditionalData(chatLocation: chatLocation, additionalData: additionalData), useRootInterfaceStateForThread: useRootInterfaceStateForThread @@ -2020,7 +2103,7 @@ public final class AccountViewTracker { topTaggedMessageIdNamespaces: [], tag: tag, appendMessagesFromTheSameGroup: false, - namespaces: .not(Namespaces.Message.allScheduled), + namespaces: .not(Namespaces.Message.allNonRegular), orderStatistics: orderStatistics, additionalData: wrappedHistoryViewAdditionalData(chatLocation: chatLocation, additionalData: additionalData), useRootInterfaceStateForThread: useRootInterfaceStateForThread @@ -2028,10 +2111,10 @@ public final class AccountViewTracker { } } - return account.postbox.aroundMessageOfInterestHistoryViewForChatLocation(chatLocation, ignoreMessagesInTimestampRange: ignoreMessagesInTimestampRange, count: count, topTaggedMessageIdNamespaces: [Namespaces.Message.Cloud], tag: tag, appendMessagesFromTheSameGroup: appendMessagesFromTheSameGroup, namespaces: .not(Namespaces.Message.allScheduled), orderStatistics: orderStatistics, customUnreadMessageId: nil, additionalData: wrappedHistoryViewAdditionalData(chatLocation: chatLocation, additionalData: additionalData), useRootInterfaceStateForThread: useRootInterfaceStateForThread) + return account.postbox.aroundMessageOfInterestHistoryViewForChatLocation(chatLocation, ignoreMessagesInTimestampRange: ignoreMessagesInTimestampRange, count: count, topTaggedMessageIdNamespaces: [Namespaces.Message.Cloud], tag: tag, appendMessagesFromTheSameGroup: appendMessagesFromTheSameGroup, namespaces: .not(Namespaces.Message.allNonRegular), orderStatistics: orderStatistics, customUnreadMessageId: nil, additionalData: wrappedHistoryViewAdditionalData(chatLocation: chatLocation, additionalData: additionalData), useRootInterfaceStateForThread: useRootInterfaceStateForThread) } } else { - signal = account.postbox.aroundMessageOfInterestHistoryViewForChatLocation(chatLocation, ignoreMessagesInTimestampRange: ignoreMessagesInTimestampRange, count: count, topTaggedMessageIdNamespaces: [Namespaces.Message.Cloud], tag: tag, appendMessagesFromTheSameGroup: appendMessagesFromTheSameGroup, namespaces: .not(Namespaces.Message.allScheduled), orderStatistics: orderStatistics, customUnreadMessageId: nil, additionalData: wrappedHistoryViewAdditionalData(chatLocation: chatLocation, additionalData: additionalData), useRootInterfaceStateForThread: useRootInterfaceStateForThread) + signal = account.postbox.aroundMessageOfInterestHistoryViewForChatLocation(chatLocation, ignoreMessagesInTimestampRange: ignoreMessagesInTimestampRange, count: count, topTaggedMessageIdNamespaces: [Namespaces.Message.Cloud], tag: tag, appendMessagesFromTheSameGroup: appendMessagesFromTheSameGroup, namespaces: .not(Namespaces.Message.allNonRegular), orderStatistics: orderStatistics, customUnreadMessageId: nil, additionalData: wrappedHistoryViewAdditionalData(chatLocation: chatLocation, additionalData: additionalData), useRootInterfaceStateForThread: useRootInterfaceStateForThread) } return wrappedMessageHistorySignal(chatLocation: chatLocation, signal: signal, fixedCombinedReadStates: nil, addHoleIfNeeded: true) } else { @@ -2041,7 +2124,7 @@ public final class AccountViewTracker { public func aroundIdMessageHistoryViewForLocation(_ chatLocation: ChatLocationInput, ignoreMessagesInTimestampRange: ClosedRange? = nil, count: Int, ignoreRelatedChats: Bool, messageId: MessageId, tag: HistoryViewInputTag? = nil, appendMessagesFromTheSameGroup: Bool = false, orderStatistics: MessageHistoryViewOrderStatistics = [], additionalData: [AdditionalMessageHistoryViewData] = [], useRootInterfaceStateForThread: Bool = false) -> Signal<(MessageHistoryView, ViewUpdateType, InitialMessageHistoryData?), NoError> { if let account = self.account { - let signal = account.postbox.aroundIdMessageHistoryViewForLocation(chatLocation, ignoreMessagesInTimestampRange: ignoreMessagesInTimestampRange, count: count, ignoreRelatedChats: ignoreRelatedChats, messageId: messageId, topTaggedMessageIdNamespaces: [Namespaces.Message.Cloud], tag: tag, appendMessagesFromTheSameGroup: appendMessagesFromTheSameGroup, namespaces: .not(Namespaces.Message.allScheduled), orderStatistics: orderStatistics, additionalData: wrappedHistoryViewAdditionalData(chatLocation: chatLocation, additionalData: additionalData), useRootInterfaceStateForThread: useRootInterfaceStateForThread) + let signal = account.postbox.aroundIdMessageHistoryViewForLocation(chatLocation, ignoreMessagesInTimestampRange: ignoreMessagesInTimestampRange, count: count, ignoreRelatedChats: ignoreRelatedChats, messageId: messageId, topTaggedMessageIdNamespaces: [Namespaces.Message.Cloud], tag: tag, appendMessagesFromTheSameGroup: appendMessagesFromTheSameGroup, namespaces: .not(Namespaces.Message.allNonRegular), orderStatistics: orderStatistics, additionalData: wrappedHistoryViewAdditionalData(chatLocation: chatLocation, additionalData: additionalData), useRootInterfaceStateForThread: useRootInterfaceStateForThread) return wrappedMessageHistorySignal(chatLocation: chatLocation, signal: signal, fixedCombinedReadStates: nil, addHoleIfNeeded: false) } else { return .never() @@ -2059,7 +2142,7 @@ public final class AccountViewTracker { case let .message(index): inputAnchor = .index(index) } - let signal = account.postbox.aroundMessageHistoryViewForLocation(chatLocation, anchor: inputAnchor, ignoreMessagesInTimestampRange: ignoreMessagesInTimestampRange, count: count, clipHoles: clipHoles, ignoreRelatedChats: ignoreRelatedChats, fixedCombinedReadStates: fixedCombinedReadStates, topTaggedMessageIdNamespaces: [Namespaces.Message.Cloud], tag: tag, appendMessagesFromTheSameGroup: appendMessagesFromTheSameGroup, namespaces: .not(Namespaces.Message.allScheduled), orderStatistics: orderStatistics, additionalData: wrappedHistoryViewAdditionalData(chatLocation: chatLocation, additionalData: additionalData), useRootInterfaceStateForThread: useRootInterfaceStateForThread) + let signal = account.postbox.aroundMessageHistoryViewForLocation(chatLocation, anchor: inputAnchor, ignoreMessagesInTimestampRange: ignoreMessagesInTimestampRange, count: count, clipHoles: clipHoles, ignoreRelatedChats: ignoreRelatedChats, fixedCombinedReadStates: fixedCombinedReadStates, topTaggedMessageIdNamespaces: [Namespaces.Message.Cloud], tag: tag, appendMessagesFromTheSameGroup: appendMessagesFromTheSameGroup, namespaces: .not(Namespaces.Message.allNonRegular), orderStatistics: orderStatistics, additionalData: wrappedHistoryViewAdditionalData(chatLocation: chatLocation, additionalData: additionalData), useRootInterfaceStateForThread: useRootInterfaceStateForThread) return wrappedMessageHistorySignal(chatLocation: chatLocation, signal: signal, fixedCombinedReadStates: fixedCombinedReadStates, addHoleIfNeeded: false) } else { return .never() @@ -2498,6 +2581,20 @@ public final class AccountViewTracker { } } } + + public func keepQuickRepliesApproximatelyUpdated() { + self.queue.async { + guard let account = self.account else { + return + } + let timestamp = CFAbsoluteTimeGetCurrent() + if self.quickRepliesUpdateTimestamp + 16 * 60 * 60 < timestamp { + self.quickRepliesUpdateTimestamp = timestamp + self.quickRepliesUpdateDisposable?.dispose() + self.quickRepliesUpdateDisposable = _internal_keepShortcutMessagesUpdated(account: account).startStrict() + } + } + } } public final class ExtractedChatListItemCachedData: Hashable { diff --git a/submodules/TelegramCore/Sources/State/ApplyUpdateMessage.swift b/submodules/TelegramCore/Sources/State/ApplyUpdateMessage.swift index 6bb8c766577..0a2a0b1fb43 100644 --- a/submodules/TelegramCore/Sources/State/ApplyUpdateMessage.swift +++ b/submodules/TelegramCore/Sources/State/ApplyUpdateMessage.swift @@ -96,7 +96,7 @@ func applyUpdateMessage(postbox: Postbox, stateManager: AccountStateManager, mes var updatedTimestamp: Int32? if let apiMessage = apiMessage { switch apiMessage { - case let .message(_, _, _, _, _, _, _, _, _, date, _, _, _, _, _, _, _, _, _, _, _, _, _): + case let .message(_, _, _, _, _, _, _, _, _, date, _, _, _, _, _, _, _, _, _, _, _, _, _, _): updatedTimestamp = date case .messageEmpty: break @@ -129,7 +129,9 @@ func applyUpdateMessage(postbox: Postbox, stateManager: AccountStateManager, mes let updatedId: MessageId if let messageId = messageId { var namespace: MessageId.Namespace = Namespaces.Message.Cloud - if let updatedTimestamp = updatedTimestamp { + if Namespaces.Message.allQuickReply.contains(message.id.namespace) { + namespace = Namespaces.Message.QuickReplyCloud + } else if let updatedTimestamp = updatedTimestamp { if message.scheduleTime != nil && message.scheduleTime == updatedTimestamp { namespace = Namespaces.Message.ScheduledCloud } @@ -141,21 +143,17 @@ func applyUpdateMessage(postbox: Postbox, stateManager: AccountStateManager, mes updatedId = currentMessage.id } - for attribute in currentMessage.attributes { - if let attribute = attribute as? OutgoingMessageInfoAttribute { - bubbleUpEmojiOrStickersets = attribute.bubbleUpEmojiOrStickersets - } - } - let media: [Media] var attributes: [MessageAttribute] let text: String let forwardInfo: StoreMessageForwardInfo? + let threadId: Int64? if let apiMessage = apiMessage, let apiMessagePeerId = apiMessage.peerId, let updatedMessage = StoreMessage(apiMessage: apiMessage, accountPeerId: accountPeerId, peerIsForum: transaction.getPeer(apiMessagePeerId)?.isForum ?? false) { media = updatedMessage.media attributes = updatedMessage.attributes text = updatedMessage.text forwardInfo = updatedMessage.forwardInfo + threadId = updatedMessage.threadId } else if case let .updateShortSentMessage(_, _, _, _, _, apiMedia, entities, ttlPeriod) = result { let (mediaValue, _, nonPremium, hasSpoiler, _) = textMediaAndExpirationTimerFromApiMedia(apiMedia, currentMessage.id.peerId) if let mediaValue = mediaValue { @@ -197,16 +195,36 @@ func applyUpdateMessage(postbox: Postbox, stateManager: AccountStateManager, mes } } } + if Namespaces.Message.allQuickReply.contains(message.id.namespace) { + for i in 0 ..< updatedAttributes.count { + if updatedAttributes[i] is OutgoingQuickReplyMessageAttribute { + updatedAttributes.remove(at: i) + break + } + } + } attributes = updatedAttributes text = currentMessage.text forwardInfo = currentMessage.forwardInfo.flatMap(StoreMessageForwardInfo.init) + threadId = currentMessage.threadId } else { media = currentMessage.media attributes = currentMessage.attributes text = currentMessage.text forwardInfo = currentMessage.forwardInfo.flatMap(StoreMessageForwardInfo.init) + threadId = currentMessage.threadId + } + + for attribute in currentMessage.attributes { + if let attribute = attribute as? OutgoingMessageInfoAttribute { + bubbleUpEmojiOrStickersets = attribute.bubbleUpEmojiOrStickersets + } else if let attribute = attribute as? OutgoingQuickReplyMessageAttribute { + if let threadId { + _internal_applySentQuickReplyMessage(transaction: transaction, shortcut: attribute.shortcut, quickReplyId: Int32(clamping: threadId)) + } + } } if let channelPts = channelPts { @@ -255,7 +273,7 @@ func applyUpdateMessage(postbox: Postbox, stateManager: AccountStateManager, mes let (tags, globalTags) = tagsForStoreMessage(incoming: currentMessage.flags.contains(.Incoming), attributes: attributes, media: media, textEntities: entitiesAttribute?.entities, isPinned: currentMessage.tags.contains(.pinned)) - if currentMessage.id.peerId.namespace == Namespaces.Peer.CloudChannel, !currentMessage.flags.contains(.Incoming), !Namespaces.Message.allScheduled.contains(currentMessage.id.namespace) { + if currentMessage.id.peerId.namespace == Namespaces.Peer.CloudChannel, !currentMessage.flags.contains(.Incoming), !Namespaces.Message.allNonRegular.contains(currentMessage.id.namespace) { let peerId = currentMessage.id.peerId if let peer = transaction.getPeer(peerId) { if let peer = peer as? TelegramChannel { @@ -344,7 +362,9 @@ func applyUpdateGroupMessages(postbox: Postbox, stateManager: AccountStateManage let updatedRawMessageIds = result.updatedRawMessageIds var namespace = Namespaces.Message.Cloud - if let message = messages.first, let apiMessage = result.messages.first, message.scheduleTime != nil && message.scheduleTime == apiMessage.timestamp { + if Namespaces.Message.allQuickReply.contains(messages[0].id.namespace) { + namespace = Namespaces.Message.QuickReplyCloud + } else if let message = messages.first, let apiMessage = result.messages.first, message.scheduleTime != nil && message.scheduleTime == apiMessage.timestamp { namespace = Namespaces.Message.ScheduledCloud } @@ -411,6 +431,16 @@ func applyUpdateGroupMessages(postbox: Postbox, stateManager: AccountStateManage var bubbleUpEmojiOrStickersets: [ItemCollectionId] = [] + if let (message, _, updatedMessage) = mapping.first { + for attribute in message.attributes { + if let attribute = attribute as? OutgoingQuickReplyMessageAttribute { + if let threadId = updatedMessage.threadId { + _internal_applySentQuickReplyMessage(transaction: transaction, shortcut: attribute.shortcut, quickReplyId: Int32(clamping: threadId)) + } + } + } + } + for (message, _, updatedMessage) in mapping { transaction.updateMessage(message.id, update: { currentMessage in let updatedId: MessageId diff --git a/submodules/TelegramCore/Sources/State/CloudChatRemoveMessagesOperation.swift b/submodules/TelegramCore/Sources/State/CloudChatRemoveMessagesOperation.swift index 58b9bd40edb..acf9a302614 100644 --- a/submodules/TelegramCore/Sources/State/CloudChatRemoveMessagesOperation.swift +++ b/submodules/TelegramCore/Sources/State/CloudChatRemoveMessagesOperation.swift @@ -1,7 +1,7 @@ import Postbox -func cloudChatAddRemoveMessagesOperation(transaction: Transaction, peerId: PeerId, messageIds: [MessageId], type: CloudChatRemoveMessagesType) { - transaction.operationLogAddEntry(peerId: peerId, tag: OperationLogTags.CloudChatRemoveMessages, tagLocalIndex: .automatic, tagMergedIndex: .automatic, contents: CloudChatRemoveMessagesOperation(messageIds: messageIds, type: type)) +func cloudChatAddRemoveMessagesOperation(transaction: Transaction, peerId: PeerId, threadId: Int64?, messageIds: [MessageId], type: CloudChatRemoveMessagesType) { + transaction.operationLogAddEntry(peerId: peerId, tag: OperationLogTags.CloudChatRemoveMessages, tagLocalIndex: .automatic, tagMergedIndex: .automatic, contents: CloudChatRemoveMessagesOperation(messageIds: messageIds, threadId: threadId, type: type)) } func cloudChatAddRemoveChatOperation(transaction: Transaction, peerId: PeerId, reportChatSpam: Bool, deleteGloballyIfPossible: Bool) { @@ -15,7 +15,26 @@ func cloudChatAddClearHistoryOperation(transaction: Transaction, peerId: PeerId, messageIds.append(message.id) return true } - cloudChatAddRemoveMessagesOperation(transaction: transaction, peerId: peerId, messageIds: messageIds, type: .forLocalPeer) + cloudChatAddRemoveMessagesOperation(transaction: transaction, peerId: peerId, threadId: threadId, messageIds: messageIds, type: .forLocalPeer) + } else if type == .quickReplyMessages { + var messageIds: [MessageId] = [] + transaction.withAllMessages(peerId: peerId, namespace: Namespaces.Message.QuickReplyCloud) { message -> Bool in + messageIds.append(message.id) + return true + } + cloudChatAddRemoveMessagesOperation(transaction: transaction, peerId: peerId, threadId: threadId, messageIds: messageIds, type: .forLocalPeer) + + let topMessageId: MessageId? + if let explicitTopMessageId = explicitTopMessageId { + topMessageId = explicitTopMessageId + } else { + topMessageId = transaction.getTopPeerMessageId(peerId: peerId, namespace: Namespaces.Message.QuickReplyCloud) + } + if let topMessageId = topMessageId { + transaction.operationLogAddEntry(peerId: peerId, tag: OperationLogTags.CloudChatRemoveMessages, tagLocalIndex: .automatic, tagMergedIndex: .automatic, contents: CloudChatClearHistoryOperation(peerId: peerId, topMessageId: topMessageId, threadId: threadId, minTimestamp: minTimestamp, maxTimestamp: maxTimestamp, type: type)) + } else if case .forEveryone = type { + transaction.operationLogAddEntry(peerId: peerId, tag: OperationLogTags.CloudChatRemoveMessages, tagLocalIndex: .automatic, tagMergedIndex: .automatic, contents: CloudChatClearHistoryOperation(peerId: peerId, topMessageId: MessageId(peerId: peerId, namespace: Namespaces.Message.Cloud, id: .max), threadId: threadId, minTimestamp: minTimestamp, maxTimestamp: maxTimestamp, type: type)) + } } else { let topMessageId: MessageId? if let explicitTopMessageId = explicitTopMessageId { diff --git a/submodules/TelegramCore/Sources/State/HistoryViewStateValidation.swift b/submodules/TelegramCore/Sources/State/HistoryViewStateValidation.swift index ddaea2d2c3e..4290ad04f4b 100644 --- a/submodules/TelegramCore/Sources/State/HistoryViewStateValidation.swift +++ b/submodules/TelegramCore/Sources/State/HistoryViewStateValidation.swift @@ -28,78 +28,60 @@ private final class HistoryStateValidationContext { private enum HistoryState { case channel(PeerId, ChannelState) - //case group(PeerGroupId, TelegramPeerGroupState) case scheduledMessages(PeerId) + case quickReplyMessages(PeerId, Int32) var hasInvalidationIndex: Bool { switch self { - case let .channel(_, state): - return state.invalidatedPts != nil - /*case let .group(_, state): - return state.invalidatedStateIndex != nil*/ - case .scheduledMessages: - return false + case let .channel(_, state): + return state.invalidatedPts != nil + case .scheduledMessages: + return false + case .quickReplyMessages: + return false } } func isMessageValid(_ message: Message) -> Bool { switch self { - case let .channel(_, state): - if let invalidatedPts = state.invalidatedPts { - var messagePts: Int32? - inner: for attribute in message.attributes { - if let attribute = attribute as? ChannelMessageStateVersionAttribute { - messagePts = attribute.pts - break inner - } - } - var requiresValidation = false - if let messagePts = messagePts { - if messagePts < invalidatedPts { - requiresValidation = true - } - } else { - requiresValidation = true + case let .channel(_, state): + if let invalidatedPts = state.invalidatedPts { + var messagePts: Int32? + inner: for attribute in message.attributes { + if let attribute = attribute as? ChannelMessageStateVersionAttribute { + messagePts = attribute.pts + break inner } - - return !requiresValidation - } else { - return true } - /*case let .group(_, state): - if let invalidatedStateIndex = state.invalidatedStateIndex { - var messageStateIndex: Int32? - inner: for attribute in message.attributes { - if let attribute = attribute as? PeerGroupMessageStateVersionAttribute { - messageStateIndex = attribute.stateIndex - break inner - } - } - var requiresValidation = false - if let messageStateIndex = messageStateIndex { - if messageStateIndex < invalidatedStateIndex { - requiresValidation = true - } - } else { + + var requiresValidation = false + if let messagePts = messagePts { + if messagePts < invalidatedPts { requiresValidation = true } - return !requiresValidation } else { - return true - }*/ - case .scheduledMessages: - return false + requiresValidation = true + } + + return !requiresValidation + } else { + return true + } + case .scheduledMessages: + return false + case .quickReplyMessages: + return false } } func matchesPeerId(_ peerId: PeerId) -> Bool { switch self { - case let .channel(statePeerId, _): - return statePeerId == peerId - /*case .group: - return true*/ - case let .scheduledMessages(statePeerId): - return statePeerId == peerId + case let .channel(statePeerId, _): + return statePeerId == peerId + case let .scheduledMessages(statePeerId): + return statePeerId == peerId + case let .quickReplyMessages(statePeerId, _): + return statePeerId == peerId } } } @@ -411,6 +393,35 @@ final class HistoryViewStateValidationContexts { })) } } + } else if view.namespaces.contains(Namespaces.Message.QuickReplyCloud) { + if let _ = self.contexts[id] { + } else if let location = location, case let .peer(peerId, threadId) = location { + guard let threadId else { + return + } + + let timestamp = self.network.context.globalTime() + if let previousTimestamp = self.previousPeerValidationTimestamps[peerId], timestamp < previousTimestamp + 60 { + } else { + self.previousPeerValidationTimestamps[peerId] = timestamp + + let context = HistoryStateValidationContext() + self.contexts[id] = context + + let disposable = MetaDisposable() + let batch = HistoryStateValidationBatch(disposable: disposable) + context.batch = batch + + let messages: [Message] = view.entries.map { $0.message }.filter { $0.id.namespace == Namespaces.Message.QuickReplyCloud } + + disposable.set((validateQuickReplyMessagesBatch(postbox: self.postbox, network: self.network, accountPeerId: peerId, tag: nil, messages: messages, historyState: .quickReplyMessages(peerId, Int32(clamping: threadId))) + |> deliverOn(self.queue)).start(completed: { [weak self] in + if let strongSelf = self, let context = strongSelf.contexts[id] { + context.batch = nil + } + })) + } + } } } } @@ -690,6 +701,53 @@ private func validateScheduledMessagesBatch(postbox: Postbox, network: Network, } |> switchToLatest } +private func validateQuickReplyMessagesBatch(postbox: Postbox, network: Network, accountPeerId: PeerId, tag: MessageTags?, messages: [Message], historyState: HistoryState) -> Signal { + return postbox.transaction { transaction -> Signal in + var signal: Signal + switch historyState { + case let .quickReplyMessages(peerId, shortcutId): + if let peer = transaction.getPeer(peerId) { + let hash = hashForScheduledMessages(messages) + signal = network.request(Api.functions.messages.getQuickReplyMessages(flags: 0, shortcutId: shortcutId, id: nil, hash: hash)) + |> map { result -> ValidatedMessages in + let messages: [Api.Message] + let chats: [Api.Chat] + let users: [Api.User] + + switch result { + case let .messages(messages: apiMessages, chats: apiChats, users: apiUsers): + messages = apiMessages + chats = apiChats + users = apiUsers + case let .messagesSlice(_, _, _, _, messages: apiMessages, chats: apiChats, users: apiUsers): + messages = apiMessages + chats = apiChats + users = apiUsers + case let .channelMessages(_, _, _, _, apiMessages, apiTopics, apiChats, apiUsers): + messages = apiMessages + let _ = apiTopics + chats = apiChats + users = apiUsers + case .messagesNotModified: + return .notModified + } + return .messages(peer, messages, chats, users, nil) + } + } else { + signal = .complete() + } + default: + signal = .complete() + } + var previous: [MessageId: Message] = [:] + for message in messages { + previous[message.id] = message + } + return validateBatch(postbox: postbox, network: network, transaction: transaction, accountPeerId: accountPeerId, tag: tag, historyState: historyState, signal: signal, previous: previous, messageNamespace: Namespaces.Message.QuickReplyCloud) + } + |> switchToLatest +} + private func validateBatch(postbox: Postbox, network: Network, transaction: Transaction, accountPeerId: PeerId, tag: MessageTags?, historyState: HistoryState, signal: Signal, previous: [MessageId: Message], messageNamespace: MessageId.Namespace) -> Signal { return signal |> map(Optional.init) diff --git a/submodules/TelegramCore/Sources/State/ManagedCloudChatRemoveMessagesOperations.swift b/submodules/TelegramCore/Sources/State/ManagedCloudChatRemoveMessagesOperations.swift index 1e4572414b7..18743c84504 100644 --- a/submodules/TelegramCore/Sources/State/ManagedCloudChatRemoveMessagesOperations.swift +++ b/submodules/TelegramCore/Sources/State/ManagedCloudChatRemoveMessagesOperations.swift @@ -127,10 +127,14 @@ func managedCloudChatRemoveMessagesOperations(postbox: Postbox, network: Network private func removeMessages(postbox: Postbox, network: Network, stateManager: AccountStateManager, peer: Peer, operation: CloudChatRemoveMessagesOperation) -> Signal { var isScheduled = false + var isQuickReply = false for id in operation.messageIds { if id.namespace == Namespaces.Message.ScheduledCloud { isScheduled = true break + } else if id.namespace == Namespaces.Message.QuickReplyCloud { + isQuickReply = true + break } } @@ -160,6 +164,32 @@ private func removeMessages(postbox: Postbox, network: Network, stateManager: Ac } else { return .complete() } + } else if isQuickReply { + if let threadId = operation.threadId { + var signal: Signal = .complete() + for s in stride(from: 0, to: operation.messageIds.count, by: 100) { + let ids = Array(operation.messageIds[s ..< min(s + 100, operation.messageIds.count)]) + let partSignal = network.request(Api.functions.messages.deleteQuickReplyMessages(shortcutId: Int32(clamping: threadId), id: ids.map(\.id))) + |> map { result -> Api.Updates? in + return result + } + |> `catch` { _ in + return .single(nil) + } + |> mapToSignal { updates -> Signal in + if let updates = updates { + stateManager.addUpdates(updates) + } + return .complete() + } + + signal = signal + |> then(partSignal) + } + return signal + } else { + return .complete() + } } else if peer.id.namespace == Namespaces.Peer.CloudChannel { if let inputChannel = apiInputChannel(peer) { var signal: Signal = .complete() @@ -329,7 +359,7 @@ private func removeChat(transaction: Transaction, postbox: Postbox, network: Net return requestClearHistory(postbox: postbox, network: network, stateManager: stateManager, inputPeer: inputPeer, maxId: operation.topMessageId?.id ?? Int32.max - 1, justClear: false, minTimestamp: nil, maxTimestamp: nil, type: operation.deleteGloballyIfPossible ? .forEveryone : .forLocalPeer) |> then(reportSignal) |> then(postbox.transaction { transaction -> Void in - _internal_clearHistory(transaction: transaction, mediaBox: postbox.mediaBox, peerId: peer.id, threadId: nil, namespaces: .not(Namespaces.Message.allScheduled)) + _internal_clearHistory(transaction: transaction, mediaBox: postbox.mediaBox, peerId: peer.id, threadId: nil, namespaces: .not(Namespaces.Message.allNonRegular)) }) } else { return .complete() @@ -386,6 +416,27 @@ private func requestClearHistory(postbox: Postbox, network: Network, stateManage private func _internal_clearHistory(transaction: Transaction, postbox: Postbox, network: Network, stateManager: AccountStateManager, peer: Peer, operation: CloudChatClearHistoryOperation) -> Signal { if peer.id.namespace == Namespaces.Peer.CloudGroup || peer.id.namespace == Namespaces.Peer.CloudUser { + if case .quickReplyMessages = operation.type { + guard let threadId = operation.threadId else { + return .complete() + } + + let signal = network.request(Api.functions.messages.deleteQuickReplyShortcut(shortcutId: Int32(clamping: threadId))) + |> map { result -> Api.Bool? in + return result + } + |> `catch` { _ -> Signal in + return .single(nil) + } + |> mapToSignal { result -> Signal in + return .fail(true) + } + return (signal |> restart) + |> `catch` { _ -> Signal in + return .complete() + } + } + if let inputPeer = apiInputPeer(peer) { if peer.id == stateManager.accountPeerId, let threadId = operation.threadId { guard let inputSubPeer = transaction.getPeer(PeerId(threadId)).flatMap(apiInputPeer) else { @@ -407,7 +458,7 @@ private func _internal_clearHistory(transaction: Transaction, postbox: Postbox, return result } |> `catch` { _ -> Signal in - return .fail(true) + return .single(nil) } |> mapToSignal { result -> Signal in if let result = result { diff --git a/submodules/TelegramCore/Sources/State/ManagedConsumePersonalMessagesActions.swift b/submodules/TelegramCore/Sources/State/ManagedConsumePersonalMessagesActions.swift index f253215b505..ace5e77447a 100644 --- a/submodules/TelegramCore/Sources/State/ManagedConsumePersonalMessagesActions.swift +++ b/submodules/TelegramCore/Sources/State/ManagedConsumePersonalMessagesActions.swift @@ -21,7 +21,7 @@ private func md5Hash(_ data: Data) -> Md5Hash { return Md5Hash(data: hashData) } -private func md5StringHash(_ string: String) -> UInt64 { +func md5StringHash(_ string: String) -> UInt64 { guard let data = string.data(using: .utf8) else { return 0 } diff --git a/submodules/TelegramCore/Sources/State/PendingMessageManager.swift b/submodules/TelegramCore/Sources/State/PendingMessageManager.swift index bab864f5a68..8ed18226d66 100644 --- a/submodules/TelegramCore/Sources/State/PendingMessageManager.swift +++ b/submodules/TelegramCore/Sources/State/PendingMessageManager.swift @@ -793,6 +793,7 @@ public final class PendingMessageManager { var replyToStoryId: StoryId? var scheduleTime: Int32? var sendAsPeerId: PeerId? + var quickReply: OutgoingQuickReplyMessageAttribute? var flags: Int32 = 0 @@ -821,6 +822,8 @@ public final class PendingMessageManager { hideCaptions = attribute.hideCaptions } else if let attribute = attribute as? SendAsMessageAttribute { sendAsPeerId = attribute.peerId + } else if let attribute = attribute as? OutgoingQuickReplyMessageAttribute { + quickReply = attribute } } @@ -871,6 +874,16 @@ public final class PendingMessageManager { topMsgId = Int32(clamping: threadId) } + var quickReplyShortcut: Api.InputQuickReplyShortcut? + if let quickReply { + if let threadId = messages[0].0.threadId { + quickReplyShortcut = .inputQuickReplyShortcutId(shortcutId: Int32(clamping: threadId)) + } else { + quickReplyShortcut = .inputQuickReplyShortcut(shortcut: quickReply.shortcut) + } + flags |= 1 << 17 + } + let forwardPeerIds = Set(forwardIds.map { $0.0.peerId }) if forwardPeerIds.count != 1 { assertionFailure() @@ -878,7 +891,7 @@ public final class PendingMessageManager { } else if let inputSourcePeerId = forwardPeerIds.first, let inputSourcePeer = transaction.getPeer(inputSourcePeerId).flatMap(apiInputPeer) { let dependencyTag = PendingMessageRequestDependencyTag(messageId: messages[0].0.id) - sendMessageRequest = network.request(Api.functions.messages.forwardMessages(flags: flags, fromPeer: inputSourcePeer, id: forwardIds.map { $0.0.id }, randomId: forwardIds.map { $0.1 }, toPeer: inputPeer, topMsgId: topMsgId, scheduleDate: scheduleTime, sendAs: sendAsInputPeer), tag: dependencyTag) + sendMessageRequest = network.request(Api.functions.messages.forwardMessages(flags: flags, fromPeer: inputSourcePeer, id: forwardIds.map { $0.0.id }, randomId: forwardIds.map { $0.1 }, toPeer: inputPeer, topMsgId: topMsgId, scheduleDate: scheduleTime, sendAs: sendAsInputPeer, quickReplyShortcut: quickReplyShortcut), tag: dependencyTag) } else { assertionFailure() sendMessageRequest = .fail(MTRpcError(errorCode: 400, errorDescription: "Invalid forward source")) @@ -993,7 +1006,17 @@ public final class PendingMessageManager { } } - sendMessageRequest = network.request(Api.functions.messages.sendMultiMedia(flags: flags, peer: inputPeer, replyTo: replyTo, multiMedia: singleMedias, scheduleDate: scheduleTime, sendAs: sendAsInputPeer)) + var quickReplyShortcut: Api.InputQuickReplyShortcut? + if let quickReply { + if let threadId = messages[0].0.threadId { + quickReplyShortcut = .inputQuickReplyShortcutId(shortcutId: Int32(clamping: threadId)) + } else { + quickReplyShortcut = .inputQuickReplyShortcut(shortcut: quickReply.shortcut) + } + flags |= 1 << 17 + } + + sendMessageRequest = network.request(Api.functions.messages.sendMultiMedia(flags: flags, peer: inputPeer, replyTo: replyTo, multiMedia: singleMedias, scheduleDate: scheduleTime, sendAs: sendAsInputPeer, quickReplyShortcut: quickReplyShortcut)) } return sendMessageRequest @@ -1168,6 +1191,7 @@ public final class PendingMessageManager { var scheduleTime: Int32? var sendAsPeerId: PeerId? var bubbleUpEmojiOrStickersets = false + var quickReply: OutgoingQuickReplyMessageAttribute? var flags: Int32 = 0 @@ -1202,6 +1226,8 @@ public final class PendingMessageManager { scheduleTime = attribute.scheduleTime } else if let attribute = attribute as? SendAsMessageAttribute { sendAsPeerId = attribute.peerId + } else if let attribute = attribute as? OutgoingQuickReplyMessageAttribute { + quickReply = attribute } } @@ -1291,7 +1317,17 @@ public final class PendingMessageManager { } } - sendMessageRequest = network.requestWithAdditionalInfo(Api.functions.messages.sendMessage(flags: flags, peer: inputPeer, replyTo: replyTo, message: message.text, randomId: uniqueId, replyMarkup: nil, entities: messageEntities, scheduleDate: scheduleTime, sendAs: sendAsInputPeer), info: .acknowledgement, tag: dependencyTag) + var quickReplyShortcut: Api.InputQuickReplyShortcut? + if let quickReply { + if let threadId = message.threadId { + quickReplyShortcut = .inputQuickReplyShortcutId(shortcutId: Int32(clamping: threadId)) + } else { + quickReplyShortcut = .inputQuickReplyShortcut(shortcut: quickReply.shortcut) + } + flags |= 1 << 17 + } + + sendMessageRequest = network.requestWithAdditionalInfo(Api.functions.messages.sendMessage(flags: flags, peer: inputPeer, replyTo: replyTo, message: message.text, randomId: uniqueId, replyMarkup: nil, entities: messageEntities, scheduleDate: scheduleTime, sendAs: sendAsInputPeer, quickReplyShortcut: quickReplyShortcut), info: .acknowledgement, tag: dependencyTag) case let .media(inputMedia, text): if bubbleUpEmojiOrStickersets { flags |= Int32(1 << 15) @@ -1355,8 +1391,18 @@ public final class PendingMessageManager { flags |= 1 << 16 } } + + var quickReplyShortcut: Api.InputQuickReplyShortcut? + if let quickReply { + if let threadId = message.threadId { + quickReplyShortcut = .inputQuickReplyShortcutId(shortcutId: Int32(clamping: threadId)) + } else { + quickReplyShortcut = .inputQuickReplyShortcut(shortcut: quickReply.shortcut) + } + flags |= 1 << 17 + } - sendMessageRequest = network.request(Api.functions.messages.sendMedia(flags: flags, peer: inputPeer, replyTo: replyTo, media: inputMedia, message: text, randomId: uniqueId, replyMarkup: nil, entities: messageEntities, scheduleDate: scheduleTime, sendAs: sendAsInputPeer), tag: dependencyTag) + sendMessageRequest = network.request(Api.functions.messages.sendMedia(flags: flags, peer: inputPeer, replyTo: replyTo, media: inputMedia, message: text, randomId: uniqueId, replyMarkup: nil, entities: messageEntities, scheduleDate: scheduleTime, sendAs: sendAsInputPeer, quickReplyShortcut: quickReplyShortcut), tag: dependencyTag) |> map(NetworkRequestResult.result) case let .forward(sourceInfo): var topMsgId: Int32? @@ -1365,8 +1411,18 @@ public final class PendingMessageManager { topMsgId = Int32(clamping: threadId) } + var quickReplyShortcut: Api.InputQuickReplyShortcut? + if let quickReply { + if let threadId = message.threadId { + quickReplyShortcut = .inputQuickReplyShortcutId(shortcutId: Int32(clamping: threadId)) + } else { + quickReplyShortcut = .inputQuickReplyShortcut(shortcut: quickReply.shortcut) + } + flags |= 1 << 17 + } + if let forwardSourceInfoAttribute = forwardSourceInfoAttribute, let sourcePeer = transaction.getPeer(forwardSourceInfoAttribute.messageId.peerId), let sourceInputPeer = apiInputPeer(sourcePeer) { - sendMessageRequest = network.request(Api.functions.messages.forwardMessages(flags: flags, fromPeer: sourceInputPeer, id: [sourceInfo.messageId.id], randomId: [uniqueId], toPeer: inputPeer, topMsgId: topMsgId, scheduleDate: scheduleTime, sendAs: sendAsInputPeer), tag: dependencyTag) + sendMessageRequest = network.request(Api.functions.messages.forwardMessages(flags: flags, fromPeer: sourceInputPeer, id: [sourceInfo.messageId.id], randomId: [uniqueId], toPeer: inputPeer, topMsgId: topMsgId, scheduleDate: scheduleTime, sendAs: sendAsInputPeer, quickReplyShortcut: quickReplyShortcut), tag: dependencyTag) |> map(NetworkRequestResult.result) } else { sendMessageRequest = .fail(MTRpcError(errorCode: 400, errorDescription: "internal")) @@ -1429,7 +1485,17 @@ public final class PendingMessageManager { } } - sendMessageRequest = network.request(Api.functions.messages.sendInlineBotResult(flags: flags, peer: inputPeer, replyTo: replyTo, randomId: uniqueId, queryId: chatContextResult.queryId, id: chatContextResult.id, scheduleDate: scheduleTime, sendAs: sendAsInputPeer)) + var quickReplyShortcut: Api.InputQuickReplyShortcut? + if let quickReply { + if let threadId = message.threadId { + quickReplyShortcut = .inputQuickReplyShortcutId(shortcutId: Int32(clamping: threadId)) + } else { + quickReplyShortcut = .inputQuickReplyShortcut(shortcut: quickReply.shortcut) + } + flags |= 1 << 17 + } + + sendMessageRequest = network.request(Api.functions.messages.sendInlineBotResult(flags: flags, peer: inputPeer, replyTo: replyTo, randomId: uniqueId, queryId: chatContextResult.queryId, id: chatContextResult.id, scheduleDate: scheduleTime, sendAs: sendAsInputPeer, quickReplyShortcut: quickReplyShortcut)) |> map(NetworkRequestResult.result) case .messageScreenshot: let replyTo: Api.InputReplyTo @@ -1557,7 +1623,16 @@ public final class PendingMessageManager { private func applySentMessage(postbox: Postbox, stateManager: AccountStateManager, message: Message, content: PendingMessageUploadedContentAndReuploadInfo, result: Api.Updates) -> Signal { var apiMessage: Api.Message? for resultMessage in result.messages { - if let id = resultMessage.id(namespace: Namespaces.Message.allScheduled.contains(message.id.namespace) ? Namespaces.Message.ScheduledCloud : Namespaces.Message.Cloud) { + let targetNamespace: MessageId.Namespace + if Namespaces.Message.allScheduled.contains(message.id.namespace) { + targetNamespace = Namespaces.Message.ScheduledCloud + } else if Namespaces.Message.allQuickReply.contains(message.id.namespace) { + targetNamespace = Namespaces.Message.QuickReplyCloud + } else { + targetNamespace = Namespaces.Message.Cloud + } + + if let id = resultMessage.id(namespace: targetNamespace) { if id.peerId == message.id.peerId { apiMessage = resultMessage break @@ -1567,7 +1642,9 @@ public final class PendingMessageManager { let silent = message.muted var namespace = Namespaces.Message.Cloud - if let apiMessage = apiMessage, let id = apiMessage.id(namespace: message.scheduleTime != nil && message.scheduleTime == apiMessage.timestamp ? Namespaces.Message.ScheduledCloud : Namespaces.Message.Cloud) { + if message.id.namespace == Namespaces.Message.QuickReplyLocal { + namespace = Namespaces.Message.QuickReplyCloud + } else if let apiMessage = apiMessage, let id = apiMessage.id(namespace: message.scheduleTime != nil && message.scheduleTime == apiMessage.timestamp ? Namespaces.Message.ScheduledCloud : Namespaces.Message.Cloud) { namespace = id.namespace if let attribute = message.attributes.first(where: { $0 is OutgoingMessageInfoAttribute }) as? OutgoingMessageInfoAttribute, let correlationId = attribute.correlationId { @@ -1594,7 +1671,9 @@ public final class PendingMessageManager { private func applySentGroupMessages(postbox: Postbox, stateManager: AccountStateManager, messages: [Message], result: Api.Updates) -> Signal { var silent = false var namespace = Namespaces.Message.Cloud - if let message = messages.first, let apiMessage = result.messages.first, message.scheduleTime != nil && message.scheduleTime == apiMessage.timestamp { + if let message = messages.first, message.id.namespace == Namespaces.Message.QuickReplyLocal { + namespace = Namespaces.Message.QuickReplyCloud + } else if let message = messages.first, let apiMessage = result.messages.first, message.scheduleTime != nil && message.scheduleTime == apiMessage.timestamp { namespace = Namespaces.Message.ScheduledCloud if message.muted { silent = true @@ -1605,7 +1684,7 @@ public final class PendingMessageManager { for i in 0 ..< messages.count { let message = messages[i] let apiMessage = result.messages[i] - if let id = apiMessage.id(namespace: message.scheduleTime != nil && message.scheduleTime == apiMessage.timestamp ? Namespaces.Message.ScheduledCloud : Namespaces.Message.Cloud) { + if let id = apiMessage.id(namespace: namespace) { if let attribute = message.attributes.first(where: { $0 is OutgoingMessageInfoAttribute }) as? OutgoingMessageInfoAttribute, let correlationId = attribute.correlationId { self.correlationIdToSentMessageId.with { value in value.mapping[correlationId] = id diff --git a/submodules/TelegramCore/Sources/State/Serialization.swift b/submodules/TelegramCore/Sources/State/Serialization.swift index ebac41c4cf2..6e1311188e7 100644 --- a/submodules/TelegramCore/Sources/State/Serialization.swift +++ b/submodules/TelegramCore/Sources/State/Serialization.swift @@ -210,7 +210,7 @@ public class BoxedMessage: NSObject { public class Serialization: NSObject, MTSerialization { public func currentLayer() -> UInt { - return 174 + return 176 } public func parseMessage(_ data: Data!) -> Any! { diff --git a/submodules/TelegramCore/Sources/State/UpdateMessageService.swift b/submodules/TelegramCore/Sources/State/UpdateMessageService.swift index d3091dad121..279064d68a6 100644 --- a/submodules/TelegramCore/Sources/State/UpdateMessageService.swift +++ b/submodules/TelegramCore/Sources/State/UpdateMessageService.swift @@ -58,7 +58,7 @@ class UpdateMessageService: NSObject, MTMessageService { self.putNext(groups) } case let .updateShortChatMessage(flags, id, fromId, chatId, message, pts, ptsCount, date, fwdFrom, viaBotId, replyHeader, entities, ttlPeriod): - let generatedMessage = Api.Message.message(flags: flags, id: id, fromId: .peerUser(userId: fromId), fromBoostsApplied: nil, peerId: Api.Peer.peerChat(chatId: chatId), savedPeerId: nil, fwdFrom: fwdFrom, viaBotId: viaBotId, replyTo: replyHeader, date: date, message: message, media: Api.MessageMedia.messageMediaEmpty, replyMarkup: nil, entities: entities, views: nil, forwards: nil, replies: nil, editDate: nil, postAuthor: nil, groupedId: nil, reactions: nil, restrictionReason: nil, ttlPeriod: ttlPeriod) + let generatedMessage = Api.Message.message(flags: flags, id: id, fromId: .peerUser(userId: fromId), fromBoostsApplied: nil, peerId: Api.Peer.peerChat(chatId: chatId), savedPeerId: nil, fwdFrom: fwdFrom, viaBotId: viaBotId, replyTo: replyHeader, date: date, message: message, media: Api.MessageMedia.messageMediaEmpty, replyMarkup: nil, entities: entities, views: nil, forwards: nil, replies: nil, editDate: nil, postAuthor: nil, groupedId: nil, reactions: nil, restrictionReason: nil, ttlPeriod: ttlPeriod, quickReplyShortcutId: nil) let update = Api.Update.updateNewMessage(message: generatedMessage, pts: pts, ptsCount: ptsCount) let groups = groupUpdates([update], users: [], chats: [], date: date, seqRange: nil) if groups.count != 0 { @@ -74,7 +74,7 @@ class UpdateMessageService: NSObject, MTMessageService { let generatedPeerId = Api.Peer.peerUser(userId: userId) - let generatedMessage = Api.Message.message(flags: flags, id: id, fromId: generatedFromId, fromBoostsApplied: nil, peerId: generatedPeerId, savedPeerId: nil, fwdFrom: fwdFrom, viaBotId: viaBotId, replyTo: replyHeader, date: date, message: message, media: Api.MessageMedia.messageMediaEmpty, replyMarkup: nil, entities: entities, views: nil, forwards: nil, replies: nil, editDate: nil, postAuthor: nil, groupedId: nil, reactions: nil, restrictionReason: nil, ttlPeriod: ttlPeriod) + let generatedMessage = Api.Message.message(flags: flags, id: id, fromId: generatedFromId, fromBoostsApplied: nil, peerId: generatedPeerId, savedPeerId: nil, fwdFrom: fwdFrom, viaBotId: viaBotId, replyTo: replyHeader, date: date, message: message, media: Api.MessageMedia.messageMediaEmpty, replyMarkup: nil, entities: entities, views: nil, forwards: nil, replies: nil, editDate: nil, postAuthor: nil, groupedId: nil, reactions: nil, restrictionReason: nil, ttlPeriod: ttlPeriod, quickReplyShortcutId: nil) let update = Api.Update.updateNewMessage(message: generatedMessage, pts: pts, ptsCount: ptsCount) let groups = groupUpdates([update], users: [], chats: [], date: date, seqRange: nil) if groups.count != 0 { diff --git a/submodules/TelegramCore/Sources/State/UpdatesApiUtils.swift b/submodules/TelegramCore/Sources/State/UpdatesApiUtils.swift index 87f132c4ca7..a7fcc880937 100644 --- a/submodules/TelegramCore/Sources/State/UpdatesApiUtils.swift +++ b/submodules/TelegramCore/Sources/State/UpdatesApiUtils.swift @@ -104,7 +104,7 @@ extension Api.MessageMedia { extension Api.Message { var rawId: Int32 { switch self { - case let .message(_, id, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _): + case let .message(_, id, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _): return id case let .messageEmpty(_, id, _): return id @@ -115,7 +115,7 @@ extension Api.Message { func id(namespace: MessageId.Namespace = Namespaces.Message.Cloud) -> MessageId? { switch self { - case let .message(_, id, _, _, messagePeerId, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _): + case let .message(_, id, _, _, messagePeerId, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _): let peerId: PeerId = messagePeerId.peerId return MessageId(peerId: peerId, namespace: namespace, id: id) case let .messageEmpty(_, id, peerId): @@ -132,7 +132,7 @@ extension Api.Message { var peerId: PeerId? { switch self { - case let .message(_, _, _, _, messagePeerId, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _): + case let .message(_, _, _, _, messagePeerId, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _): let peerId: PeerId = messagePeerId.peerId return peerId case let .messageEmpty(_, _, peerId): @@ -145,7 +145,7 @@ extension Api.Message { var timestamp: Int32? { switch self { - case let .message(_, _, _, _, _, _, _, _, _, date, _, _, _, _, _, _, _, _, _, _, _, _, _): + case let .message(_, _, _, _, _, _, _, _, _, date, _, _, _, _, _, _, _, _, _, _, _, _, _, _): return date case let .messageService(_, _, _, _, _, date, _, _): return date @@ -156,7 +156,7 @@ extension Api.Message { var preCachedResources: [(MediaResource, Data)]? { switch self { - case let .message(_, _, _, _, _, _, _, _, _, _, _, media, _, _, _, _, _, _, _, _, _, _, _): + case let .message(_, _, _, _, _, _, _, _, _, _, _, media, _, _, _, _, _, _, _, _, _, _, _, _): return media?.preCachedResources default: return nil @@ -165,7 +165,7 @@ extension Api.Message { var preCachedStories: [StoryId: Api.StoryItem]? { switch self { - case let .message(_, _, _, _, _, _, _, _, _, _, _, media, _, _, _, _, _, _, _, _, _, _, _): + case let .message(_, _, _, _, _, _, _, _, _, _, _, media, _, _, _, _, _, _, _, _, _, _, _, _): return media?.preCachedStories default: return nil @@ -271,6 +271,8 @@ extension Api.Update { return message case let .updateNewScheduledMessage(message): return message + case let .updateQuickReplyMessage(message): + return message default: return nil } @@ -334,6 +336,8 @@ extension Api.Update { return [peer.peerId] case let .updateNewScheduledMessage(message): return apiMessagePeerIds(message) + case let .updateQuickReplyMessage(message): + return apiMessagePeerIds(message) default: return [] } @@ -349,6 +353,8 @@ extension Api.Update { return apiMessageAssociatedMessageIds(message) case let .updateNewScheduledMessage(message): return apiMessageAssociatedMessageIds(message) + case let .updateQuickReplyMessage(message): + return apiMessageAssociatedMessageIds(message) default: break } diff --git a/submodules/TelegramCore/Sources/SyncCore/QuickReplyMessageAttribute.swift b/submodules/TelegramCore/Sources/SyncCore/QuickReplyMessageAttribute.swift new file mode 100644 index 00000000000..9df4ade3998 --- /dev/null +++ b/submodules/TelegramCore/Sources/SyncCore/QuickReplyMessageAttribute.swift @@ -0,0 +1,23 @@ +import Foundation +import Postbox +import TelegramApi + +public final class OutgoingQuickReplyMessageAttribute: Equatable, MessageAttribute { + public let shortcut: String + + public init(shortcut: String) { + self.shortcut = shortcut + } + + required public init(decoder: PostboxDecoder) { + self.shortcut = decoder.decodeStringForKey("s", orElse: "") + } + + public func encode(_ encoder: PostboxEncoder) { + encoder.encodeString(self.shortcut, forKey: "s") + } + + public static func ==(lhs: OutgoingQuickReplyMessageAttribute, rhs: OutgoingQuickReplyMessageAttribute) -> Bool { + return true + } +} diff --git a/submodules/TelegramCore/Sources/SyncCore/SyncCore_CachedUserData.swift b/submodules/TelegramCore/Sources/SyncCore/SyncCore_CachedUserData.swift index 87c5e650038..5e9a6f657bf 100644 --- a/submodules/TelegramCore/Sources/SyncCore/SyncCore_CachedUserData.swift +++ b/submodules/TelegramCore/Sources/SyncCore/SyncCore_CachedUserData.swift @@ -247,6 +247,211 @@ public final class EditableBotInfo: PostboxCoding, Equatable { } } +public final class TelegramBusinessHours: Equatable, Codable { + public struct WorkingTimeInterval: Equatable, Codable { + private enum CodingKeys: String, CodingKey { + case startMinute + case endMinute + } + + public let startMinute: Int + public let endMinute: Int + + public init(startMinute: Int, endMinute: Int) { + self.startMinute = startMinute + self.endMinute = endMinute + } + + public init(from decoder: Decoder) throws { + let container = try decoder.container(keyedBy: CodingKeys.self) + + self.startMinute = Int(try container.decode(Int32.self, forKey: .startMinute)) + self.endMinute = Int(try container.decode(Int32.self, forKey: .endMinute)) + } + + public func encode(to encoder: Encoder) throws { + var container = encoder.container(keyedBy: CodingKeys.self) + + try container.encode(Int32(clamping: self.startMinute), forKey: .startMinute) + try container.encode(Int32(clamping: self.endMinute), forKey: .endMinute) + } + + public static func ==(lhs: WorkingTimeInterval, rhs: WorkingTimeInterval) -> Bool { + if lhs.startMinute != rhs.startMinute { + return false + } + if lhs.endMinute != rhs.endMinute { + return false + } + return true + } + } + + public let timezoneId: String + public let weeklyTimeIntervals: [WorkingTimeInterval] + + public init(timezoneId: String, weeklyTimeIntervals: [WorkingTimeInterval]) { + self.timezoneId = timezoneId + self.weeklyTimeIntervals = weeklyTimeIntervals + } + + public static func ==(lhs: TelegramBusinessHours, rhs: TelegramBusinessHours) -> Bool { + if lhs.timezoneId != rhs.timezoneId { + return false + } + if lhs.weeklyTimeIntervals != rhs.weeklyTimeIntervals { + return false + } + return true + } + + public enum WeekDay { + case closed + case open + case intervals([WorkingTimeInterval]) + } + + public func splitIntoWeekDays() -> [WeekDay] { + var mappedDays: [[WorkingTimeInterval]] = Array(repeating: [], count: 7) + + var weekMinutes = IndexSet() + for interval in self.weeklyTimeIntervals { + weekMinutes.insert(integersIn: interval.startMinute ..< interval.endMinute) + } + + for i in 0 ..< mappedDays.count { + let dayRange = i * 24 * 60 ..< (i + 1) * 24 * 60 + var removeMinutes = IndexSet() + inner: for range in weekMinutes.rangeView { + if range.lowerBound >= dayRange.upperBound { + break inner + } else { + let clippedRange: Range + if range.lowerBound == dayRange.lowerBound { + clippedRange = range.lowerBound ..< min(range.upperBound, dayRange.upperBound) + } else { + clippedRange = range.lowerBound ..< min(range.upperBound, dayRange.upperBound + 12 * 60) + } + + let startTimeInsideDay = clippedRange.lowerBound - i * (24 * 60) + let endTimeInsideDay = clippedRange.upperBound - i * (24 * 60) + + mappedDays[i].append(WorkingTimeInterval( + startMinute: startTimeInsideDay, + endMinute: endTimeInsideDay + )) + removeMinutes.insert(integersIn: clippedRange) + } + } + + weekMinutes.subtract(removeMinutes) + } + + return mappedDays.map { day -> WeekDay in + var minutes = IndexSet() + for interval in day { + minutes.insert(integersIn: interval.startMinute ..< interval.endMinute) + } + if minutes.isEmpty { + return .closed + } else if minutes == IndexSet(integersIn: 0 ..< 24 * 60) || minutes == IndexSet(integersIn: 0 ..< (24 * 60 - 1)) { + return .open + } else { + return .intervals(day) + } + } + } + + public func weekMinuteSet() -> IndexSet { + var result = IndexSet() + + for interval in self.weeklyTimeIntervals { + result.insert(integersIn: interval.startMinute ..< interval.endMinute) + } + + return result + } +} + +public final class TelegramBusinessLocation: Equatable, Codable { + public struct Coordinates: Equatable, Codable { + public let latitude: Double + public let longitude: Double + + public init(latitude: Double, longitude: Double) { + self.latitude = latitude + self.longitude = longitude + } + } + + public let address: String + public let coordinates: Coordinates? + + public init(address: String, coordinates: Coordinates?) { + self.address = address + self.coordinates = coordinates + } + + public static func ==(lhs: TelegramBusinessLocation, rhs: TelegramBusinessLocation) -> Bool { + if lhs.address != rhs.address { + return false + } + if lhs.coordinates != rhs.coordinates { + return false + } + return true + } +} + +extension TelegramBusinessHours.WorkingTimeInterval { + init(apiInterval: Api.BusinessWeeklyOpen) { + switch apiInterval { + case let .businessWeeklyOpen(startMinute, endMinute): + self.init(startMinute: Int(startMinute), endMinute: Int(endMinute)) + } + } + + var apiInterval: Api.BusinessWeeklyOpen { + return .businessWeeklyOpen(startMinute: Int32(clamping: self.startMinute), endMinute: Int32(clamping: self.endMinute)) + } +} + +extension TelegramBusinessHours { + convenience init(apiWorkingHours: Api.BusinessWorkHours) { + switch apiWorkingHours { + case let .businessWorkHours(_, timezoneId, weeklyOpen): + self.init(timezoneId: timezoneId, weeklyTimeIntervals: weeklyOpen.map(TelegramBusinessHours.WorkingTimeInterval.init(apiInterval:))) + } + } + + var apiBusinessHours: Api.BusinessWorkHours { + return .businessWorkHours(flags: 0, timezoneId: self.timezoneId, weeklyOpen: self.weeklyTimeIntervals.map(\.apiInterval)) + } +} + +extension TelegramBusinessLocation.Coordinates { + init?(apiGeoPoint: Api.GeoPoint) { + switch apiGeoPoint { + case let .geoPoint(_, long, lat, _, _): + self.init(latitude: lat, longitude: long) + case .geoPointEmpty: + return nil + } + } + + var apiInputGeoPoint: Api.InputGeoPoint { + return .inputGeoPoint(flags: 0, lat: self.latitude, long: self.longitude, accuracyRadius: nil) + } +} + +extension TelegramBusinessLocation { + convenience init(apiLocation: Api.BusinessLocation) { + switch apiLocation { + case let .businessLocation(_, geoPoint, address): + self.init(address: address, coordinates: geoPoint.flatMap { Coordinates(apiGeoPoint: $0) }) + } + } +} public final class CachedUserData: CachedPeerData { public let about: String? @@ -270,6 +475,11 @@ public final class CachedUserData: CachedPeerData { public let voiceMessagesAvailable: Bool public let wallpaper: TelegramWallpaper? public let flags: CachedUserFlags + public let businessHours: TelegramBusinessHours? + public let businessLocation: TelegramBusinessLocation? + public let greetingMessage: TelegramBusinessGreetingMessage? + public let awayMessage: TelegramBusinessAwayMessage? + public let connectedBot: TelegramAccountConnectedBot? public let peerIds: Set public let messageIds: Set @@ -297,11 +507,16 @@ public final class CachedUserData: CachedPeerData { self.voiceMessagesAvailable = true self.wallpaper = nil self.flags = CachedUserFlags() + self.businessHours = nil + self.businessLocation = nil self.peerIds = Set() self.messageIds = Set() + self.greetingMessage = nil + self.awayMessage = nil + self.connectedBot = nil } - public init(about: String?, botInfo: BotInfo?, editableBotInfo: EditableBotInfo?, peerStatusSettings: PeerStatusSettings?, pinnedMessageId: MessageId?, isBlocked: Bool, commonGroupCount: Int32, voiceCallsAvailable: Bool, videoCallsAvailable: Bool, callsPrivate: Bool, canPinMessages: Bool, hasScheduledMessages: Bool, autoremoveTimeout: CachedPeerAutoremoveTimeout, themeEmoticon: String?, photo: CachedPeerProfilePhoto, personalPhoto: CachedPeerProfilePhoto, fallbackPhoto: CachedPeerProfilePhoto, premiumGiftOptions: [CachedPremiumGiftOption], voiceMessagesAvailable: Bool, wallpaper: TelegramWallpaper?, flags: CachedUserFlags) { + public init(about: String?, botInfo: BotInfo?, editableBotInfo: EditableBotInfo?, peerStatusSettings: PeerStatusSettings?, pinnedMessageId: MessageId?, isBlocked: Bool, commonGroupCount: Int32, voiceCallsAvailable: Bool, videoCallsAvailable: Bool, callsPrivate: Bool, canPinMessages: Bool, hasScheduledMessages: Bool, autoremoveTimeout: CachedPeerAutoremoveTimeout, themeEmoticon: String?, photo: CachedPeerProfilePhoto, personalPhoto: CachedPeerProfilePhoto, fallbackPhoto: CachedPeerProfilePhoto, premiumGiftOptions: [CachedPremiumGiftOption], voiceMessagesAvailable: Bool, wallpaper: TelegramWallpaper?, flags: CachedUserFlags, businessHours: TelegramBusinessHours?, businessLocation: TelegramBusinessLocation?, greetingMessage: TelegramBusinessGreetingMessage?, awayMessage: TelegramBusinessAwayMessage?, connectedBot: TelegramAccountConnectedBot?) { self.about = about self.botInfo = botInfo self.editableBotInfo = editableBotInfo @@ -323,6 +538,11 @@ public final class CachedUserData: CachedPeerData { self.voiceMessagesAvailable = voiceMessagesAvailable self.wallpaper = wallpaper self.flags = flags + self.businessHours = businessHours + self.businessLocation = businessLocation + self.greetingMessage = greetingMessage + self.awayMessage = awayMessage + self.connectedBot = connectedBot self.peerIds = Set() @@ -375,6 +595,13 @@ public final class CachedUserData: CachedPeerData { messageIds.insert(pinnedMessageId) } self.messageIds = messageIds + + self.businessHours = decoder.decodeCodable(TelegramBusinessHours.self, forKey: "bhrs") + self.businessLocation = decoder.decodeCodable(TelegramBusinessLocation.self, forKey: "bloc") + + self.greetingMessage = decoder.decodeCodable(TelegramBusinessGreetingMessage.self, forKey: "bgreet") + self.awayMessage = decoder.decodeCodable(TelegramBusinessAwayMessage.self, forKey: "baway") + self.connectedBot = decoder.decodeCodable(TelegramAccountConnectedBot.self, forKey: "bbot") } public func encode(_ encoder: PostboxEncoder) { @@ -435,6 +662,36 @@ public final class CachedUserData: CachedPeerData { } encoder.encodeInt32(self.flags.rawValue, forKey: "fl") + + if let businessHours = self.businessHours { + encoder.encodeCodable(businessHours, forKey: "bhrs") + } else { + encoder.encodeNil(forKey: "bhrs") + } + + if let businessLocation = self.businessLocation { + encoder.encodeCodable(businessLocation, forKey: "bloc") + } else { + encoder.encodeNil(forKey: "bloc") + } + + if let greetingMessage = self.greetingMessage { + encoder.encodeCodable(greetingMessage, forKey: "bgreet") + } else { + encoder.encodeNil(forKey: "bgreet") + } + + if let awayMessage = self.awayMessage { + encoder.encodeCodable(awayMessage, forKey: "baway") + } else { + encoder.encodeNil(forKey: "baway") + } + + if let connectedBot = self.connectedBot { + encoder.encodeCodable(connectedBot, forKey: "bbot") + } else { + encoder.encodeNil(forKey: "bbot") + } } public func isEqual(to: CachedPeerData) -> Bool { @@ -448,91 +705,126 @@ public final class CachedUserData: CachedPeerData { if other.canPinMessages != self.canPinMessages { return false } + if other.businessHours != self.businessHours { + return false + } + if other.businessLocation != self.businessLocation { + return false + } + if other.greetingMessage != self.greetingMessage { + return false + } + if other.awayMessage != self.awayMessage { + return false + } + if other.connectedBot != self.connectedBot { + return false + } return other.about == self.about && other.botInfo == self.botInfo && other.editableBotInfo == self.editableBotInfo && self.peerStatusSettings == other.peerStatusSettings && self.isBlocked == other.isBlocked && self.commonGroupCount == other.commonGroupCount && self.voiceCallsAvailable == other.voiceCallsAvailable && self.videoCallsAvailable == other.videoCallsAvailable && self.callsPrivate == other.callsPrivate && self.hasScheduledMessages == other.hasScheduledMessages && self.autoremoveTimeout == other.autoremoveTimeout && self.themeEmoticon == other.themeEmoticon && self.photo == other.photo && self.personalPhoto == other.personalPhoto && self.fallbackPhoto == other.fallbackPhoto && self.premiumGiftOptions == other.premiumGiftOptions && self.voiceMessagesAvailable == other.voiceMessagesAvailable && self.flags == other.flags && self.wallpaper == other.wallpaper } public func withUpdatedAbout(_ about: String?) -> CachedUserData { - return CachedUserData(about: about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags) + return CachedUserData(about: about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) } public func withUpdatedBotInfo(_ botInfo: BotInfo?) -> CachedUserData { - return CachedUserData(about: self.about, botInfo: botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags) + return CachedUserData(about: self.about, botInfo: botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) } public func withUpdatedEditableBotInfo(_ editableBotInfo: EditableBotInfo?) -> CachedUserData { - return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags) + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) } public func withUpdatedPeerStatusSettings(_ peerStatusSettings: PeerStatusSettings) -> CachedUserData { - return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags) + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) } public func withUpdatedPinnedMessageId(_ pinnedMessageId: MessageId?) -> CachedUserData { - return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags) + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) } public func withUpdatedIsBlocked(_ isBlocked: Bool) -> CachedUserData { - return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags) + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) } public func withUpdatedCommonGroupCount(_ commonGroupCount: Int32) -> CachedUserData { - return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags) + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) } public func withUpdatedVoiceCallsAvailable(_ voiceCallsAvailable: Bool) -> CachedUserData { - return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags) + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) } public func withUpdatedVideoCallsAvailable(_ videoCallsAvailable: Bool) -> CachedUserData { - return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags) + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) } public func withUpdatedCallsPrivate(_ callsPrivate: Bool) -> CachedUserData { - return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags) + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) } public func withUpdatedCanPinMessages(_ canPinMessages: Bool) -> CachedUserData { - return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags) + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) } public func withUpdatedHasScheduledMessages(_ hasScheduledMessages: Bool) -> CachedUserData { - return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags) + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) } public func withUpdatedAutoremoveTimeout(_ autoremoveTimeout: CachedPeerAutoremoveTimeout) -> CachedUserData { - return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags) + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) } public func withUpdatedThemeEmoticon(_ themeEmoticon: String?) -> CachedUserData { - return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags) + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) } public func withUpdatedPhoto(_ photo: CachedPeerProfilePhoto) -> CachedUserData { - return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags) + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) } public func withUpdatedPersonalPhoto(_ personalPhoto: CachedPeerProfilePhoto) -> CachedUserData { - return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags) + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) } public func withUpdatedFallbackPhoto(_ fallbackPhoto: CachedPeerProfilePhoto) -> CachedUserData { - return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags) + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) } public func withUpdatedPremiumGiftOptions(_ premiumGiftOptions: [CachedPremiumGiftOption]) -> CachedUserData { - return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags) + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) } public func withUpdatedVoiceMessagesAvailable(_ voiceMessagesAvailable: Bool) -> CachedUserData { - return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags) + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) } public func withUpdatedWallpaper(_ wallpaper: TelegramWallpaper?) -> CachedUserData { - return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: wallpaper, flags: self.flags) + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) } public func withUpdatedFlags(_ flags: CachedUserFlags) -> CachedUserData { - return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: flags) + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) + } + + public func withUpdatedBusinessHours(_ businessHours: TelegramBusinessHours?) -> CachedUserData { + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) + } + + public func withUpdatedBusinessLocation(_ businessLocation: TelegramBusinessLocation?) -> CachedUserData { + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) + } + + public func withUpdatedGreetingMessage(_ greetingMessage: TelegramBusinessGreetingMessage?) -> CachedUserData { + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: greetingMessage, awayMessage: self.awayMessage, connectedBot: self.connectedBot) + } + + public func withUpdatedAwayMessage(_ awayMessage: TelegramBusinessAwayMessage?) -> CachedUserData { + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: awayMessage, connectedBot: self.connectedBot) + } + + public func withUpdatedConnectedBot(_ connectedBot: TelegramAccountConnectedBot?) -> CachedUserData { + return CachedUserData(about: self.about, botInfo: self.botInfo, editableBotInfo: self.editableBotInfo, peerStatusSettings: self.peerStatusSettings, pinnedMessageId: self.pinnedMessageId, isBlocked: self.isBlocked, commonGroupCount: self.commonGroupCount, voiceCallsAvailable: self.voiceCallsAvailable, videoCallsAvailable: self.videoCallsAvailable, callsPrivate: self.callsPrivate, canPinMessages: self.canPinMessages, hasScheduledMessages: self.hasScheduledMessages, autoremoveTimeout: self.autoremoveTimeout, themeEmoticon: self.themeEmoticon, photo: self.photo, personalPhoto: self.personalPhoto, fallbackPhoto: self.fallbackPhoto, premiumGiftOptions: self.premiumGiftOptions, voiceMessagesAvailable: self.voiceMessagesAvailable, wallpaper: self.wallpaper, flags: self.flags, businessHours: self.businessHours, businessLocation: self.businessLocation, greetingMessage: self.greetingMessage, awayMessage: self.awayMessage, connectedBot: connectedBot) } } diff --git a/submodules/TelegramCore/Sources/SyncCore/SyncCore_CloudChatRemoveMessagesOperation.swift b/submodules/TelegramCore/Sources/SyncCore/SyncCore_CloudChatRemoveMessagesOperation.swift index 40d830bf1b8..41e23ccdcfd 100644 --- a/submodules/TelegramCore/Sources/SyncCore/SyncCore_CloudChatRemoveMessagesOperation.swift +++ b/submodules/TelegramCore/Sources/SyncCore/SyncCore_CloudChatRemoveMessagesOperation.swift @@ -24,21 +24,29 @@ public extension CloudChatRemoveMessagesType { public final class CloudChatRemoveMessagesOperation: PostboxCoding { public let messageIds: [MessageId] + public let threadId: Int64? public let type: CloudChatRemoveMessagesType - public init(messageIds: [MessageId], type: CloudChatRemoveMessagesType) { + public init(messageIds: [MessageId], threadId: Int64?, type: CloudChatRemoveMessagesType) { self.messageIds = messageIds + self.threadId = threadId self.type = type } public init(decoder: PostboxDecoder) { self.messageIds = MessageId.decodeArrayFromBuffer(decoder.decodeBytesForKeyNoCopy("i")!) + self.threadId = decoder.decodeOptionalInt64ForKey("threadId") self.type = CloudChatRemoveMessagesType(rawValue: decoder.decodeInt32ForKey("t", orElse: 0))! } public func encode(_ encoder: PostboxEncoder) { let buffer = WriteBuffer() MessageId.encodeArrayToBuffer(self.messageIds, buffer: buffer) + if let threadId = self.threadId { + encoder.encodeInt64(threadId, forKey: "threadId") + } else { + encoder.encodeNil(forKey: "threadId") + } encoder.encodeBytes(buffer, forKey: "i") encoder.encodeInt32(self.type.rawValue, forKey: "t") } @@ -88,6 +96,7 @@ public enum CloudChatClearHistoryType: Int32 { case forLocalPeer case forEveryone case scheduledMessages + case quickReplyMessages } public enum InteractiveHistoryClearingType: Int32 { diff --git a/submodules/TelegramCore/Sources/SyncCore/SyncCore_MediaReference.swift b/submodules/TelegramCore/Sources/SyncCore/SyncCore_MediaReference.swift index 66f409e0fdd..57a8473844a 100644 --- a/submodules/TelegramCore/Sources/SyncCore/SyncCore_MediaReference.swift +++ b/submodules/TelegramCore/Sources/SyncCore/SyncCore_MediaReference.swift @@ -7,7 +7,7 @@ public struct MessageReference: PostboxCoding, Hashable, Equatable { switch content { case .none: return nil - case let .message(peer, _, _, _, _, _): + case let .message(peer, _, _, _, _, _, _): return peer } } @@ -15,7 +15,7 @@ public struct MessageReference: PostboxCoding, Hashable, Equatable { switch content { case .none: return nil - case let .message(_, author, _, _, _, _): + case let .message(_, author, _, _, _, _, _): return author } } @@ -24,7 +24,7 @@ public struct MessageReference: PostboxCoding, Hashable, Equatable { switch content { case .none: return nil - case let .message(_, _, id, _, _, _): + case let .message(_, _, id, _, _, _, _): return id } } @@ -33,7 +33,7 @@ public struct MessageReference: PostboxCoding, Hashable, Equatable { switch content { case .none: return nil - case let .message(_, _, _, timestamp, _, _): + case let .message(_, _, _, timestamp, _, _, _): return timestamp } } @@ -42,7 +42,7 @@ public struct MessageReference: PostboxCoding, Hashable, Equatable { switch content { case .none: return nil - case let .message(_, _, _, _, incoming, _): + case let .message(_, _, _, _, incoming, _, _): return incoming } } @@ -51,11 +51,20 @@ public struct MessageReference: PostboxCoding, Hashable, Equatable { switch content { case .none: return nil - case let .message(_, _, _, _, _, secret): + case let .message(_, _, _, _, _, secret, _): return secret } } + public var threadId: Int64? { + switch content { + case .none: + return nil + case let .message(_, _, _, _, _, _, threadId): + return threadId + } + } + public init(_ message: Message) { if message.id.namespace != Namespaces.Message.Local, let peer = message.peers[message.id.peerId], let inputPeer = PeerReference(peer) { let author: PeerReference? @@ -64,13 +73,13 @@ public struct MessageReference: PostboxCoding, Hashable, Equatable { } else { author = nil } - self.content = .message(peer: inputPeer, author: author, id: message.id, timestamp: message.timestamp, incoming: message.flags.contains(.Incoming), secret: message.containsSecretMedia) + self.content = .message(peer: inputPeer, author: author, id: message.id, timestamp: message.timestamp, incoming: message.flags.contains(.Incoming), secret: message.containsSecretMedia, threadId: message.threadId) } else { self.content = .none } } - public init(peer: Peer, author: Peer?, id: MessageId, timestamp: Int32, incoming: Bool, secret: Bool) { + public init(peer: Peer, author: Peer?, id: MessageId, timestamp: Int32, incoming: Bool, secret: Bool, threadId: Int64?) { if let inputPeer = PeerReference(peer) { let a: PeerReference? if let peer = author { @@ -78,7 +87,7 @@ public struct MessageReference: PostboxCoding, Hashable, Equatable { } else { a = nil } - self.content = .message(peer: inputPeer, author: a, id: id, timestamp: timestamp, incoming: incoming, secret: secret) + self.content = .message(peer: inputPeer, author: a, id: id, timestamp: timestamp, incoming: incoming, secret: secret, threadId: threadId) } else { self.content = .none } @@ -95,14 +104,14 @@ public struct MessageReference: PostboxCoding, Hashable, Equatable { public enum MessageReferenceContent: PostboxCoding, Hashable, Equatable { case none - case message(peer: PeerReference, author: PeerReference?, id: MessageId, timestamp: Int32, incoming: Bool, secret: Bool) + case message(peer: PeerReference, author: PeerReference?, id: MessageId, timestamp: Int32, incoming: Bool, secret: Bool, threadId: Int64?) public init(decoder: PostboxDecoder) { switch decoder.decodeInt32ForKey("_r", orElse: 0) { case 0: self = .none case 1: - self = .message(peer: decoder.decodeObjectForKey("p", decoder: { PeerReference(decoder: $0) }) as! PeerReference, author: decoder.decodeObjectForKey("author") as? PeerReference, id: MessageId(peerId: PeerId(decoder.decodeInt64ForKey("i.p", orElse: 0)), namespace: decoder.decodeInt32ForKey("i.n", orElse: 0), id: decoder.decodeInt32ForKey("i.i", orElse: 0)), timestamp: 0, incoming: false, secret: false) + self = .message(peer: decoder.decodeObjectForKey("p", decoder: { PeerReference(decoder: $0) }) as! PeerReference, author: decoder.decodeObjectForKey("author") as? PeerReference, id: MessageId(peerId: PeerId(decoder.decodeInt64ForKey("i.p", orElse: 0)), namespace: decoder.decodeInt32ForKey("i.n", orElse: 0), id: decoder.decodeInt32ForKey("i.i", orElse: 0)), timestamp: 0, incoming: false, secret: false, threadId: decoder.decodeOptionalInt64ForKey("tid")) default: assertionFailure() self = .none @@ -113,7 +122,7 @@ public enum MessageReferenceContent: PostboxCoding, Hashable, Equatable { switch self { case .none: encoder.encodeInt32(0, forKey: "_r") - case let .message(peer, author, id, _, _, _): + case let .message(peer, author, id, _, _, _, threadId): encoder.encodeInt32(1, forKey: "_r") encoder.encodeObject(peer, forKey: "p") if let author = author { @@ -124,6 +133,11 @@ public enum MessageReferenceContent: PostboxCoding, Hashable, Equatable { encoder.encodeInt64(id.peerId.toInt64(), forKey: "i.p") encoder.encodeInt32(id.namespace, forKey: "i.n") encoder.encodeInt32(id.id, forKey: "i.i") + if let threadId { + encoder.encodeInt64(threadId, forKey: "tid") + } else { + encoder.encodeNil(forKey: "tid") + } } } } diff --git a/submodules/TelegramCore/Sources/SyncCore/SyncCore_Namespaces.swift b/submodules/TelegramCore/Sources/SyncCore/SyncCore_Namespaces.swift index c61b94706a2..958ac84482f 100644 --- a/submodules/TelegramCore/Sources/SyncCore/SyncCore_Namespaces.swift +++ b/submodules/TelegramCore/Sources/SyncCore/SyncCore_Namespaces.swift @@ -8,8 +8,12 @@ public struct Namespaces { public static let SecretIncoming: Int32 = 2 public static let ScheduledCloud: Int32 = 3 public static let ScheduledLocal: Int32 = 4 + public static let QuickReplyCloud: Int32 = 5 + public static let QuickReplyLocal: Int32 = 6 public static let allScheduled: Set = Set([Namespaces.Message.ScheduledCloud, Namespaces.Message.ScheduledLocal]) + public static let allQuickReply: Set = Set([Namespaces.Message.QuickReplyCloud, Namespaces.Message.QuickReplyLocal]) + public static let allNonRegular: Set = Set([Namespaces.Message.ScheduledCloud, Namespaces.Message.ScheduledLocal, Namespaces.Message.QuickReplyCloud, Namespaces.Message.QuickReplyLocal]) } public struct Media { @@ -280,6 +284,8 @@ private enum PreferencesKeyValues: Int32 { case audioTranscriptionTrialState = 33 case didCacheSavedMessageTagsPrefix = 34 case displaySavedChatsAsTopics = 35 + case shortcutMessages = 37 + case timezoneList = 38 } public func applicationSpecificPreferencesKey(_ value: Int32) -> ValueBoxKey { @@ -463,6 +469,18 @@ public struct PreferencesKeys { key.setInt32(0, value: PreferencesKeyValues.displaySavedChatsAsTopics.rawValue) return key } + + public static func shortcutMessages() -> ValueBoxKey { + let key = ValueBoxKey(length: 4) + key.setInt32(0, value: PreferencesKeyValues.shortcutMessages.rawValue) + return key + } + + public static func timezoneList() -> ValueBoxKey { + let key = ValueBoxKey(length: 4) + key.setInt32(0, value: PreferencesKeyValues.timezoneList.rawValue) + return key + } } private enum SharedDataKeyValues: Int32 { diff --git a/submodules/TelegramCore/Sources/TelegramEngine/AccountData/TelegramEngineAccountData.swift b/submodules/TelegramCore/Sources/TelegramEngine/AccountData/TelegramEngineAccountData.swift index f6d798ebe75..7c50cc13872 100644 --- a/submodules/TelegramCore/Sources/TelegramEngine/AccountData/TelegramEngineAccountData.swift +++ b/submodules/TelegramCore/Sources/TelegramEngine/AccountData/TelegramEngineAccountData.swift @@ -105,5 +105,105 @@ public extension TelegramEngine { |> ignoreValues |> then(remoteApply) } + + public func updateAccountBusinessHours(businessHours: TelegramBusinessHours?) -> Signal { + let peerId = self.account.peerId + + var flags: Int32 = 0 + if businessHours != nil { + flags |= 1 << 0 + } + let remoteApply: Signal = self.account.network.request(Api.functions.account.updateBusinessWorkHours(flags: flags, businessWorkHours: businessHours?.apiBusinessHours)) + |> `catch` { _ -> Signal in + return .single(.boolFalse) + } + |> ignoreValues + + return self.account.postbox.transaction { transaction -> Void in + transaction.updatePeerCachedData(peerIds: Set([peerId]), update: { _, current in + let current = current as? CachedUserData ?? CachedUserData() + return current.withUpdatedBusinessHours(businessHours) + }) + } + |> ignoreValues + |> then(remoteApply) + } + + public func updateAccountBusinessLocation(businessLocation: TelegramBusinessLocation?) -> Signal { + let peerId = self.account.peerId + + var flags: Int32 = 0 + + var inputGeoPoint: Api.InputGeoPoint? + var inputAddress: String? + if let businessLocation { + flags |= 1 << 0 + inputAddress = businessLocation.address + + inputGeoPoint = businessLocation.coordinates?.apiInputGeoPoint + if inputGeoPoint != nil { + flags |= 1 << 1 + } + } + + let remoteApply: Signal = self.account.network.request(Api.functions.account.updateBusinessLocation(flags: flags, geoPoint: inputGeoPoint, address: inputAddress)) + |> `catch` { _ -> Signal in + return .single(.boolFalse) + } + |> ignoreValues + + return self.account.postbox.transaction { transaction -> Void in + transaction.updatePeerCachedData(peerIds: Set([peerId]), update: { _, current in + let current = current as? CachedUserData ?? CachedUserData() + return current.withUpdatedBusinessLocation(businessLocation) + }) + } + |> ignoreValues + |> then(remoteApply) + } + + public func shortcutMessageList(onlyRemote: Bool) -> Signal { + return _internal_shortcutMessageList(account: self.account, onlyRemote: onlyRemote) + } + + public func keepShortcutMessageListUpdated() -> Signal { + return _internal_keepShortcutMessagesUpdated(account: self.account) + } + + public func editMessageShortcut(id: Int32, shortcut: String) { + let _ = _internal_editMessageShortcut(account: self.account, id: id, shortcut: shortcut).startStandalone() + } + + public func deleteMessageShortcuts(ids: [Int32]) { + let _ = _internal_deleteMessageShortcuts(account: self.account, ids: ids).startStandalone() + } + + public func reorderMessageShortcuts(ids: [Int32], completion: @escaping () -> Void) { + let _ = _internal_reorderMessageShortcuts(account: self.account, ids: ids, localCompletion: completion).startStandalone() + } + + public func sendMessageShortcut(peerId: EnginePeer.Id, id: Int32) { + let _ = _internal_sendMessageShortcut(account: self.account, peerId: peerId, id: id).startStandalone() + } + + public func cachedTimeZoneList() -> Signal { + return _internal_cachedTimeZoneList(account: self.account) + } + + public func keepCachedTimeZoneListUpdated() -> Signal { + return _internal_keepCachedTimeZoneListUpdated(account: self.account) + } + + public func updateBusinessGreetingMessage(greetingMessage: TelegramBusinessGreetingMessage?) -> Signal { + return _internal_updateBusinessGreetingMessage(account: self.account, greetingMessage: greetingMessage) + } + + public func updateBusinessAwayMessage(awayMessage: TelegramBusinessAwayMessage?) -> Signal { + return _internal_updateBusinessAwayMessage(account: self.account, awayMessage: awayMessage) + } + + public func setAccountConnectedBot(bot: TelegramAccountConnectedBot?) -> Signal { + return _internal_setAccountConnectedBot(account: self.account, bot: bot) + } } } diff --git a/submodules/TelegramCore/Sources/TelegramEngine/Data/PeersData.swift b/submodules/TelegramCore/Sources/TelegramEngine/Data/PeersData.swift index 08f63d5fab2..a036a33a419 100644 --- a/submodules/TelegramCore/Sources/TelegramEngine/Data/PeersData.swift +++ b/submodules/TelegramCore/Sources/TelegramEngine/Data/PeersData.swift @@ -4,6 +4,7 @@ import Postbox public typealias EngineExportedPeerInvitation = ExportedInvitation public typealias EngineSecretChatKeyFingerprint = SecretChatKeyFingerprint + public enum EnginePeerCachedInfoItem { case known(T) case unknown @@ -1113,6 +1114,91 @@ public extension TelegramEngine.EngineData.Item { } } + public struct SlowmodeTimeout: TelegramEngineDataItem, TelegramEngineMapKeyDataItem, PostboxViewDataItem { + public typealias Result = Int32? + + fileprivate var id: EnginePeer.Id + public var mapKey: EnginePeer.Id { + return self.id + } + + public init(id: EnginePeer.Id) { + self.id = id + } + + var key: PostboxViewKey { + return .cachedPeerData(peerId: self.id) + } + + func extract(view: PostboxView) -> Result { + guard let view = view as? CachedPeerDataView else { + preconditionFailure() + } + if let cachedData = view.cachedPeerData as? CachedChannelData { + return cachedData.slowModeTimeout + } else { + return nil + } + } + } + public struct SlowmodeValidUntilTimeout: TelegramEngineDataItem, TelegramEngineMapKeyDataItem, PostboxViewDataItem { + public typealias Result = Int32? + + fileprivate var id: EnginePeer.Id + public var mapKey: EnginePeer.Id { + return self.id + } + + public init(id: EnginePeer.Id) { + self.id = id + } + + var key: PostboxViewKey { + return .cachedPeerData(peerId: self.id) + } + + func extract(view: PostboxView) -> Result { + guard let view = view as? CachedPeerDataView else { + preconditionFailure() + } + if let cachedData = view.cachedPeerData as? CachedChannelData { + return cachedData.slowModeValidUntilTimestamp + } + return nil + } + } + + public struct CanAvoidGroupRestrictions: TelegramEngineDataItem, TelegramEngineMapKeyDataItem, PostboxViewDataItem { + public typealias Result = Bool + + fileprivate var id: EnginePeer.Id + public var mapKey: EnginePeer.Id { + return self.id + } + + public init(id: EnginePeer.Id) { + self.id = id + } + + var key: PostboxViewKey { + return .cachedPeerData(peerId: self.id) + } + + func extract(view: PostboxView) -> Result { + guard let view = view as? CachedPeerDataView else { + preconditionFailure() + } + if let cachedData = view.cachedPeerData as? CachedChannelData { + if let boostsToUnrestrict = cachedData.boostsToUnrestrict { + let appliedBoosts = cachedData.appliedBoosts ?? 0 + return boostsToUnrestrict <= appliedBoosts + } + } + return true + } + } + + public struct IsPremiumRequiredForMessaging: TelegramEngineDataItem, TelegramEngineMapKeyDataItem, AnyPostboxViewDataItem { public typealias Result = Bool @@ -1358,6 +1444,145 @@ public extension TelegramEngine.EngineData.Item { } } } + + public struct BusinessHours: TelegramEngineDataItem, TelegramEngineMapKeyDataItem, PostboxViewDataItem { + public typealias Result = TelegramBusinessHours? + + fileprivate var id: EnginePeer.Id + public var mapKey: EnginePeer.Id { + return self.id + } + public init(id: EnginePeer.Id) { + self.id = id + } + + var key: PostboxViewKey { + return .cachedPeerData(peerId: self.id) + } + + func extract(view: PostboxView) -> Result { + guard let view = view as? CachedPeerDataView else { + preconditionFailure() + } + if let cachedData = view.cachedPeerData as? CachedUserData { + return cachedData.businessHours + } else { + return nil + } + } + } + + public struct BusinessLocation: TelegramEngineDataItem, TelegramEngineMapKeyDataItem, PostboxViewDataItem { + public typealias Result = TelegramBusinessLocation? + + fileprivate var id: EnginePeer.Id + public var mapKey: EnginePeer.Id { + return self.id + } + + public init(id: EnginePeer.Id) { + self.id = id + } + + var key: PostboxViewKey { + return .cachedPeerData(peerId: self.id) + } + + func extract(view: PostboxView) -> Result { + guard let view = view as? CachedPeerDataView else { + preconditionFailure() + } + if let cachedData = view.cachedPeerData as? CachedUserData { + return cachedData.businessLocation + } else { + return nil + } + } + } + + public struct BusinessGreetingMessage: TelegramEngineDataItem, TelegramEngineMapKeyDataItem, PostboxViewDataItem { + public typealias Result = TelegramBusinessGreetingMessage? + + fileprivate var id: EnginePeer.Id + public var mapKey: EnginePeer.Id { + return self.id + } + + public init(id: EnginePeer.Id) { + self.id = id + } + + var key: PostboxViewKey { + return .cachedPeerData(peerId: self.id) + } + + func extract(view: PostboxView) -> Result { + guard let view = view as? CachedPeerDataView else { + preconditionFailure() + } + if let cachedData = view.cachedPeerData as? CachedUserData { + return cachedData.greetingMessage + } else { + return nil + } + } + } + + public struct BusinessAwayMessage: TelegramEngineDataItem, TelegramEngineMapKeyDataItem, PostboxViewDataItem { + public typealias Result = TelegramBusinessAwayMessage? + + fileprivate var id: EnginePeer.Id + public var mapKey: EnginePeer.Id { + return self.id + } + + public init(id: EnginePeer.Id) { + self.id = id + } + + var key: PostboxViewKey { + return .cachedPeerData(peerId: self.id) + } + + func extract(view: PostboxView) -> Result { + guard let view = view as? CachedPeerDataView else { + preconditionFailure() + } + if let cachedData = view.cachedPeerData as? CachedUserData { + return cachedData.awayMessage + } else { + return nil + } + } + } + + public struct BusinessConnectedBot: TelegramEngineDataItem, TelegramEngineMapKeyDataItem, PostboxViewDataItem { + public typealias Result = TelegramAccountConnectedBot? + + fileprivate var id: EnginePeer.Id + public var mapKey: EnginePeer.Id { + return self.id + } + + public init(id: EnginePeer.Id) { + self.id = id + } + + var key: PostboxViewKey { + return .cachedPeerData(peerId: self.id) + } + + func extract(view: PostboxView) -> Result { + guard let view = view as? CachedPeerDataView else { + preconditionFailure() + } + if let cachedData = view.cachedPeerData as? CachedUserData { + return cachedData.connectedBot + } else { + return nil + } + } + } } } diff --git a/submodules/TelegramCore/Sources/TelegramEngine/Data/TelegramEngineData.swift b/submodules/TelegramCore/Sources/TelegramEngine/Data/TelegramEngineData.swift index b72ee5fa4b2..dc59baf52f3 100644 --- a/submodules/TelegramCore/Sources/TelegramEngine/Data/TelegramEngineData.swift +++ b/submodules/TelegramCore/Sources/TelegramEngine/Data/TelegramEngineData.swift @@ -419,6 +419,7 @@ public extension TelegramEngine { T5.Result, T6.Result, T7.Result + ), NoError> { return self._subscribe(items: [ @@ -503,6 +504,72 @@ public extension TelegramEngine { ) } } + public func subscribe< + T0: TelegramEngineDataItem, + T1: TelegramEngineDataItem, + T2: TelegramEngineDataItem, + T3: TelegramEngineDataItem, + T4: TelegramEngineDataItem, + T5: TelegramEngineDataItem, + T6: TelegramEngineDataItem, + T7: TelegramEngineDataItem, + T8: TelegramEngineDataItem, + T9: TelegramEngineDataItem + + >( + _ t0: T0, + _ t1: T1, + _ t2: T2, + _ t3: T3, + _ t4: T4, + _ t5: T5, + _ t6: T6, + _ t7: T7, + _ t8: T8, + _ t9: T9 + + ) -> Signal< + ( + T0.Result, + T1.Result, + T2.Result, + T3.Result, + T4.Result, + T5.Result, + T6.Result, + T7.Result, + T8.Result, + T9.Result + ), + NoError> { + return self._subscribe(items: [ + t0 as! AnyPostboxViewDataItem, + t1 as! AnyPostboxViewDataItem, + t2 as! AnyPostboxViewDataItem, + t3 as! AnyPostboxViewDataItem, + t4 as! AnyPostboxViewDataItem, + t5 as! AnyPostboxViewDataItem, + t6 as! AnyPostboxViewDataItem, + t7 as! AnyPostboxViewDataItem, + t8 as! AnyPostboxViewDataItem, + t9 as! AnyPostboxViewDataItem + ]) + |> map { results -> (T0.Result, T1.Result, T2.Result, T3.Result, T4.Result, T5.Result, T6.Result, T7.Result, T8.Result, T9.Result) in + return ( + results[0] as! T0.Result, + results[1] as! T1.Result, + results[2] as! T2.Result, + results[3] as! T3.Result, + results[4] as! T4.Result, + results[5] as! T5.Result, + results[6] as! T6.Result, + results[7] as! T7.Result, + results[8] as! T8.Result, + results[9] as! T9.Result + ) + } + } + public func get< T0: TelegramEngineDataItem, diff --git a/submodules/TelegramCore/Sources/TelegramEngine/Messages/DeleteMessagesInteractively.swift b/submodules/TelegramCore/Sources/TelegramEngine/Messages/DeleteMessagesInteractively.swift index 12e0fa696a7..6fb62b81421 100644 --- a/submodules/TelegramCore/Sources/TelegramEngine/Messages/DeleteMessagesInteractively.swift +++ b/submodules/TelegramCore/Sources/TelegramEngine/Messages/DeleteMessagesInteractively.swift @@ -12,35 +12,51 @@ func _internal_deleteMessagesInteractively(account: Account, messageIds: [Messag } func deleteMessagesInteractively(transaction: Transaction, stateManager: AccountStateManager?, postbox: Postbox, messageIds initialMessageIds: [MessageId], type: InteractiveMessagesDeletionType, deleteAllInGroup: Bool = false, removeIfPossiblyDelivered: Bool) { - var messageIds: [MessageId] = [] + var messageIds: [MessageAndThreadId] = [] if deleteAllInGroup { + var tempIds: [MessageId] = initialMessageIds for id in initialMessageIds { if let group = transaction.getMessageGroup(id) ?? transaction.getMessageForwardedGroup(id) { for message in group { - if !messageIds.contains(message.id) { - messageIds.append(message.id) + if !tempIds.contains(message.id) { + tempIds.append(message.id) } } } else { - messageIds.append(id) + tempIds.append(id) + } + } + + messageIds = tempIds.map { id in + if id.namespace == Namespaces.Message.QuickReplyCloud { + if let message = transaction.getMessage(id) { + return MessageAndThreadId(messageId: id, threadId: message.threadId) + } else { + return MessageAndThreadId(messageId: id, threadId: nil) + } + } else { + return MessageAndThreadId(messageId: id, threadId: nil) } } } else { - messageIds = initialMessageIds - } - - var messageIdsByPeerId: [PeerId: [MessageId]] = [:] - for id in messageIds { - if messageIdsByPeerId[id.peerId] == nil { - messageIdsByPeerId[id.peerId] = [id] - } else { - messageIdsByPeerId[id.peerId]!.append(id) + messageIds = initialMessageIds.map { id in + if id.namespace == Namespaces.Message.QuickReplyCloud { + if let message = transaction.getMessage(id) { + return MessageAndThreadId(messageId: id, threadId: message.threadId) + } else { + return MessageAndThreadId(messageId: id, threadId: nil) + } + } else { + return MessageAndThreadId(messageId: id, threadId: nil) + } } } var uniqueIds: [Int64: PeerId] = [:] - for (peerId, peerMessageIds) in messageIdsByPeerId { + for (peerAndThreadId, peerMessageIds) in messagesIdsGroupedByPeerId(messageIds) { + let peerId = peerAndThreadId.peerId + let threadId = peerAndThreadId.threadId for id in peerMessageIds { if let message = transaction.getMessage(id) { for attribute in message.attributes { @@ -53,13 +69,13 @@ func deleteMessagesInteractively(transaction: Transaction, stateManager: Account if peerId.namespace == Namespaces.Peer.CloudChannel || peerId.namespace == Namespaces.Peer.CloudGroup || peerId.namespace == Namespaces.Peer.CloudUser { let remoteMessageIds = peerMessageIds.filter { id in - if id.namespace == Namespaces.Message.Local || id.namespace == Namespaces.Message.ScheduledLocal { + if id.namespace == Namespaces.Message.Local || id.namespace == Namespaces.Message.ScheduledLocal || id.namespace == Namespaces.Message.QuickReplyLocal { return false } return true } if !remoteMessageIds.isEmpty { - cloudChatAddRemoveMessagesOperation(transaction: transaction, peerId: peerId, messageIds: remoteMessageIds, type: CloudChatRemoveMessagesType(type)) + cloudChatAddRemoveMessagesOperation(transaction: transaction, peerId: peerId, threadId: threadId, messageIds: remoteMessageIds, type: CloudChatRemoveMessagesType(type)) } } else if peerId.namespace == Namespaces.Peer.SecretChat { if let state = transaction.getPeerChatState(peerId) as? SecretChatState { @@ -87,9 +103,9 @@ func deleteMessagesInteractively(transaction: Transaction, stateManager: Account } } } - _internal_deleteMessages(transaction: transaction, mediaBox: postbox.mediaBox, ids: messageIds) + _internal_deleteMessages(transaction: transaction, mediaBox: postbox.mediaBox, ids: messageIds.map(\.messageId)) - stateManager?.notifyDeletedMessages(messageIds: messageIds) + stateManager?.notifyDeletedMessages(messageIds: messageIds.map(\.messageId)) if !uniqueIds.isEmpty && removeIfPossiblyDelivered { stateManager?.removePossiblyDeliveredMessages(uniqueIds: uniqueIds) @@ -102,7 +118,7 @@ func _internal_clearHistoryInRangeInteractively(postbox: Postbox, peerId: PeerId cloudChatAddClearHistoryOperation(transaction: transaction, peerId: peerId, threadId: threadId, explicitTopMessageId: nil, minTimestamp: minTimestamp, maxTimestamp: maxTimestamp, type: CloudChatClearHistoryType(type)) if type == .scheduledMessages { } else { - _internal_clearHistoryInRange(transaction: transaction, mediaBox: postbox.mediaBox, peerId: peerId, threadId: threadId, minTimestamp: minTimestamp, maxTimestamp: maxTimestamp, namespaces: .not(Namespaces.Message.allScheduled)) + _internal_clearHistoryInRange(transaction: transaction, mediaBox: postbox.mediaBox, peerId: peerId, threadId: threadId, minTimestamp: minTimestamp, maxTimestamp: maxTimestamp, namespaces: .not(Namespaces.Message.allNonRegular)) } } else if peerId.namespace == Namespaces.Peer.SecretChat { /*_internal_clearHistory(transaction: transaction, mediaBox: postbox.mediaBox, peerId: peerId, namespaces: .all) @@ -141,7 +157,7 @@ func _internal_clearHistoryInteractively(postbox: Postbox, peerId: PeerId, threa topIndex = topMessage.index } - _internal_clearHistory(transaction: transaction, mediaBox: postbox.mediaBox, peerId: peerId, threadId: threadId, namespaces: .not(Namespaces.Message.allScheduled)) + _internal_clearHistory(transaction: transaction, mediaBox: postbox.mediaBox, peerId: peerId, threadId: threadId, namespaces: .not(Namespaces.Message.allNonRegular)) if let cachedData = transaction.getPeerCachedData(peerId: peerId) as? CachedChannelData, let migrationReference = cachedData.migrationReference { cloudChatAddClearHistoryOperation(transaction: transaction, peerId: migrationReference.maxMessageId.peerId, threadId: threadId, explicitTopMessageId: MessageId(peerId: migrationReference.maxMessageId.peerId, namespace: migrationReference.maxMessageId.namespace, id: migrationReference.maxMessageId.id + 1), minTimestamp: nil, maxTimestamp: nil, type: CloudChatClearHistoryType(type)) _internal_clearHistory(transaction: transaction, mediaBox: postbox.mediaBox, peerId: migrationReference.maxMessageId.peerId, threadId: threadId, namespaces: .all) diff --git a/submodules/TelegramCore/Sources/TelegramEngine/Messages/ForwardGame.swift b/submodules/TelegramCore/Sources/TelegramEngine/Messages/ForwardGame.swift index b61557dec8a..f7e5d428338 100644 --- a/submodules/TelegramCore/Sources/TelegramEngine/Messages/ForwardGame.swift +++ b/submodules/TelegramCore/Sources/TelegramEngine/Messages/ForwardGame.swift @@ -14,7 +14,7 @@ func _internal_forwardGameWithScore(account: Account, messageId: MessageId, to p flags |= (1 << 13) } - return account.network.request(Api.functions.messages.forwardMessages(flags: flags, fromPeer: fromInputPeer, id: [messageId.id], randomId: [Int64.random(in: Int64.min ... Int64.max)], toPeer: toInputPeer, topMsgId: threadId.flatMap { Int32(clamping: $0) }, scheduleDate: nil, sendAs: sendAsInputPeer)) + return account.network.request(Api.functions.messages.forwardMessages(flags: flags, fromPeer: fromInputPeer, id: [messageId.id], randomId: [Int64.random(in: Int64.min ... Int64.max)], toPeer: toInputPeer, topMsgId: threadId.flatMap { Int32(clamping: $0) }, scheduleDate: nil, sendAs: sendAsInputPeer, quickReplyShortcut: nil)) |> map(Optional.init) |> `catch` { _ -> Signal in return .single(nil) diff --git a/submodules/TelegramCore/Sources/TelegramEngine/Messages/Polls.swift b/submodules/TelegramCore/Sources/TelegramEngine/Messages/Polls.swift index 1ddb77458b2..d7174c3e951 100644 --- a/submodules/TelegramCore/Sources/TelegramEngine/Messages/Polls.swift +++ b/submodules/TelegramCore/Sources/TelegramEngine/Messages/Polls.swift @@ -135,7 +135,7 @@ func _internal_requestClosePoll(postbox: Postbox, network: Network, stateManager pollMediaFlags |= 1 << 1 } - return network.request(Api.functions.messages.editMessage(flags: flags, peer: inputPeer, id: messageId.id, message: nil, media: .inputMediaPoll(flags: pollMediaFlags, poll: .poll(id: poll.pollId.id, flags: pollFlags, question: poll.text, answers: poll.options.map({ $0.apiOption }), closePeriod: poll.deadlineTimeout, closeDate: nil), correctAnswers: correctAnswers, solution: mappedSolution, solutionEntities: mappedSolutionEntities), replyMarkup: nil, entities: nil, scheduleDate: nil)) + return network.request(Api.functions.messages.editMessage(flags: flags, peer: inputPeer, id: messageId.id, message: nil, media: .inputMediaPoll(flags: pollMediaFlags, poll: .poll(id: poll.pollId.id, flags: pollFlags, question: poll.text, answers: poll.options.map({ $0.apiOption }), closePeriod: poll.deadlineTimeout, closeDate: nil), correctAnswers: correctAnswers, solution: mappedSolution, solutionEntities: mappedSolutionEntities), replyMarkup: nil, entities: nil, scheduleDate: nil, quickReplyShortcutId: nil)) |> map(Optional.init) |> `catch` { _ -> Signal in return .single(nil) diff --git a/submodules/TelegramCore/Sources/TelegramEngine/Messages/QuickReplyMessages.swift b/submodules/TelegramCore/Sources/TelegramEngine/Messages/QuickReplyMessages.swift new file mode 100644 index 00000000000..a7e321c872b --- /dev/null +++ b/submodules/TelegramCore/Sources/TelegramEngine/Messages/QuickReplyMessages.swift @@ -0,0 +1,899 @@ +import Foundation +import Postbox +import SwiftSignalKit +import TelegramApi +import MtProtoKit + +public final class QuickReplyMessageShortcut: Codable, Equatable { + public let id: Int32 + public let shortcut: String + + public init(id: Int32, shortcut: String) { + self.id = id + self.shortcut = shortcut + } + + public static func ==(lhs: QuickReplyMessageShortcut, rhs: QuickReplyMessageShortcut) -> Bool { + if lhs.id != rhs.id { + return false + } + if lhs.shortcut != rhs.shortcut { + return false + } + return true + } + + public init(from decoder: Decoder) throws { + let container = try decoder.container(keyedBy: StringCodingKey.self) + + self.id = try container.decode(Int32.self, forKey: "id") + self.shortcut = try container.decode(String.self, forKey: "shortcut") + } + + public func encode(to encoder: Encoder) throws { + var container = encoder.container(keyedBy: StringCodingKey.self) + + try container.encode(self.id, forKey: "id") + try container.encode(self.shortcut, forKey: "shortcut") + } +} + +struct QuickReplyMessageShortcutsState: Codable, Equatable { + var shortcuts: [QuickReplyMessageShortcut] + + init(shortcuts: [QuickReplyMessageShortcut]) { + self.shortcuts = shortcuts + } +} + +public final class ShortcutMessageList: Equatable { + public final class Item: Equatable { + public let id: Int32? + public let shortcut: String + public let topMessage: EngineMessage + public let totalCount: Int + + public init(id: Int32?, shortcut: String, topMessage: EngineMessage, totalCount: Int) { + self.id = id + self.shortcut = shortcut + self.topMessage = topMessage + self.totalCount = totalCount + } + + public static func ==(lhs: Item, rhs: Item) -> Bool { + if lhs === rhs { + return true + } + if lhs.id != rhs.id { + return false + } + if lhs.shortcut != rhs.shortcut { + return false + } + if lhs.topMessage != rhs.topMessage { + return false + } + if lhs.totalCount != rhs.totalCount { + return false + } + return true + } + } + + public let items: [Item] + public let isLoading: Bool + + public init(items: [Item], isLoading: Bool) { + self.items = items + self.isLoading = isLoading + } + + public static func ==(lhs: ShortcutMessageList, rhs: ShortcutMessageList) -> Bool { + if lhs === rhs { + return true + } + if lhs.items != rhs.items { + return false + } + if lhs.isLoading != rhs.isLoading { + return false + } + return true + } +} + +func _internal_quickReplyMessageShortcutsState(account: Account) -> Signal { + let viewKey: PostboxViewKey = .preferences(keys: Set([PreferencesKeys.shortcutMessages()])) + return account.postbox.combinedView(keys: [viewKey]) + |> map { views -> QuickReplyMessageShortcutsState? in + guard let view = views.views[viewKey] as? PreferencesView else { + return nil + } + guard let value = view.values[PreferencesKeys.shortcutMessages()]?.get(QuickReplyMessageShortcutsState.self) else { + return nil + } + return value + } +} + +func _internal_keepShortcutMessagesUpdated(account: Account) -> Signal { + let updateSignal = _internal_shortcutMessageList(account: account, onlyRemote: true) + |> take(1) + |> mapToSignal { list -> Signal in + var acc: UInt64 = 0 + for item in list.items { + guard let itemId = item.id else { + continue + } + combineInt64Hash(&acc, with: UInt64(itemId)) + combineInt64Hash(&acc, with: md5StringHash(item.shortcut)) + combineInt64Hash(&acc, with: UInt64(item.topMessage.id.id)) + + var editTimestamp: Int32 = 0 + inner: for attribute in item.topMessage.attributes { + if let attribute = attribute as? EditedMessageAttribute { + editTimestamp = attribute.date + break inner + } + } + combineInt64Hash(&acc, with: UInt64(editTimestamp)) + } + + return account.network.request(Api.functions.messages.getQuickReplies(hash: finalizeInt64Hash(acc))) + |> map(Optional.init) + |> `catch` { _ -> Signal in + return .single(nil) + } + |> mapToSignal { result -> Signal in + guard let result else { + return .complete() + } + + return account.postbox.transaction { transaction in + var state = transaction.getPreferencesEntry(key: PreferencesKeys.shortcutMessages())?.get(QuickReplyMessageShortcutsState.self) ?? QuickReplyMessageShortcutsState(shortcuts: []) + switch result { + case let .quickReplies(quickReplies, messages, chats, users): + let previousShortcuts = state.shortcuts + state.shortcuts.removeAll() + + let parsedPeers = AccumulatedPeers(transaction: transaction, chats: chats, users: users) + updatePeers(transaction: transaction, accountPeerId: account.peerId, peers: parsedPeers) + + var storeMessages: [StoreMessage] = [] + + for message in messages { + if let message = StoreMessage(apiMessage: message, accountPeerId: account.peerId, peerIsForum: false) { + storeMessages.append(message) + } + } + let _ = transaction.addMessages(storeMessages, location: .Random) + var topMessageIds: [Int32: Int32] = [:] + + for quickReply in quickReplies { + switch quickReply { + case let .quickReply(shortcutId, shortcut, topMessage, _): + state.shortcuts.append(QuickReplyMessageShortcut( + id: shortcutId, + shortcut: shortcut + )) + topMessageIds[shortcutId] = topMessage + } + } + + if previousShortcuts != state.shortcuts { + for shortcut in previousShortcuts { + if let topMessageId = topMessageIds[shortcut.id] { + //TODO:remove earlier + let _ = topMessageId + } else { + let existingCloudMessages = transaction.getMessagesWithThreadId(peerId: account.peerId, namespace: Namespaces.Message.QuickReplyCloud, threadId: Int64(shortcut.id), from: MessageIndex.lowerBound(peerId: account.peerId, namespace: Namespaces.Message.QuickReplyCloud), includeFrom: false, to: MessageIndex.upperBound(peerId: account.peerId, namespace: Namespaces.Message.QuickReplyCloud), limit: 1000) + let existingLocalMessages = transaction.getMessagesWithThreadId(peerId: account.peerId, namespace: Namespaces.Message.QuickReplyLocal, threadId: Int64(shortcut.id), from: MessageIndex.lowerBound(peerId: account.peerId, namespace: Namespaces.Message.QuickReplyLocal), includeFrom: false, to: MessageIndex.upperBound(peerId: account.peerId, namespace: Namespaces.Message.QuickReplyLocal), limit: 1000) + + transaction.deleteMessages(existingCloudMessages.map(\.id), forEachMedia: nil) + transaction.deleteMessages(existingLocalMessages.map(\.id), forEachMedia: nil) + } + } + } + case .quickRepliesNotModified: + break + } + + transaction.setPreferencesEntry(key: PreferencesKeys.shortcutMessages(), value: PreferencesEntry(state)) + } + |> ignoreValues + } + } + + return updateSignal +} + +func _internal_shortcutMessageList(account: Account, onlyRemote: Bool) -> Signal { + let pendingShortcuts: Signal<[String: EngineMessage], NoError> + if onlyRemote { + pendingShortcuts = .single([:]) + } else { + pendingShortcuts = account.postbox.aroundMessageHistoryViewForLocation(.peer(peerId: account.peerId, threadId: nil), anchor: .upperBound, ignoreMessagesInTimestampRange: nil, count: 100, fixedCombinedReadStates: nil, topTaggedMessageIdNamespaces: Set(), tag: nil, appendMessagesFromTheSameGroup: false, namespaces: .just(Set([Namespaces.Message.QuickReplyLocal])), orderStatistics: []) + |> map { view , _, _ -> [String: EngineMessage] in + var topMessages: [String: EngineMessage] = [:] + for entry in view.entries { + var shortcut: String? + inner: for attribute in entry.message.attributes { + if let attribute = attribute as? OutgoingQuickReplyMessageAttribute { + shortcut = attribute.shortcut + break inner + } + } + if let shortcut { + if let currentTopMessage = topMessages[shortcut] { + if entry.message.index < currentTopMessage.index { + topMessages[shortcut] = EngineMessage(entry.message) + } + } else { + topMessages[shortcut] = EngineMessage(entry.message) + } + } + } + return topMessages + } + |> distinctUntilChanged + } + + return combineLatest(queue: .mainQueue(), + _internal_quickReplyMessageShortcutsState(account: account) |> distinctUntilChanged, + pendingShortcuts + ) + |> mapToSignal { state, pendingShortcuts -> Signal in + guard let state else { + return .single(ShortcutMessageList(items: [], isLoading: true)) + } + + var keys: [PostboxViewKey] = [] + var historyViewKeys: [Int32: PostboxViewKey] = [:] + var summaryKeys: [Int32: PostboxViewKey] = [:] + for shortcut in state.shortcuts { + let historyViewKey: PostboxViewKey = .historyView(PostboxViewKey.HistoryView( + peerId: account.peerId, + threadId: Int64(shortcut.id), + clipHoles: false, + trackHoles: false, + anchor: .lowerBound, + appendMessagesFromTheSameGroup: false, + namespaces: .just(Set([Namespaces.Message.QuickReplyCloud])), + count: 10 + )) + historyViewKeys[shortcut.id] = historyViewKey + keys.append(historyViewKey) + + let summaryKey: PostboxViewKey = .historyTagSummaryView(tag: [], peerId: account.peerId, threadId: Int64(shortcut.id), namespace: Namespaces.Message.QuickReplyCloud, customTag: nil) + summaryKeys[shortcut.id] = summaryKey + keys.append(summaryKey) + } + return account.postbox.combinedView( + keys: keys + ) + |> map { views -> ShortcutMessageList in + var items: [ShortcutMessageList.Item] = [] + + for shortcut in state.shortcuts { + guard let historyViewKey = historyViewKeys[shortcut.id], let historyView = views.views[historyViewKey] as? MessageHistoryView else { + continue + } + + var totalCount = 1 + if let summaryKey = summaryKeys[shortcut.id], let summaryView = views.views[summaryKey] as? MessageHistoryTagSummaryView { + if let count = summaryView.count { + totalCount = max(1, Int(count)) + } + } + + if let entry = historyView.entries.first { + items.append(ShortcutMessageList.Item(id: shortcut.id, shortcut: shortcut.shortcut, topMessage: EngineMessage(entry.message), totalCount: totalCount)) + } + } + + for (shortcut, message) in pendingShortcuts.sorted(by: { $0.key < $1.key }) { + if !items.contains(where: { $0.shortcut == shortcut }) { + items.append(ShortcutMessageList.Item( + id: nil, + shortcut: shortcut, + topMessage: message, + totalCount: 1 + )) + } + } + + return ShortcutMessageList(items: items, isLoading: false) + } + |> distinctUntilChanged + } +} + +func _internal_editMessageShortcut(account: Account, id: Int32, shortcut: String) -> Signal { + let remoteApply = account.network.request(Api.functions.messages.editQuickReplyShortcut(shortcutId: id, shortcut: shortcut)) + |> `catch` { _ -> Signal in + return .single(.boolFalse) + } + |> mapToSignal { _ -> Signal in + return .complete() + } + + return account.postbox.transaction { transaction in + var state = transaction.getPreferencesEntry(key: PreferencesKeys.shortcutMessages())?.get(QuickReplyMessageShortcutsState.self) ?? QuickReplyMessageShortcutsState(shortcuts: []) + if let index = state.shortcuts.firstIndex(where: { $0.id == id }) { + state.shortcuts[index] = QuickReplyMessageShortcut(id: id, shortcut: shortcut) + } + transaction.setPreferencesEntry(key: PreferencesKeys.shortcutMessages(), value: PreferencesEntry(state)) + } + |> ignoreValues + |> then(remoteApply) +} + +func _internal_deleteMessageShortcuts(account: Account, ids: [Int32]) -> Signal { + return account.postbox.transaction { transaction in + var state = transaction.getPreferencesEntry(key: PreferencesKeys.shortcutMessages())?.get(QuickReplyMessageShortcutsState.self) ?? QuickReplyMessageShortcutsState(shortcuts: []) + + for id in ids { + if let index = state.shortcuts.firstIndex(where: { $0.id == id }) { + state.shortcuts.remove(at: index) + } + } + transaction.setPreferencesEntry(key: PreferencesKeys.shortcutMessages(), value: PreferencesEntry(state)) + + for id in ids { + cloudChatAddClearHistoryOperation(transaction: transaction, peerId: account.peerId, threadId: Int64(id), explicitTopMessageId: nil, minTimestamp: nil, maxTimestamp: nil, type: .quickReplyMessages) + } + } + |> ignoreValues +} + +func _internal_reorderMessageShortcuts(account: Account, ids: [Int32], localCompletion: @escaping () -> Void) -> Signal { + let remoteApply = account.network.request(Api.functions.messages.reorderQuickReplies(order: ids)) + |> `catch` { _ -> Signal in + return .single(.boolFalse) + } + |> mapToSignal { _ -> Signal in + return .complete() + } + + return account.postbox.transaction { transaction in + var state = transaction.getPreferencesEntry(key: PreferencesKeys.shortcutMessages())?.get(QuickReplyMessageShortcutsState.self) ?? QuickReplyMessageShortcutsState(shortcuts: []) + + let previousShortcuts = state.shortcuts + state.shortcuts.removeAll() + for id in ids { + if let index = previousShortcuts.firstIndex(where: { $0.id == id }) { + state.shortcuts.append(previousShortcuts[index]) + } + } + for shortcut in previousShortcuts { + if !state.shortcuts.contains(where: { $0.id == shortcut.id }) { + state.shortcuts.append(shortcut) + } + } + + transaction.setPreferencesEntry(key: PreferencesKeys.shortcutMessages(), value: PreferencesEntry(state)) + } + |> ignoreValues + |> afterCompleted { + localCompletion() + } + |> then(remoteApply) +} + +func _internal_sendMessageShortcut(account: Account, peerId: PeerId, id: Int32) -> Signal { + return account.postbox.transaction { transaction -> Peer? in + return transaction.getPeer(peerId) + } + |> mapToSignal { peer -> Signal in + guard let peer, let inputPeer = apiInputPeer(peer) else { + return .complete() + } + return account.network.request(Api.functions.messages.sendQuickReplyMessages(peer: inputPeer, shortcutId: id)) + |> map(Optional.init) + |> `catch` { _ -> Signal in + return .single(nil) + } + |> mapToSignal { result -> Signal in + if let result { + account.stateManager.addUpdates(result) + } + return .complete() + } + } +} + +func _internal_applySentQuickReplyMessage(transaction: Transaction, shortcut: String, quickReplyId: Int32) { + var state = transaction.getPreferencesEntry(key: PreferencesKeys.shortcutMessages())?.get(QuickReplyMessageShortcutsState.self) ?? QuickReplyMessageShortcutsState(shortcuts: []) + + if !state.shortcuts.contains(where: { $0.id == quickReplyId }) { + state.shortcuts.append(QuickReplyMessageShortcut(id: quickReplyId, shortcut: shortcut)) + transaction.setPreferencesEntry(key: PreferencesKeys.shortcutMessages(), value: PreferencesEntry(state)) + } +} + +public final class TelegramBusinessRecipients: Codable, Equatable { + public struct Categories: OptionSet { + public var rawValue: Int32 + + public init(rawValue: Int32) { + self.rawValue = rawValue + } + + public static let existingChats = Categories(rawValue: 1 << 0) + public static let newChats = Categories(rawValue: 1 << 1) + public static let contacts = Categories(rawValue: 1 << 2) + public static let nonContacts = Categories(rawValue: 1 << 3) + } + + private enum CodingKeys: String, CodingKey { + case categories + case additionalPeers + case exclude + } + + public let categories: Categories + public let additionalPeers: Set + public let exclude: Bool + + public init(categories: Categories, additionalPeers: Set, exclude: Bool) { + self.categories = categories + self.additionalPeers = additionalPeers + self.exclude = exclude + } + + public init(from decoder: Decoder) throws { + let container = try decoder.container(keyedBy: CodingKeys.self) + + self.categories = Categories(rawValue: try container.decode(Int32.self, forKey: .categories)) + self.additionalPeers = Set(try container.decode([PeerId].self, forKey: .additionalPeers)) + self.exclude = try container.decode(Bool.self, forKey: .exclude) + } + + public func encode(to encoder: Encoder) throws { + var container = encoder.container(keyedBy: CodingKeys.self) + + try container.encode(self.categories.rawValue, forKey: .categories) + try container.encode(Array(self.additionalPeers).sorted(), forKey: .additionalPeers) + try container.encode(self.exclude, forKey: .exclude) + } + + public static func ==(lhs: TelegramBusinessRecipients, rhs: TelegramBusinessRecipients) -> Bool { + if lhs === rhs { + return true + } + + if lhs.categories != rhs.categories { + return false + } + if lhs.additionalPeers != rhs.additionalPeers { + return false + } + if lhs.exclude != rhs.exclude { + return false + } + + return true + } +} + +public final class TelegramBusinessGreetingMessage: Codable, Equatable { + private enum CodingKeys: String, CodingKey { + case shortcutId + case recipients + case inactivityDays + } + + public let shortcutId: Int32 + public let recipients: TelegramBusinessRecipients + public let inactivityDays: Int + + public init(shortcutId: Int32, recipients: TelegramBusinessRecipients, inactivityDays: Int) { + self.shortcutId = shortcutId + self.recipients = recipients + self.inactivityDays = inactivityDays + } + + public init(from decoder: Decoder) throws { + let container = try decoder.container(keyedBy: CodingKeys.self) + + self.shortcutId = try container.decode(Int32.self, forKey: .shortcutId) + self.recipients = try container.decode(TelegramBusinessRecipients.self, forKey: .recipients) + self.inactivityDays = Int(try container.decode(Int32.self, forKey: .inactivityDays)) + } + + public func encode(to encoder: Encoder) throws { + var container = encoder.container(keyedBy: CodingKeys.self) + + try container.encode(self.shortcutId, forKey: .shortcutId) + try container.encode(self.recipients, forKey: .recipients) + try container.encode(Int32(clamping: self.inactivityDays), forKey: .inactivityDays) + } + + public static func ==(lhs: TelegramBusinessGreetingMessage, rhs: TelegramBusinessGreetingMessage) -> Bool { + if lhs === rhs { + return true + } + + if lhs.shortcutId != rhs.shortcutId { + return false + } + if lhs.recipients != rhs.recipients { + return false + } + if lhs.inactivityDays != rhs.inactivityDays { + return false + } + + return true + } +} + +extension TelegramBusinessGreetingMessage { + convenience init(apiGreetingMessage: Api.BusinessGreetingMessage) { + switch apiGreetingMessage { + case let .businessGreetingMessage(shortcutId, recipients, noActivityDays): + self.init( + shortcutId: shortcutId, + recipients: TelegramBusinessRecipients(apiValue: recipients), + inactivityDays: Int(noActivityDays) + ) + } + } +} + +public final class TelegramBusinessAwayMessage: Codable, Equatable { + public enum Schedule: Codable, Equatable { + private enum CodingKeys: String, CodingKey { + case discriminator + case customBeginTimestamp + case customEndTimestamp + } + + case always + case outsideWorkingHours + case custom(beginTimestamp: Int32, endTimestamp: Int32) + + public init(from decoder: Decoder) throws { + let container = try decoder.container(keyedBy: CodingKeys.self) + + switch try container.decode(Int32.self, forKey: .discriminator) { + case 0: + self = .always + case 1: + self = .outsideWorkingHours + case 2: + self = .custom(beginTimestamp: try container.decode(Int32.self, forKey: .customBeginTimestamp), endTimestamp: try container.decode(Int32.self, forKey: .customEndTimestamp)) + default: + self = .always + } + } + + public func encode(to encoder: Encoder) throws { + var container = encoder.container(keyedBy: CodingKeys.self) + + switch self { + case .always: + try container.encode(0 as Int32, forKey: .discriminator) + case .outsideWorkingHours: + try container.encode(1 as Int32, forKey: .discriminator) + case let .custom(beginTimestamp, endTimestamp): + try container.encode(2 as Int32, forKey: .discriminator) + try container.encode(beginTimestamp, forKey: .customBeginTimestamp) + try container.encode(endTimestamp, forKey: .customEndTimestamp) + } + } + } + + private enum CodingKeys: String, CodingKey { + case shortcutId + case recipients + case schedule + case sendWhenOffline + } + + public let shortcutId: Int32 + public let recipients: TelegramBusinessRecipients + public let schedule: Schedule + public let sendWhenOffline: Bool + + public init(shortcutId: Int32, recipients: TelegramBusinessRecipients, schedule: Schedule, sendWhenOffline: Bool) { + self.shortcutId = shortcutId + self.recipients = recipients + self.schedule = schedule + self.sendWhenOffline = sendWhenOffline + } + + public init(from decoder: Decoder) throws { + let container = try decoder.container(keyedBy: CodingKeys.self) + + self.shortcutId = try container.decode(Int32.self, forKey: .shortcutId) + self.recipients = try container.decode(TelegramBusinessRecipients.self, forKey: .recipients) + self.schedule = try container.decode(Schedule.self, forKey: .schedule) + self.sendWhenOffline = try container.decodeIfPresent(Bool.self, forKey: .sendWhenOffline) ?? false + } + + public func encode(to encoder: Encoder) throws { + var container = encoder.container(keyedBy: CodingKeys.self) + + try container.encode(self.shortcutId, forKey: .shortcutId) + try container.encode(self.recipients, forKey: .recipients) + try container.encode(self.schedule, forKey: .schedule) + try container.encode(self.sendWhenOffline, forKey: .sendWhenOffline) + } + + public static func ==(lhs: TelegramBusinessAwayMessage, rhs: TelegramBusinessAwayMessage) -> Bool { + if lhs === rhs { + return true + } + + if lhs.shortcutId != rhs.shortcutId { + return false + } + if lhs.recipients != rhs.recipients { + return false + } + if lhs.schedule != rhs.schedule { + return false + } + if lhs.sendWhenOffline != rhs.sendWhenOffline { + return false + } + + return true + } +} + +extension TelegramBusinessAwayMessage { + convenience init(apiAwayMessage: Api.BusinessAwayMessage) { + switch apiAwayMessage { + case let .businessAwayMessage(flags, shortcutId, schedule, recipients): + let mappedSchedule: Schedule + switch schedule { + case .businessAwayMessageScheduleAlways: + mappedSchedule = .always + case .businessAwayMessageScheduleOutsideWorkHours: + mappedSchedule = .outsideWorkingHours + case let .businessAwayMessageScheduleCustom(startDate, endDate): + mappedSchedule = .custom(beginTimestamp: startDate, endTimestamp: endDate) + } + + let sendWhenOffline = (flags & (1 << 0)) != 0 + + self.init( + shortcutId: shortcutId, + recipients: TelegramBusinessRecipients(apiValue: recipients), + schedule: mappedSchedule, + sendWhenOffline: sendWhenOffline + ) + } + } +} + +extension TelegramBusinessRecipients { + convenience init(apiValue: Api.BusinessRecipients) { + switch apiValue { + case let .businessRecipients(flags, users): + var categories: Categories = [] + if (flags & (1 << 0)) != 0 { + categories.insert(.existingChats) + } + if (flags & (1 << 1)) != 0 { + categories.insert(.newChats) + } + if (flags & (1 << 2)) != 0 { + categories.insert(.contacts) + } + if (flags & (1 << 3)) != 0 { + categories.insert(.nonContacts) + } + + self.init( + categories: categories, + additionalPeers: Set((users ?? []).map(PeerId.init)), + exclude: (flags & (1 << 5)) != 0 + ) + } + } + + func apiInputValue(additionalPeers: [Peer]) -> Api.InputBusinessRecipients { + var users: [Api.InputUser]? + if !additionalPeers.isEmpty { + users = additionalPeers.compactMap(apiInputUser) + } + + var flags: Int32 = 0 + + if self.categories.contains(.existingChats) { + flags |= 1 << 0 + } + if self.categories.contains(.newChats) { + flags |= 1 << 1 + } + if self.categories.contains(.contacts) { + flags |= 1 << 2 + } + if self.categories.contains(.nonContacts) { + flags |= 1 << 3 + } + if self.exclude { + flags |= 1 << 5 + } + if users != nil { + flags |= 1 << 4 + } + + return .inputBusinessRecipients(flags: flags, users: users) + } +} + +func _internal_updateBusinessGreetingMessage(account: Account, greetingMessage: TelegramBusinessGreetingMessage?) -> Signal { + let remoteApply = account.postbox.transaction { transaction -> [Peer] in + guard let greetingMessage else { + return [] + } + return greetingMessage.recipients.additionalPeers.compactMap(transaction.getPeer) + } + |> mapToSignal { additionalPeers in + var mappedMessage: Api.InputBusinessGreetingMessage? + if let greetingMessage { + mappedMessage = .inputBusinessGreetingMessage( + shortcutId: greetingMessage.shortcutId, + recipients: greetingMessage.recipients.apiInputValue(additionalPeers: additionalPeers), + noActivityDays: Int32(clamping: greetingMessage.inactivityDays) + ) + } + + var flags: Int32 = 0 + if mappedMessage != nil { + flags |= 1 << 0 + } + + return account.network.request(Api.functions.account.updateBusinessGreetingMessage(flags: flags, message: mappedMessage)) + |> `catch` { _ -> Signal in + return .single(.boolFalse) + } + |> mapToSignal { _ -> Signal in + return .complete() + } + } + + return account.postbox.transaction { transaction in + transaction.updatePeerCachedData(peerIds: Set([account.peerId]), update: { _, current in + var current = (current as? CachedUserData) ?? CachedUserData() + current = current.withUpdatedGreetingMessage(greetingMessage) + return current + }) + } + |> ignoreValues + |> then(remoteApply) +} + +func _internal_updateBusinessAwayMessage(account: Account, awayMessage: TelegramBusinessAwayMessage?) -> Signal { + let remoteApply = account.postbox.transaction { transaction -> [Peer] in + guard let awayMessage else { + return [] + } + return awayMessage.recipients.additionalPeers.compactMap(transaction.getPeer) + } + |> mapToSignal { additionalPeers in + var mappedMessage: Api.InputBusinessAwayMessage? + if let awayMessage { + let mappedSchedule: Api.BusinessAwayMessageSchedule + switch awayMessage.schedule { + case .always: + mappedSchedule = .businessAwayMessageScheduleAlways + case .outsideWorkingHours: + mappedSchedule = .businessAwayMessageScheduleOutsideWorkHours + case let .custom(beginTimestamp, endTimestamp): + mappedSchedule = .businessAwayMessageScheduleCustom(startDate: beginTimestamp, endDate: endTimestamp) + } + + var flags: Int32 = 0 + if awayMessage.sendWhenOffline { + flags |= 1 << 0 + } + + mappedMessage = .inputBusinessAwayMessage( + flags: flags, + shortcutId: awayMessage.shortcutId, + schedule: mappedSchedule, + recipients: awayMessage.recipients.apiInputValue(additionalPeers: additionalPeers) + ) + } + + var flags: Int32 = 0 + if mappedMessage != nil { + flags |= 1 << 0 + } + + return account.network.request(Api.functions.account.updateBusinessAwayMessage(flags: flags, message: mappedMessage)) + |> `catch` { _ -> Signal in + return .single(.boolFalse) + } + |> mapToSignal { _ -> Signal in + return .complete() + } + } + + return account.postbox.transaction { transaction in + transaction.updatePeerCachedData(peerIds: Set([account.peerId]), update: { _, current in + var current = (current as? CachedUserData) ?? CachedUserData() + current = current.withUpdatedAwayMessage(awayMessage) + return current + }) + } + |> ignoreValues + |> then(remoteApply) +} + +public final class TelegramAccountConnectedBot: Codable, Equatable { + public let id: PeerId + public let recipients: TelegramBusinessRecipients + public let canReply: Bool + + public init(id: PeerId, recipients: TelegramBusinessRecipients, canReply: Bool) { + self.id = id + self.recipients = recipients + self.canReply = canReply + } + + public static func ==(lhs: TelegramAccountConnectedBot, rhs: TelegramAccountConnectedBot) -> Bool { + if lhs === rhs { + return true + } + if lhs.id != rhs.id { + return false + } + if lhs.recipients != rhs.recipients { + return false + } + if lhs.canReply != rhs.canReply { + return false + } + return true + } +} + +public func _internal_setAccountConnectedBot(account: Account, bot: TelegramAccountConnectedBot?) -> Signal { + let remoteApply = account.postbox.transaction { transaction -> (Peer?, [Peer]) in + guard let bot else { + return (nil, []) + } + return (transaction.getPeer(bot.id), bot.recipients.additionalPeers.compactMap(transaction.getPeer)) + } + |> mapToSignal { botUser, additionalPeers in + var flags: Int32 = 0 + var mappedBot: Api.InputUser = .inputUserEmpty + var mappedRecipients: Api.InputBusinessRecipients = .inputBusinessRecipients(flags: 0, users: nil) + + if let bot, let inputBotUser = botUser.flatMap(apiInputUser) { + mappedBot = inputBotUser + if bot.canReply { + flags |= 1 << 0 + } + mappedRecipients = bot.recipients.apiInputValue(additionalPeers: additionalPeers) + } + + return account.network.request(Api.functions.account.updateConnectedBot(flags: flags, bot: mappedBot, recipients: mappedRecipients)) + |> map(Optional.init) + |> `catch` { _ -> Signal in + return .single(nil) + } + |> mapToSignal { result -> Signal in + if let result { + account.stateManager.addUpdates(result) + } + return .complete() + } + } + + return account.postbox.transaction { transaction in + transaction.updatePeerCachedData(peerIds: Set([account.peerId]), update: { _, current in + var current = (current as? CachedUserData) ?? CachedUserData() + current = current.withUpdatedConnectedBot(bot) + return current + }) + } + |> ignoreValues + |> then(remoteApply) +} diff --git a/submodules/TelegramCore/Sources/TelegramEngine/Messages/ReplyThreadHistory.swift b/submodules/TelegramCore/Sources/TelegramEngine/Messages/ReplyThreadHistory.swift index 0695e05c0c6..4c57ab1f131 100644 --- a/submodules/TelegramCore/Sources/TelegramEngine/Messages/ReplyThreadHistory.swift +++ b/submodules/TelegramCore/Sources/TelegramEngine/Messages/ReplyThreadHistory.swift @@ -843,7 +843,7 @@ func _internal_fetchChannelReplyThreadMessage(account: Account, messageId: Messa count: 40, clipHoles: true, anchor: inputAnchor, - namespaces: .not(Namespaces.Message.allScheduled) + namespaces: .not(Namespaces.Message.allNonRegular) ) if !testView.isLoading || transaction.getMessageHistoryThreadInfo(peerId: threadMessageId.peerId, threadId: Int64(threadMessageId.id)) != nil { let initialAnchor: ChatReplyThreadMessage.Anchor diff --git a/submodules/TelegramCore/Sources/TelegramEngine/Messages/SearchMessages.swift b/submodules/TelegramCore/Sources/TelegramEngine/Messages/SearchMessages.swift index 409791f28da..d64c77acc51 100644 --- a/submodules/TelegramCore/Sources/TelegramEngine/Messages/SearchMessages.swift +++ b/submodules/TelegramCore/Sources/TelegramEngine/Messages/SearchMessages.swift @@ -584,12 +584,18 @@ func _internal_downloadMessage(accountPeerId: PeerId, postbox: Postbox, network: } func fetchRemoteMessage(accountPeerId: PeerId, postbox: Postbox, source: FetchMessageHistoryHoleSource, message: MessageReference) -> Signal { - guard case let .message(peer, _, id, _, _, _) = message.content else { + guard case let .message(peer, _, id, _, _, _, threadId) = message.content else { return .single(nil) } let signal: Signal if id.namespace == Namespaces.Message.ScheduledCloud { signal = source.request(Api.functions.messages.getScheduledMessages(peer: peer.inputPeer, id: [id.id])) + } else if id.namespace == Namespaces.Message.QuickReplyCloud { + if let threadId { + signal = source.request(Api.functions.messages.getQuickReplyMessages(flags: 1 << 0, shortcutId: Int32(clamping: threadId), id: [id.id], hash: 0)) + } else { + signal = .never() + } } else if id.peerId.namespace == Namespaces.Peer.CloudChannel { if let channel = peer.inputChannel { signal = source.request(Api.functions.channels.getMessages(channel: channel, id: [Api.InputMessage.inputMessageID(id: id.id)])) diff --git a/submodules/TelegramCore/Sources/TelegramEngine/Messages/SparseMessageList.swift b/submodules/TelegramCore/Sources/TelegramEngine/Messages/SparseMessageList.swift index 5f39622ea4e..f6f258c1546 100644 --- a/submodules/TelegramCore/Sources/TelegramEngine/Messages/SparseMessageList.swift +++ b/submodules/TelegramCore/Sources/TelegramEngine/Messages/SparseMessageList.swift @@ -192,7 +192,7 @@ public final class SparseMessageList { let location: ChatLocationInput = .peer(peerId: self.peerId, threadId: self.threadId) - self.topItemsDisposable.set((self.account.postbox.aroundMessageHistoryViewForLocation(location, anchor: .upperBound, ignoreMessagesInTimestampRange: nil, count: count, fixedCombinedReadStates: nil, topTaggedMessageIdNamespaces: Set(), tag: .tag(self.messageTag), appendMessagesFromTheSameGroup: false, namespaces: .not(Set(Namespaces.Message.allScheduled)), orderStatistics: []) + self.topItemsDisposable.set((self.account.postbox.aroundMessageHistoryViewForLocation(location, anchor: .upperBound, ignoreMessagesInTimestampRange: nil, count: count, fixedCombinedReadStates: nil, topTaggedMessageIdNamespaces: Set(), tag: .tag(self.messageTag), appendMessagesFromTheSameGroup: false, namespaces: .not(Namespaces.Message.allNonRegular), orderStatistics: []) |> deliverOn(self.queue)).start(next: { [weak self] view, updateType, _ in guard let strongSelf = self else { return diff --git a/submodules/TelegramCore/Sources/TelegramEngine/Messages/TelegramEngineMessages.swift b/submodules/TelegramCore/Sources/TelegramEngine/Messages/TelegramEngineMessages.swift index e0334d215c3..8decbe638dc 100644 --- a/submodules/TelegramCore/Sources/TelegramEngine/Messages/TelegramEngineMessages.swift +++ b/submodules/TelegramCore/Sources/TelegramEngine/Messages/TelegramEngineMessages.swift @@ -1205,7 +1205,7 @@ public extension TelegramEngine { } var selectedMedia: EngineMedia - if let alternativeMedia = itemAndPeer.item.alternativeMedia.flatMap(EngineMedia.init), !preferHighQuality { + if let alternativeMedia = itemAndPeer.item.alternativeMedia.flatMap(EngineMedia.init), (!preferHighQuality && !itemAndPeer.item.isMy) { selectedMedia = alternativeMedia } else { selectedMedia = EngineMedia(media) diff --git a/submodules/TelegramCore/Sources/TelegramEngine/Messages/TimeZones.swift b/submodules/TelegramCore/Sources/TelegramEngine/Messages/TimeZones.swift new file mode 100644 index 00000000000..5f231500db6 --- /dev/null +++ b/submodules/TelegramCore/Sources/TelegramEngine/Messages/TimeZones.swift @@ -0,0 +1,106 @@ +import Foundation +import Postbox +import SwiftSignalKit +import TelegramApi +import MtProtoKit + +public final class TimeZoneList: Codable, Equatable { + public final class Item: Codable, Equatable { + public let id: String + public let title: String + public let utcOffset: Int32 + + public init(id: String, title: String, utcOffset: Int32) { + self.id = id + self.title = title + self.utcOffset = utcOffset + } + + public static func ==(lhs: Item, rhs: Item) -> Bool { + if lhs === rhs { + return true + } + if lhs.id != rhs.id { + return false + } + if lhs.title != rhs.title { + return false + } + if lhs.utcOffset != rhs.utcOffset { + return false + } + return true + } + } + + public let items: [Item] + public let hashValue: Int32 + + public init(items: [Item], hashValue: Int32) { + self.items = items + self.hashValue = hashValue + } + + public static func ==(lhs: TimeZoneList, rhs: TimeZoneList) -> Bool { + if lhs === rhs { + return true + } + if lhs.items != rhs.items { + return false + } + if lhs.hashValue != rhs.hashValue { + return false + } + return true + } +} + +func _internal_cachedTimeZoneList(account: Account) -> Signal { + let viewKey: PostboxViewKey = .preferences(keys: Set([PreferencesKeys.timezoneList()])) + return account.postbox.combinedView(keys: [viewKey]) + |> map { views -> TimeZoneList? in + guard let view = views.views[viewKey] as? PreferencesView else { + return nil + } + guard let value = view.values[PreferencesKeys.timezoneList()]?.get(TimeZoneList.self) else { + return nil + } + return value + } +} + +func _internal_keepCachedTimeZoneListUpdated(account: Account) -> Signal { + let updateSignal = _internal_cachedTimeZoneList(account: account) + |> take(1) + |> mapToSignal { list -> Signal in + return account.network.request(Api.functions.help.getTimezonesList(hash: list?.hashValue ?? 0)) + |> map(Optional.init) + |> `catch` { _ -> Signal in + return .single(nil) + } + |> mapToSignal { result -> Signal in + guard let result else { + return .complete() + } + + return account.postbox.transaction { transaction in + switch result { + case let .timezonesList(timezones, hash): + var items: [TimeZoneList.Item] = [] + for item in timezones { + switch item { + case let .timezone(id, name, utcOffset): + items.append(TimeZoneList.Item(id: id, title: name, utcOffset: utcOffset)) + } + } + transaction.setPreferencesEntry(key: PreferencesKeys.timezoneList(), value: PreferencesEntry(TimeZoneList(items: items, hashValue: hash))) + case .timezonesListNotModified: + break + } + } + |> ignoreValues + } + } + + return updateSignal +} diff --git a/submodules/TelegramCore/Sources/TelegramEngine/Peers/ChatListFiltering.swift b/submodules/TelegramCore/Sources/TelegramEngine/Peers/ChatListFiltering.swift index cec7854829c..af88b9dffa1 100644 --- a/submodules/TelegramCore/Sources/TelegramEngine/Peers/ChatListFiltering.swift +++ b/submodules/TelegramCore/Sources/TelegramEngine/Peers/ChatListFiltering.swift @@ -306,7 +306,7 @@ extension ChatListFilter { switch apiFilter { case .dialogFilterDefault: self = .allChats - case let .dialogFilter(flags, id, title, emoticon, pinnedPeers, includePeers, excludePeers): + case let .dialogFilter(flags, id, title, emoticon, color, pinnedPeers, includePeers, excludePeers): self = .filter( id: id, title: title, @@ -353,10 +353,10 @@ extension ChatListFilter { return nil } }, - color: nil + color: color.flatMap(PeerNameColor.init(rawValue:)) ) ) - case let .dialogFilterChatlist(flags, id, title, emoticon, pinnedPeers, includePeers): + case let .dialogFilterChatlist(flags, id, title, emoticon, color, pinnedPeers, includePeers): self = .filter( id: id, title: title, @@ -392,7 +392,7 @@ extension ChatListFilter { } }), excludePeers: [], - color: nil + color: color.flatMap(PeerNameColor.init(rawValue:)) ) ) } @@ -408,7 +408,10 @@ extension ChatListFilter { if emoticon != nil { flags |= 1 << 25 } - return .dialogFilterChatlist(flags: flags, id: id, title: title, emoticon: emoticon, pinnedPeers: data.includePeers.pinnedPeers.compactMap { peerId -> Api.InputPeer? in + if data.color != nil { + flags |= 1 << 27 + } + return .dialogFilterChatlist(flags: flags, id: id, title: title, emoticon: emoticon, color: data.color?.rawValue, pinnedPeers: data.includePeers.pinnedPeers.compactMap { peerId -> Api.InputPeer? in return transaction.getPeer(peerId).flatMap(apiInputPeer) }, includePeers: data.includePeers.peers.compactMap { peerId -> Api.InputPeer? in if data.includePeers.pinnedPeers.contains(peerId) { @@ -431,7 +434,10 @@ extension ChatListFilter { if emoticon != nil { flags |= 1 << 25 } - return .dialogFilter(flags: flags, id: id, title: title, emoticon: emoticon, pinnedPeers: data.includePeers.pinnedPeers.compactMap { peerId -> Api.InputPeer? in + if data.color != nil { + flags |= 1 << 27 + } + return .dialogFilter(flags: flags, id: id, title: title, emoticon: emoticon, color: data.color?.rawValue, pinnedPeers: data.includePeers.pinnedPeers.compactMap { peerId -> Api.InputPeer? in return transaction.getPeer(peerId).flatMap(apiInputPeer) }, includePeers: data.includePeers.peers.compactMap { peerId -> Api.InputPeer? in if data.includePeers.pinnedPeers.contains(peerId) { @@ -488,111 +494,116 @@ private enum RequestChatListFiltersError { case generic } -private func requestChatListFilters(accountPeerId: PeerId, postbox: Postbox, network: Network) -> Signal<[ChatListFilter], RequestChatListFiltersError> { +private func requestChatListFilters(accountPeerId: PeerId, postbox: Postbox, network: Network) -> Signal<([ChatListFilter], Bool), RequestChatListFiltersError> { return network.request(Api.functions.messages.getDialogFilters()) |> mapError { _ -> RequestChatListFiltersError in return .generic } - |> mapToSignal { result -> Signal<[ChatListFilter], RequestChatListFiltersError> in - return postbox.transaction { transaction -> ([ChatListFilter], [Api.InputPeer], [Api.InputPeer]) in - var filters: [ChatListFilter] = [] - var missingPeers: [Api.InputPeer] = [] - var missingChats: [Api.InputPeer] = [] - var missingPeerIds = Set() - var missingChatIds = Set() - for apiFilter in result { - let filter = ChatListFilter(apiFilter: apiFilter) - filters.append(filter) - switch apiFilter { - case .dialogFilterDefault: - break - case let .dialogFilter(_, _, _, _, pinnedPeers, includePeers, excludePeers): - for peer in pinnedPeers + includePeers + excludePeers { - var peerId: PeerId? - switch peer { - case let .inputPeerUser(userId, _): - peerId = PeerId(namespace: Namespaces.Peer.CloudUser, id: PeerId.Id._internalFromInt64Value(userId)) - case let .inputPeerChat(chatId): - peerId = PeerId(namespace: Namespaces.Peer.CloudGroup, id: PeerId.Id._internalFromInt64Value(chatId)) - case let .inputPeerChannel(channelId, _): - peerId = PeerId(namespace: Namespaces.Peer.CloudChannel, id: PeerId.Id._internalFromInt64Value(channelId)) - default: - break - } - if let peerId = peerId { - if transaction.getPeer(peerId) == nil && !missingPeerIds.contains(peerId) { - missingPeerIds.insert(peerId) - missingPeers.append(peer) + |> mapToSignal { result -> Signal<([ChatListFilter], Bool), RequestChatListFiltersError> in + return postbox.transaction { transaction -> ([ChatListFilter], [Api.InputPeer], [Api.InputPeer], Bool) in + switch result { + case let .dialogFilters(flags, apiFilters): + let tagsEnabled = (flags & (1 << 0)) != 0 + + var filters: [ChatListFilter] = [] + var missingPeers: [Api.InputPeer] = [] + var missingChats: [Api.InputPeer] = [] + var missingPeerIds = Set() + var missingChatIds = Set() + for apiFilter in apiFilters { + let filter = ChatListFilter(apiFilter: apiFilter) + filters.append(filter) + switch apiFilter { + case .dialogFilterDefault: + break + case let .dialogFilter(_, _, _, _, _, pinnedPeers, includePeers, excludePeers): + for peer in pinnedPeers + includePeers + excludePeers { + var peerId: PeerId? + switch peer { + case let .inputPeerUser(userId, _): + peerId = PeerId(namespace: Namespaces.Peer.CloudUser, id: PeerId.Id._internalFromInt64Value(userId)) + case let .inputPeerChat(chatId): + peerId = PeerId(namespace: Namespaces.Peer.CloudGroup, id: PeerId.Id._internalFromInt64Value(chatId)) + case let .inputPeerChannel(channelId, _): + peerId = PeerId(namespace: Namespaces.Peer.CloudChannel, id: PeerId.Id._internalFromInt64Value(channelId)) + default: + break } - } - } - - for peer in pinnedPeers { - var peerId: PeerId? - switch peer { - case let .inputPeerUser(userId, _): - peerId = PeerId(namespace: Namespaces.Peer.CloudUser, id: PeerId.Id._internalFromInt64Value(userId)) - case let .inputPeerChat(chatId): - peerId = PeerId(namespace: Namespaces.Peer.CloudGroup, id: PeerId.Id._internalFromInt64Value(chatId)) - case let .inputPeerChannel(channelId, _): - peerId = PeerId(namespace: Namespaces.Peer.CloudChannel, id: PeerId.Id._internalFromInt64Value(channelId)) - default: - break - } - if let peerId = peerId, !missingChatIds.contains(peerId) { - if transaction.getPeerChatListIndex(peerId) == nil { - missingChatIds.insert(peerId) - missingChats.append(peer) + if let peerId = peerId { + if transaction.getPeer(peerId) == nil && !missingPeerIds.contains(peerId) { + missingPeerIds.insert(peerId) + missingPeers.append(peer) + } } } - } - case let .dialogFilterChatlist(_, _, _, _, pinnedPeers, includePeers): - for peer in pinnedPeers + includePeers { - var peerId: PeerId? - switch peer { - case let .inputPeerUser(userId, _): - peerId = PeerId(namespace: Namespaces.Peer.CloudUser, id: PeerId.Id._internalFromInt64Value(userId)) - case let .inputPeerChat(chatId): - peerId = PeerId(namespace: Namespaces.Peer.CloudGroup, id: PeerId.Id._internalFromInt64Value(chatId)) - case let .inputPeerChannel(channelId, _): - peerId = PeerId(namespace: Namespaces.Peer.CloudChannel, id: PeerId.Id._internalFromInt64Value(channelId)) - default: - break - } - if let peerId = peerId { - if transaction.getPeer(peerId) == nil && !missingPeerIds.contains(peerId) { - missingPeerIds.insert(peerId) - missingPeers.append(peer) + + for peer in pinnedPeers { + var peerId: PeerId? + switch peer { + case let .inputPeerUser(userId, _): + peerId = PeerId(namespace: Namespaces.Peer.CloudUser, id: PeerId.Id._internalFromInt64Value(userId)) + case let .inputPeerChat(chatId): + peerId = PeerId(namespace: Namespaces.Peer.CloudGroup, id: PeerId.Id._internalFromInt64Value(chatId)) + case let .inputPeerChannel(channelId, _): + peerId = PeerId(namespace: Namespaces.Peer.CloudChannel, id: PeerId.Id._internalFromInt64Value(channelId)) + default: + break + } + if let peerId = peerId, !missingChatIds.contains(peerId) { + if transaction.getPeerChatListIndex(peerId) == nil { + missingChatIds.insert(peerId) + missingChats.append(peer) + } } } - } - - for peer in pinnedPeers { - var peerId: PeerId? - switch peer { - case let .inputPeerUser(userId, _): - peerId = PeerId(namespace: Namespaces.Peer.CloudUser, id: PeerId.Id._internalFromInt64Value(userId)) - case let .inputPeerChat(chatId): - peerId = PeerId(namespace: Namespaces.Peer.CloudGroup, id: PeerId.Id._internalFromInt64Value(chatId)) - case let .inputPeerChannel(channelId, _): - peerId = PeerId(namespace: Namespaces.Peer.CloudChannel, id: PeerId.Id._internalFromInt64Value(channelId)) - default: - break + case let .dialogFilterChatlist(_, _, _, _, _, pinnedPeers, includePeers): + for peer in pinnedPeers + includePeers { + var peerId: PeerId? + switch peer { + case let .inputPeerUser(userId, _): + peerId = PeerId(namespace: Namespaces.Peer.CloudUser, id: PeerId.Id._internalFromInt64Value(userId)) + case let .inputPeerChat(chatId): + peerId = PeerId(namespace: Namespaces.Peer.CloudGroup, id: PeerId.Id._internalFromInt64Value(chatId)) + case let .inputPeerChannel(channelId, _): + peerId = PeerId(namespace: Namespaces.Peer.CloudChannel, id: PeerId.Id._internalFromInt64Value(channelId)) + default: + break + } + if let peerId = peerId { + if transaction.getPeer(peerId) == nil && !missingPeerIds.contains(peerId) { + missingPeerIds.insert(peerId) + missingPeers.append(peer) + } + } } - if let peerId = peerId, !missingChatIds.contains(peerId) { - if transaction.getPeerChatListIndex(peerId) == nil { - missingChatIds.insert(peerId) - missingChats.append(peer) + + for peer in pinnedPeers { + var peerId: PeerId? + switch peer { + case let .inputPeerUser(userId, _): + peerId = PeerId(namespace: Namespaces.Peer.CloudUser, id: PeerId.Id._internalFromInt64Value(userId)) + case let .inputPeerChat(chatId): + peerId = PeerId(namespace: Namespaces.Peer.CloudGroup, id: PeerId.Id._internalFromInt64Value(chatId)) + case let .inputPeerChannel(channelId, _): + peerId = PeerId(namespace: Namespaces.Peer.CloudChannel, id: PeerId.Id._internalFromInt64Value(channelId)) + default: + break + } + if let peerId = peerId, !missingChatIds.contains(peerId) { + if transaction.getPeerChatListIndex(peerId) == nil { + missingChatIds.insert(peerId) + missingChats.append(peer) + } } } } } + return (filters, missingPeers, missingChats, tagsEnabled) } - return (filters, missingPeers, missingChats) } |> castError(RequestChatListFiltersError.self) - |> mapToSignal { filtersAndMissingPeers -> Signal<[ChatListFilter], RequestChatListFiltersError> in - let (filters, missingPeers, missingChats) = filtersAndMissingPeers + |> mapToSignal { filtersAndMissingPeers -> Signal<([ChatListFilter], Bool), RequestChatListFiltersError> in + let (filters, missingPeers, missingChats, tagsEnabled) = filtersAndMissingPeers var missingUsers: [Api.InputUser] = [] var missingChannels: [Api.InputChannel] = [] @@ -700,13 +711,10 @@ private func requestChatListFilters(accountPeerId: PeerId, postbox: Postbox, net loadMissingChats ) |> castError(RequestChatListFiltersError.self) - |> mapToSignal { _ -> Signal<[ChatListFilter], RequestChatListFiltersError> in - #if swift(<5.1) - return .complete() - #endif + |> mapToSignal { _ -> Signal<([ChatListFilter], Bool), RequestChatListFiltersError> in } |> then( - .single(filters) + Signal<([ChatListFilter], Bool), RequestChatListFiltersError>.single((filters, tagsEnabled)) ) } } @@ -889,14 +897,16 @@ struct ChatListFiltersState: Codable, Equatable { var updates: [ChatListFilterUpdates] + var remoteDisplayTags: Bool? var displayTags: Bool - static var `default` = ChatListFiltersState(filters: [], remoteFilters: nil, updates: [], displayTags: false) + static var `default` = ChatListFiltersState(filters: [], remoteFilters: nil, updates: [], remoteDisplayTags: nil, displayTags: false) - fileprivate init(filters: [ChatListFilter], remoteFilters: [ChatListFilter]?, updates: [ChatListFilterUpdates], displayTags: Bool) { + fileprivate init(filters: [ChatListFilter], remoteFilters: [ChatListFilter]?, updates: [ChatListFilterUpdates], remoteDisplayTags: Bool?, displayTags: Bool) { self.filters = filters self.remoteFilters = remoteFilters self.updates = updates + self.remoteDisplayTags = remoteDisplayTags self.displayTags = displayTags } @@ -906,6 +916,7 @@ struct ChatListFiltersState: Codable, Equatable { self.filters = try container.decode([ChatListFilter].self, forKey: "filters") self.remoteFilters = try container.decodeIfPresent([ChatListFilter].self, forKey: "remoteFilters") self.updates = try container.decodeIfPresent([ChatListFilterUpdates].self, forKey: "updates") ?? [] + self.remoteDisplayTags = try container.decodeIfPresent(Bool.self, forKey: "remoteDisplayTags") self.displayTags = try container.decodeIfPresent(Bool.self, forKey: "displayTags") ?? false } @@ -915,6 +926,7 @@ struct ChatListFiltersState: Codable, Equatable { try container.encode(self.filters, forKey: "filters") try container.encodeIfPresent(self.remoteFilters, forKey: "remoteFilters") try container.encode(self.updates, forKey: "updates") + try container.encodeIfPresent(self.remoteDisplayTags, forKey: "remoteDisplayTags") try container.encode(self.displayTags, forKey: "displayTags") } @@ -959,6 +971,43 @@ func _internal_updateChatListFiltersInteractively(postbox: Postbox, _ f: @escapi } } +func _internal_updateChatListFiltersDisplayTagsInteractively(postbox: Postbox, displayTags: Bool) -> Signal { + return postbox.transaction { transaction -> Void in + var hasUpdates = false + transaction.updatePreferencesEntry(key: PreferencesKeys.chatListFilters, { entry in + var state = entry?.get(ChatListFiltersState.self) ?? ChatListFiltersState.default + if displayTags != state.displayTags { + state.displayTags = displayTags + + if state.displayTags { + for i in 0 ..< state.filters.count { + switch state.filters[i] { + case .allChats: + break + case let .filter(id, title, emoticon, data): + if data.color == nil { + var data = data + data.color = PeerNameColor(rawValue: Int32.random(in: 0 ... 7)) + state.filters[i] = .filter(id: id, title: title, emoticon: emoticon, data: data) + } + } + } + } + + hasUpdates = true + } + + state.normalize() + + return PreferencesEntry(state) + }) + if hasUpdates { + requestChatListFiltersSync(transaction: transaction) + } + } + |> ignoreValues +} + func _internal_updateChatListFiltersInteractively(transaction: Transaction, _ f: ([ChatListFilter]) -> [ChatListFilter]) { var hasUpdates = false transaction.updatePreferencesEntry(key: PreferencesKeys.chatListFilters, { entry in @@ -1366,18 +1415,22 @@ private func synchronizeChatListFilters(transaction: Transaction, accountPeerId: let settings = transaction.getPreferencesEntry(key: PreferencesKeys.chatListFilters)?.get(ChatListFiltersState.self) ?? ChatListFiltersState.default let localFilters = settings.filters let locallyKnownRemoteFilters = settings.remoteFilters ?? [] + let localDisplayTags = settings.displayTags + let locallyKnownRemoteDisplayTags = settings.remoteDisplayTags ?? false return requestChatListFilters(accountPeerId: accountPeerId, postbox: postbox, network: network) - |> `catch` { _ -> Signal<[ChatListFilter], NoError> in + |> `catch` { _ -> Signal<([ChatListFilter], Bool), NoError> in return .complete() } - |> mapToSignal { remoteFilters -> Signal in - if localFilters == locallyKnownRemoteFilters { + |> mapToSignal { remoteFilters, remoteTagsEnabled -> Signal in + if localFilters == locallyKnownRemoteFilters && localDisplayTags == locallyKnownRemoteDisplayTags { return postbox.transaction { transaction -> Void in let _ = updateChatListFiltersState(transaction: transaction, { state in var state = state state.filters = remoteFilters state.remoteFilters = state.filters + state.displayTags = remoteTagsEnabled + state.remoteDisplayTags = state.displayTags return state }) } @@ -1453,6 +1506,17 @@ private func synchronizeChatListFilters(transaction: Transaction, accountPeerId: reorderFilters = .complete() } + let updateTagsEnabled: Signal + if localDisplayTags != remoteTagsEnabled { + updateTagsEnabled = network.request(Api.functions.messages.toggleDialogFilterTags(enabled: localDisplayTags ? .boolTrue : .boolFalse)) + |> ignoreValues + |> `catch` { _ -> Signal in + return .complete() + } + } else { + updateTagsEnabled = .complete() + } + return deleteSignals |> then( addSignals @@ -1460,12 +1524,16 @@ private func synchronizeChatListFilters(transaction: Transaction, accountPeerId: |> then( reorderFilters ) + |> then( + updateTagsEnabled + ) |> then( postbox.transaction { transaction -> Void in let _ = updateChatListFiltersState(transaction: transaction, { state in var state = state state.filters = mergedFilters state.remoteFilters = state.filters + state.remoteDisplayTags = state.displayTags return state }) } diff --git a/submodules/TelegramCore/Sources/TelegramEngine/Peers/TelegramEnginePeers.swift b/submodules/TelegramCore/Sources/TelegramEngine/Peers/TelegramEnginePeers.swift index 0d1e3ea5d74..d0bc67b7258 100644 --- a/submodules/TelegramCore/Sources/TelegramEngine/Peers/TelegramEnginePeers.swift +++ b/submodules/TelegramCore/Sources/TelegramEngine/Peers/TelegramEnginePeers.swift @@ -589,13 +589,7 @@ public extension TelegramEngine { } public func updateChatListFiltersDisplayTags(isEnabled: Bool) { - let _ = self.account.postbox.transaction({ transaction in - updateChatListFiltersState(transaction: transaction, { state in - var state = state - state.displayTags = isEnabled - return state - }) - }).start() + let _ = _internal_updateChatListFiltersDisplayTagsInteractively(postbox: self.account.postbox, displayTags: isEnabled).startStandalone() } public func updatedChatListFilters() -> Signal<[ChatListFilter], NoError> { @@ -1082,7 +1076,7 @@ public extension TelegramEngine { transaction.setMessageHistoryThreadInfo(peerId: id, threadId: threadId, info: nil) - _internal_clearHistory(transaction: transaction, mediaBox: self.account.postbox.mediaBox, peerId: id, threadId: threadId, namespaces: .not(Namespaces.Message.allScheduled)) + _internal_clearHistory(transaction: transaction, mediaBox: self.account.postbox.mediaBox, peerId: id, threadId: threadId, namespaces: .not(Namespaces.Message.allNonRegular)) } |> ignoreValues } @@ -1094,7 +1088,7 @@ public extension TelegramEngine { transaction.setMessageHistoryThreadInfo(peerId: id, threadId: threadId, info: nil) - _internal_clearHistory(transaction: transaction, mediaBox: self.account.postbox.mediaBox, peerId: id, threadId: threadId, namespaces: .not(Namespaces.Message.allScheduled)) + _internal_clearHistory(transaction: transaction, mediaBox: self.account.postbox.mediaBox, peerId: id, threadId: threadId, namespaces: .not(Namespaces.Message.allNonRegular)) } } |> ignoreValues @@ -1373,7 +1367,7 @@ public extension TelegramEngine { return .single(false) } - return self.account.postbox.aroundMessageHistoryViewForLocation(.peer(peerId: id, threadId: nil), anchor: .upperBound, ignoreMessagesInTimestampRange: nil, count: 44, fixedCombinedReadStates: nil, topTaggedMessageIdNamespaces: Set(), tag: nil, appendMessagesFromTheSameGroup: false, namespaces: .not(Namespaces.Message.allScheduled), orderStatistics: []) + return self.account.postbox.aroundMessageHistoryViewForLocation(.peer(peerId: id, threadId: nil), anchor: .upperBound, ignoreMessagesInTimestampRange: nil, count: 44, fixedCombinedReadStates: nil, topTaggedMessageIdNamespaces: Set(), tag: nil, appendMessagesFromTheSameGroup: false, namespaces: .not(Namespaces.Message.allNonRegular), orderStatistics: []) |> map { view -> Bool in for entry in view.0.entries { if entry.message.flags.contains(.Incoming) { diff --git a/submodules/TelegramCore/Sources/TelegramEngine/Peers/UpdateCachedPeerData.swift b/submodules/TelegramCore/Sources/TelegramEngine/Peers/UpdateCachedPeerData.swift index 81af3adfb1f..775b5d7c2e5 100644 --- a/submodules/TelegramCore/Sources/TelegramEngine/Peers/UpdateCachedPeerData.swift +++ b/submodules/TelegramCore/Sources/TelegramEngine/Peers/UpdateCachedPeerData.swift @@ -196,18 +196,28 @@ func _internal_fetchAndUpdateCachedPeerData(accountPeerId: PeerId, peerId rawPee } else { editableBotInfo = .single(nil) } + + var additionalConnectedBots: Signal = .single(nil) + if rawPeerId == accountPeerId { + additionalConnectedBots = network.request(Api.functions.account.getConnectedBots()) + |> map(Optional.init) + |> `catch` { _ -> Signal in + return .single(nil) + } + } return combineLatest( network.request(Api.functions.users.getFullUser(id: inputUser)) |> retryRequest, - editableBotInfo + editableBotInfo, + additionalConnectedBots ) - |> mapToSignal { result, editableBotInfo -> Signal in + |> mapToSignal { result, editableBotInfo, additionalConnectedBots -> Signal in return postbox.transaction { transaction -> Bool in switch result { case let .userFull(fullUser, chats, users): var accountUser: Api.User? - let parsedPeers = AccumulatedPeers(transaction: transaction, chats: chats, users: users) + var parsedPeers = AccumulatedPeers(transaction: transaction, chats: chats, users: users) for user in users { if user.peerId == accountPeerId { accountUser = user @@ -215,8 +225,28 @@ func _internal_fetchAndUpdateCachedPeerData(accountPeerId: PeerId, peerId rawPee } let _ = accountUser + var mappedConnectedBot: TelegramAccountConnectedBot? + + if let additionalConnectedBots { + switch additionalConnectedBots { + case let .connectedBots(connectedBots, users): + parsedPeers = parsedPeers.union(with: AccumulatedPeers(transaction: transaction, chats: [], users: users)) + + if let apiBot = connectedBots.first { + switch apiBot { + case let .connectedBot(flags, botId, recipients): + mappedConnectedBot = TelegramAccountConnectedBot( + id: PeerId(botId), + recipients: TelegramBusinessRecipients(apiValue: recipients), + canReply: (flags & (1 << 0)) != 0 + ) + } + } + } + } + switch fullUser { - case let .userFull(_, _, _, _, _, _, _, userFullNotifySettings, _, _, _, _, _, _, _, _, _, _, _, _): + case let .userFull(_, _, _, _, _, _, _, _, userFullNotifySettings, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _): updatePeers(transaction: transaction, accountPeerId: accountPeerId, peers: parsedPeers) transaction.updateCurrentPeerNotificationSettings([peerId: TelegramPeerNotificationSettings(apiSettings: userFullNotifySettings)]) } @@ -228,7 +258,7 @@ func _internal_fetchAndUpdateCachedPeerData(accountPeerId: PeerId, peerId rawPee previous = CachedUserData() } switch fullUser { - case let .userFull(userFullFlags, _, userFullAbout, userFullSettings, personalPhoto, profilePhoto, fallbackPhoto, _, userFullBotInfo, userFullPinnedMsgId, userFullCommonChatsCount, _, userFullTtlPeriod, userFullThemeEmoticon, _, _, _, userPremiumGiftOptions, userWallpaper, stories): + case let .userFull(userFullFlags, _, _, userFullAbout, userFullSettings, personalPhoto, profilePhoto, fallbackPhoto, _, userFullBotInfo, userFullPinnedMsgId, userFullCommonChatsCount, _, userFullTtlPeriod, userFullThemeEmoticon, _, _, _, userPremiumGiftOptions, userWallpaper, stories, businessWorkHours, businessLocation, greetingMessage, awayMessage): let _ = stories let botInfo = userFullBotInfo.flatMap(BotInfo.init(apiBotInfo:)) let isBlocked = (userFullFlags & (1 << 0)) != 0 @@ -286,6 +316,26 @@ func _internal_fetchAndUpdateCachedPeerData(accountPeerId: PeerId, peerId rawPee let wallpaper = userWallpaper.flatMap { TelegramWallpaper(apiWallpaper: $0) } + var mappedBusinessHours: TelegramBusinessHours? + if let businessWorkHours { + mappedBusinessHours = TelegramBusinessHours(apiWorkingHours: businessWorkHours) + } + + var mappedBusinessLocation: TelegramBusinessLocation? + if let businessLocation { + mappedBusinessLocation = TelegramBusinessLocation(apiLocation: businessLocation) + } + + var mappedGreetingMessage: TelegramBusinessGreetingMessage? + if let greetingMessage { + mappedGreetingMessage = TelegramBusinessGreetingMessage(apiGreetingMessage: greetingMessage) + } + + var mappedAwayMessage: TelegramBusinessAwayMessage? + if let awayMessage { + mappedAwayMessage = TelegramBusinessAwayMessage(apiAwayMessage: awayMessage) + } + return previous.withUpdatedAbout(userFullAbout) .withUpdatedBotInfo(botInfo) .withUpdatedEditableBotInfo(editableBotInfo) @@ -307,6 +357,11 @@ func _internal_fetchAndUpdateCachedPeerData(accountPeerId: PeerId, peerId rawPee .withUpdatedVoiceMessagesAvailable(voiceMessagesAvailable) .withUpdatedWallpaper(wallpaper) .withUpdatedFlags(flags) + .withUpdatedBusinessHours(mappedBusinessHours) + .withUpdatedBusinessLocation(mappedBusinessLocation) + .withUpdatedGreetingMessage(mappedGreetingMessage) + .withUpdatedAwayMessage(mappedAwayMessage) + .withUpdatedConnectedBot(mappedConnectedBot) } }) } diff --git a/submodules/TelegramCore/Sources/TelegramEngine/Stickers/TelegramEngineStickers.swift b/submodules/TelegramCore/Sources/TelegramEngine/Stickers/TelegramEngineStickers.swift index 8e607e3dcea..acc87028079 100644 --- a/submodules/TelegramCore/Sources/TelegramEngine/Stickers/TelegramEngineStickers.swift +++ b/submodules/TelegramCore/Sources/TelegramEngine/Stickers/TelegramEngineStickers.swift @@ -257,6 +257,12 @@ public extension TelegramEngine { return (items.map(\.file), isFinalResult) } } + + public func addRecentlyUsedSticker(file: TelegramMediaFile) { + let _ = self.account.postbox.transaction({ transaction -> Void in + TelegramCore.addRecentlyUsedSticker(transaction: transaction, fileReference: .standalone(media: file)) + }).start() + } } } diff --git a/submodules/TelegramCore/Sources/Utils/MessageUtils.swift b/submodules/TelegramCore/Sources/Utils/MessageUtils.swift index 22173d47cb6..53f630c40d4 100644 --- a/submodules/TelegramCore/Sources/Utils/MessageUtils.swift +++ b/submodules/TelegramCore/Sources/Utils/MessageUtils.swift @@ -181,6 +181,36 @@ func messagesIdsGroupedByPeerId(_ ids: ReferencedReplyMessageIds) -> [PeerId: Re return dict } +func messagesIdsGroupedByPeerId(_ ids: Set) -> [PeerAndThreadId: [MessageId]] { + var dict: [PeerAndThreadId: [MessageId]] = [:] + + for id in ids { + let peerAndThreadId = PeerAndThreadId(peerId: id.messageId.peerId, threadId: id.threadId) + if dict[peerAndThreadId] == nil { + dict[peerAndThreadId] = [id.messageId] + } else { + dict[peerAndThreadId]!.append(id.messageId) + } + } + + return dict +} + +func messagesIdsGroupedByPeerId(_ ids: [MessageAndThreadId]) -> [PeerAndThreadId: [MessageId]] { + var dict: [PeerAndThreadId: [MessageId]] = [:] + + for id in ids { + let peerAndThreadId = PeerAndThreadId(peerId: id.messageId.peerId, threadId: id.threadId) + if dict[peerAndThreadId] == nil { + dict[peerAndThreadId] = [id.messageId] + } else { + dict[peerAndThreadId]!.append(id.messageId) + } + } + + return dict +} + func locallyRenderedMessage(message: StoreMessage, peers: [PeerId: Peer], associatedThreadInfo: Message.AssociatedThreadInfo? = nil) -> Message? { guard case let .Id(id) = message.id else { return nil diff --git a/submodules/TelegramNotices/Sources/Notices.swift b/submodules/TelegramNotices/Sources/Notices.swift index 627928f5ac2..7eee7c482ba 100644 --- a/submodules/TelegramNotices/Sources/Notices.swift +++ b/submodules/TelegramNotices/Sources/Notices.swift @@ -199,6 +199,7 @@ private enum ApplicationSpecificGlobalNotice: Int32 { case savedMessageTagLabelSuggestion = 65 case dismissedLastSeenBadge = 66 case dismissedMessagePrivacyBadge = 67 + case dismissedBusinessBadge = 68 var key: ValueBoxKey { let v = ValueBoxKey(length: 4) @@ -529,6 +530,10 @@ private struct ApplicationSpecificNoticeKeys { static func dismissedMessagePrivacyBadge() -> NoticeEntryKey { return NoticeEntryKey(namespace: noticeNamespace(namespace: globalNamespace), key: ApplicationSpecificGlobalNotice.dismissedMessagePrivacyBadge.key) } + + static func dismissedBusinessBadge() -> NoticeEntryKey { + return NoticeEntryKey(namespace: noticeNamespace(namespace: globalNamespace), key: ApplicationSpecificGlobalNotice.dismissedBusinessBadge.key) + } } public struct ApplicationSpecificNotice { @@ -2223,4 +2228,25 @@ public struct ApplicationSpecificNotice { } |> take(1) } + + public static func setDismissedBusinessBadge(accountManager: AccountManager) -> Signal { + return accountManager.transaction { transaction -> Void in + if let entry = CodableEntry(ApplicationSpecificBoolNotice()) { + transaction.setNotice(ApplicationSpecificNoticeKeys.dismissedBusinessBadge(), entry) + } + } + |> ignoreValues + } + + public static func dismissedBusinessBadge(accountManager: AccountManager) -> Signal { + return accountManager.noticeEntry(key: ApplicationSpecificNoticeKeys.dismissedBusinessBadge()) + |> map { view -> Bool in + if let _ = view.value?.get(ApplicationSpecificBoolNotice.self) { + return true + } else { + return false + } + } + |> take(1) + } } diff --git a/submodules/TelegramPresentationData/Sources/Resources/PresentationResourcesSettings.swift b/submodules/TelegramPresentationData/Sources/Resources/PresentationResourcesSettings.swift index 7e54418f9bb..9dac210e626 100644 --- a/submodules/TelegramPresentationData/Sources/Resources/PresentationResourcesSettings.swift +++ b/submodules/TelegramPresentationData/Sources/Resources/PresentationResourcesSettings.swift @@ -11,10 +11,41 @@ private func drawBorder(context: CGContext, rect: CGRect) { context.strokePath() } -private func renderIcon(name: String) -> UIImage? { +private func addRoundedRectPath(context: CGContext, rect: CGRect, radius: CGFloat) { + context.saveGState() + context.translateBy(x: rect.minX, y: rect.minY) + context.scaleBy(x: radius, y: radius) + let fw = rect.width / radius + let fh = rect.height / radius + context.move(to: CGPoint(x: fw, y: fh / 2.0)) + context.addArc(tangent1End: CGPoint(x: fw, y: fh), tangent2End: CGPoint(x: fw/2, y: fh), radius: 1.0) + context.addArc(tangent1End: CGPoint(x: 0, y: fh), tangent2End: CGPoint(x: 0, y: fh/2), radius: 1) + context.addArc(tangent1End: CGPoint(x: 0, y: 0), tangent2End: CGPoint(x: fw/2, y: 0), radius: 1) + context.addArc(tangent1End: CGPoint(x: fw, y: 0), tangent2End: CGPoint(x: fw, y: fh/2), radius: 1) + context.closePath() + context.restoreGState() +} + +private func renderIcon(name: String, backgroundColors: [UIColor]? = nil) -> UIImage? { return generateImage(CGSize(width: 29.0, height: 29.0), contextGenerator: { size, context in let bounds = CGRect(origin: CGPoint(), size: size) context.clear(bounds) + + if let backgroundColors { + addRoundedRectPath(context: context, rect: CGRect(origin: CGPoint(), size: size), radius: 7.0) + context.clip() + + var locations: [CGFloat] = [0.0, 1.0] + let colors: [CGColor] = backgroundColors.map(\.cgColor) + + let colorSpace = CGColorSpaceCreateDeviceRGB() + let gradient = CGGradient(colorsSpace: colorSpace, colors: colors as CFArray, locations: &locations)! + + context.drawLinearGradient(gradient, start: CGPoint(x: size.width, y: size.height), end: CGPoint(x: 0.0, y: 0.0), options: CGGradientDrawingOptions()) + + context.resetClip() + } + if let image = UIImage(bundleImageName: name)?.cgImage { context.draw(image, in: bounds) } @@ -44,6 +75,7 @@ public struct PresentationResourcesSettings { public static let powerSaving = renderIcon(name: "Settings/Menu/PowerSaving") public static let stories = renderIcon(name: "Settings/Menu/Stories") public static let premiumGift = renderIcon(name: "Settings/Menu/Gift") + public static let business = renderIcon(name: "Settings/Menu/Business", backgroundColors: [UIColor(rgb: 0xA95CE3), UIColor(rgb: 0xF16B80)]) public static let premium = generateImage(CGSize(width: 29.0, height: 29.0), contextGenerator: { size, context in let bounds = CGRect(origin: CGPoint(), size: size) diff --git a/submodules/TelegramStringFormatting/Sources/DateFormat.swift b/submodules/TelegramStringFormatting/Sources/DateFormat.swift index 27479dca88f..fc3e439fa5e 100644 --- a/submodules/TelegramStringFormatting/Sources/DateFormat.swift +++ b/submodules/TelegramStringFormatting/Sources/DateFormat.swift @@ -2,7 +2,7 @@ import Foundation import TelegramPresentationData import TelegramUIPreferences -public func stringForShortTimestamp(hours: Int32, minutes: Int32, dateTimeFormat: PresentationDateTimeFormat) -> String { +public func stringForShortTimestamp(hours: Int32, minutes: Int32, dateTimeFormat: PresentationDateTimeFormat, formatAsPlainText: Bool = false) -> String { switch dateTimeFormat.timeFormat { case .regular: let hourString: String @@ -20,10 +20,18 @@ public func stringForShortTimestamp(hours: Int32, minutes: Int32, dateTimeFormat } else { periodString = "AM" } + + let spaceCharacter: String + if formatAsPlainText { + spaceCharacter = " " + } else { + spaceCharacter = "\u{00a0}" + } + if minutes >= 10 { - return "\(hourString):\(minutes) \(periodString)" + return "\(hourString):\(minutes)\(spaceCharacter)\(periodString)" } else { - return "\(hourString):0\(minutes) \(periodString)" + return "\(hourString):0\(minutes)\(spaceCharacter)\(periodString)" } case .military: return String(format: "%02d:%02d", arguments: [Int(hours), Int(minutes)]) diff --git a/submodules/TelegramStringFormatting/Sources/PresenceStrings.swift b/submodules/TelegramStringFormatting/Sources/PresenceStrings.swift index b8f5bbd3db9..8cdc962093d 100644 --- a/submodules/TelegramStringFormatting/Sources/PresenceStrings.swift +++ b/submodules/TelegramStringFormatting/Sources/PresenceStrings.swift @@ -107,6 +107,46 @@ public func stringForMonth(strings: PresentationStrings, month: Int32, ofYear ye } } +private func monthAtIndex(_ index: Int, strings: PresentationStrings) -> String { + switch index { + case 0: + return strings.Month_ShortJanuary + case 1: + return strings.Month_ShortFebruary + case 2: + return strings.Month_ShortMarch + case 3: + return strings.Month_ShortApril + case 4: + return strings.Month_ShortMay + case 5: + return strings.Month_ShortJune + case 6: + return strings.Month_ShortJuly + case 7: + return strings.Month_ShortAugust + case 8: + return strings.Month_ShortSeptember + case 9: + return strings.Month_ShortOctober + case 10: + return strings.Month_ShortNovember + case 11: + return strings.Month_ShortDecember + default: + return "" + } +} + +public func stringForCompactDate(timestamp: Int32, strings: PresentationStrings, dateTimeFormat: PresentationDateTimeFormat) -> String { + var t: time_t = time_t(timestamp) + var timeinfo: tm = tm() + localtime_r(&t, &timeinfo) + + //TODO:localize + return "\(shortStringForDayOfWeek(strings: strings, day: timeinfo.tm_wday)) \(timeinfo.tm_mday) \(monthAtIndex(Int(timeinfo.tm_mon), strings: strings))" +} + public enum RelativeTimestampFormatDay { case today case yesterday @@ -362,37 +402,6 @@ public func stringForRelativeLiveLocationUpdateTimestamp(strings: PresentationSt } } -private func monthAtIndex(_ index: Int, strings: PresentationStrings) -> String { - switch index { - case 0: - return strings.Month_GenJanuary - case 1: - return strings.Month_GenFebruary - case 2: - return strings.Month_GenMarch - case 3: - return strings.Month_GenApril - case 4: - return strings.Month_GenMay - case 5: - return strings.Month_GenJune - case 6: - return strings.Month_GenJuly - case 7: - return strings.Month_GenAugust - case 8: - return strings.Month_GenSeptember - case 9: - return strings.Month_GenOctober - case 10: - return strings.Month_GenNovember - case 11: - return strings.Month_GenDecember - default: - return "" - } -} - public func stringForRelativeActivityTimestamp(strings: PresentationStrings, dateTimeFormat: PresentationDateTimeFormat, preciseTime: Bool = false, relativeTimestamp: Int32, relativeTo timestamp: Int32) -> String { let difference = timestamp - relativeTimestamp if difference < 60 { diff --git a/submodules/TelegramUI/BUILD b/submodules/TelegramUI/BUILD index e4b131d26b2..104d0a258fd 100644 --- a/submodules/TelegramUI/BUILD +++ b/submodules/TelegramUI/BUILD @@ -20,6 +20,7 @@ NGDEPS = [ "//Nicegram/NGModels:NGModels", "//Nicegram/NGWrap:NGWrap", "@FirebaseSDK//:FirebaseCrashlytics", + "@swiftpkg_nicegram_assistant_ios//:FeatAvatarGeneratorUI", "@swiftpkg_nicegram_assistant_ios//:FeatCardUI", "@swiftpkg_nicegram_assistant_ios//:FeatChatBanner", "@swiftpkg_nicegram_assistant_ios//:FeatPremium", @@ -462,8 +463,11 @@ swift_library( "//submodules/TelegramUI/Components/Chat/TopMessageReactions", "//submodules/TelegramUI/Components/Chat/SavedTagNameAlertController", "//submodules/TelegramUI/Components/Chat/ChatInlineSearchResultsListComponent", - "//submodules/TelegramUI/Components/Settings/BusinessSetupScreen", "//submodules/TelegramUI/Components/Settings/ChatbotSetupScreen", + "//submodules/TelegramUI/Components/Settings/BusinessLocationSetupScreen", + "//submodules/TelegramUI/Components/Settings/BusinessHoursSetupScreen", + "//submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen", + "//submodules/TelegramUI/Components/Settings/QuickReplyNameAlertController", ] + select({ "@build_bazel_rules_apple//apple:ios_arm64": appcenter_targets, "//build-system:ios_sim_arm64": [], diff --git a/submodules/TelegramUI/Components/AudioTranscriptionButtonComponent/Sources/AudioTranscriptionButtonComponent.swift b/submodules/TelegramUI/Components/AudioTranscriptionButtonComponent/Sources/AudioTranscriptionButtonComponent.swift index b614222d278..3696b9ffeaa 100644 --- a/submodules/TelegramUI/Components/AudioTranscriptionButtonComponent/Sources/AudioTranscriptionButtonComponent.swift +++ b/submodules/TelegramUI/Components/AudioTranscriptionButtonComponent/Sources/AudioTranscriptionButtonComponent.swift @@ -17,6 +17,12 @@ public final class AudioTranscriptionButtonComponent: Component { } else { return false } + case let .custom(lhsBackgroundColor, lhsForegroundColor): + if case let .custom(rhsBackgroundColor, rhsForegroundColor) = rhs { + return lhsBackgroundColor == rhsBackgroundColor && lhsForegroundColor == rhsForegroundColor + } else { + return false + } case let .freeform(lhsFreeform, lhsForeground): if case let .freeform(rhsFreeform, rhsForeground) = rhs, lhsFreeform == rhsFreeform, lhsForeground == rhsForeground { return true @@ -27,6 +33,7 @@ public final class AudioTranscriptionButtonComponent: Component { } case bubble(PresentationThemePartedColors) + case custom(UIColor, UIColor) case freeform((UIColor, Bool), UIColor) } @@ -101,6 +108,9 @@ public final class AudioTranscriptionButtonComponent: Component { case let .bubble(theme): foregroundColor = theme.bubble.withWallpaper.reactionActiveBackground backgroundColor = theme.bubble.withWallpaper.reactionInactiveBackground + case let .custom(backgroundColorValue, foregroundColorValue): + foregroundColor = foregroundColorValue + backgroundColor = backgroundColorValue case let .freeform(colorAndBlur, color): foregroundColor = color backgroundColor = .clear diff --git a/submodules/TelegramUI/Components/CameraScreen/Sources/CameraScreen.swift b/submodules/TelegramUI/Components/CameraScreen/Sources/CameraScreen.swift index 84de31e7780..0eb3f6033e9 100644 --- a/submodules/TelegramUI/Components/CameraScreen/Sources/CameraScreen.swift +++ b/submodules/TelegramUI/Components/CameraScreen/Sources/CameraScreen.swift @@ -2384,6 +2384,7 @@ public class CameraScreen: ViewController { bottom: bottomInset, right: layout.safeInsets.right ), + additionalInsets: layout.additionalInsets, inputHeight: layout.inputHeight ?? 0.0, metrics: layout.metrics, deviceMetrics: layout.deviceMetrics, diff --git a/submodules/TelegramUI/Components/Chat/ChatInlineSearchResultsListComponent/Sources/ChatInlineSearchResultsListComponent.swift b/submodules/TelegramUI/Components/Chat/ChatInlineSearchResultsListComponent/Sources/ChatInlineSearchResultsListComponent.swift index 0340332567c..0900afd2ae3 100644 --- a/submodules/TelegramUI/Components/Chat/ChatInlineSearchResultsListComponent/Sources/ChatInlineSearchResultsListComponent.swift +++ b/submodules/TelegramUI/Components/Chat/ChatInlineSearchResultsListComponent/Sources/ChatInlineSearchResultsListComponent.swift @@ -81,6 +81,7 @@ public final class ChatInlineSearchResultsListComponent: Component { public let loadTagMessages: (MemoryBuffer, MessageIndex?) -> Signal? public let getSearchResult: () -> Signal? public let getSavedPeers: (String) -> Signal<[(EnginePeer, MessageIndex?)], NoError>? + public let loadMoreSearchResults: () -> Void public init( context: AccountContext, @@ -92,7 +93,8 @@ public final class ChatInlineSearchResultsListComponent: Component { peerSelected: @escaping (EnginePeer) -> Void, loadTagMessages: @escaping (MemoryBuffer, MessageIndex?) -> Signal?, getSearchResult: @escaping () -> Signal?, - getSavedPeers: @escaping (String) -> Signal<[(EnginePeer, MessageIndex?)], NoError>? + getSavedPeers: @escaping (String) -> Signal<[(EnginePeer, MessageIndex?)], NoError>?, + loadMoreSearchResults: @escaping () -> Void ) { self.context = context self.presentation = presentation @@ -104,6 +106,7 @@ public final class ChatInlineSearchResultsListComponent: Component { self.loadTagMessages = loadTagMessages self.getSearchResult = getSearchResult self.getSavedPeers = getSavedPeers + self.loadMoreSearchResults = loadMoreSearchResults } public static func ==(lhs: ChatInlineSearchResultsListComponent, rhs: ChatInlineSearchResultsListComponent) -> Bool { @@ -377,6 +380,19 @@ public final class ChatInlineSearchResultsListComponent: Component { break } } + } else if let (currentIndex, disposable) = self.searchContents { + if let loadAroundIndex, loadAroundIndex != currentIndex { + switch component.contents { + case .empty: + break + case .tag: + break + case .search: + self.searchContents = (loadAroundIndex, disposable) + + component.loadMoreSearchResults() + } + } } } @@ -511,7 +527,7 @@ public final class ChatInlineSearchResultsListComponent: Component { contentId: .search(query), entries: entries, messages: messages, - hasEarlier: false, + hasEarlier: !(result?.completed ?? true), hasLater: false ) if !self.isUpdating { @@ -617,6 +633,8 @@ public final class ChatInlineSearchResultsListComponent: Component { openStories: { _, _ in }, dismissNotice: { _ in + }, + editPeer: { _ in } ) self.chatListNodeInteraction = chatListNodeInteraction diff --git a/submodules/TelegramUI/Components/Chat/ChatMessageAnimatedStickerItemNode/Sources/ChatMessageAnimatedStickerItemNode.swift b/submodules/TelegramUI/Components/Chat/ChatMessageAnimatedStickerItemNode/Sources/ChatMessageAnimatedStickerItemNode.swift index 3a4f48fb5b4..bf2ab06bd62 100644 --- a/submodules/TelegramUI/Components/Chat/ChatMessageAnimatedStickerItemNode/Sources/ChatMessageAnimatedStickerItemNode.swift +++ b/submodules/TelegramUI/Components/Chat/ChatMessageAnimatedStickerItemNode/Sources/ChatMessageAnimatedStickerItemNode.swift @@ -434,10 +434,10 @@ public class ChatMessageAnimatedStickerItemNode: ChatMessageItemView { override public func setupItem(_ item: ChatMessageItem, synchronousLoad: Bool) { super.setupItem(item, synchronousLoad: synchronousLoad) - if item.message.id.namespace == Namespaces.Message.Local || item.message.id.namespace == Namespaces.Message.ScheduledLocal { + if item.message.id.namespace == Namespaces.Message.Local || item.message.id.namespace == Namespaces.Message.ScheduledLocal || item.message.id.namespace == Namespaces.Message.QuickReplyLocal { self.wasPending = true } - if self.wasPending && (item.message.id.namespace != Namespaces.Message.Local && item.message.id.namespace != Namespaces.Message.ScheduledLocal) { + if self.wasPending && (item.message.id.namespace != Namespaces.Message.Local && item.message.id.namespace != Namespaces.Message.ScheduledLocal && item.message.id.namespace != Namespaces.Message.QuickReplyLocal) { self.didChangeFromPendingToSent = true } @@ -818,8 +818,6 @@ public class ChatMessageAnimatedStickerItemNode: ChatMessageItemView { if !isBroadcastChannel { hasAvatar = true - } else if case .feed = item.chatLocation { - hasAvatar = true } } } else if incoming { @@ -844,8 +842,8 @@ public class ChatMessageAnimatedStickerItemNode: ChatMessageItemView { } else if incoming { hasAvatar = true } - case .feed: - hasAvatar = true + case .customChatContents: + hasAvatar = false } if hasAvatar { @@ -859,7 +857,7 @@ public class ChatMessageAnimatedStickerItemNode: ChatMessageItemView { var needsShareButton = false if case .pinnedMessages = item.associatedData.subject { needsShareButton = true - } else if isFailed || Namespaces.Message.allScheduled.contains(item.message.id.namespace) { + } else if isFailed || Namespaces.Message.allNonRegular.contains(item.message.id.namespace) { needsShareButton = false } else if item.message.id.peerId.isRepliesOrSavedMessages(accountPeerId: item.context.account.peerId) { for attribute in item.content.firstMessage.attributes { @@ -1445,6 +1443,9 @@ public class ChatMessageAnimatedStickerItemNode: ChatMessageItemView { let dateAndStatusFrame = CGRect(origin: CGPoint(x: max(displayLeftInset, updatedImageFrame.maxX - dateAndStatusSize.width - 4.0), y: updatedImageFrame.maxY - dateAndStatusSize.height - 4.0 + imageBottomPadding), size: dateAndStatusSize) animation.animator.updateFrame(layer: strongSelf.dateAndStatusNode.layer, frame: dateAndStatusFrame, completion: nil) dateAndStatusApply(animation) + if case .customChatContents = item.associatedData.subject { + strongSelf.dateAndStatusNode.isHidden = true + } if needsReplyBackground { if strongSelf.replyBackgroundContent == nil, let backgroundContent = item.controllerInteraction.presentationContext.backgroundNode?.makeBubbleBackground(for: .free) { diff --git a/submodules/TelegramUI/Components/Chat/ChatMessageAttachedContentNode/Sources/ChatMessageAttachedContentNode.swift b/submodules/TelegramUI/Components/Chat/ChatMessageAttachedContentNode/Sources/ChatMessageAttachedContentNode.swift index 688e432d8d4..a6c634bea46 100644 --- a/submodules/TelegramUI/Components/Chat/ChatMessageAttachedContentNode/Sources/ChatMessageAttachedContentNode.swift +++ b/submodules/TelegramUI/Components/Chat/ChatMessageAttachedContentNode/Sources/ChatMessageAttachedContentNode.swift @@ -675,7 +675,8 @@ public final class ChatMessageAttachedContentNode: ASDisplayNode { } var statusLayoutAndContinue: (CGFloat, (CGFloat) -> (CGSize, (ListViewItemUpdateAnimation) -> ChatMessageDateAndStatusNode))? - if case let .linear(_, bottom) = position { + if case .customChatContents = associatedData.subject { + } else if case let .linear(_, bottom) = position { switch bottom { case .None, .Neighbour(_, .footer, _): if message.adAttribute == nil { @@ -727,7 +728,7 @@ public final class ChatMessageAttachedContentNode: ASDisplayNode { let contentFileSizeAndApply: (CGSize, ChatMessageInteractiveFileNode.Apply)? if let contentFileFinalizeLayout { - let (size, apply) = contentFileFinalizeLayout(resultingWidth - insets.left - insets.right) + let (size, apply) = contentFileFinalizeLayout(resultingWidth - insets.left - insets.right - 6.0) contentFileSizeAndApply = (size, apply) } else { contentFileSizeAndApply = nil @@ -845,7 +846,7 @@ public final class ChatMessageAttachedContentNode: ASDisplayNode { offsetY: actualSize.height )) - actualSize.height += contentFileSize.height + actualSize.height += contentFileSize.height + 9.0 } case .actionButton: if let (actionButtonSize, _) = actionButtonSizeAndApply { diff --git a/submodules/TelegramUI/Components/Chat/ChatMessageBubbleItemNode/Sources/ChatMessageBubbleItemNode.swift b/submodules/TelegramUI/Components/Chat/ChatMessageBubbleItemNode/Sources/ChatMessageBubbleItemNode.swift index 3fb79cbdeeb..3a0b47b7d76 100644 --- a/submodules/TelegramUI/Components/Chat/ChatMessageBubbleItemNode/Sources/ChatMessageBubbleItemNode.swift +++ b/submodules/TelegramUI/Components/Chat/ChatMessageBubbleItemNode/Sources/ChatMessageBubbleItemNode.swift @@ -338,7 +338,7 @@ private func contentNodeMessagesAndClassesForItem(_ item: ChatMessageItem) -> ([ needReactions = false } - if !isAction && !hasSeparateCommentsButton && !Namespaces.Message.allScheduled.contains(firstMessage.id.namespace) && !hideAllAdditionalInfo { + if !isAction && !hasSeparateCommentsButton && !Namespaces.Message.allNonRegular.contains(firstMessage.id.namespace) && !hideAllAdditionalInfo { if hasCommentButton(item: item) { result.append((firstMessage, ChatMessageCommentFooterContentNode.self, ChatMessageEntryAttributes(), BubbleItemAttributes(isAttachment: true, neighborType: .footer, neighborSpacing: .default))) } @@ -1478,8 +1478,8 @@ public class ChatMessageBubbleItemNode: ChatMessageItemView, ChatMessagePreviewI if !isBroadcastChannel { hasAvatar = incoming - } else if case .feed = item.chatLocation { - hasAvatar = true + } else if case .customChatContents = item.chatLocation { + hasAvatar = false } } } else if incoming { @@ -1563,7 +1563,7 @@ public class ChatMessageBubbleItemNode: ChatMessageItemView, ChatMessagePreviewI } else if case let .replyThread(replyThreadMessage) = item.chatLocation, replyThreadMessage.effectiveTopId == item.message.id { needsShareButton = false allowFullWidth = true - } else if isFailed || Namespaces.Message.allScheduled.contains(item.message.id.namespace) { + } else if isFailed || Namespaces.Message.allNonRegular.contains(item.message.id.namespace) { needsShareButton = false } else if item.message.id.peerId == item.context.account.peerId { if let _ = sourceReference { @@ -2156,7 +2156,8 @@ public class ChatMessageBubbleItemNode: ChatMessageItemView, ChatMessagePreviewI maximumNodeWidth = size.width - if mosaicRange.upperBound == contentPropertiesAndLayouts.count || contentPropertiesAndLayouts[contentPropertiesAndLayouts.count - 1].3.isAttachment { + if case .customChatContents = item.associatedData.subject { + } else if mosaicRange.upperBound == contentPropertiesAndLayouts.count || contentPropertiesAndLayouts[contentPropertiesAndLayouts.count - 1].3.isAttachment { let message = item.content.firstMessage var edited = false diff --git a/submodules/TelegramUI/Components/Chat/ChatMessageContactBubbleContentNode/Sources/ChatMessageContactBubbleContentNode.swift b/submodules/TelegramUI/Components/Chat/ChatMessageContactBubbleContentNode/Sources/ChatMessageContactBubbleContentNode.swift index ff61a4319a2..f94d9c3786c 100644 --- a/submodules/TelegramUI/Components/Chat/ChatMessageContactBubbleContentNode/Sources/ChatMessageContactBubbleContentNode.swift +++ b/submodules/TelegramUI/Components/Chat/ChatMessageContactBubbleContentNode/Sources/ChatMessageContactBubbleContentNode.swift @@ -252,7 +252,10 @@ public class ChatMessageContactBubbleContentNode: ChatMessageBubbleContentNode { let dateText = stringForMessageTimestampStatus(accountPeerId: item.context.account.peerId, message: item.message, dateTimeFormat: item.presentationData.dateTimeFormat, nameDisplayOrder: item.presentationData.nameDisplayOrder, strings: item.presentationData.strings, associatedData: item.associatedData) let statusType: ChatMessageDateAndStatusType? - switch position { + if case .customChatContents = item.associatedData.subject { + statusType = nil + } else { + switch position { case .linear(_, .None), .linear(_, .Neighbour(true, _, _)): if incoming { statusType = .BubbleIncoming @@ -267,6 +270,7 @@ public class ChatMessageContactBubbleContentNode: ChatMessageBubbleContentNode { } default: statusType = nil + } } var statusSuggestedWidthAndContinue: (CGFloat, (CGFloat) -> (CGSize, (ListViewItemUpdateAnimation) -> Void))? diff --git a/submodules/TelegramUI/Components/Chat/ChatMessageFileBubbleContentNode/Sources/ChatMessageFileBubbleContentNode.swift b/submodules/TelegramUI/Components/Chat/ChatMessageFileBubbleContentNode/Sources/ChatMessageFileBubbleContentNode.swift index 10d7db92832..c9ddf26366a 100644 --- a/submodules/TelegramUI/Components/Chat/ChatMessageFileBubbleContentNode/Sources/ChatMessageFileBubbleContentNode.swift +++ b/submodules/TelegramUI/Components/Chat/ChatMessageFileBubbleContentNode/Sources/ChatMessageFileBubbleContentNode.swift @@ -112,8 +112,11 @@ public class ChatMessageFileBubbleContentNode: ChatMessageBubbleContentNode { incoming = false } let statusType: ChatMessageDateAndStatusType? - switch preparePosition { - case .linear(_, .None), .linear(_, .Neighbour(true, _, _)): + if case .customChatContents = item.associatedData.subject { + statusType = nil + } else { + switch preparePosition { + case .linear(_, .None), .linear(_, .Neighbour(true, _, _)): if incoming { statusType = .BubbleIncoming } else { @@ -127,6 +130,7 @@ public class ChatMessageFileBubbleContentNode: ChatMessageBubbleContentNode { } default: statusType = nil + } } let automaticDownload = shouldDownloadMediaAutomatically(settings: item.controllerInteraction.automaticMediaDownloadSettings, peerType: item.associatedData.automaticDownloadPeerType, networkType: item.associatedData.automaticDownloadNetworkType, authorPeerId: item.message.author?.id, contactsPeerIds: item.associatedData.contactsPeerIds, media: selectedFile!) diff --git a/submodules/TelegramUI/Components/Chat/ChatMessageInstantVideoBubbleContentNode/Sources/ChatMessageInstantVideoBubbleContentNode.swift b/submodules/TelegramUI/Components/Chat/ChatMessageInstantVideoBubbleContentNode/Sources/ChatMessageInstantVideoBubbleContentNode.swift index 9eee510b6d3..a010d655e00 100644 --- a/submodules/TelegramUI/Components/Chat/ChatMessageInstantVideoBubbleContentNode/Sources/ChatMessageInstantVideoBubbleContentNode.swift +++ b/submodules/TelegramUI/Components/Chat/ChatMessageInstantVideoBubbleContentNode/Sources/ChatMessageInstantVideoBubbleContentNode.swift @@ -201,21 +201,25 @@ public class ChatMessageInstantVideoBubbleContentNode: ChatMessageBubbleContentN } let statusType: ChatMessageDateAndStatusType? - switch preparePosition { - case .linear(_, .None), .linear(_, .Neighbour(true, _, _)): - if incoming { - statusType = .BubbleIncoming - } else { - if item.message.flags.contains(.Failed) { - statusType = .BubbleOutgoing(.Failed) - } else if (item.message.flags.isSending && !item.message.isSentOrAcknowledged) || item.attributes.updatingMedia != nil { - statusType = .BubbleOutgoing(.Sending) + if case .customChatContents = item.associatedData.subject { + statusType = nil + } else { + switch preparePosition { + case .linear(_, .None), .linear(_, .Neighbour(true, _, _)): + if incoming { + statusType = .BubbleIncoming } else { - statusType = .BubbleOutgoing(.Sent(read: item.read)) + if item.message.flags.contains(.Failed) { + statusType = .BubbleOutgoing(.Failed) + } else if (item.message.flags.isSending && !item.message.isSentOrAcknowledged) || item.attributes.updatingMedia != nil { + statusType = .BubbleOutgoing(.Sending) + } else { + statusType = .BubbleOutgoing(.Sent(read: item.read)) + } } + default: + statusType = nil } - default: - statusType = nil } let automaticDownload = shouldDownloadMediaAutomatically(settings: item.controllerInteraction.automaticMediaDownloadSettings, peerType: item.associatedData.automaticDownloadPeerType, networkType: item.associatedData.automaticDownloadNetworkType, authorPeerId: item.message.author?.id, contactsPeerIds: item.associatedData.contactsPeerIds, media: selectedFile!) diff --git a/submodules/TelegramUI/Components/Chat/ChatMessageInstantVideoItemNode/Sources/ChatMessageInstantVideoItemNode.swift b/submodules/TelegramUI/Components/Chat/ChatMessageInstantVideoItemNode/Sources/ChatMessageInstantVideoItemNode.swift index e19651af743..c1ba69fe709 100644 --- a/submodules/TelegramUI/Components/Chat/ChatMessageInstantVideoItemNode/Sources/ChatMessageInstantVideoItemNode.swift +++ b/submodules/TelegramUI/Components/Chat/ChatMessageInstantVideoItemNode/Sources/ChatMessageInstantVideoItemNode.swift @@ -99,7 +99,7 @@ public class ChatMessageInstantVideoItemNode: ChatMessageItemView, UIGestureReco self.interactiveVideoNode.shouldOpen = { [weak self] in if let strongSelf = self { - if let item = strongSelf.item, (item.message.id.namespace == Namespaces.Message.Local || item.message.id.namespace == Namespaces.Message.ScheduledLocal) { + if let item = strongSelf.item, (item.message.id.namespace == Namespaces.Message.Local || item.message.id.namespace == Namespaces.Message.ScheduledLocal || item.message.id.namespace == Namespaces.Message.QuickReplyLocal) { return false } return !strongSelf.animatingHeight @@ -321,8 +321,8 @@ public class ChatMessageInstantVideoItemNode: ChatMessageItemView, UIGestureReco if !isBroadcastChannel { hasAvatar = true - } else if case .feed = item.chatLocation { - hasAvatar = true + } else if case .customChatContents = item.chatLocation { + hasAvatar = false } } } else if incoming { @@ -341,7 +341,7 @@ public class ChatMessageInstantVideoItemNode: ChatMessageItemView, UIGestureReco var needsShareButton = false if case .pinnedMessages = item.associatedData.subject { needsShareButton = true - } else if isFailed || Namespaces.Message.allScheduled.contains(item.message.id.namespace) { + } else if isFailed || Namespaces.Message.allNonRegular.contains(item.message.id.namespace) { needsShareButton = false } else if item.message.id.peerId == item.context.account.peerId { diff --git a/submodules/TelegramUI/Components/Chat/ChatMessageInteractiveFileNode/Sources/ChatMessageInteractiveFileNode.swift b/submodules/TelegramUI/Components/Chat/ChatMessageInteractiveFileNode/Sources/ChatMessageInteractiveFileNode.swift index 4262768be25..7c11d7204f4 100644 --- a/submodules/TelegramUI/Components/Chat/ChatMessageInteractiveFileNode/Sources/ChatMessageInteractiveFileNode.swift +++ b/submodules/TelegramUI/Components/Chat/ChatMessageInteractiveFileNode/Sources/ChatMessageInteractiveFileNode.swift @@ -883,7 +883,9 @@ public final class ChatMessageInteractiveFileNode: ASDisplayNode { var updatedAudioTranscriptionState: AudioTranscriptionButtonComponent.TranscriptionState? var displayTranscribe = false - if arguments.message.id.peerId.namespace != Namespaces.Peer.SecretChat && !isViewOnceMessage && !arguments.presentationData.isPreview { + if Namespaces.Message.allNonRegular.contains(arguments.message.id.namespace) { + displayTranscribe = false + } else if arguments.message.id.peerId.namespace != Namespaces.Peer.SecretChat && !isViewOnceMessage && !arguments.presentationData.isPreview { let premiumConfiguration = PremiumConfiguration.with(appConfiguration: arguments.context.currentAppConfiguration.with { $0 }) if arguments.associatedData.isPremium { displayTranscribe = true @@ -1469,10 +1471,16 @@ public final class ChatMessageInteractiveFileNode: ASDisplayNode { strongSelf.view.addSubview(audioTranscriptionButton) added = true } + let buttonTheme: AudioTranscriptionButtonComponent.Theme + if let customTintColor = arguments.customTintColor { + buttonTheme = .custom(customTintColor.withMultipliedAlpha(0.1), customTintColor) + } else { + buttonTheme = .bubble(arguments.incoming ? arguments.presentationData.theme.theme.chat.message.incoming : arguments.presentationData.theme.theme.chat.message.outgoing) + } let audioTranscriptionButtonSize = audioTranscriptionButton.update( transition: animation.isAnimated ? .easeInOut(duration: 0.3) : .immediate, component: AnyComponent(AudioTranscriptionButtonComponent( - theme: .bubble(arguments.incoming ? arguments.presentationData.theme.theme.chat.message.incoming : arguments.presentationData.theme.theme.chat.message.outgoing), + theme: buttonTheme, transcriptionState: effectiveAudioTranscriptionState, pressed: { guard let strongSelf = self else { diff --git a/submodules/TelegramUI/Components/Chat/ChatMessageInteractiveInstantVideoNode/Sources/ChatMessageInteractiveInstantVideoNode.swift b/submodules/TelegramUI/Components/Chat/ChatMessageInteractiveInstantVideoNode/Sources/ChatMessageInteractiveInstantVideoNode.swift index a3a48a3c5b1..e10b3d40c74 100644 --- a/submodules/TelegramUI/Components/Chat/ChatMessageInteractiveInstantVideoNode/Sources/ChatMessageInteractiveInstantVideoNode.swift +++ b/submodules/TelegramUI/Components/Chat/ChatMessageInteractiveInstantVideoNode/Sources/ChatMessageInteractiveInstantVideoNode.swift @@ -464,7 +464,7 @@ public class ChatMessageInteractiveInstantVideoNode: ASDisplayNode { break } } - if item.message.id.namespace == Namespaces.Message.Local || item.message.id.namespace == Namespaces.Message.ScheduledLocal { + if item.message.id.namespace == Namespaces.Message.Local || item.message.id.namespace == Namespaces.Message.ScheduledLocal || item.message.id.namespace == Namespaces.Message.QuickReplyLocal { notConsumed = true } @@ -949,6 +949,10 @@ public class ChatMessageInteractiveInstantVideoNode: ASDisplayNode { animation.animator.updateFrame(layer: strongSelf.dateAndStatusNode.layer, frame: dateAndStatusFrame, completion: nil) } + if case .customChatContents = item.associatedData.subject { + strongSelf.dateAndStatusNode.isHidden = true + } + if let videoNode = strongSelf.videoNode { videoNode.bounds = CGRect(origin: CGPoint(), size: videoFrame.size) if strongSelf.imageScale != imageScale { diff --git a/submodules/TelegramUI/Components/Chat/ChatMessageItemImpl/Sources/ChatMessageDateHeader.swift b/submodules/TelegramUI/Components/Chat/ChatMessageItemImpl/Sources/ChatMessageDateHeader.swift index e1f4f5b65f8..297e1a4988c 100644 --- a/submodules/TelegramUI/Components/Chat/ChatMessageItemImpl/Sources/ChatMessageDateHeader.swift +++ b/submodules/TelegramUI/Components/Chat/ChatMessageItemImpl/Sources/ChatMessageDateHeader.swift @@ -524,7 +524,7 @@ public final class ChatMessageAvatarHeaderNodeImpl: ListViewItemHeaderNode, Chat return } var messageId: MessageId? - if let messageReference = messageReference, case let .message(_, _, id, _, _, _) = messageReference.content { + if let messageReference = messageReference, case let .message(_, _, id, _, _, _, _) = messageReference.content { messageId = id } strongSelf.controllerInteraction?.openPeerContextMenu(peer, messageId, strongSelf.containerNode, strongSelf.containerNode.bounds, gesture) @@ -751,7 +751,7 @@ public final class ChatMessageAvatarHeaderNodeImpl: ListViewItemHeaderNode, Chat @objc private func tapGesture(_ recognizer: ListViewTapGestureRecognizer) { if case .ended = recognizer.state { - if self.peerId.namespace == Namespaces.Peer.Empty, case let .message(_, _, id, _, _, _) = self.messageReference?.content { + if self.peerId.namespace == Namespaces.Peer.Empty, case let .message(_, _, id, _, _, _, _) = self.messageReference?.content { self.controllerInteraction?.displayMessageTooltip(id, self.presentationData.strings.Conversation_ForwardAuthorHiddenTooltip, self, self.avatarNode.frame) } else if let peer = self.peer { if let adMessageId = self.adMessageId { diff --git a/submodules/TelegramUI/Components/Chat/ChatMessageItemImpl/Sources/ChatMessageItemImpl.swift b/submodules/TelegramUI/Components/Chat/ChatMessageItemImpl/Sources/ChatMessageItemImpl.swift index 809bc40416b..e6cd1aaaeae 100644 --- a/submodules/TelegramUI/Components/Chat/ChatMessageItemImpl/Sources/ChatMessageItemImpl.swift +++ b/submodules/TelegramUI/Components/Chat/ChatMessageItemImpl/Sources/ChatMessageItemImpl.swift @@ -372,7 +372,10 @@ public final class ChatMessageItemImpl: ChatMessageItem, CustomStringConvertible } self.avatarHeader = avatarHeader - var headers: [ListViewItemHeader] = [self.dateHeader] + var headers: [ListViewItemHeader] = [] + if !self.disableDate { + headers.append(self.dateHeader) + } if case .messageOptions = associatedData.subject { headers = [] } diff --git a/submodules/TelegramUI/Components/Chat/ChatMessageMapBubbleContentNode/Sources/ChatMessageMapBubbleContentNode.swift b/submodules/TelegramUI/Components/Chat/ChatMessageMapBubbleContentNode/Sources/ChatMessageMapBubbleContentNode.swift index 18cf1765151..c1f43229378 100644 --- a/submodules/TelegramUI/Components/Chat/ChatMessageMapBubbleContentNode/Sources/ChatMessageMapBubbleContentNode.swift +++ b/submodules/TelegramUI/Components/Chat/ChatMessageMapBubbleContentNode/Sources/ChatMessageMapBubbleContentNode.swift @@ -223,7 +223,10 @@ public class ChatMessageMapBubbleContentNode: ChatMessageBubbleContentNode { let dateText = stringForMessageTimestampStatus(accountPeerId: item.context.account.peerId, message: item.message, dateTimeFormat: item.presentationData.dateTimeFormat, nameDisplayOrder: item.presentationData.nameDisplayOrder, strings: item.presentationData.strings, format: dateFormat, associatedData: item.associatedData) let statusType: ChatMessageDateAndStatusType? - switch position { + if case .customChatContents = item.associatedData.subject { + statusType = nil + } else { + switch position { case .linear(_, .None), .linear(_, .Neighbour(true, _, _)): if selectedMedia?.venue != nil || activeLiveBroadcastingTimeout != nil { if incoming { @@ -252,6 +255,7 @@ public class ChatMessageMapBubbleContentNode: ChatMessageBubbleContentNode { } default: statusType = nil + } } var statusSize = CGSize() diff --git a/submodules/TelegramUI/Components/Chat/ChatMessageMediaBubbleContentNode/Sources/ChatMessageMediaBubbleContentNode.swift b/submodules/TelegramUI/Components/Chat/ChatMessageMediaBubbleContentNode/Sources/ChatMessageMediaBubbleContentNode.swift index 85a3503874d..61fe835f664 100644 --- a/submodules/TelegramUI/Components/Chat/ChatMessageMediaBubbleContentNode/Sources/ChatMessageMediaBubbleContentNode.swift +++ b/submodules/TelegramUI/Components/Chat/ChatMessageMediaBubbleContentNode/Sources/ChatMessageMediaBubbleContentNode.swift @@ -284,7 +284,10 @@ public class ChatMessageMediaBubbleContentNode: ChatMessageBubbleContentNode { let dateText = stringForMessageTimestampStatus(accountPeerId: item.context.account.peerId, message: item.message, dateTimeFormat: item.presentationData.dateTimeFormat, nameDisplayOrder: item.presentationData.nameDisplayOrder, strings: item.presentationData.strings, format: dateFormat, associatedData: item.associatedData) let statusType: ChatMessageDateAndStatusType? - switch preparePosition { + if case .customChatContents = item.associatedData.subject { + statusType = nil + } else { + switch preparePosition { case .linear(_, .None), .linear(_, .Neighbour(true, _, _)): if item.message.effectivelyIncoming(item.context.account.peerId) { statusType = .ImageIncoming @@ -301,6 +304,7 @@ public class ChatMessageMediaBubbleContentNode: ChatMessageBubbleContentNode { statusType = nil default: statusType = nil + } } var isReplyThread = false diff --git a/submodules/TelegramUI/Components/Chat/ChatMessagePollBubbleContentNode/Sources/ChatMessagePollBubbleContentNode.swift b/submodules/TelegramUI/Components/Chat/ChatMessagePollBubbleContentNode/Sources/ChatMessagePollBubbleContentNode.swift index a0261a4de67..940ca9feb7b 100644 --- a/submodules/TelegramUI/Components/Chat/ChatMessagePollBubbleContentNode/Sources/ChatMessagePollBubbleContentNode.swift +++ b/submodules/TelegramUI/Components/Chat/ChatMessagePollBubbleContentNode/Sources/ChatMessagePollBubbleContentNode.swift @@ -494,7 +494,7 @@ private final class ChatMessagePollOptionNode: ASDisplayNode { let shouldHaveRadioNode = optionResult == nil let isSelectable: Bool - if shouldHaveRadioNode, case .poll(multipleAnswers: true) = poll.kind, !Namespaces.Message.allScheduled.contains(message.id.namespace) { + if shouldHaveRadioNode, case .poll(multipleAnswers: true) = poll.kind, !Namespaces.Message.allNonRegular.contains(message.id.namespace) { isSelectable = true } else { isSelectable = false @@ -905,7 +905,7 @@ public class ChatMessagePollBubbleContentNode: ChatMessageBubbleContentNode { } } if !hasSelection { - if !Namespaces.Message.allScheduled.contains(item.message.id.namespace) { + if !Namespaces.Message.allNonRegular.contains(item.message.id.namespace) { item.controllerInteraction.requestOpenMessagePollResults(item.message.id, pollId) } } else if !selectedOpaqueIdentifiers.isEmpty { @@ -986,7 +986,10 @@ public class ChatMessagePollBubbleContentNode: ChatMessageBubbleContentNode { let dateText = stringForMessageTimestampStatus(accountPeerId: item.context.account.peerId, message: item.message, dateTimeFormat: item.presentationData.dateTimeFormat, nameDisplayOrder: item.presentationData.nameDisplayOrder, strings: item.presentationData.strings, format: dateFormat, associatedData: item.associatedData) let statusType: ChatMessageDateAndStatusType? - switch position { + if case .customChatContents = item.associatedData.subject { + statusType = nil + } else { + switch position { case .linear(_, .None), .linear(_, .Neighbour(true, _, _)): if incoming { statusType = .BubbleIncoming @@ -1001,8 +1004,9 @@ public class ChatMessagePollBubbleContentNode: ChatMessageBubbleContentNode { } default: statusType = nil + } } - + var statusSuggestedWidthAndContinue: (CGFloat, (CGFloat) -> (CGSize, (ListViewItemUpdateAnimation) -> Void))? if let statusType = statusType { @@ -1207,7 +1211,7 @@ public class ChatMessagePollBubbleContentNode: ChatMessageBubbleContentNode { boundingSize.width = max(boundingSize.width, min(270.0, constrainedSize.width)) var canVote = false - if (item.message.id.namespace == Namespaces.Message.Cloud || Namespaces.Message.allScheduled.contains(item.message.id.namespace)), let poll = poll, poll.pollId.namespace == Namespaces.Media.CloudPoll, !isClosed { + if (item.message.id.namespace == Namespaces.Message.Cloud || Namespaces.Message.allNonRegular.contains(item.message.id.namespace)), let poll = poll, poll.pollId.namespace == Namespaces.Media.CloudPoll, !isClosed { var hasVoted = false if let voters = poll.results.voters { for voter in voters { @@ -1576,7 +1580,7 @@ public class ChatMessagePollBubbleContentNode: ChatMessageBubbleContentNode { self.buttonNode.isHidden = false } - if Namespaces.Message.allScheduled.contains(item.message.id.namespace) { + if Namespaces.Message.allNonRegular.contains(item.message.id.namespace) { self.buttonNode.isUserInteractionEnabled = false } else { self.buttonNode.isUserInteractionEnabled = true @@ -1639,7 +1643,7 @@ public class ChatMessagePollBubbleContentNode: ChatMessageBubbleContentNode { if optionNode.frame.contains(point), case .tap = gesture { if optionNode.isUserInteractionEnabled { return ChatMessageBubbleContentTapAction(content: .ignore) - } else if let result = optionNode.currentResult, let item = self.item, !Namespaces.Message.allScheduled.contains(item.message.id.namespace), let poll = self.poll, let option = optionNode.option, !isBotChat { + } else if let result = optionNode.currentResult, let item = self.item, !Namespaces.Message.allNonRegular.contains(item.message.id.namespace), let poll = self.poll, let option = optionNode.option, !isBotChat { switch poll.publicity { case .anonymous: let string: String diff --git a/submodules/TelegramUI/Components/Chat/ChatMessageRestrictedBubbleContentNode/Sources/ChatMessageRestrictedBubbleContentNode.swift b/submodules/TelegramUI/Components/Chat/ChatMessageRestrictedBubbleContentNode/Sources/ChatMessageRestrictedBubbleContentNode.swift index 6dbf1521f66..5871ffeff55 100644 --- a/submodules/TelegramUI/Components/Chat/ChatMessageRestrictedBubbleContentNode/Sources/ChatMessageRestrictedBubbleContentNode.swift +++ b/submodules/TelegramUI/Components/Chat/ChatMessageRestrictedBubbleContentNode/Sources/ChatMessageRestrictedBubbleContentNode.swift @@ -77,21 +77,25 @@ public class ChatMessageRestrictedBubbleContentNode: ChatMessageBubbleContentNod let dateText = stringForMessageTimestampStatus(accountPeerId: item.context.account.peerId, message: item.message, dateTimeFormat: item.presentationData.dateTimeFormat, nameDisplayOrder: item.presentationData.nameDisplayOrder, strings: item.presentationData.strings, associatedData: item.associatedData) let statusType: ChatMessageDateAndStatusType? - switch position { - case .linear(_, .None), .linear(_, .Neighbour(true, _, _)): - if incoming { - statusType = .BubbleIncoming - } else { - if message.flags.contains(.Failed) { - statusType = .BubbleOutgoing(.Failed) - } else if message.flags.isSending && !message.isSentOrAcknowledged { - statusType = .BubbleOutgoing(.Sending) + if case .customChatContents = item.associatedData.subject { + statusType = nil + } else { + switch position { + case .linear(_, .None), .linear(_, .Neighbour(true, _, _)): + if incoming { + statusType = .BubbleIncoming } else { - statusType = .BubbleOutgoing(.Sent(read: item.read)) + if message.flags.contains(.Failed) { + statusType = .BubbleOutgoing(.Failed) + } else if message.flags.isSending && !message.isSentOrAcknowledged { + statusType = .BubbleOutgoing(.Sending) + } else { + statusType = .BubbleOutgoing(.Sent(read: item.read)) + } } + default: + statusType = nil } - default: - statusType = nil } let entities = [MessageTextEntity(range: 0.. Void public let openRecommendedChannelContextMenu: (EnginePeer, UIView, ContextGesture?) -> Void public let openGroupBoostInfo: (EnginePeer.Id?, Int) -> Void + public let openStickerEditor: () -> Void public let requestMessageUpdate: (MessageId, Bool) -> Void public let cancelInteractiveKeyboardGestures: () -> Void @@ -362,6 +363,7 @@ public final class ChatControllerInteraction: ChatControllerInteractionProtocol openPremiumStatusInfo: @escaping (EnginePeer.Id, UIView, Int64?, PeerNameColor) -> Void, openRecommendedChannelContextMenu: @escaping (EnginePeer, UIView, ContextGesture?) -> Void, openGroupBoostInfo: @escaping (EnginePeer.Id?, Int) -> Void, + openStickerEditor: @escaping () -> Void, requestMessageUpdate: @escaping (MessageId, Bool) -> Void, cancelInteractiveKeyboardGestures: @escaping () -> Void, dismissTextInput: @escaping () -> Void, @@ -468,6 +470,7 @@ public final class ChatControllerInteraction: ChatControllerInteractionProtocol self.openPremiumStatusInfo = openPremiumStatusInfo self.openRecommendedChannelContextMenu = openRecommendedChannelContextMenu self.openGroupBoostInfo = openGroupBoostInfo + self.openStickerEditor = openStickerEditor self.requestMessageUpdate = requestMessageUpdate self.cancelInteractiveKeyboardGestures = cancelInteractiveKeyboardGestures diff --git a/submodules/TelegramUI/Components/ChatEntityKeyboardInputNode/Sources/ChatEntityKeyboardInputNode.swift b/submodules/TelegramUI/Components/ChatEntityKeyboardInputNode/Sources/ChatEntityKeyboardInputNode.swift index 70f925a52e5..d6e954fdb2d 100644 --- a/submodules/TelegramUI/Components/ChatEntityKeyboardInputNode/Sources/ChatEntityKeyboardInputNode.swift +++ b/submodules/TelegramUI/Components/ChatEntityKeyboardInputNode/Sources/ChatEntityKeyboardInputNode.swift @@ -56,6 +56,7 @@ public final class ChatEntityKeyboardInputNode: ChatInputNode { let dismissTextInput: () -> Void let insertText: (NSAttributedString) -> Void let backwardsDeleteText: () -> Void + let openStickerEditor: () -> Void let presentController: (ViewController, Any?) -> Void let presentGlobalOverlayController: (ViewController, Any?) -> Void let getNavigationController: () -> NavigationController? @@ -72,6 +73,7 @@ public final class ChatEntityKeyboardInputNode: ChatInputNode { dismissTextInput: @escaping () -> Void, insertText: @escaping (NSAttributedString) -> Void, backwardsDeleteText: @escaping () -> Void, + openStickerEditor: @escaping () -> Void, presentController: @escaping (ViewController, Any?) -> Void, presentGlobalOverlayController: @escaping (ViewController, Any?) -> Void, getNavigationController: @escaping () -> NavigationController?, @@ -86,6 +88,7 @@ public final class ChatEntityKeyboardInputNode: ChatInputNode { self.dismissTextInput = dismissTextInput self.insertText = insertText self.backwardsDeleteText = backwardsDeleteText + self.openStickerEditor = openStickerEditor self.presentController = presentController self.presentGlobalOverlayController = presentGlobalOverlayController self.getNavigationController = getNavigationController @@ -106,6 +109,7 @@ public final class ChatEntityKeyboardInputNode: ChatInputNode { self.dismissTextInput = chatControllerInteraction.dismissTextInput self.insertText = panelInteraction.insertText self.backwardsDeleteText = panelInteraction.backwardsDeleteText + self.openStickerEditor = chatControllerInteraction.openStickerEditor self.presentController = chatControllerInteraction.presentController self.presentGlobalOverlayController = chatControllerInteraction.presentGlobalOverlayController self.getNavigationController = chatControllerInteraction.navigationController @@ -1148,6 +1152,9 @@ public final class ChatEntityKeyboardInputNode: ChatInputNode { return } guard let file = item.itemFile else { + if groupId == AnyHashable("recent"), case .icon(.add) = item.content { + interaction.openStickerEditor() + } return } diff --git a/submodules/TelegramUI/Components/ChatListHeaderComponent/Sources/ChatListHeaderComponent.swift b/submodules/TelegramUI/Components/ChatListHeaderComponent/Sources/ChatListHeaderComponent.swift index e41442d0ae3..ae0f24162d2 100644 --- a/submodules/TelegramUI/Components/ChatListHeaderComponent/Sources/ChatListHeaderComponent.swift +++ b/submodules/TelegramUI/Components/ChatListHeaderComponent/Sources/ChatListHeaderComponent.swift @@ -267,7 +267,9 @@ public final class ChatListHeaderComponent: Component { } func update(title: String, theme: PresentationTheme, availableSize: CGSize, transition: Transition) -> CGSize { - self.titleView.attributedText = NSAttributedString(string: title, font: Font.regular(17.0), textColor: theme.rootController.navigationBar.accentTextColor) + let titleText = NSAttributedString(string: title, font: Font.regular(17.0), textColor: theme.rootController.navigationBar.accentTextColor) + let titleTextUpdated = self.titleView.attributedText != titleText + self.titleView.attributedText = titleText let titleSize = self.titleView.updateLayout(CGSize(width: 100.0, height: 44.0)) self.accessibilityLabel = title @@ -287,7 +289,12 @@ public final class ChatListHeaderComponent: Component { transition.setPosition(view: self.arrowView, position: arrowFrame.center) transition.setBounds(view: self.arrowView, bounds: CGRect(origin: CGPoint(), size: arrowFrame.size)) - transition.setFrame(view: self.titleView, frame: CGRect(origin: CGPoint(x: iconOffset - 3.0 + arrowSize.width + iconSpacing, y: floor((availableSize.height - titleSize.height) / 2.0)), size: titleSize)) + let titleFrame = CGRect(origin: CGPoint(x: iconOffset - 3.0 + arrowSize.width + iconSpacing, y: floor((availableSize.height - titleSize.height) / 2.0)), size: titleSize) + if titleTextUpdated { + self.titleView.frame = titleFrame + } else { + transition.setFrame(view: self.titleView, frame: titleFrame) + } return CGSize(width: iconOffset + arrowSize.width + iconSpacing + titleSize.width, height: availableSize.height) } @@ -479,7 +486,9 @@ public final class ChatListHeaderComponent: Component { transition.setPosition(view: self.titleScaleContainer, position: CGPoint(x: size.width * 0.5, y: size.height * 0.5)) transition.setBounds(view: self.titleScaleContainer, bounds: CGRect(origin: self.titleScaleContainer.bounds.origin, size: size)) - self.titleTextView.attributedText = NSAttributedString(string: content.title, font: Font.semibold(17.0), textColor: theme.rootController.navigationBar.primaryTextColor) + let titleText = NSAttributedString(string: content.title, font: Font.semibold(17.0), textColor: theme.rootController.navigationBar.primaryTextColor) + let titleTextUpdated = self.titleTextView.attributedText != titleText + self.titleTextView.attributedText = titleText let buttonSpacing: CGFloat = 8.0 @@ -616,7 +625,11 @@ public final class ChatListHeaderComponent: Component { let titleTextSize = self.titleTextView.updateLayout(CGSize(width: remainingWidth, height: size.height)) let titleFrame = CGRect(origin: CGPoint(x: floor((size.width - titleTextSize.width) / 2.0) + sideContentWidth, y: floor((size.height - titleTextSize.height) / 2.0)), size: titleTextSize) - transition.setFrame(view: self.titleTextView, frame: titleFrame) + if titleTextUpdated { + self.titleTextView.frame = titleFrame + } else { + transition.setFrame(view: self.titleTextView, frame: titleFrame) + } if let titleComponent = content.titleComponent { var titleContentTransition = transition diff --git a/submodules/TelegramUI/Components/ChatListHeaderComponent/Sources/ChatListNavigationBar.swift b/submodules/TelegramUI/Components/ChatListHeaderComponent/Sources/ChatListNavigationBar.swift index 4f8e504ce4d..d59b4fe7729 100644 --- a/submodules/TelegramUI/Components/ChatListHeaderComponent/Sources/ChatListNavigationBar.swift +++ b/submodules/TelegramUI/Components/ChatListHeaderComponent/Sources/ChatListNavigationBar.swift @@ -27,6 +27,7 @@ public final class ChatListNavigationBar: Component { public let statusBarHeight: CGFloat public let sideInset: CGFloat public let isSearchActive: Bool + public let isSearchEnabled: Bool public let primaryContent: ChatListHeaderComponent.Content? public let secondaryContent: ChatListHeaderComponent.Content? public let secondaryTransition: CGFloat @@ -48,6 +49,7 @@ public final class ChatListNavigationBar: Component { statusBarHeight: CGFloat, sideInset: CGFloat, isSearchActive: Bool, + isSearchEnabled: Bool, primaryContent: ChatListHeaderComponent.Content?, secondaryContent: ChatListHeaderComponent.Content?, secondaryTransition: CGFloat, @@ -68,6 +70,7 @@ public final class ChatListNavigationBar: Component { self.statusBarHeight = statusBarHeight self.sideInset = sideInset self.isSearchActive = isSearchActive + self.isSearchEnabled = isSearchEnabled self.primaryContent = primaryContent self.secondaryContent = secondaryContent self.secondaryTransition = secondaryTransition @@ -102,6 +105,9 @@ public final class ChatListNavigationBar: Component { if lhs.isSearchActive != rhs.isSearchActive { return false } + if lhs.isSearchEnabled != rhs.isSearchEnabled { + return false + } if lhs.primaryContent != rhs.primaryContent { return false } @@ -316,6 +322,9 @@ public final class ChatListNavigationBar: Component { searchContentNode.updateLayout(size: searchSize, leftInset: component.sideInset, rightInset: component.sideInset, transition: transition.containedViewLayoutTransition) + transition.setAlpha(view: searchContentNode.view, alpha: component.isSearchEnabled ? 1.0 : 0.5) + searchContentNode.isUserInteractionEnabled = component.isSearchEnabled + let headerTransition = transition let storiesOffsetFraction: CGFloat @@ -520,6 +529,7 @@ public final class ChatListNavigationBar: Component { statusBarHeight: component.statusBarHeight, sideInset: component.sideInset, isSearchActive: component.isSearchActive, + isSearchEnabled: component.isSearchEnabled, primaryContent: component.primaryContent, secondaryContent: component.secondaryContent, secondaryTransition: component.secondaryTransition, diff --git a/submodules/TelegramUI/Components/EmptyStateIndicatorComponent/Sources/EmptyStateIndicatorComponent.swift b/submodules/TelegramUI/Components/EmptyStateIndicatorComponent/Sources/EmptyStateIndicatorComponent.swift index 4cb104ad66a..01ac92dbe0c 100644 --- a/submodules/TelegramUI/Components/EmptyStateIndicatorComponent/Sources/EmptyStateIndicatorComponent.swift +++ b/submodules/TelegramUI/Components/EmptyStateIndicatorComponent/Sources/EmptyStateIndicatorComponent.swift @@ -16,6 +16,8 @@ public final class EmptyStateIndicatorComponent: Component { public let text: String public let actionTitle: String? public let action: () -> Void + public let additionalActionTitle: String? + public let additionalAction: () -> Void public init( context: AccountContext, @@ -24,7 +26,9 @@ public final class EmptyStateIndicatorComponent: Component { title: String, text: String, actionTitle: String?, - action: @escaping () -> Void + action: @escaping () -> Void, + additionalActionTitle: String?, + additionalAction: @escaping () -> Void ) { self.context = context self.theme = theme @@ -33,6 +37,8 @@ public final class EmptyStateIndicatorComponent: Component { self.text = text self.actionTitle = actionTitle self.action = action + self.additionalActionTitle = additionalActionTitle + self.additionalAction = additionalAction } public static func ==(lhs: EmptyStateIndicatorComponent, rhs: EmptyStateIndicatorComponent) -> Bool { @@ -54,6 +60,9 @@ public final class EmptyStateIndicatorComponent: Component { if lhs.actionTitle != rhs.actionTitle { return false } + if lhs.additionalActionTitle != rhs.additionalActionTitle { + return false + } return true } @@ -65,6 +74,7 @@ public final class EmptyStateIndicatorComponent: Component { private let title = ComponentView() private let text = ComponentView() private var button: ComponentView? + private var additionalButton: ComponentView? override public init(frame: CGRect) { super.init(frame: frame) @@ -139,7 +149,7 @@ public final class EmptyStateIndicatorComponent: Component { } )), environment: {}, - containerSize: CGSize(width: 240.0, height: 50.0) + containerSize: CGSize(width: 260.0, height: 50.0) ) } else { if let button = self.button { @@ -148,14 +158,52 @@ public final class EmptyStateIndicatorComponent: Component { } } + var additionalButtonSize: CGSize? + if let additionalActionTitle = component.additionalActionTitle { + let additionalButton: ComponentView + if let current = self.additionalButton { + additionalButton = current + } else { + additionalButton = ComponentView() + self.additionalButton = additionalButton + } + + additionalButtonSize = additionalButton.update( + transition: transition, + component: AnyComponent(Button( + content: AnyComponent(Text( + text: additionalActionTitle, font: + Font.regular(17.0), + color: component.theme.list.itemAccentColor) + ), + action: { [weak self] in + guard let self, let component = self.component else { + return + } + component.additionalAction() + } + )), + environment: {}, + containerSize: CGSize(width: 262.0, height: 50.0) + ) + } else { + if let additionalButton = self.additionalButton { + self.additionalButton = nil + additionalButton.view?.removeFromSuperview() + } + } + let animationSpacing: CGFloat = 11.0 let titleSpacing: CGFloat = 17.0 - let buttonSpacing: CGFloat = 17.0 + let buttonSpacing: CGFloat = 21.0 var totalHeight: CGFloat = animationSize.height + animationSpacing + titleSize.height + titleSpacing + textSize.height if let buttonSize { totalHeight += buttonSpacing + buttonSize.height } + if let additionalButtonSize { + totalHeight += buttonSpacing + additionalButtonSize.height + } var contentY = floor((availableSize.height - totalHeight) * 0.5) @@ -185,7 +233,14 @@ public final class EmptyStateIndicatorComponent: Component { self.addSubview(buttonView) } transition.setFrame(view: buttonView, frame: CGRect(origin: CGPoint(x: floor((availableSize.width - buttonSize.width) * 0.5), y: contentY), size: buttonSize)) - contentY += buttonSize.height + contentY += buttonSize.height + buttonSpacing + } + if let additionalButtonSize, let additionalButtonView = self.additionalButton?.view { + if additionalButtonView.superview == nil { + self.addSubview(additionalButtonView) + } + transition.setFrame(view: additionalButtonView, frame: CGRect(origin: CGPoint(x: floor((availableSize.width - additionalButtonSize.width) * 0.5), y: contentY), size: additionalButtonSize)) + contentY += additionalButtonSize.height } return availableSize diff --git a/submodules/TelegramUI/Components/EntityKeyboard/Sources/EmojiPagerContentComponent.swift b/submodules/TelegramUI/Components/EntityKeyboard/Sources/EmojiPagerContentComponent.swift index 13dc762e1f6..7c141f89eee 100644 --- a/submodules/TelegramUI/Components/EntityKeyboard/Sources/EmojiPagerContentComponent.swift +++ b/submodules/TelegramUI/Components/EntityKeyboard/Sources/EmojiPagerContentComponent.swift @@ -2519,6 +2519,7 @@ public final class EmojiPagerContentComponent: Component { case premiumStar case topic(String, Int32) case stop + case add } case animation(EntityKeyboardAnimationData) @@ -3559,6 +3560,15 @@ public final class EmojiPagerContentComponent: Component { let imageSize = image.size.aspectFitted(CGSize(width: size.width - 6.0, height: size.height - 6.0)) image.draw(in: CGRect(origin: CGPoint(x: floor((size.width - imageSize.width) / 2.0), y: floor((size.height - imageSize.height) / 2.0)), size: imageSize)) } + case .add: + context.setFillColor(UIColor.black.withAlphaComponent(0.08).cgColor) + context.fillEllipse(in: CGRect(origin: .zero, size: size).insetBy(dx: 8.0, dy: 8.0)) + context.setFillColor(UIColor.black.withAlphaComponent(0.16).cgColor) + + let plusSize = CGSize(width: 4.5, height: 31.5) + context.addPath(UIBezierPath(roundedRect: CGRect(x: floorToScreenPixels((size.width - plusSize.width) / 2.0), y: floorToScreenPixels((size.height - plusSize.height) / 2.0), width: plusSize.width, height: plusSize.height), cornerRadius: plusSize.width / 2.0).cgPath) + context.addPath(UIBezierPath(roundedRect: CGRect(x: floorToScreenPixels((size.width - plusSize.height) / 2.0), y: floorToScreenPixels((size.height - plusSize.width) / 2.0), width: plusSize.height, height: plusSize.width), cornerRadius: plusSize.width / 2.0).cgPath) + context.fillPath() } UIGraphicsPopContext() diff --git a/submodules/TelegramUI/Components/EntityKeyboard/Sources/EmojiPagerContentSignals.swift b/submodules/TelegramUI/Components/EntityKeyboard/Sources/EmojiPagerContentSignals.swift index 7f7ec18d10c..4fbf2390c14 100644 --- a/submodules/TelegramUI/Components/EntityKeyboard/Sources/EmojiPagerContentSignals.swift +++ b/submodules/TelegramUI/Components/EntityKeyboard/Sources/EmojiPagerContentSignals.swift @@ -1784,6 +1784,7 @@ public extension EmojiPagerContentComponent { } if let recentStickers = recentStickers { + let groupId = "recent" for item in recentStickers.items { guard let item = item.contents.get(RecentMediaItem.self) else { continue @@ -1807,7 +1808,6 @@ public extension EmojiPagerContentComponent { tintMode: tintMode ) - let groupId = "recent" if let groupIndex = itemGroupIndexById[groupId] { itemGroups[groupIndex].items.append(resultItem) } else { diff --git a/submodules/TelegramUI/Components/LegacyCamera/Sources/LegacyCamera.swift b/submodules/TelegramUI/Components/LegacyCamera/Sources/LegacyCamera.swift index a9c14e92e52..17eccfc3ea3 100644 --- a/submodules/TelegramUI/Components/LegacyCamera/Sources/LegacyCamera.swift +++ b/submodules/TelegramUI/Components/LegacyCamera/Sources/LegacyCamera.swift @@ -10,7 +10,7 @@ import ShareController import LegacyUI import LegacyMediaPickerUI -public func presentedLegacyCamera(context: AccountContext, peer: Peer, chatLocation: ChatLocation, cameraView: TGAttachmentCameraView?, menuController: TGMenuSheetController?, parentController: ViewController, attachmentController: ViewController? = nil, editingMedia: Bool, saveCapturedPhotos: Bool, mediaGrouping: Bool, initialCaption: NSAttributedString, hasSchedule: Bool, enablePhoto: Bool, enableVideo: Bool, sendMessagesWithSignals: @escaping ([Any]?, Bool, Int32) -> Void, recognizedQRCode: @escaping (String) -> Void = { _ in }, presentSchedulePicker: @escaping (Bool, @escaping (Int32) -> Void) -> Void, presentTimerPicker: @escaping (@escaping (Int32) -> Void) -> Void, getCaptionPanelView: @escaping () -> TGCaptionPanelView?, dismissedWithResult: @escaping () -> Void = {}, finishedTransitionIn: @escaping () -> Void = {}) { +public func presentedLegacyCamera(context: AccountContext, peer: Peer?, chatLocation: ChatLocation, cameraView: TGAttachmentCameraView?, menuController: TGMenuSheetController?, parentController: ViewController, attachmentController: ViewController? = nil, editingMedia: Bool, saveCapturedPhotos: Bool, mediaGrouping: Bool, initialCaption: NSAttributedString, hasSchedule: Bool, enablePhoto: Bool, enableVideo: Bool, sendMessagesWithSignals: @escaping ([Any]?, Bool, Int32) -> Void, recognizedQRCode: @escaping (String) -> Void = { _ in }, presentSchedulePicker: @escaping (Bool, @escaping (Int32) -> Void) -> Void, presentTimerPicker: @escaping (@escaping (Int32) -> Void) -> Void, getCaptionPanelView: @escaping () -> TGCaptionPanelView?, dismissedWithResult: @escaping () -> Void = {}, finishedTransitionIn: @escaping () -> Void = {}) { let presentationData = context.sharedContext.currentPresentationData.with { $0 } let legacyController = LegacyController(presentation: .custom, theme: presentationData.theme) legacyController.supportedOrientations = ViewControllerSupportedOrientations(regularSize: .portrait, compactSize: .portrait) @@ -18,7 +18,7 @@ public func presentedLegacyCamera(context: AccountContext, peer: Peer, chatLocat legacyController.deferScreenEdgeGestures = [.top] - let isSecretChat = peer.id.namespace == Namespaces.Peer.SecretChat + let isSecretChat = peer?.id.namespace == Namespaces.Peer.SecretChat let controller: TGCameraController if let cameraView = cameraView, let previewView = cameraView.previewView() { @@ -87,15 +87,18 @@ public func presentedLegacyCamera(context: AccountContext, peer: Peer, chatLocat controller.allowCaptionEntities = true controller.allowGrouping = mediaGrouping controller.inhibitDocumentCaptions = false - controller.recipientName = EnginePeer(peer).displayTitle(strings: presentationData.strings, displayOrder: presentationData.nameDisplayOrder) - if peer.id != context.account.peerId { - if peer is TelegramUser { - controller.hasTimer = hasSchedule + + if let peer { + controller.recipientName = EnginePeer(peer).displayTitle(strings: presentationData.strings, displayOrder: presentationData.nameDisplayOrder) + if peer.id != context.account.peerId { + if peer is TelegramUser { + controller.hasTimer = hasSchedule + } + controller.hasSilentPosting = true } - controller.hasSilentPosting = true } controller.hasSchedule = hasSchedule - controller.reminder = peer.id == context.account.peerId + controller.reminder = peer?.id == context.account.peerId let screenSize = parentController.view.bounds.size var startFrame = CGRect(x: 0, y: screenSize.height, width: screenSize.width, height: screenSize.height) diff --git a/submodules/TelegramUI/Components/LegacyMessageInputPanel/Sources/LegacyMessageInputPanel.swift b/submodules/TelegramUI/Components/LegacyMessageInputPanel/Sources/LegacyMessageInputPanel.swift index 2df3092fd4a..2472f96dfca 100644 --- a/submodules/TelegramUI/Components/LegacyMessageInputPanel/Sources/LegacyMessageInputPanel.swift +++ b/submodules/TelegramUI/Components/LegacyMessageInputPanel/Sources/LegacyMessageInputPanel.swift @@ -57,7 +57,7 @@ public class LegacyMessageInputPanelNode: ASDisplayNode, TGCaptionPanelView { super.init() - self.state._updated = { [weak self] transition in + self.state._updated = { [weak self] transition, _ in if let self { self.update(transition: transition.containedViewLayoutTransition) } diff --git a/submodules/TelegramUI/Components/ListActionItemComponent/Sources/ListActionItemComponent.swift b/submodules/TelegramUI/Components/ListActionItemComponent/Sources/ListActionItemComponent.swift index b6c169f3886..0b0fe45934f 100644 --- a/submodules/TelegramUI/Components/ListActionItemComponent/Sources/ListActionItemComponent.swift +++ b/submodules/TelegramUI/Components/ListActionItemComponent/Sources/ListActionItemComponent.swift @@ -7,28 +7,84 @@ import ListSectionComponent import SwitchNode public final class ListActionItemComponent: Component { + public enum ToggleStyle { + case regular + case icons + } + + public struct Toggle: Equatable { + public var style: ToggleStyle + public var isOn: Bool + public var isInteractive: Bool + public var action: ((Bool) -> Void)? + + public init(style: ToggleStyle, isOn: Bool, isInteractive: Bool = true, action: ((Bool) -> Void)? = nil) { + self.style = style + self.isOn = isOn + self.isInteractive = isInteractive + self.action = action + } + + public static func ==(lhs: Toggle, rhs: Toggle) -> Bool { + if lhs.style != rhs.style { + return false + } + if lhs.isOn != rhs.isOn { + return false + } + if lhs.isInteractive != rhs.isInteractive { + return false + } + if (lhs.action == nil) != (rhs.action == nil) { + return false + } + return true + } + } + public enum Accessory: Equatable { case arrow - case toggle(Bool) + case toggle(Toggle) + case activity + } + + public enum IconInsets: Equatable { + case `default` + case custom(UIEdgeInsets) + } + + public struct Icon: Equatable { + public var component: AnyComponentWithIdentity + public var insets: IconInsets + public var allowUserInteraction: Bool + + public init(component: AnyComponentWithIdentity, insets: IconInsets = .default, allowUserInteraction: Bool = false) { + self.component = component + self.insets = insets + self.allowUserInteraction = allowUserInteraction + } } public let theme: PresentationTheme public let title: AnyComponent + public let contentInsets: UIEdgeInsets public let leftIcon: AnyComponentWithIdentity? - public let icon: AnyComponentWithIdentity? + public let icon: Icon? public let accessory: Accessory? public let action: ((UIView) -> Void)? public init( theme: PresentationTheme, title: AnyComponent, + contentInsets: UIEdgeInsets = UIEdgeInsets(top: 12.0, left: 0.0, bottom: 12.0, right: 0.0), leftIcon: AnyComponentWithIdentity? = nil, - icon: AnyComponentWithIdentity? = nil, + icon: Icon? = nil, accessory: Accessory? = .arrow, action: ((UIView) -> Void)? ) { self.theme = theme self.title = title + self.contentInsets = contentInsets self.leftIcon = leftIcon self.icon = icon self.accessory = accessory @@ -42,6 +98,9 @@ public final class ListActionItemComponent: Component { if lhs.title != rhs.title { return false } + if lhs.contentInsets != rhs.contentInsets { + return false + } if lhs.leftIcon != rhs.leftIcon { return false } @@ -63,7 +122,9 @@ public final class ListActionItemComponent: Component { private var icon: ComponentView? private var arrowView: UIImageView? - private var switchNode: IconSwitchNode? + private var switchNode: SwitchNode? + private var iconSwitchNode: IconSwitchNode? + private var activityIndicatorView: UIActivityIndicatorView? private var component: ListActionItemComponent? @@ -83,7 +144,10 @@ public final class ListActionItemComponent: Component { self.addTarget(self, action: #selector(self.pressed), for: .touchUpInside) self.internalHighligthedChanged = { [weak self] isHighlighted in - guard let self else { + guard let self, let component = self.component, component.action != nil else { + return + } + if case .toggle = component.accessory, component.action == nil { return } if let customUpdateIsHighlighted = self.customUpdateIsHighlighted { @@ -97,32 +161,44 @@ public final class ListActionItemComponent: Component { } @objc private func pressed() { - self.component?.action?(self) + guard let component, let action = component.action else { + return + } + action(self) + } + + override public func hitTest(_ point: CGPoint, with event: UIEvent?) -> UIView? { + guard let result = super.hitTest(point, with: event) else { + return nil + } + return result } func update(component: ListActionItemComponent, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { let previousComponent = self.component self.component = component - self.isEnabled = component.action != nil - let themeUpdated = component.theme !== previousComponent?.theme - let verticalInset: CGFloat = 12.0 - var contentLeftInset: CGFloat = 16.0 let contentRightInset: CGFloat switch component.accessory { case .none: - contentRightInset = 16.0 + if let _ = component.icon { + contentRightInset = 42.0 + } else { + contentRightInset = 16.0 + } case .arrow: contentRightInset = 30.0 case .toggle: - contentRightInset = 42.0 + contentRightInset = 76.0 + case .activity: + contentRightInset = 76.0 } var contentHeight: CGFloat = 0.0 - contentHeight += verticalInset + contentHeight += component.contentInsets.top if component.leftIcon != nil { contentLeftInset += 46.0 @@ -134,7 +210,7 @@ public final class ListActionItemComponent: Component { environment: {}, containerSize: CGSize(width: availableSize.width - contentLeftInset - contentRightInset, height: availableSize.height) ) - let titleFrame = CGRect(origin: CGPoint(x: contentLeftInset, y: verticalInset), size: titleSize) + let titleFrame = CGRect(origin: CGPoint(x: contentLeftInset, y: contentHeight), size: titleSize) if let titleView = self.title.view { if titleView.superview == nil { titleView.isUserInteractionEnabled = false @@ -144,10 +220,10 @@ public final class ListActionItemComponent: Component { } contentHeight += titleSize.height - contentHeight += verticalInset + contentHeight += component.contentInsets.bottom if let iconValue = component.icon { - if previousComponent?.icon?.id != iconValue.id, let icon = self.icon { + if previousComponent?.icon?.component.id != iconValue.component.id, let icon = self.icon { self.icon = nil if let iconView = icon.view { transition.setAlpha(view: iconView, alpha: 0.0, completion: { [weak iconView] _ in @@ -168,17 +244,23 @@ public final class ListActionItemComponent: Component { let iconSize = icon.update( transition: iconTransition, - component: iconValue.component, + component: iconValue.component.component, environment: {}, containerSize: CGSize(width: availableSize.width, height: availableSize.height) ) - let iconFrame = CGRect(origin: CGPoint(x: availableSize.width - contentRightInset - iconSize.width, y: floor((contentHeight - iconSize.height) * 0.5)), size: iconSize) + + var iconOffset: CGFloat = 0.0 + if case .none = component.accessory { + iconOffset = 26.0 + } + + let iconFrame = CGRect(origin: CGPoint(x: availableSize.width - contentRightInset - iconSize.width + iconOffset, y: floor((contentHeight - iconSize.height) * 0.5)), size: iconSize) if let iconView = icon.view { if iconView.superview == nil { - iconView.isUserInteractionEnabled = false self.addSubview(iconView) transition.animateAlpha(view: iconView, from: 0.0, to: 1.0) } + iconView.isUserInteractionEnabled = iconValue.allowUserInteraction iconTransition.setFrame(view: iconView, frame: iconFrame) } } else { @@ -243,15 +325,16 @@ public final class ListActionItemComponent: Component { var arrowTransition = transition if let current = self.arrowView { arrowView = current + if themeUpdated { + arrowView.image = PresentationResourcesItemList.disclosureArrowImage(component.theme) + } } else { arrowTransition = arrowTransition.withAnimation(.none) - arrowView = UIImageView(image: PresentationResourcesItemList.disclosureArrowImage(component.theme)?.withRenderingMode(.alwaysTemplate)) + arrowView = UIImageView(image: PresentationResourcesItemList.disclosureArrowImage(component.theme)) self.arrowView = arrowView self.addSubview(arrowView) } - - arrowView.tintColor = component.theme.list.disclosureArrowColor - + if let image = arrowView.image { let arrowFrame = CGRect(origin: CGPoint(x: availableSize.width - 7.0 - image.size.width, y: floor((contentHeight - image.size.height) * 0.5)), size: image.size) arrowTransition.setFrame(view: arrowView, frame: arrowFrame) @@ -263,32 +346,85 @@ public final class ListActionItemComponent: Component { } } - if case let .toggle(isOn) = component.accessory { - let switchNode: IconSwitchNode - var switchTransition = transition - var updateSwitchTheme = themeUpdated - if let current = self.switchNode { - switchNode = current - switchNode.setOn(isOn, animated: !transition.animation.isImmediate) - } else { - switchTransition = switchTransition.withAnimation(.none) - updateSwitchTheme = true - switchNode = IconSwitchNode() - switchNode.setOn(isOn, animated: false) - self.addSubview(switchNode.view) - } - - if updateSwitchTheme { - switchNode.frameColor = component.theme.list.itemSwitchColors.frameColor - switchNode.contentColor = component.theme.list.itemSwitchColors.contentColor - switchNode.handleColor = component.theme.list.itemSwitchColors.handleColor - switchNode.positiveContentColor = component.theme.list.itemSwitchColors.positiveColor - switchNode.negativeContentColor = component.theme.list.itemSwitchColors.negativeColor + if case let .toggle(toggle) = component.accessory { + switch toggle.style { + case .regular: + let switchNode: SwitchNode + var switchTransition = transition + var updateSwitchTheme = themeUpdated + if let current = self.switchNode { + switchNode = current + switchNode.setOn(toggle.isOn, animated: !transition.animation.isImmediate) + } else { + switchTransition = switchTransition.withAnimation(.none) + updateSwitchTheme = true + switchNode = SwitchNode() + switchNode.setOn(toggle.isOn, animated: false) + self.switchNode = switchNode + self.addSubview(switchNode.view) + + switchNode.valueUpdated = { [weak self] value in + guard let self, let component = self.component else { + return + } + if case let .toggle(toggle) = component.accessory, let action = toggle.action { + action(value) + } else { + component.action?(self) + } + } + } + switchNode.isUserInteractionEnabled = toggle.isInteractive + + if updateSwitchTheme { + switchNode.frameColor = component.theme.list.itemSwitchColors.frameColor + switchNode.contentColor = component.theme.list.itemSwitchColors.contentColor + switchNode.handleColor = component.theme.list.itemSwitchColors.handleColor + } + + let switchSize = CGSize(width: 51.0, height: 31.0) + let switchFrame = CGRect(origin: CGPoint(x: availableSize.width - 16.0 - switchSize.width, y: floor((min(60.0, contentHeight) - switchSize.height) * 0.5)), size: switchSize) + switchTransition.setFrame(view: switchNode.view, frame: switchFrame) + case .icons: + let switchNode: IconSwitchNode + var switchTransition = transition + var updateSwitchTheme = themeUpdated + if let current = self.iconSwitchNode { + switchNode = current + switchNode.setOn(toggle.isOn, animated: !transition.animation.isImmediate) + } else { + switchTransition = switchTransition.withAnimation(.none) + updateSwitchTheme = true + switchNode = IconSwitchNode() + switchNode.setOn(toggle.isOn, animated: false) + self.iconSwitchNode = switchNode + self.addSubview(switchNode.view) + + switchNode.valueUpdated = { [weak self] value in + guard let self, let component = self.component else { + return + } + if case let .toggle(toggle) = component.accessory, let action = toggle.action { + action(value) + } else { + component.action?(self) + } + } + } + switchNode.isUserInteractionEnabled = toggle.isInteractive + + if updateSwitchTheme { + switchNode.frameColor = component.theme.list.itemSwitchColors.frameColor + switchNode.contentColor = component.theme.list.itemSwitchColors.contentColor + switchNode.handleColor = component.theme.list.itemSwitchColors.handleColor + switchNode.positiveContentColor = component.theme.list.itemSwitchColors.positiveColor + switchNode.negativeContentColor = component.theme.list.itemSwitchColors.negativeColor + } + + let switchSize = CGSize(width: 51.0, height: 31.0) + let switchFrame = CGRect(origin: CGPoint(x: availableSize.width - 16.0 - switchSize.width, y: floor((min(60.0, contentHeight) - switchSize.height) * 0.5)), size: switchSize) + switchTransition.setFrame(view: switchNode.view, frame: switchFrame) } - - let switchSize = CGSize(width: 51.0, height: 31.0) - let switchFrame = CGRect(origin: CGPoint(x: availableSize.width - 16.0 - switchSize.width, y: floor((min(60.0, contentHeight) - switchSize.height) * 0.5)), size: switchSize) - switchTransition.setFrame(view: switchNode.view, frame: switchFrame) } else { if let switchNode = self.switchNode { self.switchNode = nil @@ -296,6 +432,40 @@ public final class ListActionItemComponent: Component { } } + if case .activity = component.accessory { + let activityIndicatorView: UIActivityIndicatorView + var activityIndicatorTransition = transition + if let current = self.activityIndicatorView { + activityIndicatorView = current + } else { + activityIndicatorTransition = activityIndicatorTransition.withAnimation(.none) + if #available(iOS 13.0, *) { + activityIndicatorView = UIActivityIndicatorView(style: .medium) + } else { + activityIndicatorView = UIActivityIndicatorView(style: .gray) + } + self.activityIndicatorView = activityIndicatorView + self.addSubview(activityIndicatorView) + activityIndicatorView.sizeToFit() + } + + let activityIndicatorSize = activityIndicatorView.bounds.size + let activityIndicatorFrame = CGRect(origin: CGPoint(x: availableSize.width - 16.0 - activityIndicatorSize.width, y: floor((min(60.0, contentHeight) - activityIndicatorSize.height) * 0.5)), size: activityIndicatorSize) + + activityIndicatorView.tintColor = component.theme.list.itemPrimaryTextColor.withMultipliedAlpha(0.5) + + activityIndicatorTransition.setFrame(view: activityIndicatorView, frame: activityIndicatorFrame) + + if !activityIndicatorView.isAnimating { + activityIndicatorView.startAnimating() + } + } else { + if let activityIndicatorView = self.activityIndicatorView { + self.activityIndicatorView = nil + activityIndicatorView.removeFromSuperview() + } + } + self.separatorInset = contentLeftInset return CGSize(width: availableSize.width, height: contentHeight) diff --git a/submodules/TelegramUI/Components/ListItemSliderSelectorComponent/BUILD b/submodules/TelegramUI/Components/ListItemSliderSelectorComponent/BUILD new file mode 100644 index 00000000000..9e20c794e64 --- /dev/null +++ b/submodules/TelegramUI/Components/ListItemSliderSelectorComponent/BUILD @@ -0,0 +1,23 @@ +load("@build_bazel_rules_swift//swift:swift.bzl", "swift_library") + +swift_library( + name = "ListItemSliderSelectorComponent", + module_name = "ListItemSliderSelectorComponent", + srcs = glob([ + "Sources/**/*.swift", + ]), + copts = [ + "-warnings-as-errors", + ], + deps = [ + "//submodules/Display", + "//submodules/TelegramPresentationData", + "//submodules/ComponentFlow", + "//submodules/Components/MultilineTextComponent", + "//submodules/TelegramUI/Components/SliderComponent", + "//submodules/TelegramUI/Components/ListSectionComponent", + ], + visibility = [ + "//visibility:public", + ], +) diff --git a/submodules/TelegramUI/Components/ListItemSliderSelectorComponent/Sources/ListItemSliderSelectorComponent.swift b/submodules/TelegramUI/Components/ListItemSliderSelectorComponent/Sources/ListItemSliderSelectorComponent.swift new file mode 100644 index 00000000000..80b67d862a0 --- /dev/null +++ b/submodules/TelegramUI/Components/ListItemSliderSelectorComponent/Sources/ListItemSliderSelectorComponent.swift @@ -0,0 +1,151 @@ +import Foundation +import UIKit +import Display +import AsyncDisplayKit +import TelegramPresentationData +import ComponentFlow +import MultilineTextComponent +import ListSectionComponent +import SliderComponent + +public final class ListItemSliderSelectorComponent: Component { + public let theme: PresentationTheme + public let values: [String] + public let selectedIndex: Int + public let selectedIndexUpdated: (Int) -> Void + + public init( + theme: PresentationTheme, + values: [String], + selectedIndex: Int, + selectedIndexUpdated: @escaping (Int) -> Void + ) { + self.theme = theme + self.values = values + self.selectedIndex = selectedIndex + self.selectedIndexUpdated = selectedIndexUpdated + } + + public static func ==(lhs: ListItemSliderSelectorComponent, rhs: ListItemSliderSelectorComponent) -> Bool { + if lhs.theme !== rhs.theme { + return false + } + if lhs.values != rhs.values { + return false + } + if lhs.selectedIndex != rhs.selectedIndex { + return false + } + return true + } + + public final class View: UIView, ListSectionComponent.ChildView { + private var titles: [ComponentView] = [] + private var slider = ComponentView() + + private var component: ListItemSliderSelectorComponent? + private weak var state: EmptyComponentState? + + public var customUpdateIsHighlighted: ((Bool) -> Void)? + public private(set) var separatorInset: CGFloat = 0.0 + + override public init(frame: CGRect) { + super.init(frame: frame) + } + + required public init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + func update(component: ListItemSliderSelectorComponent, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + self.component = component + self.state = state + + let sideInset: CGFloat = 13.0 + let titleSideInset: CGFloat = 20.0 + let titleClippingSideInset: CGFloat = 14.0 + + let titleAreaWidth: CGFloat = availableSize.width - titleSideInset * 2.0 + + for i in 0 ..< component.values.count { + var titleTransition = transition + let title: ComponentView + if self.titles.count > i { + title = self.titles[i] + } else { + titleTransition = titleTransition.withAnimation(.none) + title = ComponentView() + self.titles.append(title) + } + let titleSize = title.update( + transition: .immediate, + component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: component.values[i], font: Font.regular(13.0), textColor: component.theme.list.itemSecondaryTextColor)) + )), + environment: {}, + containerSize: CGSize(width: 100.0, height: 100.0) + ) + var titleFrame = CGRect(origin: CGPoint(x: titleSideInset - floor(titleSize.width * 0.5), y: 14.0), size: titleSize) + if component.values.count > 1 { + titleFrame.origin.x += floor(CGFloat(i) / CGFloat(component.values.count - 1) * titleAreaWidth) + } + if titleFrame.minX < titleClippingSideInset { + titleFrame.origin.x = titleSideInset + } + if titleFrame.maxX > availableSize.width - titleClippingSideInset { + titleFrame.origin.x = availableSize.width - titleClippingSideInset - titleSize.width + } + if let titleView = title.view { + if titleView.superview == nil { + self.addSubview(titleView) + } + titleView.bounds = CGRect(origin: CGPoint(), size: titleFrame.size) + titleTransition.setPosition(view: titleView, position: titleFrame.center) + } + } + if self.titles.count > component.values.count { + for i in component.values.count ..< self.titles.count { + self.titles[i].view?.removeFromSuperview() + } + self.titles.removeLast(self.titles.count - component.values.count) + } + + let sliderSize = self.slider.update( + transition: transition, + component: AnyComponent(SliderComponent( + valueCount: component.values.count, + value: component.selectedIndex, + trackBackgroundColor: component.theme.list.controlSecondaryColor, + trackForegroundColor: component.theme.list.itemAccentColor, + valueUpdated: { [weak self] value in + guard let self, let component = self.component else { + return + } + component.selectedIndexUpdated(value) + } + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 100.0) + ) + let sliderFrame = CGRect(origin: CGPoint(x: sideInset, y: 36.0), size: sliderSize) + if let sliderView = self.slider.view { + if sliderView.superview == nil { + self.addSubview(sliderView) + } + transition.setFrame(view: sliderView, frame: sliderFrame) + } + + self.separatorInset = 16.0 + + return CGSize(width: availableSize.width, height: 88.0) + } + } + + public func makeView() -> View { + return View(frame: CGRect()) + } + + public func update(view: View, availableSize: CGSize, state: State, environment: Environment, transition: Transition) -> CGSize { + return view.update(component: self, availableSize: availableSize, state: state, environment: environment, transition: transition) + } +} diff --git a/submodules/TelegramUI/Components/ListItemSwipeOptionContainer/BUILD b/submodules/TelegramUI/Components/ListItemSwipeOptionContainer/BUILD new file mode 100644 index 00000000000..c5634af4055 --- /dev/null +++ b/submodules/TelegramUI/Components/ListItemSwipeOptionContainer/BUILD @@ -0,0 +1,21 @@ +load("@build_bazel_rules_swift//swift:swift.bzl", "swift_library") + +swift_library( + name = "ListItemSwipeOptionContainer", + module_name = "ListItemSwipeOptionContainer", + srcs = glob([ + "Sources/**/*.swift", + ]), + copts = [ + "-warnings-as-errors", + ], + deps = [ + "//submodules/Display", + "//submodules/ComponentFlow", + "//submodules/Components/MultilineTextComponent", + "//submodules/Components/ComponentDisplayAdapters", + ], + visibility = [ + "//visibility:public", + ], +) diff --git a/submodules/TelegramUI/Components/ListItemSwipeOptionContainer/Sources/ListItemSwipeOptionContainer.swift b/submodules/TelegramUI/Components/ListItemSwipeOptionContainer/Sources/ListItemSwipeOptionContainer.swift new file mode 100644 index 00000000000..eeec875a32a --- /dev/null +++ b/submodules/TelegramUI/Components/ListItemSwipeOptionContainer/Sources/ListItemSwipeOptionContainer.swift @@ -0,0 +1,905 @@ +import Foundation +import UIKit +import Display +import ComponentFlow +import ComponentDisplayAdapters +import AppBundle +import MultilineTextComponent + +private let titleFontWithIcon = Font.medium(13.0) +private let titleFontWithoutIcon = Font.regular(17.0) + +private final class SwipeOptionsGestureRecognizer: UIPanGestureRecognizer { + public var validatedGesture = false + public var firstLocation: CGPoint = CGPoint() + + public var allowAnyDirection = false + public var lastVelocity: CGPoint = CGPoint() + + override public init(target: Any?, action: Selector?) { + super.init(target: target, action: action) + + if #available(iOS 13.4, *) { + self.allowedScrollTypesMask = .continuous + } + + self.maximumNumberOfTouches = 1 + } + + override public func reset() { + super.reset() + + self.validatedGesture = false + } + + public func becomeCancelled() { + self.state = .cancelled + } + + override public func touchesBegan(_ touches: Set, with event: UIEvent) { + super.touchesBegan(touches, with: event) + + let touch = touches.first! + self.firstLocation = touch.location(in: self.view) + } + + override public func touchesMoved(_ touches: Set, with event: UIEvent) { + let location = touches.first!.location(in: self.view) + let translation = CGPoint(x: location.x - self.firstLocation.x, y: location.y - self.firstLocation.y) + + if !self.validatedGesture { + if !self.allowAnyDirection && translation.x > 0.0 { + self.state = .failed + } else if abs(translation.y) > 4.0 && abs(translation.y) > abs(translation.x) * 2.5 { + self.state = .failed + } else if abs(translation.x) > 4.0 && abs(translation.y) * 2.5 < abs(translation.x) { + self.validatedGesture = true + } + } + + if self.validatedGesture { + self.lastVelocity = self.velocity(in: self.view) + super.touchesMoved(touches, with: event) + } + } +} + +open class ListItemSwipeOptionContainer: UIView, UIGestureRecognizerDelegate { + public struct Option: Equatable { + public enum Icon: Equatable { + case none + case image(image: UIImage) + + public static func ==(lhs: Icon, rhs: Icon) -> Bool { + switch lhs { + case .none: + if case .none = rhs { + return true + } else { + return false + } + case let .image(lhsImage): + if case let .image(rhsImage) = rhs, lhsImage == rhsImage { + return true + } else { + return false + } + } + } + } + + public let key: AnyHashable + public let title: String + public let icon: Icon + public let color: UIColor + public let textColor: UIColor + + public init(key: AnyHashable, title: String, icon: Icon, color: UIColor, textColor: UIColor) { + self.key = key + self.title = title + self.icon = icon + self.color = color + self.textColor = textColor + } + + public static func ==(lhs: Option, rhs: Option) -> Bool { + if lhs.key != rhs.key { + return false + } + if lhs.title != rhs.title { + return false + } + if !lhs.color.isEqual(rhs.color) { + return false + } + if !lhs.textColor.isEqual(rhs.textColor) { + return false + } + if lhs.icon != rhs.icon { + return false + } + return true + } + } + + private enum OptionAlignment { + case left + case right + } + + private final class OptionView: UIView { + private let backgroundView: UIView + private let title = ComponentView() + private var iconView: UIImageView? + + private let titleString: String + private let textColor: UIColor + + private var titleSize: CGSize? + + var alignment: OptionAlignment? + var isExpanded: Bool = false + + init(title: String, icon: Option.Icon, color: UIColor, textColor: UIColor) { + self.titleString = title + self.textColor = textColor + + self.backgroundView = UIView() + + switch icon { + case let .image(image): + let iconView = UIImageView() + iconView.image = image.withRenderingMode(.alwaysTemplate) + iconView.tintColor = textColor + self.iconView = iconView + case .none: + self.iconView = nil + } + + super.init(frame: CGRect()) + + self.addSubview(self.backgroundView) + if let iconView = self.iconView { + self.addSubview(iconView) + } + self.backgroundView.backgroundColor = color + } + + required init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + func updateLayout( + isFirst: Bool, + isLeft: Bool, + baseSize: CGSize, + alignment: OptionAlignment, + isExpanded: Bool, + extendedWidth: CGFloat, + sideInset: CGFloat, + transition: Transition, + additive: Bool, + revealFactor: CGFloat, + animateIconMovement: Bool + ) { + var animateAdditive = false + if additive && !transition.animation.isImmediate && self.isExpanded != isExpanded { + animateAdditive = true + } + + let backgroundFrame: CGRect + if isFirst { + backgroundFrame = CGRect(origin: CGPoint(x: isLeft ? -400.0 : 0.0, y: 0.0), size: CGSize(width: extendedWidth + 400.0, height: baseSize.height)) + } else { + backgroundFrame = CGRect(origin: CGPoint(x: 0.0, y: 0.0), size: CGSize(width: extendedWidth, height: baseSize.height)) + } + let deltaX: CGFloat + if animateAdditive { + let previousFrame = self.backgroundView.frame + self.backgroundView.frame = backgroundFrame + if isLeft { + deltaX = previousFrame.width - backgroundFrame.width + } else { + deltaX = -(previousFrame.width - backgroundFrame.width) + } + if !animateIconMovement { + transition.animatePosition(view: self.backgroundView, from: CGPoint(x: deltaX, y: 0.0), to: CGPoint(), additive: true) + } + } else { + deltaX = 0.0 + transition.setFrame(view: self.backgroundView, frame: backgroundFrame) + } + + self.alignment = alignment + self.isExpanded = isExpanded + let titleSize = self.titleSize ?? CGSize(width: 32.0, height: 10.0) + var contentRect = CGRect(origin: CGPoint(), size: baseSize) + switch alignment { + case .left: + contentRect.origin.x = 0.0 + case .right: + contentRect.origin.x = extendedWidth - contentRect.width + } + + if let iconView = self.iconView, let imageSize = iconView.image?.size { + let iconOffset: CGFloat = -9.0 + let titleIconSpacing: CGFloat = 11.0 + let iconFrame = CGRect(origin: CGPoint(x: contentRect.minX + floor((baseSize.width - imageSize.width + sideInset) / 2.0), y: contentRect.midY - imageSize.height / 2.0 + iconOffset), size: imageSize) + if animateAdditive { + let iconOffsetX = animateIconMovement ? iconView.frame.minX - iconFrame.minX : deltaX + iconView.frame = iconFrame + transition.animatePosition(view: iconView, from: CGPoint(x: iconOffsetX, y: 0.0), to: CGPoint(), additive: true) + } else { + transition.setFrame(view: iconView, frame: iconFrame) + } + + let titleFrame = CGRect(origin: CGPoint(x: contentRect.minX + floor((baseSize.width - titleSize.width + sideInset) / 2.0), y: contentRect.midY + titleIconSpacing), size: titleSize) + if let titleView = self.title.view { + if titleView.superview == nil { + self.addSubview(titleView) + } + if animateAdditive { + let titleOffsetX = animateIconMovement ? titleView.frame.minX - titleFrame.minX : deltaX + titleView.frame = titleFrame + transition.animatePosition(view: titleView, from: CGPoint(x: titleOffsetX, y: 0.0), to: CGPoint(), additive: true) + } else { + transition.setFrame(view: titleView, frame: titleFrame) + } + } + } else { + let titleFrame = CGRect(origin: CGPoint(x: contentRect.minX + floor((baseSize.width - titleSize.width + sideInset) / 2.0), y: contentRect.minY + floor((baseSize.height - titleSize.height) / 2.0)), size: titleSize) + + if let titleView = self.title.view { + if titleView.superview == nil { + self.addSubview(titleView) + } + if animateAdditive { + let titleOffsetX = animateIconMovement ? titleView.frame.minX - titleFrame.minX : deltaX + titleView.frame = titleFrame + transition.animatePosition(view: titleView, from: CGPoint(x: titleOffsetX, y: 0.0), to: CGPoint(), additive: true) + } else { + transition.setFrame(view: titleView, frame: titleFrame) + } + } + } + } + + func calculateSize(_ constrainedSize: CGSize) -> CGSize { + let titleSize = self.title.update( + transition: .immediate, + component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: self.titleString, font: self.iconView == nil ? titleFontWithoutIcon : titleFontWithIcon, textColor: self.textColor)) + )), + environment: {}, + containerSize: CGSize(width: 200.0, height: 100.0) + ) + self.titleSize = titleSize + + var maxWidth = titleSize.width + if let iconView = self.iconView, let image = iconView.image { + maxWidth = max(image.size.width, maxWidth) + } + return CGSize(width: max(74.0, maxWidth + 20.0), height: constrainedSize.height) + } + } + + public final class OptionsView: UIView { + private let optionSelected: (Option) -> Void + private let tapticAction: () -> Void + + private var options: [Option] = [] + private var isLeft: Bool = false + + private var optionViews: [OptionView] = [] + private var revealOffset: CGFloat = 0.0 + private var sideInset: CGFloat = 0.0 + + public init(optionSelected: @escaping (Option) -> Void, tapticAction: @escaping () -> Void) { + self.optionSelected = optionSelected + self.tapticAction = tapticAction + + super.init(frame: CGRect()) + + let gestureRecognizer = TapLongTapOrDoubleTapGestureRecognizer(target: self, action: #selector(self.tapGesture(_:))) + gestureRecognizer.tapActionAtPoint = { _ in + return .waitForSingleTap + } + self.addGestureRecognizer(gestureRecognizer) + } + + required public init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + public func setOptions(_ options: [Option], isLeft: Bool) { + if self.options != options || self.isLeft != isLeft { + self.options = options + self.isLeft = isLeft + for optionView in self.optionViews { + optionView.removeFromSuperview() + } + self.optionViews = options.map { option in + return OptionView(title: option.title, icon: option.icon, color: option.color, textColor: option.textColor) + } + if isLeft { + for optionView in self.optionViews.reversed() { + self.addSubview(optionView) + } + } else { + for optionView in self.optionViews { + self.addSubview(optionView) + } + } + } + } + + func calculateSize(_ constrainedSize: CGSize) -> CGSize { + var maxWidth: CGFloat = 0.0 + for optionView in self.optionViews { + let nodeSize = optionView.calculateSize(constrainedSize) + maxWidth = max(nodeSize.width, maxWidth) + } + return CGSize(width: maxWidth * CGFloat(self.optionViews.count), height: constrainedSize.height) + } + + public func updateRevealOffset(offset: CGFloat, sideInset: CGFloat, transition: Transition) { + self.revealOffset = offset + self.sideInset = sideInset + self.updateNodesLayout(transition: transition) + } + + private func updateNodesLayout(transition: Transition) { + let size = self.bounds.size + if size.width.isLessThanOrEqualTo(0.0) || self.optionViews.isEmpty { + return + } + let basicNodeWidth = floor((size.width - abs(self.sideInset)) / CGFloat(self.optionViews.count)) + let lastNodeWidth = size.width - basicNodeWidth * CGFloat(self.optionViews.count - 1) + let revealFactor = self.revealOffset / size.width + let boundaryRevealFactor: CGFloat + if self.optionViews.count > 2 { + boundaryRevealFactor = 1.0 + 16.0 / size.width + } else { + boundaryRevealFactor = 1.0 + basicNodeWidth / size.width + } + let startingOffset: CGFloat + if self.isLeft { + startingOffset = size.width + max(0.0, abs(revealFactor) - 1.0) * size.width + } else { + startingOffset = 0.0 + } + + var completionCount = self.optionViews.count + let intermediateCompletion = { + } + + var i = self.isLeft ? (self.optionViews.count - 1) : 0 + while i >= 0 && i < self.optionViews.count { + let optionView = self.optionViews[i] + let nodeWidth = i == (self.optionViews.count - 1) ? lastNodeWidth : basicNodeWidth + var nodeTransition = transition + var isExpanded = false + if (self.isLeft && i == 0) || (!self.isLeft && i == self.optionViews.count - 1) { + if abs(revealFactor) > boundaryRevealFactor { + isExpanded = true + } + } + if let _ = optionView.alignment, optionView.isExpanded != isExpanded { + nodeTransition = !transition.animation.isImmediate ? transition : .easeInOut(duration: 0.2) + if transition.animation.isImmediate { + self.tapticAction() + } + } + + var sideInset: CGFloat = 0.0 + if i == self.optionViews.count - 1 { + sideInset = self.sideInset + } + + let extendedWidth: CGFloat + let nodeLeftOffset: CGFloat + if isExpanded { + nodeLeftOffset = 0.0 + extendedWidth = size.width * max(1.0, abs(revealFactor)) + } else if self.isLeft { + let offset = basicNodeWidth * CGFloat(self.optionViews.count - 1 - i) + extendedWidth = (size.width - offset) * max(1.0, abs(revealFactor)) + nodeLeftOffset = startingOffset - extendedWidth - floorToScreenPixels(offset * abs(revealFactor)) + } else { + let offset = basicNodeWidth * CGFloat(i) + extendedWidth = (size.width - offset) * max(1.0, abs(revealFactor)) + nodeLeftOffset = startingOffset + floorToScreenPixels(offset * abs(revealFactor)) + } + + transition.setFrame(view: optionView, frame: CGRect(origin: CGPoint(x: nodeLeftOffset, y: 0.0), size: CGSize(width: extendedWidth, height: size.height)), completion: { _ in + completionCount -= 1 + intermediateCompletion() + }) + + var nodeAlignment: OptionAlignment + if (self.optionViews.count > 1) { + nodeAlignment = self.isLeft ? .right : .left + } else { + if self.isLeft { + nodeAlignment = isExpanded ? .right : .left + } else { + nodeAlignment = isExpanded ? .left : .right + } + } + let animateIconMovement = self.optionViews.count == 1 + optionView.updateLayout(isFirst: (self.isLeft && i == 0) || (!self.isLeft && i == self.optionViews.count - 1), isLeft: self.isLeft, baseSize: CGSize(width: nodeWidth, height: size.height), alignment: nodeAlignment, isExpanded: isExpanded, extendedWidth: extendedWidth, sideInset: sideInset, transition: nodeTransition, additive: transition.animation.isImmediate, revealFactor: revealFactor, animateIconMovement: animateIconMovement) + + if self.isLeft { + i -= 1 + } else { + i += 1 + } + } + } + + @objc private func tapGesture(_ recognizer: TapLongTapOrDoubleTapGestureRecognizer) { + if case .ended = recognizer.state, let gesture = recognizer.lastRecognizedGestureAndLocation?.0, case .tap = gesture { + let location = recognizer.location(in: self) + var selectedOption: Int? + + var i = self.isLeft ? 0 : (self.optionViews.count - 1) + while i >= 0 && i < self.optionViews.count { + if self.optionViews[i].frame.contains(location) { + selectedOption = i + break + } + if self.isLeft { + i += 1 + } else { + i -= 1 + } + } + if let selectedOption { + self.optionSelected(self.options[selectedOption]) + } + } + } + + public func isDisplayingExtendedAction() -> Bool { + return self.optionViews.contains(where: { $0.isExpanded }) + } + } + + private var validLayout: (size: CGSize, leftInset: CGFloat, reftInset: CGFloat)? + + private var leftRevealView: OptionsView? + private var rightRevealView: OptionsView? + private var revealOptions: (left: [Option], right: [Option]) = ([], []) + + private var initialRevealOffset: CGFloat = 0.0 + public private(set) var revealOffset: CGFloat = 0.0 + + private var recognizer: SwipeOptionsGestureRecognizer? + private var tapRecognizer: UITapGestureRecognizer? + private var hapticFeedback: HapticFeedback? + + private var allowAnyDirection: Bool = false + + public var updateRevealOffset: ((CGFloat, Transition) -> Void)? + public var revealOptionsInteractivelyOpened: (() -> Void)? + public var revealOptionsInteractivelyClosed: (() -> Void)? + public var revealOptionSelected: ((Option, Bool) -> Void)? + + open var controlsContainer: UIView { + return self + } + + public var isDisplayingRevealedOptions: Bool { + return !self.revealOffset.isZero + } + + override public init(frame: CGRect) { + super.init(frame: frame) + + let recognizer = SwipeOptionsGestureRecognizer(target: self, action: #selector(self.revealGesture(_:))) + self.recognizer = recognizer + recognizer.delegate = self + recognizer.allowAnyDirection = self.allowAnyDirection + self.addGestureRecognizer(recognizer) + + let tapRecognizer = UITapGestureRecognizer(target: self, action: #selector(self.revealTapGesture(_:))) + self.tapRecognizer = tapRecognizer + tapRecognizer.delegate = self + self.addGestureRecognizer(tapRecognizer) + + self.disablesInteractiveTransitionGestureRecognizer = self.allowAnyDirection + + self.disablesInteractiveTransitionGestureRecognizerNow = { [weak self] in + guard let self else { + return false + } + if !self.revealOffset.isZero { + return true + } + return false + } + } + + required public init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + open func setRevealOptions(_ options: (left: [Option], right: [Option])) { + if self.revealOptions == options { + return + } + let previousOptions = self.revealOptions + let wasEmpty = self.revealOptions.left.isEmpty && self.revealOptions.right.isEmpty + self.revealOptions = options + let isEmpty = options.left.isEmpty && options.right.isEmpty + if options.left.isEmpty { + if let _ = self.leftRevealView { + self.recognizer?.becomeCancelled() + self.updateRevealOffsetInternal(offset: 0.0, transition: .spring(duration: 0.3)) + } + } else if previousOptions.left != options.left { + } + if options.right.isEmpty { + if let _ = self.rightRevealView { + self.recognizer?.becomeCancelled() + self.updateRevealOffsetInternal(offset: 0.0, transition: .spring(duration: 0.3)) + } + } else if previousOptions.right != options.right { + if let _ = self.rightRevealView { + } + } + if wasEmpty != isEmpty { + self.recognizer?.isEnabled = !isEmpty + } + let allowAnyDirection = !options.left.isEmpty || !self.revealOffset.isZero + if allowAnyDirection != self.allowAnyDirection { + self.allowAnyDirection = allowAnyDirection + self.recognizer?.allowAnyDirection = allowAnyDirection + self.disablesInteractiveTransitionGestureRecognizer = allowAnyDirection + } + } + + override open func gestureRecognizerShouldBegin(_ gestureRecognizer: UIGestureRecognizer) -> Bool { + if let recognizer = self.recognizer, gestureRecognizer == self.tapRecognizer { + return abs(self.revealOffset) > 0.0 && !recognizer.validatedGesture + } else if let recognizer = self.recognizer, gestureRecognizer == self.recognizer, recognizer.numberOfTouches == 0 { + let translation = recognizer.velocity(in: recognizer.view) + if abs(translation.y) > 4.0 && abs(translation.y) > abs(translation.x) * 2.5 { + return false + } + } + return true + } + + open func gestureRecognizer(_ gestureRecognizer: UIGestureRecognizer, shouldRecognizeSimultaneouslyWith otherGestureRecognizer: UIGestureRecognizer) -> Bool { + if let recognizer = self.recognizer, otherGestureRecognizer == recognizer { + return true + } else { + return false + } + } + + open func gestureRecognizer(_ gestureRecognizer: UIGestureRecognizer, shouldBeRequiredToFailBy otherGestureRecognizer: UIGestureRecognizer) -> Bool { + /*if gestureRecognizer === self.recognizer && otherGestureRecognizer is InteractiveTransitionGestureRecognizer { + return true + }*/ + return false + } + + @objc private func revealTapGesture(_ recognizer: UITapGestureRecognizer) { + if case .ended = recognizer.state { + self.updateRevealOffsetInternal(offset: 0.0, transition: .spring(duration: 0.3)) + self.revealOptionsInteractivelyClosed?() + } + } + + @objc private func revealGesture(_ recognizer: SwipeOptionsGestureRecognizer) { + guard let (size, _, _) = self.validLayout else { + return + } + switch recognizer.state { + case .began: + if let leftRevealView = self.leftRevealView { + let revealSize = leftRevealView.bounds.size + let location = recognizer.location(in: self) + if location.x < revealSize.width { + recognizer.becomeCancelled() + } else { + self.initialRevealOffset = self.revealOffset + } + } else if let rightRevealView = self.rightRevealView { + let revealSize = rightRevealView.bounds.size + let location = recognizer.location(in: self) + if location.x > size.width - revealSize.width { + recognizer.becomeCancelled() + } else { + self.initialRevealOffset = self.revealOffset + } + } else { + if self.revealOptions.left.isEmpty && self.revealOptions.right.isEmpty { + recognizer.becomeCancelled() + } + self.initialRevealOffset = self.revealOffset + } + case .changed: + var translation = recognizer.translation(in: self) + translation.x += self.initialRevealOffset + if self.revealOptions.left.isEmpty { + translation.x = min(0.0, translation.x) + } + if self.leftRevealView == nil && CGFloat(0.0).isLess(than: translation.x) { + self.setupAndAddLeftRevealNode() + self.revealOptionsInteractivelyOpened?() + } else if self.rightRevealView == nil && translation.x.isLess(than: 0.0) { + self.setupAndAddRightRevealNode() + self.revealOptionsInteractivelyOpened?() + } + self.updateRevealOffsetInternal(offset: translation.x, transition: .immediate) + if self.leftRevealView == nil && self.rightRevealView == nil { + self.revealOptionsInteractivelyClosed?() + } + case .ended, .cancelled: + guard let recognizer = self.recognizer else { + break + } + + if let leftRevealView = self.leftRevealView { + let velocity = recognizer.velocity(in: self) + let revealSize = leftRevealView.bounds.size + var reveal = false + if abs(velocity.x) < 100.0 { + if self.initialRevealOffset.isZero && self.revealOffset > 0.0 { + reveal = true + } else if self.revealOffset > revealSize.width { + reveal = true + } else { + reveal = false + } + } else { + if velocity.x > 0.0 { + reveal = true + } else { + reveal = false + } + } + + var selectedOption: Option? + if reveal && leftRevealView.isDisplayingExtendedAction() { + reveal = false + selectedOption = self.revealOptions.left.first + } else { + self.updateRevealOffsetInternal(offset: reveal ? revealSize.width : 0.0, transition: .spring(duration: 0.3)) + } + + if let selectedOption = selectedOption { + self.revealOptionSelected?(selectedOption, true) + } else { + if !reveal { + self.revealOptionsInteractivelyClosed?() + } + } + } else if let rightRevealView = self.rightRevealView { + let velocity = recognizer.velocity(in: self) + let revealSize = rightRevealView.bounds.size + var reveal = false + if abs(velocity.x) < 100.0 { + if self.initialRevealOffset.isZero && self.revealOffset < 0.0 { + reveal = true + } else if self.revealOffset < -revealSize.width { + reveal = true + } else { + reveal = false + } + } else { + if velocity.x < 0.0 { + reveal = true + } else { + reveal = false + } + } + + var selectedOption: Option? + if reveal && rightRevealView.isDisplayingExtendedAction() { + reveal = false + selectedOption = self.revealOptions.right.last + } else { + self.updateRevealOffsetInternal(offset: reveal ? -revealSize.width : 0.0, transition: .spring(duration: 0.3)) + } + + if let selectedOption = selectedOption { + self.revealOptionSelected?(selectedOption, true) + } else { + if !reveal { + self.revealOptionsInteractivelyClosed?() + } + } + } + default: + break + } + } + + private func setupAndAddLeftRevealNode() { + if !self.revealOptions.left.isEmpty { + let revealView = OptionsView(optionSelected: { [weak self] option in + self?.revealOptionSelected?(option, false) + }, tapticAction: { [weak self] in + self?.hapticImpact() + }) + revealView.setOptions(self.revealOptions.left, isLeft: true) + self.leftRevealView = revealView + + if let (size, leftInset, _) = self.validLayout { + var revealSize = revealView.calculateSize(CGSize(width: CGFloat.greatestFiniteMagnitude, height: size.height)) + revealSize.width += leftInset + + revealView.frame = CGRect(origin: CGPoint(x: min(self.revealOffset - revealSize.width, 0.0), y: 0.0), size: revealSize) + revealView.updateRevealOffset(offset: 0.0, sideInset: leftInset, transition: .immediate) + } + + self.controlsContainer.addSubview(revealView) + } + } + + private func setupAndAddRightRevealNode() { + if !self.revealOptions.right.isEmpty { + let revealView = OptionsView(optionSelected: { [weak self] option in + self?.revealOptionSelected?(option, false) + }, tapticAction: { [weak self] in + self?.hapticImpact() + }) + revealView.setOptions(self.revealOptions.right, isLeft: false) + self.rightRevealView = revealView + + if let (size, _, rightInset) = self.validLayout { + var revealSize = revealView.calculateSize(CGSize(width: CGFloat.greatestFiniteMagnitude, height: size.height)) + revealSize.width += rightInset + + revealView.frame = CGRect(origin: CGPoint(x: size.width + max(self.revealOffset, -revealSize.width), y: 0.0), size: revealSize) + revealView.updateRevealOffset(offset: 0.0, sideInset: -rightInset, transition: .immediate) + } + + self.controlsContainer.addSubview(revealView) + } + } + + public func updateLayout(size: CGSize, leftInset: CGFloat, rightInset: CGFloat) { + self.validLayout = (size, leftInset, rightInset) + + if let leftRevealView = self.leftRevealView { + var revealSize = leftRevealView.calculateSize(CGSize(width: CGFloat.greatestFiniteMagnitude, height: size.height)) + revealSize.width += leftInset + leftRevealView.frame = CGRect(origin: CGPoint(x: min(self.revealOffset - revealSize.width, 0.0), y: 0.0), size: revealSize) + } + + if let rightRevealView = self.rightRevealView { + var revealSize = rightRevealView.calculateSize(CGSize(width: CGFloat.greatestFiniteMagnitude, height: size.height)) + revealSize.width += rightInset + rightRevealView.frame = CGRect(origin: CGPoint(x: size.width + max(self.revealOffset, -revealSize.width), y: 0.0), size: revealSize) + } + } + + open func updateRevealOffsetInternal(offset: CGFloat, transition: Transition, completion: (() -> Void)? = nil) { + self.revealOffset = offset + guard let (size, leftInset, rightInset) = self.validLayout else { + return + } + + var leftRevealCompleted = true + var rightRevealCompleted = true + let intermediateCompletion = { + if leftRevealCompleted && rightRevealCompleted { + completion?() + } + } + + if let leftRevealView = self.leftRevealView { + leftRevealCompleted = false + + let revealSize = leftRevealView.bounds.size + + let revealFrame = CGRect(origin: CGPoint(x: min(self.revealOffset - revealSize.width, 0.0), y: 0.0), size: revealSize) + let revealNodeOffset = -self.revealOffset + leftRevealView.updateRevealOffset(offset: revealNodeOffset, sideInset: leftInset, transition: transition) + + if CGFloat(offset).isLessThanOrEqualTo(0.0) { + self.leftRevealView = nil + transition.setFrame(view: leftRevealView, frame: revealFrame, completion: { [weak leftRevealView] _ in + leftRevealView?.removeFromSuperview() + + leftRevealCompleted = true + intermediateCompletion() + }) + } else { + transition.setFrame(view: leftRevealView, frame: revealFrame, completion: { _ in + leftRevealCompleted = true + intermediateCompletion() + }) + } + } + if let rightRevealView = self.rightRevealView { + rightRevealCompleted = false + + let revealSize = rightRevealView.bounds.size + + let revealFrame = CGRect(origin: CGPoint(x: min(size.width, size.width + self.revealOffset), y: 0.0), size: revealSize) + let revealNodeOffset = -self.revealOffset + rightRevealView.updateRevealOffset(offset: revealNodeOffset, sideInset: -rightInset, transition: transition) + + if CGFloat(0.0).isLessThanOrEqualTo(offset) { + self.rightRevealView = nil + transition.setFrame(view: rightRevealView, frame: revealFrame, completion: { [weak rightRevealView] _ in + rightRevealView?.removeFromSuperview() + + rightRevealCompleted = true + intermediateCompletion() + }) + } else { + transition.setFrame(view: rightRevealView, frame: revealFrame, completion: { _ in + rightRevealCompleted = true + intermediateCompletion() + }) + } + } + let allowAnyDirection = !self.revealOptions.left.isEmpty || !offset.isZero + if allowAnyDirection != self.allowAnyDirection { + self.allowAnyDirection = allowAnyDirection + self.recognizer?.allowAnyDirection = allowAnyDirection + self.disablesInteractiveTransitionGestureRecognizer = allowAnyDirection + } + + self.updateRevealOffset?(offset, transition) + } + + open func setRevealOptionsOpened(_ value: Bool, animated: Bool) { + if value != !self.revealOffset.isZero { + if !self.revealOffset.isZero { + self.recognizer?.becomeCancelled() + } + let transition: Transition + if animated { + transition = .spring(duration: 0.3) + } else { + transition = .immediate + } + if value { + if self.rightRevealView == nil { + self.setupAndAddRightRevealNode() + if let rightRevealView = self.rightRevealView, let validLayout = self.validLayout { + let revealSize = rightRevealView.calculateSize(CGSize(width: CGFloat.greatestFiniteMagnitude, height: validLayout.size.height)) + self.updateRevealOffsetInternal(offset: -revealSize.width, transition: transition) + } + } + } else if !self.revealOffset.isZero { + self.updateRevealOffsetInternal(offset: 0.0, transition: transition) + } + } + } + + open func animateRevealOptionsFill(completion: (() -> Void)? = nil) { + if let validLayout = self.validLayout { + self.layer.allowsGroupOpacity = true + self.updateRevealOffsetInternal(offset: -validLayout.0.width - 74.0, transition: .spring(duration: 0.3), completion: { + self.layer.allowsGroupOpacity = false + completion?() + }) + } + } + + open var preventsTouchesToOtherItems: Bool { + return self.isDisplayingRevealedOptions + } + + open func touchesToOtherItemsPrevented() { + if self.isDisplayingRevealedOptions { + self.setRevealOptionsOpened(false, animated: true) + } + } + + private func hapticImpact() { + if self.hapticFeedback == nil { + self.hapticFeedback = HapticFeedback() + } + self.hapticFeedback?.impact(.medium) + } +} diff --git a/submodules/TelegramUI/Components/ListMultilineTextFieldItemComponent/BUILD b/submodules/TelegramUI/Components/ListMultilineTextFieldItemComponent/BUILD new file mode 100644 index 00000000000..cab35652152 --- /dev/null +++ b/submodules/TelegramUI/Components/ListMultilineTextFieldItemComponent/BUILD @@ -0,0 +1,24 @@ +load("@build_bazel_rules_swift//swift:swift.bzl", "swift_library") + +swift_library( + name = "ListMultilineTextFieldItemComponent", + module_name = "ListMultilineTextFieldItemComponent", + srcs = glob([ + "Sources/**/*.swift", + ]), + copts = [ + "-warnings-as-errors", + ], + deps = [ + "//submodules/Display", + "//submodules/ComponentFlow", + "//submodules/TelegramPresentationData", + "//submodules/Components/MultilineTextComponent", + "//submodules/TelegramUI/Components/ListSectionComponent", + "//submodules/TelegramUI/Components/TextFieldComponent", + "//submodules/AccountContext", + ], + visibility = [ + "//visibility:public", + ], +) diff --git a/submodules/TelegramUI/Components/ListMultilineTextFieldItemComponent/Sources/ListMultilineTextFieldItemComponent.swift b/submodules/TelegramUI/Components/ListMultilineTextFieldItemComponent/Sources/ListMultilineTextFieldItemComponent.swift new file mode 100644 index 00000000000..d2c964545d2 --- /dev/null +++ b/submodules/TelegramUI/Components/ListMultilineTextFieldItemComponent/Sources/ListMultilineTextFieldItemComponent.swift @@ -0,0 +1,281 @@ +import Foundation +import UIKit +import Display +import ComponentFlow +import TelegramPresentationData +import MultilineTextComponent +import ListSectionComponent +import TextFieldComponent +import AccountContext + +public final class ListMultilineTextFieldItemComponent: Component { + public final class ExternalState { + public fileprivate(set) var hasText: Bool = false + + public init() { + } + } + + public final class ResetText: Equatable { + public let value: String + + public init(value: String) { + self.value = value + } + + public static func ==(lhs: ResetText, rhs: ResetText) -> Bool { + return lhs === rhs + } + } + + public let externalState: ExternalState? + public let context: AccountContext + public let theme: PresentationTheme + public let strings: PresentationStrings + public let initialText: String + public let resetText: ResetText? + public let placeholder: String + public let autocapitalizationType: UITextAutocapitalizationType + public let autocorrectionType: UITextAutocorrectionType + public let characterLimit: Int? + public let allowEmptyLines: Bool + public let updated: ((String) -> Void)? + public let textUpdateTransition: Transition + public let tag: AnyObject? + + public init( + externalState: ExternalState? = nil, + context: AccountContext, + theme: PresentationTheme, + strings: PresentationStrings, + initialText: String, + resetText: ResetText? = nil, + placeholder: String, + autocapitalizationType: UITextAutocapitalizationType = .sentences, + autocorrectionType: UITextAutocorrectionType = .default, + characterLimit: Int? = nil, + allowEmptyLines: Bool = true, + updated: ((String) -> Void)?, + textUpdateTransition: Transition = .immediate, + tag: AnyObject? = nil + ) { + self.externalState = externalState + self.context = context + self.theme = theme + self.strings = strings + self.initialText = initialText + self.resetText = resetText + self.placeholder = placeholder + self.autocapitalizationType = autocapitalizationType + self.autocorrectionType = autocorrectionType + self.characterLimit = characterLimit + self.allowEmptyLines = allowEmptyLines + self.updated = updated + self.textUpdateTransition = textUpdateTransition + self.tag = tag + } + + public static func ==(lhs: ListMultilineTextFieldItemComponent, rhs: ListMultilineTextFieldItemComponent) -> Bool { + if lhs.externalState !== rhs.externalState { + return false + } + if lhs.context !== rhs.context { + return false + } + if lhs.theme !== rhs.theme { + return false + } + if lhs.strings !== rhs.strings { + return false + } + if lhs.initialText != rhs.initialText { + return false + } + if lhs.resetText != rhs.resetText { + return false + } + if lhs.placeholder != rhs.placeholder { + return false + } + if lhs.autocapitalizationType != rhs.autocapitalizationType { + return false + } + if lhs.autocorrectionType != rhs.autocorrectionType { + return false + } + if lhs.characterLimit != rhs.characterLimit { + return false + } + if lhs.allowEmptyLines != rhs.allowEmptyLines { + return false + } + if (lhs.updated == nil) != (rhs.updated == nil) { + return false + } + return true + } + + private final class TextField: UITextField { + var sideInset: CGFloat = 0.0 + + override func textRect(forBounds bounds: CGRect) -> CGRect { + return CGRect(origin: CGPoint(x: self.sideInset, y: 0.0), size: CGSize(width: bounds.width - self.sideInset * 2.0, height: bounds.height)) + } + + override func editingRect(forBounds bounds: CGRect) -> CGRect { + return CGRect(origin: CGPoint(x: self.sideInset, y: 0.0), size: CGSize(width: bounds.width - self.sideInset * 2.0, height: bounds.height)) + } + } + + public final class View: UIView, UITextFieldDelegate, ListSectionComponent.ChildView, ComponentTaggedView { + private let textField = ComponentView() + private let textFieldExternalState = TextFieldComponent.ExternalState() + + private let placeholder = ComponentView() + + private var component: ListMultilineTextFieldItemComponent? + private weak var state: EmptyComponentState? + private var isUpdating: Bool = false + + public var currentText: String { + if let textFieldView = self.textField.view as? TextFieldComponent.View { + return textFieldView.inputState.inputText.string + } else { + return "" + } + } + + public var customUpdateIsHighlighted: ((Bool) -> Void)? + public private(set) var separatorInset: CGFloat = 0.0 + + public override init(frame: CGRect) { + super.init(frame: CGRect()) + } + + required public init?(coder: NSCoder) { + preconditionFailure() + } + + public func textFieldShouldReturn(_ textField: UITextField) -> Bool { + return true + } + + @objc private func textDidChange() { + if !self.isUpdating { + self.state?.updated(transition: self.component?.textUpdateTransition ?? .immediate) + } + self.component?.updated?(self.currentText) + } + + public func setText(text: String, updateState: Bool) { + if let textFieldView = self.textField.view as? TextFieldComponent.View { + //TODO + let _ = textFieldView + } + + if updateState { + self.component?.updated?(self.currentText) + } else { + self.state?.updated(transition: .immediate, isLocal: true) + } + } + + public func matches(tag: Any) -> Bool { + if let component = self.component, let componentTag = component.tag { + let tag = tag as AnyObject + if componentTag === tag { + return true + } + } + return false + } + + func update(component: ListMultilineTextFieldItemComponent, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + self.isUpdating = true + defer { + self.isUpdating = false + } + + self.component = component + self.state = state + + let verticalInset: CGFloat = 12.0 + let sideInset: CGFloat = 16.0 + + let textFieldSize = self.textField.update( + transition: transition, + component: AnyComponent(TextFieldComponent( + context: component.context, + strings: component.strings, + externalState: self.textFieldExternalState, + fontSize: 17.0, + textColor: component.theme.list.itemPrimaryTextColor, + insets: UIEdgeInsets(top: verticalInset, left: sideInset - 8.0, bottom: verticalInset, right: sideInset - 8.0), + hideKeyboard: false, + customInputView: nil, + resetText: component.resetText.flatMap { resetText in + return NSAttributedString(string: resetText.value, font: Font.regular(17.0), textColor: component.theme.list.itemPrimaryTextColor) + }, + isOneLineWhenUnfocused: false, + characterLimit: component.characterLimit, + allowEmptyLines: component.allowEmptyLines, + formatMenuAvailability: .none, + lockedFormatAction: { + }, + present: { _ in + }, + paste: { _ in + } + )), + environment: {}, + containerSize: availableSize + ) + + let size = CGSize(width: textFieldSize.width, height: textFieldSize.height - 1.0) + let textFieldFrame = CGRect(origin: CGPoint(), size: textFieldSize) + + if let textFieldView = self.textField.view { + if textFieldView.superview == nil { + self.addSubview(textFieldView) + self.textField.parentState = state + } + transition.setFrame(view: textFieldView, frame: textFieldFrame) + } + + let placeholderSize = self.placeholder.update( + transition: .immediate, + component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: component.placeholder.isEmpty ? " " : component.placeholder, font: Font.regular(17.0), textColor: component.theme.list.itemPlaceholderTextColor)) + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 100.0) + ) + let placeholderFrame = CGRect(origin: CGPoint(x: sideInset, y: verticalInset), size: placeholderSize) + if let placeholderView = self.placeholder.view { + if placeholderView.superview == nil { + placeholderView.layer.anchorPoint = CGPoint() + placeholderView.isUserInteractionEnabled = false + self.insertSubview(placeholderView, at: 0) + } + transition.setPosition(view: placeholderView, position: placeholderFrame.origin) + placeholderView.bounds = CGRect(origin: CGPoint(), size: placeholderFrame.size) + + placeholderView.isHidden = self.textFieldExternalState.hasText + } + + self.separatorInset = 16.0 + + component.externalState?.hasText = self.textFieldExternalState.hasText + + return size + } + } + + public func makeView() -> View { + return View(frame: CGRect()) + } + + public func update(view: View, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + return view.update(component: self, availableSize: availableSize, state: state, environment: environment, transition: transition) + } +} diff --git a/submodules/TelegramUI/Components/ListSectionComponent/Sources/ListSectionComponent.swift b/submodules/TelegramUI/Components/ListSectionComponent/Sources/ListSectionComponent.swift index ce6b71a1ef3..714250a551e 100644 --- a/submodules/TelegramUI/Components/ListSectionComponent/Sources/ListSectionComponent.swift +++ b/submodules/TelegramUI/Components/ListSectionComponent/Sources/ListSectionComponent.swift @@ -217,6 +217,7 @@ public final class ListSectionComponent: Component { itemTransition = itemTransition.withAnimation(.none) itemView = ItemView() self.itemViews[itemId] = itemView + itemView.contents.parentState = state } let itemSize = itemView.contents.update( diff --git a/submodules/TelegramUI/Components/ListTextFieldItemComponent/BUILD b/submodules/TelegramUI/Components/ListTextFieldItemComponent/BUILD index c0256f3093c..bbb2f2118f7 100644 --- a/submodules/TelegramUI/Components/ListTextFieldItemComponent/BUILD +++ b/submodules/TelegramUI/Components/ListTextFieldItemComponent/BUILD @@ -14,6 +14,9 @@ swift_library( "//submodules/ComponentFlow", "//submodules/TelegramPresentationData", "//submodules/Components/MultilineTextComponent", + "//submodules/TelegramUI/Components/ListSectionComponent", + "//submodules/TelegramUI/Components/PlainButtonComponent", + "//submodules/Components/BundleIconComponent", ], visibility = [ "//visibility:public", diff --git a/submodules/TelegramUI/Components/ListTextFieldItemComponent/Sources/ListTextFieldItemComponent.swift b/submodules/TelegramUI/Components/ListTextFieldItemComponent/Sources/ListTextFieldItemComponent.swift index c91f9b94147..3f6e7c42655 100644 --- a/submodules/TelegramUI/Components/ListTextFieldItemComponent/Sources/ListTextFieldItemComponent.swift +++ b/submodules/TelegramUI/Components/ListTextFieldItemComponent/Sources/ListTextFieldItemComponent.swift @@ -4,23 +4,50 @@ import Display import ComponentFlow import TelegramPresentationData import MultilineTextComponent +import ListSectionComponent +import PlainButtonComponent +import BundleIconComponent public final class ListTextFieldItemComponent: Component { + public final class ResetText: Equatable { + public let value: String + + public init(value: String) { + self.value = value + } + + public static func ==(lhs: ResetText, rhs: ResetText) -> Bool { + return lhs === rhs + } + } + public let theme: PresentationTheme public let initialText: String + public let resetText: ResetText? public let placeholder: String + public let autocapitalizationType: UITextAutocapitalizationType + public let autocorrectionType: UITextAutocorrectionType public let updated: ((String) -> Void)? + public let tag: AnyObject? public init( theme: PresentationTheme, initialText: String, + resetText: ResetText? = nil, placeholder: String, - updated: ((String) -> Void)? + autocapitalizationType: UITextAutocapitalizationType = .sentences, + autocorrectionType: UITextAutocorrectionType = .default, + updated: ((String) -> Void)?, + tag: AnyObject? = nil ) { self.theme = theme self.initialText = initialText + self.resetText = resetText self.placeholder = placeholder + self.autocapitalizationType = autocapitalizationType + self.autocorrectionType = autocorrectionType self.updated = updated + self.tag = tag } public static func ==(lhs: ListTextFieldItemComponent, rhs: ListTextFieldItemComponent) -> Bool { @@ -30,9 +57,18 @@ public final class ListTextFieldItemComponent: Component { if lhs.initialText != rhs.initialText { return false } + if lhs.resetText !== rhs.resetText { + return false + } if lhs.placeholder != rhs.placeholder { return false } + if lhs.autocapitalizationType != rhs.autocapitalizationType { + return false + } + if lhs.autocorrectionType != rhs.autocorrectionType { + return false + } if (lhs.updated == nil) != (rhs.updated == nil) { return false } @@ -51,9 +87,10 @@ public final class ListTextFieldItemComponent: Component { } } - public final class View: UIView, UITextFieldDelegate { + public final class View: UIView, UITextFieldDelegate, ListSectionComponent.ChildView, ComponentTaggedView { private let textField: TextField private let placeholder = ComponentView() + private let clearButton = ComponentView() private var component: ListTextFieldItemComponent? private weak var state: EmptyComponentState? @@ -63,6 +100,9 @@ public final class ListTextFieldItemComponent: Component { return self.textField.text ?? "" } + public var customUpdateIsHighlighted: ((Bool) -> Void)? + public private(set) var separatorInset: CGFloat = 0.0 + public override init(frame: CGRect) { self.textField = TextField() @@ -81,6 +121,27 @@ public final class ListTextFieldItemComponent: Component { if !self.isUpdating { self.state?.updated(transition: .immediate) } + self.component?.updated?(self.currentText) + } + + public func setText(text: String, updateState: Bool) { + self.textField.text = text + if updateState { + self.state?.updated(transition: .immediate, isLocal: true) + self.component?.updated?(self.currentText) + } else { + self.state?.updated(transition: .immediate, isLocal: true) + } + } + + public func matches(tag: Any) -> Bool { + if let component = self.component, let componentTag = component.tag { + let tag = tag as AnyObject + if componentTag === tag { + return true + } + } + return false } func update(component: ListTextFieldItemComponent, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { @@ -101,6 +162,16 @@ public final class ListTextFieldItemComponent: Component { self.textField.delegate = self self.textField.addTarget(self, action: #selector(self.textDidChange), for: .editingChanged) } + if let resetText = component.resetText, previousComponent?.resetText !== component.resetText { + self.textField.text = resetText.value + } + + if self.textField.autocapitalizationType != component.autocapitalizationType { + self.textField.autocapitalizationType = component.autocapitalizationType + } + if self.textField.autocorrectionType != component.autocorrectionType { + self.textField.autocorrectionType = component.autocorrectionType + } let themeUpdated = component.theme !== previousComponent?.theme @@ -120,7 +191,7 @@ public final class ListTextFieldItemComponent: Component { text: .plain(NSAttributedString(string: component.placeholder.isEmpty ? " " : component.placeholder, font: Font.regular(17.0), textColor: component.theme.list.itemPlaceholderTextColor)) )), environment: {}, - containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 100.0) + containerSize: CGSize(width: availableSize.width - sideInset * 2.0 - 30.0, height: 100.0) ) let contentHeight: CGFloat = placeholderSize.height + verticalInset * 2.0 let placeholderFrame = CGRect(origin: CGPoint(x: sideInset, y: floor((contentHeight - placeholderSize.height) * 0.5)), size: placeholderSize) @@ -138,6 +209,37 @@ public final class ListTextFieldItemComponent: Component { transition.setFrame(view: self.textField, frame: CGRect(origin: CGPoint(x: 0.0, y: 0.0), size: CGSize(width: availableSize.width, height: contentHeight))) + let clearButtonSize = self.clearButton.update( + transition: transition, + component: AnyComponent(PlainButtonComponent( + content: AnyComponent(BundleIconComponent( + name: "Components/Search Bar/Clear", + tintColor: component.theme.list.itemPrimaryTextColor.withMultipliedAlpha(0.4) + )), + effectAlignment: .center, + minSize: CGSize(width: 44.0, height: 44.0), + action: { [weak self] in + guard let self else { + return + } + self.setText(text: "", updateState: true) + }, + animateAlpha: false, + animateScale: true + )), + environment: {}, + containerSize: CGSize(width: 44.0, height: 44.0) + ) + if let clearButtonView = self.clearButton.view { + if clearButtonView.superview == nil { + self.addSubview(clearButtonView) + } + transition.setFrame(view: clearButtonView, frame: CGRect(origin: CGPoint(x: availableSize.width - 0.0 - clearButtonSize.width, y: floor((contentHeight - clearButtonSize.height) * 0.5)), size: clearButtonSize)) + clearButtonView.isHidden = self.currentText.isEmpty + } + + self.separatorInset = 16.0 + return CGSize(width: availableSize.width, height: contentHeight) } } diff --git a/submodules/TelegramUI/Components/MediaEditor/MetalResources/EditorDefault.metal b/submodules/TelegramUI/Components/MediaEditor/MetalResources/EditorDefault.metal index d03c2f7e8f5..6559709cc87 100644 --- a/submodules/TelegramUI/Components/MediaEditor/MetalResources/EditorDefault.metal +++ b/submodules/TelegramUI/Components/MediaEditor/MetalResources/EditorDefault.metal @@ -25,8 +25,8 @@ vertex RasterizerData defaultVertexShader(uint vertexID [[vertex_id]], fragment half4 defaultFragmentShader(RasterizerData in [[stage_in]], texture2d texture [[texture(0)]]) { constexpr sampler samplr(filter::linear, mag_filter::linear, min_filter::linear); - half3 color = texture.sample(samplr, in.texCoord).rgb; - return half4(color, 1.0); + half4 color = texture.sample(samplr, in.texCoord); + return color; } fragment half histogramPrepareFragmentShader(RasterizerData in [[stage_in]], @@ -39,12 +39,12 @@ fragment half histogramPrepareFragmentShader(RasterizerData in [[stage_in]], } typedef struct { - float3 topColor; - float3 bottomColor; + float4 topColor; + float4 bottomColor; } GradientColors; fragment half4 gradientFragmentShader(RasterizerData in [[stage_in]], constant GradientColors& colors [[buffer(0)]]) { - return half4(half3(mix(colors.topColor, colors.bottomColor, in.texCoord.y)), 1.0); + return half4(half3(mix(colors.topColor.rgb, colors.bottomColor.rgb, in.texCoord.y)), 1.0); } diff --git a/submodules/TelegramUI/Components/MediaEditor/MetalResources/EditorDual.metal b/submodules/TelegramUI/Components/MediaEditor/MetalResources/EditorDual.metal index c45b5ec9ffc..a4fa037c62d 100644 --- a/submodules/TelegramUI/Components/MediaEditor/MetalResources/EditorDual.metal +++ b/submodules/TelegramUI/Components/MediaEditor/MetalResources/EditorDual.metal @@ -3,6 +3,14 @@ using namespace metal; +typedef struct { + float2 dimensions; + float roundness; + float alpha; + float isOpaque; + float empty; +} VideoEncodeParameters; + typedef struct { float4 pos; float2 texCoord; @@ -17,11 +25,10 @@ float sdfRoundedRectangle(float2 uv, float2 position, float2 size, float radius) fragment half4 dualFragmentShader(RasterizerData in [[stage_in]], texture2d texture [[texture(0)]], - constant uint2 &resolution[[buffer(0)]], - constant float &roundness[[buffer(1)]], - constant float &alpha[[buffer(2)]] + texture2d mask [[texture(1)]], + constant VideoEncodeParameters& adjustments [[buffer(0)]] ) { - float2 R = float2(resolution.x, resolution.y); + float2 R = float2(adjustments.dimensions.x, adjustments.dimensions.y); float2 uv = (in.localPos - float2(0.5, 0.5)) * 2.0; if (R.x > R.y) { @@ -33,10 +40,11 @@ fragment half4 dualFragmentShader(RasterizerData in [[stage_in]], constexpr sampler samplr(filter::linear, mag_filter::linear, min_filter::linear); half3 color = texture.sample(samplr, in.texCoord).rgb; + float colorAlpha = min(1.0, adjustments.isOpaque + mask.sample(samplr, in.texCoord).r); - float t = 1.0 / resolution.y; + float t = 1.0 / adjustments.dimensions.y; float side = 1.0 * aspectRatio; - float distance = smoothstep(t, -t, sdfRoundedRectangle(uv, float2(0.0, 0.0), float2(side, mix(1.0, side, roundness)), side * roundness)); + float distance = smoothstep(t, -t, sdfRoundedRectangle(uv, float2(0.0, 0.0), float2(side, mix(1.0, side, adjustments.roundness)), side * adjustments.roundness)); - return mix(half4(color, 0.0), half4(color, 1.0 * alpha), distance); + return mix(half4(color, 0.0), half4(color, colorAlpha * adjustments.alpha), distance); } diff --git a/submodules/TelegramUI/Components/MediaEditor/Sources/ImageObjectSeparation.swift b/submodules/TelegramUI/Components/MediaEditor/Sources/ImageObjectSeparation.swift new file mode 100644 index 00000000000..fe2f1100f02 --- /dev/null +++ b/submodules/TelegramUI/Components/MediaEditor/Sources/ImageObjectSeparation.swift @@ -0,0 +1,153 @@ +import Foundation +import UIKit +import Vision +import CoreImage +import CoreImage.CIFilterBuiltins +import SwiftSignalKit +import VideoToolbox + +private let queue = Queue() + +public func cutoutStickerImage(from image: UIImage, onlyCheck: Bool = false) -> Signal { + if #available(iOS 17.0, *) { + guard let cgImage = image.cgImage else { + return .single(nil) + } + return Signal { subscriber in + let ciContext = CIContext(options: nil) + let inputImage = CIImage(cgImage: cgImage) + let handler = VNImageRequestHandler(cgImage: cgImage, options: [:]) + let request = VNGenerateForegroundInstanceMaskRequest { [weak handler] request, error in + guard let handler, let result = request.results?.first as? VNInstanceMaskObservation else { + subscriber.putNext(nil) + subscriber.putCompletion() + return + } + if onlyCheck { + subscriber.putNext(UIImage()) + subscriber.putCompletion() + } else { + let instances = instances(atPoint: nil, inObservation: result) + if let mask = try? result.generateScaledMaskForImage(forInstances: instances, from: handler) { + let filter = CIFilter.blendWithMask() + filter.inputImage = inputImage + filter.backgroundImage = CIImage(color: .clear) + filter.maskImage = CIImage(cvPixelBuffer: mask) + if let output = filter.outputImage, let cgImage = ciContext.createCGImage(output, from: inputImage.extent) { + let image = UIImage(cgImage: cgImage) + subscriber.putNext(image) + subscriber.putCompletion() + return + } + } + subscriber.putNext(nil) + subscriber.putCompletion() + } + } + try? handler.perform([request]) + return ActionDisposable { + request.cancel() + } + } + |> runOn(queue) + } else { + return .single(nil) + } +} + +public enum CutoutResult { + case image(UIImage) + case pixelBuffer(CVPixelBuffer) +} + +public func cutoutImage(from image: UIImage, atPoint point: CGPoint?, asImage: Bool) -> Signal { + if #available(iOS 17.0, *) { + guard let cgImage = image.cgImage else { + return .single(nil) + } + return Signal { subscriber in + let ciContext = CIContext(options: nil) + let inputImage = CIImage(cgImage: cgImage) + let handler = VNImageRequestHandler(cgImage: cgImage, options: [:]) + let request = VNGenerateForegroundInstanceMaskRequest { [weak handler] request, error in + guard let handler, let result = request.results?.first as? VNInstanceMaskObservation else { + subscriber.putNext(nil) + subscriber.putCompletion() + return + } + + let instances = IndexSet(instances(atPoint: point, inObservation: result).prefix(1)) + if let mask = try? result.generateScaledMaskForImage(forInstances: instances, from: handler) { + if asImage { + let filter = CIFilter.blendWithMask() + filter.inputImage = inputImage + filter.backgroundImage = CIImage(color: .clear) + filter.maskImage = CIImage(cvPixelBuffer: mask) + if let output = filter.outputImage, let cgImage = ciContext.createCGImage(output, from: inputImage.extent) { + let image = UIImage(cgImage: cgImage) + subscriber.putNext(.image(image)) + subscriber.putCompletion() + return + } + } else { + let filter = CIFilter.blendWithMask() + filter.inputImage = CIImage(color: .white) + filter.backgroundImage = CIImage(color: .black) + filter.maskImage = CIImage(cvPixelBuffer: mask) + if let output = filter.outputImage, let cgImage = ciContext.createCGImage(output, from: inputImage.extent) { + let image = UIImage(cgImage: cgImage) + subscriber.putNext(.image(image)) + subscriber.putCompletion() + return + } +// subscriber.putNext(.pixelBuffer(mask)) +// subscriber.putCompletion() + } + } + subscriber.putNext(nil) + subscriber.putCompletion() + } + + try? handler.perform([request]) + return ActionDisposable { + request.cancel() + } + } + |> runOn(queue) + } else { + return .single(nil) + } +} + +@available(iOS 17.0, *) +private func instances(atPoint maybePoint: CGPoint?, inObservation observation: VNInstanceMaskObservation) -> IndexSet { + guard let point = maybePoint else { + return observation.allInstances + } + + let instanceMap = observation.instanceMask + let coords = VNImagePointForNormalizedPoint(point, CVPixelBufferGetWidth(instanceMap) - 1, CVPixelBufferGetHeight(instanceMap) - 1) + + CVPixelBufferLockBaseAddress(instanceMap, .readOnly) + guard let pixels = CVPixelBufferGetBaseAddress(instanceMap) else { + fatalError() + } + let bytesPerRow = CVPixelBufferGetBytesPerRow(instanceMap) + let instanceLabel = pixels.load(fromByteOffset: Int(coords.y) * bytesPerRow + Int(coords.x), as: UInt8.self) + CVPixelBufferUnlockBaseAddress(instanceMap, .readOnly) + + return instanceLabel == 0 ? observation.allInstances : [Int(instanceLabel)] +} + +private extension UIImage { + convenience init?(pixelBuffer: CVPixelBuffer) { + var cgImage: CGImage? + VTCreateCGImageFromCVPixelBuffer(pixelBuffer, options: nil, imageOut: &cgImage) + + guard let cgImage = cgImage else { + return nil + } + + self.init(cgImage: cgImage) + } +} diff --git a/submodules/TelegramUI/Components/MediaEditor/Sources/MediaEditor.swift b/submodules/TelegramUI/Components/MediaEditor/Sources/MediaEditor.swift index 6dba11a9ea3..183b2405e3b 100644 --- a/submodules/TelegramUI/Components/MediaEditor/Sources/MediaEditor.swift +++ b/submodules/TelegramUI/Components/MediaEditor/Sources/MediaEditor.swift @@ -94,6 +94,11 @@ public final class MediaEditor { } } + public enum Mode { + case `default` + case sticker + } + public enum Subject { case image(UIImage, PixelDimensions) case video(String, UIImage?, Bool, String?, PixelDimensions, Double) @@ -116,6 +121,7 @@ public final class MediaEditor { } private let context: AccountContext + private let mode: Mode private let subject: Subject private let clock = CMClockGetHostTimeClock() @@ -182,6 +188,10 @@ public final class MediaEditor { } } + public private(set) var canCutout: Bool = false + public var canCutoutUpdated: (Bool) -> Void = { _ in } + public var isCutoutUpdated: (Bool) -> Void = { _ in } + private var textureCache: CVMetalTextureCache! public var hasPortraitMask: Bool { @@ -391,8 +401,9 @@ public final class MediaEditor { } } - public init(context: AccountContext, subject: Subject, values: MediaEditorValues? = nil, hasHistogram: Bool = false) { + public init(context: AccountContext, mode: Mode, subject: Subject, values: MediaEditorValues? = nil, hasHistogram: Bool = false) { self.context = context + self.mode = mode self.subject = subject if let values { self.values = values @@ -668,6 +679,19 @@ public final class MediaEditor { } else { textureSource.setMainInput(.image(image)) } + + + if case .sticker = self.mode { + let _ = (cutoutStickerImage(from: image, onlyCheck: true) + |> deliverOnMainQueue).start(next: { [weak self] result in + guard let self, result != nil else { + return + } + self.canCutout = true + self.canCutoutUpdated(true) + }) + } + } if let player, let playerItem = player.currentItem, !textureSourceResult.playerIsReference { textureSource.setMainInput(.video(playerItem)) @@ -677,7 +701,12 @@ public final class MediaEditor { } self.renderer.textureSource = textureSource - self.setGradientColors(textureSourceResult.gradientColors) + switch self.mode { + case .default: + self.setGradientColors(textureSourceResult.gradientColors) + case .sticker: + self.setGradientColors(GradientColors(top: .clear, bottom: .clear)) + } if let _ = textureSourceResult.player { self.updateRenderChain() @@ -1615,7 +1644,7 @@ public final class MediaEditor { public func setGradientColors(_ gradientColors: GradientColors) { self.gradientColorsPromise.set(.single(gradientColors)) - self.updateValues(mode: .skipRendering) { values in + self.updateValues(mode: self.sourceIsVideo ? .skipRendering : .generic) { values in return values.withUpdatedGradientColors(gradientColors: gradientColors.array) } } @@ -1654,6 +1683,51 @@ public final class MediaEditor { self.renderer.renderFrame() } + public func getSeparatedImage(point: CGPoint?) -> Signal { + guard let textureSource = self.renderer.textureSource as? UniversalTextureSource, let image = textureSource.mainImage else { + return .single(nil) + } + return cutoutImage(from: image, atPoint: point, asImage: true) + |> map { result in + if let result, case let .image(image) = result { + return image + } else { + return nil + } + } + } + + public func removeSeparationMask() { + self.isCutoutUpdated(false) + + self.renderer.currentMainInputMask = nil + if !self.skipRendering { + self.updateRenderChain() + } + } + + public func setSeparationMask(point: CGPoint?) { + guard let renderTarget = self.previewView, let device = renderTarget.mtlDevice else { + return + } + guard let textureSource = self.renderer.textureSource as? UniversalTextureSource, let image = textureSource.mainImage else { + return + } + self.isCutoutUpdated(true) + + let _ = (cutoutImage(from: image, atPoint: point, asImage: false) + |> deliverOnMainQueue).start(next: { [weak self] result in + guard let self, let result, case let .image(image) = result else { + return + } + //TODO:replace with pixelbuffer + self.renderer.currentMainInputMask = loadTexture(image: image, device: device) + if !self.skipRendering { + self.updateRenderChain() + } + }) + } + private func maybeGeneratePersonSegmentation(_ image: UIImage?) { if #available(iOS 15.0, *), let cgImage = image?.cgImage { let faceRequest = VNDetectFaceRectanglesRequest { [weak self] request, _ in diff --git a/submodules/TelegramUI/Components/MediaEditor/Sources/MediaEditorComposer.swift b/submodules/TelegramUI/Components/MediaEditor/Sources/MediaEditorComposer.swift index eb276b8a087..d97aa11d7d0 100644 --- a/submodules/TelegramUI/Components/MediaEditor/Sources/MediaEditorComposer.swift +++ b/submodules/TelegramUI/Components/MediaEditor/Sources/MediaEditorComposer.swift @@ -178,7 +178,7 @@ final class MediaEditorComposer { } } -public func makeEditorImageComposition(context: CIContext, postbox: Postbox, inputImage: UIImage, dimensions: CGSize, values: MediaEditorValues, time: CMTime, textScale: CGFloat, completion: @escaping (UIImage?) -> Void) { +public func makeEditorImageComposition(context: CIContext, postbox: Postbox, inputImage: UIImage, dimensions: CGSize, outputDimensions: CGSize? = nil, values: MediaEditorValues, time: CMTime, textScale: CGFloat, completion: @escaping (UIImage?) -> Void) { let colorSpace = CGColorSpaceCreateDeviceRGB() let inputImage = CIImage(image: inputImage, options: [.colorSpace: colorSpace])! var drawingImage: CIImage? @@ -192,7 +192,7 @@ public func makeEditorImageComposition(context: CIContext, postbox: Postbox, inp entities.append(contentsOf: composerEntitiesForDrawingEntity(postbox: postbox, textScale: textScale, entity: entity.entity, colorSpace: colorSpace)) } - makeEditorImageFrameComposition(context: context, inputImage: inputImage, drawingImage: drawingImage, dimensions: dimensions, outputDimensions: dimensions, values: values, entities: entities, time: time, textScale: textScale, completion: { ciImage in + makeEditorImageFrameComposition(context: context, inputImage: inputImage, drawingImage: drawingImage, dimensions: dimensions, outputDimensions: outputDimensions ?? dimensions, values: values, entities: entities, time: time, textScale: textScale, completion: { ciImage in if let ciImage { if let cgImage = context.createCGImage(ciImage, from: CGRect(origin: .zero, size: ciImage.extent.size)) { Queue.mainQueue().async { @@ -206,20 +206,19 @@ public func makeEditorImageComposition(context: CIContext, postbox: Postbox, inp } private func makeEditorImageFrameComposition(context: CIContext, inputImage: CIImage, drawingImage: CIImage?, dimensions: CGSize, outputDimensions: CGSize, values: MediaEditorValues, entities: [MediaEditorComposerEntity], time: CMTime, textScale: CGFloat = 1.0, completion: @escaping (CIImage?) -> Void) { - var resultImage = CIImage(color: .black).cropped(to: CGRect(origin: .zero, size: dimensions)).transformed(by: CGAffineTransform(translationX: -dimensions.width / 2.0, y: -dimensions.height / 2.0)) + var isClear = false + if let gradientColor = values.gradientColors?.first, gradientColor.alpha.isZero { + isClear = true + } - var mediaImage = inputImage.samplingLinear().transformed(by: CGAffineTransform(translationX: -inputImage.extent.midX, y: -inputImage.extent.midY)) + var resultImage = CIImage(color: isClear ? .clear : .black).cropped(to: CGRect(origin: .zero, size: dimensions)).transformed(by: CGAffineTransform(translationX: -dimensions.width / 2.0, y: -dimensions.height / 2.0)) - var initialScale: CGFloat - if mediaImage.extent.height > mediaImage.extent.width && values.isStory { - initialScale = max(dimensions.width / mediaImage.extent.width, dimensions.height / mediaImage.extent.height) - } else { - initialScale = dimensions.width / mediaImage.extent.width - } + var mediaImage = inputImage.samplingLinear().transformed(by: CGAffineTransform(translationX: -inputImage.extent.midX, y: -inputImage.extent.midY)) if values.isStory { resultImage = mediaImage.samplingLinear().composited(over: resultImage) } else { + let initialScale = dimensions.width / mediaImage.extent.width var horizontalScale = initialScale if values.cropMirroring { horizontalScale *= -1.0 diff --git a/submodules/TelegramUI/Components/MediaEditor/Sources/MediaEditorRenderer.swift b/submodules/TelegramUI/Components/MediaEditor/Sources/MediaEditorRenderer.swift index 0242d8dc452..c4bddd47257 100644 --- a/submodules/TelegramUI/Components/MediaEditor/Sources/MediaEditorRenderer.swift +++ b/submodules/TelegramUI/Components/MediaEditor/Sources/MediaEditorRenderer.swift @@ -98,6 +98,7 @@ final class MediaEditorRenderer { } private var currentMainInput: Input? + var currentMainInputMask: MTLTexture? private var currentAdditionalInput: Input? private(set) var resultTexture: MTLTexture? @@ -202,7 +203,7 @@ final class MediaEditorRenderer { } if let mainTexture { - return self.videoFinishPass.process(input: mainTexture, secondInput: additionalTexture, timestamp: mainInput.timestamp, device: device, commandBuffer: commandBuffer) + return self.videoFinishPass.process(input: mainTexture, inputMask: self.currentMainInputMask, secondInput: additionalTexture, timestamp: mainInput.timestamp, device: device, commandBuffer: commandBuffer) } else { return nil } diff --git a/submodules/TelegramUI/Components/MediaEditor/Sources/MediaEditorValues.swift b/submodules/TelegramUI/Components/MediaEditor/Sources/MediaEditorValues.swift index cbd042f9759..fcf114e4dfb 100644 --- a/submodules/TelegramUI/Components/MediaEditor/Sources/MediaEditorValues.swift +++ b/submodules/TelegramUI/Components/MediaEditor/Sources/MediaEditorValues.swift @@ -634,6 +634,10 @@ public final class MediaEditorValues: Codable, Equatable { return MediaEditorValues(peerId: self.peerId, originalDimensions: self.originalDimensions, cropOffset: offset, cropRect: self.cropRect, cropScale: scale, cropRotation: rotation, cropMirroring: mirroring, cropOrientation: self.cropOrientation, gradientColors: self.gradientColors, videoTrimRange: self.videoTrimRange, videoIsMuted: self.videoIsMuted, videoIsFullHd: self.videoIsFullHd, videoIsMirrored: self.videoIsMirrored, videoVolume: self.videoVolume, additionalVideoPath: self.additionalVideoPath, additionalVideoIsDual: self.additionalVideoIsDual, additionalVideoPosition: self.additionalVideoPosition, additionalVideoScale: self.additionalVideoScale, additionalVideoRotation: self.additionalVideoRotation, additionalVideoPositionChanges: self.additionalVideoPositionChanges, additionalVideoTrimRange: self.additionalVideoTrimRange, additionalVideoOffset: self.additionalVideoOffset, additionalVideoVolume: self.additionalVideoVolume, nightTheme: self.nightTheme, drawing: self.drawing, entities: self.entities, toolValues: self.toolValues, audioTrack: self.audioTrack, audioTrackTrimRange: self.audioTrackTrimRange, audioTrackOffset: self.audioTrackOffset, audioTrackVolume: self.audioTrackVolume, audioTrackSamples: self.audioTrackSamples, qualityPreset: self.qualityPreset) } + public func withUpdatedCropRect(cropRect: CGRect, rotation: CGFloat, mirroring: Bool) -> MediaEditorValues { + return MediaEditorValues(peerId: self.peerId, originalDimensions: self.originalDimensions, cropOffset: .zero, cropRect: cropRect, cropScale: 1.0, cropRotation: rotation, cropMirroring: mirroring, cropOrientation: self.cropOrientation, gradientColors: self.gradientColors, videoTrimRange: self.videoTrimRange, videoIsMuted: self.videoIsMuted, videoIsFullHd: self.videoIsFullHd, videoIsMirrored: self.videoIsMirrored, videoVolume: self.videoVolume, additionalVideoPath: self.additionalVideoPath, additionalVideoIsDual: self.additionalVideoIsDual, additionalVideoPosition: self.additionalVideoPosition, additionalVideoScale: self.additionalVideoScale, additionalVideoRotation: self.additionalVideoRotation, additionalVideoPositionChanges: self.additionalVideoPositionChanges, additionalVideoTrimRange: self.additionalVideoTrimRange, additionalVideoOffset: self.additionalVideoOffset, additionalVideoVolume: self.additionalVideoVolume, nightTheme: self.nightTheme, drawing: self.drawing, entities: self.entities, toolValues: self.toolValues, audioTrack: self.audioTrack, audioTrackTrimRange: self.audioTrackTrimRange, audioTrackOffset: self.audioTrackOffset, audioTrackVolume: self.audioTrackVolume, audioTrackSamples: self.audioTrackSamples, qualityPreset: self.qualityPreset) + } + func withUpdatedGradientColors(gradientColors: [UIColor]) -> MediaEditorValues { return MediaEditorValues(peerId: self.peerId, originalDimensions: self.originalDimensions, cropOffset: self.cropOffset, cropRect: self.cropRect, cropScale: self.cropScale, cropRotation: self.cropRotation, cropMirroring: self.cropMirroring, cropOrientation: self.cropOrientation, gradientColors: gradientColors, videoTrimRange: self.videoTrimRange, videoIsMuted: self.videoIsMuted, videoIsFullHd: self.videoIsFullHd, videoIsMirrored: self.videoIsMirrored, videoVolume: self.videoVolume, additionalVideoPath: self.additionalVideoPath, additionalVideoIsDual: self.additionalVideoIsDual, additionalVideoPosition: self.additionalVideoPosition, additionalVideoScale: self.additionalVideoScale, additionalVideoRotation: self.additionalVideoRotation, additionalVideoPositionChanges: self.additionalVideoPositionChanges, additionalVideoTrimRange: self.additionalVideoTrimRange, additionalVideoOffset: self.additionalVideoOffset, additionalVideoVolume: self.additionalVideoVolume, nightTheme: self.nightTheme, drawing: self.drawing, entities: self.entities, toolValues: self.toolValues, audioTrack: self.audioTrack, audioTrackTrimRange: self.audioTrackTrimRange, audioTrackOffset: self.audioTrackOffset, audioTrackVolume: self.audioTrackVolume, audioTrackSamples: self.audioTrackSamples, qualityPreset: self.qualityPreset) } @@ -716,6 +720,10 @@ public final class MediaEditorValues: Codable, Equatable { return MediaEditorValues(peerId: self.peerId, originalDimensions: self.originalDimensions, cropOffset: self.cropOffset, cropRect: self.cropRect, cropScale: self.cropScale, cropRotation: self.cropRotation, cropMirroring: self.cropMirroring, cropOrientation: self.cropOrientation, gradientColors: self.gradientColors, videoTrimRange: videoTrimRange, videoIsMuted: self.videoIsMuted, videoIsFullHd: self.videoIsFullHd, videoIsMirrored: self.videoIsMirrored, videoVolume: self.videoVolume, additionalVideoPath: self.additionalVideoPath, additionalVideoIsDual: self.additionalVideoIsDual, additionalVideoPosition: self.additionalVideoPosition, additionalVideoScale: self.additionalVideoScale, additionalVideoRotation: self.additionalVideoRotation, additionalVideoPositionChanges: self.additionalVideoPositionChanges, additionalVideoTrimRange: self.additionalVideoTrimRange, additionalVideoOffset: self.additionalVideoOffset, additionalVideoVolume: self.additionalVideoVolume, nightTheme: self.nightTheme, drawing: self.drawing, entities: entities, toolValues: self.toolValues, audioTrack: self.audioTrack, audioTrackTrimRange: self.audioTrackTrimRange, audioTrackOffset: self.audioTrackOffset, audioTrackVolume: self.audioTrackVolume, audioTrackSamples: self.audioTrackSamples, qualityPreset: self.qualityPreset) } + public func withUpdatedQualityPreset(_ qualityPreset: MediaQualityPreset?) -> MediaEditorValues { + return MediaEditorValues(peerId: self.peerId, originalDimensions: self.originalDimensions, cropOffset: self.cropOffset, cropRect: self.cropRect, cropScale: self.cropScale, cropRotation: self.cropRotation, cropMirroring: self.cropMirroring, cropOrientation: self.cropOrientation, gradientColors: self.gradientColors, videoTrimRange: videoTrimRange, videoIsMuted: self.videoIsMuted, videoIsFullHd: self.videoIsFullHd, videoIsMirrored: self.videoIsMirrored, videoVolume: self.videoVolume, additionalVideoPath: self.additionalVideoPath, additionalVideoIsDual: self.additionalVideoIsDual, additionalVideoPosition: self.additionalVideoPosition, additionalVideoScale: self.additionalVideoScale, additionalVideoRotation: self.additionalVideoRotation, additionalVideoPositionChanges: self.additionalVideoPositionChanges, additionalVideoTrimRange: self.additionalVideoTrimRange, additionalVideoOffset: self.additionalVideoOffset, additionalVideoVolume: self.additionalVideoVolume, nightTheme: self.nightTheme, drawing: self.drawing, entities: self.entities, toolValues: self.toolValues, audioTrack: self.audioTrack, audioTrackTrimRange: self.audioTrackTrimRange, audioTrackOffset: self.audioTrackOffset, audioTrackVolume: self.audioTrackVolume, audioTrackSamples: self.audioTrackSamples, qualityPreset: qualityPreset) + } + public var resultDimensions: PixelDimensions { if self.videoIsFullHd { return PixelDimensions(width: 1080, height: 1920) diff --git a/submodules/TelegramUI/Components/MediaEditor/Sources/RenderPass.swift b/submodules/TelegramUI/Components/MediaEditor/Sources/RenderPass.swift index 4611bc5f57f..54a849f09e5 100644 --- a/submodules/TelegramUI/Components/MediaEditor/Sources/RenderPass.swift +++ b/submodules/TelegramUI/Components/MediaEditor/Sources/RenderPass.swift @@ -165,6 +165,7 @@ final class OutputRenderPass: DefaultRenderPass { renderPassDescriptor.colorAttachments[0].texture = (drawable as? CAMetalDrawable)?.texture renderPassDescriptor.colorAttachments[0].loadAction = .clear renderPassDescriptor.colorAttachments[0].storeAction = .store + renderPassDescriptor.colorAttachments[0].clearColor = MTLClearColor(red: 0.0, green: 0.0, blue: 0.0, alpha: 0.0) let drawableSize = renderTarget.drawableSize diff --git a/submodules/TelegramUI/Components/MediaEditor/Sources/UniversalTextureSource.swift b/submodules/TelegramUI/Components/MediaEditor/Sources/UniversalTextureSource.swift index 737dd4ee674..4e7a97f844e 100644 --- a/submodules/TelegramUI/Components/MediaEditor/Sources/UniversalTextureSource.swift +++ b/submodules/TelegramUI/Components/MediaEditor/Sources/UniversalTextureSource.swift @@ -42,6 +42,13 @@ final class UniversalTextureSource: TextureSource { ) } + var mainImage: UIImage? { + if let mainInput = self.mainInputContext?.input, case let .image(image) = mainInput { + return image + } + return nil + } + func setMainInput(_ input: Input) { guard let renderTarget = self.renderTarget else { return @@ -128,15 +135,6 @@ final class UniversalTextureSource: TextureSource { self.update() } } -// -// private func setupDisplayLink(frameRate: Int) { -// self.displayLink?.invalidate() -// self.displayLink = nil -// -// if self.playerItemOutput != nil { - -// } -// } } private protocol InputContext { diff --git a/submodules/TelegramUI/Components/MediaEditor/Sources/VideoFinishPass.swift b/submodules/TelegramUI/Components/MediaEditor/Sources/VideoFinishPass.swift index 45f313016f3..d6e3fcbe436 100644 --- a/submodules/TelegramUI/Components/MediaEditor/Sources/VideoFinishPass.swift +++ b/submodules/TelegramUI/Components/MediaEditor/Sources/VideoFinishPass.swift @@ -144,6 +144,14 @@ private var transitionDuration = 0.5 private var apperanceDuration = 0.2 private var videoRemovalDuration: Double = 0.2 +struct VideoEncodeParameters { + var dimensions: simd_float2 + var roundness: simd_float1 + var alpha: simd_float1 + var isOpaque: simd_float1 + var empty: simd_float1 +} + final class VideoFinishPass: RenderPass { private var cachedTexture: MTLTexture? @@ -195,6 +203,7 @@ final class VideoFinishPass: RenderPass { containerSize: CGSize, texture: MTLTexture, textureRotation: TextureRotation, + maskTexture: MTLTexture?, position: VideoPosition, roundness: Float, alpha: Float, @@ -202,6 +211,11 @@ final class VideoFinishPass: RenderPass { device: MTLDevice ) { encoder.setFragmentTexture(texture, index: 0) + if let maskTexture { + encoder.setFragmentTexture(maskTexture, index: 1) + } else { + encoder.setFragmentTexture(texture, index: 1) + } let center = CGPoint( x: position.position.x - containerSize.width / 2.0, @@ -220,20 +234,31 @@ final class VideoFinishPass: RenderPass { options: []) encoder.setVertexBuffer(buffer, offset: 0, index: 0) - var resolution = simd_uint2(UInt32(size.width), UInt32(size.height)) - encoder.setFragmentBytes(&resolution, length: MemoryLayout.size * 2, index: 0) - - var roundness = roundness - encoder.setFragmentBytes(&roundness, length: MemoryLayout.size, index: 1) - - var alpha = alpha - encoder.setFragmentBytes(&alpha, length: MemoryLayout.size, index: 2) + var parameters = VideoEncodeParameters( + dimensions: simd_float2(Float(size.width), Float(size.height)), + roundness: roundness, + alpha: alpha, + isOpaque: maskTexture == nil ? 1.0 : 0.0, + empty: 0 + ) + encoder.setFragmentBytes(¶meters, length: MemoryLayout.size, index: 0) +// var resolution = simd_uint2(UInt32(size.width), UInt32(size.height)) +// encoder.setFragmentBytes(&resolution, length: MemoryLayout.size * 2, index: 0) +// +// var roundness = roundness +// encoder.setFragmentBytes(&roundness, length: MemoryLayout.size, index: 1) +// +// var alpha = alpha +// encoder.setFragmentBytes(&alpha, length: MemoryLayout.size, index: 2) +// +// var isOpaque = maskTexture == nil ? 1.0 : 0.0 +// encoder.setFragmentBytes(&isOpaque, length: MemoryLayout.size, index: 3) encoder.drawPrimitives(type: .triangleStrip, vertexStart: 0, vertexCount: 4) } private let canvasSize = CGSize(width: 1080.0, height: 1920.0) - private var gradientColors = GradientColors(topColor: simd_float3(0.0, 0.0, 0.0), bottomColor: simd_float3(0.0, 0.0, 0.0)) + private var gradientColors = GradientColors(topColor: simd_float4(0.0, 0.0, 0.0, 0.0), bottomColor: simd_float4(0.0, 0.0, 0.0, 0.0)) func update(values: MediaEditorValues, videoDuration: Double?, additionalVideoDuration: Double?) { let position = CGPoint( x: canvasSize.width / 2.0 + values.cropOffset.x, @@ -241,6 +266,7 @@ final class VideoFinishPass: RenderPass { ) self.isStory = values.isStory + self.isSticker = values.gradientColors?.first?.alpha == 0.0 self.mainPosition = VideoFinishPass.VideoPosition(position: position, size: self.mainPosition.size, scale: values.cropScale, rotation: values.cropRotation, baseScale: self.mainPosition.baseScale) if let position = values.additionalVideoPosition, let scale = values.additionalVideoScale, let rotation = values.additionalVideoRotation { @@ -262,12 +288,12 @@ final class VideoFinishPass: RenderPass { } if let gradientColors = values.gradientColors, let top = gradientColors.first, let bottom = gradientColors.last { - let (topRed, topGreen, topBlue, _) = top.components - let (bottomRed, bottomGreen, bottomBlue, _) = bottom.components + let (topRed, topGreen, topBlue, topAlpha) = top.components + let (bottomRed, bottomGreen, bottomBlue, bottomAlpha) = bottom.components self.gradientColors = GradientColors( - topColor: simd_float3(Float(topRed), Float(topGreen), Float(topBlue)), - bottomColor: simd_float3(Float(bottomRed), Float(bottomGreen), Float(bottomBlue)) + topColor: simd_float4(Float(topRed), Float(topGreen), Float(topBlue), Float(topAlpha)), + bottomColor: simd_float4(Float(bottomRed), Float(bottomGreen), Float(bottomBlue), Float(bottomAlpha)) ) } } @@ -289,6 +315,7 @@ final class VideoFinishPass: RenderPass { ) private var isStory = true + private var isSticker = true private var videoPositionChanges: [VideoPositionChange] = [] private var videoDuration: Double? private var additionalVideoDuration: Double? @@ -476,16 +503,31 @@ final class VideoFinishPass: RenderPass { return (backgroundVideoState, foregroundVideoState, disappearingVideoState) } - func process(input: MTLTexture, secondInput: MTLTexture?, timestamp: CMTime, device: MTLDevice, commandBuffer: MTLCommandBuffer) -> MTLTexture? { + func process( + input: MTLTexture, + inputMask: MTLTexture?, + secondInput: MTLTexture?, + timestamp: CMTime, + device: MTLDevice, + commandBuffer: MTLCommandBuffer + ) -> MTLTexture? { if !self.isStory { return input } let baseScale: CGFloat - if input.height > input.width { - baseScale = max(canvasSize.width / CGFloat(input.width), canvasSize.height / CGFloat(input.height)) + if !self.isSticker { + if input.height > input.width { + baseScale = max(canvasSize.width / CGFloat(input.width), canvasSize.height / CGFloat(input.height)) + } else { + baseScale = canvasSize.width / CGFloat(input.width) + } } else { - baseScale = canvasSize.width / CGFloat(input.width) + if input.height > input.width { + baseScale = canvasSize.width / CGFloat(input.width) + } else { + baseScale = canvasSize.width / CGFloat(input.height) + } } self.mainPosition = self.mainPosition.with(size: CGSize(width: input.width, height: input.height), baseScale: baseScale) @@ -508,9 +550,13 @@ final class VideoFinishPass: RenderPass { let renderPassDescriptor = MTLRenderPassDescriptor() renderPassDescriptor.colorAttachments[0].texture = self.cachedTexture! - renderPassDescriptor.colorAttachments[0].loadAction = .dontCare + if self.gradientColors.topColor.w > 0.0 { + renderPassDescriptor.colorAttachments[0].loadAction = .dontCare + } else { + renderPassDescriptor.colorAttachments[0].loadAction = .clear + } renderPassDescriptor.colorAttachments[0].storeAction = .store - renderPassDescriptor.colorAttachments[0].clearColor = MTLClearColor(red: 0.0, green: 0.0, blue: 0.0, alpha: 1.0) + renderPassDescriptor.colorAttachments[0].clearColor = MTLClearColor(red: 0.0, green: 0.0, blue: 0.0, alpha: 0.0) guard let renderCommandEncoder = commandBuffer.makeRenderCommandEncoder(descriptor: renderPassDescriptor) else { return input } @@ -521,12 +567,13 @@ final class VideoFinishPass: RenderPass { znear: -1.0, zfar: 1.0) ) - renderCommandEncoder.setRenderPipelineState(self.gradientPipelineState!) - self.encodeGradient( - using: renderCommandEncoder, - containerSize: containerSize, - device: device - ) + if self.gradientColors.topColor.w > 0.0 { + self.encodeGradient( + using: renderCommandEncoder, + containerSize: containerSize, + device: device + ) + } renderCommandEncoder.setRenderPipelineState(self.mainPipelineState!) @@ -538,6 +585,7 @@ final class VideoFinishPass: RenderPass { containerSize: containerSize, texture: transitionVideoState.texture, textureRotation: transitionVideoState.textureRotation, + maskTexture: nil, position: transitionVideoState.position, roundness: transitionVideoState.roundness, alpha: transitionVideoState.alpha, @@ -551,6 +599,7 @@ final class VideoFinishPass: RenderPass { containerSize: containerSize, texture: mainVideoState.texture, textureRotation: mainVideoState.textureRotation, + maskTexture: inputMask, position: mainVideoState.position, roundness: mainVideoState.roundness, alpha: mainVideoState.alpha, @@ -564,6 +613,7 @@ final class VideoFinishPass: RenderPass { containerSize: containerSize, texture: additionalVideoState.texture, textureRotation: additionalVideoState.textureRotation, + maskTexture: nil, position: additionalVideoState.position, roundness: additionalVideoState.roundness, alpha: additionalVideoState.alpha, @@ -578,8 +628,8 @@ final class VideoFinishPass: RenderPass { } struct GradientColors { - var topColor: simd_float3 - var bottomColor: simd_float3 + var topColor: simd_float4 + var bottomColor: simd_float4 } func encodeGradient( @@ -587,6 +637,7 @@ final class VideoFinishPass: RenderPass { containerSize: CGSize, device: MTLDevice ) { + encoder.setRenderPipelineState(self.gradientPipelineState!) let vertices = verticesDataForRotation(.rotate0Degrees) let buffer = device.makeBuffer( diff --git a/submodules/TelegramUI/Components/MediaEditorScreen/BUILD b/submodules/TelegramUI/Components/MediaEditorScreen/BUILD index e0bf2ad2537..88ab5ba5efb 100644 --- a/submodules/TelegramUI/Components/MediaEditorScreen/BUILD +++ b/submodules/TelegramUI/Components/MediaEditorScreen/BUILD @@ -50,6 +50,8 @@ swift_library( "//submodules/TelegramUI/Components/Stories/ForwardInfoPanelComponent", "//submodules/TelegramUI/Components/ContextReferenceButtonComponent", "//submodules/TelegramUI/Components/MediaScrubberComponent", + "//submodules/Components/BlurredBackgroundComponent", + "//submodules/TelegramUI/Components/DustEffect", ], visibility = [ "//visibility:public", diff --git a/submodules/TelegramUI/Components/MediaEditorScreen/Sources/MediaCutoutScreen.swift b/submodules/TelegramUI/Components/MediaEditorScreen/Sources/MediaCutoutScreen.swift new file mode 100644 index 00000000000..fd68d35fbbb --- /dev/null +++ b/submodules/TelegramUI/Components/MediaEditorScreen/Sources/MediaCutoutScreen.swift @@ -0,0 +1,437 @@ +import Foundation +import UIKit +import Display +import AsyncDisplayKit +import ComponentFlow +import SwiftSignalKit +import ViewControllerComponent +import ComponentDisplayAdapters +import TelegramPresentationData +import AccountContext +import TelegramCore +import MultilineTextComponent +import DrawingUI +import MediaEditor +import Photos +import LottieAnimationComponent +import MessageInputPanelComponent +import DustEffect + +private final class MediaCutoutScreenComponent: Component { + typealias EnvironmentType = ViewControllerComponentContainer.Environment + + let context: AccountContext + let mediaEditor: MediaEditor + + init( + context: AccountContext, + mediaEditor: MediaEditor + ) { + self.context = context + self.mediaEditor = mediaEditor + } + + static func ==(lhs: MediaCutoutScreenComponent, rhs: MediaCutoutScreenComponent) -> Bool { + if lhs.context !== rhs.context { + return false + } + return true + } + + public final class View: UIView { + private let buttonsContainerView = UIView() + private let buttonsBackgroundView = UIView() + private let previewContainerView = UIView() + private let cancelButton = ComponentView() + private let label = ComponentView() + private let doneButton = ComponentView() + + private let fadeView = UIView() + private let separatedImageView = UIImageView() + + private var component: MediaCutoutScreenComponent? + private weak var state: State? + private var environment: ViewControllerComponentContainer.Environment? + + override init(frame: CGRect) { + self.buttonsContainerView.clipsToBounds = true + + self.fadeView.alpha = 0.0 + self.fadeView.backgroundColor = UIColor(rgb: 0x000000, alpha: 0.6) + + self.separatedImageView.contentMode = .scaleAspectFit + + super.init(frame: frame) + + self.backgroundColor = .clear + + self.addSubview(self.buttonsContainerView) + self.buttonsContainerView.addSubview(self.buttonsBackgroundView) + + self.addSubview(self.fadeView) + self.addSubview(self.separatedImageView) + self.addSubview(self.previewContainerView) + + self.previewContainerView.addGestureRecognizer(UITapGestureRecognizer(target: self, action: #selector(self.previewTap(_:)))) + } + + required init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + @objc private func previewTap(_ gestureRecognizer: UITapGestureRecognizer) { + guard let component = self.component else { + return + } + let location = gestureRecognizer.location(in: gestureRecognizer.view) + + let point = CGPoint( + x: location.x / self.previewContainerView.frame.width, + y: location.y / self.previewContainerView.frame.height + ) + component.mediaEditor.setSeparationMask(point: point) + + self.playDissolveAnimation() + } + + func animateInFromEditor() { + self.buttonsBackgroundView.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.2) + self.label.view?.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.2) + } + + private var animatingOut = false + func animateOutToEditor(completion: @escaping () -> Void) { + self.animatingOut = true + + self.cancelButton.view?.isHidden = true + + self.fadeView.layer.animateAlpha(from: self.fadeView.alpha, to: 0.0, duration: 0.2, removeOnCompletion: false) + self.buttonsBackgroundView.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.2, removeOnCompletion: false, completion: { _ in + completion() + }) + self.label.view?.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.2, removeOnCompletion: false) + + self.state?.updated() + } + + public func playDissolveAnimation() { + guard let component = self.component, let resultImage = component.mediaEditor.resultImage, let environment = self.environment, let controller = environment.controller() as? MediaCutoutScreen else { + return + } + let previewView = controller.previewView + + let dustEffectLayer = DustEffectLayer() + dustEffectLayer.position = previewView.center + dustEffectLayer.bounds = previewView.bounds + previewView.superview?.layer.insertSublayer(dustEffectLayer, below: previewView.layer) + + dustEffectLayer.animationSpeed = 2.2 + dustEffectLayer.becameEmpty = { [weak dustEffectLayer] in + dustEffectLayer?.removeFromSuperlayer() + } + + dustEffectLayer.addItem(frame: previewView.bounds, image: resultImage) + + controller.requestDismiss(animated: true) + } + + func update(component: MediaCutoutScreenComponent, availableSize: CGSize, state: State, environment: Environment, transition: Transition) -> CGSize { + let environment = environment[ViewControllerComponentContainer.Environment.self].value + self.environment = environment + + let isFirstTime = self.component == nil + self.component = component + self.state = state + + let isTablet: Bool + if case .regular = environment.metrics.widthClass { + isTablet = true + } else { + isTablet = false + } + + let buttonSideInset: CGFloat + let buttonBottomInset: CGFloat = 8.0 + var controlsBottomInset: CGFloat = 0.0 + let previewSize: CGSize + var topInset: CGFloat = environment.statusBarHeight + 5.0 + if isTablet { + let previewHeight = availableSize.height - topInset - 75.0 + previewSize = CGSize(width: floorToScreenPixels(previewHeight / 1.77778), height: previewHeight) + buttonSideInset = 30.0 + } else { + previewSize = CGSize(width: availableSize.width, height: floorToScreenPixels(availableSize.width * 1.77778)) + buttonSideInset = 10.0 + if availableSize.height < previewSize.height + 30.0 { + topInset = 0.0 + controlsBottomInset = -75.0 + } else { + self.buttonsBackgroundView.backgroundColor = .clear + } + } + + let previewContainerFrame = CGRect(origin: CGPoint(x: floorToScreenPixels((availableSize.width - previewSize.width) / 2.0), y: environment.safeInsets.top), size: CGSize(width: previewSize.width, height: availableSize.height - environment.safeInsets.top - environment.safeInsets.bottom + controlsBottomInset)) + let buttonsContainerFrame = CGRect(origin: CGPoint(x: 0.0, y: availableSize.height - environment.safeInsets.bottom + controlsBottomInset), size: CGSize(width: availableSize.width, height: environment.safeInsets.bottom - controlsBottomInset)) + + let cancelButtonSize = self.cancelButton.update( + transition: transition, + component: AnyComponent(Button( + content: AnyComponent( + LottieAnimationComponent( + animation: LottieAnimationComponent.AnimationItem( + name: "media_backToCancel", + mode: .animating(loop: false), + range: self.animatingOut ? (0.5, 1.0) : (0.0, 0.5) + ), + colors: ["__allcolors__": .white], + size: CGSize(width: 33.0, height: 33.0) + ) + ), + action: { + guard let controller = environment.controller() as? MediaCutoutScreen else { + return + } + controller.requestDismiss(animated: true) + } + )), + environment: {}, + containerSize: CGSize(width: 44.0, height: 44.0) + ) + let cancelButtonFrame = CGRect( + origin: CGPoint(x: buttonSideInset, y: buttonBottomInset), + size: cancelButtonSize + ) + if let cancelButtonView = self.cancelButton.view { + if cancelButtonView.superview == nil { + self.buttonsContainerView.addSubview(cancelButtonView) + } + transition.setFrame(view: cancelButtonView, frame: cancelButtonFrame) + } + + let labelSize = self.label.update( + transition: transition, + component: AnyComponent(Text(text: "Tap an object to cut it out", font: Font.regular(17.0), color: .white)), + environment: {}, + containerSize: CGSize(width: availableSize.width - 88.0, height: 44.0) + ) + let labelFrame = CGRect( + origin: CGPoint(x: floorToScreenPixels((availableSize.width - labelSize.width) / 2.0), y: buttonBottomInset + 4.0), + size: labelSize + ) + if let labelView = self.label.view { + if labelView.superview == nil { + self.buttonsContainerView.addSubview(labelView) + } + transition.setFrame(view: labelView, frame: labelFrame) + } + + transition.setFrame(view: self.buttonsContainerView, frame: buttonsContainerFrame) + transition.setFrame(view: self.buttonsBackgroundView, frame: CGRect(origin: .zero, size: buttonsContainerFrame.size)) + + transition.setFrame(view: self.previewContainerView, frame: previewContainerFrame) + transition.setFrame(view: self.separatedImageView, frame: previewContainerFrame) + + let frameWidth = floor(previewContainerFrame.width * 0.97) + + self.fadeView.frame = CGRect(x: floorToScreenPixels((previewContainerFrame.width - frameWidth) / 2.0), y: previewContainerFrame.minY + floorToScreenPixels((previewContainerFrame.height - frameWidth) / 2.0), width: frameWidth, height: frameWidth) + self.fadeView.layer.cornerRadius = frameWidth / 8.0 + + if isFirstTime { + let _ = (component.mediaEditor.getSeparatedImage(point: nil) + |> deliverOnMainQueue).start(next: { [weak self] image in + guard let self else { + return + } + self.separatedImageView.image = image + self.state?.updated(transition: .easeInOut(duration: 0.2)) + }) + } else { + if let _ = self.separatedImageView.image { + transition.setAlpha(view: self.fadeView, alpha: 1.0) + } else { + transition.setAlpha(view: self.fadeView, alpha: 0.0) + } + } + return availableSize + } + } + + func makeView() -> View { + return View() + } + + public func update(view: View, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + return view.update(component: self, availableSize: availableSize, state: state, environment: environment, transition: transition) + } +} + +public final class MediaCutoutScreen: ViewController { + fileprivate final class Node: ViewControllerTracingNode, UIGestureRecognizerDelegate { + private weak var controller: MediaCutoutScreen? + private let context: AccountContext + + fileprivate let componentHost: ComponentView + + private var presentationData: PresentationData + private var validLayout: ContainerViewLayout? + + init(controller: MediaCutoutScreen) { + self.controller = controller + self.context = controller.context + + self.presentationData = self.context.sharedContext.currentPresentationData.with { $0 } + + self.componentHost = ComponentView() + + super.init() + + self.backgroundColor = .clear + } + + override func didLoad() { + super.didLoad() + + self.view.disablesInteractiveModalDismiss = true + self.view.disablesInteractiveKeyboardGestureRecognizer = true + } + + @objc func gestureRecognizer(_ gestureRecognizer: UIGestureRecognizer, shouldRecognizeSimultaneouslyWith otherGestureRecognizer: UIGestureRecognizer) -> Bool { + return true + } + + func animateInFromEditor() { + if let view = self.componentHost.view as? MediaCutoutScreenComponent.View { + view.animateInFromEditor() + } + } + + func animateOutToEditor(completion: @escaping () -> Void) { + if let mediaEditor = self.controller?.mediaEditor { + mediaEditor.play() + } + if let view = self.componentHost.view as? MediaCutoutScreenComponent.View { + view.animateOutToEditor(completion: completion) + } + } + + func containerLayoutUpdated(layout: ContainerViewLayout, forceUpdate: Bool = false, animateOut: Bool = false, transition: Transition) { + guard let controller = self.controller else { + return + } + let isFirstTime = self.validLayout == nil + self.validLayout = layout + + let isTablet = layout.metrics.isTablet + + let previewSize: CGSize + let topInset: CGFloat = (layout.statusBarHeight ?? 0.0) + 5.0 + if isTablet { + let previewHeight = layout.size.height - topInset - 75.0 + previewSize = CGSize(width: floorToScreenPixels(previewHeight / 1.77778), height: previewHeight) + } else { + previewSize = CGSize(width: layout.size.width, height: floorToScreenPixels(layout.size.width * 1.77778)) + } + let bottomInset = layout.size.height - previewSize.height - topInset + + let environment = ViewControllerComponentContainer.Environment( + statusBarHeight: layout.statusBarHeight ?? 0.0, + navigationHeight: 0.0, + safeInsets: UIEdgeInsets( + top: topInset, + left: layout.safeInsets.left, + bottom: bottomInset, + right: layout.safeInsets.right + ), + additionalInsets: layout.additionalInsets, + inputHeight: layout.inputHeight ?? 0.0, + metrics: layout.metrics, + deviceMetrics: layout.deviceMetrics, + orientation: nil, + isVisible: true, + theme: self.presentationData.theme, + strings: self.presentationData.strings, + dateTimeFormat: self.presentationData.dateTimeFormat, + controller: { [weak self] in + return self?.controller + } + ) + + let componentSize = self.componentHost.update( + transition: transition, + component: AnyComponent( + MediaCutoutScreenComponent( + context: self.context, + mediaEditor: controller.mediaEditor + ) + ), + environment: { + environment + }, + forceUpdate: forceUpdate || animateOut, + containerSize: layout.size + ) + if let componentView = self.componentHost.view { + if componentView.superview == nil { + self.view.insertSubview(componentView, at: 3) + componentView.clipsToBounds = true + } + let componentFrame = CGRect(origin: .zero, size: componentSize) + transition.setFrame(view: componentView, frame: CGRect(origin: componentFrame.origin, size: CGSize(width: componentFrame.width, height: componentFrame.height))) + } + + if isFirstTime { + self.animateInFromEditor() + } + } + } + + fileprivate var node: Node { + return self.displayNode as! Node + } + + fileprivate let context: AccountContext + fileprivate let mediaEditor: MediaEditor + fileprivate let previewView: MediaEditorPreviewView + + public var dismissed: () -> Void = {} + + private var initialValues: MediaEditorValues + + public init(context: AccountContext, mediaEditor: MediaEditor, previewView: MediaEditorPreviewView) { + self.context = context + self.mediaEditor = mediaEditor + self.previewView = previewView + self.initialValues = mediaEditor.values.makeCopy() + + super.init(navigationBarPresentationData: nil) + self.navigationPresentation = .flatModal + + self.supportedOrientations = ViewControllerSupportedOrientations(regularSize: .all, compactSize: .portrait) + + self.statusBar.statusBarStyle = .White + } + + required public init(coder aDecoder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + override public func loadDisplayNode() { + self.displayNode = Node(controller: self) + + super.displayNodeDidLoad() + } + + func requestDismiss(animated: Bool) { + self.dismissed() + + self.node.animateOutToEditor(completion: { + self.dismiss() + }) + } + + override public func containerLayoutUpdated(_ layout: ContainerViewLayout, transition: ContainedViewLayoutTransition) { + super.containerLayoutUpdated(layout, transition: transition) + + (self.displayNode as! Node).containerLayoutUpdated(layout: layout, transition: Transition(transition)) + } +} diff --git a/submodules/TelegramUI/Components/MediaEditorScreen/Sources/MediaEditorScreen.swift b/submodules/TelegramUI/Components/MediaEditorScreen/Sources/MediaEditorScreen.swift index 52e1f0ef041..cfd9e042dfe 100644 --- a/submodules/TelegramUI/Components/MediaEditorScreen/Sources/MediaEditorScreen.swift +++ b/submodules/TelegramUI/Components/MediaEditorScreen/Sources/MediaEditorScreen.swift @@ -40,6 +40,7 @@ import TelegramStringFormatting import ForwardInfoPanelComponent import ContextReferenceButtonComponent import MediaScrubberComponent +import BlurredBackgroundComponent private let playbackButtonTag = GenericComponentViewTag() private let muteButtonTag = GenericComponentViewTag() @@ -78,6 +79,7 @@ final class MediaEditorScreenComponent: Component { let entityViewForEntity: (DrawingEntity) -> DrawingEntityView? let openDrawing: (DrawingScreenType) -> Void let openTools: () -> Void + let openCutout: () -> Void init( context: AccountContext, @@ -93,7 +95,8 @@ final class MediaEditorScreenComponent: Component { selectedEntity: DrawingEntity?, entityViewForEntity: @escaping (DrawingEntity) -> DrawingEntityView?, openDrawing: @escaping (DrawingScreenType) -> Void, - openTools: @escaping () -> Void + openTools: @escaping () -> Void, + openCutout: @escaping () -> Void ) { self.context = context self.externalState = externalState @@ -109,6 +112,7 @@ final class MediaEditorScreenComponent: Component { self.entityViewForEntity = entityViewForEntity self.openDrawing = openDrawing self.openTools = openTools + self.openCutout = openCutout } static func ==(lhs: MediaEditorScreenComponent, rhs: MediaEditorScreenComponent) -> Bool { @@ -149,6 +153,8 @@ final class MediaEditorScreenComponent: Component { case sticker case tools case done + case cutout + case undo } private var cachedImages: [ImageKey: UIImage] = [:] func image(_ key: ImageKey) -> UIImage { @@ -165,6 +171,10 @@ final class MediaEditorScreenComponent: Component { image = generateTintedImage(image: UIImage(bundleImageName: "Media Editor/AddSticker"), color: .white)! case .tools: image = generateTintedImage(image: UIImage(bundleImageName: "Media Editor/Tools"), color: .white)! + case .cutout: + image = generateTintedImage(image: UIImage(bundleImageName: "Media Editor/Cutout"), color: .white)! + case .undo: + image = generateTintedImage(image: UIImage(bundleImageName: "Media Editor/CutoutUndo"), color: .white)! case .done: image = generateImage(CGSize(width: 33.0, height: 33.0), rotatedContext: { size, context in context.clear(CGRect(origin: CGPoint(), size: size)) @@ -254,6 +264,7 @@ final class MediaEditorScreenComponent: Component { private let stickerButton = ComponentView() private let toolsButton = ComponentView() private let doneButton = ComponentView() + private let cutoutButton = ComponentView() private let fadeView = UIButton() @@ -385,6 +396,7 @@ final class MediaEditorScreenComponent: Component { self.inputPanelExternalState.deleteBackward() } }, + openStickerEditor: {}, presentController: { [weak self] c, a in if let self { self.environment?.controller()?.present(c, in: .window(.root), with: a) @@ -580,7 +592,7 @@ final class MediaEditorScreenComponent: Component { } } - func animateOutToTool(transition: Transition) { + func animateOutToTool(inPlace: Bool, transition: Transition) { if let view = self.cancelButton.view { view.alpha = 0.0 } @@ -592,7 +604,9 @@ final class MediaEditorScreenComponent: Component { ] for button in buttons { if let view = button.view { - view.layer.animatePosition(from: .zero, to: CGPoint(x: 0.0, y: -44.0), duration: 0.3, timingFunction: kCAMediaTimingFunctionSpring, additive: true) + if !inPlace { + view.layer.animatePosition(from: .zero, to: CGPoint(x: 0.0, y: -44.0), duration: 0.3, timingFunction: kCAMediaTimingFunctionSpring, additive: true) + } view.layer.animateScale(from: 1.0, to: 0.1, duration: 0.2) } } @@ -607,7 +621,7 @@ final class MediaEditorScreenComponent: Component { } } - func animateInFromTool(transition: Transition) { + func animateInFromTool(inPlace: Bool, transition: Transition) { if let view = self.cancelButton.view { view.alpha = 1.0 } @@ -622,7 +636,9 @@ final class MediaEditorScreenComponent: Component { ] for button in buttons { if let view = button.view { - view.layer.animatePosition(from: CGPoint(x: 0.0, y: -44.0), to: .zero, duration: 0.3, timingFunction: kCAMediaTimingFunctionSpring, additive: true) + if !inPlace { + view.layer.animatePosition(from: CGPoint(x: 0.0, y: -44.0), to: .zero, duration: 0.3, timingFunction: kCAMediaTimingFunctionSpring, additive: true) + } view.layer.animateScale(from: 0.1, to: 1.0, duration: 0.2) } } @@ -685,6 +701,7 @@ final class MediaEditorScreenComponent: Component { let openDrawing = component.openDrawing let openTools = component.openTools + let openCutout = component.openCutout let buttonSideInset: CGFloat let buttonBottomInset: CGFloat = 8.0 @@ -748,33 +765,48 @@ final class MediaEditorScreenComponent: Component { transition.setAlpha(view: cancelButtonView, alpha: buttonsAreHidden ? 0.0 : bottomButtonsAlpha) } - let doneButtonTitle = isEditingStory ? environment.strings.Story_Editor_Done : environment.strings.Story_Editor_Next + var doneButtonTitle: String? + var doneButtonIcon: UIImage + switch controller.mode { + case .storyEditor: + doneButtonTitle = isEditingStory ? environment.strings.Story_Editor_Done.uppercased() : environment.strings.Story_Editor_Next.uppercased() + doneButtonIcon = UIImage(bundleImageName: "Media Editor/Next")! + case .stickerEditor: + doneButtonTitle = nil + doneButtonIcon = generateTintedImage(image: UIImage(bundleImageName: "Media Editor/Apply"), color: .white)! + } + let doneButtonSize = self.doneButton.update( transition: transition, component: AnyComponent(PlainButtonComponent( content: AnyComponent(DoneButtonContentComponent( backgroundColor: UIColor(rgb: 0x007aff), - icon: UIImage(bundleImageName: "Media Editor/Next")!, - title: doneButtonTitle.uppercased())), + icon: doneButtonIcon, + title: doneButtonTitle)), effectAlignment: .center, action: { [weak self] in guard let environment = self?.environment, let controller = environment.controller() as? MediaEditorScreen else { return } - guard !controller.node.recording.isActive else { - return - } - guard controller.checkCaptionLimit() else { - return - } - if controller.isEditingStory { - controller.requestCompletion(animated: true) - } else { - if controller.checkIfCompletionIsAllowed() { - controller.openPrivacySettings(completion: { [weak controller] in - controller?.requestCompletion(animated: true) - }) + switch controller.mode { + case .storyEditor: + guard !controller.node.recording.isActive else { + return + } + guard controller.checkCaptionLimit() else { + return + } + if controller.isEditingStory { + controller.requestStoryCompletion(animated: true) + } else { + if controller.checkIfCompletionIsAllowed() { + controller.openPrivacySettings(completion: { [weak controller] in + controller?.requestStoryCompletion(animated: true) + }) + } } + case .stickerEditor: + controller.requestStickerCompletion(animated: true) } } )), @@ -827,7 +859,7 @@ final class MediaEditorScreenComponent: Component { let drawButtonFrame = CGRect( origin: CGPoint(x: buttonsLeftOffset + floorToScreenPixels(buttonsAvailableWidth / 5.0 - drawButtonSize.width / 2.0 - 3.0), y: availableSize.height - environment.safeInsets.bottom + buttonBottomInset + controlsBottomInset + 1.0), size: drawButtonSize - ) + ) if let drawButtonView = self.drawButton.view { if drawButtonView.superview == nil { self.addSubview(drawButtonView) @@ -839,6 +871,7 @@ final class MediaEditorScreenComponent: Component { } } + let textButtonSize = self.textButton.update( transition: transition, component: AnyComponent(Button( @@ -950,6 +983,42 @@ final class MediaEditorScreenComponent: Component { } } + if controller.node.canCutout { + let isCutout = controller.node.isCutout + let cutoutButtonSize = self.cutoutButton.update( + transition: transition, + component: AnyComponent(PlainButtonComponent( + content: AnyComponent(CutoutButtonContentComponent( + backgroundColor: UIColor(rgb: 0xffffff, alpha: 0.18), + icon: state.image(isCutout ? .undo : .cutout), + title: isCutout ? "Undo Cut Out" : "Cut Out an Object" + )), + effectAlignment: .center, + action: { + openCutout() + } + )), + environment: {}, + containerSize: CGSize(width: availableSize.width, height: 44.0) + ) + let cutoutButtonFrame = CGRect( + origin: CGPoint(x: floorToScreenPixels((availableSize.width - cutoutButtonSize.width) / 2.0), y: availableSize.height - environment.safeInsets.bottom + buttonBottomInset + controlsBottomInset - cutoutButtonSize.height - 14.0), + size: cutoutButtonSize + ) + if let cutoutButtonView = self.cutoutButton.view { + if cutoutButtonView.superview == nil { + self.addSubview(cutoutButtonView) + + cutoutButtonView.layer.animatePosition(from: CGPoint(x: 0.0, y: 64.0), to: .zero, duration: 0.3, delay: 0.0, timingFunction: kCAMediaTimingFunctionSpring, additive: true) + cutoutButtonView.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.2, delay: 0.0) + cutoutButtonView.layer.animateScale(from: 0.1, to: 1.0, duration: 0.2, delay: 0.0) + } + transition.setPosition(view: cutoutButtonView, position: cutoutButtonFrame.center) + transition.setBounds(view: cutoutButtonView, bounds: CGRect(origin: .zero, size: cutoutButtonFrame.size)) + transition.setAlpha(view: cutoutButtonView, alpha: buttonsAreHidden ? 0.0 : bottomButtonsAlpha) + } + } + let mediaEditor = controller.node.mediaEditor var timeoutValue: String @@ -1083,224 +1152,6 @@ final class MediaEditorScreenComponent: Component { ) } - let nextInputMode: MessageInputPanelComponent.InputMode - switch self.currentInputMode { - case .text: - nextInputMode = .emoji - case .emoji: - nextInputMode = .text - default: - nextInputMode = .emoji - } - - var canRecordVideo = true - if let subject = controller.node.subject { - if case let .video(_, _, _, additionalPath, _, _, _, _, _) = subject, additionalPath != nil { - canRecordVideo = false - } - } - - self.inputPanel.parentState = state - let inputPanelSize = self.inputPanel.update( - transition: transition, - component: AnyComponent(MessageInputPanelComponent( - externalState: self.inputPanelExternalState, - context: component.context, - theme: environment.theme, - strings: environment.strings, - style: .editor, - placeholder: .plain(environment.strings.Story_Editor_InputPlaceholderAddCaption), - maxLength: Int(component.context.userLimits.maxStoryCaptionLength), - queryTypes: [.mention], - alwaysDarkWhenHasText: false, - resetInputContents: nil, - nextInputMode: { _ in return nextInputMode }, - areVoiceMessagesAvailable: false, - presentController: { [weak controller] c in - guard let controller else { - return - } - controller.present(c, in: .window(.root)) - }, - presentInGlobalOverlay: { [weak controller] c in - guard let controller else { - return - } - controller.presentInGlobalOverlay(c) - }, - sendMessageAction: { [weak self] in - guard let self else { - return - } - self.deactivateInput() - }, - sendMessageOptionsAction: nil, - sendStickerAction: { _ in }, - setMediaRecordingActive: canRecordVideo ? { [weak controller] isActive, _, finished, sourceView in - guard let controller else { - return - } - controller.node.recording.setMediaRecordingActive(isActive, finished: finished, sourceView: sourceView) - } : nil, - lockMediaRecording: { [weak controller, weak self] in - guard let controller, let self else { - return - } - controller.node.recording.isLocked = true - self.state?.updated(transition: .easeInOut(duration: 0.2)) - }, - stopAndPreviewMediaRecording: { [weak controller] in - guard let controller else { - return - } - controller.node.recording.setMediaRecordingActive(false, finished: true, sourceView: nil) - }, - discardMediaRecordingPreview: nil, - attachmentAction: nil, - myReaction: nil, - likeAction: nil, - likeOptionsAction: nil, - inputModeAction: { [weak self] in - if let self { - switch self.currentInputMode { - case .text: - self.currentInputMode = .emoji - case .emoji: - self.currentInputMode = .text - default: - self.currentInputMode = .emoji - } - if self.currentInputMode == .text { - self.activateInput() - } else { - self.state?.updated(transition: .immediate) - } - } - }, - timeoutAction: isEditingStory ? nil : { [weak controller] view, gesture in - guard let controller else { - return - } - controller.presentTimeoutSetup(sourceView: view, gesture: gesture) - }, - forwardAction: nil, - moreAction: nil, - presentVoiceMessagesUnavailableTooltip: nil, - presentTextLengthLimitTooltip: { [weak controller] in - guard let controller else { - return - } - controller.presentCaptionLimitPremiumSuggestion(isPremium: controller.context.isPremium) - }, - presentTextFormattingTooltip: { [weak controller] in - guard let controller else { - return - } - controller.presentCaptionEntitiesPremiumSuggestion() - }, - paste: { [weak self, weak controller] data in - guard let self, let controller else { - return - } - switch data { - case let .sticker(image, _): - if max(image.size.width, image.size.height) > 1.0 { - let entity = DrawingStickerEntity(content: .image(image, .sticker)) - controller.node.interaction?.insertEntity(entity, scale: 1.0) - self.deactivateInput() - } - case let .images(images): - if images.count == 1, let image = images.first, max(image.size.width, image.size.height) > 1.0 { - let entity = DrawingStickerEntity(content: .image(image, .rectangle)) - controller.node.interaction?.insertEntity(entity, scale: 2.5) - self.deactivateInput() - } - case .text: - Queue.mainQueue().after(0.1) { - let text = self.getInputText() - if text.length > component.context.userLimits.maxStoryCaptionLength { - controller.presentCaptionLimitPremiumSuggestion(isPremium: self.state?.isPremium ?? false) - } - } - default: - break - } - }, - audioRecorder: nil, - videoRecordingStatus: controller.node.recording.status, - isRecordingLocked: controller.node.recording.isLocked, - hasRecordedVideo: mediaEditor?.values.additionalVideoPath != nil, - recordedAudioPreview: nil, - hasRecordedVideoPreview: false, - wasRecordingDismissed: false, - timeoutValue: timeoutValue, - timeoutSelected: false, - displayGradient: false, - bottomInset: 0.0, - isFormattingLocked: !state.isPremium, - hideKeyboard: self.currentInputMode == .emoji, - customInputView: nil, - forceIsEditing: self.currentInputMode == .emoji, - disabledPlaceholder: nil, - header: header, - isChannel: false, - storyItem: nil, - chatLocation: nil - )), - environment: {}, - containerSize: CGSize(width: inputPanelAvailableWidth, height: inputPanelAvailableHeight) - ) - - if self.inputPanelExternalState.isEditing && controller.node.entitiesView.hasSelection { - Queue.mainQueue().justDispatch { - controller.node.entitiesView.selectEntity(nil) - } - } - - if self.inputPanelExternalState.isEditing { - if self.currentInputMode == .emoji || (inputHeight.isZero && keyboardWasHidden) { - inputHeight = max(inputHeight, environment.deviceMetrics.standardInputHeight(inLandscape: false)) - } - } - keyboardHeight = inputHeight - - let fadeTransition = Transition(animation: .curve(duration: 0.3, curve: .easeInOut)) - if self.inputPanelExternalState.isEditing { - fadeTransition.setAlpha(view: self.fadeView, alpha: 1.0) - } else { - fadeTransition.setAlpha(view: self.fadeView, alpha: 0.0) - } - transition.setFrame(view: self.fadeView, frame: CGRect(origin: .zero, size: availableSize)) - - let isEditingCaption = self.inputPanelExternalState.isEditing - if self.isEditingCaption != isEditingCaption { - self.isEditingCaption = isEditingCaption - - if isEditingCaption { - controller.dismissAllTooltips() - mediaEditor?.maybePauseVideo() - } else { - mediaEditor?.maybeUnpauseVideo() - } - } - - let inputPanelBackgroundSize = self.inputPanelBackground.update( - transition: transition, - component: AnyComponent(BlurredGradientComponent(position: .bottom, tag: nil)), - environment: {}, - containerSize: CGSize(width: availableSize.width, height: keyboardHeight + 60.0) - ) - if let inputPanelBackgroundView = self.inputPanelBackground.view { - if inputPanelBackgroundView.superview == nil { - self.addSubview(inputPanelBackgroundView) - } - let isVisible = isEditingCaption && inputHeight > 44.0 - transition.setFrame(view: inputPanelBackgroundView, frame: CGRect(origin: CGPoint(x: 0.0, y: isVisible ? availableSize.height - inputPanelBackgroundSize.height : availableSize.height), size: inputPanelBackgroundSize)) - if !self.animatingButtons { - transition.setAlpha(view: inputPanelBackgroundView, alpha: isVisible ? 1.0 : 0.0, delay: isVisible ? 0.0 : 0.4) - } - } - var isEditingTextEntity = false var sizeSliderVisible = false var sizeValue: CGFloat? @@ -1310,562 +1161,783 @@ final class MediaEditorScreenComponent: Component { sizeValue = textEntity.fontSize } - var inputPanelBottomInset: CGFloat = -controlsBottomInset - if inputHeight > 0.0 { - inputPanelBottomInset = inputHeight - environment.safeInsets.bottom - } - let inputPanelFrame = CGRect(origin: CGPoint(x: floorToScreenPixels((availableSize.width - inputPanelSize.width) / 2.0), y: availableSize.height - environment.safeInsets.bottom - inputPanelBottomInset - inputPanelSize.height - 3.0), size: inputPanelSize) - if let inputPanelView = self.inputPanel.view { - if inputPanelView.superview == nil { - self.addSubview(inputPanelView) - } - transition.setFrame(view: inputPanelView, frame: inputPanelFrame) - transition.setAlpha(view: inputPanelView, alpha: isEditingTextEntity || component.isDisplayingTool || component.isDismissing || component.isInteractingWithEntities ? 0.0 : 1.0) - } - - if let playerState = state.playerState { - let scrubberInset: CGFloat = 9.0 - - let minDuration: Double - let maxDuration: Double - if playerState.isAudioOnly { - minDuration = 5.0 - maxDuration = 15.0 - } else { - minDuration = 1.0 - maxDuration = storyMaxVideoDuration - } - - let previousTrackCount = self.currentVisibleTracks?.count - let visibleTracks = playerState.tracks.filter { $0.visibleInTimeline }.map { MediaScrubberComponent.Track($0) } - self.currentVisibleTracks = visibleTracks - - var scrubberTransition = transition - if let previousTrackCount, previousTrackCount != visibleTracks.count { - scrubberTransition = .easeInOut(duration: 0.2) + if case .storyEditor = controller.mode { + let nextInputMode: MessageInputPanelComponent.InputMode + switch self.currentInputMode { + case .text: + nextInputMode = .emoji + case .emoji: + nextInputMode = .text + default: + nextInputMode = .emoji } - let isAudioOnly = playerState.isAudioOnly - let hasMainVideoTrack = playerState.tracks.contains(where: { $0.id == 0 }) - - let scrubber: ComponentView - if let current = self.scrubber { - scrubber = current - } else { - scrubber = ComponentView() - self.scrubber = scrubber + var canRecordVideo = true + if let subject = controller.node.subject { + if case let .video(_, _, _, additionalPath, _, _, _, _, _) = subject, additionalPath != nil { + canRecordVideo = false + } } - let scrubberSize = scrubber.update( - transition: scrubberTransition, - component: AnyComponent(MediaScrubberComponent( + self.inputPanel.parentState = state + let inputPanelSize = self.inputPanel.update( + transition: transition, + component: AnyComponent(MessageInputPanelComponent( + externalState: self.inputPanelExternalState, context: component.context, - style: .editor, theme: environment.theme, - generationTimestamp: playerState.generationTimestamp, - position: playerState.position, - minDuration: minDuration, - maxDuration: maxDuration, - isPlaying: playerState.isPlaying, - tracks: visibleTracks, - positionUpdated: { [weak mediaEditor] position, apply in - if let mediaEditor { - mediaEditor.seek(position, andPlay: apply) + strings: environment.strings, + style: .editor, + placeholder: .plain(environment.strings.Story_Editor_InputPlaceholderAddCaption), + maxLength: Int(component.context.userLimits.maxStoryCaptionLength), + queryTypes: [.mention], + alwaysDarkWhenHasText: false, + resetInputContents: nil, + nextInputMode: { _ in return nextInputMode }, + areVoiceMessagesAvailable: false, + presentController: { [weak controller] c in + guard let controller else { + return } + controller.present(c, in: .window(.root)) }, - trackTrimUpdated: { [weak mediaEditor] trackId, start, end, updatedEnd, apply in - guard let mediaEditor else { + presentInGlobalOverlay: { [weak controller] c in + guard let controller else { return } - let trimRange = start..= upperBound { - start = lowerBound - } else if start < lowerBound { - start = lowerBound - } - } - - mediaEditor.seek(start, andPlay: true) - mediaEditor.play() - } else { - mediaEditor.stop() - } - } - } else if trackId == 1 { - mediaEditor.setAdditionalVideoOffset(offset, apply: apply) + controller.presentTimeoutSetup(sourceView: view, gesture: gesture) + }, + forwardAction: nil, + moreAction: nil, + presentVoiceMessagesUnavailableTooltip: nil, + presentTextLengthLimitTooltip: { [weak controller] in + guard let controller else { + return } + controller.presentCaptionLimitPremiumSuggestion(isPremium: controller.context.isPremium) }, - trackLongPressed: { [weak self] trackId, sourceView in - guard let self, let controller = self.environment?.controller() as? MediaEditorScreen else { + presentTextFormattingTooltip: { [weak controller] in + guard let controller else { return } - controller.node.presentTrackOptions(trackId: trackId, sourceView: sourceView) - } + controller.presentCaptionEntitiesPremiumSuggestion() + }, + paste: { [weak self, weak controller] data in + guard let self, let controller else { + return + } + switch data { + case let .sticker(image, _): + if max(image.size.width, image.size.height) > 1.0 { + let entity = DrawingStickerEntity(content: .image(image, .sticker)) + controller.node.interaction?.insertEntity(entity, scale: 1.0) + self.deactivateInput() + } + case let .images(images): + if images.count == 1, let image = images.first, max(image.size.width, image.size.height) > 1.0 { + let entity = DrawingStickerEntity(content: .image(image, .rectangle)) + controller.node.interaction?.insertEntity(entity, scale: 2.5) + self.deactivateInput() + } + case .text: + Queue.mainQueue().after(0.1) { + let text = self.getInputText() + if text.length > component.context.userLimits.maxStoryCaptionLength { + controller.presentCaptionLimitPremiumSuggestion(isPremium: self.state?.isPremium ?? false) + } + } + default: + break + } + }, + audioRecorder: nil, + videoRecordingStatus: controller.node.recording.status, + isRecordingLocked: controller.node.recording.isLocked, + hasRecordedVideo: mediaEditor?.values.additionalVideoPath != nil, + recordedAudioPreview: nil, + hasRecordedVideoPreview: false, + wasRecordingDismissed: false, + timeoutValue: timeoutValue, + timeoutSelected: false, + displayGradient: false, + bottomInset: 0.0, + isFormattingLocked: !state.isPremium, + hideKeyboard: self.currentInputMode == .emoji, + customInputView: nil, + forceIsEditing: self.currentInputMode == .emoji, + disabledPlaceholder: nil, + header: header, + isChannel: false, + storyItem: nil, + chatLocation: nil )), environment: {}, - containerSize: CGSize(width: previewSize.width - scrubberInset * 2.0, height: availableSize.height) + containerSize: CGSize(width: inputPanelAvailableWidth, height: inputPanelAvailableHeight) ) - let scrubberFrame = CGRect(origin: CGPoint(x: floorToScreenPixels((availableSize.width - scrubberSize.width) / 2.0), y: availableSize.height - environment.safeInsets.bottom - scrubberSize.height + controlsBottomInset - inputPanelSize.height + 3.0), size: scrubberSize) - if let scrubberView = scrubber.view { - var animateIn = false - if scrubberView.superview == nil { - animateIn = true - if let inputPanelBackgroundView = self.inputPanelBackground.view, inputPanelBackgroundView.superview != nil { - self.insertSubview(scrubberView, belowSubview: inputPanelBackgroundView) - } else { - self.addSubview(scrubberView) - } + if self.inputPanelExternalState.isEditing && controller.node.entitiesView.hasSelection { + Queue.mainQueue().justDispatch { + controller.node.entitiesView.selectEntity(nil) } - if animateIn { - scrubberView.frame = scrubberFrame + } + + if self.inputPanelExternalState.isEditing { + if self.currentInputMode == .emoji || (inputHeight.isZero && keyboardWasHidden) { + inputHeight = max(inputHeight, environment.deviceMetrics.standardInputHeight(inLandscape: false)) + } + } + keyboardHeight = inputHeight + + let fadeTransition = Transition(animation: .curve(duration: 0.3, curve: .easeInOut)) + if self.inputPanelExternalState.isEditing { + fadeTransition.setAlpha(view: self.fadeView, alpha: 1.0) + } else { + fadeTransition.setAlpha(view: self.fadeView, alpha: 0.0) + } + transition.setFrame(view: self.fadeView, frame: CGRect(origin: .zero, size: availableSize)) + + let isEditingCaption = self.inputPanelExternalState.isEditing + if self.isEditingCaption != isEditingCaption { + self.isEditingCaption = isEditingCaption + + if isEditingCaption { + controller.dismissAllTooltips() + mediaEditor?.maybePauseVideo() } else { - scrubberTransition.setFrame(view: scrubberView, frame: scrubberFrame) + mediaEditor?.maybeUnpauseVideo() + } + } + + let inputPanelBackgroundSize = self.inputPanelBackground.update( + transition: transition, + component: AnyComponent(BlurredGradientComponent(position: .bottom, tag: nil)), + environment: {}, + containerSize: CGSize(width: availableSize.width, height: keyboardHeight + 60.0) + ) + if let inputPanelBackgroundView = self.inputPanelBackground.view { + if inputPanelBackgroundView.superview == nil { + self.addSubview(inputPanelBackgroundView) } - if !self.animatingButtons && !(!hasMainVideoTrack && animateIn) { - transition.setAlpha(view: scrubberView, alpha: component.isDisplayingTool || component.isDismissing || component.isInteractingWithEntities || isEditingCaption || isRecordingAdditionalVideo || isEditingTextEntity ? 0.0 : 1.0) - } else if animateIn { - scrubberView.layer.animatePosition(from: CGPoint(x: 0.0, y: 44.0), to: .zero, duration: 0.3, timingFunction: kCAMediaTimingFunctionSpring, additive: true) - scrubberView.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.2) - scrubberView.layer.animateScale(from: 0.6, to: 1.0, duration: 0.2) + let isVisible = isEditingCaption && inputHeight > 44.0 + transition.setFrame(view: inputPanelBackgroundView, frame: CGRect(origin: CGPoint(x: 0.0, y: isVisible ? availableSize.height - inputPanelBackgroundSize.height : availableSize.height), size: inputPanelBackgroundSize)) + if !self.animatingButtons { + transition.setAlpha(view: inputPanelBackgroundView, alpha: isVisible ? 1.0 : 0.0, delay: isVisible ? 0.0 : 0.4) } } - } else { - if let scrubber = self.scrubber { - self.scrubber = nil - if let scrubberView = scrubber.view { - scrubberView.layer.animatePosition(from: .zero, to: CGPoint(x: 0.0, y: 44.0), duration: 0.3, timingFunction: kCAMediaTimingFunctionSpring, removeOnCompletion: false, additive: true) - scrubberView.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.2, removeOnCompletion: false, completion: { _ in - scrubberView.removeFromSuperview() - }) - scrubberView.layer.animateScale(from: 1.0, to: 0.6, duration: 0.2, removeOnCompletion: false) + + var inputPanelBottomInset: CGFloat = -controlsBottomInset + if inputHeight > 0.0 { + inputPanelBottomInset = inputHeight - environment.safeInsets.bottom + } + let inputPanelFrame = CGRect(origin: CGPoint(x: floorToScreenPixels((availableSize.width - inputPanelSize.width) / 2.0), y: availableSize.height - environment.safeInsets.bottom - inputPanelBottomInset - inputPanelSize.height - 3.0), size: inputPanelSize) + if let inputPanelView = self.inputPanel.view { + if inputPanelView.superview == nil { + self.addSubview(inputPanelView) } + transition.setFrame(view: inputPanelView, frame: inputPanelFrame) + transition.setAlpha(view: inputPanelView, alpha: isEditingTextEntity || component.isDisplayingTool || component.isDismissing || component.isInteractingWithEntities ? 0.0 : 1.0) } - } - - let displayTopButtons = !(self.inputPanelExternalState.isEditing || isEditingTextEntity || component.isDisplayingTool) - let saveContentComponent: AnyComponentWithIdentity - if component.hasAppeared { - saveContentComponent = AnyComponentWithIdentity( - id: "animatedIcon", - component: AnyComponent( - LottieAnimationComponent( - animation: LottieAnimationComponent.AnimationItem( - name: "anim_storysave", - mode: .still(position: .begin), - range: nil - ), - colors: ["__allcolors__": .white], - size: CGSize(width: 30.0, height: 30.0) - ).tagged(saveButtonTag) - ) - ) - } else { - saveContentComponent = AnyComponentWithIdentity( - id: "staticIcon", - component: AnyComponent( - BundleIconComponent( - name: "Media Editor/SaveIcon", - tintColor: nil - ) + if let playerState = state.playerState { + let scrubberInset: CGFloat = 9.0 + + let minDuration: Double + let maxDuration: Double + if playerState.isAudioOnly { + minDuration = 5.0 + maxDuration = 15.0 + } else { + minDuration = 1.0 + maxDuration = storyMaxVideoDuration + } + + let previousTrackCount = self.currentVisibleTracks?.count + let visibleTracks = playerState.tracks.filter { $0.visibleInTimeline }.map { MediaScrubberComponent.Track($0) } + self.currentVisibleTracks = visibleTracks + + var scrubberTransition = transition + if let previousTrackCount, previousTrackCount != visibleTracks.count { + scrubberTransition = .easeInOut(duration: 0.2) + } + + let isAudioOnly = playerState.isAudioOnly + let hasMainVideoTrack = playerState.tracks.contains(where: { $0.id == 0 }) + + let scrubber: ComponentView + if let current = self.scrubber { + scrubber = current + } else { + scrubber = ComponentView() + self.scrubber = scrubber + } + + let scrubberSize = scrubber.update( + transition: scrubberTransition, + component: AnyComponent(MediaScrubberComponent( + context: component.context, + style: .editor, + theme: environment.theme, + generationTimestamp: playerState.generationTimestamp, + position: playerState.position, + minDuration: minDuration, + maxDuration: maxDuration, + isPlaying: playerState.isPlaying, + tracks: visibleTracks, + positionUpdated: { [weak mediaEditor] position, apply in + if let mediaEditor { + mediaEditor.seek(position, andPlay: apply) + } + }, + trackTrimUpdated: { [weak mediaEditor] trackId, start, end, updatedEnd, apply in + guard let mediaEditor else { + return + } + let trimRange = start..= upperBound { + start = lowerBound + } else if start < lowerBound { + start = lowerBound + } + } + + mediaEditor.seek(start, andPlay: true) + mediaEditor.play() + } else { + mediaEditor.stop() + } + } + } else if trackId == 1 { + mediaEditor.setAdditionalVideoOffset(offset, apply: apply) + } + }, + trackLongPressed: { [weak self] trackId, sourceView in + guard let self, let controller = self.environment?.controller() as? MediaEditorScreen else { + return + } + controller.node.presentTrackOptions(trackId: trackId, sourceView: sourceView) + } + )), + environment: {}, + containerSize: CGSize(width: previewSize.width - scrubberInset * 2.0, height: availableSize.height) ) - ) - } - - let saveButtonSize = self.saveButton.update( - transition: transition, - component: AnyComponent(CameraButton( - content: saveContentComponent, - action: { [weak self] in - guard let environment = self?.environment, let controller = environment.controller() as? MediaEditorScreen else { - return + + let scrubberFrame = CGRect(origin: CGPoint(x: floorToScreenPixels((availableSize.width - scrubberSize.width) / 2.0), y: availableSize.height - environment.safeInsets.bottom - scrubberSize.height + controlsBottomInset - inputPanelSize.height + 3.0), size: scrubberSize) + if let scrubberView = scrubber.view { + var animateIn = false + if scrubberView.superview == nil { + animateIn = true + if let inputPanelBackgroundView = self.inputPanelBackground.view, inputPanelBackgroundView.superview != nil { + self.insertSubview(scrubberView, belowSubview: inputPanelBackgroundView) + } else { + self.addSubview(scrubberView) + } } - guard !controller.node.recording.isActive else { - return + if animateIn { + scrubberView.frame = scrubberFrame + } else { + scrubberTransition.setFrame(view: scrubberView, frame: scrubberFrame) } - if let view = self?.saveButton.findTaggedView(tag: saveButtonTag) as? LottieAnimationComponent.View { - view.playOnce() + if !self.animatingButtons && !(!hasMainVideoTrack && animateIn) { + transition.setAlpha(view: scrubberView, alpha: component.isDisplayingTool || component.isDismissing || component.isInteractingWithEntities || isEditingCaption || isRecordingAdditionalVideo || isEditingTextEntity ? 0.0 : 1.0) + } else if animateIn { + scrubberView.layer.animatePosition(from: CGPoint(x: 0.0, y: 44.0), to: .zero, duration: 0.3, timingFunction: kCAMediaTimingFunctionSpring, additive: true) + scrubberView.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.2) + scrubberView.layer.animateScale(from: 0.6, to: 1.0, duration: 0.2) + } + } + } else { + if let scrubber = self.scrubber { + self.scrubber = nil + if let scrubberView = scrubber.view { + scrubberView.layer.animatePosition(from: .zero, to: CGPoint(x: 0.0, y: 44.0), duration: 0.3, timingFunction: kCAMediaTimingFunctionSpring, removeOnCompletion: false, additive: true) + scrubberView.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.2, removeOnCompletion: false, completion: { _ in + scrubberView.removeFromSuperview() + }) + scrubberView.layer.animateScale(from: 1.0, to: 0.6, duration: 0.2, removeOnCompletion: false) } - controller.requestSave() } - )), - environment: {}, - containerSize: CGSize(width: 44.0, height: 44.0) - ) - let saveButtonFrame = CGRect( - origin: CGPoint(x: availableSize.width - 20.0 - saveButtonSize.width, y: max(environment.statusBarHeight + 10.0, environment.safeInsets.top + 20.0)), - size: saveButtonSize - ) - if let saveButtonView = self.saveButton.view { - if saveButtonView.superview == nil { - setupButtonShadow(saveButtonView) - self.addSubview(saveButtonView) } - - let saveButtonAlpha = component.isSavingAvailable ? topButtonsAlpha : 0.3 - saveButtonView.isUserInteractionEnabled = component.isSavingAvailable - - transition.setPosition(view: saveButtonView, position: saveButtonFrame.center) - transition.setBounds(view: saveButtonView, bounds: CGRect(origin: .zero, size: saveButtonFrame.size)) - transition.setScale(view: saveButtonView, scale: displayTopButtons ? 1.0 : 0.01) - transition.setAlpha(view: saveButtonView, alpha: displayTopButtons && !component.isDismissing && !component.isInteractingWithEntities ? saveButtonAlpha : 0.0) - } - - var topButtonOffsetX: CGFloat = 0.0 - - if let subject = controller.node.subject, case .message = subject { - let isNightTheme = mediaEditor?.values.nightTheme == true - let dayNightContentComponent: AnyComponentWithIdentity + let displayTopButtons = !(self.inputPanelExternalState.isEditing || isEditingTextEntity || component.isDisplayingTool) + + let saveContentComponent: AnyComponentWithIdentity if component.hasAppeared { - dayNightContentComponent = AnyComponentWithIdentity( + saveContentComponent = AnyComponentWithIdentity( id: "animatedIcon", component: AnyComponent( LottieAnimationComponent( animation: LottieAnimationComponent.AnimationItem( - name: isNightTheme ? "anim_sun" : "anim_sun_reverse", - mode: state.dayNightDidChange ? .animating(loop: false) : .still(position: .end) + name: "anim_storysave", + mode: .still(position: .begin), + range: nil ), colors: ["__allcolors__": .white], size: CGSize(width: 30.0, height: 30.0) - ).tagged(dayNightButtonTag) + ).tagged(saveButtonTag) ) ) } else { - dayNightContentComponent = AnyComponentWithIdentity( + saveContentComponent = AnyComponentWithIdentity( id: "staticIcon", component: AnyComponent( BundleIconComponent( - name: "Media Editor/MuteIcon", + name: "Media Editor/SaveIcon", tintColor: nil ) ) ) } - let dayNightButtonSize = self.dayNightButton.update( + let saveButtonSize = self.saveButton.update( transition: transition, component: AnyComponent(CameraButton( - content: dayNightContentComponent, - action: { [weak self, weak state, weak mediaEditor] in + content: saveContentComponent, + action: { [weak self] in guard let environment = self?.environment, let controller = environment.controller() as? MediaEditorScreen else { return } guard !controller.node.recording.isActive else { return } - - if let mediaEditor { - state?.dayNightDidChange = true - - if let snapshotView = controller.node.previewContainerView.snapshotView(afterScreenUpdates: false) { - controller.node.previewContainerView.addSubview(snapshotView) - - snapshotView.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.25, delay: 0.1, removeOnCompletion: false, completion: { _ in - snapshotView.removeFromSuperview() - }) - } - - Queue.mainQueue().after(0.1) { - mediaEditor.toggleNightTheme() - controller.node.entitiesView.eachView { view in - if let stickerEntityView = view as? DrawingStickerEntityView { - stickerEntityView.isNightTheme = mediaEditor.values.nightTheme - } - } - } + if let view = self?.saveButton.findTaggedView(tag: saveButtonTag) as? LottieAnimationComponent.View { + view.playOnce() } + controller.requestSave() } )), environment: {}, containerSize: CGSize(width: 44.0, height: 44.0) ) - let dayNightButtonFrame = CGRect( - origin: CGPoint(x: availableSize.width - 20.0 - dayNightButtonSize.width - 50.0, y: max(environment.statusBarHeight + 10.0, environment.safeInsets.top + 20.0)), - size: dayNightButtonSize + let saveButtonFrame = CGRect( + origin: CGPoint(x: availableSize.width - 20.0 - saveButtonSize.width, y: max(environment.statusBarHeight + 10.0, environment.safeInsets.top + 20.0)), + size: saveButtonSize ) - if let dayNightButtonView = self.dayNightButton.view { - if dayNightButtonView.superview == nil { - setupButtonShadow(dayNightButtonView) - self.addSubview(dayNightButtonView) - - dayNightButtonView.layer.animateAlpha(from: 0.0, to: dayNightButtonView.alpha, duration: self.animatingButtons ? 0.1 : 0.2) - dayNightButtonView.layer.animateScale(from: 0.4, to: 1.0, duration: self.animatingButtons ? 0.1 : 0.2) + if let saveButtonView = self.saveButton.view { + if saveButtonView.superview == nil { + setupButtonShadow(saveButtonView) + self.addSubview(saveButtonView) } - transition.setPosition(view: dayNightButtonView, position: dayNightButtonFrame.center) - transition.setBounds(view: dayNightButtonView, bounds: CGRect(origin: .zero, size: dayNightButtonFrame.size)) - transition.setScale(view: dayNightButtonView, scale: displayTopButtons ? 1.0 : 0.01) - transition.setAlpha(view: dayNightButtonView, alpha: displayTopButtons && !component.isDismissing && !component.isInteractingWithEntities ? topButtonsAlpha : 0.0) - } - - topButtonOffsetX += 50.0 - } else { - if let dayNightButtonView = self.dayNightButton.view, dayNightButtonView.superview != nil { - dayNightButtonView.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.2, removeOnCompletion: false, completion: { [weak dayNightButtonView] _ in - dayNightButtonView?.removeFromSuperview() - }) - dayNightButtonView.layer.animateScale(from: 1.0, to: 0.01, duration: 0.2, removeOnCompletion: false) + + let saveButtonAlpha = component.isSavingAvailable ? topButtonsAlpha : 0.3 + saveButtonView.isUserInteractionEnabled = component.isSavingAvailable + + transition.setPosition(view: saveButtonView, position: saveButtonFrame.center) + transition.setBounds(view: saveButtonView, bounds: CGRect(origin: .zero, size: saveButtonFrame.size)) + transition.setScale(view: saveButtonView, scale: displayTopButtons ? 1.0 : 0.01) + transition.setAlpha(view: saveButtonView, alpha: displayTopButtons && !component.isDismissing && !component.isInteractingWithEntities ? saveButtonAlpha : 0.0) } - } - - if let playerState = state.playerState, playerState.hasAudio { - let isVideoMuted = mediaEditor?.values.videoIsMuted ?? false + + var topButtonOffsetX: CGFloat = 0.0 - let muteContentComponent: AnyComponentWithIdentity - if component.hasAppeared { - muteContentComponent = AnyComponentWithIdentity( - id: "animatedIcon", - component: AnyComponent( - LottieAnimationComponent( - animation: LottieAnimationComponent.AnimationItem( - name: "anim_storymute", - mode: state.muteDidChange ? .animating(loop: false) : .still(position: .begin), - range: isVideoMuted ? (0.0, 0.5) : (0.5, 1.0) - ), - colors: ["__allcolors__": .white], - size: CGSize(width: 30.0, height: 30.0) - ).tagged(muteButtonTag) + if let subject = controller.node.subject, case .message = subject { + let isNightTheme = mediaEditor?.values.nightTheme == true + + let dayNightContentComponent: AnyComponentWithIdentity + if component.hasAppeared { + dayNightContentComponent = AnyComponentWithIdentity( + id: "animatedIcon", + component: AnyComponent( + LottieAnimationComponent( + animation: LottieAnimationComponent.AnimationItem( + name: isNightTheme ? "anim_sun" : "anim_sun_reverse", + mode: state.dayNightDidChange ? .animating(loop: false) : .still(position: .end) + ), + colors: ["__allcolors__": .white], + size: CGSize(width: 30.0, height: 30.0) + ).tagged(dayNightButtonTag) + ) ) - ) - } else { - muteContentComponent = AnyComponentWithIdentity( - id: "staticIcon", - component: AnyComponent( - BundleIconComponent( - name: "Media Editor/MuteIcon", - tintColor: nil + } else { + dayNightContentComponent = AnyComponentWithIdentity( + id: "staticIcon", + component: AnyComponent( + BundleIconComponent( + name: "Media Editor/MuteIcon", + tintColor: nil + ) ) ) + } + + let dayNightButtonSize = self.dayNightButton.update( + transition: transition, + component: AnyComponent(CameraButton( + content: dayNightContentComponent, + action: { [weak self, weak state, weak mediaEditor] in + guard let environment = self?.environment, let controller = environment.controller() as? MediaEditorScreen else { + return + } + guard !controller.node.recording.isActive else { + return + } + + if let mediaEditor { + state?.dayNightDidChange = true + + if let snapshotView = controller.node.previewContainerView.snapshotView(afterScreenUpdates: false) { + controller.node.previewContainerView.addSubview(snapshotView) + + snapshotView.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.25, delay: 0.1, removeOnCompletion: false, completion: { _ in + snapshotView.removeFromSuperview() + }) + } + + Queue.mainQueue().after(0.1) { + mediaEditor.toggleNightTheme() + controller.node.entitiesView.eachView { view in + if let stickerEntityView = view as? DrawingStickerEntityView { + stickerEntityView.isNightTheme = mediaEditor.values.nightTheme + } + } + } + } + } + )), + environment: {}, + containerSize: CGSize(width: 44.0, height: 44.0) + ) + let dayNightButtonFrame = CGRect( + origin: CGPoint(x: availableSize.width - 20.0 - dayNightButtonSize.width - 50.0, y: max(environment.statusBarHeight + 10.0, environment.safeInsets.top + 20.0)), + size: dayNightButtonSize ) + if let dayNightButtonView = self.dayNightButton.view { + if dayNightButtonView.superview == nil { + setupButtonShadow(dayNightButtonView) + self.addSubview(dayNightButtonView) + + dayNightButtonView.layer.animateAlpha(from: 0.0, to: dayNightButtonView.alpha, duration: self.animatingButtons ? 0.1 : 0.2) + dayNightButtonView.layer.animateScale(from: 0.4, to: 1.0, duration: self.animatingButtons ? 0.1 : 0.2) + } + transition.setPosition(view: dayNightButtonView, position: dayNightButtonFrame.center) + transition.setBounds(view: dayNightButtonView, bounds: CGRect(origin: .zero, size: dayNightButtonFrame.size)) + transition.setScale(view: dayNightButtonView, scale: displayTopButtons ? 1.0 : 0.01) + transition.setAlpha(view: dayNightButtonView, alpha: displayTopButtons && !component.isDismissing && !component.isInteractingWithEntities ? topButtonsAlpha : 0.0) + } + + topButtonOffsetX += 50.0 + } else { + if let dayNightButtonView = self.dayNightButton.view, dayNightButtonView.superview != nil { + dayNightButtonView.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.2, removeOnCompletion: false, completion: { [weak dayNightButtonView] _ in + dayNightButtonView?.removeFromSuperview() + }) + dayNightButtonView.layer.animateScale(from: 1.0, to: 0.01, duration: 0.2, removeOnCompletion: false) + } } - let muteButtonSize = self.muteButton.update( - transition: transition, - component: AnyComponent(CameraButton( - content: muteContentComponent, - action: { [weak self, weak state, weak mediaEditor] in - guard let environment = self?.environment, let controller = environment.controller() as? MediaEditorScreen else { - return - } - guard !controller.node.recording.isActive else { - return - } - - if let mediaEditor { - state?.muteDidChange = true - let isMuted = !mediaEditor.values.videoIsMuted - mediaEditor.setVideoIsMuted(isMuted) - state?.updated() + if let playerState = state.playerState, playerState.hasAudio { + let isVideoMuted = mediaEditor?.values.videoIsMuted ?? false + + let muteContentComponent: AnyComponentWithIdentity + if component.hasAppeared { + muteContentComponent = AnyComponentWithIdentity( + id: "animatedIcon", + component: AnyComponent( + LottieAnimationComponent( + animation: LottieAnimationComponent.AnimationItem( + name: "anim_storymute", + mode: state.muteDidChange ? .animating(loop: false) : .still(position: .begin), + range: isVideoMuted ? (0.0, 0.5) : (0.5, 1.0) + ), + colors: ["__allcolors__": .white], + size: CGSize(width: 30.0, height: 30.0) + ).tagged(muteButtonTag) + ) + ) + } else { + muteContentComponent = AnyComponentWithIdentity( + id: "staticIcon", + component: AnyComponent( + BundleIconComponent( + name: "Media Editor/MuteIcon", + tintColor: nil + ) + ) + ) + } + + let muteButtonSize = self.muteButton.update( + transition: transition, + component: AnyComponent(CameraButton( + content: muteContentComponent, + action: { [weak self, weak state, weak mediaEditor] in + guard let environment = self?.environment, let controller = environment.controller() as? MediaEditorScreen else { + return + } + guard !controller.node.recording.isActive else { + return + } - controller.node.presentMutedTooltip() + if let mediaEditor { + state?.muteDidChange = true + let isMuted = !mediaEditor.values.videoIsMuted + mediaEditor.setVideoIsMuted(isMuted) + state?.updated() + + controller.node.presentMutedTooltip() + } } - } - )), - environment: {}, - containerSize: CGSize(width: 44.0, height: 44.0) - ) - let muteButtonFrame = CGRect( - origin: CGPoint(x: availableSize.width - 20.0 - muteButtonSize.width - 50.0 - topButtonOffsetX, y: max(environment.statusBarHeight + 10.0, environment.safeInsets.top + 20.0)), - size: muteButtonSize - ) - if let muteButtonView = self.muteButton.view { - if muteButtonView.superview == nil { - setupButtonShadow(muteButtonView) - self.addSubview(muteButtonView) - - muteButtonView.layer.animateAlpha(from: 0.0, to: muteButtonView.alpha, duration: self.animatingButtons ? 0.1 : 0.2) - muteButtonView.layer.animateScale(from: 0.4, to: 1.0, duration: self.animatingButtons ? 0.1 : 0.2) + )), + environment: {}, + containerSize: CGSize(width: 44.0, height: 44.0) + ) + let muteButtonFrame = CGRect( + origin: CGPoint(x: availableSize.width - 20.0 - muteButtonSize.width - 50.0 - topButtonOffsetX, y: max(environment.statusBarHeight + 10.0, environment.safeInsets.top + 20.0)), + size: muteButtonSize + ) + if let muteButtonView = self.muteButton.view { + if muteButtonView.superview == nil { + setupButtonShadow(muteButtonView) + self.addSubview(muteButtonView) + + muteButtonView.layer.animateAlpha(from: 0.0, to: muteButtonView.alpha, duration: self.animatingButtons ? 0.1 : 0.2) + muteButtonView.layer.animateScale(from: 0.4, to: 1.0, duration: self.animatingButtons ? 0.1 : 0.2) + } + transition.setPosition(view: muteButtonView, position: muteButtonFrame.center) + transition.setBounds(view: muteButtonView, bounds: CGRect(origin: .zero, size: muteButtonFrame.size)) + transition.setScale(view: muteButtonView, scale: displayTopButtons ? 1.0 : 0.01) + transition.setAlpha(view: muteButtonView, alpha: displayTopButtons && !component.isDismissing && !component.isInteractingWithEntities ? topButtonsAlpha : 0.0) + } + + topButtonOffsetX += 50.0 + } else { + if let muteButtonView = self.muteButton.view, muteButtonView.superview != nil { + muteButtonView.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.2, removeOnCompletion: false, completion: { [weak muteButtonView] _ in + muteButtonView?.removeFromSuperview() + }) + muteButtonView.layer.animateScale(from: 1.0, to: 0.01, duration: 0.2, removeOnCompletion: false) } - transition.setPosition(view: muteButtonView, position: muteButtonFrame.center) - transition.setBounds(view: muteButtonView, bounds: CGRect(origin: .zero, size: muteButtonFrame.size)) - transition.setScale(view: muteButtonView, scale: displayTopButtons ? 1.0 : 0.01) - transition.setAlpha(view: muteButtonView, alpha: displayTopButtons && !component.isDismissing && !component.isInteractingWithEntities ? topButtonsAlpha : 0.0) } - topButtonOffsetX += 50.0 - } else { - if let muteButtonView = self.muteButton.view, muteButtonView.superview != nil { - muteButtonView.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.2, removeOnCompletion: false, completion: { [weak muteButtonView] _ in - muteButtonView?.removeFromSuperview() - }) - muteButtonView.layer.animateScale(from: 1.0, to: 0.01, duration: 0.2, removeOnCompletion: false) - } - } - - if let playerState = state.playerState { - let playbackContentComponent: AnyComponentWithIdentity - if component.hasAppeared { - playbackContentComponent = AnyComponentWithIdentity( - id: "animatedIcon", - component: AnyComponent( - LottieAnimationComponent( - animation: LottieAnimationComponent.AnimationItem( - name: "anim_storyplayback", - mode: state.playbackDidChange ? .animating(loop: false) : .still(position: .end), - range: playerState.isPlaying ? (0.5, 1.0) : (0.0, 0.5) - ), - colors: ["__allcolors__": .white], - size: CGSize(width: 30.0, height: 30.0) - ).tagged(playbackButtonTag) + if let playerState = state.playerState { + let playbackContentComponent: AnyComponentWithIdentity + if component.hasAppeared { + playbackContentComponent = AnyComponentWithIdentity( + id: "animatedIcon", + component: AnyComponent( + LottieAnimationComponent( + animation: LottieAnimationComponent.AnimationItem( + name: "anim_storyplayback", + mode: state.playbackDidChange ? .animating(loop: false) : .still(position: .end), + range: playerState.isPlaying ? (0.5, 1.0) : (0.0, 0.5) + ), + colors: ["__allcolors__": .white], + size: CGSize(width: 30.0, height: 30.0) + ).tagged(playbackButtonTag) + ) ) - ) - } else { - playbackContentComponent = AnyComponentWithIdentity( - id: "staticIcon", - component: AnyComponent( - BundleIconComponent( - name: playerState.isPlaying ? "Media Editor/Pause" : "Media Editor/Play", - tintColor: nil + } else { + playbackContentComponent = AnyComponentWithIdentity( + id: "staticIcon", + component: AnyComponent( + BundleIconComponent( + name: playerState.isPlaying ? "Media Editor/Pause" : "Media Editor/Play", + tintColor: nil + ) ) ) + } + + let playbackButtonSize = self.playbackButton.update( + transition: transition, + component: AnyComponent(CameraButton( + content: playbackContentComponent, + action: { [weak self, weak mediaEditor, weak state] in + guard let environment = self?.environment, let controller = environment.controller() as? MediaEditorScreen else { + return + } + guard !controller.node.recording.isActive else { + return + } + if let mediaEditor { + state?.playbackDidChange = true + mediaEditor.togglePlayback() + } + } + )), + environment: {}, + containerSize: CGSize(width: 44.0, height: 44.0) + ) + let playbackButtonFrame = CGRect( + origin: CGPoint(x: availableSize.width - 20.0 - playbackButtonSize.width - 50.0 - topButtonOffsetX, y: max(environment.statusBarHeight + 10.0, environment.safeInsets.top + 20.0)), + size: playbackButtonSize ) + if let playbackButtonView = self.playbackButton.view { + if playbackButtonView.superview == nil { + setupButtonShadow(playbackButtonView) + self.addSubview(playbackButtonView) + + playbackButtonView.layer.animateAlpha(from: 0.0, to: playbackButtonView.alpha, duration: self.animatingButtons ? 0.1 : 0.2) + playbackButtonView.layer.animateScale(from: 0.4, to: 1.0, duration: self.animatingButtons ? 0.1 : 0.2) + } + transition.setPosition(view: playbackButtonView, position: playbackButtonFrame.center) + transition.setBounds(view: playbackButtonView, bounds: CGRect(origin: .zero, size: playbackButtonFrame.size)) + transition.setScale(view: playbackButtonView, scale: displayTopButtons ? 1.0 : 0.01) + transition.setAlpha(view: playbackButtonView, alpha: displayTopButtons && !component.isDismissing && !component.isInteractingWithEntities ? topButtonsAlpha : 0.0) + } + + topButtonOffsetX += 50.0 + } else { + if let playbackButtonView = self.playbackButton.view, playbackButtonView.superview != nil { + playbackButtonView.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.2, removeOnCompletion: false, completion: { [weak playbackButtonView] _ in + playbackButtonView?.removeFromSuperview() + }) + playbackButtonView.layer.animateScale(from: 1.0, to: 0.01, duration: 0.2, removeOnCompletion: false) + } } - let playbackButtonSize = self.playbackButton.update( + let switchCameraButtonSize = self.switchCameraButton.update( transition: transition, - component: AnyComponent(CameraButton( - content: playbackContentComponent, - action: { [weak self, weak mediaEditor, weak state] in - guard let environment = self?.environment, let controller = environment.controller() as? MediaEditorScreen else { - return - } - guard !controller.node.recording.isActive else { - return - } - if let mediaEditor { - state?.playbackDidChange = true - mediaEditor.togglePlayback() + component: AnyComponent(Button( + content: AnyComponent( + FlipButtonContentComponent(tag: switchCameraButtonTag) + ), + action: { [weak self] in + if let self, let environment = self.environment, let controller = environment.controller() as? MediaEditorScreen { + controller.node.recording.togglePosition() + + if let view = self.switchCameraButton.findTaggedView(tag: switchCameraButtonTag) as? FlipButtonContentComponent.View { + view.playAnimation() + } } } - )), + ).withIsExclusive(false)), environment: {}, - containerSize: CGSize(width: 44.0, height: 44.0) + containerSize: CGSize(width: 48.0, height: 48.0) ) - let playbackButtonFrame = CGRect( - origin: CGPoint(x: availableSize.width - 20.0 - playbackButtonSize.width - 50.0 - topButtonOffsetX, y: max(environment.statusBarHeight + 10.0, environment.safeInsets.top + 20.0)), - size: playbackButtonSize + let switchCameraButtonFrame = CGRect( + origin: CGPoint(x: 12.0, y: max(environment.statusBarHeight + 10.0, inputPanelFrame.minY - switchCameraButtonSize.height - 3.0)), + size: switchCameraButtonSize ) - if let playbackButtonView = self.playbackButton.view { - if playbackButtonView.superview == nil { - setupButtonShadow(playbackButtonView) - self.addSubview(playbackButtonView) - - playbackButtonView.layer.animateAlpha(from: 0.0, to: playbackButtonView.alpha, duration: self.animatingButtons ? 0.1 : 0.2) - playbackButtonView.layer.animateScale(from: 0.4, to: 1.0, duration: self.animatingButtons ? 0.1 : 0.2) + if let switchCameraButtonView = self.switchCameraButton.view { + if switchCameraButtonView.superview == nil { + self.addSubview(switchCameraButtonView) } - transition.setPosition(view: playbackButtonView, position: playbackButtonFrame.center) - transition.setBounds(view: playbackButtonView, bounds: CGRect(origin: .zero, size: playbackButtonFrame.size)) - transition.setScale(view: playbackButtonView, scale: displayTopButtons ? 1.0 : 0.01) - transition.setAlpha(view: playbackButtonView, alpha: displayTopButtons && !component.isDismissing && !component.isInteractingWithEntities ? topButtonsAlpha : 0.0) + transition.setPosition(view: switchCameraButtonView, position: switchCameraButtonFrame.center) + transition.setBounds(view: switchCameraButtonView, bounds: CGRect(origin: .zero, size: switchCameraButtonFrame.size)) + transition.setScale(view: switchCameraButtonView, scale: isRecordingAdditionalVideo ? 1.0 : 0.01) + transition.setAlpha(view: switchCameraButtonView, alpha: isRecordingAdditionalVideo ? 1.0 : 0.0) } - topButtonOffsetX += 50.0 - } else { - if let playbackButtonView = self.playbackButton.view, playbackButtonView.superview != nil { - playbackButtonView.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.2, removeOnCompletion: false, completion: { [weak playbackButtonView] _ in - playbackButtonView?.removeFromSuperview() - }) - playbackButtonView.layer.animateScale(from: 1.0, to: 0.01, duration: 0.2, removeOnCompletion: false) - } - } - - let switchCameraButtonSize = self.switchCameraButton.update( - transition: transition, - component: AnyComponent(Button( - content: AnyComponent( - FlipButtonContentComponent(tag: switchCameraButtonTag) - ), - action: { [weak self] in - if let self, let environment = self.environment, let controller = environment.controller() as? MediaEditorScreen { - controller.node.recording.togglePosition() - - if let view = self.switchCameraButton.findTaggedView(tag: switchCameraButtonTag) as? FlipButtonContentComponent.View { - view.playAnimation() - } - } - } - ).withIsExclusive(false)), - environment: {}, - containerSize: CGSize(width: 48.0, height: 48.0) - ) - let switchCameraButtonFrame = CGRect( - origin: CGPoint(x: 12.0, y: max(environment.statusBarHeight + 10.0, inputPanelFrame.minY - switchCameraButtonSize.height - 3.0)), - size: switchCameraButtonSize - ) - if let switchCameraButtonView = self.switchCameraButton.view { - if switchCameraButtonView.superview == nil { - self.addSubview(switchCameraButtonView) - } - transition.setPosition(view: switchCameraButtonView, position: switchCameraButtonFrame.center) - transition.setBounds(view: switchCameraButtonView, bounds: CGRect(origin: .zero, size: switchCameraButtonFrame.size)) - transition.setScale(view: switchCameraButtonView, scale: isRecordingAdditionalVideo ? 1.0 : 0.01) - transition.setAlpha(view: switchCameraButtonView, alpha: isRecordingAdditionalVideo ? 1.0 : 0.0) } let textCancelButtonSize = self.textCancelButton.update( @@ -1977,6 +2049,11 @@ let storyDimensions = CGSize(width: 1080.0, height: 1920.0) let storyMaxVideoDuration: Double = 60.0 public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate { + public enum Mode { + case storyEditor + case stickerEditor + } + public enum TransitionIn { public final class GalleryTransitionIn { public weak var sourceView: UIView? @@ -2057,6 +2134,8 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate private let gradientView: UIImageView private var gradientColorsDisposable: Disposable? + private let stickerTransparentView: UIImageView + fileprivate let entitiesContainerView: UIView let entitiesView: DrawingEntitiesView fileprivate let selectionContainerView: DrawingSelectionContainerView @@ -2085,6 +2164,9 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate private var isDismissed = false private var isDismissBySwipeSuppressed = false + fileprivate var canCutout = false + fileprivate var isCutout = false + private (set) var hasAnyChanges = false private var playbackPositionDisposable: Disposable? @@ -2122,9 +2204,11 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate } self.gradientView = UIImageView() + self.stickerTransparentView = UIImageView() + self.stickerTransparentView.clipsToBounds = true self.entitiesContainerView = UIView(frame: CGRect(origin: .zero, size: storyDimensions)) - self.entitiesView = DrawingEntitiesView(context: controller.context, size: storyDimensions, hasBin: true) + self.entitiesView = DrawingEntitiesView(context: controller.context, size: storyDimensions, hasBin: true, isStickerEditor: controller.mode == .stickerEditor) self.entitiesView.getEntityCenterPosition = { return CGPoint(x: storyDimensions.width / 2.0, y: storyDimensions.height / 2.0) } @@ -2132,6 +2216,10 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate return UIEdgeInsets(top: 160.0, left: 36.0, bottom: storyDimensions.height - 160.0, right: storyDimensions.width - 36.0) } self.previewView = MediaEditorPreviewView(frame: .zero) + if case .stickerEditor = controller.mode { + self.previewView.isOpaque = false + self.previewView.backgroundColor = .clear + } self.drawingView = DrawingView(size: storyDimensions) self.drawingView.isUserInteractionEnabled = false @@ -2147,7 +2235,31 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate self.view.addSubview(self.backgroundDimView) self.view.addSubview(self.containerView) self.containerView.addSubview(self.previewContainerView) - self.previewContainerView.addSubview(self.gradientView) + + if case .stickerEditor = controller.mode { + let rowsCount = 40 + self.stickerTransparentView.image = generateImage(CGSize(width: rowsCount, height: rowsCount), opaque: true, scale: 1.0, rotatedContext: { size, context in + context.setFillColor(UIColor.black.cgColor) + context.fill(CGRect(origin: .zero, size: size)) + context.setFillColor(UIColor(rgb: 0x2b2b2d).cgColor) + + for row in 0 ..< rowsCount { + for column in 0 ..< rowsCount { + if (row + column).isMultiple(of: 2) { + context.addRect(CGRect(x: column, y: row, width: 1, height: 1)) + } + } + } + context.fillPath() + }) + self.stickerTransparentView.layer.magnificationFilter = .nearest + self.stickerTransparentView.layer.shouldRasterize = true + self.stickerTransparentView.layer.rasterizationScale = UIScreenScale + self.previewContainerView.addSubview(self.stickerTransparentView) + } else { + self.previewContainerView.addSubview(self.gradientView) + } + self.previewContainerView.addSubview(self.previewView) self.previewContainerView.addSubview(self.entitiesContainerView) self.entitiesContainerView.addSubview(self.entitiesView) @@ -2275,8 +2387,12 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate } private func setup(with subject: MediaEditorScreen.Subject) { - self.actualSubject = subject + guard let controller = self.controller else { + return + } + self.actualSubject = subject + var effectiveSubject = subject if case let .draft(draft, _ ) = subject { for entity in draft.values.entities { @@ -2288,10 +2404,6 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate } self.subject = effectiveSubject - guard let controller = self.controller else { - return - } - Queue.mainQueue().justDispatch { controller.setupAudioSessionIfNeeded() } @@ -2321,10 +2433,19 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate let fittedSize = mediaDimensions.cgSize.fitted(CGSize(width: maxSide, height: maxSide)) let mediaEntity = DrawingMediaEntity(size: fittedSize) mediaEntity.position = CGPoint(x: storyDimensions.width / 2.0, y: storyDimensions.height / 2.0) - if fittedSize.height > fittedSize.width { - mediaEntity.scale = max(storyDimensions.width / fittedSize.width, storyDimensions.height / fittedSize.height) - } else { - mediaEntity.scale = storyDimensions.width / fittedSize.width + switch controller.mode { + case .storyEditor: + if fittedSize.height > fittedSize.width { + mediaEntity.scale = max(storyDimensions.width / fittedSize.width, storyDimensions.height / fittedSize.height) + } else { + mediaEntity.scale = storyDimensions.width / fittedSize.width + } + case .stickerEditor: + if fittedSize.height > fittedSize.width { + mediaEntity.scale = storyDimensions.width / fittedSize.width + } else { + mediaEntity.scale = storyDimensions.width / fittedSize.height + } } let initialPosition = mediaEntity.position @@ -2370,7 +2491,7 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate } } - let mediaEditor = MediaEditor(context: self.context, subject: effectiveSubject.editorSubject, values: initialValues, hasHistogram: true) + let mediaEditor = MediaEditor(context: self.context, mode: controller.mode == .stickerEditor ? .sticker : .default, subject: effectiveSubject.editorSubject, values: initialValues, hasHistogram: true) if let initialVideoPosition = controller.initialVideoPosition { mediaEditor.seek(initialVideoPosition, andPlay: true) } @@ -2389,6 +2510,20 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate } controller.requestLayout(transition: .animated(duration: 0.25, curve: .easeInOut)) } + } + mediaEditor.canCutoutUpdated = { [weak self] canCutout in + guard let self, let controller = self.controller else { + return + } + self.canCutout = canCutout + controller.requestLayout(transition: .animated(duration: 0.25, curve: .easeInOut)) + } + mediaEditor.isCutoutUpdated = { [weak self] isCutout in + guard let self else { + return + } + self.isCutout = isCutout + self.requestLayout(forceUpdate: true, transition: .immediate) } if case .message = effectiveSubject { @@ -2765,6 +2900,12 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate hasSwipeToDismiss = true } } + + var hasSwipeToEnhance = true + if case .stickerEditor = controller.mode { + hasSwipeToDismiss = false + hasSwipeToEnhance = false + } let translation = gestureRecognizer.translation(in: self.view) let velocity = gestureRecognizer.velocity(in: self.view) @@ -2776,7 +2917,7 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate self.isDismissBySwipeSuppressed = controller.isEligibleForDraft() controller.requestLayout(transition: .animated(duration: 0.25, curve: .easeInOut)) } - } else if abs(translation.x) > 10.0 && !self.isDismissing && !self.isEnhancing && self.canEnhance { + } else if abs(translation.x) > 10.0 && !self.isDismissing && !self.isEnhancing && self.canEnhance && hasSwipeToEnhance { self.isEnhancing = true controller.requestLayout(transition: .animated(duration: 0.3, curve: .easeInOut)) } @@ -2855,7 +2996,7 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate } @objc func handleTap(_ gestureRecognizer: UITapGestureRecognizer) { - guard !self.recording.isActive else { + guard !self.recording.isActive, let controller = self.controller else { return } let location = gestureRecognizer.location(in: self.view) @@ -2872,7 +3013,9 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate if let layout = self.validLayout, (layout.inputHeight ?? 0.0) > 0.0 { self.view.endEditing(true) } else { - self.insertTextEntity() + if case .storyEditor = controller.mode { + self.insertTextEntity() + } } } } @@ -2900,16 +3043,29 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate } private func setupTransitionImage(_ image: UIImage) { + guard let controller = self.controller else { + return + } self.previewContainerView.alpha = 1.0 let transitionInView = UIImageView(image: image) transitionInView.contentMode = .scaleAspectFill var initialScale: CGFloat - if image.size.height > image.size.width { - initialScale = max(self.previewContainerView.bounds.width / image.size.width, self.previewContainerView.bounds.height / image.size.height) - } else { - initialScale = self.previewContainerView.bounds.width / image.size.width + switch controller.mode { + case .storyEditor: + if image.size.height > image.size.width { + initialScale = max(self.previewContainerView.bounds.width / image.size.width, self.previewContainerView.bounds.height / image.size.height) + } else { + initialScale = self.previewContainerView.bounds.width / image.size.width + } + case .stickerEditor: + if image.size.height > image.size.width { + initialScale = self.previewContainerView.bounds.width / image.size.width + } else { + initialScale = self.previewContainerView.bounds.width / image.size.height + } } + transitionInView.center = CGPoint(x: self.previewContainerView.bounds.width / 2.0, y: self.previewContainerView.bounds.height / 2.0) transitionInView.transform = CGAffineTransformMakeScale(initialScale, initialScale) self.previewContainerView.addSubview(transitionInView) @@ -3199,22 +3355,22 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate } } - func animateOutToTool() { + func animateOutToTool(inPlace: Bool = false) { self.isDisplayingTool = true let transition: Transition = .easeInOut(duration: 0.2) if let view = self.componentHost.view as? MediaEditorScreenComponent.View { - view.animateOutToTool(transition: transition) + view.animateOutToTool(inPlace: inPlace, transition: transition) } self.requestUpdate(transition: transition) } - func animateInFromTool() { + func animateInFromTool(inPlace: Bool = false) { self.isDisplayingTool = false let transition: Transition = .easeInOut(duration: 0.2) if let view = self.componentHost.view as? MediaEditorScreenComponent.View { - view.animateInFromTool(transition: transition) + view.animateInFromTool(inPlace: inPlace, transition: transition) } self.requestUpdate(transition: transition) } @@ -3378,7 +3534,7 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate let entity = DrawingStickerEntity(content: .image(updatedImage, .rectangle)) entity.canCutOut = false - let _ = (cutoutStickerImage(from: image) + let _ = (cutoutStickerImage(from: image, onlyCheck: true) |> deliverOnMainQueue).start(next: { [weak entity] result in if result != nil, let entity { entity.canCutOut = true @@ -3853,6 +4009,7 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate bottom: bottomInset, right: layout.safeInsets.right ), + additionalInsets: layout.additionalInsets, inputHeight: layoutInputHeight, metrics: layout.metrics, deviceMetrics: layout.deviceMetrics, @@ -4079,6 +4236,33 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate self.controller?.present(controller, in: .window(.root)) self.animateOutToTool() } + }, + openCutout: { [weak self] in + if let self, let mediaEditor = self.mediaEditor { + if self.entitiesView.hasSelection { + self.entitiesView.selectEntity(nil) + } + + if controller.node.isCutout { + let snapshotView = self.previewView.snapshotView(afterScreenUpdates: false) + if let snapshotView { + self.previewView.superview?.addSubview(snapshotView) + } + self.previewView.layer.animateAlpha(from: 0.0, to: 1.0, duration: 0.3, completion: { _ in + snapshotView?.removeFromSuperview() + }) + mediaEditor.removeSeparationMask() + } else { + let controller = MediaCutoutScreen(context: self.context, mediaEditor: mediaEditor, previewView: self.previewView) + controller.dismissed = { [weak self] in + if let self { + self.animateInFromTool(inPlace: true) + } + } + self.controller?.present(controller, in: .window(.root)) + self.animateOutToTool(inPlace: true) + } + } } ) ), @@ -4172,6 +4356,10 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate transition.setFrame(view: self.selectionContainerView, frame: CGRect(origin: .zero, size: previewFrame.size)) + let stickerFrameWidth = floor(previewSize.width * 0.97) + transition.setFrame(view: self.stickerTransparentView, frame: CGRect(origin: CGPoint(x: floorToScreenPixels((previewSize.width - stickerFrameWidth) / 2.0), y: floorToScreenPixels((previewSize.height - stickerFrameWidth) / 2.0)), size: CGSize(width: stickerFrameWidth, height: stickerFrameWidth))) + self.stickerTransparentView.layer.cornerRadius = stickerFrameWidth / 8.0 + self.interaction?.containerLayoutUpdated(layout: layout, transition: transition) var layout = layout @@ -4292,6 +4480,7 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate } let context: AccountContext + let mode: Mode let subject: Signal let isEditingStory: Bool fileprivate let customTarget: EnginePeer.Id? @@ -4322,6 +4511,7 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate public init( context: AccountContext, + mode: Mode, subject: Signal, customTarget: EnginePeer.Id? = nil, isEditing: Bool = false, @@ -4335,6 +4525,7 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate completion: @escaping (MediaEditorScreen.Result, @escaping (@escaping () -> Void) -> Void) -> Void ) { self.context = context + self.mode = mode self.subject = subject self.customTarget = customTarget self.isEditingStory = isEditing @@ -4916,35 +5107,46 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate } let presentationData = self.context.sharedContext.currentPresentationData.with { $0 } - let title: String - let save: String - if case .draft = self.node.actualSubject { - title = presentationData.strings.Story_Editor_DraftDiscardDraft - save = presentationData.strings.Story_Editor_DraftKeepDraft - } else { + var title: String + var text: String + var save: String? + switch self.mode { + case .storyEditor: + if case .draft = self.node.actualSubject { + title = presentationData.strings.Story_Editor_DraftDiscardDraft + save = presentationData.strings.Story_Editor_DraftKeepDraft + } else { + title = presentationData.strings.Story_Editor_DraftDiscardMedia + save = presentationData.strings.Story_Editor_DraftKeepMedia + } + text = presentationData.strings.Story_Editor_DraftDiscaedText + case .stickerEditor: title = presentationData.strings.Story_Editor_DraftDiscardMedia - save = presentationData.strings.Story_Editor_DraftKeepMedia + text = presentationData.strings.Story_Editor_DiscardText + } + + var actions: [TextAlertAction] = [] + actions.append(TextAlertAction(type: .destructiveAction, title: presentationData.strings.Story_Editor_DraftDiscard, action: { [weak self] in + if let self { + self.requestDismiss(saveDraft: false, animated: true) + } + })) + if let save { + actions.append(TextAlertAction(type: .genericAction, title: save, action: { [weak self] in + if let self { + self.requestDismiss(saveDraft: true, animated: true) + } + })) } + actions.append(TextAlertAction(type: .genericAction, title: presentationData.strings.Common_Cancel, action: { + + })) let controller = textAlertController( context: self.context, forceTheme: defaultDarkPresentationTheme, title: title, - text: presentationData.strings.Story_Editor_DraftDiscaedText, - actions: [ - TextAlertAction(type: .destructiveAction, title: presentationData.strings.Story_Editor_DraftDiscard, action: { [weak self] in - if let self { - self.requestDismiss(saveDraft: false, animated: true) - } - }), - TextAlertAction(type: .genericAction, title: save, action: { [weak self] in - if let self { - self.requestDismiss(saveDraft: true, animated: true) - } - }), - TextAlertAction(type: .genericAction, title: presentationData.strings.Common_Cancel, action: { - - }) - ], + text: text, + actions: actions, actionLayout: .vertical ) self.present(controller, in: .window(.root)) @@ -5012,7 +5214,7 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate } private var didComplete = false - func requestCompletion(animated: Bool) { + func requestStoryCompletion(animated: Bool) { guard let mediaEditor = self.node.mediaEditor, let subject = self.node.subject, let actualSubject = self.node.actualSubject, !self.didComplete else { return } @@ -5364,6 +5566,55 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate } } + func requestStickerCompletion(animated: Bool) { + guard let mediaEditor = self.node.mediaEditor, !self.didComplete else { + return + } + + self.didComplete = true + + self.dismissAllTooltips() + + mediaEditor.stop() + mediaEditor.invalidate() + self.node.entitiesView.invalidate() + + if let navigationController = self.navigationController as? NavigationController { + navigationController.updateRootContainerTransitionOffset(0.0, transition: .immediate) + } + + let entities = self.node.entitiesView.entities.filter { !($0 is DrawingMediaEntity) } + let codableEntities = DrawingEntitiesView.encodeEntities(entities, entitiesView: self.node.entitiesView) + mediaEditor.setDrawingAndEntities(data: nil, image: mediaEditor.values.drawing, entities: codableEntities) + + if let image = mediaEditor.resultImage { + makeEditorImageComposition(context: self.node.ciContext, postbox: self.context.account.postbox, inputImage: image, dimensions: storyDimensions, values: mediaEditor.values, time: .zero, textScale: 2.0, completion: { [weak self] resultImage in + if let self, let resultImage { + Logger.shared.log("MediaEditor", "Completed with image \(resultImage)") + + let scaledImage = generateImage(CGSize(width: 512, height: 512), contextGenerator: { size, context in + context.clear(CGRect(origin: CGPoint(), size: size)) + + context.addPath(CGPath(roundedRect: CGRect(origin: .zero, size: size), cornerWidth: size.width / 8.0, cornerHeight: size.width / 8.0, transform: nil)) + context.clip() + + let scaledSize = resultImage.size.aspectFilled(size) + context.draw(resultImage.cgImage!, in: CGRect(origin: CGPoint(x: floor((size.width - scaledSize.width) / 2.0), y: floor((size.height - scaledSize.height) / 2.0)), size: scaledSize)) + }, opaque: false, scale: 1.0)! + + self.completion(MediaEditorScreen.Result(media: .image(image: scaledImage, dimensions: PixelDimensions(scaledImage.size)), mediaAreas: [], caption: NSAttributedString(), options: MediaEditorResultPrivacy(sendAsPeerId: nil, privacy: EngineStoryPrivacy(base: .everyone, additionallyIncludePeers: []), timeout: 0, isForwardingDisabled: false, pin: false), stickers: [], randomId: 0), { [weak self] finished in + self?.node.animateOut(finished: true, saveDraft: false, completion: { [weak self] in + self?.dismiss() + Queue.mainQueue().justDispatch { + finished() + } + }) + }) + } + }) + } + } + private var videoExport: MediaEditorVideoExport? private var exportDisposable = MetaDisposable() @@ -5615,7 +5866,7 @@ public final class MediaEditorScreen: ViewController, UIDropInteractionDelegate } } -final class DoneButtonContentComponent: CombinedComponent { +private final class DoneButtonContentComponent: CombinedComponent { let backgroundColor: UIColor let icon: UIImage let title: String? @@ -5646,8 +5897,9 @@ final class DoneButtonContentComponent: CombinedComponent { let text = Child(Text.self) return { context in + let iconSize = context.component.icon.size let icon = icon.update( - component: Image(image: context.component.icon, tintColor: .white, size: CGSize(width: 10.0, height: 16.0)), + component: Image(image: context.component.icon, tintColor: .white, size: iconSize), availableSize: CGSize(width: 180.0, height: 100.0), transition: .immediate ) @@ -5705,6 +5957,91 @@ final class DoneButtonContentComponent: CombinedComponent { } } +private final class CutoutButtonContentComponent: CombinedComponent { + let backgroundColor: UIColor + let icon: UIImage + let title: String? + + init( + backgroundColor: UIColor, + icon: UIImage, + title: String? + ) { + self.backgroundColor = backgroundColor + self.icon = icon + self.title = title + } + + static func ==(lhs: CutoutButtonContentComponent, rhs: CutoutButtonContentComponent) -> Bool { + if lhs.backgroundColor != rhs.backgroundColor { + return false + } + if lhs.title != rhs.title { + return false + } + return true + } + + static var body: Body { + let background = Child(BlurredBackgroundComponent.self) + let icon = Child(Image.self) + let text = Child(Text.self) + + return { context in + let iconSize = context.component.icon.size + let icon = icon.update( + component: Image(image: context.component.icon, tintColor: .white, size: iconSize), + availableSize: CGSize(width: 180.0, height: 40.0), + transition: .immediate + ) + + let backgroundHeight: CGFloat = 40.0 + var backgroundSize = CGSize(width: backgroundHeight, height: backgroundHeight) + + let textSpacing: CGFloat = 8.0 + + var title: _UpdatedChildComponent? + if let titleText = context.component.title { + title = text.update( + component: Text( + text: titleText, + font: Font.with(size: 17.0, weight: .semibold), + color: .white + ), + availableSize: CGSize(width: 240.0, height: 100.0), + transition: .immediate + ) + + let updatedBackgroundWidth = backgroundSize.width + textSpacing + title!.size.width + backgroundSize.width = updatedBackgroundWidth + 32.0 + } + + let background = background.update( + component: BlurredBackgroundComponent(color: context.component.backgroundColor, tintContainerView: nil, cornerRadius: backgroundHeight / 2.0), + availableSize: backgroundSize, + transition: .immediate + ) + context.add(background + .position(CGPoint(x: backgroundSize.width / 2.0, y: backgroundSize.height / 2.0)) + .cornerRadius(min(backgroundSize.width, backgroundSize.height) / 2.0) + .clipsToBounds(true) + ) + + if let title { + context.add(title + .position(CGPoint(x: title.size.width / 2.0 + 54.0, y: backgroundHeight / 2.0)) + ) + } + + context.add(icon + .position(CGPoint(x: 36.0, y: backgroundSize.height / 2.0)) + ) + + return backgroundSize + } + } +} + private final class HeaderContextReferenceContentSource: ContextReferenceContentSource { private let controller: ViewController diff --git a/submodules/TelegramUI/Components/MediaEditorScreen/Sources/MediaToolsScreen.swift b/submodules/TelegramUI/Components/MediaEditorScreen/Sources/MediaToolsScreen.swift index 250f26efe5c..11f3086dcb7 100644 --- a/submodules/TelegramUI/Components/MediaEditorScreen/Sources/MediaToolsScreen.swift +++ b/submodules/TelegramUI/Components/MediaEditorScreen/Sources/MediaToolsScreen.swift @@ -1031,6 +1031,7 @@ public final class MediaToolsScreen: ViewController { bottom: bottomInset, right: layout.safeInsets.right ), + additionalInsets: layout.additionalInsets, inputHeight: layout.inputHeight ?? 0.0, metrics: layout.metrics, deviceMetrics: layout.deviceMetrics, diff --git a/submodules/TelegramUI/Components/MediaEditorScreen/Sources/SaveProgressScreen.swift b/submodules/TelegramUI/Components/MediaEditorScreen/Sources/SaveProgressScreen.swift index 7de762ec791..2aeb3f1dfae 100644 --- a/submodules/TelegramUI/Components/MediaEditorScreen/Sources/SaveProgressScreen.swift +++ b/submodules/TelegramUI/Components/MediaEditorScreen/Sources/SaveProgressScreen.swift @@ -448,6 +448,7 @@ public final class SaveProgressScreen: ViewController { bottom: topInset, right: layout.safeInsets.right ), + additionalInsets: layout.additionalInsets, inputHeight: layout.inputHeight ?? 0.0, metrics: layout.metrics, deviceMetrics: layout.deviceMetrics, diff --git a/submodules/TelegramUI/Components/PeerInfo/PeerInfoScreen/BUILD b/submodules/TelegramUI/Components/PeerInfo/PeerInfoScreen/BUILD index 2b772973384..b0a86ff36e6 100644 --- a/submodules/TelegramUI/Components/PeerInfo/PeerInfoScreen/BUILD +++ b/submodules/TelegramUI/Components/PeerInfo/PeerInfoScreen/BUILD @@ -150,6 +150,7 @@ swift_library( "//submodules/TelegramUI/Components/Settings/BoostLevelIconComponent", "//submodules/Components/MultilineTextComponent", "//submodules/TelegramUI/Components/Settings/PeerNameColorItem", + "//submodules/TelegramUI/Components/PlainButtonComponent", ], visibility = [ "//visibility:public", diff --git a/submodules/TelegramUI/Components/PeerInfo/PeerInfoScreen/Sources/ListItems/PeerInfoScreenAddressItem.swift b/submodules/TelegramUI/Components/PeerInfo/PeerInfoScreen/Sources/ListItems/PeerInfoScreenAddressItem.swift index 0b1f06d683c..c6042821136 100644 --- a/submodules/TelegramUI/Components/PeerInfo/PeerInfoScreen/Sources/ListItems/PeerInfoScreenAddressItem.swift +++ b/submodules/TelegramUI/Components/PeerInfo/PeerInfoScreen/Sources/ListItems/PeerInfoScreenAddressItem.swift @@ -13,7 +13,7 @@ final class PeerInfoScreenAddressItem: PeerInfoScreenItem { let text: String let imageSignal: Signal<(TransformImageArguments) -> DrawingContext?, NoError>? let action: (() -> Void)? - let longTapAction: (() -> Void)? + let longTapAction: ((ASDisplayNode, String) -> Void)? let linkItemAction: ((TextLinkItemActionType, TextLinkItem) -> Void)? init( @@ -22,7 +22,7 @@ final class PeerInfoScreenAddressItem: PeerInfoScreenItem { text: String, imageSignal: Signal<(TransformImageArguments) -> DrawingContext?, NoError>?, action: (() -> Void)?, - longTapAction: (() -> Void)? = nil, + longTapAction: ((ASDisplayNode, String) -> Void)? = nil, linkItemAction: ((TextLinkItemActionType, TextLinkItem) -> Void)? = nil ) { self.id = id @@ -66,6 +66,16 @@ private final class PeerInfoScreenAddressItemNode: PeerInfoScreenItemNode { self.addSubnode(self.bottomSeparatorNode) self.addSubnode(self.selectionNode) self.addSubnode(self.maskNode) + + self.view.addGestureRecognizer(UILongPressGestureRecognizer(target: self, action: #selector(self.longPressGesture(_:)))) + } + + @objc private func longPressGesture(_ recognizer: UILongPressGestureRecognizer) { + if case .began = recognizer.state { + if let item = self.item { + item.longTapAction?(self, item.text) + } + } } override func update(width: CGFloat, safeInsets: UIEdgeInsets, presentationData: PresentationData, item: PeerInfoScreenItem, topItem: PeerInfoScreenItem?, bottomItem: PeerInfoScreenItem?, hasCorners: Bool, transition: ContainedViewLayoutTransition) -> CGFloat { @@ -81,7 +91,7 @@ private final class PeerInfoScreenAddressItemNode: PeerInfoScreenItemNode { self.bottomSeparatorNode.backgroundColor = presentationData.theme.list.itemBlocksSeparatorColor - let addressItem = ItemListAddressItem(theme: presentationData.theme, label: item.label, text: item.text, imageSignal: item.imageSignal, sectionId: 0, style: .blocks, displayDecorations: false, action: nil, longTapAction: item.longTapAction, linkItemAction: item.linkItemAction) + let addressItem = ItemListAddressItem(theme: presentationData.theme, label: item.label, text: item.text, imageSignal: item.imageSignal, sectionId: 0, style: .blocks, displayDecorations: false, action: nil, longTapAction: nil, linkItemAction: item.linkItemAction) let params = ListViewItemLayoutParams(width: width, leftInset: safeInsets.left, rightInset: safeInsets.right, availableHeight: 1000.0) @@ -90,7 +100,7 @@ private final class PeerInfoScreenAddressItemNode: PeerInfoScreenItemNode { itemNode = current addressItem.updateNode(async: { $0() }, node: { return itemNode - }, params: params, previousItem: nil, nextItem: nil, animation: .None, completion: { (layout, apply) in + }, params: params, previousItem: topItem != nil ? addressItem : nil, nextItem: bottomItem != nil ? addressItem : nil, animation: .None, completion: { (layout, apply) in let nodeFrame = CGRect(origin: CGPoint(), size: CGSize(width: width, height: layout.size.height)) itemNode.contentSize = layout.contentSize @@ -119,7 +129,7 @@ private final class PeerInfoScreenAddressItemNode: PeerInfoScreenItemNode { self.maskNode.image = hasCorners ? PresentationResourcesItemList.cornersImage(presentationData.theme, top: hasTopCorners, bottom: hasBottomCorners) : nil self.maskNode.frame = CGRect(origin: CGPoint(x: safeInsets.left, y: 0.0), size: CGSize(width: width - safeInsets.left - safeInsets.right, height: height)) - self.bottomSeparatorNode.isHidden = hasBottomCorners + self.bottomSeparatorNode.isHidden = hasBottomCorners || bottomItem != nil transition.updateFrame(node: itemNode, frame: CGRect(origin: CGPoint(), size: itemNode.bounds.size)) diff --git a/submodules/TelegramUI/Components/PeerInfo/PeerInfoScreen/Sources/ListItems/PeerInfoScreenBusinessHoursItem.swift b/submodules/TelegramUI/Components/PeerInfo/PeerInfoScreen/Sources/ListItems/PeerInfoScreenBusinessHoursItem.swift new file mode 100644 index 00000000000..9e43592add3 --- /dev/null +++ b/submodules/TelegramUI/Components/PeerInfo/PeerInfoScreen/Sources/ListItems/PeerInfoScreenBusinessHoursItem.swift @@ -0,0 +1,624 @@ +import AsyncDisplayKit +import Display +import TelegramPresentationData +import AccountContext +import TextFormat +import UIKit +import AppBundle +import TelegramStringFormatting +import ContextUI +import TelegramCore +import ComponentFlow +import MultilineTextComponent +import BundleIconComponent +import PlainButtonComponent + +private func dayBusinessHoursText(presentationData: PresentationData, day: TelegramBusinessHours.WeekDay, offsetMinutes: Int, formatAsPlainText: Bool = false) -> String { + var businessHoursText: String = "" + switch day { + case .open: + businessHoursText += presentationData.strings.PeerInfo_BusinessHours_DayOpen24h + case .closed: + businessHoursText += presentationData.strings.PeerInfo_BusinessHours_DayClosed + case let .intervals(intervals): + func clipMinutes(_ value: Int) -> Int { + var value = value + if value < 0 { + value = 24 * 60 + value + } + return value % (24 * 60) + } + + var resultText: String = "" + for range in intervals { + let range = TelegramBusinessHours.WorkingTimeInterval(startMinute: range.startMinute + offsetMinutes, endMinute: range.endMinute + offsetMinutes) + + if !resultText.isEmpty { + if formatAsPlainText { + resultText.append(", ") + } else { + resultText.append("\n") + } + } + let startHours = clipMinutes(range.startMinute) / 60 + let startMinutes = clipMinutes(range.startMinute) % 60 + let startText = stringForShortTimestamp(hours: Int32(startHours), minutes: Int32(startMinutes), dateTimeFormat: presentationData.dateTimeFormat, formatAsPlainText: formatAsPlainText) + let endHours = clipMinutes(range.endMinute) / 60 + let endMinutes = clipMinutes(range.endMinute) % 60 + let endText = stringForShortTimestamp(hours: Int32(endHours), minutes: Int32(endMinutes), dateTimeFormat: presentationData.dateTimeFormat, formatAsPlainText: formatAsPlainText) + resultText.append("\(startText) - \(endText)") + } + businessHoursText += resultText + } + + return businessHoursText +} + +final class PeerInfoScreenBusinessHoursItem: PeerInfoScreenItem { + let id: AnyHashable + let label: String + let businessHours: TelegramBusinessHours + let requestLayout: (Bool) -> Void + let longTapAction: (ASDisplayNode, String) -> Void + + init( + id: AnyHashable, + label: String, + businessHours: TelegramBusinessHours, + requestLayout: @escaping (Bool) -> Void, + longTapAction: @escaping (ASDisplayNode, String) -> Void + ) { + self.id = id + self.label = label + self.businessHours = businessHours + self.requestLayout = requestLayout + self.longTapAction = longTapAction + } + + func node() -> PeerInfoScreenItemNode { + return PeerInfoScreenBusinessHoursItemNode() + } +} + +private final class PeerInfoScreenBusinessHoursItemNode: PeerInfoScreenItemNode { + private let containerNode: ContextControllerSourceNode + private let contextSourceNode: ContextExtractedContentContainingNode + + private let extractedBackgroundImageNode: ASImageNode + + private var extractedRect: CGRect? + private var nonExtractedRect: CGRect? + + private let maskNode: ASImageNode + private let labelNode: ImmediateTextNode + private let currentStatusText = ComponentView() + private let currentDayText = ComponentView() + private var timezoneSwitchButton: ComponentView? + private var dayTitles: [ComponentView] = [] + private var dayValues: [ComponentView] = [] + private let arrowIcon = ComponentView() + + private let bottomSeparatorNode: ASDisplayNode + + private let activateArea: AccessibilityAreaNode + + private var item: PeerInfoScreenBusinessHoursItem? + private var presentationData: PresentationData? + private var theme: PresentationTheme? + + private var currentTimezone: TimeZone + private var displayLocalTimezone: Bool = false + private var cachedDays: [TelegramBusinessHours.WeekDay] = [] + private var cachedWeekMinuteSet = IndexSet() + + private var isExpanded: Bool = false + + override init() { + self.contextSourceNode = ContextExtractedContentContainingNode() + self.containerNode = ContextControllerSourceNode() + + self.extractedBackgroundImageNode = ASImageNode() + self.extractedBackgroundImageNode.displaysAsynchronously = false + self.extractedBackgroundImageNode.alpha = 0.0 + + self.maskNode = ASImageNode() + self.maskNode.isUserInteractionEnabled = false + + self.labelNode = ImmediateTextNode() + self.labelNode.displaysAsynchronously = false + self.labelNode.isUserInteractionEnabled = false + + self.bottomSeparatorNode = ASDisplayNode() + self.bottomSeparatorNode.isLayerBacked = true + + self.activateArea = AccessibilityAreaNode() + + self.currentTimezone = TimeZone.current + + super.init() + + self.addSubnode(self.bottomSeparatorNode) + + self.containerNode.addSubnode(self.contextSourceNode) + self.containerNode.targetNodeForActivationProgress = self.contextSourceNode.contentNode + self.addSubnode(self.containerNode) + + self.addSubnode(self.maskNode) + + self.contextSourceNode.contentNode.clipsToBounds = true + + self.contextSourceNode.contentNode.addSubnode(self.extractedBackgroundImageNode) + self.contextSourceNode.contentNode.addSubnode(self.labelNode) + + self.addSubnode(self.activateArea) + + self.containerNode.isGestureEnabled = false + + self.contextSourceNode.willUpdateIsExtractedToContextPreview = { [weak self] isExtracted, transition in + guard let strongSelf = self, let theme = strongSelf.theme else { + return + } + + if isExtracted { + strongSelf.extractedBackgroundImageNode.image = generateStretchableFilledCircleImage(diameter: 28.0, color: theme.list.plainBackgroundColor) + } + + if let extractedRect = strongSelf.extractedRect, let nonExtractedRect = strongSelf.nonExtractedRect { + let rect = isExtracted ? extractedRect : nonExtractedRect + transition.updateFrame(node: strongSelf.extractedBackgroundImageNode, frame: rect) + } + + transition.updateAlpha(node: strongSelf.extractedBackgroundImageNode, alpha: isExtracted ? 1.0 : 0.0, completion: { _ in + if !isExtracted { + self?.extractedBackgroundImageNode.image = nil + } + }) + } + } + + override func didLoad() { + super.didLoad() + + let recognizer = TapLongTapOrDoubleTapGestureRecognizer(target: self, action: #selector(self.tapLongTapOrDoubleTapGesture(_:))) + recognizer.tapActionAtPoint = { _ in + return .waitForSingleTap + } + recognizer.highlight = { [weak self] point in + guard let strongSelf = self else { + return + } + strongSelf.updateTouchesAtPoint(point) + } + self.view.addGestureRecognizer(recognizer) + } + + @objc private func tapLongTapOrDoubleTapGesture(_ recognizer: TapLongTapOrDoubleTapGestureRecognizer) { + switch recognizer.state { + case .ended: + if let (gesture, _) = recognizer.lastRecognizedGestureAndLocation { + switch gesture { + case .tap: + self.isExpanded = !self.isExpanded + self.item?.requestLayout(true) + case .longTap: + if let item = self.item, let presentationData = self.presentationData { + var text = "" + + var timezoneOffsetMinutes: Int = 0 + if self.displayLocalTimezone { + var currentCalendar = Calendar(identifier: .gregorian) + currentCalendar.timeZone = TimeZone(identifier: item.businessHours.timezoneId) ?? TimeZone.current + + timezoneOffsetMinutes = (self.currentTimezone.secondsFromGMT() - currentCalendar.timeZone.secondsFromGMT()) / 60 + } + + let businessDays: [TelegramBusinessHours.WeekDay] = self.cachedDays + + for i in 0 ..< businessDays.count { + let dayTitleValue: String + switch i { + case 0: + dayTitleValue = presentationData.strings.Weekday_Monday + case 1: + dayTitleValue = presentationData.strings.Weekday_Tuesday + case 2: + dayTitleValue = presentationData.strings.Weekday_Wednesday + case 3: + dayTitleValue = presentationData.strings.Weekday_Thursday + case 4: + dayTitleValue = presentationData.strings.Weekday_Friday + case 5: + dayTitleValue = presentationData.strings.Weekday_Saturday + case 6: + dayTitleValue = presentationData.strings.Weekday_Sunday + default: + dayTitleValue = " " + } + + let businessHoursText = dayBusinessHoursText(presentationData: presentationData, day: businessDays[i], offsetMinutes: timezoneOffsetMinutes, formatAsPlainText: true) + + if !text.isEmpty { + text.append("\n") + } + text.append("\(dayTitleValue): \(businessHoursText)") + } + + item.longTapAction(self, text) + } + default: + break + } + } + default: + break + } + } + + override func update(width: CGFloat, safeInsets: UIEdgeInsets, presentationData: PresentationData, item: PeerInfoScreenItem, topItem: PeerInfoScreenItem?, bottomItem: PeerInfoScreenItem?, hasCorners: Bool, transition: ContainedViewLayoutTransition) -> CGFloat { + guard let item = item as? PeerInfoScreenBusinessHoursItem else { + return 10.0 + } + + let businessDays: [TelegramBusinessHours.WeekDay] + if self.item?.businessHours != item.businessHours { + businessDays = item.businessHours.splitIntoWeekDays() + self.cachedDays = businessDays + self.cachedWeekMinuteSet = item.businessHours.weekMinuteSet() + } else { + businessDays = self.cachedDays + } + + self.item = item + self.presentationData = presentationData + self.theme = presentationData.theme + + let sideInset: CGFloat = 16.0 + safeInsets.left + + self.bottomSeparatorNode.backgroundColor = presentationData.theme.list.itemBlocksSeparatorColor + + self.labelNode.attributedText = NSAttributedString(string: item.label, font: Font.regular(14.0), textColor: presentationData.theme.list.itemPrimaryTextColor) + + let labelSize = self.labelNode.updateLayout(CGSize(width: width - sideInset * 2.0, height: .greatestFiniteMagnitude)) + + var topOffset = 10.0 + let labelFrame = CGRect(origin: CGPoint(x: sideInset, y: topOffset), size: labelSize) + if labelSize.height > 0.0 { + topOffset += labelSize.height + topOffset += 3.0 + } + + let arrowIconSize = self.arrowIcon.update( + transition: .immediate, + component: AnyComponent(BundleIconComponent( + name: "Item List/DownArrow", + tintColor: presentationData.theme.list.disclosureArrowColor + )), + environment: {}, + containerSize: CGSize(width: 100.0, height: 100.0) + ) + let arrowIconFrame = CGRect(origin: CGPoint(x: width - sideInset + 1.0 - arrowIconSize.width, y: topOffset + 5.0), size: arrowIconSize) + if let arrowIconView = self.arrowIcon.view { + if arrowIconView.superview == nil { + self.contextSourceNode.contentNode.view.addSubview(arrowIconView) + arrowIconView.frame = arrowIconFrame + } + transition.updatePosition(layer: arrowIconView.layer, position: arrowIconFrame.center) + transition.updateBounds(layer: arrowIconView.layer, bounds: CGRect(origin: CGPoint(), size: arrowIconFrame.size)) + transition.updateTransformRotation(view: arrowIconView, angle: self.isExpanded ? CGFloat.pi * 1.0 : CGFloat.pi * 0.0) + } + + var currentCalendar = Calendar(identifier: .gregorian) + currentCalendar.timeZone = TimeZone(identifier: item.businessHours.timezoneId) ?? TimeZone.current + let currentDate = Date() + var currentDayIndex = currentCalendar.component(.weekday, from: currentDate) + if currentDayIndex == 1 { + currentDayIndex = 6 + } else { + currentDayIndex -= 2 + } + + let currentMinute = currentCalendar.component(.minute, from: currentDate) + let currentHour = currentCalendar.component(.hour, from: currentDate) + let currentWeekMinute = currentDayIndex * 24 * 60 + currentHour * 60 + currentMinute + + var timezoneOffsetMinutes: Int = 0 + if self.displayLocalTimezone { + timezoneOffsetMinutes = (self.currentTimezone.secondsFromGMT() - currentCalendar.timeZone.secondsFromGMT()) / 60 + } + + let isOpen = self.cachedWeekMinuteSet.contains(currentWeekMinute) + let openStatusText = isOpen ? presentationData.strings.PeerInfo_BusinessHours_StatusOpen : presentationData.strings.PeerInfo_BusinessHours_StatusClosed + + var currentDayStatusText = currentDayIndex >= 0 && currentDayIndex < businessDays.count ? dayBusinessHoursText(presentationData: presentationData, day: businessDays[currentDayIndex], offsetMinutes: timezoneOffsetMinutes) : " " + + if !isOpen { + for range in self.cachedWeekMinuteSet.rangeView { + if range.lowerBound > currentWeekMinute { + let openInMinutes = range.lowerBound - currentWeekMinute + if openInMinutes < 60 { + currentDayStatusText = presentationData.strings.PeerInfo_BusinessHours_StatusOpensInMinutes(Int32(openInMinutes)) + } else if openInMinutes < 6 * 60 { + currentDayStatusText = presentationData.strings.PeerInfo_BusinessHours_StatusOpensInHours(Int32(openInMinutes / 60)) + } else { + let openDate = currentDate.addingTimeInterval(Double(openInMinutes * 60)) + let openTimestamp = Int32(openDate.timeIntervalSince1970) + Int32(currentCalendar.timeZone.secondsFromGMT() - TimeZone.current.secondsFromGMT()) + + let dateText = humanReadableStringForTimestamp(strings: presentationData.strings, dateTimeFormat: presentationData.dateTimeFormat, timestamp: openTimestamp, alwaysShowTime: true, allowYesterday: false, format: HumanReadableStringFormat( + dateFormatString: { value in + let text = PresentationStrings.FormattedString(string: presentationData.strings.Chat_MessageSeenTimestamp_Date(value).string, ranges: []) + return presentationData.strings.PeerInfo_BusinessHours_StatusOpensOnDate(text.string) + }, + tomorrowFormatString: { value in + return PresentationStrings.FormattedString(string: presentationData.strings.PeerInfo_BusinessHours_StatusOpensTomorrowAt(value).string, ranges: []) + }, + todayFormatString: { value in + return PresentationStrings.FormattedString(string: presentationData.strings.PeerInfo_BusinessHours_StatusOpensTodayAt(value).string, ranges: []) + }, + yesterdayFormatString: { value in + return PresentationStrings.FormattedString(string: presentationData.strings.PeerInfo_BusinessHours_StatusOpensTodayAt(value).string, ranges: []) + } + )).string + currentDayStatusText = dateText + } + break + } + } + } + + let currentStatusTextSize = self.currentStatusText.update( + transition: .immediate, + component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: openStatusText, font: Font.regular(15.0), textColor: isOpen ? presentationData.theme.list.freeTextSuccessColor : presentationData.theme.list.itemDestructiveColor)) + )), + environment: {}, + containerSize: CGSize(width: width - sideInset * 2.0, height: 100.0) + ) + let currentStatusTextFrame = CGRect(origin: CGPoint(x: sideInset, y: topOffset), size: currentStatusTextSize) + if let currentStatusTextView = self.currentStatusText.view { + if currentStatusTextView.superview == nil { + currentStatusTextView.layer.anchorPoint = CGPoint() + self.contextSourceNode.contentNode.view.addSubview(currentStatusTextView) + } + transition.updatePosition(layer: currentStatusTextView.layer, position: currentStatusTextFrame.origin) + currentStatusTextView.bounds = CGRect(origin: CGPoint(), size: currentStatusTextFrame.size) + } + + let dayRightInset = sideInset + 17.0 + + let currentDayTextSize = self.currentDayText.update( + transition: .immediate, + component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: currentDayStatusText, font: Font.regular(15.0), textColor: presentationData.theme.list.itemSecondaryTextColor)), + horizontalAlignment: .right, + maximumNumberOfLines: 0 + )), + environment: {}, + containerSize: CGSize(width: width - sideInset - dayRightInset, height: 100.0) + ) + + var timezoneSwitchButtonSize: CGSize? + if item.businessHours.timezoneId != self.currentTimezone.identifier { + let timezoneSwitchButton: ComponentView + if let current = self.timezoneSwitchButton { + timezoneSwitchButton = current + } else { + timezoneSwitchButton = ComponentView() + self.timezoneSwitchButton = timezoneSwitchButton + } + let timezoneSwitchTitle: String + if self.displayLocalTimezone { + timezoneSwitchTitle = presentationData.strings.PeerInfo_BusinessHours_TimezoneSwitchMy + } else { + timezoneSwitchTitle = presentationData.strings.PeerInfo_BusinessHours_TimezoneSwitchBusiness + } + timezoneSwitchButtonSize = timezoneSwitchButton.update( + transition: .immediate, + component: AnyComponent(PlainButtonComponent( + content: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: timezoneSwitchTitle, font: Font.regular(12.0), textColor: presentationData.theme.list.itemAccentColor)) + )), + background: AnyComponent(RoundedRectangle( + color: presentationData.theme.list.itemAccentColor.withMultipliedAlpha(0.1), + cornerRadius: nil + )), + effectAlignment: .center, + contentInsets: UIEdgeInsets(top: 1.0, left: 7.0, bottom: 2.0, right: 7.0), + action: { [weak self] in + guard let self else { + return + } + self.displayLocalTimezone = !self.displayLocalTimezone + self.item?.requestLayout(false) + }, + animateAlpha: true, + animateScale: false, + animateContents: false + )), + environment: {}, + containerSize: CGSize(width: 100.0, height: 100.0) + ) + } else { + if let timezoneSwitchButton = self.timezoneSwitchButton { + self.timezoneSwitchButton = nil + timezoneSwitchButton.view?.removeFromSuperview() + } + } + + let timezoneSwitchButtonSpacing: CGFloat = 3.0 + if timezoneSwitchButtonSize != nil { + topOffset -= 20.0 + } + + let currentDayTextFrame = CGRect(origin: CGPoint(x: width - dayRightInset - currentDayTextSize.width, y: topOffset), size: currentDayTextSize) + if let currentDayTextView = self.currentDayText.view { + if currentDayTextView.superview == nil { + currentDayTextView.layer.anchorPoint = CGPoint() + self.contextSourceNode.contentNode.view.addSubview(currentDayTextView) + } + transition.updatePosition(layer: currentDayTextView.layer, position: currentDayTextFrame.origin) + currentDayTextView.bounds = CGRect(origin: CGPoint(), size: currentDayTextFrame.size) + } + + topOffset += max(currentStatusTextSize.height, currentDayTextSize.height) + + if let timezoneSwitchButtonView = self.timezoneSwitchButton?.view, let timezoneSwitchButtonSize { + topOffset += timezoneSwitchButtonSpacing + + var timezoneSwitchButtonTransition = transition + if timezoneSwitchButtonView.superview == nil { + timezoneSwitchButtonTransition = .immediate + self.contextSourceNode.contentNode.view.addSubview(timezoneSwitchButtonView) + } + let timezoneSwitchButtonFrame = CGRect(origin: CGPoint(x: width - dayRightInset - timezoneSwitchButtonSize.width, y: topOffset), size: timezoneSwitchButtonSize) + timezoneSwitchButtonTransition.updateFrame(view: timezoneSwitchButtonView, frame: timezoneSwitchButtonFrame) + + topOffset += timezoneSwitchButtonSize.height + } + + let daySpacing: CGFloat = 15.0 + + var dayHeights: CGFloat = 0.0 + + for rawI in 0 ..< businessDays.count { + if rawI == 0 { + //skip current day + continue + } + let i = (rawI + currentDayIndex) % businessDays.count + + dayHeights += daySpacing + + var dayTransition = transition + let dayTitle: ComponentView + if self.dayTitles.count > i { + dayTitle = self.dayTitles[i] + } else { + dayTransition = .immediate + dayTitle = ComponentView() + self.dayTitles.append(dayTitle) + } + + let dayValue: ComponentView + if self.dayValues.count > i { + dayValue = self.dayValues[i] + } else { + dayValue = ComponentView() + self.dayValues.append(dayValue) + } + + let dayTitleValue: String + switch i { + case 0: + dayTitleValue = presentationData.strings.Weekday_Monday + case 1: + dayTitleValue = presentationData.strings.Weekday_Tuesday + case 2: + dayTitleValue = presentationData.strings.Weekday_Wednesday + case 3: + dayTitleValue = presentationData.strings.Weekday_Thursday + case 4: + dayTitleValue = presentationData.strings.Weekday_Friday + case 5: + dayTitleValue = presentationData.strings.Weekday_Saturday + case 6: + dayTitleValue = presentationData.strings.Weekday_Sunday + default: + dayTitleValue = " " + } + + let businessHoursText = dayBusinessHoursText(presentationData: presentationData, day: businessDays[i], offsetMinutes: timezoneOffsetMinutes) + + let dayTitleSize = dayTitle.update( + transition: .immediate, + component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: dayTitleValue, font: Font.regular(15.0), textColor: presentationData.theme.list.itemPrimaryTextColor)) + )), + environment: {}, + containerSize: CGSize(width: width - sideInset * 2.0, height: 100.0) + ) + let dayTitleFrame = CGRect(origin: CGPoint(x: sideInset, y: topOffset + dayHeights), size: dayTitleSize) + if let dayTitleView = dayTitle.view { + if dayTitleView.superview == nil { + dayTitleView.layer.anchorPoint = CGPoint() + self.contextSourceNode.contentNode.view.addSubview(dayTitleView) + dayTitleView.alpha = 0.0 + } + dayTransition.updatePosition(layer: dayTitleView.layer, position: dayTitleFrame.origin) + dayTitleView.bounds = CGRect(origin: CGPoint(), size: dayTitleFrame.size) + + transition.updateAlpha(layer: dayTitleView.layer, alpha: self.isExpanded ? 1.0 : 0.0) + } + + let dayValueSize = dayValue.update( + transition: .immediate, + component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: businessHoursText, font: Font.regular(15.0), textColor: presentationData.theme.list.itemSecondaryTextColor, paragraphAlignment: .right)), + horizontalAlignment: .right, + maximumNumberOfLines: 0 + )), + environment: {}, + containerSize: CGSize(width: width - sideInset - dayRightInset, height: 100.0) + ) + let dayValueFrame = CGRect(origin: CGPoint(x: width - dayRightInset - dayValueSize.width, y: topOffset + dayHeights), size: dayValueSize) + if let dayValueView = dayValue.view { + if dayValueView.superview == nil { + dayValueView.layer.anchorPoint = CGPoint(x: 1.0, y: 0.0) + self.contextSourceNode.contentNode.view.addSubview(dayValueView) + dayValueView.alpha = 0.0 + } + dayTransition.updatePosition(layer: dayValueView.layer, position: CGPoint(x: dayValueFrame.maxX, y: dayValueFrame.minY)) + dayValueView.bounds = CGRect(origin: CGPoint(), size: dayValueFrame.size) + + transition.updateAlpha(layer: dayValueView.layer, alpha: self.isExpanded ? 1.0 : 0.0) + } + + dayHeights += max(dayTitleSize.height, dayValueSize.height) + } + + if self.isExpanded { + topOffset += dayHeights + } + + topOffset += 11.0 + + transition.updateFrame(node: self.labelNode, frame: labelFrame) + + let height = topOffset + + transition.updateFrame(node: self.bottomSeparatorNode, frame: CGRect(origin: CGPoint(x: sideInset, y: height - UIScreenPixel), size: CGSize(width: width - sideInset, height: UIScreenPixel))) + transition.updateAlpha(node: self.bottomSeparatorNode, alpha: bottomItem == nil ? 0.0 : 1.0) + + let hasCorners = hasCorners && (topItem == nil || bottomItem == nil) + let hasTopCorners = hasCorners && topItem == nil + let hasBottomCorners = hasCorners && bottomItem == nil + + self.maskNode.image = hasCorners ? PresentationResourcesItemList.cornersImage(presentationData.theme, top: hasTopCorners, bottom: hasBottomCorners) : nil + self.maskNode.frame = CGRect(origin: CGPoint(x: safeInsets.left, y: 0.0), size: CGSize(width: width - safeInsets.left - safeInsets.right, height: height)) + self.bottomSeparatorNode.isHidden = hasBottomCorners + + self.activateArea.frame = CGRect(origin: CGPoint(), size: CGSize(width: width, height: height)) + self.activateArea.accessibilityLabel = item.label + + let contentSize = CGSize(width: width, height: height) + self.containerNode.frame = CGRect(origin: CGPoint(), size: contentSize) + self.contextSourceNode.frame = CGRect(origin: CGPoint(), size: contentSize) + transition.updateFrame(node: self.contextSourceNode.contentNode, frame: CGRect(origin: CGPoint(), size: contentSize)) + + let nonExtractedRect = CGRect(origin: CGPoint(), size: CGSize(width: contentSize.width, height: contentSize.height)) + let extractedRect = nonExtractedRect + self.extractedRect = extractedRect + self.nonExtractedRect = nonExtractedRect + + if self.contextSourceNode.isExtractedToContextPreview { + self.extractedBackgroundImageNode.frame = extractedRect + } else { + self.extractedBackgroundImageNode.frame = nonExtractedRect + } + self.contextSourceNode.contentRect = extractedRect + + return height + } + + private func updateTouchesAtPoint(_ point: CGPoint?) { + } +} diff --git a/submodules/TelegramUI/Components/PeerInfo/PeerInfoScreen/Sources/PeerInfoScreen.swift b/submodules/TelegramUI/Components/PeerInfo/PeerInfoScreen/Sources/PeerInfoScreen.swift index 8db7a04c952..7730007aced 100644 --- a/submodules/TelegramUI/Components/PeerInfo/PeerInfoScreen/Sources/PeerInfoScreen.swift +++ b/submodules/TelegramUI/Components/PeerInfo/PeerInfoScreen/Sources/PeerInfoScreen.swift @@ -360,6 +360,8 @@ final class PeerInfoSelectionPanelNode: ASDisplayNode { }, sendContextResult: { _, _, _, _ in return false }, sendBotCommand: { _, _ in + }, sendShortcut: { _ in + }, openEditShortcuts: { }, sendBotStart: { _ in }, botSwitchChatWithPayload: { _, _ in }, beginMediaRecording: { _ in @@ -495,6 +497,8 @@ private enum PeerInfoContextSubject { case ngId(String) case regDate(String) case link(customLink: String?) + case businessHours(String) + case businessLocation(String) } private enum PeerInfoSettingsSection { @@ -1002,10 +1006,13 @@ private func settingsItems(data: PeerInfoScreenData?, context: AccountContext, p items[.payment]!.append(PeerInfoScreenDisclosureItem(id: 100, label: .text(""), text: presentationData.strings.Settings_Premium, icon: PresentationResourcesSettings.premium, action: { interaction.openSettings(.premium) })) + items[.payment]!.append(PeerInfoScreenDisclosureItem(id: 101, label: .text(""), additionalBadgeLabel: presentationData.strings.Settings_New, text: presentationData.strings.Settings_Business, icon: PresentationResourcesSettings.business, action: { + interaction.openSettings(.businessSetup) + })) // MARK: Nicegram, comment this item (hide "Gift Premium") /* - items[.payment]!.append(PeerInfoScreenDisclosureItem(id: 102, label: .text(""), additionalBadgeLabel: presentationData.strings.Settings_New, text: presentationData.strings.Settings_PremiumGift, icon: PresentationResourcesSettings.premiumGift, action: { + items[.payment]!.append(PeerInfoScreenDisclosureItem(id: 102, label: .text(""), text: presentationData.strings.Settings_PremiumGift, icon: PresentationResourcesSettings.premiumGift, action: { interaction.openSettings(.premiumGift) })) */ @@ -1262,6 +1269,49 @@ private func infoItems(data: PeerInfoScreenData?, context: AccountContext, prese interaction.requestLayout(false) })) } + + if let businessHours = cachedData.businessHours { + items[.peerInfo]!.append(PeerInfoScreenBusinessHoursItem(id: 300, label: presentationData.strings.PeerInfo_BusinessHours_Label, businessHours: businessHours, requestLayout: { animated in + interaction.requestLayout(animated) + }, longTapAction: { sourceNode, text in + if !text.isEmpty { + interaction.openPeerInfoContextMenu(.businessHours(text), sourceNode, nil) + } + })) + } + + if let businessLocation = cachedData.businessLocation { + if let coordinates = businessLocation.coordinates { + let imageSignal = chatMapSnapshotImage(engine: context.engine, resource: MapSnapshotMediaResource(latitude: coordinates.latitude, longitude: coordinates.longitude, width: 90, height: 90)) + items[.peerInfo]!.append(PeerInfoScreenAddressItem( + id: 301, + label: presentationData.strings.PeerInfo_Location_Label, + text: businessLocation.address, + imageSignal: imageSignal, + action: { + interaction.openLocation() + }, + longTapAction: { sourceNode, text in + if !text.isEmpty { + interaction.openPeerInfoContextMenu(.businessLocation(text), sourceNode, nil) + } + } + )) + } else { + items[.peerInfo]!.append(PeerInfoScreenAddressItem( + id: 301, + label: presentationData.strings.PeerInfo_Location_Label, + text: businessLocation.address, + imageSignal: nil, + action: nil, + longTapAction: { sourceNode, text in + if !text.isEmpty { + interaction.openPeerInfoContextMenu(.businessLocation(text), sourceNode, nil) + } + } + )) + } + } } if let reactionSourceMessageId = reactionSourceMessageId, !data.isContact { items[.peerInfo]!.append(PeerInfoScreenActionItem(id: 3, text: presentationData.strings.UserInfo_SendMessage, action: { @@ -3210,6 +3260,7 @@ final class PeerInfoScreenNode: ViewControllerTracingNode, PeerInfoScreenNodePro }, openPremiumStatusInfo: { _, _, _, _ in }, openRecommendedChannelContextMenu: { _, _, _ in }, openGroupBoostInfo: { _, _ in + }, openStickerEditor: { }, requestMessageUpdate: { _, _ in }, cancelInteractiveKeyboardGestures: { }, dismissTextInput: { @@ -5918,16 +5969,24 @@ final class PeerInfoScreenNode: ViewControllerTracingNode, PeerInfoScreenNodePro } } else if let channel = peer as? TelegramChannel { if let cachedData = strongSelf.data?.cachedData as? CachedChannelData { - if case .group = channel.info { - items.append(.action(ContextMenuActionItem(text: presentationData.strings.PeerInfo_Group_Boost, badge: ContextMenuActionBadge(value: presentationData.strings.Settings_New, color: .accent, style: .label), icon: { theme in - generateTintedImage(image: UIImage(bundleImageName: "Chat/Context Menu/Boost"), color: theme.contextMenu.primaryColor) - }, action: { [weak self] _, f in - f(.dismissWithoutContent) - - self?.openBoost() - }))) + let boostTitle: String + var isNew = false + switch channel.info { + case .group: + boostTitle = presentationData.strings.PeerInfo_Group_Boost + isNew = true + case .broadcast: + boostTitle = presentationData.strings.PeerInfo_Channel_Boost } + items.append(.action(ContextMenuActionItem(text: boostTitle, badge: isNew ? ContextMenuActionBadge(value: presentationData.strings.Settings_New, color: .accent, style: .label) : nil, icon: { theme in + generateTintedImage(image: UIImage(bundleImageName: "Chat/Context Menu/Boost"), color: theme.contextMenu.primaryColor) + }, action: { [weak self] _, f in + f(.dismissWithoutContent) + + self?.openBoost() + }))) + if channel.hasPermission(.editStories) { items.append(.action(ContextMenuActionItem(text: presentationData.strings.PeerInfo_Channel_ArchivedStories, icon: { theme in generateTintedImage(image: UIImage(bundleImageName: "Chat/Context Menu/Archive"), color: theme.contextMenu.primaryColor) @@ -7786,9 +7845,21 @@ final class PeerInfoScreenNode: ViewControllerTracingNode, PeerInfoScreenNodePro } private func openLocation() { - guard let data = self.data, let peer = data.peer, let cachedData = data.cachedData as? CachedChannelData, let location = cachedData.peerGeoLocation else { + guard let data = self.data, let peer = data.peer else { + return + } + + var location: PeerGeoLocation? + if let cachedData = data.cachedData as? CachedChannelData, let locationValue = cachedData.peerGeoLocation { + location = locationValue + } else if let cachedData = data.cachedData as? CachedUserData, let businessLocation = cachedData.businessLocation, let coordinates = businessLocation.coordinates { + location = PeerGeoLocation(latitude: coordinates.latitude, longitude: coordinates.longitude, address: businessLocation.address) + } + + guard let location else { return } + let context = self.context let presentationData = self.presentationData let map = TelegramMediaMap(latitude: location.latitude, longitude: location.longitude, heading: nil, accuracyRadius: nil, geoPlace: nil, venue: MapVenue(title: EnginePeer(peer).displayTitle(strings: presentationData.strings, displayOrder: presentationData.nameDisplayOrder), address: location.address, provider: nil, id: nil, type: nil), liveBroadcastingTimeout: nil, liveProximityNotificationRadius: nil) @@ -8112,6 +8183,27 @@ final class PeerInfoScreenNode: ViewControllerTracingNode, PeerInfoScreenNodePro return nil } })) + case .businessHours(let text), .businessLocation(let text): + let presentationData = context.sharedContext.currentPresentationData.with { $0 } + + let actions: [ContextMenuAction] = [ContextMenuAction(content: .text(title: presentationData.strings.Conversation_ContextMenuCopy, accessibilityLabel: presentationData.strings.Conversation_ContextMenuCopy), action: { [weak self] in + UIPasteboard.general.string = text + + self?.controller?.present(UndoOverlayController(presentationData: presentationData, content: .copy(text: presentationData.strings.Conversation_TextCopied), elevatedLayout: false, animateInAsReplacement: false, action: { _ in return false }), in: .current) + })] + + let contextMenuController = makeContextMenuController(actions: actions) + controller.present(contextMenuController, in: .window(.root), with: ContextMenuControllerPresentationArguments(sourceNodeAndRect: { [weak self, weak sourceNode] in + if let controller = self?.controller, let sourceNode = sourceNode { + var rect = sourceNode.bounds.insetBy(dx: 0.0, dy: 2.0) + if let sourceRect = sourceRect { + rect = sourceRect.insetBy(dx: 0.0, dy: 2.0) + } + return (sourceNode, rect, controller.displayNode, controller.view.bounds) + } else { + return nil + } + })) } } @@ -9037,7 +9129,8 @@ final class PeerInfoScreenNode: ViewControllerTracingNode, PeerInfoScreenNodePro } }) case .chatFolders: - push(chatListFilterPresetListController(context: self.context, mode: .default)) + let controller = self.context.sharedContext.makeFilterSettingsController(context: self.context, modal: false, scrollToTags: false, dismissed: nil) + push(controller) case .notificationsAndSounds: if let settings = self.data?.globalSettings { push(notificationsAndSoundsController(context: self.context, exceptionsList: settings.notificationExceptions)) diff --git a/submodules/TelegramUI/Components/PeerInfo/PeerInfoStoryGridScreen/Sources/PeerInfoStoryGridScreen.swift b/submodules/TelegramUI/Components/PeerInfo/PeerInfoStoryGridScreen/Sources/PeerInfoStoryGridScreen.swift index 8dba1993637..1ae74c8100a 100644 --- a/submodules/TelegramUI/Components/PeerInfo/PeerInfoStoryGridScreen/Sources/PeerInfoStoryGridScreen.swift +++ b/submodules/TelegramUI/Components/PeerInfo/PeerInfoStoryGridScreen/Sources/PeerInfoStoryGridScreen.swift @@ -295,7 +295,23 @@ final class PeerInfoStoryGridScreenComponent: Component { let _ = paneNode.scrollToTop() } + func openCreateStory() { + guard let component = self.component else { + return + } + if let rootController = component.context.sharedContext.mainWindow?.viewController as? TelegramRootControllerInterface { + let coordinator = rootController.openStoryCamera(customTarget: nil, transitionIn: nil, transitionedIn: {}, transitionOut: { _, _ in return nil }) + coordinator?.animateIn() + } + } + + private var isUpdating = false func update(component: PeerInfoStoryGridScreenComponent, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + self.isUpdating = true + defer { + self.isUpdating = false + } + self.component = component self.state = state @@ -313,7 +329,7 @@ final class PeerInfoStoryGridScreenComponent: Component { var bottomInset: CGFloat = environment.safeInsets.bottom - if self.selectedCount != 0 { + if self.selectedCount != 0 || (component.scope == .saved && self.paneNode?.isEmpty == false) { let selectionPanel: ComponentView var selectionPanelTransition = transition if let current = self.selectionPanel { @@ -327,7 +343,7 @@ final class PeerInfoStoryGridScreenComponent: Component { let buttonText: String switch component.scope { case .saved: - buttonText = environment.strings.ChatList_Context_Archive + buttonText = self.selectedCount > 0 ? environment.strings.ChatList_Context_Archive : environment.strings.StoryList_SavedAddAction case .archive: buttonText = environment.strings.StoryList_SaveToProfile } @@ -344,7 +360,7 @@ final class PeerInfoStoryGridScreenComponent: Component { guard let self, let component = self.component, let environment = self.environment else { return } - guard let paneNode = self.paneNode, !paneNode.selectedIds.isEmpty else { + guard let paneNode = self.paneNode else { return } @@ -361,21 +377,25 @@ final class PeerInfoStoryGridScreenComponent: Component { switch component.scope { case .saved: let selectedCount = paneNode.selectedItems.count - let _ = component.context.engine.messages.updateStoriesArePinned(peerId: component.peerId, ids: paneNode.selectedItems, isPinned: false).start() - - paneNode.setIsSelectionModeActive(false) - (self.environment?.controller() as? PeerInfoStoryGridScreen)?.updateTitle() - - let presentationData = component.context.sharedContext.currentPresentationData.with({ $0 }).withUpdated(theme: environment.theme) - - let title: String = presentationData.strings.StoryList_TooltipStoriesSavedToProfile(Int32(selectedCount)) - environment.controller()?.present(UndoOverlayController( - presentationData: presentationData, - content: .info(title: nil, text: title, timeout: nil, customUndoText: nil), - elevatedLayout: false, - animateInAsReplacement: false, - action: { _ in return false } - ), in: .current) + if selectedCount == 0 { + self.openCreateStory() + } else { + let _ = component.context.engine.messages.updateStoriesArePinned(peerId: component.peerId, ids: paneNode.selectedItems, isPinned: false).start() + + paneNode.setIsSelectionModeActive(false) + (self.environment?.controller() as? PeerInfoStoryGridScreen)?.updateTitle() + + let presentationData = component.context.sharedContext.currentPresentationData.with({ $0 }).withUpdated(theme: environment.theme) + + let title: String = presentationData.strings.StoryList_TooltipStoriesSavedToProfile(Int32(selectedCount)) + environment.controller()?.present(UndoOverlayController( + presentationData: presentationData, + content: .info(title: nil, text: title, timeout: nil, customUndoText: nil), + elevatedLayout: false, + animateInAsReplacement: false, + action: { _ in return false } + ), in: .current) + } case .archive: let _ = component.context.engine.messages.updateStoriesArePinned(peerId: component.peerId, ids: paneNode.selectedItems, isPinned: true).start() @@ -449,10 +469,28 @@ final class PeerInfoStoryGridScreenComponent: Component { }, listContext: nil ) + paneNode.isEmptyUpdated = { [weak self] _ in + guard let self else { + return + } + if !self.isUpdating { + self.state?.updated(transition: .immediate) + } + } self.paneNode = paneNode - self.addSubview(paneNode.view) + if let selectionPanelView = self.selectionPanel?.view { + self.insertSubview(paneNode.view, belowSubview: selectionPanelView) + } else { + self.addSubview(paneNode.view) + } paneNode.emptyAction = { [weak self] in + guard let self else { + return + } + self.openCreateStory() + } + paneNode.additionalEmptyAction = { [weak self] in guard let self, let component = self.component else { return } diff --git a/submodules/TelegramUI/Components/PeerInfo/PeerInfoVisualMediaPaneNode/Sources/PeerInfoStoryPaneNode.swift b/submodules/TelegramUI/Components/PeerInfo/PeerInfoVisualMediaPaneNode/Sources/PeerInfoStoryPaneNode.swift index b7d525ded01..99d64be4437 100644 --- a/submodules/TelegramUI/Components/PeerInfo/PeerInfoVisualMediaPaneNode/Sources/PeerInfoStoryPaneNode.swift +++ b/submodules/TelegramUI/Components/PeerInfo/PeerInfoVisualMediaPaneNode/Sources/PeerInfoStoryPaneNode.swift @@ -948,6 +948,8 @@ public final class PeerInfoStoryPaneNode: ASDisplayNode, PeerInfoPaneNode, UIScr } } + public var isEmptyUpdated: (Bool) -> Void = { _ in } + public private(set) var isSelectionModeActive: Bool private var currentParams: (size: CGSize, topInset: CGFloat, sideInset: CGFloat, bottomInset: CGFloat, deviceMetrics: DeviceMetrics, visibleHeight: CGFloat, isScrollingLockedAtTop: Bool, expandProgress: CGFloat, navigationHeight: CGFloat, presentationData: PresentationData)? @@ -985,6 +987,7 @@ public final class PeerInfoStoryPaneNode: ASDisplayNode, PeerInfoPaneNode, UIScr public var openCurrentDate: (() -> Void)? public var paneDidScroll: (() -> Void)? public var emptyAction: (() -> Void)? + public var additionalEmptyAction: (() -> Void)? public var ensureRectVisible: ((UIView, CGRect) -> Void)? @@ -1729,6 +1732,7 @@ public final class PeerInfoStoryPaneNode: ASDisplayNode, PeerInfoPaneNode, UIScr private func updateHistory(items: SparseItemGrid.Items, synchronous: Bool, reloadAtTop: Bool) { self.items = items + self.isEmptyUpdated(self.isEmpty) if let (size, topInset, sideInset, bottomInset, deviceMetrics, visibleHeight, isScrollingLockedAtTop, expandProgress, navigationHeight, presentationData) = self.currentParams { var gridSnapshot: UIView? @@ -2027,14 +2031,21 @@ public final class PeerInfoStoryPaneNode: ASDisplayNode, PeerInfoPaneNode, UIScr context: self.context, theme: presentationData.theme, animationName: "StoryListEmpty", - title: self.isArchive ? presentationData.strings.StoryList_ArchivedEmptyState_Title : presentationData.strings.StoryList_SavedEmptyState_Title, - text: self.isArchive ? presentationData.strings.StoryList_ArchivedEmptyState_Text : presentationData.strings.StoryList_SavedEmptyState_Text, - actionTitle: self.isArchive ? nil : presentationData.strings.StoryList_SavedEmptyAction, + title: self.isArchive ? presentationData.strings.StoryList_ArchivedEmptyState_Title : presentationData.strings.StoryList_SavedEmptyPosts_Title, + text: self.isArchive ? presentationData.strings.StoryList_ArchivedEmptyState_Text : presentationData.strings.StoryList_SavedEmptyPosts_Text, + actionTitle: self.isArchive ? nil : presentationData.strings.StoryList_SavedAddAction, action: { [weak self] in guard let self else { return } self.emptyAction?() + }, + additionalActionTitle: self.isArchive ? nil : presentationData.strings.StoryList_SavedEmptyAction, + additionalAction: { [weak self] in + guard let self else { + return + } + self.additionalEmptyAction?() } )), environment: {}, diff --git a/submodules/TelegramUI/Components/PeerSelectionController/Sources/PeerSelectionControllerNode.swift b/submodules/TelegramUI/Components/PeerSelectionController/Sources/PeerSelectionControllerNode.swift index fa7762a1894..f57a661601f 100644 --- a/submodules/TelegramUI/Components/PeerSelectionController/Sources/PeerSelectionControllerNode.swift +++ b/submodules/TelegramUI/Components/PeerSelectionController/Sources/PeerSelectionControllerNode.swift @@ -591,6 +591,8 @@ final class PeerSelectionControllerNode: ASDisplayNode { }, sendContextResult: { _, _, _, _ in return false }, sendBotCommand: { _, _ in + }, sendShortcut: { _ in + }, openEditShortcuts: { }, sendBotStart: { _ in }, botSwitchChatWithPayload: { _, _ in }, beginMediaRecording: { _ in diff --git a/submodules/TelegramUI/Components/PlainButtonComponent/Sources/PlainButtonComponent.swift b/submodules/TelegramUI/Components/PlainButtonComponent/Sources/PlainButtonComponent.swift index cfce4784d2a..f91488ec89c 100644 --- a/submodules/TelegramUI/Components/PlainButtonComponent/Sources/PlainButtonComponent.swift +++ b/submodules/TelegramUI/Components/PlainButtonComponent/Sources/PlainButtonComponent.swift @@ -11,6 +11,7 @@ public final class PlainButtonComponent: Component { } public let content: AnyComponent + public let background: AnyComponent? public let effectAlignment: EffectAlignment public let minSize: CGSize? public let contentInsets: UIEdgeInsets @@ -23,6 +24,7 @@ public final class PlainButtonComponent: Component { public init( content: AnyComponent, + background: AnyComponent? = nil, effectAlignment: EffectAlignment, minSize: CGSize? = nil, contentInsets: UIEdgeInsets = UIEdgeInsets(), @@ -34,6 +36,7 @@ public final class PlainButtonComponent: Component { tag: AnyObject? = nil ) { self.content = content + self.background = background self.effectAlignment = effectAlignment self.minSize = minSize self.contentInsets = contentInsets @@ -49,6 +52,9 @@ public final class PlainButtonComponent: Component { if lhs.content != rhs.content { return false } + if lhs.background != rhs.background { + return false + } if lhs.effectAlignment != rhs.effectAlignment { return false } @@ -92,6 +98,7 @@ public final class PlainButtonComponent: Component { private let contentContainer = UIView() private let content = ComponentView() + private var background: ComponentView? public var contentView: UIView? { return self.content.view @@ -220,7 +227,12 @@ public final class PlainButtonComponent: Component { } let contentFrame = CGRect(origin: CGPoint(x: component.contentInsets.left + floor((size.width - component.contentInsets.left - component.contentInsets.right - contentSize.width) * 0.5), y: component.contentInsets.top + floor((size.height - component.contentInsets.top - component.contentInsets.bottom - contentSize.height) * 0.5)), size: contentSize) - contentTransition.setPosition(view: contentView, position: CGPoint(x: contentFrame.minX + contentFrame.width * contentView.layer.anchorPoint.x, y: contentFrame.minY + contentFrame.height * contentView.layer.anchorPoint.y)) + let contentPosition = CGPoint(x: contentFrame.minX + contentFrame.width * contentView.layer.anchorPoint.x, y: contentFrame.minY + contentFrame.height * contentView.layer.anchorPoint.y) + if !component.animateContents && (abs(contentView.center.x - contentPosition.x) <= 2.0 && abs(contentView.center.y - contentPosition.y) <= 2.0){ + contentView.center = contentPosition + } else { + contentTransition.setPosition(view: contentView, position: contentPosition) + } if component.animateContents { contentTransition.setBounds(view: contentView, bounds: CGRect(origin: CGPoint(), size: contentFrame.size)) @@ -243,6 +255,33 @@ public final class PlainButtonComponent: Component { transition.setBounds(view: self.contentContainer, bounds: CGRect(origin: CGPoint(), size: size)) transition.setPosition(view: self.contentContainer, position: CGPoint(x: size.width * anchorX, y: size.height * 0.5)) + if let backgroundValue = component.background { + var backgroundTransition = transition + let background: ComponentView + if let current = self.background { + background = current + } else { + backgroundTransition = .immediate + background = ComponentView() + self.background = background + } + let _ = background.update( + transition: backgroundTransition, + component: backgroundValue, + environment: {}, + containerSize: size + ) + if let backgroundView = background.view { + if backgroundView.superview == nil { + self.contentContainer.insertSubview(backgroundView, at: 0) + } + backgroundTransition.setFrame(view: backgroundView, frame: CGRect(origin: CGPoint(), size: size)) + } + } else if let background = self.background { + self.background = nil + background.view?.removeFromSuperview() + } + return size } } diff --git a/submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/BUILD b/submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/BUILD new file mode 100644 index 00000000000..b7f87546ebf --- /dev/null +++ b/submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/BUILD @@ -0,0 +1,55 @@ +load("@build_bazel_rules_swift//swift:swift.bzl", "swift_library") + +swift_library( + name = "AutomaticBusinessMessageSetupScreen", + module_name = "AutomaticBusinessMessageSetupScreen", + srcs = glob([ + "Sources/**/*.swift", + ]), + copts = [ + "-warnings-as-errors", + ], + deps = [ + "//submodules/AsyncDisplayKit", + "//submodules/Display", + "//submodules/Postbox", + "//submodules/TelegramCore", + "//submodules/SSignalKit/SwiftSignalKit", + "//submodules/TelegramPresentationData", + "//submodules/AccountContext", + "//submodules/PresentationDataUtils", + "//submodules/Markdown", + "//submodules/ComponentFlow", + "//submodules/Components/ViewControllerComponent", + "//submodules/Components/BundleIconComponent", + "//submodules/Components/MultilineTextComponent", + "//submodules/Components/BalancedTextComponent", + "//submodules/TelegramUI/Components/AnimatedTextComponent", + "//submodules/TelegramUI/Components/ButtonComponent", + "//submodules/TelegramUI/Components/BackButtonComponent", + "//submodules/TelegramUI/Components/ListSectionComponent", + "//submodules/TelegramUI/Components/ListActionItemComponent", + "//submodules/TelegramUI/Components/ListTextFieldItemComponent", + "//submodules/TelegramUI/Components/LottieComponent", + "//submodules/AvatarNode", + "//submodules/TelegramUI/Components/PlainButtonComponent", + "//submodules/TelegramUI/Components/Stories/PeerListItemComponent", + "//submodules/TelegramUI/Components/ListItemSliderSelectorComponent", + "//submodules/ShimmerEffect", + "//submodules/ChatListUI", + "//submodules/MergeLists", + "//submodules/Components/ComponentDisplayAdapters", + "//submodules/ItemListPeerActionItem", + "//submodules/ItemListUI", + "//submodules/TelegramUI/Components/Settings/QuickReplyNameAlertController", + "//submodules/DateSelectionUI", + "//submodules/TelegramStringFormatting", + "//submodules/TelegramUI/Components/TimeSelectionActionSheet", + "//submodules/TelegramUI/Components/ChatListHeaderComponent", + "//submodules/AttachmentUI", + "//submodules/SearchBarNode", + ], + visibility = [ + "//visibility:public", + ], +) diff --git a/submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/AutomaticBusinessMessageListItemComponent.swift b/submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/AutomaticBusinessMessageListItemComponent.swift new file mode 100644 index 00000000000..ad53e21b11b --- /dev/null +++ b/submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/AutomaticBusinessMessageListItemComponent.swift @@ -0,0 +1,318 @@ +import Foundation +import UIKit +import Display +import ComponentFlow +import ListSectionComponent +import TelegramPresentationData +import AppBundle +import ChatListUI +import AccountContext +import Postbox +import TelegramCore + +final class GreetingMessageListItemComponent: Component { + let context: AccountContext + let theme: PresentationTheme + let strings: PresentationStrings + let accountPeer: EnginePeer + let message: EngineMessage + let count: Int + let action: (() -> Void)? + + init( + context: AccountContext, + theme: PresentationTheme, + strings: PresentationStrings, + accountPeer: EnginePeer, + message: EngineMessage, + count: Int, + action: (() -> Void)? = nil + ) { + self.context = context + self.theme = theme + self.strings = strings + self.accountPeer = accountPeer + self.message = message + self.count = count + self.action = action + } + + static func ==(lhs: GreetingMessageListItemComponent, rhs: GreetingMessageListItemComponent) -> Bool { + if lhs.context !== rhs.context { + return false + } + if lhs.theme !== rhs.theme { + return false + } + if lhs.strings !== rhs.strings { + return false + } + if lhs.accountPeer != rhs.accountPeer { + return false + } + if lhs.message != rhs.message { + return false + } + if lhs.count != rhs.count { + return false + } + if (lhs.action == nil) != (rhs.action == nil) { + return false + } + return true + } + + final class View: HighlightTrackingButton, ListSectionComponent.ChildView { + private var component: GreetingMessageListItemComponent? + private weak var componentState: EmptyComponentState? + + private var chatListPresentationData: ChatListPresentationData? + private var chatListNodeInteraction: ChatListNodeInteraction? + + private var itemNode: ListViewItemNode? + + var customUpdateIsHighlighted: ((Bool) -> Void)? + private(set) var separatorInset: CGFloat = 0.0 + + override init(frame: CGRect) { + super.init(frame: frame) + + self.addTarget(self, action: #selector(self.pressed), for: .touchUpInside) + self.internalHighligthedChanged = { [weak self] isHighlighted in + guard let self, let component = self.component, component.action != nil else { + return + } + if let customUpdateIsHighlighted = self.customUpdateIsHighlighted { + customUpdateIsHighlighted(isHighlighted) + } + } + } + + required init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + @objc private func pressed() { + self.component?.action?() + } + + func update(component: GreetingMessageListItemComponent, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + let previousComponent = self.component + self.component = component + self.componentState = state + + self.isEnabled = component.action != nil + + let chatListPresentationData: ChatListPresentationData + if let current = self.chatListPresentationData, let previousComponent, previousComponent.theme === component.theme { + chatListPresentationData = current + } else { + let presentationData = component.context.sharedContext.currentPresentationData.with({ $0 }) + chatListPresentationData = ChatListPresentationData( + theme: component.theme, + fontSize: presentationData.listsFontSize, + strings: component.strings, + dateTimeFormat: presentationData.dateTimeFormat, + nameSortOrder: presentationData.nameSortOrder, + nameDisplayOrder: presentationData.nameDisplayOrder, + disableAnimations: false + ) + self.chatListPresentationData = chatListPresentationData + } + + let chatListNodeInteraction: ChatListNodeInteraction + if let current = self.chatListNodeInteraction { + chatListNodeInteraction = current + } else { + chatListNodeInteraction = ChatListNodeInteraction( + context: component.context, + animationCache: component.context.animationCache, + animationRenderer: component.context.animationRenderer, + activateSearch: { + }, + peerSelected: { _, _, _, _ in + }, + disabledPeerSelected: { _, _, _ in + }, + togglePeerSelected: { _, _ in + }, + togglePeersSelection: { _, _ in + }, + additionalCategorySelected: { _ in + }, + messageSelected: { _, _, _, _ in + }, + groupSelected: { _ in + }, + addContact: { _ in + }, + setPeerIdWithRevealedOptions: { _, _ in + }, + setItemPinned: { _, _ in + }, + setPeerMuted: { _, _ in + }, + setPeerThreadMuted: { _, _, _ in + }, + deletePeer: { _, _ in + }, + deletePeerThread: { _, _ in + }, + setPeerThreadStopped: { _, _, _ in + }, + setPeerThreadPinned: { _, _, _ in + }, + setPeerThreadHidden: { _, _, _ in + }, + updatePeerGrouping: { _, _ in + }, + togglePeerMarkedUnread: { _, _ in + }, + toggleArchivedFolderHiddenByDefault: { + }, + toggleThreadsSelection: { _, _ in + }, + hidePsa: { _ in + }, + activateChatPreview: { _, _, _, _, _ in + }, + present: { _ in + }, + openForumThread: { _, _ in + }, + openStorageManagement: { + }, + openPasswordSetup: { + }, + openPremiumIntro: { + }, + openPremiumGift: { + }, + openActiveSessions: { + }, + performActiveSessionAction: { _, _ in + }, + openChatFolderUpdates: { + }, + hideChatFolderUpdates: { + }, + openStories: { _, _ in + }, + dismissNotice: { _ in + }, + editPeer: { _ in + } + ) + self.chatListNodeInteraction = chatListNodeInteraction + } + + let chatListItem = ChatListItem( + presentationData: chatListPresentationData, + context: component.context, + chatListLocation: .chatList(groupId: .root), + filterData: nil, + index: EngineChatList.Item.Index.chatList(ChatListIndex(pinningIndex: nil, messageIndex: component.message.index)), + content: .peer(ChatListItemContent.PeerData( + messages: [component.message], + peer: EngineRenderedPeer(peer: component.accountPeer), + threadInfo: nil, + combinedReadState: nil, + isRemovedFromTotalUnreadCount: false, + presence: nil, + hasUnseenMentions: false, + hasUnseenReactions: false, + draftState: nil, + mediaDraftContentType: nil, + inputActivities: nil, + promoInfo: nil, + ignoreUnreadBadge: false, + displayAsMessage: false, + hasFailedMessages: false, + forumTopicData: nil, + topForumTopicItems: [], + autoremoveTimeout: nil, + storyState: nil, + requiresPremiumForMessaging: false, + displayAsTopicList: false, + tags: [], + customMessageListData: ChatListItemContent.CustomMessageListData( + commandPrefix: nil, + searchQuery: nil, + messageCount: component.count, + hideSeparator: true + ) + )), + editing: false, + hasActiveRevealControls: false, + selected: false, + header: nil, + enableContextActions: false, + hiddenOffset: false, + interaction: chatListNodeInteraction + ) + var itemNode: ListViewItemNode? + let params = ListViewItemLayoutParams(width: availableSize.width, leftInset: 0.0, rightInset: 0.0, availableHeight: 1000.0) + if let current = self.itemNode { + itemNode = current + chatListItem.updateNode( + async: { f in f () }, + node: { + return current + }, + params: params, + previousItem: nil, + nextItem: nil, animation: .None, + completion: { layout, apply in + let nodeFrame = CGRect(origin: current.frame.origin, size: CGSize(width: layout.size.width, height: layout.size.height)) + + current.contentSize = layout.contentSize + current.insets = layout.insets + current.frame = nodeFrame + + apply(ListViewItemApply(isOnScreen: true)) + }) + } else { + var outItemNode: ListViewItemNode? + chatListItem.nodeConfiguredForParams( + async: { f in f() }, + params: params, + synchronousLoads: true, + previousItem: nil, + nextItem: nil, + completion: { node, apply in + outItemNode = node + apply().1(ListViewItemApply(isOnScreen: true)) + } + ) + itemNode = outItemNode + } + + let size = CGSize(width: availableSize.width, height: itemNode?.contentSize.height ?? 44.0) + + if self.itemNode !== itemNode { + self.itemNode?.removeFromSupernode() + + self.itemNode = itemNode + if let itemNode { + itemNode.isUserInteractionEnabled = false + self.addSubview(itemNode.view) + } + } + if let itemNode = self.itemNode { + itemNode.frame = CGRect(origin: CGPoint(), size: size) + } + + self.separatorInset = 76.0 + + return size + } + } + + func makeView() -> View { + return View(frame: CGRect()) + } + + func update(view: View, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + return view.update(component: self, availableSize: availableSize, state: state, environment: environment, transition: transition) + } +} diff --git a/submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/AutomaticBusinessMessageSetupChatContents.swift b/submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/AutomaticBusinessMessageSetupChatContents.swift new file mode 100644 index 00000000000..b002aa3cc56 --- /dev/null +++ b/submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/AutomaticBusinessMessageSetupChatContents.swift @@ -0,0 +1,252 @@ +import Foundation +import UIKit +import SwiftSignalKit +import Postbox +import TelegramCore +import AccountContext + +final class AutomaticBusinessMessageSetupChatContents: ChatCustomContentsProtocol { + private final class PendingMessageContext { + let disposable = MetaDisposable() + var message: Message? + + init() { + } + } + + private final class Impl { + let queue: Queue + let context: AccountContext + + private var shortcut: String + private var shortcutId: Int32? + + private(set) var mergedHistoryView: MessageHistoryView? + private var sourceHistoryView: MessageHistoryView? + + private var pendingMessages: [PendingMessageContext] = [] + private var historyViewDisposable: Disposable? + private var pendingHistoryViewDisposable: Disposable? + let historyViewStream = ValuePipe<(MessageHistoryView, ViewUpdateType)>() + private var nextUpdateIsHoleFill: Bool = false + + init(queue: Queue, context: AccountContext, shortcut: String, shortcutId: Int32?) { + self.queue = queue + self.context = context + self.shortcut = shortcut + self.shortcutId = shortcutId + + self.updateHistoryViewRequest(reload: false) + } + + deinit { + for context in self.pendingMessages { + context.disposable.dispose() + } + self.historyViewDisposable?.dispose() + self.pendingHistoryViewDisposable?.dispose() + } + + private func updateHistoryViewRequest(reload: Bool) { + if let shortcutId = self.shortcutId { + self.pendingHistoryViewDisposable?.dispose() + self.pendingHistoryViewDisposable = nil + + if self.historyViewDisposable == nil || reload { + self.historyViewDisposable?.dispose() + + self.historyViewDisposable = (self.context.account.viewTracker.quickReplyMessagesViewForLocation(quickReplyId: shortcutId) + |> deliverOn(self.queue)).start(next: { [weak self] view, update, _ in + guard let self else { + return + } + if update == .FillHole { + self.nextUpdateIsHoleFill = true + self.updateHistoryViewRequest(reload: true) + return + } + + let nextUpdateIsHoleFill = self.nextUpdateIsHoleFill + self.nextUpdateIsHoleFill = false + + self.sourceHistoryView = view + + if !view.entries.contains(where: { $0.message.id.namespace == Namespaces.Message.QuickReplyCloud }) { + self.shortcutId = nil + } + + self.updateHistoryView(updateType: nextUpdateIsHoleFill ? .FillHole : .Generic) + }) + } + } else { + self.historyViewDisposable?.dispose() + self.historyViewDisposable = nil + + self.pendingHistoryViewDisposable = (self.context.account.viewTracker.pendingQuickReplyMessagesViewForLocation(shortcut: self.shortcut) + |> deliverOn(self.queue)).start(next: { [weak self] view, _, _ in + guard let self else { + return + } + + let nextUpdateIsHoleFill = self.nextUpdateIsHoleFill + self.nextUpdateIsHoleFill = false + + self.sourceHistoryView = view + + self.updateHistoryView(updateType: nextUpdateIsHoleFill ? .FillHole : .Generic) + }) + + /*if self.sourceHistoryView == nil { + let sourceHistoryView = MessageHistoryView(tag: nil, namespaces: .just(Namespaces.Message.allQuickReply), entries: [], holeEarlier: false, holeLater: false, isLoading: false) + self.sourceHistoryView = sourceHistoryView + self.updateHistoryView(updateType: .Initial) + }*/ + } + } + + private func updateHistoryView(updateType: ViewUpdateType) { + var entries = self.sourceHistoryView?.entries ?? [] + for pendingMessage in self.pendingMessages { + if let message = pendingMessage.message { + if !entries.contains(where: { $0.message.stableId == message.stableId }) { + entries.append(MessageHistoryEntry( + message: message, + isRead: true, + location: nil, + monthLocation: nil, + attributes: MutableMessageHistoryEntryAttributes( + authorIsContact: false + ) + )) + } + } + } + entries.sort(by: { $0.message.index < $1.message.index }) + + let mergedHistoryView = MessageHistoryView(tag: nil, namespaces: .just(Namespaces.Message.allQuickReply), entries: entries, holeEarlier: false, holeLater: false, isLoading: false) + self.mergedHistoryView = mergedHistoryView + + self.historyViewStream.putNext((mergedHistoryView, updateType)) + } + + func enqueueMessages(messages: [EnqueueMessage]) { + let threadId = self.shortcutId.flatMap(Int64.init) + let _ = (TelegramCore.enqueueMessages(account: self.context.account, peerId: self.context.account.peerId, messages: messages.map { message in + return message.withUpdatedThreadId(threadId).withUpdatedAttributes { attributes in + var attributes = attributes + attributes.removeAll(where: { $0 is OutgoingQuickReplyMessageAttribute }) + attributes.append(OutgoingQuickReplyMessageAttribute(shortcut: self.shortcut)) + return attributes + } + }) + |> deliverOn(self.queue)).startStandalone(next: { [weak self] result in + guard let self else { + return + } + if self.shortcutId != nil { + return + } + for id in result { + if let id { + let pendingMessage = PendingMessageContext() + self.pendingMessages.append(pendingMessage) + pendingMessage.disposable.set(( + self.context.account.postbox.messageView(id) + |> deliverOn(self.queue) + ).startStrict(next: { [weak self, weak pendingMessage] messageView in + guard let self else { + return + } + guard let pendingMessage else { + return + } + pendingMessage.message = messageView.message + if let message = pendingMessage.message, message.id.namespace == Namespaces.Message.QuickReplyCloud, let threadId = message.threadId { + self.shortcutId = Int32(clamping: threadId) + self.updateHistoryViewRequest(reload: true) + } else { + self.updateHistoryView(updateType: .Generic) + } + })) + } + } + }) + } + + func deleteMessages(ids: [EngineMessage.Id]) { + let _ = self.context.engine.messages.deleteMessagesInteractively(messageIds: ids, type: .forEveryone).startStandalone() + } + + func editMessage(id: EngineMessage.Id, text: String, media: RequestEditMessageMedia, entities: TextEntitiesMessageAttribute?, webpagePreviewAttribute: WebpagePreviewMessageAttribute?, disableUrlPreview: Bool) { + } + + func quickReplyUpdateShortcut(value: String) { + self.shortcut = value + if let shortcutId = self.shortcutId { + self.context.engine.accountData.editMessageShortcut(id: shortcutId, shortcut: value) + } + } + } + + var kind: ChatCustomContentsKind + + var historyView: Signal<(MessageHistoryView, ViewUpdateType), NoError> { + return self.impl.signalWith({ impl, subscriber in + if let mergedHistoryView = impl.mergedHistoryView { + subscriber.putNext((mergedHistoryView, .Initial)) + } + return impl.historyViewStream.signal().start(next: subscriber.putNext) + }) + } + + var messageLimit: Int? { + return 20 + } + + private let queue: Queue + private let impl: QueueLocalObject + + init(context: AccountContext, kind: ChatCustomContentsKind, shortcutId: Int32?) { + self.kind = kind + + let initialShortcut: String + switch kind { + case let .quickReplyMessageInput(shortcut, _): + initialShortcut = shortcut + } + + let queue = Queue() + self.queue = queue + self.impl = QueueLocalObject(queue: queue, generate: { + return Impl(queue: queue, context: context, shortcut: initialShortcut, shortcutId: shortcutId) + }) + } + + func enqueueMessages(messages: [EnqueueMessage]) { + self.impl.with { impl in + impl.enqueueMessages(messages: messages) + } + } + + func deleteMessages(ids: [EngineMessage.Id]) { + self.impl.with { impl in + impl.deleteMessages(ids: ids) + } + } + + func editMessage(id: EngineMessage.Id, text: String, media: RequestEditMessageMedia, entities: TextEntitiesMessageAttribute?, webpagePreviewAttribute: WebpagePreviewMessageAttribute?, disableUrlPreview: Bool) { + self.impl.with { impl in + impl.editMessage(id: id, text: text, media: media, entities: entities, webpagePreviewAttribute: webpagePreviewAttribute, disableUrlPreview: disableUrlPreview) + } + } + + func quickReplyUpdateShortcut(value: String) { + switch self.kind { + case let .quickReplyMessageInput(_, shortcutType): + self.kind = .quickReplyMessageInput(shortcut: value, shortcutType: shortcutType) + self.impl.with { impl in + impl.quickReplyUpdateShortcut(value: value) + } + } + } +} diff --git a/submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/AutomaticBusinessMessageSetupScreen.swift b/submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/AutomaticBusinessMessageSetupScreen.swift new file mode 100644 index 00000000000..7e9fe2c3f2a --- /dev/null +++ b/submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/AutomaticBusinessMessageSetupScreen.swift @@ -0,0 +1,1667 @@ +import Foundation +import UIKit +import Photos +import Display +import AsyncDisplayKit +import SwiftSignalKit +import Postbox +import TelegramCore +import TelegramPresentationData +import TelegramUIPreferences +import PresentationDataUtils +import AccountContext +import ComponentFlow +import ViewControllerComponent +import MultilineTextComponent +import BalancedTextComponent +import BackButtonComponent +import ListSectionComponent +import ListActionItemComponent +import ListTextFieldItemComponent +import BundleIconComponent +import LottieComponent +import Markdown +import PeerListItemComponent +import AvatarNode +import ListItemSliderSelectorComponent +import DateSelectionUI +import PlainButtonComponent +import TelegramStringFormatting +import TimeSelectionActionSheet + +private let checkIcon: UIImage = { + return generateImage(CGSize(width: 12.0, height: 10.0), rotatedContext: { size, context in + context.clear(CGRect(origin: CGPoint(), size: size)) + context.setStrokeColor(UIColor.white.cgColor) + context.setLineWidth(1.98) + context.setLineCap(.round) + context.setLineJoin(.round) + context.translateBy(x: 1.0, y: 1.0) + + let _ = try? drawSvgPath(context, path: "M0.215053763,4.36080467 L3.31621263,7.70466293 L3.31621263,7.70466293 C3.35339229,7.74475231 3.41603123,7.74711109 3.45612061,7.70993143 C3.45920681,7.70706923 3.46210733,7.70401312 3.46480451,7.70078171 L9.89247312,0 S ") + })!.withRenderingMode(.alwaysTemplate) +}() + +final class AutomaticBusinessMessageSetupScreenComponent: Component { + typealias EnvironmentType = ViewControllerComponentContainer.Environment + + let context: AccountContext + let initialData: AutomaticBusinessMessageSetupScreen.InitialData + let mode: AutomaticBusinessMessageSetupScreen.Mode + + init( + context: AccountContext, + initialData: AutomaticBusinessMessageSetupScreen.InitialData, + mode: AutomaticBusinessMessageSetupScreen.Mode + ) { + self.context = context + self.initialData = initialData + self.mode = mode + } + + static func ==(lhs: AutomaticBusinessMessageSetupScreenComponent, rhs: AutomaticBusinessMessageSetupScreenComponent) -> Bool { + if lhs.context !== rhs.context { + return false + } + if lhs.mode != rhs.mode { + return false + } + + return true + } + + private final class ScrollView: UIScrollView { + override func touchesShouldCancel(in view: UIView) -> Bool { + return true + } + } + + struct AdditionalPeerList { + enum Category: Int { + case newChats = 0 + case existingChats = 1 + case contacts = 2 + case nonContacts = 3 + } + + struct Peer { + var peer: EnginePeer + var isContact: Bool + + init(peer: EnginePeer, isContact: Bool) { + self.peer = peer + self.isContact = isContact + } + } + + var categories: Set + var peers: [Peer] + + init(categories: Set, peers: [Peer]) { + self.categories = categories + self.peers = peers + } + } + + private enum Schedule { + case always + case outsideBusinessHours + case custom + } + + final class View: UIView, UIScrollViewDelegate { + private let topOverscrollLayer = SimpleLayer() + private let scrollView: ScrollView + + private let navigationTitle = ComponentView() + private let icon = ComponentView() + private let subtitle = ComponentView() + private let generalSection = ComponentView() + private let messagesSection = ComponentView() + private let scheduleSection = ComponentView() + private let customScheduleSection = ComponentView() + private let sendWhenOfflineSection = ComponentView() + private let accessSection = ComponentView() + private let excludedSection = ComponentView() + private let periodSection = ComponentView() + + private var ignoreScrolling: Bool = false + private var isUpdating: Bool = false + + private var component: AutomaticBusinessMessageSetupScreenComponent? + private(set) weak var state: EmptyComponentState? + private var environment: EnvironmentType? + + private var isOn: Bool = false + private var accountPeer: EnginePeer? + private var currentShortcut: ShortcutMessageList.Item? + private var currentShortcutDisposable: Disposable? + + private var schedule: Schedule = .always + private var customScheduleStart: Date? + private var customScheduleEnd: Date? + + private var sendWhenOffline: Bool = true + + private var hasAccessToAllChatsByDefault: Bool = true + private var additionalPeerList = AdditionalPeerList( + categories: Set(), + peers: [] + ) + + private var replyToMessages: Bool = true + + private var inactivityDays: Int = 7 + + override init(frame: CGRect) { + self.scrollView = ScrollView() + self.scrollView.showsVerticalScrollIndicator = true + self.scrollView.showsHorizontalScrollIndicator = false + self.scrollView.scrollsToTop = false + self.scrollView.delaysContentTouches = false + self.scrollView.canCancelContentTouches = true + self.scrollView.contentInsetAdjustmentBehavior = .never + if #available(iOS 13.0, *) { + self.scrollView.automaticallyAdjustsScrollIndicatorInsets = false + } + self.scrollView.alwaysBounceVertical = true + + super.init(frame: frame) + + self.scrollView.delegate = self + self.addSubview(self.scrollView) + + self.scrollView.layer.addSublayer(self.topOverscrollLayer) + } + + required init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + deinit { + self.currentShortcutDisposable?.dispose() + } + + func scrollToTop() { + self.scrollView.setContentOffset(CGPoint(), animated: true) + } + + func attemptNavigation(complete: @escaping () -> Void) -> Bool { + guard let component = self.component else { + return true + } + + let presentationData = component.context.sharedContext.currentPresentationData.with { $0 } + + if self.isOn { + if !self.hasAccessToAllChatsByDefault && self.additionalPeerList.categories.isEmpty && self.additionalPeerList.peers.isEmpty { + self.environment?.controller()?.present(standardTextAlertController(theme: AlertControllerTheme(presentationData: presentationData), title: nil, text: presentationData.strings.BusinessMessageSetup_ErrorNoRecipients_Text, actions: [ + TextAlertAction(type: .genericAction, title: presentationData.strings.Common_Cancel, action: { + }), + TextAlertAction(type: .defaultAction, title: presentationData.strings.BusinessMessageSetup_ErrorNoRecipients_ResetAction, action: { + complete() + }) + ]), in: .window(.root)) + + return false + } + + if case .away = component.mode, case .custom = self.schedule { + var errorText: String? + if let customScheduleStart = self.customScheduleStart, let customScheduleEnd = self.customScheduleEnd { + if customScheduleStart >= customScheduleEnd { + errorText = presentationData.strings.BusinessMessageSetup_ErrorScheduleEndTimeBeforeStartTime_Text + } + } else { + if self.customScheduleStart == nil && self.customScheduleEnd == nil { + errorText = presentationData.strings.BusinessMessageSetup_ErrorScheduleTimeMissing_Text + } else if self.customScheduleStart == nil { + errorText = presentationData.strings.BusinessMessageSetup_ErrorScheduleStartTimeMissing_Text + } else { + errorText = presentationData.strings.BusinessMessageSetup_ErrorScheduleEndTimeMissing_Text + } + } + + if let errorText { + self.environment?.controller()?.present(standardTextAlertController(theme: AlertControllerTheme(presentationData: presentationData), title: nil, text: errorText, actions: [ + TextAlertAction(type: .genericAction, title: presentationData.strings.Common_Cancel, action: { + }), + TextAlertAction(type: .defaultAction, title: presentationData.strings.BusinessMessageSetup_ErrorScheduleTime_ResetAction, action: { + complete() + }) + ]), in: .window(.root)) + return false + } + } + } + + var mappedCategories: TelegramBusinessRecipients.Categories = [] + if self.additionalPeerList.categories.contains(.existingChats) { + mappedCategories.insert(.existingChats) + } + if self.additionalPeerList.categories.contains(.newChats) { + mappedCategories.insert(.newChats) + } + if self.additionalPeerList.categories.contains(.contacts) { + mappedCategories.insert(.contacts) + } + if self.additionalPeerList.categories.contains(.nonContacts) { + mappedCategories.insert(.nonContacts) + } + let recipients = TelegramBusinessRecipients( + categories: mappedCategories, + additionalPeers: Set(self.additionalPeerList.peers.map(\.peer.id)), + exclude: self.hasAccessToAllChatsByDefault + ) + + switch component.mode { + case .greeting: + var greetingMessage: TelegramBusinessGreetingMessage? + if self.isOn, let currentShortcut = self.currentShortcut, let shortcutId = currentShortcut.id { + greetingMessage = TelegramBusinessGreetingMessage( + shortcutId: shortcutId, + recipients: recipients, + inactivityDays: self.inactivityDays + ) + } + let _ = component.context.engine.accountData.updateBusinessGreetingMessage(greetingMessage: greetingMessage).startStandalone() + case .away: + var awayMessage: TelegramBusinessAwayMessage? + if self.isOn, let currentShortcut = self.currentShortcut, let shortcutId = currentShortcut.id { + let mappedSchedule: TelegramBusinessAwayMessage.Schedule + switch self.schedule { + case .always: + mappedSchedule = .always + case .outsideBusinessHours: + mappedSchedule = .outsideWorkingHours + case .custom: + if let customScheduleStart = self.customScheduleStart, let customScheduleEnd = self.customScheduleEnd { + mappedSchedule = .custom(beginTimestamp: Int32(customScheduleStart.timeIntervalSince1970), endTimestamp: Int32(customScheduleEnd.timeIntervalSince1970)) + } else { + return false + } + } + awayMessage = TelegramBusinessAwayMessage( + shortcutId: shortcutId, + recipients: recipients, + schedule: mappedSchedule, + sendWhenOffline: self.sendWhenOffline + ) + } + let _ = component.context.engine.accountData.updateBusinessAwayMessage(awayMessage: awayMessage).startStandalone() + } + + return true + } + + func scrollViewDidScroll(_ scrollView: UIScrollView) { + if !self.ignoreScrolling { + self.updateScrolling(transition: .immediate) + } + } + + var scrolledUp = true + private func updateScrolling(transition: Transition) { + let navigationRevealOffsetY: CGFloat = 0.0 + + let navigationAlphaDistance: CGFloat = 16.0 + let navigationAlpha: CGFloat = max(0.0, min(1.0, (self.scrollView.contentOffset.y - navigationRevealOffsetY) / navigationAlphaDistance)) + if let controller = self.environment?.controller(), let navigationBar = controller.navigationBar { + transition.setAlpha(layer: navigationBar.backgroundNode.layer, alpha: navigationAlpha) + transition.setAlpha(layer: navigationBar.stripeNode.layer, alpha: navigationAlpha) + } + + var scrolledUp = false + if navigationAlpha < 0.5 { + scrolledUp = true + } else if navigationAlpha > 0.5 { + scrolledUp = false + } + + if self.scrolledUp != scrolledUp { + self.scrolledUp = scrolledUp + if !self.isUpdating { + self.state?.updated() + } + } + + if let navigationTitleView = self.navigationTitle.view { + transition.setAlpha(view: navigationTitleView, alpha: 1.0) + } + } + + private func openAdditionalPeerListSetup() { + guard let component = self.component, let enviroment = self.environment else { + return + } + + enum AdditionalCategoryId: Int { + case existingChats + case newChats + case contacts + case nonContacts + } + + let additionalCategories: [ChatListNodeAdditionalCategory] = [ + ChatListNodeAdditionalCategory( + id: self.hasAccessToAllChatsByDefault ? AdditionalCategoryId.existingChats.rawValue : AdditionalCategoryId.newChats.rawValue, + icon: generateAvatarImage(size: CGSize(width: 40.0, height: 40.0), icon: generateTintedImage(image: UIImage(bundleImageName: self.hasAccessToAllChatsByDefault ? "Chat List/Filters/Chats" : "Chat List/Filters/NewChats"), color: .white), cornerRadius: 12.0, color: .purple), + smallIcon: generateAvatarImage(size: CGSize(width: 22.0, height: 22.0), icon: generateTintedImage(image: UIImage(bundleImageName: self.hasAccessToAllChatsByDefault ? "Chat List/Filters/Chats" : "Chat List/Filters/NewChats"), color: .white), iconScale: 0.6, cornerRadius: 6.0, circleCorners: true, color: .purple), + title: self.hasAccessToAllChatsByDefault ? enviroment.strings.BusinessMessageSetup_Recipients_CategoryExistingChats : enviroment.strings.BusinessMessageSetup_Recipients_CategoryNewChats + ), + ChatListNodeAdditionalCategory( + id: AdditionalCategoryId.contacts.rawValue, + icon: generateAvatarImage(size: CGSize(width: 40.0, height: 40.0), icon: generateTintedImage(image: UIImage(bundleImageName: "Chat List/Filters/Contact"), color: .white), cornerRadius: 12.0, color: .blue), + smallIcon: generateAvatarImage(size: CGSize(width: 22.0, height: 22.0), icon: generateTintedImage(image: UIImage(bundleImageName: "Chat List/Filters/Contact"), color: .white), iconScale: 0.6, cornerRadius: 6.0, circleCorners: true, color: .blue), + title: enviroment.strings.BusinessMessageSetup_Recipients_CategoryContacts + ), + ChatListNodeAdditionalCategory( + id: AdditionalCategoryId.nonContacts.rawValue, + icon: generateAvatarImage(size: CGSize(width: 40.0, height: 40.0), icon: generateTintedImage(image: UIImage(bundleImageName: "Chat List/Filters/User"), color: .white), cornerRadius: 12.0, color: .yellow), + smallIcon: generateAvatarImage(size: CGSize(width: 22.0, height: 22.0), icon: generateTintedImage(image: UIImage(bundleImageName: "Chat List/Filters/User"), color: .white), iconScale: 0.6, cornerRadius: 6.0, circleCorners: true, color: .yellow), + title: enviroment.strings.BusinessMessageSetup_Recipients_CategoryNonContacts + ) + ] + var selectedCategories = Set() + for category in self.additionalPeerList.categories { + switch category { + case .existingChats: + selectedCategories.insert(AdditionalCategoryId.existingChats.rawValue) + case .newChats: + selectedCategories.insert(AdditionalCategoryId.newChats.rawValue) + case .contacts: + selectedCategories.insert(AdditionalCategoryId.contacts.rawValue) + case .nonContacts: + selectedCategories.insert(AdditionalCategoryId.nonContacts.rawValue) + } + } + + let controller = component.context.sharedContext.makeContactMultiselectionController(ContactMultiselectionControllerParams(context: component.context, mode: .chatSelection(ContactMultiselectionControllerMode.ChatSelection( + title: self.hasAccessToAllChatsByDefault ? enviroment.strings.BusinessMessageSetup_Recipients_ExcludeSearchTitle : enviroment.strings.BusinessMessageSetup_Recipients_IncludeSearchTitle, + searchPlaceholder: enviroment.strings.ChatListFilter_AddChatsSearchPlaceholder, + selectedChats: Set(self.additionalPeerList.peers.map(\.peer.id)), + additionalCategories: ContactMultiselectionControllerAdditionalCategories(categories: additionalCategories, selectedCategories: selectedCategories), + chatListFilters: nil, + onlyUsers: true + )), options: [], filters: [], alwaysEnabled: true, limit: 100, reachedLimit: { _ in + })) + controller.navigationPresentation = .modal + + let _ = (controller.result + |> take(1) + |> deliverOnMainQueue).startStandalone(next: { [weak self, weak controller] result in + guard let self, let component = self.component, case let .result(rawPeerIds, additionalCategoryIds) = result else { + controller?.dismiss() + return + } + + let peerIds = rawPeerIds.compactMap { id -> EnginePeer.Id? in + switch id { + case let .peer(id): + return id + case .deviceContact: + return nil + } + } + + let _ = (component.context.engine.data.get( + EngineDataMap( + peerIds.map(TelegramEngine.EngineData.Item.Peer.Peer.init(id:)) + ), + EngineDataMap( + peerIds.map(TelegramEngine.EngineData.Item.Peer.IsContact.init(id:)) + ) + ) + |> deliverOnMainQueue).start(next: { [weak self] peerMap, isContactMap in + guard let self else { + return + } + + let mappedCategories = additionalCategoryIds.compactMap { item -> AdditionalPeerList.Category? in + switch item { + case AdditionalCategoryId.existingChats.rawValue: + return .existingChats + case AdditionalCategoryId.newChats.rawValue: + return .newChats + case AdditionalCategoryId.contacts.rawValue: + return .contacts + case AdditionalCategoryId.nonContacts.rawValue: + return .nonContacts + default: + return nil + } + } + + self.additionalPeerList.categories = Set(mappedCategories) + + self.additionalPeerList.peers.removeAll() + for id in peerIds { + guard let maybePeer = peerMap[id], let peer = maybePeer else { + continue + } + self.additionalPeerList.peers.append(AdditionalPeerList.Peer( + peer: peer, + isContact: isContactMap[id] ?? false + )) + } + self.additionalPeerList.peers.sort(by: { lhs, rhs in + return lhs.peer.debugDisplayTitle < rhs.peer.debugDisplayTitle + }) + self.state?.updated(transition: .immediate) + + controller?.dismiss() + }) + }) + + self.environment?.controller()?.push(controller) + } + + private func openMessageList() { + guard let component = self.component else { + return + } + + let shortcutName: String + let shortcutType: ChatQuickReplyShortcutType + switch component.mode { + case .greeting: + shortcutName = "hello" + shortcutType = .greeting + case .away: + shortcutName = "away" + shortcutType = .away + } + + let contents = AutomaticBusinessMessageSetupChatContents( + context: component.context, + kind: .quickReplyMessageInput(shortcut: shortcutName, shortcutType: shortcutType), + shortcutId: self.currentShortcut?.id + ) + let chatController = component.context.sharedContext.makeChatController( + context: component.context, + chatLocation: .customChatContents, + subject: .customChatContents(contents: contents), + botStart: nil, + mode: .standard(.default) + ) + chatController.navigationPresentation = .modal + self.environment?.controller()?.push(chatController) + } + + private func openCustomScheduleDateSetup(isStartTime: Bool, isDate: Bool) { + guard let component = self.component else { + return + } + + let currentValue: Date = (isStartTime ? self.customScheduleStart : self.customScheduleEnd) ?? Date() + + if isDate { + var calendar = Calendar(identifier: .gregorian) + calendar.timeZone = TimeZone(secondsFromGMT: 0)! + let components = calendar.dateComponents([.year, .month, .day], from: currentValue) + guard let clippedDate = calendar.date(from: components) else { + return + } + + let controller = DateSelectionActionSheetController( + context: component.context, + title: nil, + currentValue: Int32(clippedDate.timeIntervalSince1970), + minimumDate: nil, + maximumDate: nil, + emptyTitle: nil, + applyValue: { [weak self] value in + guard let self else { + return + } + guard let value else { + return + } + let updatedDate = Date(timeIntervalSince1970: Double(value)) + let calendar = Calendar.current + var updatedComponents = calendar.dateComponents([.year, .month, .day], from: updatedDate) + let currentComponents = calendar.dateComponents([.hour, .minute], from: currentValue) + updatedComponents.hour = currentComponents.hour + updatedComponents.minute = currentComponents.minute + guard let updatedClippedDate = calendar.date(from: updatedComponents) else { + return + } + + if isStartTime { + self.customScheduleStart = updatedClippedDate + } else { + self.customScheduleEnd = updatedClippedDate + } + self.state?.updated(transition: .immediate) + } + ) + self.environment?.controller()?.present(controller, in: .window(.root)) + } else { + let calendar = Calendar.current + let components = calendar.dateComponents([.hour, .minute], from: currentValue) + let hour = components.hour ?? 0 + let minute = components.minute ?? 0 + + let controller = TimeSelectionActionSheet(context: component.context, currentValue: Int32(hour * 60 * 60 + minute * 60), applyValue: { [weak self] value in + guard let self else { + return + } + guard let value else { + return + } + + let updatedHour = value / (60 * 60) + let updatedMinute = (value % (60 * 60)) / 60 + + let calendar = Calendar.current + var updatedComponents = calendar.dateComponents([.year, .month, .day], from: currentValue) + updatedComponents.hour = Int(updatedHour) + updatedComponents.minute = Int(updatedMinute) + + guard let updatedClippedDate = calendar.date(from: updatedComponents) else { + return + } + + if isStartTime { + self.customScheduleStart = updatedClippedDate + } else { + self.customScheduleEnd = updatedClippedDate + } + self.state?.updated(transition: .immediate) + }) + self.environment?.controller()?.present(controller, in: .window(.root)) + } + } + + func update(component: AutomaticBusinessMessageSetupScreenComponent, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + self.isUpdating = true + defer { + self.isUpdating = false + } + + if self.component == nil { + self.accountPeer = component.initialData.accountPeer + + var initialRecipients: TelegramBusinessRecipients? + + let shortcutName: String + switch component.mode { + case .greeting: + shortcutName = "hello" + + if let greetingMessage = component.initialData.greetingMessage { + self.isOn = true + initialRecipients = greetingMessage.recipients + + self.inactivityDays = greetingMessage.inactivityDays + } + case .away: + shortcutName = "away" + + if let awayMessage = component.initialData.awayMessage { + self.isOn = true + + self.sendWhenOffline = awayMessage.sendWhenOffline + + initialRecipients = awayMessage.recipients + + switch awayMessage.schedule { + case .always: + self.schedule = .always + case let .custom(beginTimestamp, endTimestamp): + self.schedule = .custom + self.customScheduleStart = Date(timeIntervalSince1970: Double(beginTimestamp)) + self.customScheduleEnd = Date(timeIntervalSince1970: Double(endTimestamp)) + case .outsideWorkingHours: + if component.initialData.businessHours != nil { + self.schedule = .outsideBusinessHours + } else { + self.schedule = .always + } + } + } + } + + if let initialRecipients { + var mappedCategories = Set() + if initialRecipients.categories.contains(.existingChats) { + mappedCategories.insert(.existingChats) + } + if initialRecipients.categories.contains(.newChats) { + mappedCategories.insert(.newChats) + } + if initialRecipients.categories.contains(.contacts) { + mappedCategories.insert(.contacts) + } + if initialRecipients.categories.contains(.nonContacts) { + mappedCategories.insert(.nonContacts) + } + + var additionalPeers: [AdditionalPeerList.Peer] = [] + for peerId in initialRecipients.additionalPeers { + if let peer = component.initialData.additionalPeers[peerId] { + additionalPeers.append(peer) + } + } + + self.additionalPeerList = AdditionalPeerList( + categories: mappedCategories, + peers: additionalPeers + ) + + self.hasAccessToAllChatsByDefault = initialRecipients.exclude + } + + self.currentShortcut = component.initialData.shortcutMessageList.items.first(where: { $0.shortcut == shortcutName }) + + self.currentShortcutDisposable = (component.context.engine.accountData.shortcutMessageList(onlyRemote: false) + |> deliverOnMainQueue).start(next: { [weak self] shortcutMessageList in + guard let self else { + return + } + let shortcut = shortcutMessageList.items.first(where: { $0.shortcut == shortcutName }) + if shortcut != self.currentShortcut { + self.currentShortcut = shortcut + self.state?.updated(transition: .immediate) + } + }) + } + + let environment = environment[EnvironmentType.self].value + let themeUpdated = self.environment?.theme !== environment.theme + self.environment = environment + + self.component = component + self.state = state + + let alphaTransition: Transition = transition.animation.isImmediate ? transition : transition.withAnimation(.curve(duration: 0.25, curve: .easeInOut)) + + if themeUpdated { + self.backgroundColor = environment.theme.list.blocksBackgroundColor + } + + let presentationData = component.context.sharedContext.currentPresentationData.with { $0 } + + let navigationTitleSize = self.navigationTitle.update( + transition: transition, + component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: component.mode == .greeting ? environment.strings.BusinessMessageSetup_TitleGreetingMessage : environment.strings.BusinessMessageSetup_TitleAwayMessage, font: Font.semibold(17.0), textColor: environment.theme.rootController.navigationBar.primaryTextColor)), + horizontalAlignment: .center + )), + environment: {}, + containerSize: CGSize(width: availableSize.width, height: 100.0) + ) + let navigationTitleFrame = CGRect(origin: CGPoint(x: floor((availableSize.width - navigationTitleSize.width) / 2.0), y: environment.statusBarHeight + floor((environment.navigationHeight - environment.statusBarHeight - navigationTitleSize.height) / 2.0)), size: navigationTitleSize) + if let navigationTitleView = self.navigationTitle.view { + if navigationTitleView.superview == nil { + if let controller = self.environment?.controller(), let navigationBar = controller.navigationBar { + navigationBar.view.addSubview(navigationTitleView) + } + } + transition.setFrame(view: navigationTitleView, frame: navigationTitleFrame) + } + + let bottomContentInset: CGFloat = 24.0 + let sideInset: CGFloat = 16.0 + environment.safeInsets.left + let sectionSpacing: CGFloat = 32.0 + + var contentHeight: CGFloat = 0.0 + + contentHeight += environment.navigationHeight + + let iconSize = self.icon.update( + transition: .immediate, + component: AnyComponent(LottieComponent( + content: LottieComponent.AppBundleContent(name: component.mode == .greeting ? "HandWaveEmoji" : "ZzzEmoji"), + loop: false + )), + environment: {}, + containerSize: CGSize(width: 100.0, height: 100.0) + ) + let iconFrame = CGRect(origin: CGPoint(x: floor((availableSize.width - iconSize.width) * 0.5), y: contentHeight + 8.0), size: iconSize) + if let iconView = self.icon.view as? LottieComponent.View { + if iconView.superview == nil { + self.scrollView.addSubview(iconView) + iconView.playOnce() + } + transition.setPosition(view: iconView, position: iconFrame.center) + iconView.bounds = CGRect(origin: CGPoint(), size: iconFrame.size) + } + + contentHeight += 124.0 + + let subtitleString = NSMutableAttributedString(attributedString: parseMarkdownIntoAttributedString(component.mode == .greeting ? environment.strings.BusinessMessageSetup_TextGreetingMessage : environment.strings.BusinessMessageSetup_TextAwayMessage, attributes: MarkdownAttributes( + body: MarkdownAttributeSet(font: Font.regular(15.0), textColor: environment.theme.list.freeTextColor), + bold: MarkdownAttributeSet(font: Font.semibold(15.0), textColor: environment.theme.list.freeTextColor), + link: MarkdownAttributeSet(font: Font.regular(15.0), textColor: environment.theme.list.itemAccentColor), + linkAttribute: { attributes in + return ("URL", "") + }), textAlignment: .center + )) + + let subtitleSize = self.subtitle.update( + transition: .immediate, + component: AnyComponent(BalancedTextComponent( + text: .plain(subtitleString), + horizontalAlignment: .center, + maximumNumberOfLines: 0, + lineSpacing: 0.25, + highlightColor: environment.theme.list.itemAccentColor.withMultipliedAlpha(0.1), + highlightAction: { attributes in + if let _ = attributes[NSAttributedString.Key(rawValue: "URL")] { + return NSAttributedString.Key(rawValue: "URL") + } else { + return nil + } + }, + tapAction: { [weak self] _, _ in + guard let self, let component = self.component else { + return + } + let _ = component + } + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 1000.0) + ) + let subtitleFrame = CGRect(origin: CGPoint(x: floor((availableSize.width - subtitleSize.width) * 0.5), y: contentHeight), size: subtitleSize) + if let subtitleView = self.subtitle.view { + if subtitleView.superview == nil { + self.scrollView.addSubview(subtitleView) + } + transition.setPosition(view: subtitleView, position: subtitleFrame.center) + subtitleView.bounds = CGRect(origin: CGPoint(), size: subtitleFrame.size) + } + contentHeight += subtitleSize.height + contentHeight += 27.0 + + var generalSectionItems: [AnyComponentWithIdentity] = [] + generalSectionItems.append(AnyComponentWithIdentity(id: 0, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: component.mode == .greeting ? environment.strings.BusinessMessageSetup_ToggleGreetingMessage : environment.strings.BusinessMessageSetup_ToggleAwayMessage, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemPrimaryTextColor + )), + maximumNumberOfLines: 1 + ))), + ], alignment: .left, spacing: 2.0)), + accessory: .toggle(ListActionItemComponent.Toggle(style: .regular, isOn: self.isOn, action: { [weak self] _ in + guard let self else { + return + } + self.isOn = !self.isOn + self.state?.updated(transition: .spring(duration: 0.4)) + })), + action: nil + )))) + + let generalSectionSize = self.generalSection.update( + transition: transition, + component: AnyComponent(ListSectionComponent( + theme: environment.theme, + header: nil, + footer: nil, + items: generalSectionItems + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 10000.0) + ) + let generalSectionFrame = CGRect(origin: CGPoint(x: sideInset, y: contentHeight), size: generalSectionSize) + if let generalSectionView = self.generalSection.view { + if generalSectionView.superview == nil { + self.scrollView.addSubview(generalSectionView) + } + transition.setFrame(view: generalSectionView, frame: generalSectionFrame) + } + contentHeight += generalSectionSize.height + contentHeight += sectionSpacing + + var otherSectionsHeight: CGFloat = 0.0 + + var messagesSectionItems: [AnyComponentWithIdentity] = [] + if let currentShortcut = self.currentShortcut { + if let accountPeer = self.accountPeer { + messagesSectionItems.append(AnyComponentWithIdentity(id: 1, component: AnyComponent(GreetingMessageListItemComponent( + context: component.context, + theme: environment.theme, + strings: environment.strings, + accountPeer: accountPeer, + message: currentShortcut.topMessage, + count: currentShortcut.totalCount, + action: { [weak self] in + guard let self else { + return + } + self.openMessageList() + } + )))) + } + } else { + messagesSectionItems.append(AnyComponentWithIdentity(id: 0, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: component.mode == .greeting ? environment.strings.BusinessMessageSetup_CreateGreetingMessage : environment.strings.BusinessMessageSetup_CreateAwayMessage, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemAccentColor + )), + maximumNumberOfLines: 1 + ))), + ], alignment: .left, spacing: 2.0)), + leftIcon: AnyComponentWithIdentity(id: 0, component: AnyComponent(BundleIconComponent( + name: "Chat List/ComposeIcon", + tintColor: environment.theme.list.itemAccentColor + ))), + accessory: nil, + action: { [weak self] _ in + guard let self else { + return + } + self.openMessageList() + } + )))) + } + let messagesSectionSize = self.messagesSection.update( + transition: transition, + component: AnyComponent(ListSectionComponent( + theme: environment.theme, + header: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: component.mode == .greeting ? environment.strings.BusinessMessageSetup_GreetingMessageSectionHeader : environment.strings.BusinessMessageSetup_AwayMessageSectionHeader, + font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), + textColor: environment.theme.list.freeTextColor + )), + maximumNumberOfLines: 0 + )), + footer: nil, + items: messagesSectionItems + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 10000.0) + ) + let messagesSectionFrame = CGRect(origin: CGPoint(x: sideInset, y: contentHeight + otherSectionsHeight), size: messagesSectionSize) + if let messagesSectionView = self.messagesSection.view { + if messagesSectionView.superview == nil { + messagesSectionView.layer.allowsGroupOpacity = true + self.scrollView.addSubview(messagesSectionView) + } + transition.setFrame(view: messagesSectionView, frame: messagesSectionFrame) + alphaTransition.setAlpha(view: messagesSectionView, alpha: self.isOn ? 1.0 : 0.0) + } + otherSectionsHeight += messagesSectionSize.height + otherSectionsHeight += sectionSpacing + + if case .away = component.mode { + var scheduleSectionItems: [AnyComponentWithIdentity] = [] + optionLoop: for i in 0 ..< 3 { + let title: String + let schedule: Schedule + switch i { + case 0: + title = environment.strings.BusinessMessageSetup_ScheduleAlways + schedule = .always + case 1: + if component.initialData.businessHours == nil { + continue optionLoop + } + + title = environment.strings.BusinessMessageSetup_ScheduleOutsideBusinessHours + schedule = .outsideBusinessHours + default: + title = environment.strings.BusinessMessageSetup_ScheduleCustom + schedule = .custom + } + let isSelected = self.schedule == schedule + scheduleSectionItems.append(AnyComponentWithIdentity(id: scheduleSectionItems.count, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: title, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemPrimaryTextColor + )), + maximumNumberOfLines: 1 + ))), + ], alignment: .left, spacing: 2.0)), + leftIcon: AnyComponentWithIdentity(id: 0, component: AnyComponent(Image( + image: checkIcon, + tintColor: !isSelected ? .clear : environment.theme.list.itemAccentColor, + contentMode: .center + ))), + accessory: nil, + action: { [weak self] _ in + guard let self else { + return + } + + if self.schedule != schedule { + self.schedule = schedule + self.state?.updated(transition: .immediate) + } + } + )))) + } + let scheduleSectionSize = self.scheduleSection.update( + transition: transition, + component: AnyComponent(ListSectionComponent( + theme: environment.theme, + header: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: environment.strings.BusinessMessageSetup_ScheduleSectionHeader, + font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), + textColor: environment.theme.list.freeTextColor + )), + maximumNumberOfLines: 0 + )), + footer: nil, + items: scheduleSectionItems + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 10000.0) + ) + let scheduleSectionFrame = CGRect(origin: CGPoint(x: sideInset, y: contentHeight + otherSectionsHeight), size: scheduleSectionSize) + if let scheduleSectionView = self.scheduleSection.view { + if scheduleSectionView.superview == nil { + scheduleSectionView.layer.allowsGroupOpacity = true + self.scrollView.addSubview(scheduleSectionView) + } + transition.setFrame(view: scheduleSectionView, frame: scheduleSectionFrame) + alphaTransition.setAlpha(view: scheduleSectionView, alpha: self.isOn ? 1.0 : 0.0) + } + otherSectionsHeight += scheduleSectionSize.height + otherSectionsHeight += sectionSpacing + + var customScheduleSectionsHeight: CGFloat = 0.0 + var customScheduleSectionItems: [AnyComponentWithIdentity] = [] + for i in 0 ..< 2 { + let title: String + let itemDate: Date? + let isStartTime: Bool + switch i { + case 0: + title = environment.strings.BusinessMessageSetup_ScheduleStartTime + itemDate = self.customScheduleStart + isStartTime = true + default: + title = environment.strings.BusinessMessageSetup_ScheduleEndTime + itemDate = self.customScheduleEnd + isStartTime = false + } + + var icon: ListActionItemComponent.Icon? + var accessory: ListActionItemComponent.Accessory? + if let itemDate { + let calendar = Calendar.current + let hours = calendar.component(.hour, from: itemDate) + let minutes = calendar.component(.minute, from: itemDate) + + let presentationData = component.context.sharedContext.currentPresentationData.with({ $0 }) + + let timeText = stringForShortTimestamp(hours: Int32(hours), minutes: Int32(minutes), dateTimeFormat: presentationData.dateTimeFormat) + + let dateFormatter = DateFormatter() + dateFormatter.timeStyle = .none + dateFormatter.dateStyle = .medium + let dateText = stringForCompactDate(timestamp: Int32(itemDate.timeIntervalSince1970), strings: environment.strings, dateTimeFormat: presentationData.dateTimeFormat) + + icon = ListActionItemComponent.Icon(component: AnyComponentWithIdentity(id: 0, component: AnyComponent(HStack([ + AnyComponentWithIdentity(id: 0, component: AnyComponent(PlainButtonComponent( + content: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: dateText, font: Font.regular(17.0), textColor: environment.theme.list.itemPrimaryTextColor)) + )), + background: AnyComponent(RoundedRectangle(color: environment.theme.list.itemPrimaryTextColor.withMultipliedAlpha(0.1), cornerRadius: 6.0)), + effectAlignment: .center, + minSize: nil, + contentInsets: UIEdgeInsets(top: 7.0, left: 8.0, bottom: 7.0, right: 8.0), + action: { [weak self] in + guard let self else { + return + } + self.openCustomScheduleDateSetup(isStartTime: isStartTime, isDate: true) + }, + animateAlpha: true, + animateScale: false + ))), + AnyComponentWithIdentity(id: 1, component: AnyComponent(PlainButtonComponent( + content: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: timeText, font: Font.regular(17.0), textColor: environment.theme.list.itemPrimaryTextColor)) + )), + background: AnyComponent(RoundedRectangle(color: environment.theme.list.itemPrimaryTextColor.withMultipliedAlpha(0.1), cornerRadius: 6.0)), + effectAlignment: .center, + minSize: nil, + contentInsets: UIEdgeInsets(top: 7.0, left: 8.0, bottom: 7.0, right: 8.0), + action: { [weak self] in + guard let self else { + return + } + self.openCustomScheduleDateSetup(isStartTime: isStartTime, isDate: false) + }, + animateAlpha: true, + animateScale: false + ))) + ], spacing: 4.0))), insets: .custom(UIEdgeInsets(top: 4.0, left: 0.0, bottom: 4.0, right: 0.0)), allowUserInteraction: true) + } else { + icon = ListActionItemComponent.Icon(component: AnyComponentWithIdentity(id: 1, component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: environment.strings.BusinessMessageSetup_ScheduleTimePlaceholder, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemSecondaryTextColor + )), + maximumNumberOfLines: 1 + )))) + accessory = .arrow + } + + customScheduleSectionItems.append(AnyComponentWithIdentity(id: customScheduleSectionItems.count, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: title, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemPrimaryTextColor + )), + maximumNumberOfLines: 1 + ))), + ], alignment: .left, spacing: 2.0)), + icon: icon, + accessory: accessory, + action: itemDate != nil ? nil : { [weak self] _ in + guard let self else { + return + } + self.openCustomScheduleDateSetup(isStartTime: isStartTime, isDate: true) + } + )))) + } + let customScheduleSectionSize = self.customScheduleSection.update( + transition: transition, + component: AnyComponent(ListSectionComponent( + theme: environment.theme, + header: nil, + footer: nil, + items: customScheduleSectionItems + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 10000.0) + ) + let customScheduleSectionFrame = CGRect(origin: CGPoint(x: sideInset, y: contentHeight + otherSectionsHeight + customScheduleSectionsHeight), size: customScheduleSectionSize) + if let customScheduleSectionView = self.customScheduleSection.view { + if customScheduleSectionView.superview == nil { + customScheduleSectionView.layer.allowsGroupOpacity = true + self.scrollView.addSubview(customScheduleSectionView) + } + transition.setFrame(view: customScheduleSectionView, frame: customScheduleSectionFrame) + alphaTransition.setAlpha(view: customScheduleSectionView, alpha: (self.isOn && self.schedule == .custom) ? 1.0 : 0.0) + } + customScheduleSectionsHeight += customScheduleSectionSize.height + customScheduleSectionsHeight += sectionSpacing + + if self.schedule == .custom { + otherSectionsHeight += customScheduleSectionsHeight + } + } + + if case .away = component.mode { + let sendWhenOfflineSectionSize = self.sendWhenOfflineSection.update( + transition: transition, + component: AnyComponent(ListSectionComponent( + theme: environment.theme, + header: nil, + footer: AnyComponent(MultilineTextComponent( + text: .markdown( + text: environment.strings.BusinessMessageSetup_SendWhenOfflineFooter, + attributes: MarkdownAttributes( + body: MarkdownAttributeSet(font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), textColor: environment.theme.list.freeTextColor), + bold: MarkdownAttributeSet(font: Font.semibold(presentationData.listsFontSize.itemListBaseHeaderFontSize), textColor: environment.theme.list.freeTextColor), + link: MarkdownAttributeSet(font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), textColor: environment.theme.list.itemAccentColor), + linkAttribute: { _ in + return nil + } + ) + ), + maximumNumberOfLines: 0 + )), + items: [ + AnyComponentWithIdentity(id: 0, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: environment.strings.BusinessMessageSetup_SendWhenOffline, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemPrimaryTextColor + )), + maximumNumberOfLines: 1 + ))), + ], alignment: .left, spacing: 2.0)), + leftIcon: nil, + accessory: .toggle(ListActionItemComponent.Toggle( + style: .regular, + isOn: self.sendWhenOffline, + action: { [weak self] value in + guard let self else { + return + } + self.sendWhenOffline = value + self.state?.updated(transition: .spring(duration: 0.4)) + } + )), + action: nil + ))) + ] + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 10000.0) + ) + let sendWhenOfflineSectionFrame = CGRect(origin: CGPoint(x: sideInset, y: contentHeight + otherSectionsHeight), size: sendWhenOfflineSectionSize) + if let sendWhenOfflineSectionView = self.sendWhenOfflineSection.view { + if sendWhenOfflineSectionView.superview == nil { + sendWhenOfflineSectionView.layer.allowsGroupOpacity = true + self.scrollView.addSubview(sendWhenOfflineSectionView) + } + transition.setFrame(view: sendWhenOfflineSectionView, frame: sendWhenOfflineSectionFrame) + alphaTransition.setAlpha(view: sendWhenOfflineSectionView, alpha: self.isOn ? 1.0 : 0.0) + } + otherSectionsHeight += sendWhenOfflineSectionSize.height + otherSectionsHeight += sectionSpacing + } + + let accessSectionSize = self.accessSection.update( + transition: transition, + component: AnyComponent(ListSectionComponent( + theme: environment.theme, + header: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: environment.strings.BusinessMessageSetup_RecipientsSectionHeader, + font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), + textColor: environment.theme.list.freeTextColor + )), + maximumNumberOfLines: 0 + )), + footer: nil, + items: [ + AnyComponentWithIdentity(id: 0, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: environment.strings.BusinessMessageSetup_RecipientsOptionAllExcept, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemPrimaryTextColor + )), + maximumNumberOfLines: 1 + ))), + ], alignment: .left, spacing: 2.0)), + leftIcon: AnyComponentWithIdentity(id: 0, component: AnyComponent(Image( + image: checkIcon, + tintColor: !self.hasAccessToAllChatsByDefault ? .clear : environment.theme.list.itemAccentColor, + contentMode: .center + ))), + accessory: nil, + action: { [weak self] _ in + guard let self else { + return + } + if !self.hasAccessToAllChatsByDefault { + self.hasAccessToAllChatsByDefault = true + self.additionalPeerList.categories.removeAll() + self.additionalPeerList.peers.removeAll() + self.state?.updated(transition: .immediate) + } + } + ))), + AnyComponentWithIdentity(id: 1, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: environment.strings.BusinessMessageSetup_RecipientsOptionOnly, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemPrimaryTextColor + )), + maximumNumberOfLines: 1 + ))), + ], alignment: .left, spacing: 2.0)), + leftIcon: AnyComponentWithIdentity(id: 0, component: AnyComponent(Image( + image: checkIcon, + tintColor: self.hasAccessToAllChatsByDefault ? .clear : environment.theme.list.itemAccentColor, + contentMode: .center + ))), + accessory: nil, + action: { [weak self] _ in + guard let self else { + return + } + if self.hasAccessToAllChatsByDefault { + self.hasAccessToAllChatsByDefault = false + self.additionalPeerList.categories.removeAll() + self.additionalPeerList.peers.removeAll() + self.state?.updated(transition: .immediate) + } + } + ))) + ] + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 10000.0) + ) + let accessSectionFrame = CGRect(origin: CGPoint(x: sideInset, y: contentHeight + otherSectionsHeight), size: accessSectionSize) + if let accessSectionView = self.accessSection.view { + if accessSectionView.superview == nil { + accessSectionView.layer.allowsGroupOpacity = true + self.scrollView.addSubview(accessSectionView) + } + transition.setFrame(view: accessSectionView, frame: accessSectionFrame) + alphaTransition.setAlpha(view: accessSectionView, alpha: self.isOn ? 1.0 : 0.0) + } + otherSectionsHeight += accessSectionSize.height + otherSectionsHeight += sectionSpacing + + var excludedSectionItems: [AnyComponentWithIdentity] = [] + excludedSectionItems.append(AnyComponentWithIdentity(id: 0, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: self.hasAccessToAllChatsByDefault ? environment.strings.BusinessMessageSetup_Recipients_AddExclude : environment.strings.BusinessMessageSetup_Recipients_AddInclude, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemAccentColor + )), + maximumNumberOfLines: 1 + ))), + ], alignment: .left, spacing: 2.0)), + leftIcon: AnyComponentWithIdentity(id: 0, component: AnyComponent(BundleIconComponent( + name: "Chat List/AddIcon", + tintColor: environment.theme.list.itemAccentColor + ))), + accessory: nil, + action: { [weak self] _ in + guard let self else { + return + } + self.openAdditionalPeerListSetup() + } + )))) + for category in self.additionalPeerList.categories.sorted(by: { $0.rawValue < $1.rawValue }) { + let title: String + let icon: String + let color: AvatarBackgroundColor + switch category { + case .newChats: + title = environment.strings.BusinessMessageSetup_Recipients_CategoryNewChats + icon = "Chat List/Filters/NewChats" + color = .purple + case .existingChats: + title = environment.strings.BusinessMessageSetup_Recipients_CategoryExistingChats + icon = "Chat List/Filters/Chats" + color = .purple + case .contacts: + title = environment.strings.BusinessMessageSetup_Recipients_CategoryContacts + icon = "Chat List/Filters/Contact" + color = .blue + case .nonContacts: + title = environment.strings.BusinessMessageSetup_Recipients_CategoryNonContacts + icon = "Chat List/Filters/User" + color = .yellow + } + excludedSectionItems.append(AnyComponentWithIdentity(id: category, component: AnyComponent(PeerListItemComponent( + context: component.context, + theme: environment.theme, + strings: environment.strings, + style: .generic, + sideInset: 0.0, + title: title, + avatar: PeerListItemComponent.Avatar( + icon: icon, + color: color, + clipStyle: .roundedRect + ), + peer: nil, + subtitle: nil, + subtitleAccessory: .none, + presence: nil, + selectionState: .none, + hasNext: false, + action: { peer, _, _ in + }, + inlineActions: PeerListItemComponent.InlineActionsState( + actions: [PeerListItemComponent.InlineAction( + id: AnyHashable(0), + title: environment.strings.Common_Delete, + color: .destructive, + action: { [weak self] in + guard let self else { + return + } + self.additionalPeerList.categories.remove(category) + self.state?.updated(transition: .spring(duration: 0.4)) + } + )] + ) + )))) + } + for peer in self.additionalPeerList.peers { + excludedSectionItems.append(AnyComponentWithIdentity(id: peer.peer.id, component: AnyComponent(PeerListItemComponent( + context: component.context, + theme: environment.theme, + strings: environment.strings, + style: .generic, + sideInset: 0.0, + title: peer.peer.displayTitle(strings: environment.strings, displayOrder: .firstLast), + peer: peer.peer, + subtitle: peer.isContact ? environment.strings.ChatList_PeerTypeContact : environment.strings.ChatList_PeerTypeNonContact, + subtitleAccessory: .none, + presence: nil, + selectionState: .none, + hasNext: false, + action: { peer, _, _ in + }, + inlineActions: PeerListItemComponent.InlineActionsState( + actions: [PeerListItemComponent.InlineAction( + id: AnyHashable(0), + title: environment.strings.Common_Delete, + color: .destructive, + action: { [weak self] in + guard let self else { + return + } + self.additionalPeerList.peers.removeAll(where: { $0.peer.id == peer.peer.id }) + self.state?.updated(transition: .spring(duration: 0.4)) + } + )] + ) + )))) + } + + let excludedSectionSize = self.excludedSection.update( + transition: transition, + component: AnyComponent(ListSectionComponent( + theme: environment.theme, + header: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: self.hasAccessToAllChatsByDefault ? environment.strings.BusinessMessageSetup_Recipients_ExcludedSectionHeader : environment.strings.BusinessMessageSetup_Recipients_IncludedSectionHeader, + font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), + textColor: environment.theme.list.freeTextColor + )), + maximumNumberOfLines: 0 + )), + footer: AnyComponent(MultilineTextComponent( + text: .markdown( + text: component.mode == .greeting ? environment.strings.BusinessMessageSetup_Recipients_GreetingMessageFooter : environment.strings.BusinessMessageSetup_Recipients_AwayMessageFooter, + attributes: MarkdownAttributes( + body: MarkdownAttributeSet(font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), textColor: environment.theme.list.freeTextColor), + bold: MarkdownAttributeSet(font: Font.semibold(presentationData.listsFontSize.itemListBaseHeaderFontSize), textColor: environment.theme.list.freeTextColor), + link: MarkdownAttributeSet(font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), textColor: environment.theme.list.itemAccentColor), + linkAttribute: { _ in + return nil + } + ) + ), + maximumNumberOfLines: 0 + )), + items: excludedSectionItems + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 10000.0) + ) + let excludedSectionFrame = CGRect(origin: CGPoint(x: sideInset, y: contentHeight + otherSectionsHeight), size: excludedSectionSize) + if let excludedSectionView = self.excludedSection.view { + if excludedSectionView.superview == nil { + excludedSectionView.layer.allowsGroupOpacity = true + self.scrollView.addSubview(excludedSectionView) + } + transition.setFrame(view: excludedSectionView, frame: excludedSectionFrame) + alphaTransition.setAlpha(view: excludedSectionView, alpha: self.isOn ? 1.0 : 0.0) + } + otherSectionsHeight += excludedSectionSize.height + otherSectionsHeight += sectionSpacing + + if case .greeting = component.mode { + var selectedInactivityIndex = 0 + let valueList: [Int] = [ + 7, + 14, + 21, + 28 + ] + for i in 0 ..< valueList.count { + if valueList[i] <= self.inactivityDays { + selectedInactivityIndex = i + } + } + + let periodSectionSize = self.periodSection.update( + transition: transition, + component: AnyComponent(ListSectionComponent( + theme: environment.theme, + header: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: environment.strings.BusinessMessageSetup_InactivitySectionHeader, + font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), + textColor: environment.theme.list.freeTextColor + )), + maximumNumberOfLines: 0 + )), + footer: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: environment.strings.BusinessMessageSetup_InactivitySectionFooter, + font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), + textColor: environment.theme.list.freeTextColor + )), + maximumNumberOfLines: 0 + )), + items: [ + AnyComponentWithIdentity(id: 0, component: AnyComponent(ListItemSliderSelectorComponent( + theme: environment.theme, + values: valueList.map { item in + return environment.strings.MessageTimer_Days(Int32(item)) + }, + selectedIndex: selectedInactivityIndex, + selectedIndexUpdated: { [weak self] index in + guard let self else { + return + } + let index = max(0, min(valueList.count - 1, index)) + self.inactivityDays = valueList[index] + self.state?.updated(transition: .immediate) + } + ))) + ] + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 10000.0) + ) + let periodSectionFrame = CGRect(origin: CGPoint(x: sideInset, y: contentHeight + otherSectionsHeight), size: periodSectionSize) + if let periodSectionView = self.periodSection.view { + if periodSectionView.superview == nil { + periodSectionView.layer.allowsGroupOpacity = true + self.scrollView.addSubview(periodSectionView) + } + transition.setFrame(view: periodSectionView, frame: periodSectionFrame) + alphaTransition.setAlpha(view: periodSectionView, alpha: self.isOn ? 1.0 : 0.0) + } + otherSectionsHeight += periodSectionSize.height + otherSectionsHeight += sectionSpacing + } + + if self.isOn { + contentHeight += otherSectionsHeight + } + + contentHeight += bottomContentInset + contentHeight += environment.safeInsets.bottom + + let previousBounds = self.scrollView.bounds + + let contentSize = CGSize(width: availableSize.width, height: contentHeight) + self.ignoreScrolling = true + if self.scrollView.frame != CGRect(origin: CGPoint(), size: availableSize) { + self.scrollView.frame = CGRect(origin: CGPoint(), size: availableSize) + } + if self.scrollView.contentSize != contentSize { + self.scrollView.contentSize = contentSize + } + let scrollInsets = UIEdgeInsets(top: environment.navigationHeight, left: 0.0, bottom: 0.0, right: 0.0) + if self.scrollView.scrollIndicatorInsets != scrollInsets { + self.scrollView.scrollIndicatorInsets = scrollInsets + } + self.ignoreScrolling = false + + if !previousBounds.isEmpty, !transition.animation.isImmediate { + let bounds = self.scrollView.bounds + if bounds.maxY != previousBounds.maxY { + let offsetY = previousBounds.maxY - bounds.maxY + transition.animateBoundsOrigin(view: self.scrollView, from: CGPoint(x: 0.0, y: offsetY), to: CGPoint(), additive: true) + } + } + + self.topOverscrollLayer.frame = CGRect(origin: CGPoint(x: 0.0, y: -3000.0), size: CGSize(width: availableSize.width, height: 3000.0)) + + self.updateScrolling(transition: transition) + + return availableSize + } + } + + func makeView() -> View { + return View() + } + + func update(view: View, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + return view.update(component: self, availableSize: availableSize, state: state, environment: environment, transition: transition) + } +} + +public final class AutomaticBusinessMessageSetupScreen: ViewControllerComponentContainer { + public final class InitialData: AutomaticBusinessMessageSetupScreenInitialData { + fileprivate let accountPeer: EnginePeer? + fileprivate let shortcutMessageList: ShortcutMessageList + fileprivate let greetingMessage: TelegramBusinessGreetingMessage? + fileprivate let awayMessage: TelegramBusinessAwayMessage? + fileprivate let additionalPeers: [EnginePeer.Id: AutomaticBusinessMessageSetupScreenComponent.AdditionalPeerList.Peer] + fileprivate let businessHours: TelegramBusinessHours? + + fileprivate init( + accountPeer: EnginePeer?, + shortcutMessageList: ShortcutMessageList, + greetingMessage: TelegramBusinessGreetingMessage?, + awayMessage: TelegramBusinessAwayMessage?, + additionalPeers: [EnginePeer.Id: AutomaticBusinessMessageSetupScreenComponent.AdditionalPeerList.Peer], + businessHours: TelegramBusinessHours? + ) { + self.accountPeer = accountPeer + self.shortcutMessageList = shortcutMessageList + self.greetingMessage = greetingMessage + self.awayMessage = awayMessage + self.additionalPeers = additionalPeers + self.businessHours = businessHours + } + } + + public enum Mode { + case greeting + case away + } + + private let context: AccountContext + + public init(context: AccountContext, initialData: InitialData, mode: Mode) { + self.context = context + + super.init(context: context, component: AutomaticBusinessMessageSetupScreenComponent( + context: context, + initialData: initialData, + mode: mode + ), navigationBarAppearance: .default, theme: .default, updatedPresentationData: nil) + + let presentationData = context.sharedContext.currentPresentationData.with { $0 } + self.title = "" + self.navigationItem.backBarButtonItem = UIBarButtonItem(title: presentationData.strings.Common_Back, style: .plain, target: nil, action: nil) + + self.scrollToTop = { [weak self] in + guard let self, let componentView = self.node.hostView.componentView as? AutomaticBusinessMessageSetupScreenComponent.View else { + return + } + componentView.scrollToTop() + } + + self.attemptNavigation = { [weak self] complete in + guard let self, let componentView = self.node.hostView.componentView as? AutomaticBusinessMessageSetupScreenComponent.View else { + return true + } + + return componentView.attemptNavigation(complete: complete) + } + } + + required public init(coder aDecoder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + deinit { + } + + @objc private func cancelPressed() { + self.dismiss() + } + + override public func containerLayoutUpdated(_ layout: ContainerViewLayout, transition: ContainedViewLayoutTransition) { + super.containerLayoutUpdated(layout, transition: transition) + } + + public static func initialData(context: AccountContext) -> Signal { + return combineLatest( + context.engine.data.get( + TelegramEngine.EngineData.Item.Peer.Peer(id: context.account.peerId), + TelegramEngine.EngineData.Item.Peer.BusinessGreetingMessage(id: context.account.peerId), + TelegramEngine.EngineData.Item.Peer.BusinessAwayMessage(id: context.account.peerId), + TelegramEngine.EngineData.Item.Peer.BusinessHours(id: context.account.peerId) + ), + context.engine.accountData.shortcutMessageList(onlyRemote: true) + |> take(1) + ) + |> mapToSignal { data, shortcutMessageList -> Signal in + let (accountPeer, greetingMessage, awayMessage, businessHours) = data + + var additionalPeerIds = Set() + if let greetingMessage { + additionalPeerIds.formUnion(greetingMessage.recipients.additionalPeers) + } + if let awayMessage { + additionalPeerIds.formUnion(awayMessage.recipients.additionalPeers) + } + + return context.engine.data.get( + EngineDataMap(additionalPeerIds.map(TelegramEngine.EngineData.Item.Peer.Peer.init(id:))), + EngineDataMap(additionalPeerIds.map(TelegramEngine.EngineData.Item.Peer.IsContact.init(id:))) + ) + |> map { peers, isContacts -> AutomaticBusinessMessageSetupScreenInitialData in + var additionalPeers: [EnginePeer.Id: AutomaticBusinessMessageSetupScreenComponent.AdditionalPeerList.Peer] = [:] + for id in additionalPeerIds { + guard let peer = peers[id], let peer else { + continue + } + additionalPeers[id] = AutomaticBusinessMessageSetupScreenComponent.AdditionalPeerList.Peer( + peer: peer, + isContact: isContacts[id] ?? false + ) + } + + return InitialData( + accountPeer: accountPeer, + shortcutMessageList: shortcutMessageList, + greetingMessage: greetingMessage, + awayMessage: awayMessage, + additionalPeers: additionalPeers, + businessHours: businessHours + ) + } + } + } +} diff --git a/submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/BottomPanelComponent.swift b/submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/BottomPanelComponent.swift new file mode 100644 index 00000000000..2dd0f3bae71 --- /dev/null +++ b/submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/BottomPanelComponent.swift @@ -0,0 +1,117 @@ +import Foundation +import UIKit +import Display +import ComponentFlow +import TelegramPresentationData +import ComponentDisplayAdapters + +final class BottomPanelComponent: Component { + let theme: PresentationTheme + let content: AnyComponentWithIdentity + let insets: UIEdgeInsets + + init( + theme: PresentationTheme, + content: AnyComponentWithIdentity, + insets: UIEdgeInsets + ) { + self.theme = theme + self.content = content + self.insets = insets + } + + static func ==(lhs: BottomPanelComponent, rhs: BottomPanelComponent) -> Bool { + if lhs.theme !== rhs.theme { + return false + } + if lhs.content != rhs.content { + return false + } + if lhs.insets != rhs.insets { + return false + } + return true + } + + final class View: UIView { + private let separatorLayer: SimpleLayer + private let backgroundView: BlurredBackgroundView + private var content = ComponentView() + + private var component: BottomPanelComponent? + private weak var componentState: EmptyComponentState? + + override init(frame: CGRect) { + self.separatorLayer = SimpleLayer() + self.backgroundView = BlurredBackgroundView(color: .clear, enableBlur: true) + + super.init(frame: frame) + + self.addSubview(self.backgroundView) + self.layer.addSublayer(self.separatorLayer) + } + + required init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + func update(component: BottomPanelComponent, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + let previousComponent = self.component + self.component = component + self.componentState = state + + let themeUpdated = previousComponent?.theme !== component.theme + + var contentHeight: CGFloat = 0.0 + + contentHeight += component.insets.top + + var contentTransition = transition + if let previousComponent, previousComponent.content.id != component.content.id { + contentTransition = contentTransition.withAnimation(.none) + self.content.view?.removeFromSuperview() + self.content = ComponentView() + } + + let contentSize = self.content.update( + transition: contentTransition, + component: component.content.component, + environment: {}, + containerSize: CGSize(width: availableSize.width - component.insets.left - component.insets.right, height: availableSize.height - component.insets.top - component.insets.bottom) + ) + let contentFrame = CGRect(origin: CGPoint(x: floor((availableSize.width - contentSize.width) * 0.5), y: contentHeight), size: contentSize) + if let contentView = self.content.view { + if contentView.superview == nil { + self.addSubview(contentView) + } + contentTransition.setFrame(view: contentView, frame: contentFrame) + } + contentHeight += contentSize.height + + contentHeight += component.insets.bottom + + let size = CGSize(width: availableSize.width, height: contentHeight) + + if themeUpdated { + self.backgroundView.updateColor(color: component.theme.rootController.navigationBar.blurredBackgroundColor, transition: .immediate) + self.separatorLayer.backgroundColor = component.theme.rootController.navigationBar.separatorColor.cgColor + } + + let backgroundFrame = CGRect(origin: CGPoint(x: 0.0, y: 0.0), size: size) + transition.setFrame(view: self.backgroundView, frame: backgroundFrame) + self.backgroundView.update(size: backgroundFrame.size, transition: transition.containedViewLayoutTransition) + + transition.setFrame(layer: self.separatorLayer, frame: CGRect(origin: CGPoint(x: 0.0, y: -UIScreenPixel), size: CGSize(width: availableSize.width, height: UIScreenPixel))) + + return size + } + } + + func makeView() -> View { + return View(frame: CGRect()) + } + + func update(view: View, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + return view.update(component: self, availableSize: availableSize, state: state, environment: environment, transition: transition) + } +} diff --git a/submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/QuickReplyEmptyStateComponent.swift b/submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/QuickReplyEmptyStateComponent.swift new file mode 100644 index 00000000000..4582101ed6f --- /dev/null +++ b/submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/QuickReplyEmptyStateComponent.swift @@ -0,0 +1,186 @@ +import Foundation +import UIKit +import Display +import ComponentFlow +import TelegramPresentationData +import AppBundle +import ButtonComponent +import MultilineTextComponent +import BalancedTextComponent +import LottieComponent + +final class QuickReplyEmptyStateComponent: Component { + let theme: PresentationTheme + let strings: PresentationStrings + let insets: UIEdgeInsets + let action: () -> Void + + init( + theme: PresentationTheme, + strings: PresentationStrings, + insets: UIEdgeInsets, + action: @escaping () -> Void + ) { + self.theme = theme + self.strings = strings + self.insets = insets + self.action = action + } + + static func ==(lhs: QuickReplyEmptyStateComponent, rhs: QuickReplyEmptyStateComponent) -> Bool { + if lhs.theme !== rhs.theme { + return false + } + if lhs.strings !== rhs.strings { + return false + } + if lhs.insets != rhs.insets { + return false + } + return true + } + + final class View: UIView { + private let icon = ComponentView() + private let title = ComponentView() + private let text = ComponentView() + private let button = ComponentView() + + private var component: QuickReplyEmptyStateComponent? + private weak var componentState: EmptyComponentState? + + override init(frame: CGRect) { + super.init(frame: frame) + } + + required init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + func update(component: QuickReplyEmptyStateComponent, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + let previousComponent = self.component + self.component = component + self.componentState = state + + let _ = previousComponent + + let buttonSize = self.button.update( + transition: transition, + component: AnyComponent(ButtonComponent( + background: ButtonComponent.Background( + color: component.theme.list.itemCheckColors.fillColor, + foreground: component.theme.list.itemCheckColors.foregroundColor, + pressedColor: component.theme.list.itemCheckColors.fillColor.withMultipliedAlpha(0.9) + ), + content: AnyComponentWithIdentity( + id: AnyHashable(0), + component: AnyComponent(ButtonTextContentComponent( + text: component.strings.QuickReplies_EmptyState_AddButton, + badge: 0, + textColor: component.theme.list.itemCheckColors.foregroundColor, + badgeBackground: component.theme.list.itemCheckColors.foregroundColor, + badgeForeground: component.theme.list.itemCheckColors.fillColor + )) + ), + isEnabled: true, + displaysProgress: false, + action: { [weak self] in + guard let self, let component = self.component else { + return + } + component.action() + } + )), + environment: {}, + containerSize: CGSize(width: min(availableSize.width - 16.0 * 2.0, 280.0), height: 50.0) + ) + let buttonFrame = CGRect(origin: CGPoint(x: floor((availableSize.width - buttonSize.width) * 0.5), y: availableSize.height - component.insets.bottom - 14.0 - buttonSize.height), size: buttonSize) + if let buttonView = self.button.view { + if buttonView.superview == nil { + self.addSubview(buttonView) + } + transition.setFrame(view: buttonView, frame: buttonFrame) + } + + let iconTitleSpacing: CGFloat = 13.0 + let titleTextSpacing: CGFloat = 9.0 + + let iconSize = self.icon.update( + transition: .immediate, + component: AnyComponent(LottieComponent( + content: LottieComponent.AppBundleContent(name: "WriteEmoji"), + loop: false + )), + environment: {}, + containerSize: CGSize(width: 120.0, height: 120.0) + ) + + let titleSize = self.title.update( + transition: .immediate, + component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: component.strings.QuickReplies_EmptyState_Title, font: Font.semibold(17.0), textColor: component.theme.rootController.navigationBar.primaryTextColor)), + horizontalAlignment: .center + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - 16.0 * 2.0, height: 100.0) + ) + + let textSize = self.text.update( + transition: .immediate, + component: AnyComponent(BalancedTextComponent( + text: .plain(NSAttributedString(string: component.strings.QuickReplies_EmptyState_Text, font: Font.regular(15.0), textColor: component.theme.list.itemSecondaryTextColor)), + horizontalAlignment: .center, + maximumNumberOfLines: 20, + lineSpacing: 0.2 + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - 16.0 * 2.0, height: 100.0) + ) + + let topInset: CGFloat = component.insets.top + + let centralContentsHeight: CGFloat = iconSize.height + iconTitleSpacing + titleSize.height + titleTextSpacing + var centralContentsY: CGFloat = topInset + floor((buttonFrame.minY - topInset - centralContentsHeight) * 0.426) + + let iconFrame = CGRect(origin: CGPoint(x: floor((availableSize.width - iconSize.width) * 0.5), y: centralContentsY), size: iconSize) + + if let iconView = self.icon.view as? LottieComponent.View { + if iconView.superview == nil { + self.addSubview(iconView) + iconView.playOnce() + } + transition.setFrame(view: iconView, frame: iconFrame) + } + centralContentsY += iconSize.height + iconTitleSpacing + + let titleFrame = CGRect(origin: CGPoint(x: floor((availableSize.width - titleSize.width) * 0.5), y: centralContentsY), size: titleSize) + if let titleView = self.title.view { + if titleView.superview == nil { + self.addSubview(titleView) + } + titleView.bounds = CGRect(origin: CGPoint(), size: titleFrame.size) + transition.setPosition(view: titleView, position: titleFrame.center) + } + centralContentsY += titleSize.height + titleTextSpacing + + let textFrame = CGRect(origin: CGPoint(x: floor((availableSize.width - textSize.width) * 0.5), y: centralContentsY), size: textSize) + if let textView = self.text.view { + if textView.superview == nil { + self.addSubview(textView) + } + textView.bounds = CGRect(origin: CGPoint(), size: textFrame.size) + transition.setPosition(view: textView, position: textFrame.center) + } + + return availableSize + } + } + + func makeView() -> View { + return View(frame: CGRect()) + } + + func update(view: View, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + return view.update(component: self, availableSize: availableSize, state: state, environment: environment, transition: transition) + } +} diff --git a/submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/QuickReplySetupScreen.swift b/submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/QuickReplySetupScreen.swift new file mode 100644 index 00000000000..22d8f7e0cae --- /dev/null +++ b/submodules/TelegramUI/Components/Settings/AutomaticBusinessMessageSetupScreen/Sources/QuickReplySetupScreen.swift @@ -0,0 +1,1384 @@ +import Foundation +import UIKit +import Photos +import Display +import AsyncDisplayKit +import SwiftSignalKit +import Postbox +import TelegramCore +import TelegramPresentationData +import TelegramUIPreferences +import PresentationDataUtils +import AccountContext +import ComponentFlow +import ViewControllerComponent +import MergeLists +import ComponentDisplayAdapters +import ItemListPeerActionItem +import ItemListUI +import ChatListUI +import QuickReplyNameAlertController +import ChatListHeaderComponent +import PlainButtonComponent +import MultilineTextComponent +import AttachmentUI +import SearchBarNode +import BalancedTextComponent + +final class QuickReplySetupScreenComponent: Component { + typealias EnvironmentType = ViewControllerComponentContainer.Environment + + let context: AccountContext + let initialData: QuickReplySetupScreen.InitialData + let mode: QuickReplySetupScreen.Mode + + init( + context: AccountContext, + initialData: QuickReplySetupScreen.InitialData, + mode: QuickReplySetupScreen.Mode + ) { + self.context = context + self.initialData = initialData + self.mode = mode + } + + static func ==(lhs: QuickReplySetupScreenComponent, rhs: QuickReplySetupScreenComponent) -> Bool { + if lhs.context !== rhs.context { + return false + } + + return true + } + + private enum ContentEntry: Comparable, Identifiable { + enum Id: Hashable { + case add + case item(Int32) + case pendingItem(String) + } + + var stableId: Id { + switch self { + case .add: + return .add + case let .item(item, _, _, _, _): + if let itemId = item.id { + return .item(itemId) + } else { + return .pendingItem(item.shortcut) + } + } + } + + case add + case item(item: ShortcutMessageList.Item, accountPeer: EnginePeer, sortIndex: Int, isEditing: Bool, isSelected: Bool) + + static func <(lhs: ContentEntry, rhs: ContentEntry) -> Bool { + switch lhs { + case .add: + return false + case let .item(lhsItem, _, lhsSortIndex, _, _): + switch rhs { + case .add: + return false + case let .item(rhsItem, _, rhsSortIndex, _, _): + if lhsSortIndex != rhsSortIndex { + return lhsSortIndex < rhsSortIndex + } + return lhsItem.shortcut < rhsItem.shortcut + } + } + } + + func item(listNode: ContentListNode) -> ListViewItem { + switch self { + case .add: + return ItemListPeerActionItem( + presentationData: ItemListPresentationData(listNode.presentationData), + icon: PresentationResourcesItemList.plusIconImage(listNode.presentationData.theme), + iconSignal: nil, + title: listNode.presentationData.strings.QuickReply_InlineCreateAction, + additionalBadgeIcon: nil, + alwaysPlain: true, + hasSeparator: true, + sectionId: 0, + height: .generic, + color: .accent, + editing: false, + action: { [weak listNode] in + guard let listNode, let parentView = listNode.parentView else { + return + } + parentView.openQuickReplyChat(shortcut: nil, shortcutId: nil) + } + ) + case let .item(item, accountPeer, _, isEditing, isSelected): + let chatListNodeInteraction = ChatListNodeInteraction( + context: listNode.context, + animationCache: listNode.context.animationCache, + animationRenderer: listNode.context.animationRenderer, + activateSearch: { + }, + peerSelected: { [weak listNode] _, _, _, _ in + guard let listNode, let parentView = listNode.parentView else { + return + } + parentView.openQuickReplyChat(shortcut: item.shortcut, shortcutId: item.id) + }, + disabledPeerSelected: { _, _, _ in + }, + togglePeerSelected: { [weak listNode] _, _ in + guard let listNode, let parentView = listNode.parentView else { + return + } + if let itemId = item.id { + parentView.toggleShortcutSelection(id: itemId) + } + }, + togglePeersSelection: { [weak listNode] _, _ in + guard let listNode, let parentView = listNode.parentView else { + return + } + if let itemId = item.id { + parentView.toggleShortcutSelection(id: itemId) + } + }, + additionalCategorySelected: { _ in + }, + messageSelected: { [weak listNode] _, _, _, _ in + guard let listNode, let parentView = listNode.parentView else { + return + } + parentView.openQuickReplyChat(shortcut: item.shortcut, shortcutId: item.id) + }, + groupSelected: { _ in + }, + addContact: { _ in + }, + setPeerIdWithRevealedOptions: { _, _ in + }, + setItemPinned: { _, _ in + }, + setPeerMuted: { _, _ in + }, + setPeerThreadMuted: { _, _, _ in + }, + deletePeer: { [weak listNode] _, _ in + guard let listNode, let parentView = listNode.parentView else { + return + } + if let itemId = item.id { + parentView.openDeleteShortcuts(ids: [itemId]) + } + }, + deletePeerThread: { _, _ in + }, + setPeerThreadStopped: { _, _, _ in + }, + setPeerThreadPinned: { _, _, _ in + }, + setPeerThreadHidden: { _, _, _ in + }, + updatePeerGrouping: { _, _ in + }, + togglePeerMarkedUnread: { _, _ in + }, + toggleArchivedFolderHiddenByDefault: { + }, + toggleThreadsSelection: { _, _ in + }, + hidePsa: { _ in + }, + activateChatPreview: { _, _, _, _, _ in + }, + present: { _ in + }, + openForumThread: { _, _ in + }, + openStorageManagement: { + }, + openPasswordSetup: { + }, + openPremiumIntro: { + }, + openPremiumGift: { + }, + openActiveSessions: { + }, + performActiveSessionAction: { _, _ in + }, + openChatFolderUpdates: { + }, + hideChatFolderUpdates: { + }, + openStories: { _, _ in + }, + dismissNotice: { _ in + }, + editPeer: { [weak listNode] _ in + guard let listNode, let parentView = listNode.parentView else { + return + } + if let itemId = item.id { + parentView.openEditShortcut(id: itemId, currentValue: item.shortcut) + } + } + ) + + let presentationData = listNode.context.sharedContext.currentPresentationData.with({ $0 }) + let chatListPresentationData = ChatListPresentationData( + theme: presentationData.theme, + fontSize: presentationData.listsFontSize, + strings: presentationData.strings, + dateTimeFormat: presentationData.dateTimeFormat, + nameSortOrder: presentationData.nameSortOrder, + nameDisplayOrder: presentationData.nameDisplayOrder, + disableAnimations: false + ) + + return ChatListItem( + presentationData: chatListPresentationData, + context: listNode.context, + chatListLocation: .chatList(groupId: .root), + filterData: nil, + index: EngineChatList.Item.Index.chatList(ChatListIndex(pinningIndex: nil, messageIndex: MessageIndex(id: MessageId(peerId: listNode.context.account.peerId, namespace: 0, id: 0), timestamp: 0))), + content: .peer(ChatListItemContent.PeerData( + messages: [item.topMessage], + peer: EngineRenderedPeer(peer: accountPeer), + threadInfo: nil, + combinedReadState: nil, + isRemovedFromTotalUnreadCount: false, + presence: nil, + hasUnseenMentions: false, + hasUnseenReactions: false, + draftState: nil, + mediaDraftContentType: nil, + inputActivities: nil, + promoInfo: nil, + ignoreUnreadBadge: false, + displayAsMessage: false, + hasFailedMessages: false, + forumTopicData: nil, + topForumTopicItems: [], + autoremoveTimeout: nil, + storyState: nil, + requiresPremiumForMessaging: false, + displayAsTopicList: false, + tags: [], + customMessageListData: ChatListItemContent.CustomMessageListData( + commandPrefix: "/\(item.shortcut)", + searchQuery: nil, + messageCount: item.totalCount, + hideSeparator: false + ) + )), + editing: isEditing, + hasActiveRevealControls: false, + selected: isSelected, + header: nil, + enableContextActions: true, + hiddenOffset: false, + interaction: chatListNodeInteraction + ) + } + } + } + + private final class ContentListNode: ListView { + weak var parentView: View? + let context: AccountContext + var presentationData: PresentationData + private var currentEntries: [ContentEntry] = [] + private var originalEntries: [ContentEntry] = [] + private var tempOrder: [Int32]? + private var pendingRemoveItems: [Int32]? + private var resetTempOrderOnNextUpdate: Bool = false + + init(parentView: View, context: AccountContext) { + self.parentView = parentView + self.context = context + self.presentationData = context.sharedContext.currentPresentationData.with({ $0 }) + + super.init() + + self.reorderBegan = { [weak self] in + guard let self else { + return + } + self.tempOrder = nil + } + self.reorderCompleted = { [weak self] _ in + guard let self, let tempOrder = self.tempOrder else { + return + } + self.resetTempOrderOnNextUpdate = true + self.context.engine.accountData.reorderMessageShortcuts(ids: tempOrder, completion: {}) + } + self.reorderItem = { [weak self] fromIndex, toIndex, transactionOpaqueState -> Signal in + guard let self else { + return .single(false) + } + guard fromIndex >= 0 && fromIndex < self.currentEntries.count && toIndex >= 0 && toIndex < self.currentEntries.count else { + return .single(false) + } + + let fromEntry = self.currentEntries[fromIndex] + let toEntry = self.currentEntries[toIndex] + + var referenceId: Int32? + var beforeAll = false + switch toEntry { + case let .item(item, _, _, _, _): + referenceId = item.id + case .add: + beforeAll = true + } + + if case let .item(item, _, _, _, _) = fromEntry { + var itemIds = self.currentEntries.compactMap { entry -> Int32? in + switch entry { + case .add: + return nil + case let .item(item, _, _, _, _): + return item.id + } + } + let itemId: Int32? = item.id + + if let itemId { + itemIds = itemIds.filter({ $0 != itemId }) + if let referenceId { + var inserted = false + for i in 0 ..< itemIds.count { + if itemIds[i] == referenceId { + if fromIndex < toIndex { + itemIds.insert(itemId, at: i + 1) + } else { + itemIds.insert(itemId, at: i) + } + inserted = true + break + } + } + if !inserted { + itemIds.append(itemId) + } + } else if beforeAll { + itemIds.insert(itemId, at: 0) + } else { + itemIds.append(itemId) + } + if self.tempOrder != itemIds { + self.tempOrder = itemIds + self.setEntries(entries: self.originalEntries, animated: true) + } + + return .single(true) + } else { + return .single(false) + } + } else { + return .single(false) + } + } + } + + func update(size: CGSize, insets: UIEdgeInsets, transition: Transition) { + let (listViewDuration, listViewCurve) = listViewAnimationDurationAndCurve(transition: transition.containedViewLayoutTransition) + self.transaction( + deleteIndices: [], + insertIndicesAndItems: [], + updateIndicesAndItems: [], + options: [.Synchronous, .LowLatency, .PreferSynchronousResourceLoading], + additionalScrollDistance: 0.0, + updateSizeAndInsets: ListViewUpdateSizeAndInsets(size: size, insets: insets, duration: listViewDuration, curve: listViewCurve), + updateOpaqueState: nil + ) + } + + func setPendingRemoveItems(itemIds: [Int32]) { + self.pendingRemoveItems = itemIds + self.setEntries(entries: self.originalEntries, animated: true) + } + + func setEntries(entries: [ContentEntry], animated: Bool) { + if self.resetTempOrderOnNextUpdate { + self.resetTempOrderOnNextUpdate = false + self.tempOrder = nil + } + let pendingRemoveItems = self.pendingRemoveItems + self.pendingRemoveItems = nil + + self.originalEntries = entries + + var entries = entries + if let pendingRemoveItems { + entries = entries.filter { entry in + switch entry.stableId { + case .add: + return true + case let .item(id): + return !pendingRemoveItems.contains(id) + case .pendingItem: + return true + } + } + } + + if let tempOrder = self.tempOrder { + let originalList = entries + entries.removeAll() + + if let entry = originalList.first(where: { entry in + if case .add = entry { + return true + } else { + return false + } + }) { + entries.append(entry) + } + + for id in tempOrder { + if let entry = originalList.first(where: { entry in + if case let .item(listId) = entry.stableId, listId == id { + return true + } else { + return false + } + }) { + entries.append(entry) + } + } + for entry in originalList { + if !entries.contains(where: { listEntry in + listEntry.stableId == entry.stableId + }) { + entries.append(entry) + } + } + } + + let (deleteIndices, indicesAndItems, updateIndices) = mergeListsStableWithUpdates(leftList: self.currentEntries, rightList: entries) + self.currentEntries = entries + + let deletions = deleteIndices.map { ListViewDeleteItem(index: $0, directionHint: nil) } + let insertions = indicesAndItems.map { ListViewInsertItem(index: $0.0, previousIndex: $0.2, item: $0.1.item(listNode: self), directionHint: nil) } + let updates = updateIndices.map { ListViewUpdateItem(index: $0.0, previousIndex: $0.2, item: $0.1.item(listNode: self), directionHint: nil) } + + var options: ListViewDeleteAndInsertOptions = [.Synchronous, .LowLatency] + if animated { + options.insert(.AnimateInsertion) + } else { + options.insert(.PreferSynchronousResourceLoading) + } + + self.transaction( + deleteIndices: deletions, + insertIndicesAndItems: insertions, + updateIndicesAndItems: updates, + options: options, + scrollToItem: nil, + stationaryItemRange: nil, + updateOpaqueState: nil, + completion: { _ in + } + ) + } + } + + final class View: UIView { + private var emptyState: ComponentView? + private var contentListNode: ContentListNode? + private var emptySearchState: ComponentView? + + private let navigationBarView = ComponentView() + private var navigationHeight: CGFloat? + + private var searchBarNode: SearchBarNode? + + private var selectionPanel: ComponentView? + + private var isUpdating: Bool = false + + private var component: QuickReplySetupScreenComponent? + private(set) weak var state: EmptyComponentState? + private var environment: EnvironmentType? + + private var shortcutMessageList: ShortcutMessageList? + private var shortcutMessageListDisposable: Disposable? + private var keepUpdatedDisposable: Disposable? + + private var selectedIds = Set() + + private var isEditing: Bool = false + private var isSearchDisplayControllerActive: Bool = false + private var searchQuery: String = "" + private let searchQueryComponentSeparationCharacterSet: CharacterSet + + private var accountPeer: EnginePeer? + + override init(frame: CGRect) { + self.searchQueryComponentSeparationCharacterSet = CharacterSet(charactersIn: " _.:/") + + super.init(frame: frame) + } + + required init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + deinit { + self.shortcutMessageListDisposable?.dispose() + self.keepUpdatedDisposable?.dispose() + } + + func scrollToTop() { + } + + func attemptNavigation(complete: @escaping () -> Void) -> Bool { + return true + } + + func openQuickReplyChat(shortcut: String?, shortcutId: Int32?) { + guard let component = self.component, let environment = self.environment else { + return + } + + if case let .select(completion) = component.mode { + if let shortcutId { + completion(shortcutId) + } + return + } + + if let shortcut { + let shortcutType: ChatQuickReplyShortcutType + if shortcut == "hello" { + shortcutType = .greeting + } else if shortcut == "away" { + shortcutType = .away + } else { + shortcutType = .generic + } + + let contents = AutomaticBusinessMessageSetupChatContents( + context: component.context, + kind: .quickReplyMessageInput(shortcut: shortcut, shortcutType: shortcutType), + shortcutId: shortcutId + ) + let chatController = component.context.sharedContext.makeChatController( + context: component.context, + chatLocation: .customChatContents, + subject: .customChatContents(contents: contents), + botStart: nil, + mode: .standard(.default) + ) + chatController.navigationPresentation = .modal + self.environment?.controller()?.push(chatController) + } else { + var completion: ((String?) -> Void)? + let alertController = quickReplyNameAlertController( + context: component.context, + text: environment.strings.QuickReply_CreateShortcutTitle, + subtext: environment.strings.QuickReply_CreateShortcutText, + value: "", + characterLimit: 32, + apply: { value in + completion?(value) + } + ) + completion = { [weak self, weak alertController] value in + guard let self, let environment = self.environment else { + alertController?.dismissAnimated() + return + } + if let value, !value.isEmpty { + guard let shortcutMessageList = self.shortcutMessageList else { + alertController?.dismissAnimated() + return + } + + if shortcutMessageList.items.contains(where: { $0.shortcut.lowercased() == value.lowercased() }) { + if let contentNode = alertController?.contentNode as? QuickReplyNameAlertContentNode { + contentNode.setErrorText(errorText: environment.strings.QuickReply_ShortcutExistsInlineError) + } + return + } + + alertController?.view.endEditing(true) + alertController?.dismissAnimated() + self.openQuickReplyChat(shortcut: value, shortcutId: nil) + } + } + self.environment?.controller()?.present(alertController, in: .window(.root)) + } + + self.contentListNode?.clearHighlightAnimated(true) + } + + func openEditShortcut(id: Int32, currentValue: String) { + guard let component = self.component, let environment = self.environment else { + return + } + + var completion: ((String?) -> Void)? + let alertController = quickReplyNameAlertController( + context: component.context, + text: environment.strings.QuickReply_EditShortcutTitle, + subtext: environment.strings.QuickReply_EditShortcutText, + value: currentValue, + characterLimit: 32, + apply: { value in + completion?(value) + } + ) + completion = { [weak self, weak alertController] value in + guard let self, let component = self.component, let environment = self.environment else { + alertController?.dismissAnimated() + return + } + if let value, !value.isEmpty { + if value == currentValue { + alertController?.dismissAnimated() + return + } + guard let shortcutMessageList = self.shortcutMessageList else { + alertController?.dismissAnimated() + return + } + + if shortcutMessageList.items.contains(where: { $0.shortcut.lowercased() == value.lowercased() }) { + if let contentNode = alertController?.contentNode as? QuickReplyNameAlertContentNode { + contentNode.setErrorText(errorText: environment.strings.QuickReply_ShortcutExistsInlineError) + } + } else { + component.context.engine.accountData.editMessageShortcut(id: id, shortcut: value) + + alertController?.view.endEditing(true) + alertController?.dismissAnimated() + } + } + } + self.environment?.controller()?.present(alertController, in: .window(.root)) + } + + func toggleShortcutSelection(id: Int32) { + if self.selectedIds.contains(id) { + self.selectedIds.remove(id) + } else { + self.selectedIds.insert(id) + } + self.state?.updated(transition: .spring(duration: 0.4)) + } + + func openDeleteShortcuts(ids: [Int32]) { + guard let component = self.component else { + return + } + + let presentationData = component.context.sharedContext.currentPresentationData.with { $0 } + let actionSheet = ActionSheetController(presentationData: presentationData) + var items: [ActionSheetItem] = [] + + items.append(ActionSheetButtonItem(title: ids.count == 1 ? presentationData.strings.QuickReply_DeleteConfirmationSingle : presentationData.strings.QuickReply_DeleteConfirmationMultiple, color: .destructive, action: { [weak self, weak actionSheet] in + actionSheet?.dismissAnimated() + guard let self, let component = self.component else { + return + } + + for id in ids { + self.selectedIds.remove(id) + } + self.contentListNode?.setPendingRemoveItems(itemIds: ids) + component.context.engine.accountData.deleteMessageShortcuts(ids: ids) + self.state?.updated(transition: .spring(duration: 0.4)) + })) + + actionSheet.setItemGroups([ + ActionSheetItemGroup(items: items), + ActionSheetItemGroup(items: [ + ActionSheetButtonItem(title: presentationData.strings.Common_Cancel, color: .accent, font: .bold, action: { [weak actionSheet] in + actionSheet?.dismissAnimated() + }) + ]) + ]) + self.environment?.controller()?.present(actionSheet, in: .window(.root)) + } + + private func updateNavigationBar( + component: QuickReplySetupScreenComponent, + theme: PresentationTheme, + strings: PresentationStrings, + size: CGSize, + insets: UIEdgeInsets, + statusBarHeight: CGFloat, + isModal: Bool, + transition: Transition, + deferScrollApplication: Bool + ) -> CGFloat { + var rightButtons: [AnyComponentWithIdentity] = [] + if let shortcutMessageList = self.shortcutMessageList, !shortcutMessageList.items.isEmpty { + if self.isEditing { + rightButtons.append(AnyComponentWithIdentity(id: "done", component: AnyComponent(NavigationButtonComponent( + content: .text(title: strings.Common_Done, isBold: true), + pressed: { [weak self] _ in + guard let self else { + return + } + self.isEditing = false + self.selectedIds.removeAll() + self.state?.updated(transition: .spring(duration: 0.4)) + } + )))) + } else { + rightButtons.append(AnyComponentWithIdentity(id: "edit", component: AnyComponent(NavigationButtonComponent( + content: .text(title: strings.Common_Edit, isBold: false), + pressed: { [weak self] _ in + guard let self else { + return + } + self.isEditing = true + self.state?.updated(transition: .spring(duration: 0.4)) + } + )))) + } + } + + let titleText: String + if !self.selectedIds.isEmpty { + titleText = strings.QuickReply_SelectedTitle(Int32(self.selectedIds.count)) + } else { + titleText = strings.QuickReply_Title + } + + let closeTitle: String + switch component.mode { + case .manage: + closeTitle = strings.Common_Close + case .select: + closeTitle = strings.Common_Cancel + } + let headerContent: ChatListHeaderComponent.Content? = ChatListHeaderComponent.Content( + title: titleText, + navigationBackTitle: nil, + titleComponent: nil, + chatListTitle: nil, + leftButton: isModal ? AnyComponentWithIdentity(id: "close", component: AnyComponent(NavigationButtonComponent( + content: .text(title: closeTitle, isBold: false), + pressed: { [weak self] _ in + guard let self else { + return + } + if self.attemptNavigation(complete: {}) { + self.environment?.controller()?.dismiss() + } + } + ))) : nil, + rightButtons: rightButtons, + backTitle: isModal ? nil : strings.Common_Back, + backPressed: { [weak self] in + guard let self else { + return + } + + if self.attemptNavigation(complete: {}) { + self.environment?.controller()?.dismiss() + } + } + ) + + let navigationBarSize = self.navigationBarView.update( + transition: transition, + component: AnyComponent(ChatListNavigationBar( + context: component.context, + theme: theme, + strings: strings, + statusBarHeight: statusBarHeight, + sideInset: insets.left, + isSearchActive: self.isSearchDisplayControllerActive, + isSearchEnabled: !self.isEditing, + primaryContent: headerContent, + secondaryContent: nil, + secondaryTransition: 0.0, + storySubscriptions: nil, + storiesIncludeHidden: false, + uploadProgress: [:], + tabsNode: nil, + tabsNodeIsSearch: false, + accessoryPanelContainer: nil, + accessoryPanelContainerHeight: 0.0, + activateSearch: { [weak self] _ in + guard let self else { + return + } + + self.isSearchDisplayControllerActive = true + self.state?.updated(transition: .spring(duration: 0.4)) + }, + openStatusSetup: { _ in + }, + allowAutomaticOrder: { + } + )), + environment: {}, + containerSize: size + ) + if let navigationBarComponentView = self.navigationBarView.view as? ChatListNavigationBar.View { + if deferScrollApplication { + navigationBarComponentView.deferScrollApplication = true + } + + if navigationBarComponentView.superview == nil { + self.addSubview(navigationBarComponentView) + } + transition.setFrame(view: navigationBarComponentView, frame: CGRect(origin: CGPoint(), size: navigationBarSize)) + + return navigationBarSize.height + } else { + return 0.0 + } + } + + private func updateNavigationScrolling(navigationHeight: CGFloat, transition: Transition) { + var mainOffset: CGFloat + if let shortcutMessageList = self.shortcutMessageList, !shortcutMessageList.items.isEmpty { + if let contentListNode = self.contentListNode { + switch contentListNode.visibleContentOffset() { + case .none: + mainOffset = 0.0 + case .unknown: + mainOffset = navigationHeight + case let .known(value): + mainOffset = value + } + } else { + mainOffset = navigationHeight + } + } else { + mainOffset = navigationHeight + } + + mainOffset = min(mainOffset, ChatListNavigationBar.searchScrollHeight) + if abs(mainOffset) < 0.1 { + mainOffset = 0.0 + } + + let resultingOffset = mainOffset + + var offset = resultingOffset + if self.isSearchDisplayControllerActive { + offset = 0.0 + } + + if let navigationBarComponentView = self.navigationBarView.view as? ChatListNavigationBar.View { + navigationBarComponentView.applyScroll(offset: offset, allowAvatarsExpansion: false, forceUpdate: false, transition: transition.withUserData(ChatListNavigationBar.AnimationHint( + disableStoriesAnimations: false, + crossfadeStoryPeers: false + ))) + } + } + + func update(component: QuickReplySetupScreenComponent, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + self.isUpdating = true + defer { + self.isUpdating = false + } + + if self.component == nil { + self.accountPeer = component.initialData.accountPeer + self.shortcutMessageList = component.initialData.shortcutMessageList + + self.shortcutMessageListDisposable = (component.context.engine.accountData.shortcutMessageList(onlyRemote: false) + |> deliverOnMainQueue).startStrict(next: { [weak self] shortcutMessageList in + guard let self else { + return + } + self.shortcutMessageList = shortcutMessageList + if !self.isUpdating { + self.state?.updated(transition: .immediate) + } + }) + + self.keepUpdatedDisposable = component.context.engine.accountData.keepShortcutMessageListUpdated().startStrict() + } + + let environment = environment[EnvironmentType.self].value + let themeUpdated = self.environment?.theme !== environment.theme + self.environment = environment + + self.component = component + self.state = state + + let alphaTransition: Transition = transition.animation.isImmediate ? transition : transition.withAnimation(.curve(duration: 0.25, curve: .easeInOut)) + let _ = alphaTransition + + if themeUpdated { + self.backgroundColor = environment.theme.list.plainBackgroundColor + } + + if let shortcutMessageList = self.shortcutMessageList, !shortcutMessageList.items.isEmpty { + if let emptyState = self.emptyState { + self.emptyState = nil + emptyState.view?.removeFromSuperview() + } + } else { + let emptyState: ComponentView + var emptyStateTransition = transition + if let current = self.emptyState { + emptyState = current + } else { + emptyState = ComponentView() + self.emptyState = emptyState + emptyStateTransition = emptyStateTransition.withAnimation(.none) + } + + let emptyStateFrame = CGRect(origin: CGPoint(x: 0.0, y: 0.0), size: CGSize(width: availableSize.width, height: availableSize.height)) + let _ = emptyState.update( + transition: emptyStateTransition, + component: AnyComponent(QuickReplyEmptyStateComponent( + theme: environment.theme, + strings: environment.strings, + insets: UIEdgeInsets(top: environment.navigationHeight, left: environment.safeInsets.left, bottom: environment.safeInsets.bottom + environment.additionalInsets.bottom, right: environment.safeInsets.right), + action: { [weak self] in + guard let self else { + return + } + self.openQuickReplyChat(shortcut: nil, shortcutId: nil) + } + )), + environment: {}, + containerSize: emptyStateFrame.size + ) + if let emptyStateView = emptyState.view { + if emptyStateView.superview == nil { + if let navigationBarComponentView = self.navigationBarView.view { + self.insertSubview(emptyStateView, belowSubview: navigationBarComponentView) + } else { + self.addSubview(emptyStateView) + } + } + emptyStateTransition.setFrame(view: emptyStateView, frame: emptyStateFrame) + } + } + + var isModal = false + if let controller = environment.controller(), controller.navigationPresentation == .modal { + isModal = true + } + if case .select = component.mode { + isModal = true + } + + var statusBarHeight = environment.statusBarHeight + if isModal { + statusBarHeight = max(statusBarHeight, 1.0) + } + + var listBottomInset = environment.safeInsets.bottom + environment.additionalInsets.bottom + let navigationHeight = self.updateNavigationBar( + component: component, + theme: environment.theme, + strings: environment.strings, + size: availableSize, + insets: environment.safeInsets, + statusBarHeight: statusBarHeight, + isModal: isModal, + transition: transition, + deferScrollApplication: true + ) + self.navigationHeight = navigationHeight + + var removedSearchBar: SearchBarNode? + if self.isSearchDisplayControllerActive { + let searchBarNode: SearchBarNode + var searchBarTransition = transition + if let current = self.searchBarNode { + searchBarNode = current + } else { + searchBarTransition = .immediate + let searchBarTheme = SearchBarNodeTheme(theme: environment.theme, hasSeparator: false) + searchBarNode = SearchBarNode( + theme: searchBarTheme, + strings: environment.strings, + fieldStyle: .modern, + displayBackground: false + ) + searchBarNode.placeholderString = NSAttributedString(string: environment.strings.Common_Search, font: Font.regular(17.0), textColor: searchBarTheme.placeholder) + self.searchBarNode = searchBarNode + searchBarNode.cancel = { [weak self] in + guard let self else { + return + } + self.isSearchDisplayControllerActive = false + self.state?.updated(transition: .spring(duration: 0.4)) + } + searchBarNode.textUpdated = { [weak self] query, _ in + guard let self else { + return + } + if self.searchQuery != query { + self.searchQuery = query.lowercased().trimmingCharacters(in: .whitespacesAndNewlines) + self.state?.updated(transition: .immediate) + } + } + DispatchQueue.main.async { [weak self, weak searchBarNode] in + guard let self, let searchBarNode, self.searchBarNode === searchBarNode else { + return + } + searchBarNode.activate() + + if let controller = self.environment?.controller() as? QuickReplySetupScreen { + controller.requestAttachmentMenuExpansion() + } + } + } + + var searchBarFrame = CGRect(origin: CGPoint(x: 0.0, y: navigationHeight - 54.0 + 2.0), size: CGSize(width: availableSize.width, height: 54.0)) + if isModal { + searchBarFrame.origin.y += 2.0 + } + searchBarNode.updateLayout(boundingSize: searchBarFrame.size, leftInset: environment.safeInsets.left + 6.0, rightInset: environment.safeInsets.right, transition: searchBarTransition.containedViewLayoutTransition) + searchBarTransition.setFrame(view: searchBarNode.view, frame: searchBarFrame) + if searchBarNode.view.superview == nil { + self.addSubview(searchBarNode.view) + + if case let .curve(duration, curve) = transition.animation, let navigationBarView = self.navigationBarView.view as? ChatListNavigationBar.View, let placeholderNode = navigationBarView.searchContentNode?.placeholderNode { + let timingFunction: String + switch curve { + case .easeInOut: + timingFunction = CAMediaTimingFunctionName.easeOut.rawValue + case .linear: + timingFunction = CAMediaTimingFunctionName.linear.rawValue + case .spring: + timingFunction = kCAMediaTimingFunctionSpring + case .custom: + timingFunction = kCAMediaTimingFunctionSpring + } + + searchBarNode.animateIn(from: placeholderNode, duration: duration, timingFunction: timingFunction) + } + } + } else { + self.searchQuery = "" + if let searchBarNode = self.searchBarNode { + self.searchBarNode = nil + removedSearchBar = searchBarNode + } + } + + if !self.selectedIds.isEmpty { + let selectionPanel: ComponentView + var selectionPanelTransition = transition + if let current = self.selectionPanel { + selectionPanel = current + } else { + selectionPanelTransition = selectionPanelTransition.withAnimation(.none) + selectionPanel = ComponentView() + self.selectionPanel = selectionPanel + } + + let buttonTitle: String = environment.strings.QuickReply_DeleteAction(Int32(self.selectedIds.count)) + + let selectionPanelSize = selectionPanel.update( + transition: selectionPanelTransition, + component: AnyComponent(BottomPanelComponent( + theme: environment.theme, + content: AnyComponentWithIdentity(id: 0, component: AnyComponent(PlainButtonComponent( + content: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: buttonTitle, font: Font.regular(17.0), textColor: environment.theme.list.itemDestructiveColor)) + )), + background: nil, + effectAlignment: .center, + minSize: CGSize(width: availableSize.width - environment.safeInsets.left - environment.safeInsets.right, height: 44.0), + contentInsets: UIEdgeInsets(), + action: { [weak self] in + guard let self else { + return + } + if self.selectedIds.isEmpty { + return + } + self.openDeleteShortcuts(ids: Array(self.selectedIds)) + }, + animateAlpha: true, + animateScale: false, + animateContents: false + ))), + insets: UIEdgeInsets(top: 4.0, left: environment.safeInsets.left, bottom: environment.safeInsets.bottom + environment.additionalInsets.bottom, right: environment.safeInsets.right) + )), + environment: {}, + containerSize: availableSize + ) + let selectionPanelFrame = CGRect(origin: CGPoint(x: 0.0, y: availableSize.height - selectionPanelSize.height), size: selectionPanelSize) + listBottomInset = selectionPanelSize.height + if let selectionPanelView = selectionPanel.view { + var animateIn = false + if selectionPanelView.superview == nil { + animateIn = true + self.addSubview(selectionPanelView) + } + selectionPanelTransition.setFrame(view: selectionPanelView, frame: selectionPanelFrame) + if animateIn { + transition.animatePosition(view: selectionPanelView, from: CGPoint(x: 0.0, y: selectionPanelFrame.height), to: CGPoint(), additive: true) + } + } + } else { + if let selectionPanel = self.selectionPanel { + self.selectionPanel = nil + if let selectionPanelView = selectionPanel.view { + transition.setPosition(view: selectionPanelView, position: CGPoint(x: selectionPanelView.center.x, y: availableSize.height + selectionPanelView.bounds.height * 0.5), completion: { [weak selectionPanelView] _ in + selectionPanelView?.removeFromSuperview() + }) + } + } + } + + let contentListNode: ContentListNode + if let current = self.contentListNode { + contentListNode = current + } else { + contentListNode = ContentListNode(parentView: self, context: component.context) + self.contentListNode = contentListNode + + contentListNode.visibleContentOffsetChanged = { [weak self] offset in + guard let self else { + return + } + guard let navigationHeight = self.navigationHeight else { + return + } + + self.updateNavigationScrolling(navigationHeight: navigationHeight, transition: .immediate) + } + + if let selectionPanelView = self.selectionPanel?.view { + self.insertSubview(contentListNode.view, belowSubview: selectionPanelView) + } else if let navigationBarComponentView = self.navigationBarView.view { + self.insertSubview(contentListNode.view, belowSubview: navigationBarComponentView) + } else { + self.addSubview(contentListNode.view) + } + } + + transition.setFrame(view: contentListNode.view, frame: CGRect(origin: CGPoint(), size: availableSize)) + contentListNode.update(size: availableSize, insets: UIEdgeInsets(top: navigationHeight, left: environment.safeInsets.left, bottom: listBottomInset, right: environment.safeInsets.right), transition: transition) + + var entries: [ContentEntry] = [] + if let shortcutMessageList = self.shortcutMessageList, let accountPeer = self.accountPeer { + switch component.mode { + case .manage: + if self.searchQuery.isEmpty { + entries.append(.add) + } + case .select: + break + } + for item in shortcutMessageList.items { + if !self.searchQuery.isEmpty { + var matches = false + inner: for nameComponent in item.shortcut.lowercased().components(separatedBy: self.searchQueryComponentSeparationCharacterSet) { + if nameComponent.lowercased().hasPrefix(self.searchQuery) { + matches = true + break inner + } + } + if !matches { + continue + } + } + var isItemSelected = false + if let itemId = item.id { + isItemSelected = self.selectedIds.contains(itemId) + } + entries.append(.item(item: item, accountPeer: accountPeer, sortIndex: entries.count, isEditing: self.isEditing, isSelected: isItemSelected)) + } + } + contentListNode.setEntries(entries: entries, animated: !transition.animation.isImmediate) + + if !self.searchQuery.isEmpty && entries.isEmpty { + var emptySearchStateTransition = transition + let emptySearchState: ComponentView + if let current = self.emptySearchState { + emptySearchState = current + } else { + emptySearchStateTransition = emptySearchStateTransition.withAnimation(.none) + emptySearchState = ComponentView() + self.emptySearchState = emptySearchState + } + let emptySearchStateSize = emptySearchState.update( + transition: .immediate, + component: AnyComponent(BalancedTextComponent( + text: .plain(NSAttributedString(string: environment.strings.Conversation_SearchNoResults, font: Font.regular(17.0), textColor: environment.theme.list.freeTextColor, paragraphAlignment: .center)), + horizontalAlignment: .center, + maximumNumberOfLines: 0 + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - 16.0 * 2.0, height: availableSize.height) + ) + var emptySearchStateBottomInset = listBottomInset + emptySearchStateBottomInset = max(emptySearchStateBottomInset, environment.inputHeight) + let emptySearchStateFrame = CGRect(origin: CGPoint(x: floor((availableSize.width - emptySearchStateSize.width) * 0.5), y: navigationHeight + floor((availableSize.height - emptySearchStateBottomInset - navigationHeight) * 0.5)), size: emptySearchStateSize) + if let emptySearchStateView = emptySearchState.view { + if emptySearchStateView.superview == nil { + if let navigationBarComponentView = self.navigationBarView.view { + self.insertSubview(emptySearchStateView, belowSubview: navigationBarComponentView) + } else { + self.addSubview(emptySearchStateView) + } + } + emptySearchStateTransition.containedViewLayoutTransition.updatePosition(layer: emptySearchStateView.layer, position: emptySearchStateFrame.center) + emptySearchStateView.bounds = CGRect(origin: CGPoint(), size: emptySearchStateFrame.size) + } + } else if let emptySearchState = self.emptySearchState { + self.emptySearchState = nil + emptySearchState.view?.removeFromSuperview() + } + + if let shortcutMessageList = self.shortcutMessageList, !shortcutMessageList.items.isEmpty { + contentListNode.isHidden = false + } else { + contentListNode.isHidden = true + } + + self.updateNavigationScrolling(navigationHeight: navigationHeight, transition: transition) + + if let navigationBarComponentView = self.navigationBarView.view as? ChatListNavigationBar.View { + navigationBarComponentView.deferScrollApplication = false + navigationBarComponentView.applyCurrentScroll(transition: transition) + } + + if let removedSearchBar { + if !transition.animation.isImmediate, let navigationBarView = self.navigationBarView.view as? ChatListNavigationBar.View, let placeholderNode = + navigationBarView.searchContentNode?.placeholderNode { + removedSearchBar.transitionOut(to: placeholderNode, transition: transition.containedViewLayoutTransition, completion: { [weak removedSearchBar] in + removedSearchBar?.view.removeFromSuperview() + }) + } else { + removedSearchBar.view.removeFromSuperview() + } + } + + return availableSize + } + } + + func makeView() -> View { + return View() + } + + func update(view: View, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + return view.update(component: self, availableSize: availableSize, state: state, environment: environment, transition: transition) + } +} + +public final class QuickReplySetupScreen: ViewControllerComponentContainer, AttachmentContainable { + public final class InitialData: QuickReplySetupScreenInitialData { + let accountPeer: EnginePeer? + let shortcutMessageList: ShortcutMessageList + + init( + accountPeer: EnginePeer?, + shortcutMessageList: ShortcutMessageList + ) { + self.accountPeer = accountPeer + self.shortcutMessageList = shortcutMessageList + } + } + + public enum Mode { + case manage + case select(completion: (Int32) -> Void) + } + + private let context: AccountContext + + public var requestAttachmentMenuExpansion: () -> Void = { + } + public var updateNavigationStack: (@escaping ([AttachmentContainable]) -> ([AttachmentContainable], AttachmentMediaPickerContext?)) -> Void = { _ in + } + public var updateTabBarAlpha: (CGFloat, ContainedViewLayoutTransition) -> Void = { _, _ in + } + public var cancelPanGesture: () -> Void = { + } + public var isContainerPanning: () -> Bool = { + return false + } + public var isContainerExpanded: () -> Bool = { + return false + } + public var mediaPickerContext: AttachmentMediaPickerContext? + + public init(context: AccountContext, initialData: InitialData, mode: Mode) { + self.context = context + + super.init(context: context, component: QuickReplySetupScreenComponent( + context: context, + initialData: initialData, + mode: mode + ), navigationBarAppearance: .none, theme: .default, updatedPresentationData: nil) + + self.scrollToTop = { [weak self] in + guard let self, let componentView = self.node.hostView.componentView as? QuickReplySetupScreenComponent.View else { + return + } + componentView.scrollToTop() + } + + self.attemptNavigation = { [weak self] complete in + guard let self, let componentView = self.node.hostView.componentView as? QuickReplySetupScreenComponent.View else { + return true + } + + return componentView.attemptNavigation(complete: complete) + } + } + + required public init(coder aDecoder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + deinit { + } + + @objc private func cancelPressed() { + self.dismiss() + } + + override public func containerLayoutUpdated(_ layout: ContainerViewLayout, transition: ContainedViewLayoutTransition) { + super.containerLayoutUpdated(layout, transition: transition) + } + + public static func initialData(context: AccountContext) -> Signal { + return combineLatest( + context.engine.data.get( + TelegramEngine.EngineData.Item.Peer.Peer(id: context.account.peerId) + ), + context.engine.accountData.shortcutMessageList(onlyRemote: false) + |> take(1) + ) + |> map { accountPeer, shortcutMessageList -> QuickReplySetupScreenInitialData in + return InitialData( + accountPeer: accountPeer, + shortcutMessageList: shortcutMessageList + ) + } + } + + public func isContainerPanningUpdated(_ panning: Bool) { + } + + public func resetForReuse() { + } + + public func prepareForReuse() { + } + + public func requestDismiss(completion: @escaping () -> Void) { + completion() + } + + public func shouldDismissImmediately() -> Bool { + return true + } +} diff --git a/submodules/TelegramUI/Components/Settings/BusinessHoursSetupScreen/BUILD b/submodules/TelegramUI/Components/Settings/BusinessHoursSetupScreen/BUILD new file mode 100644 index 00000000000..426f4b106b8 --- /dev/null +++ b/submodules/TelegramUI/Components/Settings/BusinessHoursSetupScreen/BUILD @@ -0,0 +1,43 @@ +load("@build_bazel_rules_swift//swift:swift.bzl", "swift_library") + +swift_library( + name = "BusinessHoursSetupScreen", + module_name = "BusinessHoursSetupScreen", + srcs = glob([ + "Sources/**/*.swift", + ]), + copts = [ + "-warnings-as-errors", + ], + deps = [ + "//submodules/AsyncDisplayKit", + "//submodules/Display", + "//submodules/Postbox", + "//submodules/TelegramCore", + "//submodules/SSignalKit/SwiftSignalKit", + "//submodules/TelegramPresentationData", + "//submodules/TelegramUIPreferences", + "//submodules/AccountContext", + "//submodules/PresentationDataUtils", + "//submodules/Markdown", + "//submodules/ComponentFlow", + "//submodules/Components/ViewControllerComponent", + "//submodules/Components/BundleIconComponent", + "//submodules/Components/MultilineTextComponent", + "//submodules/Components/BalancedTextComponent", + "//submodules/TelegramUI/Components/ListSectionComponent", + "//submodules/TelegramUI/Components/ListActionItemComponent", + "//submodules/TelegramUI/Components/ListTextFieldItemComponent", + "//submodules/TelegramUI/Components/LottieComponent", + "//submodules/TelegramUI/Components/PlainButtonComponent", + "//submodules/LocationUI", + "//submodules/AppBundle", + "//submodules/TelegramStringFormatting", + "//submodules/UIKitRuntimeUtils", + "//submodules/TelegramUI/Components/Settings/TimezoneSelectionScreen", + "//submodules/TelegramUI/Components/TimeSelectionActionSheet", + ], + visibility = [ + "//visibility:public", + ], +) diff --git a/submodules/TelegramUI/Components/Settings/BusinessHoursSetupScreen/Sources/BusinessDaySetupScreen.swift b/submodules/TelegramUI/Components/Settings/BusinessHoursSetupScreen/Sources/BusinessDaySetupScreen.swift new file mode 100644 index 00000000000..e663082df39 --- /dev/null +++ b/submodules/TelegramUI/Components/Settings/BusinessHoursSetupScreen/Sources/BusinessDaySetupScreen.swift @@ -0,0 +1,677 @@ +import Foundation +import UIKit +import Display +import AsyncDisplayKit +import SwiftSignalKit +import Postbox +import TelegramCore +import TelegramPresentationData +import TelegramUIPreferences +import PresentationDataUtils +import AccountContext +import ComponentFlow +import ViewControllerComponent +import MultilineTextComponent +import BalancedTextComponent +import ListSectionComponent +import ListActionItemComponent +import BundleIconComponent +import LottieComponent +import Markdown +import LocationUI +import TelegramStringFormatting +import PlainButtonComponent +import TimeSelectionActionSheet + +func clipMinutes(_ value: Int) -> Int { + return value % (24 * 60) +} + +final class BusinessDaySetupScreenComponent: Component { + typealias EnvironmentType = ViewControllerComponentContainer.Environment + + let context: AccountContext + let dayIndex: Int + let day: BusinessHoursSetupScreenComponent.Day + + init( + context: AccountContext, + dayIndex: Int, + day: BusinessHoursSetupScreenComponent.Day + ) { + self.context = context + self.dayIndex = dayIndex + self.day = day + } + + static func ==(lhs: BusinessDaySetupScreenComponent, rhs: BusinessDaySetupScreenComponent) -> Bool { + if lhs.context !== rhs.context { + return false + } + if lhs.dayIndex != rhs.dayIndex { + return false + } + if lhs.day != rhs.day { + return false + } + + return true + } + + private final class ScrollView: UIScrollView { + override func touchesShouldCancel(in view: UIView) -> Bool { + return true + } + } + + final class View: UIView, UIScrollViewDelegate { + private let topOverscrollLayer = SimpleLayer() + private let scrollView: ScrollView + + private let navigationTitle = ComponentView() + private let generalSection = ComponentView() + private var rangeSections: [Int: ComponentView] = [:] + private let addSection = ComponentView() + + private var isUpdating: Bool = false + + private var component: BusinessDaySetupScreenComponent? + private(set) weak var state: EmptyComponentState? + private var environment: EnvironmentType? + + private(set) var isOpen: Bool = false + private(set) var ranges: [BusinessHoursSetupScreenComponent.WorkingHourRange] = [] + private var intersectingRanges = Set() + private var nextRangeId: Int = 0 + + override init(frame: CGRect) { + self.scrollView = ScrollView() + self.scrollView.showsVerticalScrollIndicator = true + self.scrollView.showsHorizontalScrollIndicator = false + self.scrollView.scrollsToTop = false + self.scrollView.delaysContentTouches = false + self.scrollView.canCancelContentTouches = true + self.scrollView.contentInsetAdjustmentBehavior = .never + if #available(iOS 13.0, *) { + self.scrollView.automaticallyAdjustsScrollIndicatorInsets = false + } + self.scrollView.alwaysBounceVertical = true + + super.init(frame: frame) + + self.scrollView.delegate = self + self.addSubview(self.scrollView) + + self.scrollView.layer.addSublayer(self.topOverscrollLayer) + } + + required init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + deinit { + } + + func scrollToTop() { + self.scrollView.setContentOffset(CGPoint(), animated: true) + } + + func attemptNavigation(complete: @escaping () -> Void) -> Bool { + guard let component = self.component, let enviroment = self.environment else { + return true + } + + if self.isOpen { + if self.intersectingRanges.isEmpty { + return true + } + } + + let presentationData = component.context.sharedContext.currentPresentationData.with { $0 } + self.environment?.controller()?.present(standardTextAlertController(theme: AlertControllerTheme(presentationData: presentationData), title: nil, text: enviroment.strings.BusinessHoursSetup_ErrorIntersectingHours_Text, actions: [ + TextAlertAction(type: .genericAction, title: enviroment.strings.Common_Cancel, action: { + }), + TextAlertAction(type: .defaultAction, title: enviroment.strings.BusinessHoursSetup_ErrorIntersectingHours_ResetAction, action: { + complete() + }) + ]), in: .window(.root)) + + return false + } + + func scrollViewDidScroll(_ scrollView: UIScrollView) { + self.updateScrolling(transition: .immediate) + } + + var scrolledUp = true + private func updateScrolling(transition: Transition) { + let navigationRevealOffsetY: CGFloat = 0.0 + + let navigationAlphaDistance: CGFloat = 16.0 + let navigationAlpha: CGFloat = max(0.0, min(1.0, (self.scrollView.contentOffset.y - navigationRevealOffsetY) / navigationAlphaDistance)) + if let controller = self.environment?.controller(), let navigationBar = controller.navigationBar { + transition.setAlpha(layer: navigationBar.backgroundNode.layer, alpha: navigationAlpha) + transition.setAlpha(layer: navigationBar.stripeNode.layer, alpha: navigationAlpha) + } + + var scrolledUp = false + if navigationAlpha < 0.5 { + scrolledUp = true + } else if navigationAlpha > 0.5 { + scrolledUp = false + } + + if self.scrolledUp != scrolledUp { + self.scrolledUp = scrolledUp + if !self.isUpdating { + self.state?.updated() + } + } + + if let navigationTitleView = self.navigationTitle.view { + transition.setAlpha(view: navigationTitleView, alpha: 1.0) + } + } + + private func openRangeDateSetup(rangeId: Int, isStartTime: Bool) { + guard let component = self.component else { + return + } + guard let range = self.ranges.first(where: { $0.id == rangeId }) else { + return + } + + let controller = TimeSelectionActionSheet(context: component.context, currentValue: Int32(isStartTime ? clipMinutes(range.startMinute) : clipMinutes(range.endMinute)) * 60, applyValue: { [weak self] value in + guard let self else { + return + } + guard let value else { + return + } + if let index = self.ranges.firstIndex(where: { $0.id == rangeId }) { + var startMinute = range.startMinute + var endMinute = range.endMinute + if isStartTime { + startMinute = Int(value) / 60 + } else { + endMinute = Int(value) / 60 + } + if endMinute < startMinute { + endMinute = endMinute + 24 * 60 + } + self.ranges[index].startMinute = startMinute + self.ranges[index].endMinute = endMinute + self.validateRanges() + + self.state?.updated(transition: .immediate) + } + }) + self.environment?.controller()?.present(controller, in: .window(.root)) + } + + private func validateRanges() { + self.ranges.sort(by: { $0.startMinute < $1.startMinute }) + + self.intersectingRanges.removeAll() + for i in 0 ..< self.ranges.count { + var minuteSet = IndexSet() + inner: for j in 0 ..< self.ranges.count { + if i == j { + continue inner + } + let range = self.ranges[j] + let rangeMinutes = range.startMinute ..< range.endMinute + minuteSet.insert(integersIn: rangeMinutes) + } + + let range = self.ranges[i] + let rangeMinutes = range.startMinute ..< range.endMinute + + if minuteSet.intersects(integersIn: rangeMinutes) { + self.intersectingRanges.insert(range.id) + } + } + } + + func update(component: BusinessDaySetupScreenComponent, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + self.isUpdating = true + defer { + self.isUpdating = false + } + + let environment = environment[EnvironmentType.self].value + let themeUpdated = self.environment?.theme !== environment.theme + self.environment = environment + + if self.component == nil { + self.isOpen = component.day.ranges != nil + self.ranges = component.day.ranges ?? [] + self.nextRangeId = (self.ranges.map(\.id).max() ?? 0) + 1 + self.validateRanges() + } + + self.component = component + self.state = state + + if themeUpdated { + self.backgroundColor = environment.theme.list.blocksBackgroundColor + } + + let presentationData = component.context.sharedContext.currentPresentationData.with { $0 } + + let title: String + switch component.dayIndex { + case 0: + title = environment.strings.Weekday_Monday + case 1: + title = environment.strings.Weekday_Tuesday + case 2: + title = environment.strings.Weekday_Wednesday + case 3: + title = environment.strings.Weekday_Thursday + case 4: + title = environment.strings.Weekday_Friday + case 5: + title = environment.strings.Weekday_Saturday + case 6: + title = environment.strings.Weekday_Sunday + default: + title = " " + } + let navigationTitleSize = self.navigationTitle.update( + transition: transition, + component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: title, font: Font.semibold(17.0), textColor: environment.theme.rootController.navigationBar.primaryTextColor)), + horizontalAlignment: .center + )), + environment: {}, + containerSize: CGSize(width: availableSize.width, height: 100.0) + ) + let navigationTitleFrame = CGRect(origin: CGPoint(x: floor((availableSize.width - navigationTitleSize.width) / 2.0), y: environment.statusBarHeight + floor((environment.navigationHeight - environment.statusBarHeight - navigationTitleSize.height) / 2.0)), size: navigationTitleSize) + if let navigationTitleView = self.navigationTitle.view { + if navigationTitleView.superview == nil { + if let controller = self.environment?.controller(), let navigationBar = controller.navigationBar { + navigationBar.view.addSubview(navigationTitleView) + } + } + transition.setFrame(view: navigationTitleView, frame: navigationTitleFrame) + } + + let bottomContentInset: CGFloat = 24.0 + let sideInset: CGFloat = 16.0 + environment.safeInsets.left + let sectionSpacing: CGFloat = 24.0 + + let _ = bottomContentInset + let _ = sectionSpacing + + var contentHeight: CGFloat = 0.0 + + contentHeight += environment.navigationHeight + contentHeight += 16.0 + + let generalSectionSize = self.generalSection.update( + transition: transition, + component: AnyComponent(ListSectionComponent( + theme: environment.theme, + header: nil, + footer: nil, + items: [ + AnyComponentWithIdentity(id: 0, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: environment.strings.BusinessHoursSetup_DaySwitch, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemPrimaryTextColor + )), + maximumNumberOfLines: 1 + ))), + ], alignment: .left, spacing: 2.0)), + accessory: .toggle(ListActionItemComponent.Toggle(style: .regular, isOn: self.isOpen, action: { [weak self] _ in + guard let self else { + return + } + self.isOpen = !self.isOpen + self.state?.updated(transition: .spring(duration: 0.4)) + })), + action: nil + ))) + ] + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 10000.0) + ) + let generalSectionFrame = CGRect(origin: CGPoint(x: sideInset, y: contentHeight), size: generalSectionSize) + if let generalSectionView = self.generalSection.view { + if generalSectionView.superview == nil { + self.scrollView.addSubview(generalSectionView) + } + transition.setFrame(view: generalSectionView, frame: generalSectionFrame) + } + contentHeight += generalSectionSize.height + contentHeight += sectionSpacing + + var rangesSectionsHeight: CGFloat = 0.0 + for range in self.ranges { + let rangeId = range.id + var rangeSectionTransition = transition + let rangeSection: ComponentView + if let current = self.rangeSections[range.id] { + rangeSection = current + } else { + rangeSection = ComponentView() + self.rangeSections[range.id] = rangeSection + rangeSectionTransition = rangeSectionTransition.withAnimation(.none) + } + + let startHours = clipMinutes(range.startMinute) / 60 + let startMinutes = clipMinutes(range.startMinute) % 60 + let startText = stringForShortTimestamp(hours: Int32(startHours), minutes: Int32(startMinutes), dateTimeFormat: PresentationDateTimeFormat()) + let endHours = clipMinutes(range.endMinute) / 60 + let endMinutes = clipMinutes(range.endMinute) % 60 + let endText = stringForShortTimestamp(hours: Int32(endHours), minutes: Int32(endMinutes), dateTimeFormat: PresentationDateTimeFormat()) + + var rangeSectionItems: [AnyComponentWithIdentity] = [] + for i in 0 ..< 2 { + let isOpenTime = i == 0 + rangeSectionItems.append(AnyComponentWithIdentity(id: rangeSectionItems.count, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: isOpenTime ? environment.strings.BusinessHoursSetup_DayIntervalStart : environment.strings.BusinessHoursSetup_DayIntervalEnd, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemPrimaryTextColor + )), + maximumNumberOfLines: 1 + ))), + ], alignment: .left, spacing: 2.0)), + icon: ListActionItemComponent.Icon(component: AnyComponentWithIdentity(id: 0, component: AnyComponent(PlainButtonComponent( + content: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: isOpenTime ? startText : endText, font: Font.regular(17.0), textColor: self.intersectingRanges.contains(range.id) ? environment.theme.list.itemDestructiveColor : environment.theme.list.itemPrimaryTextColor)) + )), + background: AnyComponent(RoundedRectangle(color: environment.theme.list.itemPrimaryTextColor.withMultipliedAlpha(0.1), cornerRadius: 6.0)), + effectAlignment: .center, + minSize: nil, + contentInsets: UIEdgeInsets(top: 7.0, left: 8.0, bottom: 7.0, right: 8.0), + action: { [weak self] in + guard let self else { + return + } + self.openRangeDateSetup(rangeId: rangeId, isStartTime: isOpenTime) + }, + animateAlpha: true, + animateScale: false + ))), insets: .custom(UIEdgeInsets(top: 4.0, left: 0.0, bottom: 4.0, right: 0.0)), allowUserInteraction: true), + accessory: nil, + action: { [weak self] _ in + guard let self else { + return + } + self.openRangeDateSetup(rangeId: rangeId, isStartTime: isOpenTime) + } + )))) + } + + rangeSectionItems.append(AnyComponentWithIdentity(id: rangeSectionItems.count, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: environment.strings.BusinessHoursSetup_DayIntervalRemove, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemDestructiveColor + )), + maximumNumberOfLines: 1 + ))), + ], alignment: .left, spacing: 2.0)), + accessory: nil, + action: { [weak self] _ in + guard let self else { + return + } + self.ranges.removeAll(where: { $0.id == rangeId }) + self.validateRanges() + self.state?.updated(transition: .spring(duration: 0.4)) + } + )))) + + let rangeSectionSize = rangeSection.update( + transition: rangeSectionTransition, + component: AnyComponent(ListSectionComponent( + theme: environment.theme, + header: nil, + footer: nil, + items: rangeSectionItems + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 10000.0) + ) + let rangeSectionFrame = CGRect(origin: CGPoint(x: sideInset, y: contentHeight + rangesSectionsHeight), size: rangeSectionSize) + if let rangeSectionView = rangeSection.view { + var animateIn = false + if rangeSectionView.superview == nil { + animateIn = true + rangeSectionView.layer.allowsGroupOpacity = true + self.scrollView.addSubview(rangeSectionView) + } + rangeSectionTransition.setFrame(view: rangeSectionView, frame: rangeSectionFrame) + + let alphaTransition = transition.animation.isImmediate ? transition : .easeInOut(duration: 0.25) + if self.isOpen { + if animateIn { + if !transition.animation.isImmediate { + alphaTransition.animateAlpha(view: rangeSectionView, from: 0.0, to: 1.0) + transition.animateScale(view: rangeSectionView, from: 0.001, to: 1.0) + } + } else { + alphaTransition.setAlpha(view: rangeSectionView, alpha: 1.0) + } + } else { + alphaTransition.setAlpha(view: rangeSectionView, alpha: 0.0) + } + } + + rangesSectionsHeight += rangeSectionSize.height + rangesSectionsHeight += sectionSpacing + } + var removeRangeSectionIds: [Int] = [] + for (id, rangeSection) in self.rangeSections { + if !self.ranges.contains(where: { $0.id == id }) { + removeRangeSectionIds.append(id) + + if let rangeSectionView = rangeSection.view { + if !transition.animation.isImmediate { + Transition.easeInOut(duration: 0.2).setAlpha(view: rangeSectionView, alpha: 0.0, completion: { [weak rangeSectionView] _ in + rangeSectionView?.removeFromSuperview() + }) + transition.setScale(view: rangeSectionView, scale: 0.001) + } else { + rangeSectionView.removeFromSuperview() + } + } + } + } + for id in removeRangeSectionIds { + self.rangeSections.removeValue(forKey: id) + } + + var canAddRanges = true + if let lastRange = self.ranges.last, lastRange.endMinute >= 24 * 60 * 2 - 1 { + canAddRanges = false + } + + let addSectionSize = self.addSection.update( + transition: transition, + component: AnyComponent(ListSectionComponent( + theme: environment.theme, + header: nil, + footer: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: environment.strings.BusinessHoursSetup_AddSectionFooter, + font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), + textColor: environment.theme.list.freeTextColor + )), + maximumNumberOfLines: 0 + )), + items: [ + AnyComponentWithIdentity(id: 0, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: environment.strings.BusinessHoursSetup_AddAction, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemAccentColor + )), + maximumNumberOfLines: 1 + ))), + ], alignment: .left, spacing: 2.0)), + leftIcon: AnyComponentWithIdentity(id: 0, component: AnyComponent(BundleIconComponent( + name: "Item List/AddTimeIcon", + tintColor: environment.theme.list.itemAccentColor + ))), + accessory: nil, + action: { [weak self] _ in + guard let self else { + return + } + + var rangeStart = 9 * 60 + if let lastRange = self.ranges.last { + rangeStart = lastRange.endMinute + 1 + } + if rangeStart >= 2 * 24 * 60 - 1 { + return + } + let rangeEnd = min(rangeStart + 9 * 60, 2 * 24 * 60) + + let rangeId = self.nextRangeId + self.nextRangeId += 1 + + self.ranges.append(BusinessHoursSetupScreenComponent.WorkingHourRange( + id: rangeId, startMinute: rangeStart, endMinute: rangeEnd)) + self.validateRanges() + self.state?.updated(transition: .spring(duration: 0.4)) + } + ))) + ] + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 10000.0) + ) + let addSectionFrame = CGRect(origin: CGPoint(x: sideInset, y: contentHeight + rangesSectionsHeight), size: addSectionSize) + if let addSectionView = self.addSection.view { + if addSectionView.superview == nil { + self.scrollView.addSubview(addSectionView) + } + transition.setFrame(view: addSectionView, frame: addSectionFrame) + + let alphaTransition = transition.animation.isImmediate ? transition : .easeInOut(duration: 0.25) + alphaTransition.setAlpha(view: addSectionView, alpha: (self.isOpen && canAddRanges) ? 1.0 : 0.0) + } + if canAddRanges { + rangesSectionsHeight += addSectionSize.height + } + + if self.isOpen { + contentHeight += rangesSectionsHeight + } + + contentHeight += bottomContentInset + contentHeight += environment.safeInsets.bottom + + let previousBounds = self.scrollView.bounds + + let contentSize = CGSize(width: availableSize.width, height: contentHeight) + if self.scrollView.frame != CGRect(origin: CGPoint(), size: availableSize) { + self.scrollView.frame = CGRect(origin: CGPoint(), size: availableSize) + } + if self.scrollView.contentSize != contentSize { + self.scrollView.contentSize = contentSize + } + let scrollInsets = UIEdgeInsets(top: environment.navigationHeight, left: 0.0, bottom: 0.0, right: 0.0) + if self.scrollView.scrollIndicatorInsets != scrollInsets { + self.scrollView.scrollIndicatorInsets = scrollInsets + } + + if !previousBounds.isEmpty, !transition.animation.isImmediate { + let bounds = self.scrollView.bounds + if bounds.maxY != previousBounds.maxY { + let offsetY = previousBounds.maxY - bounds.maxY + transition.animateBoundsOrigin(view: self.scrollView, from: CGPoint(x: 0.0, y: offsetY), to: CGPoint(), additive: true) + } + } + + self.topOverscrollLayer.frame = CGRect(origin: CGPoint(x: 0.0, y: -3000.0), size: CGSize(width: availableSize.width, height: 3000.0)) + + self.updateScrolling(transition: transition) + + return availableSize + } + } + + func makeView() -> View { + return View() + } + + func update(view: View, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + return view.update(component: self, availableSize: availableSize, state: state, environment: environment, transition: transition) + } +} + +final class BusinessDaySetupScreen: ViewControllerComponentContainer { + private let context: AccountContext + fileprivate let updateDay: (BusinessHoursSetupScreenComponent.Day) -> Void + + init(context: AccountContext, dayIndex: Int, day: BusinessHoursSetupScreenComponent.Day, updateDay: @escaping (BusinessHoursSetupScreenComponent.Day) -> Void) { + self.context = context + self.updateDay = updateDay + + super.init(context: context, component: BusinessDaySetupScreenComponent( + context: context, + dayIndex: dayIndex, + day: day + ), navigationBarAppearance: .default, theme: .default, updatedPresentationData: nil) + + let presentationData = context.sharedContext.currentPresentationData.with { $0 } + self.title = "" + self.navigationItem.backBarButtonItem = UIBarButtonItem(title: presentationData.strings.Common_Back, style: .plain, target: nil, action: nil) + + self.scrollToTop = { [weak self] in + guard let self, let componentView = self.node.hostView.componentView as? BusinessDaySetupScreenComponent.View else { + return + } + componentView.scrollToTop() + } + + self.attemptNavigation = { [weak self] complete in + guard let self, let componentView = self.node.hostView.componentView as? BusinessDaySetupScreenComponent.View else { + return true + } + + if componentView.attemptNavigation(complete: complete) { + self.updateDay(BusinessHoursSetupScreenComponent.Day(ranges: componentView.isOpen ? componentView.ranges : nil)) + return true + } else { + return false + } + } + } + + required public init(coder aDecoder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + deinit { + } + + @objc private func cancelPressed() { + self.dismiss() + } + + override public func containerLayoutUpdated(_ layout: ContainerViewLayout, transition: ContainedViewLayoutTransition) { + super.containerLayoutUpdated(layout, transition: transition) + } +} diff --git a/submodules/TelegramUI/Components/Settings/BusinessHoursSetupScreen/Sources/BusinessHoursSetupScreen.swift b/submodules/TelegramUI/Components/Settings/BusinessHoursSetupScreen/Sources/BusinessHoursSetupScreen.swift new file mode 100644 index 00000000000..7166538490d --- /dev/null +++ b/submodules/TelegramUI/Components/Settings/BusinessHoursSetupScreen/Sources/BusinessHoursSetupScreen.swift @@ -0,0 +1,871 @@ +import Foundation +import UIKit +import Display +import AsyncDisplayKit +import SwiftSignalKit +import Postbox +import TelegramCore +import TelegramPresentationData +import TelegramUIPreferences +import PresentationDataUtils +import AccountContext +import ComponentFlow +import ViewControllerComponent +import MultilineTextComponent +import BalancedTextComponent +import ListSectionComponent +import ListActionItemComponent +import BundleIconComponent +import LottieComponent +import Markdown +import LocationUI +import TelegramStringFormatting +import TimezoneSelectionScreen + +private func wrappedMinuteRange(range: Range, dayIndexOffset: Int = 0) -> IndexSet { + let mappedRange = (range.lowerBound + dayIndexOffset * 24 * 60) ..< (range.upperBound + dayIndexOffset * 24 * 60) + + var result = IndexSet() + if mappedRange.upperBound > 7 * 24 * 60 { + if mappedRange.lowerBound < 7 * 24 * 60 { + result.insert(integersIn: mappedRange.lowerBound ..< 7 * 24 * 60) + } + result.insert(integersIn: 0 ..< (mappedRange.upperBound - 7 * 24 * 60)) + } else { + result.insert(integersIn: mappedRange) + } + return result +} + +private func getDayRanges(days: [BusinessHoursSetupScreenComponent.Day], index: Int) -> [BusinessHoursSetupScreenComponent.WorkingHourRange] { + let day = days[index] + if let ranges = day.ranges { + if ranges.isEmpty { + return [BusinessHoursSetupScreenComponent.WorkingHourRange(id: 0, startMinute: 0, endMinute: 24 * 60)] + } else { + return ranges + } + } else { + return [] + } +} + +final class BusinessHoursSetupScreenComponent: Component { + typealias EnvironmentType = ViewControllerComponentContainer.Environment + + let context: AccountContext + let initialValue: TelegramBusinessHours? + let completion: (TelegramBusinessHours?) -> Void + + init( + context: AccountContext, + initialValue: TelegramBusinessHours?, + completion: @escaping (TelegramBusinessHours?) -> Void + ) { + self.context = context + self.initialValue = initialValue + self.completion = completion + } + + static func ==(lhs: BusinessHoursSetupScreenComponent, rhs: BusinessHoursSetupScreenComponent) -> Bool { + if lhs.context !== rhs.context { + return false + } + if lhs.initialValue != rhs.initialValue { + return false + } + + return true + } + + private final class ScrollView: UIScrollView { + override func touchesShouldCancel(in view: UIView) -> Bool { + return true + } + } + + struct WorkingHourRange: Equatable { + var id: Int + var startMinute: Int + var endMinute: Int + + init(id: Int, startMinute: Int, endMinute: Int) { + self.id = id + self.startMinute = startMinute + self.endMinute = endMinute + } + } + + struct DayRangeIndex: Hashable { + var day: Int + var id: Int + + init(day: Int, id: Int) { + self.day = day + self.id = id + } + } + + struct Day: Equatable { + var ranges: [WorkingHourRange]? + + init(ranges: [WorkingHourRange]?) { + self.ranges = ranges + } + } + + struct DaysState: Equatable { + enum ValidationError: Error { + case intersectingRanges + } + + var timezoneId: String + private(set) var days: [Day] + private(set) var intersectingRanges = Set() + + init(timezoneId: String, days: [Day]) { + self.timezoneId = timezoneId + self.days = days + + self.validate() + } + + init(businessHours: TelegramBusinessHours) { + self.timezoneId = businessHours.timezoneId + + self.days = businessHours.splitIntoWeekDays().map { day in + switch day { + case .closed: + return Day(ranges: nil) + case .open: + return Day(ranges: []) + case let .intervals(intervals): + var nextIntervalId = 0 + return Day(ranges: intervals.map { interval in + let intervalId = nextIntervalId + nextIntervalId += 1 + return WorkingHourRange(id: intervalId, startMinute: interval.startMinute, endMinute: interval.endMinute) + }) + } + } + + if let value = try? self.asBusinessHours() { + if value != businessHours { + assertionFailure("Inconsistent representation") + } + } + + self.validate() + } + + mutating func validate() { + self.intersectingRanges.removeAll() + + for dayIndex in 0 ..< self.days.count { + var otherDaysMinutes = IndexSet() + inner: for otherDayIndex in 0 ..< self.days.count { + if dayIndex == otherDayIndex { + continue inner + } + for range in getDayRanges(days: self.days, index: otherDayIndex) { + otherDaysMinutes.formUnion(wrappedMinuteRange(range: range.startMinute ..< range.endMinute, dayIndexOffset: otherDayIndex)) + } + } + + let dayRanges = getDayRanges(days: self.days, index: dayIndex) + for i in 0 ..< dayRanges.count { + var currentDayOtherMinutes = IndexSet() + inner: for j in 0 ..< dayRanges.count { + if i == j { + continue inner + } + currentDayOtherMinutes.formUnion(wrappedMinuteRange(range: dayRanges[j].startMinute ..< dayRanges[j].endMinute, dayIndexOffset: dayIndex)) + } + + let currentDayIndices = wrappedMinuteRange(range: dayRanges[i].startMinute ..< dayRanges[i].endMinute, dayIndexOffset: dayIndex) + if !otherDaysMinutes.intersection(currentDayIndices).isEmpty || !currentDayOtherMinutes.intersection(currentDayIndices).isEmpty { + self.intersectingRanges.insert(DayRangeIndex(day: dayIndex, id: dayRanges[i].id)) + } + } + } + } + + mutating func update(days: [Day]) { + self.days = days + self.validate() + } + + func asBusinessHours() throws -> TelegramBusinessHours { + var mappedIntervals: [TelegramBusinessHours.WorkingTimeInterval] = [] + + var filledMinutes = IndexSet() + for i in 0 ..< self.days.count { + let dayStartMinute = i * 24 * 60 + guard var effectiveRanges = self.days[i].ranges else { + continue + } + if effectiveRanges.isEmpty { + effectiveRanges = [WorkingHourRange(id: 0, startMinute: 0, endMinute: 24 * 60)] + } + for range in effectiveRanges { + let minuteRange: Range = (dayStartMinute + range.startMinute) ..< (dayStartMinute + range.endMinute) + + let wrappedMinutes = wrappedMinuteRange(range: minuteRange) + + if !filledMinutes.intersection(wrappedMinutes).isEmpty { + throw ValidationError.intersectingRanges + } + filledMinutes.formUnion(wrappedMinutes) + mappedIntervals.append(TelegramBusinessHours.WorkingTimeInterval(startMinute: minuteRange.lowerBound, endMinute: minuteRange.upperBound)) + } + } + + var mergedIntervals: [TelegramBusinessHours.WorkingTimeInterval] = [] + for interval in mappedIntervals { + if mergedIntervals.isEmpty { + mergedIntervals.append(interval) + } else { + if mergedIntervals[mergedIntervals.count - 1].endMinute >= interval.startMinute { + mergedIntervals[mergedIntervals.count - 1] = TelegramBusinessHours.WorkingTimeInterval(startMinute: mergedIntervals[mergedIntervals.count - 1].startMinute, endMinute: interval.endMinute) + } else { + mergedIntervals.append(interval) + } + } + } + + return TelegramBusinessHours(timezoneId: self.timezoneId, weeklyTimeIntervals: mergedIntervals) + } + } + + final class View: UIView, UIScrollViewDelegate { + private let topOverscrollLayer = SimpleLayer() + private let scrollView: ScrollView + + private let navigationTitle = ComponentView() + private let icon = ComponentView() + private let subtitle = ComponentView() + private let generalSection = ComponentView() + private let daysSection = ComponentView() + private let timezoneSection = ComponentView() + + private var ignoreScrolling: Bool = false + private var isUpdating: Bool = false + + private var component: BusinessHoursSetupScreenComponent? + private(set) weak var state: EmptyComponentState? + private var environment: EnvironmentType? + + private var showHours: Bool = false + private var daysState = DaysState(timezoneId: "", days: []) + + private var timeZoneList: TimeZoneList? + private var timezonesDisposable: Disposable? + private var keepTimezonesUpdatedDisposable: Disposable? + + override init(frame: CGRect) { + self.scrollView = ScrollView() + self.scrollView.showsVerticalScrollIndicator = true + self.scrollView.showsHorizontalScrollIndicator = false + self.scrollView.scrollsToTop = false + self.scrollView.delaysContentTouches = false + self.scrollView.canCancelContentTouches = true + self.scrollView.contentInsetAdjustmentBehavior = .never + if #available(iOS 13.0, *) { + self.scrollView.automaticallyAdjustsScrollIndicatorInsets = false + } + self.scrollView.alwaysBounceVertical = true + + super.init(frame: frame) + + self.scrollView.delegate = self + self.addSubview(self.scrollView) + + self.scrollView.layer.addSublayer(self.topOverscrollLayer) + } + + required init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + deinit { + self.timezonesDisposable?.dispose() + self.keepTimezonesUpdatedDisposable?.dispose() + } + + func scrollToTop() { + self.scrollView.setContentOffset(CGPoint(), animated: true) + } + + func attemptNavigation(complete: @escaping () -> Void) -> Bool { + guard let component = self.component, let environment = self.environment else { + return true + } + + if self.showHours { + do { + let businessHours = try self.daysState.asBusinessHours() + let _ = component.context.engine.accountData.updateAccountBusinessHours(businessHours: businessHours).startStandalone() + return true + } catch _ { + let presentationData = component.context.sharedContext.currentPresentationData.with { $0 } + self.environment?.controller()?.present(standardTextAlertController(theme: AlertControllerTheme(presentationData: presentationData), title: nil, text: environment.strings.BusinessHoursSetup_ErrorIntersectingDays_Text, actions: [ + TextAlertAction(type: .genericAction, title: environment.strings.Common_Cancel, action: { + }), + TextAlertAction(type: .defaultAction, title: environment.strings.BusinessHoursSetup_ErrorIntersectingDays_ResetAction, action: { [weak self] in + guard let self else { + return + } + let _ = self + complete() + }) + ]), in: .window(.root)) + + return false + } + } else { + if component.initialValue != nil { + let _ = component.context.engine.accountData.updateAccountBusinessHours(businessHours: nil).startStandalone() + return true + } else { + return true + } + } + } + + func scrollViewDidScroll(_ scrollView: UIScrollView) { + if !self.ignoreScrolling { + self.updateScrolling(transition: .immediate) + } + } + + var scrolledUp = true + private func updateScrolling(transition: Transition) { + let navigationRevealOffsetY: CGFloat = 0.0 + + let navigationAlphaDistance: CGFloat = 16.0 + let navigationAlpha: CGFloat = max(0.0, min(1.0, (self.scrollView.contentOffset.y - navigationRevealOffsetY) / navigationAlphaDistance)) + if let controller = self.environment?.controller(), let navigationBar = controller.navigationBar { + transition.setAlpha(layer: navigationBar.backgroundNode.layer, alpha: navigationAlpha) + transition.setAlpha(layer: navigationBar.stripeNode.layer, alpha: navigationAlpha) + } + + var scrolledUp = false + if navigationAlpha < 0.5 { + scrolledUp = true + } else if navigationAlpha > 0.5 { + scrolledUp = false + } + + if self.scrolledUp != scrolledUp { + self.scrolledUp = scrolledUp + if !self.isUpdating { + self.state?.updated() + } + } + + if let navigationTitleView = self.navigationTitle.view { + transition.setAlpha(view: navigationTitleView, alpha: 1.0) + } + } + + func update(component: BusinessHoursSetupScreenComponent, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + self.isUpdating = true + defer { + self.isUpdating = false + } + + if self.component == nil { + if let initialValue = component.initialValue { + self.showHours = true + self.daysState = DaysState(businessHours: initialValue) + } else { + self.showHours = false + self.daysState.timezoneId = TimeZone.current.identifier + self.daysState.update(days: (0 ..< 7).map { _ in + return Day(ranges: []) + }) + } + + self.timezonesDisposable = (component.context.engine.accountData.cachedTimeZoneList() + |> deliverOnMainQueue).start(next: { [weak self] timeZoneList in + guard let self else { + return + } + self.timeZoneList = timeZoneList + self.state?.updated(transition: .immediate) + }) + self.keepTimezonesUpdatedDisposable = component.context.engine.accountData.keepCachedTimeZoneListUpdated().startStrict() + } + + let environment = environment[EnvironmentType.self].value + let themeUpdated = self.environment?.theme !== environment.theme + self.environment = environment + + self.component = component + self.state = state + + if themeUpdated { + self.backgroundColor = environment.theme.list.blocksBackgroundColor + } + + let presentationData = component.context.sharedContext.currentPresentationData.with { $0 } + + let navigationTitleSize = self.navigationTitle.update( + transition: transition, + component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: environment.strings.BusinessHoursSetup_Title, font: Font.semibold(17.0), textColor: environment.theme.rootController.navigationBar.primaryTextColor)), + horizontalAlignment: .center + )), + environment: {}, + containerSize: CGSize(width: availableSize.width, height: 100.0) + ) + let navigationTitleFrame = CGRect(origin: CGPoint(x: floor((availableSize.width - navigationTitleSize.width) / 2.0), y: environment.statusBarHeight + floor((environment.navigationHeight - environment.statusBarHeight - navigationTitleSize.height) / 2.0)), size: navigationTitleSize) + if let navigationTitleView = self.navigationTitle.view { + if navigationTitleView.superview == nil { + if let controller = self.environment?.controller(), let navigationBar = controller.navigationBar { + navigationBar.view.addSubview(navigationTitleView) + } + } + transition.setFrame(view: navigationTitleView, frame: navigationTitleFrame) + } + + let bottomContentInset: CGFloat = 24.0 + let sideInset: CGFloat = 16.0 + environment.safeInsets.left + let sectionSpacing: CGFloat = 30.0 + + let _ = bottomContentInset + let _ = sectionSpacing + + var contentHeight: CGFloat = 0.0 + + contentHeight += environment.navigationHeight + + let iconSize = self.icon.update( + transition: .immediate, + component: AnyComponent(LottieComponent( + content: LottieComponent.AppBundleContent(name: "BusinessHoursEmoji"), + loop: false + )), + environment: {}, + containerSize: CGSize(width: 100.0, height: 100.0) + ) + let iconFrame = CGRect(origin: CGPoint(x: floor((availableSize.width - iconSize.width) * 0.5), y: contentHeight + 10.0), size: iconSize) + if let iconView = self.icon.view as? LottieComponent.View { + if iconView.superview == nil { + self.scrollView.addSubview(iconView) + iconView.playOnce() + } + transition.setPosition(view: iconView, position: iconFrame.center) + iconView.bounds = CGRect(origin: CGPoint(), size: iconFrame.size) + } + + contentHeight += 126.0 + + let subtitleString = NSMutableAttributedString(attributedString: parseMarkdownIntoAttributedString(environment.strings.BusinessHoursSetup_Text, attributes: MarkdownAttributes( + body: MarkdownAttributeSet(font: Font.regular(15.0), textColor: environment.theme.list.freeTextColor), + bold: MarkdownAttributeSet(font: Font.semibold(15.0), textColor: environment.theme.list.freeTextColor), + link: MarkdownAttributeSet(font: Font.regular(15.0), textColor: environment.theme.list.itemAccentColor), + linkAttribute: { attributes in + return ("URL", "") + }), textAlignment: .center + )) + + let subtitleSize = self.subtitle.update( + transition: .immediate, + component: AnyComponent(BalancedTextComponent( + text: .plain(subtitleString), + horizontalAlignment: .center, + maximumNumberOfLines: 0, + lineSpacing: 0.25, + highlightColor: environment.theme.list.itemAccentColor.withMultipliedAlpha(0.1), + highlightAction: { attributes in + if let _ = attributes[NSAttributedString.Key(rawValue: "URL")] { + return NSAttributedString.Key(rawValue: "URL") + } else { + return nil + } + }, + tapAction: { [weak self] _, _ in + guard let self, let component = self.component else { + return + } + let _ = component + } + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 1000.0) + ) + let subtitleFrame = CGRect(origin: CGPoint(x: floor((availableSize.width - subtitleSize.width) * 0.5), y: contentHeight), size: subtitleSize) + if let subtitleView = self.subtitle.view { + if subtitleView.superview == nil { + self.scrollView.addSubview(subtitleView) + } + transition.setPosition(view: subtitleView, position: subtitleFrame.center) + subtitleView.bounds = CGRect(origin: CGPoint(), size: subtitleFrame.size) + } + contentHeight += subtitleSize.height + contentHeight += 27.0 + + let generalSectionSize = self.generalSection.update( + transition: transition, + component: AnyComponent(ListSectionComponent( + theme: environment.theme, + header: nil, + footer: nil, + items: [ + AnyComponentWithIdentity(id: 0, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: environment.strings.BusinessHoursSetup_MainToggle, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemPrimaryTextColor + )), + maximumNumberOfLines: 1 + ))), + ], alignment: .left, spacing: 2.0)), + accessory: .toggle(ListActionItemComponent.Toggle(style: .regular, isOn: self.showHours, action: { [weak self] _ in + guard let self else { + return + } + self.showHours = !self.showHours + self.state?.updated(transition: .spring(duration: 0.4)) + })), + action: nil + ))) + ] + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 10000.0) + ) + let generalSectionFrame = CGRect(origin: CGPoint(x: sideInset, y: contentHeight), size: generalSectionSize) + if let generalSectionView = self.generalSection.view { + if generalSectionView.superview == nil { + self.scrollView.addSubview(generalSectionView) + } + transition.setFrame(view: generalSectionView, frame: generalSectionFrame) + } + contentHeight += generalSectionSize.height + contentHeight += sectionSpacing + + var daysContentHeight: CGFloat = 0.0 + + var daysSectionItems: [AnyComponentWithIdentity] = [] + for day in self.daysState.days { + let dayIndex = daysSectionItems.count + + let title: String + switch dayIndex { + case 0: + title = environment.strings.Weekday_Monday + case 1: + title = environment.strings.Weekday_Tuesday + case 2: + title = environment.strings.Weekday_Wednesday + case 3: + title = environment.strings.Weekday_Thursday + case 4: + title = environment.strings.Weekday_Friday + case 5: + title = environment.strings.Weekday_Saturday + case 6: + title = environment.strings.Weekday_Sunday + default: + title = " " + } + + let subtitle = NSMutableAttributedString() + + var invalidIndices: [Int] = [] + let effectiveDayRanges = getDayRanges(days: self.daysState.days, index: dayIndex) + for range in effectiveDayRanges { + if self.daysState.intersectingRanges.contains(DayRangeIndex(day: dayIndex, id: range.id)) { + invalidIndices.append(range.id) + } + } + + let subtitleFont = Font.regular(floor(presentationData.listsFontSize.baseDisplaySize * 15.0 / 17.0)) + + if let ranges = self.daysState.days[dayIndex].ranges { + if ranges.isEmpty { + subtitle.append(NSAttributedString(string: environment.strings.BusinessHoursSetup_DayOpen24h, font: subtitleFont, textColor: invalidIndices.contains(0) ? environment.theme.list.itemDestructiveColor : environment.theme.list.itemAccentColor)) + } else { + for i in 0 ..< ranges.count { + let range = ranges[i] + + let startHours = clipMinutes(range.startMinute) / 60 + let startMinutes = clipMinutes(range.startMinute) % 60 + let startText = stringForShortTimestamp(hours: Int32(startHours), minutes: Int32(startMinutes), dateTimeFormat: presentationData.dateTimeFormat) + let endHours = clipMinutes(range.endMinute) / 60 + let endMinutes = clipMinutes(range.endMinute) % 60 + let endText = stringForShortTimestamp(hours: Int32(endHours), minutes: Int32(endMinutes), dateTimeFormat: presentationData.dateTimeFormat) + + var rangeString = "\(startText)\u{00a0}- \(endText)" + if i != ranges.count - 1 { + rangeString.append(", ") + } + + subtitle.append(NSAttributedString(string: rangeString, font: subtitleFont, textColor: invalidIndices.contains(range.id) ? environment.theme.list.itemDestructiveColor : environment.theme.list.itemAccentColor)) + } + } + } else { + subtitle.append(NSAttributedString(string: environment.strings.BusinessHoursSetup_DayClosed, font: subtitleFont, textColor: environment.theme.list.itemAccentColor)) + } + + daysSectionItems.append(AnyComponentWithIdentity(id: dayIndex, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: title, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemPrimaryTextColor + )), + maximumNumberOfLines: 1 + ))), + AnyComponentWithIdentity(id: AnyHashable(1), component: AnyComponent(MultilineTextComponent( + text: .plain(subtitle), + maximumNumberOfLines: 20 + ))) + ], alignment: .left, spacing: 3.0)), + contentInsets: UIEdgeInsets(top: 9.0, left: 0.0, bottom: 10.0, right: 0.0), + accessory: .toggle(ListActionItemComponent.Toggle(style: .regular, isOn: day.ranges != nil, action: { [weak self] _ in + guard let self else { + return + } + if dayIndex < self.daysState.days.count { + var days = self.daysState.days + if days[dayIndex].ranges == nil { + days[dayIndex].ranges = [] + } else { + days[dayIndex].ranges = nil + } + self.daysState.update(days: days) + } + self.state?.updated(transition: .immediate) + })), + action: { [weak self] _ in + guard let self, let component = self.component else { + return + } + self.environment?.controller()?.push(BusinessDaySetupScreen( + context: component.context, + dayIndex: dayIndex, + day: self.daysState.days[dayIndex], + updateDay: { [weak self] day in + guard let self else { + return + } + if self.daysState.days[dayIndex] != day { + var days = self.daysState.days + days[dayIndex] = day + self.daysState.update(days: days) + self.state?.updated(transition: .immediate) + } + } + )) + } + )))) + } + let daysSectionSize = self.daysSection.update( + transition: transition, + component: AnyComponent(ListSectionComponent( + theme: environment.theme, + header: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: environment.strings.BusinessHoursSetup_DaysSectionTitle, + font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), + textColor: environment.theme.list.freeTextColor + )), + maximumNumberOfLines: 0 + )), + footer: nil, + items: daysSectionItems + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 10000.0) + ) + let daysSectionFrame = CGRect(origin: CGPoint(x: sideInset, y: contentHeight + daysContentHeight), size: daysSectionSize) + if let daysSectionView = self.daysSection.view { + if daysSectionView.superview == nil { + daysSectionView.layer.allowsGroupOpacity = true + self.scrollView.addSubview(daysSectionView) + } + transition.setFrame(view: daysSectionView, frame: daysSectionFrame) + + let alphaTransition = transition.animation.isImmediate ? transition : .easeInOut(duration: 0.25) + alphaTransition.setAlpha(view: daysSectionView, alpha: self.showHours ? 1.0 : 0.0) + } + daysContentHeight += daysSectionSize.height + daysContentHeight += sectionSpacing + + let timezoneValueText: String + if let timeZoneList = self.timeZoneList { + if let item = timeZoneList.items.first(where: { $0.id == self.daysState.timezoneId }) { + timezoneValueText = item.title + } else { + timezoneValueText = TimeZone(identifier: self.daysState.timezoneId)?.localizedName(for: .shortStandard, locale: Locale.current) ?? " " + } + } else { + timezoneValueText = "..." + } + + let timezoneSectionSize = self.timezoneSection.update( + transition: transition, + component: AnyComponent(ListSectionComponent( + theme: environment.theme, + header: nil, + footer: nil, + items: [ + AnyComponentWithIdentity(id: 0, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: environment.strings.BusinessHoursSetup_TimeZone, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemPrimaryTextColor + )), + maximumNumberOfLines: 1 + )), + icon: ListActionItemComponent.Icon(component: AnyComponentWithIdentity(id: 0, component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: timezoneValueText, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemSecondaryTextColor + )), + maximumNumberOfLines: 1 + )))), + accessory: .arrow, + action: { [weak self] _ in + guard let self, let component = self.component else { + return + } + var completed: ((String) -> Void)? + let controller = TimezoneSelectionScreen(context: component.context, completed: { timezoneId in + completed?(timezoneId) + }) + controller.navigationPresentation = .modal + self.environment?.controller()?.push(controller) + completed = { [weak self, weak controller] timezoneId in + guard let self else { + controller?.dismiss() + return + } + self.daysState.timezoneId = timezoneId + self.state?.updated(transition: .immediate) + controller?.dismiss() + } + } + ))) + ] + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 10000.0) + ) + let timezoneSectionFrame = CGRect(origin: CGPoint(x: sideInset, y: contentHeight + daysContentHeight), size: timezoneSectionSize) + if let timezoneSectionView = self.timezoneSection.view { + if timezoneSectionView.superview == nil { + self.scrollView.addSubview(timezoneSectionView) + } + transition.setFrame(view: timezoneSectionView, frame: timezoneSectionFrame) + let alphaTransition = transition.animation.isImmediate ? transition : .easeInOut(duration: 0.25) + alphaTransition.setAlpha(view: timezoneSectionView, alpha: self.showHours ? 1.0 : 0.0) + } + daysContentHeight += timezoneSectionSize.height + + if self.showHours { + contentHeight += daysContentHeight + } + + contentHeight += bottomContentInset + contentHeight += environment.safeInsets.bottom + + self.ignoreScrolling = true + let previousBounds = self.scrollView.bounds + + let contentSize = CGSize(width: availableSize.width, height: contentHeight) + if self.scrollView.frame != CGRect(origin: CGPoint(), size: availableSize) { + self.scrollView.frame = CGRect(origin: CGPoint(), size: availableSize) + } + if self.scrollView.contentSize != contentSize { + self.scrollView.contentSize = contentSize + } + let scrollInsets = UIEdgeInsets(top: environment.navigationHeight, left: 0.0, bottom: 0.0, right: 0.0) + if self.scrollView.scrollIndicatorInsets != scrollInsets { + self.scrollView.scrollIndicatorInsets = scrollInsets + } + self.ignoreScrolling = false + + if !previousBounds.isEmpty, !transition.animation.isImmediate { + let bounds = self.scrollView.bounds + if bounds.maxY != previousBounds.maxY { + let offsetY = previousBounds.maxY - bounds.maxY + transition.animateBoundsOrigin(view: self.scrollView, from: CGPoint(x: 0.0, y: offsetY), to: CGPoint(), additive: true) + } + } + + self.topOverscrollLayer.frame = CGRect(origin: CGPoint(x: 0.0, y: -3000.0), size: CGSize(width: availableSize.width, height: 3000.0)) + + self.updateScrolling(transition: transition) + + return availableSize + } + } + + func makeView() -> View { + return View() + } + + func update(view: View, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + return view.update(component: self, availableSize: availableSize, state: state, environment: environment, transition: transition) + } +} + +public final class BusinessHoursSetupScreen: ViewControllerComponentContainer { + private let context: AccountContext + + public init(context: AccountContext, initialValue: TelegramBusinessHours?, completion: @escaping (TelegramBusinessHours?) -> Void) { + self.context = context + + super.init(context: context, component: BusinessHoursSetupScreenComponent( + context: context, + initialValue: initialValue, + completion: completion + ), navigationBarAppearance: .default, theme: .default, updatedPresentationData: nil) + + let presentationData = context.sharedContext.currentPresentationData.with { $0 } + self.title = "" + self.navigationItem.backBarButtonItem = UIBarButtonItem(title: presentationData.strings.Common_Back, style: .plain, target: nil, action: nil) + + self.scrollToTop = { [weak self] in + guard let self, let componentView = self.node.hostView.componentView as? BusinessHoursSetupScreenComponent.View else { + return + } + componentView.scrollToTop() + } + + self.attemptNavigation = { [weak self] complete in + guard let self, let componentView = self.node.hostView.componentView as? BusinessHoursSetupScreenComponent.View else { + return true + } + + return componentView.attemptNavigation(complete: complete) + } + } + + required public init(coder aDecoder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + deinit { + } + + @objc private func cancelPressed() { + self.dismiss() + } + + override public func containerLayoutUpdated(_ layout: ContainerViewLayout, transition: ContainedViewLayoutTransition) { + super.containerLayoutUpdated(layout, transition: transition) + } +} diff --git a/submodules/TelegramUI/Components/Settings/BusinessSetupScreen/BUILD b/submodules/TelegramUI/Components/Settings/BusinessLocationSetupScreen/BUILD similarity index 66% rename from submodules/TelegramUI/Components/Settings/BusinessSetupScreen/BUILD rename to submodules/TelegramUI/Components/Settings/BusinessLocationSetupScreen/BUILD index 6866f506c47..83cfb9c9c7c 100644 --- a/submodules/TelegramUI/Components/Settings/BusinessSetupScreen/BUILD +++ b/submodules/TelegramUI/Components/Settings/BusinessLocationSetupScreen/BUILD @@ -1,8 +1,8 @@ load("@build_bazel_rules_swift//swift:swift.bzl", "swift_library") swift_library( - name = "BusinessSetupScreen", - module_name = "BusinessSetupScreen", + name = "BusinessLocationSetupScreen", + module_name = "BusinessLocationSetupScreen", srcs = glob([ "Sources/**/*.swift", ]), @@ -10,24 +10,28 @@ swift_library( "-warnings-as-errors", ], deps = [ - "//submodules/AsyncDisplayKit", "//submodules/Display", "//submodules/Postbox", "//submodules/TelegramCore", "//submodules/SSignalKit/SwiftSignalKit", "//submodules/TelegramPresentationData", + "//submodules/TelegramUIPreferences", "//submodules/AccountContext", "//submodules/PresentationDataUtils", + "//submodules/Markdown", "//submodules/ComponentFlow", "//submodules/Components/ViewControllerComponent", "//submodules/Components/BundleIconComponent", - "//submodules/TelegramUI/Components/AnimatedTextComponent", "//submodules/Components/MultilineTextComponent", "//submodules/Components/BalancedTextComponent", - "//submodules/TelegramUI/Components/ButtonComponent", - "//submodules/TelegramUI/Components/BackButtonComponent", "//submodules/TelegramUI/Components/ListSectionComponent", "//submodules/TelegramUI/Components/ListActionItemComponent", + "//submodules/TelegramUI/Components/ListMultilineTextFieldItemComponent", + "//submodules/TelegramUI/Components/LottieComponent", + "//submodules/TelegramUI/Components/PlainButtonComponent", + "//submodules/LocationUI", + "//submodules/AppBundle", + "//submodules/Geocoding", ], visibility = [ "//visibility:public", diff --git a/submodules/TelegramUI/Components/Settings/BusinessLocationSetupScreen/Sources/BusinessLocationSetupScreen.swift b/submodules/TelegramUI/Components/Settings/BusinessLocationSetupScreen/Sources/BusinessLocationSetupScreen.swift new file mode 100644 index 00000000000..95f51e3f2c7 --- /dev/null +++ b/submodules/TelegramUI/Components/Settings/BusinessLocationSetupScreen/Sources/BusinessLocationSetupScreen.swift @@ -0,0 +1,682 @@ +import Foundation +import UIKit +import Display +import AsyncDisplayKit +import SwiftSignalKit +import Postbox +import TelegramCore +import TelegramPresentationData +import TelegramUIPreferences +import PresentationDataUtils +import AccountContext +import ComponentFlow +import ViewControllerComponent +import MultilineTextComponent +import BalancedTextComponent +import ListSectionComponent +import ListActionItemComponent +import ListMultilineTextFieldItemComponent +import BundleIconComponent +import LottieComponent +import Markdown +import LocationUI +import CoreLocation +import Geocoding + +final class BusinessLocationSetupScreenComponent: Component { + typealias EnvironmentType = ViewControllerComponentContainer.Environment + + let context: AccountContext + let initialValue: TelegramBusinessLocation? + let completion: (TelegramBusinessLocation?) -> Void + + init( + context: AccountContext, + initialValue: TelegramBusinessLocation?, + completion: @escaping (TelegramBusinessLocation?) -> Void + ) { + self.context = context + self.initialValue = initialValue + self.completion = completion + } + + static func ==(lhs: BusinessLocationSetupScreenComponent, rhs: BusinessLocationSetupScreenComponent) -> Bool { + if lhs.context !== rhs.context { + return false + } + if lhs.initialValue != rhs.initialValue { + return false + } + + return true + } + + private final class ScrollView: UIScrollView { + override func touchesShouldCancel(in view: UIView) -> Bool { + return true + } + } + + final class View: UIView, UIScrollViewDelegate { + private let topOverscrollLayer = SimpleLayer() + private let scrollView: ScrollView + + private let navigationTitle = ComponentView() + private let icon = ComponentView() + private let subtitle = ComponentView() + private let addressSection = ComponentView() + private let mapSection = ComponentView() + private let deleteSection = ComponentView() + + private var isUpdating: Bool = false + + private var component: BusinessLocationSetupScreenComponent? + private(set) weak var state: EmptyComponentState? + private var environment: EnvironmentType? + + private let addressTextInputState = ListMultilineTextFieldItemComponent.ExternalState() + private let textFieldTag = NSObject() + private var resetAddressText: String? + + private var isLoadingGeocodedAddress: Bool = false + private var geocodeDisposable: Disposable? + + private var mapCoordinates: TelegramBusinessLocation.Coordinates? + private var mapCoordinatesManuallySet: Bool = false + + private var applyButtonItem: UIBarButtonItem? + + override init(frame: CGRect) { + self.scrollView = ScrollView() + self.scrollView.showsVerticalScrollIndicator = true + self.scrollView.showsHorizontalScrollIndicator = false + self.scrollView.scrollsToTop = false + self.scrollView.delaysContentTouches = false + self.scrollView.canCancelContentTouches = true + self.scrollView.contentInsetAdjustmentBehavior = .never + if #available(iOS 13.0, *) { + self.scrollView.automaticallyAdjustsScrollIndicatorInsets = false + } + self.scrollView.alwaysBounceVertical = true + + super.init(frame: frame) + + self.scrollView.delegate = self + self.addSubview(self.scrollView) + + self.scrollView.layer.addSublayer(self.topOverscrollLayer) + } + + required init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + deinit { + self.geocodeDisposable?.dispose() + } + + func scrollToTop() { + self.scrollView.setContentOffset(CGPoint(), animated: true) + } + + func attemptNavigation(complete: @escaping () -> Void) -> Bool { + guard let component = self.component, let environment = self.environment else { + return true + } + + let businessLocation = self.currentBusinessLocation() + + if businessLocation != component.initialValue { + let presentationData = component.context.sharedContext.currentPresentationData.with { $0 } + self.environment?.controller()?.present(standardTextAlertController(theme: AlertControllerTheme(presentationData: presentationData), title: nil, text: environment.strings.BusinessLocationSetup_AlertUnsavedChanges_Text, actions: [ + TextAlertAction(type: .genericAction, title: environment.strings.Common_Cancel, action: { + }), + TextAlertAction(type: .destructiveAction, title: environment.strings.BusinessLocationSetup_AlertUnsavedChanges_ResetAction, action: { + complete() + }) + ]), in: .window(.root)) + + return false + } + + return true + } + + func scrollViewDidScroll(_ scrollView: UIScrollView) { + self.updateScrolling(transition: .immediate) + } + + var scrolledUp = true + private func updateScrolling(transition: Transition) { + let navigationRevealOffsetY: CGFloat = 0.0 + + let navigationAlphaDistance: CGFloat = 16.0 + let navigationAlpha: CGFloat = max(0.0, min(1.0, (self.scrollView.contentOffset.y - navigationRevealOffsetY) / navigationAlphaDistance)) + if let controller = self.environment?.controller(), let navigationBar = controller.navigationBar { + transition.setAlpha(layer: navigationBar.backgroundNode.layer, alpha: navigationAlpha) + transition.setAlpha(layer: navigationBar.stripeNode.layer, alpha: navigationAlpha) + } + + var scrolledUp = false + if navigationAlpha < 0.5 { + scrolledUp = true + } else if navigationAlpha > 0.5 { + scrolledUp = false + } + + if self.scrolledUp != scrolledUp { + self.scrolledUp = scrolledUp + if !self.isUpdating { + self.state?.updated() + } + } + + if let navigationTitleView = self.navigationTitle.view { + transition.setAlpha(view: navigationTitleView, alpha: 1.0) + } + } + + private func currentBusinessLocation() -> TelegramBusinessLocation? { + var address = "" + if let textView = self.addressSection.findTaggedView(tag: self.textFieldTag) as? ListMultilineTextFieldItemComponent.View { + address = textView.currentText + } + + var businessLocation: TelegramBusinessLocation? + if !address.isEmpty || self.mapCoordinates != nil { + businessLocation = TelegramBusinessLocation(address: address, coordinates: self.mapCoordinates) + } + return businessLocation + } + + private func openLocationPicker() { + var initialLocation: CLLocationCoordinate2D? + var initialGeocodedLocation: String? + if let mapCoordinates = self.mapCoordinates { + initialLocation = CLLocationCoordinate2D(latitude: mapCoordinates.latitude, longitude: mapCoordinates.longitude) + } else if let textView = self.addressSection.findTaggedView(tag: self.textFieldTag) as? ListMultilineTextFieldItemComponent.View, textView.currentText.count >= 2 { + initialGeocodedLocation = textView.currentText + } + + if let initialGeocodedLocation { + self.isLoadingGeocodedAddress = true + self.state?.updated(transition: .immediate) + + self.geocodeDisposable?.dispose() + self.geocodeDisposable = (geocodeLocation(address: initialGeocodedLocation) + |> deliverOnMainQueue).startStrict(next: { [weak self] venues in + guard let self else { + return + } + self.isLoadingGeocodedAddress = false + self.state?.updated(transition: .immediate) + self.presentLocationPicker(initialLocation: venues?.first?.location?.coordinate) + }) + } else { + self.presentLocationPicker(initialLocation: initialLocation) + } + } + + private func presentLocationPicker(initialLocation: CLLocationCoordinate2D?) { + guard let component = self.component else { + return + } + let controller = LocationPickerController(context: component.context, updatedPresentationData: nil, mode: .pick, initialLocation: initialLocation, completion: { [weak self] location, _, _, address, _ in + guard let self else { + return + } + + self.mapCoordinates = TelegramBusinessLocation.Coordinates(latitude: location.latitude, longitude: location.longitude) + self.mapCoordinatesManuallySet = true + if let textView = self.addressSection.findTaggedView(tag: self.textFieldTag) as? ListMultilineTextFieldItemComponent.View, textView.currentText.isEmpty { + self.resetAddressText = address + } + + self.state?.updated(transition: .immediate) + }) + self.environment?.controller()?.push(controller) + } + + @objc private func savePressed() { + guard let component = self.component, let environment = self.environment else { + return + } + + var address = "" + if let textView = self.addressSection.findTaggedView(tag: self.textFieldTag) as? ListMultilineTextFieldItemComponent.View { + address = textView.currentText + } + + let businessLocation = self.currentBusinessLocation() + + if businessLocation != nil && address.isEmpty { + let presentationData = component.context.sharedContext.currentPresentationData.with { $0 } + self.environment?.controller()?.present(standardTextAlertController(theme: AlertControllerTheme(presentationData: presentationData), title: nil, text: environment.strings.BusinessLocationSetup_ErrorAddressEmpty_Text, actions: [ + TextAlertAction(type: .genericAction, title: environment.strings.Common_OK, action: { + }) + ]), in: .window(.root)) + + return + } + + let _ = component.context.engine.accountData.updateAccountBusinessLocation(businessLocation: businessLocation).startStandalone() + environment.controller()?.dismiss() + } + + func update(component: BusinessLocationSetupScreenComponent, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + self.isUpdating = true + defer { + self.isUpdating = false + } + + if self.component == nil { + if let initialValue = component.initialValue { + self.mapCoordinates = initialValue.coordinates + if self.mapCoordinates != nil { + self.mapCoordinatesManuallySet = true + } + self.resetAddressText = initialValue.address + } + } + + let environment = environment[EnvironmentType.self].value + let themeUpdated = self.environment?.theme !== environment.theme + self.environment = environment + + self.component = component + self.state = state + + let alphaTransition: Transition + if !transition.animation.isImmediate { + alphaTransition = .easeInOut(duration: 0.25) + } else { + alphaTransition = .immediate + } + + if themeUpdated { + self.backgroundColor = environment.theme.list.blocksBackgroundColor + } + + let presentationData = component.context.sharedContext.currentPresentationData.with { $0 } + + let navigationTitleSize = self.navigationTitle.update( + transition: transition, + component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: environment.strings.BusinessLocationSetup_Title, font: Font.semibold(17.0), textColor: environment.theme.rootController.navigationBar.primaryTextColor)), + horizontalAlignment: .center + )), + environment: {}, + containerSize: CGSize(width: availableSize.width, height: 100.0) + ) + let navigationTitleFrame = CGRect(origin: CGPoint(x: floor((availableSize.width - navigationTitleSize.width) / 2.0), y: environment.statusBarHeight + floor((environment.navigationHeight - environment.statusBarHeight - navigationTitleSize.height) / 2.0)), size: navigationTitleSize) + if let navigationTitleView = self.navigationTitle.view { + if navigationTitleView.superview == nil { + if let controller = self.environment?.controller(), let navigationBar = controller.navigationBar { + navigationBar.view.addSubview(navigationTitleView) + } + } + transition.setFrame(view: navigationTitleView, frame: navigationTitleFrame) + } + + let bottomContentInset: CGFloat = 24.0 + let sideInset: CGFloat = 16.0 + environment.safeInsets.left + let sectionSpacing: CGFloat = 24.0 + + var contentHeight: CGFloat = 0.0 + + contentHeight += environment.navigationHeight + + let iconSize = self.icon.update( + transition: .immediate, + component: AnyComponent(LottieComponent( + content: LottieComponent.AppBundleContent(name: "MapEmoji"), + loop: false + )), + environment: {}, + containerSize: CGSize(width: 100.0, height: 100.0) + ) + let iconFrame = CGRect(origin: CGPoint(x: floor((availableSize.width - iconSize.width) * 0.5), y: contentHeight + 11.0), size: iconSize) + if let iconView = self.icon.view as? LottieComponent.View { + if iconView.superview == nil { + self.scrollView.addSubview(iconView) + iconView.playOnce() + } + transition.setPosition(view: iconView, position: iconFrame.center) + iconView.bounds = CGRect(origin: CGPoint(), size: iconFrame.size) + } + + contentHeight += 129.0 + + let subtitleString = NSMutableAttributedString(attributedString: parseMarkdownIntoAttributedString(environment.strings.BusinessLocationSetup_Text, attributes: MarkdownAttributes( + body: MarkdownAttributeSet(font: Font.regular(15.0), textColor: environment.theme.list.freeTextColor), + bold: MarkdownAttributeSet(font: Font.semibold(15.0), textColor: environment.theme.list.freeTextColor), + link: MarkdownAttributeSet(font: Font.regular(15.0), textColor: environment.theme.list.itemAccentColor), + linkAttribute: { attributes in + return ("URL", "") + }), textAlignment: .center + )) + + let subtitleSize = self.subtitle.update( + transition: .immediate, + component: AnyComponent(BalancedTextComponent( + text: .plain(subtitleString), + horizontalAlignment: .center, + maximumNumberOfLines: 0, + lineSpacing: 0.25, + highlightColor: environment.theme.list.itemAccentColor.withMultipliedAlpha(0.1), + highlightAction: { attributes in + if let _ = attributes[NSAttributedString.Key(rawValue: "URL")] { + return NSAttributedString.Key(rawValue: "URL") + } else { + return nil + } + }, + tapAction: { [weak self] _, _ in + guard let self, let component = self.component else { + return + } + let _ = component + } + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 1000.0) + ) + let subtitleFrame = CGRect(origin: CGPoint(x: floor((availableSize.width - subtitleSize.width) * 0.5), y: contentHeight), size: subtitleSize) + if let subtitleView = self.subtitle.view { + if subtitleView.superview == nil { + self.scrollView.addSubview(subtitleView) + } + transition.setPosition(view: subtitleView, position: subtitleFrame.center) + subtitleView.bounds = CGRect(origin: CGPoint(), size: subtitleFrame.size) + } + contentHeight += subtitleSize.height + contentHeight += 27.0 + + var addressSectionItems: [AnyComponentWithIdentity] = [] + addressSectionItems.append(AnyComponentWithIdentity(id: 0, component: AnyComponent(ListMultilineTextFieldItemComponent( + externalState: self.addressTextInputState, + context: component.context, + theme: environment.theme, + strings: environment.strings, + initialText: "", + resetText: self.resetAddressText.flatMap { resetAddressText in + return ListMultilineTextFieldItemComponent.ResetText(value: resetAddressText) + }, + placeholder: environment.strings.BusinessLocationSetup_AddressPlaceholder, + autocapitalizationType: .none, + autocorrectionType: .no, + characterLimit: 256, + allowEmptyLines: false, + updated: { _ in + }, + textUpdateTransition: .spring(duration: 0.4), + tag: self.textFieldTag + )))) + self.resetAddressText = nil + + let addressSectionSize = self.addressSection.update( + transition: transition, + component: AnyComponent(ListSectionComponent( + theme: environment.theme, + header: nil, + footer: nil, + items: addressSectionItems + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 10000.0) + ) + let addressSectionFrame = CGRect(origin: CGPoint(x: sideInset, y: contentHeight), size: addressSectionSize) + if let addressSectionView = self.addressSection.view { + if addressSectionView.superview == nil { + self.scrollView.addSubview(addressSectionView) + self.addressSection.parentState = state + } + transition.setFrame(view: addressSectionView, frame: addressSectionFrame) + } + contentHeight += addressSectionSize.height + contentHeight += sectionSpacing + + var mapSectionItems: [AnyComponentWithIdentity] = [] + + let mapSelectionAccessory: ListActionItemComponent.Accessory? + if self.isLoadingGeocodedAddress { + mapSelectionAccessory = .activity + } else { + mapSelectionAccessory = .toggle(ListActionItemComponent.Toggle(style: .regular, isOn: self.mapCoordinates != nil, isInteractive: self.mapCoordinates != nil)) + } + + mapSectionItems.append(AnyComponentWithIdentity(id: 0, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: environment.strings.BusinessLocationSetup_SetLocationOnMap, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemPrimaryTextColor + )), + maximumNumberOfLines: 1 + ))), + ], alignment: .left, spacing: 2.0)), + accessory: mapSelectionAccessory, + action: { [weak self] _ in + guard let self else { + return + } + if self.mapCoordinates == nil { + self.openLocationPicker() + } else { + self.mapCoordinates = nil + self.mapCoordinatesManuallySet = false + self.state?.updated(transition: .spring(duration: 0.4)) + } + } + )))) + if let mapCoordinates = self.mapCoordinates { + mapSectionItems.append(AnyComponentWithIdentity(id: 1, component: AnyComponent(MapPreviewComponent( + theme: environment.theme, + location: MapPreviewComponent.Location( + latitude: mapCoordinates.latitude, + longitude: mapCoordinates.longitude + ), + action: { [weak self] in + guard let self else { + return + } + self.openLocationPicker() + } + )))) + } + + let mapSectionSize = self.mapSection.update( + transition: transition, + component: AnyComponent(ListSectionComponent( + theme: environment.theme, + header: nil, + footer: nil, + items: mapSectionItems, + displaySeparators: false + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 10000.0) + ) + let mapSectionFrame = CGRect(origin: CGPoint(x: sideInset, y: contentHeight), size: mapSectionSize) + if let mapSectionView = self.mapSection.view { + if mapSectionView.superview == nil { + self.scrollView.addSubview(mapSectionView) + } + transition.setFrame(view: mapSectionView, frame: mapSectionFrame) + } + contentHeight += mapSectionSize.height + + var deleteSectionHeight: CGFloat = 0.0 + + deleteSectionHeight += sectionSpacing + let deleteSectionSize = self.deleteSection.update( + transition: transition, + component: AnyComponent(ListSectionComponent( + theme: environment.theme, + header: nil, + footer: nil, + items: [ + AnyComponentWithIdentity(id: 0, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: environment.strings.BusinessLocationSetup_DeleteLocation, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemDestructiveColor + )), + maximumNumberOfLines: 1 + ))), + ], alignment: .left, spacing: 2.0)), + accessory: nil, + action: { [weak self] _ in + guard let self else { + return + } + + self.resetAddressText = "" + self.mapCoordinates = nil + self.mapCoordinatesManuallySet = false + self.state?.updated(transition: .spring(duration: 0.4)) + } + ))) + ], + displaySeparators: false + )), + environment: {}, + containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 10000.0) + ) + let deleteSectionFrame = CGRect(origin: CGPoint(x: sideInset, y: contentHeight + deleteSectionHeight), size: deleteSectionSize) + if let deleteSectionView = self.deleteSection.view { + if deleteSectionView.superview == nil { + self.scrollView.addSubview(deleteSectionView) + } + transition.setFrame(view: deleteSectionView, frame: deleteSectionFrame) + + if self.mapCoordinates != nil || self.addressTextInputState.hasText { + alphaTransition.setAlpha(view: deleteSectionView, alpha: 1.0) + } else { + alphaTransition.setAlpha(view: deleteSectionView, alpha: 0.0) + } + } + deleteSectionHeight += deleteSectionSize.height + + if self.mapCoordinates != nil || self.addressTextInputState.hasText { + contentHeight += deleteSectionHeight + } + + contentHeight += bottomContentInset + contentHeight += environment.safeInsets.bottom + + let previousBounds = self.scrollView.bounds + + let contentSize = CGSize(width: availableSize.width, height: contentHeight) + if self.scrollView.frame != CGRect(origin: CGPoint(), size: availableSize) { + self.scrollView.frame = CGRect(origin: CGPoint(), size: availableSize) + } + if self.scrollView.contentSize != contentSize { + self.scrollView.contentSize = contentSize + } + let scrollInsets = UIEdgeInsets(top: environment.navigationHeight, left: 0.0, bottom: 0.0, right: 0.0) + if self.scrollView.scrollIndicatorInsets != scrollInsets { + self.scrollView.scrollIndicatorInsets = scrollInsets + } + + if !previousBounds.isEmpty, !transition.animation.isImmediate { + let bounds = self.scrollView.bounds + if bounds.maxY != previousBounds.maxY { + let offsetY = previousBounds.maxY - bounds.maxY + transition.animateBoundsOrigin(view: self.scrollView, from: CGPoint(x: 0.0, y: offsetY), to: CGPoint(), additive: true) + } + } + + self.topOverscrollLayer.frame = CGRect(origin: CGPoint(x: 0.0, y: -3000.0), size: CGSize(width: availableSize.width, height: 3000.0)) + + self.updateScrolling(transition: transition) + + if let controller = environment.controller() as? BusinessLocationSetupScreen { + let businessLocation = self.currentBusinessLocation() + + if businessLocation == component.initialValue { + if controller.navigationItem.rightBarButtonItem != nil { + controller.navigationItem.setRightBarButton(nil, animated: true) + } + } else { + let applyButtonItem: UIBarButtonItem + if let current = self.applyButtonItem { + applyButtonItem = current + } else { + applyButtonItem = UIBarButtonItem(title: environment.strings.Common_Save, style: .done, target: self, action: #selector(self.savePressed)) + } + if controller.navigationItem.rightBarButtonItem !== applyButtonItem { + controller.navigationItem.setRightBarButton(applyButtonItem, animated: true) + } + } + } + + return availableSize + } + } + + func makeView() -> View { + return View() + } + + func update(view: View, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + return view.update(component: self, availableSize: availableSize, state: state, environment: environment, transition: transition) + } +} + +public final class BusinessLocationSetupScreen: ViewControllerComponentContainer { + private let context: AccountContext + + public init( + context: AccountContext, + initialValue: TelegramBusinessLocation?, + completion: @escaping (TelegramBusinessLocation?) -> Void + ) { + self.context = context + + super.init(context: context, component: BusinessLocationSetupScreenComponent( + context: context, + initialValue: initialValue, + completion: completion + ), navigationBarAppearance: .default, theme: .default, updatedPresentationData: nil) + + let presentationData = context.sharedContext.currentPresentationData.with { $0 } + self.title = "" + self.navigationItem.backBarButtonItem = UIBarButtonItem(title: presentationData.strings.Common_Back, style: .plain, target: nil, action: nil) + + self.scrollToTop = { [weak self] in + guard let self, let componentView = self.node.hostView.componentView as? BusinessLocationSetupScreenComponent.View else { + return + } + componentView.scrollToTop() + } + + self.attemptNavigation = { [weak self] complete in + guard let self, let componentView = self.node.hostView.componentView as? BusinessLocationSetupScreenComponent.View else { + return true + } + + return componentView.attemptNavigation(complete: complete) + } + } + + required public init(coder aDecoder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + deinit { + } + + @objc private func cancelPressed() { + self.dismiss() + } + + override public func containerLayoutUpdated(_ layout: ContainerViewLayout, transition: ContainedViewLayoutTransition) { + super.containerLayoutUpdated(layout, transition: transition) + } +} diff --git a/submodules/TelegramUI/Components/Settings/BusinessLocationSetupScreen/Sources/MapPreviewComponent.swift b/submodules/TelegramUI/Components/Settings/BusinessLocationSetupScreen/Sources/MapPreviewComponent.swift new file mode 100644 index 00000000000..9e887c62796 --- /dev/null +++ b/submodules/TelegramUI/Components/Settings/BusinessLocationSetupScreen/Sources/MapPreviewComponent.swift @@ -0,0 +1,139 @@ +import Foundation +import UIKit +import Display +import ComponentFlow +import ListSectionComponent +import MapKit +import TelegramPresentationData +import AppBundle + +final class MapPreviewComponent: Component { + struct Location: Equatable { + var latitude: Double + var longitude: Double + + init(latitude: Double, longitude: Double) { + self.latitude = latitude + self.longitude = longitude + } + } + + let theme: PresentationTheme + let location: Location + let action: (() -> Void)? + + init( + theme: PresentationTheme, + location: Location, + action: (() -> Void)? = nil + ) { + self.theme = theme + self.location = location + self.action = action + } + + static func ==(lhs: MapPreviewComponent, rhs: MapPreviewComponent) -> Bool { + if lhs.theme !== rhs.theme { + return false + } + if lhs.location != rhs.location { + return false + } + if (lhs.action == nil) != (rhs.action == nil) { + return false + } + return true + } + + final class View: HighlightTrackingButton, ListSectionComponent.ChildView { + private var component: MapPreviewComponent? + private weak var componentState: EmptyComponentState? + + private var mapView: MKMapView? + + private let pinShadowView: UIImageView + private let pinView: UIImageView + private let pinForegroundView: UIImageView + + var customUpdateIsHighlighted: ((Bool) -> Void)? + private(set) var separatorInset: CGFloat = 0.0 + + override init(frame: CGRect) { + self.pinShadowView = UIImageView() + self.pinView = UIImageView() + self.pinForegroundView = UIImageView() + + super.init(frame: frame) + + self.addSubview(self.pinShadowView) + self.addSubview(self.pinView) + self.addSubview(self.pinForegroundView) + + self.pinShadowView.image = UIImage(bundleImageName: "Chat/Message/LocationPinShadow") + self.pinView.image = UIImage(bundleImageName: "Chat/Message/LocationPinBackground")?.withRenderingMode(.alwaysTemplate) + self.pinForegroundView.image = UIImage(bundleImageName: "Chat/Message/LocationPinForeground")?.withRenderingMode(.alwaysTemplate) + + self.addTarget(self, action: #selector(self.pressed), for: .touchUpInside) + } + + required init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + @objc private func pressed() { + self.component?.action?() + } + + func update(component: MapPreviewComponent, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + let previousComponent = self.component + self.component = component + self.componentState = state + + self.isEnabled = component.action != nil + + let size = CGSize(width: availableSize.width, height: 160.0) + + let mapView: MKMapView + if let current = self.mapView { + mapView = current + } else { + mapView = MKMapView() + mapView.isUserInteractionEnabled = false + self.mapView = mapView + self.insertSubview(mapView, at: 0) + } + transition.setFrame(view: mapView, frame: CGRect(origin: CGPoint(), size: size)) + + let defaultMapSpan = MKCoordinateSpan(latitudeDelta: 0.016, longitudeDelta: 0.016) + + let region = MKCoordinateRegion(center: CLLocationCoordinate2D(latitude: component.location.latitude, longitude: component.location.longitude), span: defaultMapSpan) + if previousComponent?.location != component.location { + mapView.setRegion(region, animated: false) + mapView.setVisibleMapRect(mapView.visibleMapRect, edgePadding: UIEdgeInsets(top: 70.0, left: 0.0, bottom: 0.0, right: 0.0), animated: true) + } + + let pinImageSize = self.pinView.image?.size ?? CGSize(width: 62.0, height: 74.0) + let pinFrame = CGRect(origin: CGPoint(x: floor((size.width - pinImageSize.width) * 0.5), y: floor((size.height - pinImageSize.height) * 0.5)), size: pinImageSize) + transition.setFrame(view: self.pinShadowView, frame: pinFrame) + + transition.setFrame(view: self.pinView, frame: pinFrame) + self.pinView.tintColor = component.theme.list.itemCheckColors.fillColor + + if let image = pinForegroundView.image { + let pinIconFrame = CGRect(origin: CGPoint(x: pinFrame.minX + floor((pinFrame.width - image.size.width) * 0.5), y: pinFrame.minY + 15.0), size: image.size) + transition.setFrame(view: self.pinForegroundView, frame: pinIconFrame) + self.pinForegroundView.tintColor = component.theme.list.itemCheckColors.foregroundColor + } + + return size + } + } + + func makeView() -> View { + return View(frame: CGRect()) + } + + func update(view: View, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + return view.update(component: self, availableSize: availableSize, state: state, environment: environment, transition: transition) + } +} diff --git a/submodules/TelegramUI/Components/Settings/BusinessSetupScreen/Sources/BusinessSetupScreen.swift b/submodules/TelegramUI/Components/Settings/BusinessSetupScreen/Sources/BusinessSetupScreen.swift deleted file mode 100644 index 26e1df5d667..00000000000 --- a/submodules/TelegramUI/Components/Settings/BusinessSetupScreen/Sources/BusinessSetupScreen.swift +++ /dev/null @@ -1,426 +0,0 @@ -import Foundation -import UIKit -import Photos -import Display -import AsyncDisplayKit -import SwiftSignalKit -import Postbox -import TelegramCore -import TelegramPresentationData -import TelegramUIPreferences -import PresentationDataUtils -import AccountContext -import ComponentFlow -import ViewControllerComponent -import MultilineTextComponent -import BalancedTextComponent -import BackButtonComponent -import ListSectionComponent -import ListActionItemComponent -import BundleIconComponent - -final class BusinessSetupScreenComponent: Component { - typealias EnvironmentType = ViewControllerComponentContainer.Environment - - let context: AccountContext - - init( - context: AccountContext - ) { - self.context = context - } - - static func ==(lhs: BusinessSetupScreenComponent, rhs: BusinessSetupScreenComponent) -> Bool { - if lhs.context !== rhs.context { - return false - } - - return true - } - - private final class ScrollView: UIScrollView { - override func touchesShouldCancel(in view: UIView) -> Bool { - return true - } - } - - final class View: UIView, UIScrollViewDelegate { - private let topOverscrollLayer = SimpleLayer() - private let scrollView: ScrollView - - private let navigationTitle = ComponentView() - private let title = ComponentView() - private let subtitle = ComponentView() - private let actionsSection = ComponentView() - - private var isUpdating: Bool = false - - private var component: BusinessSetupScreenComponent? - private(set) weak var state: EmptyComponentState? - private var environment: EnvironmentType? - - override init(frame: CGRect) { - self.scrollView = ScrollView() - self.scrollView.showsVerticalScrollIndicator = true - self.scrollView.showsHorizontalScrollIndicator = false - self.scrollView.scrollsToTop = false - self.scrollView.delaysContentTouches = false - self.scrollView.canCancelContentTouches = true - self.scrollView.contentInsetAdjustmentBehavior = .never - if #available(iOS 13.0, *) { - self.scrollView.automaticallyAdjustsScrollIndicatorInsets = false - } - self.scrollView.alwaysBounceVertical = true - - super.init(frame: frame) - - self.scrollView.delegate = self - self.addSubview(self.scrollView) - - self.scrollView.layer.addSublayer(self.topOverscrollLayer) - } - - required init?(coder: NSCoder) { - fatalError("init(coder:) has not been implemented") - } - - deinit { - } - - func scrollToTop() { - self.scrollView.setContentOffset(CGPoint(), animated: true) - } - - func attemptNavigation(complete: @escaping () -> Void) -> Bool { - return true - } - - func scrollViewDidScroll(_ scrollView: UIScrollView) { - self.updateScrolling(transition: .immediate) - } - - var scrolledUp = true - private func updateScrolling(transition: Transition) { - guard let environment = self.environment else { - return - } - - let navigationRevealOffsetY: CGFloat = -(environment.statusBarHeight + (environment.navigationHeight - environment.statusBarHeight) * 0.5) + (self.title.view?.frame.midY ?? 0.0) - - let navigationAlphaDistance: CGFloat = 16.0 - let navigationAlpha: CGFloat = max(0.0, min(1.0, (self.scrollView.contentOffset.y - navigationRevealOffsetY) / navigationAlphaDistance)) - if let controller = self.environment?.controller(), let navigationBar = controller.navigationBar { - transition.setAlpha(layer: navigationBar.backgroundNode.layer, alpha: navigationAlpha) - transition.setAlpha(layer: navigationBar.stripeNode.layer, alpha: navigationAlpha) - } - - var scrolledUp = false - if navigationAlpha < 0.5 { - scrolledUp = true - } else if navigationAlpha > 0.5 { - scrolledUp = false - } - - if self.scrolledUp != scrolledUp { - self.scrolledUp = scrolledUp - if !self.isUpdating { - self.state?.updated() - } - } - - if let navigationTitleView = self.navigationTitle.view { - transition.setAlpha(view: navigationTitleView, alpha: navigationAlpha) - } - } - - func update(component: BusinessSetupScreenComponent, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { - self.isUpdating = true - defer { - self.isUpdating = false - } - - let environment = environment[EnvironmentType.self].value - let themeUpdated = self.environment?.theme !== environment.theme - self.environment = environment - - self.component = component - self.state = state - - if themeUpdated { - self.backgroundColor = environment.theme.list.blocksBackgroundColor - } - - let presentationData = component.context.sharedContext.currentPresentationData.with { $0 } - - //TODO:localize - let navigationTitleSize = self.navigationTitle.update( - transition: transition, - component: AnyComponent(MultilineTextComponent( - text: .plain(NSAttributedString(string: "Telegram Business", font: Font.semibold(17.0), textColor: environment.theme.rootController.navigationBar.primaryTextColor)), - horizontalAlignment: .center - )), - environment: {}, - containerSize: CGSize(width: availableSize.width, height: 100.0) - ) - let navigationTitleFrame = CGRect(origin: CGPoint(x: floor((availableSize.width - navigationTitleSize.width) / 2.0), y: environment.statusBarHeight + floor((environment.navigationHeight - environment.statusBarHeight - navigationTitleSize.height) / 2.0)), size: navigationTitleSize) - if let navigationTitleView = self.navigationTitle.view { - if navigationTitleView.superview == nil { - if let controller = self.environment?.controller(), let navigationBar = controller.navigationBar { - navigationBar.view.addSubview(navigationTitleView) - } - } - transition.setFrame(view: navigationTitleView, frame: navigationTitleFrame) - } - - let bottomContentInset: CGFloat = 24.0 - let sideInset: CGFloat = 16.0 + environment.safeInsets.left - let sectionSpacing: CGFloat = 32.0 - - let _ = bottomContentInset - let _ = sectionSpacing - - var contentHeight: CGFloat = 0.0 - - contentHeight += environment.navigationHeight - contentHeight += 81.0 - - //TODO:localize - let titleSize = self.title.update( - transition: .immediate, - component: AnyComponent(MultilineTextComponent( - text: .plain(NSAttributedString(string: "Telegram Business", font: Font.bold(29.0), textColor: environment.theme.list.itemPrimaryTextColor)), - horizontalAlignment: .center, - maximumNumberOfLines: 1 - )), - environment: {}, - containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 1000.0) - ) - let titleFrame = CGRect(origin: CGPoint(x: floor((availableSize.width - titleSize.width) * 0.5), y: contentHeight), size: titleSize) - if let titleView = self.title.view { - if titleView.superview == nil { - self.scrollView.addSubview(titleView) - } - transition.setPosition(view: titleView, position: titleFrame.center) - titleView.bounds = CGRect(origin: CGPoint(), size: titleFrame.size) - } - contentHeight += titleSize.height - contentHeight += 17.0 - - //TODO:localize - let subtitleSize = self.subtitle.update( - transition: .immediate, - component: AnyComponent(BalancedTextComponent( - text: .plain(NSAttributedString(string: "You have now unlocked these additional business features.", font: Font.regular(15.0), textColor: environment.theme.list.itemPrimaryTextColor)), - horizontalAlignment: .center, - maximumNumberOfLines: 0, - lineSpacing: 0.25 - )), - environment: {}, - containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 1000.0) - ) - let subtitleFrame = CGRect(origin: CGPoint(x: floor((availableSize.width - subtitleSize.width) * 0.5), y: contentHeight), size: subtitleSize) - if let subtitleView = self.subtitle.view { - if subtitleView.superview == nil { - self.scrollView.addSubview(subtitleView) - } - transition.setPosition(view: subtitleView, position: subtitleFrame.center) - subtitleView.bounds = CGRect(origin: CGPoint(), size: subtitleFrame.size) - } - contentHeight += subtitleSize.height - contentHeight += 21.0 - - struct Item { - var icon: String - var title: String - var subtitle: String - var action: () -> Void - } - var items: [Item] = [] - //TODO:localize - items.append(Item( - icon: "Settings/Menu/AddAccount", - title: "Location", - subtitle: "Display the location of your business on your account.", - action: { - } - )) - items.append(Item( - icon: "Settings/Menu/DataVoice", - title: "Opening Hours", - subtitle: "Show to your customers when you are open for business.", - action: { - } - )) - items.append(Item( - icon: "Settings/Menu/Photos", - title: "Quick Replies", - subtitle: "Set up shortcuts with rich text and media to respond to messages faster.", - action: { - } - )) - items.append(Item( - icon: "Settings/Menu/Stories", - title: "Greeting Messages", - subtitle: "Create greetings that will be automatically sent to new customers.", - action: { - } - )) - items.append(Item( - icon: "Settings/Menu/Trending", - title: "Away Messages", - subtitle: "Define messages that are automatically sent when you are off.", - action: { - } - )) - items.append(Item( - icon: "Settings/Menu/DataStickers", - title: "Chatbots", - subtitle: "Add any third-party chatbots that will process customer interactions.", - action: { [weak self] in - guard let self, let component = self.component, let environment = self.environment else { - return - } - environment.controller()?.push(component.context.sharedContext.makeChatbotSetupScreen(context: component.context)) - } - )) - - var actionsSectionItems: [AnyComponentWithIdentity] = [] - for item in items { - actionsSectionItems.append(AnyComponentWithIdentity(id: actionsSectionItems.count, component: AnyComponent(ListActionItemComponent( - theme: environment.theme, - title: AnyComponent(VStack([ - AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( - text: .plain(NSAttributedString( - string: item.title, - font: Font.regular(presentationData.listsFontSize.baseDisplaySize), - textColor: environment.theme.list.itemPrimaryTextColor - )), - maximumNumberOfLines: 0 - ))), - AnyComponentWithIdentity(id: AnyHashable(1), component: AnyComponent(MultilineTextComponent( - text: .plain(NSAttributedString( - string: item.subtitle, - font: Font.regular(floor(presentationData.listsFontSize.baseDisplaySize * 13.0 / 17.0)), - textColor: environment.theme.list.itemSecondaryTextColor - )), - maximumNumberOfLines: 0, - lineSpacing: 0.18 - ))) - ], alignment: .left, spacing: 2.0)), - leftIcon: AnyComponentWithIdentity(id: 0, component: AnyComponent(BundleIconComponent( - name: item.icon, - tintColor: nil - ))), - action: { _ in - item.action() - } - )))) - } - - let actionsSectionSize = self.actionsSection.update( - transition: transition, - component: AnyComponent(ListSectionComponent( - theme: environment.theme, - header: nil, - footer: nil, - items: actionsSectionItems - )), - environment: {}, - containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 10000.0) - ) - let actionsSectionFrame = CGRect(origin: CGPoint(x: sideInset, y: contentHeight), size: actionsSectionSize) - if let actionsSectionView = self.actionsSection.view { - if actionsSectionView.superview == nil { - self.scrollView.addSubview(actionsSectionView) - } - transition.setFrame(view: actionsSectionView, frame: actionsSectionFrame) - } - contentHeight += actionsSectionSize.height - - contentHeight += bottomContentInset - contentHeight += environment.safeInsets.bottom - - let previousBounds = self.scrollView.bounds - - let contentSize = CGSize(width: availableSize.width, height: contentHeight) - if self.scrollView.frame != CGRect(origin: CGPoint(), size: availableSize) { - self.scrollView.frame = CGRect(origin: CGPoint(), size: availableSize) - } - if self.scrollView.contentSize != contentSize { - self.scrollView.contentSize = contentSize - } - let scrollInsets = UIEdgeInsets(top: environment.navigationHeight, left: 0.0, bottom: 0.0, right: 0.0) - if self.scrollView.scrollIndicatorInsets != scrollInsets { - self.scrollView.scrollIndicatorInsets = scrollInsets - } - - if !previousBounds.isEmpty, !transition.animation.isImmediate { - let bounds = self.scrollView.bounds - if bounds.maxY != previousBounds.maxY { - let offsetY = previousBounds.maxY - bounds.maxY - transition.animateBoundsOrigin(view: self.scrollView, from: CGPoint(x: 0.0, y: offsetY), to: CGPoint(), additive: true) - } - } - - self.topOverscrollLayer.frame = CGRect(origin: CGPoint(x: 0.0, y: -3000.0), size: CGSize(width: availableSize.width, height: 3000.0)) - - self.updateScrolling(transition: transition) - - return availableSize - } - } - - func makeView() -> View { - return View() - } - - func update(view: View, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { - return view.update(component: self, availableSize: availableSize, state: state, environment: environment, transition: transition) - } -} - -public final class BusinessSetupScreen: ViewControllerComponentContainer { - private let context: AccountContext - - public init(context: AccountContext) { - self.context = context - - super.init(context: context, component: BusinessSetupScreenComponent( - context: context - ), navigationBarAppearance: .default, theme: .default, updatedPresentationData: nil) - - let presentationData = context.sharedContext.currentPresentationData.with { $0 } - self.title = "" - self.navigationItem.backBarButtonItem = UIBarButtonItem(title: presentationData.strings.Common_Back, style: .plain, target: nil, action: nil) - - self.scrollToTop = { [weak self] in - guard let self, let componentView = self.node.hostView.componentView as? BusinessSetupScreenComponent.View else { - return - } - componentView.scrollToTop() - } - - self.attemptNavigation = { [weak self] complete in - guard let self, let componentView = self.node.hostView.componentView as? BusinessSetupScreenComponent.View else { - return true - } - - return componentView.attemptNavigation(complete: complete) - } - } - - required public init(coder aDecoder: NSCoder) { - fatalError("init(coder:) has not been implemented") - } - - deinit { - } - - @objc private func cancelPressed() { - self.dismiss() - } - - override public func containerLayoutUpdated(_ layout: ContainerViewLayout, transition: ContainedViewLayoutTransition) { - super.containerLayoutUpdated(layout, transition: transition) - } -} diff --git a/submodules/TelegramUI/Components/Settings/ChatbotSetupScreen/BUILD b/submodules/TelegramUI/Components/Settings/ChatbotSetupScreen/BUILD index 9cc671937c2..c83d5538a0e 100644 --- a/submodules/TelegramUI/Components/Settings/ChatbotSetupScreen/BUILD +++ b/submodules/TelegramUI/Components/Settings/ChatbotSetupScreen/BUILD @@ -31,6 +31,10 @@ swift_library( "//submodules/TelegramUI/Components/ListActionItemComponent", "//submodules/TelegramUI/Components/ListTextFieldItemComponent", "//submodules/TelegramUI/Components/LottieComponent", + "//submodules/AvatarNode", + "//submodules/TelegramUI/Components/PlainButtonComponent", + "//submodules/TelegramUI/Components/Stories/PeerListItemComponent", + "//submodules/ShimmerEffect", ], visibility = [ "//visibility:public", diff --git a/submodules/TelegramUI/Components/Settings/ChatbotSetupScreen/Sources/ChatbotSearchResultItemComponent.swift b/submodules/TelegramUI/Components/Settings/ChatbotSetupScreen/Sources/ChatbotSearchResultItemComponent.swift new file mode 100644 index 00000000000..c9b085c8ed0 --- /dev/null +++ b/submodules/TelegramUI/Components/Settings/ChatbotSetupScreen/Sources/ChatbotSearchResultItemComponent.swift @@ -0,0 +1,402 @@ +import Foundation +import UIKit +import Display +import ComponentFlow +import MultilineTextComponent +import AvatarNode +import BundleIconComponent +import TelegramPresentationData +import TelegramCore +import AccountContext +import ListSectionComponent +import PlainButtonComponent +import ShimmerEffect + +final class ChatbotSearchResultItemComponent: Component { + enum Content: Equatable { + case searching + case found(peer: EnginePeer, isInstalled: Bool) + case notFound + } + + let context: AccountContext + let theme: PresentationTheme + let strings: PresentationStrings + let content: Content + let installAction: () -> Void + let removeAction: () -> Void + + init( + context: AccountContext, + theme: PresentationTheme, + strings: PresentationStrings, + content: Content, + installAction: @escaping () -> Void, + removeAction: @escaping () -> Void + ) { + self.context = context + self.theme = theme + self.strings = strings + self.content = content + self.installAction = installAction + self.removeAction = removeAction + } + + static func ==(lhs: ChatbotSearchResultItemComponent, rhs: ChatbotSearchResultItemComponent) -> Bool { + if lhs.context !== rhs.context { + return false + } + if lhs.theme !== rhs.theme { + return false + } + if lhs.strings !== rhs.strings { + return false + } + if lhs.content != rhs.content { + return false + } + return true + } + + final class View: UIView, ListSectionComponent.ChildView { + private var notFoundLabel: ComponentView? + private let titleLabel = ComponentView() + private let subtitleLabel = ComponentView() + + private var shimmerEffectNode: ShimmerEffectNode? + + private var avatarNode: AvatarNode? + + private var addButton: ComponentView? + private var removeButton: ComponentView? + + private var component: ChatbotSearchResultItemComponent? + private weak var state: EmptyComponentState? + + var customUpdateIsHighlighted: ((Bool) -> Void)? + private(set) var separatorInset: CGFloat = 0.0 + + override init(frame: CGRect) { + super.init(frame: frame) + } + + required init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + func update(component: ChatbotSearchResultItemComponent, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + self.component = component + self.state = state + + let sideInset: CGFloat = 10.0 + let avatarDiameter: CGFloat = 40.0 + let avatarTextSpacing: CGFloat = 12.0 + let titleSubtitleSpacing: CGFloat = 1.0 + let verticalInset: CGFloat = 11.0 + + let maxTextWidth: CGFloat = availableSize.width - sideInset * 2.0 - avatarDiameter - avatarTextSpacing + + var addButtonSize: CGSize? + if case .found(_, false) = component.content { + let addButton: ComponentView + var addButtonTransition = transition + if let current = self.addButton { + addButton = current + } else { + addButtonTransition = addButtonTransition.withAnimation(.none) + addButton = ComponentView() + self.addButton = addButton + } + + addButtonSize = addButton.update( + transition: addButtonTransition, + component: AnyComponent(PlainButtonComponent( + content: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: component.strings.ChatbotSetup_BotAddAction, font: Font.semibold(15.0), textColor: component.theme.list.itemCheckColors.foregroundColor)) + )), + background: AnyComponent(RoundedRectangle(color: component.theme.list.itemCheckColors.fillColor, cornerRadius: nil)), + effectAlignment: .center, + minSize: nil, + contentInsets: UIEdgeInsets(top: 4.0, left: 8.0, bottom: 4.0, right: 8.0), + action: { [weak self] in + guard let self, let component = self.component else { + return + } + component.installAction() + }, + animateAlpha: true, + animateScale: false + )), + environment: {}, + containerSize: CGSize(width: 100.0, height: 100.0) + ) + } else { + if let addButton = self.addButton { + self.addButton = nil + if let addButtonView = addButton.view { + if !transition.animation.isImmediate { + transition.setScale(view: addButtonView, scale: 0.001) + Transition.easeInOut(duration: 0.2).setAlpha(view: addButtonView, alpha: 0.0, completion: { [weak addButtonView] _ in + addButtonView?.removeFromSuperview() + }) + } else { + addButtonView.removeFromSuperview() + } + } + } + } + + var removeButtonSize: CGSize? + if case .found(_, true) = component.content { + let removeButton: ComponentView + var removeButtonTransition = transition + if let current = self.removeButton { + removeButton = current + } else { + removeButtonTransition = removeButtonTransition.withAnimation(.none) + removeButton = ComponentView() + self.removeButton = removeButton + } + + removeButtonSize = removeButton.update( + transition: removeButtonTransition, + component: AnyComponent(PlainButtonComponent( + content: AnyComponent(BundleIconComponent( + name: "Chat/Message/SideCloseIcon", + tintColor: component.theme.list.controlSecondaryColor + )), + effectAlignment: .center, + minSize: nil, + contentInsets: UIEdgeInsets(top: 4.0, left: 4.0, bottom: 4.0, right: 4.0), + action: { [weak self] in + guard let self, let component = self.component else { + return + } + component.removeAction() + }, + animateAlpha: true, + animateScale: false + )), + environment: {}, + containerSize: CGSize(width: 100.0, height: 100.0) + ) + } else { + if let removeButton = self.removeButton { + self.removeButton = nil + if let removeButtonView = removeButton.view { + if !transition.animation.isImmediate { + transition.setScale(view: removeButtonView, scale: 0.001) + Transition.easeInOut(duration: 0.2).setAlpha(view: removeButtonView, alpha: 0.0, completion: { [weak removeButtonView] _ in + removeButtonView?.removeFromSuperview() + }) + } else { + removeButtonView.removeFromSuperview() + } + } + } + } + + let titleValue: String + let subtitleValue: String + let isTextVisible: Bool + switch component.content { + case .searching, .notFound: + isTextVisible = false + titleValue = "AAAAAAAAA" + subtitleValue = component.strings.Bot_GenericBotStatus + case let .found(peer, _): + isTextVisible = true + titleValue = peer.displayTitle(strings: component.strings, displayOrder: .firstLast) + subtitleValue = component.strings.Bot_GenericBotStatus + } + + let titleSize = self.titleLabel.update( + transition: .immediate, + component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: titleValue, font: Font.semibold(17.0), textColor: component.theme.list.itemPrimaryTextColor)), + maximumNumberOfLines: 1 + )), + environment: {}, + containerSize: CGSize(width: maxTextWidth, height: 100.0) + ) + let subtitleSize = self.subtitleLabel.update( + transition: .immediate, + component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: subtitleValue, font: Font.regular(15.0), textColor: component.theme.list.itemSecondaryTextColor)), + maximumNumberOfLines: 1 + )), + environment: {}, + containerSize: CGSize(width: maxTextWidth, height: 100.0) + ) + + let size = CGSize(width: availableSize.width, height: verticalInset * 2.0 + titleSize.height + titleSubtitleSpacing + subtitleSize.height) + + let titleFrame = CGRect(origin: CGPoint(x: sideInset + avatarDiameter + avatarTextSpacing, y: verticalInset), size: titleSize) + if let titleView = self.titleLabel.view { + var titleTransition = transition + if titleView.superview == nil { + titleTransition = .immediate + titleView.layer.anchorPoint = CGPoint() + self.addSubview(titleView) + } + if titleView.isHidden != !isTextVisible { + titleTransition = .immediate + } + + titleView.bounds = CGRect(origin: CGPoint(), size: titleFrame.size) + titleTransition.setPosition(view: titleView, position: titleFrame.origin) + titleView.isHidden = !isTextVisible + } + + let subtitleFrame = CGRect(origin: CGPoint(x: sideInset + avatarDiameter + avatarTextSpacing, y: verticalInset + titleSize.height + titleSubtitleSpacing), size: subtitleSize) + if let subtitleView = self.subtitleLabel.view { + var subtitleTransition = transition + if subtitleView.superview == nil { + subtitleTransition = .immediate + subtitleView.layer.anchorPoint = CGPoint() + self.addSubview(subtitleView) + } + if subtitleView.isHidden != !isTextVisible { + subtitleTransition = .immediate + } + + subtitleView.bounds = CGRect(origin: CGPoint(), size: subtitleFrame.size) + subtitleTransition.setPosition(view: subtitleView, position: subtitleFrame.origin) + subtitleView.isHidden = !isTextVisible + } + + let avatarFrame = CGRect(origin: CGPoint(x: sideInset, y: floor((size.height - avatarDiameter) * 0.5)), size: CGSize(width: avatarDiameter, height: avatarDiameter)) + + if case let .found(peer, _) = component.content { + var avatarTransition = transition + let avatarNode: AvatarNode + if let current = self.avatarNode { + avatarNode = current + } else { + avatarTransition = .immediate + avatarNode = AvatarNode(font: avatarPlaceholderFont(size: 17.0)) + self.avatarNode = avatarNode + self.addSubview(avatarNode.view) + } + avatarTransition.setFrame(view: avatarNode.view, frame: avatarFrame) + avatarNode.setPeer(context: component.context, theme: component.theme, peer: peer, synchronousLoad: true, displayDimensions: avatarFrame.size) + avatarNode.updateSize(size: avatarFrame.size) + } else { + if let avatarNode = self.avatarNode { + self.avatarNode = nil + avatarNode.view.removeFromSuperview() + } + } + + if case .notFound = component.content { + let notFoundLabel: ComponentView + if let current = self.notFoundLabel { + notFoundLabel = current + } else { + notFoundLabel = ComponentView() + self.notFoundLabel = notFoundLabel + } + let notFoundLabelSize = notFoundLabel.update( + transition: .immediate, + component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: component.strings.ChatbotSetup_BotNotFoundStatus, font: Font.regular(13.0), textColor: component.theme.list.itemSecondaryTextColor)) + )), + environment: {}, + containerSize: CGSize(width: maxTextWidth, height: 100.0) + ) + let notFoundLabelFrame = CGRect(origin: CGPoint(x: floor((size.width - notFoundLabelSize.width) * 0.5), y: floor((size.height - notFoundLabelSize.height) * 0.5)), size: notFoundLabelSize) + if let notFoundLabelView = notFoundLabel.view { + var notFoundLabelTransition = transition + if notFoundLabelView.superview == nil { + notFoundLabelTransition = .immediate + self.addSubview(notFoundLabelView) + } + notFoundLabelTransition.setPosition(view: notFoundLabelView, position: notFoundLabelFrame.center) + notFoundLabelView.bounds = CGRect(origin: CGPoint(), size: notFoundLabelFrame.size) + } + } else { + if let notFoundLabel = self.notFoundLabel { + self.notFoundLabel = nil + notFoundLabel.view?.removeFromSuperview() + } + } + + if let addButton = self.addButton, let addButtonSize { + var addButtonTransition = transition + let addButtonFrame = CGRect(origin: CGPoint(x: availableSize.width - sideInset - addButtonSize.width, y: floor((size.height - addButtonSize.height) * 0.5)), size: addButtonSize) + if let addButtonView = addButton.view { + if addButtonView.superview == nil { + addButtonTransition = addButtonTransition.withAnimation(.none) + self.addSubview(addButtonView) + if !transition.animation.isImmediate { + transition.animateScale(view: addButtonView, from: 0.001, to: 1.0) + Transition.easeInOut(duration: 0.2).animateAlpha(view: addButtonView, from: 0.0, to: 1.0) + } + } + addButtonTransition.setFrame(view: addButtonView, frame: addButtonFrame) + } + } + + if let removeButton = self.removeButton, let removeButtonSize { + var removeButtonTransition = transition + let removeButtonFrame = CGRect(origin: CGPoint(x: availableSize.width - sideInset - removeButtonSize.width, y: floor((size.height - removeButtonSize.height) * 0.5)), size: removeButtonSize) + if let removeButtonView = removeButton.view { + if removeButtonView.superview == nil { + removeButtonTransition = removeButtonTransition.withAnimation(.none) + self.addSubview(removeButtonView) + if !transition.animation.isImmediate { + transition.animateScale(view: removeButtonView, from: 0.001, to: 1.0) + Transition.easeInOut(duration: 0.2).animateAlpha(view: removeButtonView, from: 0.0, to: 1.0) + } + } + removeButtonTransition.setFrame(view: removeButtonView, frame: removeButtonFrame) + } + } + + if case .searching = component.content { + let shimmerEffectNode: ShimmerEffectNode + if let current = self.shimmerEffectNode { + shimmerEffectNode = current + } else { + shimmerEffectNode = ShimmerEffectNode() + self.shimmerEffectNode = shimmerEffectNode + self.addSubview(shimmerEffectNode.view) + } + + shimmerEffectNode.frame = CGRect(origin: CGPoint(), size: size) + shimmerEffectNode.updateAbsoluteRect(CGRect(origin: CGPoint(), size: size), within: size) + + var shapes: [ShimmerEffectNode.Shape] = [] + + let titleLineWidth: CGFloat = titleFrame.width + let subtitleLineWidth: CGFloat = subtitleFrame.width + let lineDiameter: CGFloat = 10.0 + + shapes.append(.circle(avatarFrame)) + + shapes.append(.roundedRectLine(startPoint: CGPoint(x: titleFrame.minX, y: titleFrame.minY + floor((titleFrame.height - lineDiameter) / 2.0)), width: titleLineWidth, diameter: lineDiameter)) + + shapes.append(.roundedRectLine(startPoint: CGPoint(x: subtitleFrame.minX, y: subtitleFrame.minY + floor((subtitleFrame.height - lineDiameter) / 2.0)), width: subtitleLineWidth, diameter: lineDiameter)) + + shimmerEffectNode.update(backgroundColor: component.theme.list.itemBlocksBackgroundColor, foregroundColor: component.theme.list.mediaPlaceholderColor, shimmeringColor: component.theme.list.itemBlocksBackgroundColor.withAlphaComponent(0.4), shapes: shapes, size: size) + } else { + if let shimmerEffectNode = self.shimmerEffectNode { + self.shimmerEffectNode = nil + shimmerEffectNode.view.removeFromSuperview() + } + } + + self.separatorInset = 16.0 + + return size + } + } + + func makeView() -> View { + return View(frame: CGRect()) + } + + func update(view: View, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + return view.update(component: self, availableSize: availableSize, state: state, environment: environment, transition: transition) + } +} diff --git a/submodules/TelegramUI/Components/Settings/ChatbotSetupScreen/Sources/ChatbotSetupScreen.swift b/submodules/TelegramUI/Components/Settings/ChatbotSetupScreen/Sources/ChatbotSetupScreen.swift index 44b119f8950..02dc866138f 100644 --- a/submodules/TelegramUI/Components/Settings/ChatbotSetupScreen/Sources/ChatbotSetupScreen.swift +++ b/submodules/TelegramUI/Components/Settings/ChatbotSetupScreen/Sources/ChatbotSetupScreen.swift @@ -21,6 +21,8 @@ import ListTextFieldItemComponent import BundleIconComponent import LottieComponent import Markdown +import PeerListItemComponent +import AvatarNode private let checkIcon: UIImage = { return generateImage(CGSize(width: 12.0, height: 10.0), rotatedContext: { size, context in @@ -39,11 +41,14 @@ final class ChatbotSetupScreenComponent: Component { typealias EnvironmentType = ViewControllerComponentContainer.Environment let context: AccountContext + let initialData: ChatbotSetupScreen.InitialData init( - context: AccountContext + context: AccountContext, + initialData: ChatbotSetupScreen.InitialData ) { self.context = context + self.initialData = initialData } static func ==(lhs: ChatbotSetupScreenComponent, rhs: ChatbotSetupScreenComponent) -> Bool { @@ -60,6 +65,49 @@ final class ChatbotSetupScreenComponent: Component { } } + private struct BotResolutionState: Equatable { + enum State: Equatable { + case searching + case notFound + case found(peer: EnginePeer, isInstalled: Bool) + } + + var query: String + var state: State + + init(query: String, state: State) { + self.query = query + self.state = state + } + } + + struct AdditionalPeerList { + enum Category: Int { + case newChats = 0 + case existingChats = 1 + case contacts = 2 + case nonContacts = 3 + } + + struct Peer { + var peer: EnginePeer + var isContact: Bool + + init(peer: EnginePeer, isContact: Bool) { + self.peer = peer + self.isContact = isContact + } + } + + var categories: Set + var peers: [Peer] + + init(categories: Set, peers: [Peer]) { + self.categories = categories + self.peers = peers + } + } + final class View: UIView, UIScrollViewDelegate { private let topOverscrollLayer = SimpleLayer() private let scrollView: ScrollView @@ -79,6 +127,19 @@ final class ChatbotSetupScreenComponent: Component { private var environment: EnvironmentType? private var chevronImage: UIImage? + private let textFieldTag = NSObject() + + private var botResolutionState: BotResolutionState? + private var botResolutionDisposable: Disposable? + private var resetQueryText: String? + + private var hasAccessToAllChatsByDefault: Bool = true + private var additionalPeerList = AdditionalPeerList( + categories: Set(), + peers: [] + ) + + private var replyToMessages: Bool = true override init(frame: CGRect) { self.scrollView = ScrollView() @@ -113,6 +174,39 @@ final class ChatbotSetupScreenComponent: Component { } func attemptNavigation(complete: @escaping () -> Void) -> Bool { + guard let component = self.component else { + return true + } + + var mappedCategories: TelegramBusinessRecipients.Categories = [] + if self.additionalPeerList.categories.contains(.existingChats) { + mappedCategories.insert(.existingChats) + } + if self.additionalPeerList.categories.contains(.newChats) { + mappedCategories.insert(.newChats) + } + if self.additionalPeerList.categories.contains(.contacts) { + mappedCategories.insert(.contacts) + } + if self.additionalPeerList.categories.contains(.nonContacts) { + mappedCategories.insert(.nonContacts) + } + let recipients = TelegramBusinessRecipients( + categories: mappedCategories, + additionalPeers: Set(self.additionalPeerList.peers.map(\.peer.id)), + exclude: self.hasAccessToAllChatsByDefault + ) + + if let botResolutionState = self.botResolutionState, case let .found(peer, isInstalled) = botResolutionState.state, isInstalled { + let _ = component.context.engine.accountData.setAccountConnectedBot(bot: TelegramAccountConnectedBot( + id: peer.id, + recipients: recipients, + canReply: self.replyToMessages + )).startStandalone() + } else { + let _ = component.context.engine.accountData.setAccountConnectedBot(bot: nil).startStandalone() + } + return true } @@ -150,12 +244,228 @@ final class ChatbotSetupScreenComponent: Component { } } + private func updateBotQuery(query: String) { + guard let component = self.component else { + return + } + + if !query.isEmpty { + if self.botResolutionState?.query != query { + let previousState = self.botResolutionState?.state + self.botResolutionState = BotResolutionState( + query: query, + state: self.botResolutionState?.state ?? .searching + ) + self.botResolutionDisposable?.dispose() + + if previousState != self.botResolutionState?.state { + self.state?.updated(transition: .spring(duration: 0.35)) + } + + self.botResolutionDisposable = (component.context.engine.peers.resolvePeerByName(name: query) + |> deliverOnMainQueue).start(next: { [weak self] result in + guard let self else { + return + } + switch result { + case .progress: + break + case let .result(peer): + let previousState = self.botResolutionState?.state + if let peer { + self.botResolutionState?.state = .found(peer: peer, isInstalled: false) + } else { + self.botResolutionState?.state = .notFound + } + if previousState != self.botResolutionState?.state { + self.state?.updated(transition: .spring(duration: 0.35)) + } + } + }) + } + } else { + if let botResolutionDisposable = self.botResolutionDisposable { + self.botResolutionDisposable = nil + botResolutionDisposable.dispose() + } + if self.botResolutionState != nil { + self.botResolutionState = nil + self.state?.updated(transition: .spring(duration: 0.35)) + } + } + } + + private func openAdditionalPeerListSetup() { + guard let component = self.component, let environment = self.environment else { + return + } + + enum AdditionalCategoryId: Int { + case existingChats + case newChats + case contacts + case nonContacts + } + + let additionalCategories: [ChatListNodeAdditionalCategory] = [ + ChatListNodeAdditionalCategory( + id: self.hasAccessToAllChatsByDefault ? AdditionalCategoryId.existingChats.rawValue : AdditionalCategoryId.newChats.rawValue, + icon: generateAvatarImage(size: CGSize(width: 40.0, height: 40.0), icon: generateTintedImage(image: UIImage(bundleImageName: self.hasAccessToAllChatsByDefault ? "Chat List/Filters/Chats" : "Chat List/Filters/NewChats"), color: .white), cornerRadius: 12.0, color: .purple), + smallIcon: generateAvatarImage(size: CGSize(width: 22.0, height: 22.0), icon: generateTintedImage(image: UIImage(bundleImageName: self.hasAccessToAllChatsByDefault ? "Chat List/Filters/Chats" : "Chat List/Filters/NewChats"), color: .white), iconScale: 0.6, cornerRadius: 6.0, circleCorners: true, color: .purple), + title: self.hasAccessToAllChatsByDefault ? environment.strings.BusinessMessageSetup_Recipients_CategoryExistingChats : environment.strings.BusinessMessageSetup_Recipients_CategoryNewChats + ), + ChatListNodeAdditionalCategory( + id: AdditionalCategoryId.contacts.rawValue, + icon: generateAvatarImage(size: CGSize(width: 40.0, height: 40.0), icon: generateTintedImage(image: UIImage(bundleImageName: "Chat List/Filters/Contact"), color: .white), cornerRadius: 12.0, color: .blue), + smallIcon: generateAvatarImage(size: CGSize(width: 22.0, height: 22.0), icon: generateTintedImage(image: UIImage(bundleImageName: "Chat List/Filters/Contact"), color: .white), iconScale: 0.6, cornerRadius: 6.0, circleCorners: true, color: .blue), + title: environment.strings.BusinessMessageSetup_Recipients_CategoryContacts + ), + ChatListNodeAdditionalCategory( + id: AdditionalCategoryId.nonContacts.rawValue, + icon: generateAvatarImage(size: CGSize(width: 40.0, height: 40.0), icon: generateTintedImage(image: UIImage(bundleImageName: "Chat List/Filters/User"), color: .white), cornerRadius: 12.0, color: .yellow), + smallIcon: generateAvatarImage(size: CGSize(width: 22.0, height: 22.0), icon: generateTintedImage(image: UIImage(bundleImageName: "Chat List/Filters/User"), color: .white), iconScale: 0.6, cornerRadius: 6.0, circleCorners: true, color: .yellow), + title: environment.strings.BusinessMessageSetup_Recipients_CategoryNonContacts + ) + ] + var selectedCategories = Set() + for category in self.additionalPeerList.categories { + switch category { + case .existingChats: + selectedCategories.insert(AdditionalCategoryId.existingChats.rawValue) + case .newChats: + selectedCategories.insert(AdditionalCategoryId.newChats.rawValue) + case .contacts: + selectedCategories.insert(AdditionalCategoryId.contacts.rawValue) + case .nonContacts: + selectedCategories.insert(AdditionalCategoryId.nonContacts.rawValue) + } + } + + let controller = component.context.sharedContext.makeContactMultiselectionController(ContactMultiselectionControllerParams(context: component.context, mode: .chatSelection(ContactMultiselectionControllerMode.ChatSelection( + title: self.hasAccessToAllChatsByDefault ? environment.strings.BusinessMessageSetup_Recipients_ExcludeSearchTitle : environment.strings.BusinessMessageSetup_Recipients_IncludeSearchTitle, + searchPlaceholder: environment.strings.ChatListFilter_AddChatsSearchPlaceholder, + selectedChats: Set(self.additionalPeerList.peers.map(\.peer.id)), + additionalCategories: ContactMultiselectionControllerAdditionalCategories(categories: additionalCategories, selectedCategories: selectedCategories), + chatListFilters: nil, + onlyUsers: true + )), options: [], filters: [], alwaysEnabled: true, limit: 100, reachedLimit: { _ in + })) + controller.navigationPresentation = .modal + + let _ = (controller.result + |> take(1) + |> deliverOnMainQueue).startStandalone(next: { [weak self, weak controller] result in + guard let self, let component = self.component, case let .result(rawPeerIds, additionalCategoryIds) = result else { + controller?.dismiss() + return + } + + let peerIds = rawPeerIds.compactMap { id -> EnginePeer.Id? in + switch id { + case let .peer(id): + return id + case .deviceContact: + return nil + } + } + + let _ = (component.context.engine.data.get( + EngineDataMap( + peerIds.map(TelegramEngine.EngineData.Item.Peer.Peer.init(id:)) + ), + EngineDataMap( + peerIds.map(TelegramEngine.EngineData.Item.Peer.IsContact.init(id:)) + ) + ) + |> deliverOnMainQueue).start(next: { [weak self] peerMap, isContactMap in + guard let self else { + return + } + + let mappedCategories = additionalCategoryIds.compactMap { item -> AdditionalPeerList.Category? in + switch item { + case AdditionalCategoryId.existingChats.rawValue: + return .existingChats + case AdditionalCategoryId.newChats.rawValue: + return .newChats + case AdditionalCategoryId.contacts.rawValue: + return .contacts + case AdditionalCategoryId.nonContacts.rawValue: + return .nonContacts + default: + return nil + } + } + + self.additionalPeerList.categories = Set(mappedCategories) + + self.additionalPeerList.peers.removeAll() + for id in peerIds { + guard let maybePeer = peerMap[id], let peer = maybePeer else { + continue + } + self.additionalPeerList.peers.append(AdditionalPeerList.Peer( + peer: peer, + isContact: isContactMap[id] ?? false + )) + } + self.additionalPeerList.peers.sort(by: { lhs, rhs in + return lhs.peer.debugDisplayTitle < rhs.peer.debugDisplayTitle + }) + self.state?.updated(transition: .immediate) + + controller?.dismiss() + }) + }) + + self.environment?.controller()?.push(controller) + } + func update(component: ChatbotSetupScreenComponent, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { self.isUpdating = true defer { self.isUpdating = false } + if self.component == nil { + if let bot = component.initialData.bot, let botPeer = component.initialData.botPeer, let addressName = botPeer.addressName { + self.botResolutionState = BotResolutionState(query: addressName, state: .found(peer: botPeer, isInstalled: true)) + self.resetQueryText = addressName.lowercased() + + self.replyToMessages = bot.canReply + + let initialRecipients = bot.recipients + + var mappedCategories = Set() + if initialRecipients.categories.contains(.existingChats) { + mappedCategories.insert(.existingChats) + } + if initialRecipients.categories.contains(.newChats) { + mappedCategories.insert(.newChats) + } + if initialRecipients.categories.contains(.contacts) { + mappedCategories.insert(.contacts) + } + if initialRecipients.categories.contains(.nonContacts) { + mappedCategories.insert(.nonContacts) + } + + var additionalPeers: [AdditionalPeerList.Peer] = [] + for peerId in initialRecipients.additionalPeers { + if let peer = component.initialData.additionalPeers[peerId] { + additionalPeers.append(peer) + } + } + + self.additionalPeerList = AdditionalPeerList( + categories: mappedCategories, + peers: additionalPeers + ) + + self.hasAccessToAllChatsByDefault = initialRecipients.exclude + } + } + let environment = environment[EnvironmentType.self].value let themeUpdated = self.environment?.theme !== environment.theme self.environment = environment @@ -169,11 +479,10 @@ final class ChatbotSetupScreenComponent: Component { let presentationData = component.context.sharedContext.currentPresentationData.with { $0 } - //TODO:localize let navigationTitleSize = self.navigationTitle.update( transition: transition, component: AnyComponent(MultilineTextComponent( - text: .plain(NSAttributedString(string: "Chatbots", font: Font.semibold(17.0), textColor: environment.theme.rootController.navigationBar.primaryTextColor)), + text: .plain(NSAttributedString(string: environment.strings.ChatbotSetup_Title, font: Font.semibold(17.0), textColor: environment.theme.rootController.navigationBar.primaryTextColor)), horizontalAlignment: .center )), environment: {}, @@ -204,15 +513,16 @@ final class ChatbotSetupScreenComponent: Component { transition: .immediate, component: AnyComponent(LottieComponent( content: LottieComponent.AppBundleContent(name: "BotEmoji"), - loop: true + loop: false )), environment: {}, containerSize: CGSize(width: 100.0, height: 100.0) ) - let iconFrame = CGRect(origin: CGPoint(x: floor((availableSize.width - iconSize.width) * 0.5), y: contentHeight + 2.0), size: iconSize) - if let iconView = self.icon.view { + let iconFrame = CGRect(origin: CGPoint(x: floor((availableSize.width - iconSize.width) * 0.5), y: contentHeight + 8.0), size: iconSize) + if let iconView = self.icon.view as? LottieComponent.View { if iconView.superview == nil { self.scrollView.addSubview(iconView) + iconView.playOnce() } transition.setPosition(view: iconView, position: iconFrame.center) iconView.bounds = CGRect(origin: CGPoint(), size: iconFrame.size) @@ -220,8 +530,7 @@ final class ChatbotSetupScreenComponent: Component { contentHeight += 129.0 - //TODO:localize - let subtitleString = NSMutableAttributedString(attributedString: parseMarkdownIntoAttributedString("Add a bot to your account to help you automatically process and respond to the messages you receive. [Learn More>]()", attributes: MarkdownAttributes( + let subtitleString = NSMutableAttributedString(attributedString: parseMarkdownIntoAttributedString(environment.strings.ChatbotSetup_Text, attributes: MarkdownAttributes( body: MarkdownAttributeSet(font: Font.regular(15.0), textColor: environment.theme.list.freeTextColor), bold: MarkdownAttributeSet(font: Font.semibold(15.0), textColor: environment.theme.list.freeTextColor), link: MarkdownAttributeSet(font: Font.regular(15.0), textColor: environment.theme.list.itemAccentColor), @@ -236,10 +545,9 @@ final class ChatbotSetupScreenComponent: Component { subtitleString.addAttribute(.attachment, value: chevronImage, range: NSRange(range, in: subtitleString.string)) } - //TODO:localize let subtitleSize = self.subtitle.update( transition: .immediate, - component: AnyComponent(MultilineTextComponent( + component: AnyComponent(BalancedTextComponent( text: .plain(subtitleString), horizontalAlignment: .center, maximumNumberOfLines: 0, @@ -253,10 +561,10 @@ final class ChatbotSetupScreenComponent: Component { } }, tapAction: { [weak self] _, _ in - guard let self, let component = self.component else { + guard let self, let component = self.component, let environment = self.environment else { return } - let _ = component + component.context.sharedContext.applicationBindings.openUrl(environment.strings.ChatbotSetup_TextLink) } )), environment: {}, @@ -273,7 +581,69 @@ final class ChatbotSetupScreenComponent: Component { contentHeight += subtitleSize.height contentHeight += 27.0 - //TODO:localize + let resetQueryText = self.resetQueryText + self.resetQueryText = nil + var nameSectionItems: [AnyComponentWithIdentity] = [] + nameSectionItems.append(AnyComponentWithIdentity(id: 0, component: AnyComponent(ListTextFieldItemComponent( + theme: environment.theme, + initialText: "", + resetText: resetQueryText.flatMap { ListTextFieldItemComponent.ResetText(value: $0) }, + placeholder: environment.strings.ChatbotSetup_BotSearchPlaceholder, + autocapitalizationType: .none, + autocorrectionType: .no, + updated: { [weak self] value in + guard let self else { + return + } + self.updateBotQuery(query: value) + }, + tag: self.textFieldTag + )))) + if let botResolutionState = self.botResolutionState { + let mappedContent: ChatbotSearchResultItemComponent.Content + switch botResolutionState.state { + case .searching: + mappedContent = .searching + case .notFound: + mappedContent = .notFound + case let .found(peer, isInstalled): + mappedContent = .found(peer: peer, isInstalled: isInstalled) + } + nameSectionItems.append(AnyComponentWithIdentity(id: 1, component: AnyComponent(ChatbotSearchResultItemComponent( + context: component.context, + theme: environment.theme, + strings: environment.strings, + content: mappedContent, + installAction: { [weak self] in + guard let self else { + return + } + if var botResolutionState = self.botResolutionState, case let .found(peer, isInstalled) = botResolutionState.state, !isInstalled { + botResolutionState.state = .found(peer: peer, isInstalled: true) + self.botResolutionState = botResolutionState + self.state?.updated(transition: .spring(duration: 0.3)) + } + }, + removeAction: { [weak self] in + guard let self else { + return + } + if let botResolutionState = self.botResolutionState, case let .found(_, isInstalled) = botResolutionState.state, isInstalled { + self.botResolutionState = nil + if let botResolutionDisposable = self.botResolutionDisposable { + self.botResolutionDisposable = nil + botResolutionDisposable.dispose() + } + + if let textFieldView = self.nameSection.findTaggedView(tag: self.textFieldTag) as? ListTextFieldItemComponent.View { + textFieldView.setText(text: "", updateState: false) + } + self.state?.updated(transition: .spring(duration: 0.3)) + } + } + )))) + } + let nameSectionSize = self.nameSection.update( transition: transition, component: AnyComponent(ListSectionComponent( @@ -281,21 +651,13 @@ final class ChatbotSetupScreenComponent: Component { header: nil, footer: AnyComponent(MultilineTextComponent( text: .plain(NSAttributedString( - string: "Enter the username or URL of the Telegram bot that you want to automatically process your chats.", + string: environment.strings.ChatbotSetup_BotSectionFooter, font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), textColor: environment.theme.list.freeTextColor )), maximumNumberOfLines: 0 )), - items: [ - AnyComponentWithIdentity(id: 0, component: AnyComponent(ListTextFieldItemComponent( - theme: environment.theme, - initialText: "", - placeholder: "Bot Username", - updated: { value in - } - ))) - ] + items: nameSectionItems )), environment: {}, containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 10000.0) @@ -310,14 +672,13 @@ final class ChatbotSetupScreenComponent: Component { contentHeight += nameSectionSize.height contentHeight += sectionSpacing - //TODO:localize let accessSectionSize = self.accessSection.update( transition: transition, component: AnyComponent(ListSectionComponent( theme: environment.theme, header: AnyComponent(MultilineTextComponent( text: .plain(NSAttributedString( - string: "CHATS ACCESSIBLE FOR THE BOT", + string: environment.strings.ChatbotSetup_RecipientsSectionHeader, font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), textColor: environment.theme.list.freeTextColor )), @@ -330,7 +691,7 @@ final class ChatbotSetupScreenComponent: Component { title: AnyComponent(VStack([ AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( text: .plain(NSAttributedString( - string: "All 1-to-1 Chats Except...", + string: environment.strings.BusinessMessageSetup_RecipientsOptionAllExcept, font: Font.regular(presentationData.listsFontSize.baseDisplaySize), textColor: environment.theme.list.itemPrimaryTextColor )), @@ -339,11 +700,20 @@ final class ChatbotSetupScreenComponent: Component { ], alignment: .left, spacing: 2.0)), leftIcon: AnyComponentWithIdentity(id: 0, component: AnyComponent(Image( image: checkIcon, - tintColor: environment.theme.list.itemAccentColor, + tintColor: !self.hasAccessToAllChatsByDefault ? .clear : environment.theme.list.itemAccentColor, contentMode: .center ))), accessory: nil, - action: { _ in + action: { [weak self] _ in + guard let self else { + return + } + if !self.hasAccessToAllChatsByDefault { + self.hasAccessToAllChatsByDefault = true + self.additionalPeerList.categories.removeAll() + self.additionalPeerList.peers.removeAll() + self.state?.updated(transition: .immediate) + } } ))), AnyComponentWithIdentity(id: 1, component: AnyComponent(ListActionItemComponent( @@ -351,7 +721,7 @@ final class ChatbotSetupScreenComponent: Component { title: AnyComponent(VStack([ AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( text: .plain(NSAttributedString( - string: "Only Selected Chats", + string: environment.strings.BusinessMessageSetup_RecipientsOptionOnly, font: Font.regular(presentationData.listsFontSize.baseDisplaySize), textColor: environment.theme.list.itemPrimaryTextColor )), @@ -360,11 +730,20 @@ final class ChatbotSetupScreenComponent: Component { ], alignment: .left, spacing: 2.0)), leftIcon: AnyComponentWithIdentity(id: 0, component: AnyComponent(Image( image: checkIcon, - tintColor: .clear, + tintColor: self.hasAccessToAllChatsByDefault ? .clear : environment.theme.list.itemAccentColor, contentMode: .center ))), accessory: nil, - action: { _ in + action: { [weak self] _ in + guard let self else { + return + } + if self.hasAccessToAllChatsByDefault { + self.hasAccessToAllChatsByDefault = false + self.additionalPeerList.categories.removeAll() + self.additionalPeerList.peers.removeAll() + self.state?.updated(transition: .immediate) + } } ))) ] @@ -382,49 +761,149 @@ final class ChatbotSetupScreenComponent: Component { contentHeight += accessSectionSize.height contentHeight += sectionSpacing - //TODO:localize + var excludedSectionItems: [AnyComponentWithIdentity] = [] + excludedSectionItems.append(AnyComponentWithIdentity(id: 0, component: AnyComponent(ListActionItemComponent( + theme: environment.theme, + title: AnyComponent(VStack([ + AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString( + string: self.hasAccessToAllChatsByDefault ? environment.strings.BusinessMessageSetup_Recipients_AddExclude : environment.strings.BusinessMessageSetup_Recipients_AddInclude, + font: Font.regular(presentationData.listsFontSize.baseDisplaySize), + textColor: environment.theme.list.itemAccentColor + )), + maximumNumberOfLines: 1 + ))), + ], alignment: .left, spacing: 2.0)), + leftIcon: AnyComponentWithIdentity(id: 0, component: AnyComponent(BundleIconComponent( + name: "Chat List/AddIcon", + tintColor: environment.theme.list.itemAccentColor + ))), + accessory: nil, + action: { [weak self] _ in + guard let self else { + return + } + self.openAdditionalPeerListSetup() + } + )))) + for category in self.additionalPeerList.categories.sorted(by: { $0.rawValue < $1.rawValue }) { + let title: String + let icon: String + let color: AvatarBackgroundColor + switch category { + case .newChats: + title = environment.strings.BusinessMessageSetup_Recipients_CategoryNewChats + icon = "Chat List/Filters/NewChats" + color = .purple + case .existingChats: + title = environment.strings.BusinessMessageSetup_Recipients_CategoryExistingChats + icon = "Chat List/Filters/Chats" + color = .purple + case .contacts: + title = environment.strings.BusinessMessageSetup_Recipients_CategoryContacts + icon = "Chat List/Filters/Contact" + color = .blue + case .nonContacts: + title = environment.strings.BusinessMessageSetup_Recipients_CategoryNonContacts + icon = "Chat List/Filters/User" + color = .yellow + } + excludedSectionItems.append(AnyComponentWithIdentity(id: category, component: AnyComponent(PeerListItemComponent( + context: component.context, + theme: environment.theme, + strings: environment.strings, + style: .generic, + sideInset: 0.0, + title: title, + avatar: PeerListItemComponent.Avatar( + icon: icon, + color: color, + clipStyle: .roundedRect + ), + peer: nil, + subtitle: nil, + subtitleAccessory: .none, + presence: nil, + selectionState: .none, + hasNext: false, + action: { peer, _, _ in + }, + inlineActions: PeerListItemComponent.InlineActionsState( + actions: [PeerListItemComponent.InlineAction( + id: AnyHashable(0), + title: environment.strings.Common_Delete, + color: .destructive, + action: { [weak self] in + guard let self else { + return + } + self.additionalPeerList.categories.remove(category) + self.state?.updated(transition: .spring(duration: 0.4)) + } + )] + ) + )))) + } + for peer in self.additionalPeerList.peers { + excludedSectionItems.append(AnyComponentWithIdentity(id: peer.peer.id, component: AnyComponent(PeerListItemComponent( + context: component.context, + theme: environment.theme, + strings: environment.strings, + style: .generic, + sideInset: 0.0, + title: peer.peer.displayTitle(strings: environment.strings, displayOrder: .firstLast), + peer: peer.peer, + subtitle: peer.isContact ? environment.strings.ChatList_PeerTypeContact : environment.strings.ChatList_PeerTypeNonContact, + subtitleAccessory: .none, + presence: nil, + selectionState: .none, + hasNext: false, + action: { peer, _, _ in + }, + inlineActions: PeerListItemComponent.InlineActionsState( + actions: [PeerListItemComponent.InlineAction( + id: AnyHashable(0), + title: environment.strings.Common_Delete, + color: .destructive, + action: { [weak self] in + guard let self else { + return + } + self.additionalPeerList.peers.removeAll(where: { $0.peer.id == peer.peer.id }) + self.state?.updated(transition: .spring(duration: 0.4)) + } + )] + ) + )))) + } + let excludedSectionSize = self.excludedSection.update( transition: transition, component: AnyComponent(ListSectionComponent( theme: environment.theme, header: AnyComponent(MultilineTextComponent( text: .plain(NSAttributedString( - string: "EXCLUDED CHATS", + string: self.hasAccessToAllChatsByDefault ? environment.strings.BusinessMessageSetup_Recipients_ExcludedSectionHeader : environment.strings.BusinessMessageSetup_Recipients_IncludedSectionHeader, font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), textColor: environment.theme.list.freeTextColor )), maximumNumberOfLines: 0 )), footer: AnyComponent(MultilineTextComponent( - text: .plain(NSAttributedString( - string: "Select chats or entire chat categories which the bot WILL NOT have access to.", - font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), - textColor: environment.theme.list.freeTextColor - )), + text: .markdown( + text: self.hasAccessToAllChatsByDefault ? environment.strings.ChatbotSetup_Recipients_ExcludedSectionFooter : environment.strings.ChatbotSetup_Recipients_IncludedSectionFooter, + attributes: MarkdownAttributes( + body: MarkdownAttributeSet(font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), textColor: environment.theme.list.freeTextColor), + bold: MarkdownAttributeSet(font: Font.semibold(presentationData.listsFontSize.itemListBaseHeaderFontSize), textColor: environment.theme.list.freeTextColor), + link: MarkdownAttributeSet(font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), textColor: environment.theme.list.itemAccentColor), + linkAttribute: { _ in + return nil + } + ) + ), maximumNumberOfLines: 0 )), - items: [ - AnyComponentWithIdentity(id: 0, component: AnyComponent(ListActionItemComponent( - theme: environment.theme, - title: AnyComponent(VStack([ - AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( - text: .plain(NSAttributedString( - string: "Exclude Chats...", - font: Font.regular(presentationData.listsFontSize.baseDisplaySize), - textColor: environment.theme.list.itemAccentColor - )), - maximumNumberOfLines: 1 - ))), - ], alignment: .left, spacing: 2.0)), - leftIcon: AnyComponentWithIdentity(id: 0, component: AnyComponent(BundleIconComponent( - name: "Chat List/AddIcon", - tintColor: environment.theme.list.itemAccentColor - ))), - accessory: nil, - action: { _ in - } - ))), - ] + items: excludedSectionItems )), environment: {}, containerSize: CGSize(width: availableSize.width - sideInset * 2.0, height: 10000.0) @@ -439,14 +918,13 @@ final class ChatbotSetupScreenComponent: Component { contentHeight += excludedSectionSize.height contentHeight += sectionSpacing - //TODO:localize let permissionsSectionSize = self.permissionsSection.update( transition: transition, component: AnyComponent(ListSectionComponent( theme: environment.theme, header: AnyComponent(MultilineTextComponent( text: .plain(NSAttributedString( - string: "BOT PERMISSIONS", + string: environment.strings.ChatbotSetup_PermissionsSectionHeader, font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), textColor: environment.theme.list.freeTextColor )), @@ -454,7 +932,7 @@ final class ChatbotSetupScreenComponent: Component { )), footer: AnyComponent(MultilineTextComponent( text: .plain(NSAttributedString( - string: "The bot will be able to view all new incoming messages, but not the messages that had been sent before you added the bot.", + string: environment.strings.ChatbotSetup_PermissionsSectionFooter, font: Font.regular(presentationData.listsFontSize.itemListBaseHeaderFontSize), textColor: environment.theme.list.freeTextColor )), @@ -466,14 +944,20 @@ final class ChatbotSetupScreenComponent: Component { title: AnyComponent(VStack([ AnyComponentWithIdentity(id: AnyHashable(0), component: AnyComponent(MultilineTextComponent( text: .plain(NSAttributedString( - string: "Reply to Messages", + string: environment.strings.ChatbotSetup_Permission_ReplyToMessages, font: Font.regular(presentationData.listsFontSize.baseDisplaySize), textColor: environment.theme.list.itemPrimaryTextColor )), maximumNumberOfLines: 1 ))), ], alignment: .left, spacing: 2.0)), - accessory: .toggle(true), + accessory: .toggle(ListActionItemComponent.Toggle(style: .icons, isOn: self.replyToMessages, action: { [weak self] _ in + guard let self else { + return + } + self.replyToMessages = !self.replyToMessages + self.state?.updated(transition: .spring(duration: 0.4)) + })), action: nil ))), ] @@ -533,13 +1017,30 @@ final class ChatbotSetupScreenComponent: Component { } public final class ChatbotSetupScreen: ViewControllerComponentContainer { + public final class InitialData: ChatbotSetupScreenInitialData { + fileprivate let bot: TelegramAccountConnectedBot? + fileprivate let botPeer: EnginePeer? + fileprivate let additionalPeers: [EnginePeer.Id: ChatbotSetupScreenComponent.AdditionalPeerList.Peer] + + fileprivate init( + bot: TelegramAccountConnectedBot?, + botPeer: EnginePeer?, + additionalPeers: [EnginePeer.Id: ChatbotSetupScreenComponent.AdditionalPeerList.Peer] + ) { + self.bot = bot + self.botPeer = botPeer + self.additionalPeers = additionalPeers + } + } + private let context: AccountContext - public init(context: AccountContext) { + public init(context: AccountContext, initialData: InitialData) { self.context = context super.init(context: context, component: ChatbotSetupScreenComponent( - context: context + context: context, + initialData: initialData ), navigationBarAppearance: .default, theme: .default, updatedPresentationData: nil) let presentationData = context.sharedContext.currentPresentationData.with { $0 } @@ -576,4 +1077,48 @@ public final class ChatbotSetupScreen: ViewControllerComponentContainer { override public func containerLayoutUpdated(_ layout: ContainerViewLayout, transition: ContainedViewLayoutTransition) { super.containerLayoutUpdated(layout, transition: transition) } + + public static func initialData(context: AccountContext) -> Signal { + return context.engine.data.get( + TelegramEngine.EngineData.Item.Peer.BusinessConnectedBot(id: context.account.peerId) + ) + |> mapToSignal { connectedBot -> Signal in + guard let connectedBot else { + return .single( + InitialData( + bot: nil, + botPeer: nil, + additionalPeers: [:] + ) + ) + } + + var additionalPeerIds = Set() + additionalPeerIds.formUnion(connectedBot.recipients.additionalPeers) + + return context.engine.data.get( + TelegramEngine.EngineData.Item.Peer.Peer(id: connectedBot.id), + EngineDataMap(additionalPeerIds.map(TelegramEngine.EngineData.Item.Peer.Peer.init(id:))), + EngineDataMap(additionalPeerIds.map(TelegramEngine.EngineData.Item.Peer.IsContact.init(id:))) + ) + |> map { botPeer, peers, isContacts -> ChatbotSetupScreenInitialData in + var additionalPeers: [EnginePeer.Id: ChatbotSetupScreenComponent.AdditionalPeerList.Peer] = [:] + for id in additionalPeerIds { + guard let peer = peers[id], let peer else { + continue + } + additionalPeers[id] = ChatbotSetupScreenComponent.AdditionalPeerList.Peer( + peer: peer, + isContact: isContacts[id] ?? false + ) + } + + return InitialData( + bot: connectedBot, + botPeer: botPeer, + additionalPeers: additionalPeers + ) + } + } + } } diff --git a/submodules/TelegramUI/Components/Settings/PeerNameColorItem/Sources/PeerNameColorItem.swift b/submodules/TelegramUI/Components/Settings/PeerNameColorItem/Sources/PeerNameColorItem.swift index 0df4e747c14..95cfb9911da 100644 --- a/submodules/TelegramUI/Components/Settings/PeerNameColorItem/Sources/PeerNameColorItem.swift +++ b/submodules/TelegramUI/Components/Settings/PeerNameColorItem/Sources/PeerNameColorItem.swift @@ -13,33 +13,35 @@ import AccountContext import ListItemComponentAdaptor private class PeerNameColorIconItem { - let index: PeerNameColor - let colors: PeerNameColors.Colors + let index: PeerNameColor? + let colors: PeerNameColors.Colors? let isDark: Bool let selected: Bool - let action: (PeerNameColor) -> Void + let isLocked: Bool + let action: (PeerNameColor?) -> Void - public init(index: PeerNameColor, colors: PeerNameColors.Colors, isDark: Bool, selected: Bool, action: @escaping (PeerNameColor) -> Void) { + public init(index: PeerNameColor?, colors: PeerNameColors.Colors?, isDark: Bool, selected: Bool, isLocked: Bool, action: @escaping (PeerNameColor?) -> Void) { self.index = index self.colors = colors self.isDark = isDark self.selected = selected + self.isLocked = isLocked self.action = action } } -private func generateRingImage(nameColor: PeerNameColors.Colors, size: CGSize = CGSize(width: 40.0, height: 40.0)) -> UIImage? { +private func generateRingImage(color: UIColor, size: CGSize = CGSize(width: 40.0, height: 40.0)) -> UIImage? { return generateImage(size, rotatedContext: { size, context in let bounds = CGRect(origin: CGPoint(), size: size) context.clear(bounds) - context.setStrokeColor(nameColor.main.cgColor) + context.setStrokeColor(color.cgColor) context.setLineWidth(2.0) context.strokeEllipse(in: bounds.insetBy(dx: 1.0, dy: 1.0)) }) } -public func generatePeerNameColorImage(nameColor: PeerNameColors.Colors, isDark: Bool, bounds: CGSize = CGSize(width: 40.0, height: 40.0), size: CGSize = CGSize(width: 40.0, height: 40.0)) -> UIImage? { +public func generatePeerNameColorImage(nameColor: PeerNameColors.Colors?, isDark: Bool, isLocked: Bool = false, isEmpty: Bool = false, bounds: CGSize = CGSize(width: 40.0, height: 40.0), size: CGSize = CGSize(width: 40.0, height: 40.0)) -> UIImage? { return generateImage(bounds, rotatedContext: { contextSize, context in let bounds = CGRect(origin: CGPoint(), size: contextSize) context.clear(bounds) @@ -48,7 +50,7 @@ public func generatePeerNameColorImage(nameColor: PeerNameColors.Colors, isDark: context.addEllipse(in: circleBounds) context.clip() - if let secondColor = nameColor.secondary { + if let nameColor, let secondColor = nameColor.secondary { var firstColor = nameColor.main var secondColor = secondColor if isDark, nameColor.tertiary == nil { @@ -84,9 +86,51 @@ public func generatePeerNameColorImage(nameColor: PeerNameColors.Colors, isDark: context.setFillColor(firstColor.cgColor) context.fillPath() } - } else { + } else if let nameColor { context.setFillColor(nameColor.main.cgColor) context.fill(circleBounds) + } else { + context.setFillColor(UIColor(rgb: 0x798896).cgColor) + context.fill(circleBounds) + } + + if isLocked { + if let image = UIImage(bundleImageName: "Chat/Input/Accessory Panels/TextLockIcon") { + let scaleFactor: CGFloat = 1.58 + let imageSize = CGSize(width: floor(image.size.width * scaleFactor), height: floor(image.size.height * scaleFactor)) + var imageFrame = CGRect(origin: CGPoint(x: circleBounds.minX + floor((circleBounds.width - imageSize.width) * 0.5), y: circleBounds.minY + floor((circleBounds.height - imageSize.height) * 0.5)), size: imageSize) + imageFrame.origin.y += -0.5 + + context.translateBy(x: imageFrame.midX, y: imageFrame.midY) + context.scaleBy(x: 1.0, y: -1.0) + context.translateBy(x: -imageFrame.midX, y: -imageFrame.midY) + + if let cgImage = image.cgImage { + context.clip(to: imageFrame, mask: cgImage) + context.setFillColor(UIColor.clear.cgColor) + context.setBlendMode(.copy) + context.fill(imageFrame) + } + } + } else if isEmpty { + if let image = UIImage(bundleImageName: "Chat/Message/SideCloseIcon") { + let scaleFactor: CGFloat = 1.0 + let imageSize = CGSize(width: floor(image.size.width * scaleFactor), height: floor(image.size.height * scaleFactor)) + var imageFrame = CGRect(origin: CGPoint(x: circleBounds.minX + floor((circleBounds.width - imageSize.width) * 0.5), y: circleBounds.minY + floor((circleBounds.height - imageSize.height) * 0.5)), size: imageSize) + imageFrame.origin.y += 0.5 + imageFrame.origin.x += 0.5 + + context.translateBy(x: imageFrame.midX, y: imageFrame.midY) + context.scaleBy(x: 1.0, y: -1.0) + context.translateBy(x: -imageFrame.midX, y: -imageFrame.midY) + + if let cgImage = image.cgImage { + context.clip(to: imageFrame, mask: cgImage) + context.setFillColor(UIColor.clear.cgColor) + context.setBlendMode(.copy) + context.fill(imageFrame) + } + } } }) } @@ -187,8 +231,8 @@ private final class PeerNameColorIconItemNode : ASDisplayNode { self.item = item if updatedAccentColor { - self.fillNode.image = generatePeerNameColorImage(nameColor: item.colors, isDark: item.isDark, bounds: size, size: size) - self.ringNode.image = generateRingImage(nameColor: item.colors, size: size) + self.fillNode.image = generatePeerNameColorImage(nameColor: item.colors, isDark: item.isDark, isLocked: item.selected && item.isLocked, isEmpty: item.colors == nil, bounds: size, size: size) + self.ringNode.image = generateRingImage(color: item.colors?.main ?? UIColor(rgb: 0x798896), size: size) } let center = CGPoint(x: size.width / 2.0, y: size.height / 2.0) @@ -208,19 +252,29 @@ private final class PeerNameColorIconItemNode : ASDisplayNode { } public final class PeerNameColorItem: ListViewItem, ItemListItem, ListItemComponentAdaptor.ItemGenerator { + public enum Mode { + case name + case profile + case folderTag + } + public var sectionId: ItemListSectionId public let theme: PresentationTheme public let colors: PeerNameColors - public let isProfile: Bool + public let mode: Mode + public let displayEmptyColor: Bool + public let isLocked: Bool public let currentColor: PeerNameColor? - public let updated: (PeerNameColor) -> Void + public let updated: (PeerNameColor?) -> Void public let tag: ItemListItemTag? - public init(theme: PresentationTheme, colors: PeerNameColors, isProfile: Bool, currentColor: PeerNameColor?, updated: @escaping (PeerNameColor) -> Void, tag: ItemListItemTag? = nil, sectionId: ItemListSectionId) { + public init(theme: PresentationTheme, colors: PeerNameColors, mode: Mode, displayEmptyColor: Bool = false, currentColor: PeerNameColor?, isLocked: Bool = false, updated: @escaping (PeerNameColor?) -> Void, tag: ItemListItemTag? = nil, sectionId: ItemListSectionId) { self.theme = theme self.colors = colors - self.isProfile = isProfile + self.mode = mode + self.displayEmptyColor = displayEmptyColor + self.isLocked = isLocked self.currentColor = currentColor self.updated = updated self.tag = tag @@ -271,7 +325,7 @@ public final class PeerNameColorItem: ListViewItem, ItemListItem, ListItemCompon if lhs.colors != rhs.colors { return false } - if lhs.isProfile != rhs.isProfile { + if lhs.mode != rhs.mode { return false } if lhs.currentColor != rhs.currentColor { @@ -334,15 +388,24 @@ public final class PeerNameColorItemNode: ListViewItemNode, ItemListItemNode { let itemsPerRow: Int let displayOrder: [Int32] - if item.isProfile { - displayOrder = item.colors.profileDisplayOrder - itemsPerRow = 8 - } else { + switch item.mode { + case .name: displayOrder = item.colors.displayOrder itemsPerRow = 7 + case .profile: + displayOrder = item.colors.profileDisplayOrder + itemsPerRow = 8 + case .folderTag: + displayOrder = item.colors.chatFolderTagDisplayOrder + itemsPerRow = 8 + } + + var numItems = displayOrder.count + if item.displayEmptyColor { + numItems += 1 } - let rowsCount = ceil(CGFloat(displayOrder.count) / CGFloat(itemsPerRow)) + let rowsCount = ceil(CGFloat(numItems) / CGFloat(itemsPerRow)) contentSize = CGSize(width: params.width, height: 48.0 * rowsCount) insets = itemListNeighborsGroupedInsets(neighbors, params) @@ -419,22 +482,30 @@ public final class PeerNameColorItemNode: ListViewItemNode, ItemListItemNode { 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)) - let action: (PeerNameColor) -> Void = { color in + let action: (PeerNameColor?) -> Void = { color in item.updated(color) } var items: [PeerNameColorIconItem] = [] var i: Int = 0 + for index in displayOrder { let color = PeerNameColor(rawValue: index) let colors: PeerNameColors.Colors - if item.isProfile { - colors = item.colors.getProfile(color, dark: item.theme.overallDarkAppearance, subject: .palette) - } else { + switch item.mode { + case .name: colors = item.colors.get(color, dark: item.theme.overallDarkAppearance) + case .profile: + colors = item.colors.getProfile(color, dark: item.theme.overallDarkAppearance, subject: .palette) + case .folderTag: + colors = item.colors.getChatFolderTag(color, dark: item.theme.overallDarkAppearance) } - items.append(PeerNameColorIconItem(index: color, colors: colors, isDark: item.theme.overallDarkAppearance, selected: color == item.currentColor, action: action)) + items.append(PeerNameColorIconItem(index: color, colors: colors, isDark: item.theme.overallDarkAppearance, selected: color == item.currentColor, isLocked: item.isLocked, action: action)) + i += 1 + } + if item.displayEmptyColor { + items.append(PeerNameColorIconItem(index: nil, colors: nil, isDark: item.theme.overallDarkAppearance, selected: item.currentColor == nil, isLocked: item.isLocked, action: action)) i += 1 } strongSelf.items = items @@ -442,18 +513,24 @@ public final class PeerNameColorItemNode: ListViewItemNode, ItemListItemNode { let sideInset: CGFloat = params.leftInset + 10.0 let iconSize = CGSize(width: 32.0, height: 32.0) - let spacing = (params.width - sideInset * 2.0 - iconSize.width * CGFloat(itemsPerRow)) / CGFloat(itemsPerRow - 1) + let spacing = floorToScreenPixels((params.width - sideInset * 2.0 - iconSize.width * CGFloat(itemsPerRow)) / CGFloat(itemsPerRow - 1)) var origin = CGPoint(x: sideInset, y: 10.0) i = 0 for item in items { let iconItemNode: PeerNameColorIconItemNode - if let current = strongSelf.itemNodes[item.index.rawValue] { + let indexKey: Int32 + if let index = item.index { + indexKey = index.rawValue + } else { + indexKey = Int32.min + } + if let current = strongSelf.itemNodes[indexKey] { iconItemNode = current } else { iconItemNode = PeerNameColorIconItemNode() - strongSelf.itemNodes[item.index.rawValue] = iconItemNode + strongSelf.itemNodes[indexKey] = iconItemNode strongSelf.containerNode.addSubnode(iconItemNode) } diff --git a/submodules/TelegramUI/Components/Settings/PeerNameColorScreen/Sources/ChannelAppearanceScreen.swift b/submodules/TelegramUI/Components/Settings/PeerNameColorScreen/Sources/ChannelAppearanceScreen.swift index 1d7f49bec23..e805cbd4b79 100644 --- a/submodules/TelegramUI/Components/Settings/PeerNameColorScreen/Sources/ChannelAppearanceScreen.swift +++ b/submodules/TelegramUI/Components/Settings/PeerNameColorScreen/Sources/ChannelAppearanceScreen.swift @@ -1261,10 +1261,10 @@ final class ChannelAppearanceScreenComponent: Component { itemGenerator: PeerNameColorItem( theme: environment.theme, colors: component.context.peerNameColors, - isProfile: true, + mode: .profile, currentColor: profileColor, updated: { [weak self] value in - guard let self else { + guard let self, let value else { return } self.updatedPeerProfileColor = value @@ -1277,14 +1277,14 @@ final class ChannelAppearanceScreenComponent: Component { AnyComponentWithIdentity(id: 2, component: AnyComponent(ListActionItemComponent( theme: environment.theme, title: AnyComponent(HStack(profileLogoContents, spacing: 6.0)), - icon: AnyComponentWithIdentity(id: 0, component: AnyComponent(EmojiActionIconComponent( + icon: ListActionItemComponent.Icon(component: AnyComponentWithIdentity(id: 0, component: AnyComponent(EmojiActionIconComponent( context: component.context, color: profileColor.flatMap { profileColor in component.context.peerNameColors.getProfile(profileColor, dark: environment.theme.overallDarkAppearance, subject: .palette).main } ?? environment.theme.list.itemAccentColor, fileId: backgroundFileId, file: backgroundFileId.flatMap { self.cachedIconFiles[$0] } - ))), + )))), action: { [weak self] view in guard let self, let resolvedState = self.resolveState(), let view = view as? ListActionItemComponent.View, let iconView = view.iconView else { return @@ -1406,12 +1406,12 @@ final class ChannelAppearanceScreenComponent: Component { AnyComponentWithIdentity(id: 0, component: AnyComponent(ListActionItemComponent( theme: environment.theme, title: AnyComponent(HStack(emojiPackContents, spacing: 6.0)), - icon: AnyComponentWithIdentity(id: 0, component: AnyComponent(EmojiActionIconComponent( + icon: ListActionItemComponent.Icon(component: AnyComponentWithIdentity(id: 0, component: AnyComponent(EmojiActionIconComponent( context: component.context, color: environment.theme.list.itemAccentColor, fileId: emojiPack?.thumbnailFileId, file: emojiPackFile - ))), + )))), action: { [weak self] view in guard let self, let resolvedState = self.resolveState() else { return @@ -1470,12 +1470,12 @@ final class ChannelAppearanceScreenComponent: Component { AnyComponentWithIdentity(id: 0, component: AnyComponent(ListActionItemComponent( theme: environment.theme, title: AnyComponent(HStack(emojiStatusContents, spacing: 6.0)), - icon: AnyComponentWithIdentity(id: 0, component: AnyComponent(EmojiActionIconComponent( + icon: ListActionItemComponent.Icon(component: AnyComponentWithIdentity(id: 0, component: AnyComponent(EmojiActionIconComponent( context: component.context, color: environment.theme.list.itemAccentColor, fileId: statusFileId, file: statusFileId.flatMap { self.cachedIconFiles[$0] } - ))), + )))), action: { [weak self] view in guard let self, let resolvedState = self.resolveState(), let view = view as? ListActionItemComponent.View, let iconView = view.iconView else { return @@ -1578,10 +1578,10 @@ final class ChannelAppearanceScreenComponent: Component { itemGenerator: PeerNameColorItem( theme: environment.theme, colors: component.context.peerNameColors, - isProfile: false, + mode: .name, currentColor: resolvedState.nameColor, updated: { [weak self] value in - guard let self else { + guard let self, let value else { return } self.updatedPeerNameColor = value @@ -1594,12 +1594,12 @@ final class ChannelAppearanceScreenComponent: Component { AnyComponentWithIdentity(id: 2, component: AnyComponent(ListActionItemComponent( theme: environment.theme, title: AnyComponent(HStack(replyLogoContents, spacing: 6.0)), - icon: AnyComponentWithIdentity(id: 0, component: AnyComponent(EmojiActionIconComponent( + icon: ListActionItemComponent.Icon(component: AnyComponentWithIdentity(id: 0, component: AnyComponent(EmojiActionIconComponent( context: component.context, color: component.context.peerNameColors.get(resolvedState.nameColor, dark: environment.theme.overallDarkAppearance).main, fileId: replyFileId, file: replyFileId.flatMap { self.cachedIconFiles[$0] } - ))), + )))), action: { [weak self] view in guard let self, let resolvedState = self.resolveState(), let view = view as? ListActionItemComponent.View, let iconView = view.iconView else { return diff --git a/submodules/TelegramUI/Components/Settings/PeerNameColorScreen/Sources/PeerNameColorScreen.swift b/submodules/TelegramUI/Components/Settings/PeerNameColorScreen/Sources/PeerNameColorScreen.swift index 945edb7b487..72cfb99bc27 100644 --- a/submodules/TelegramUI/Components/Settings/PeerNameColorScreen/Sources/PeerNameColorScreen.swift +++ b/submodules/TelegramUI/Components/Settings/PeerNameColorScreen/Sources/PeerNameColorScreen.swift @@ -212,10 +212,12 @@ private enum PeerNameColorScreenEntry: ItemListNodeEntry { return PeerNameColorItem( theme: presentationData.theme, colors: colors, - isProfile: isProfile, + mode: isProfile ? .profile : .name, currentColor: currentColor, updated: { color in - arguments.updateNameColor(color) + if let color { + arguments.updateNameColor(color) + } }, sectionId: self.section ) diff --git a/submodules/TelegramUI/Components/Settings/QuickReplyNameAlertController/BUILD b/submodules/TelegramUI/Components/Settings/QuickReplyNameAlertController/BUILD new file mode 100644 index 00000000000..a9f96d6606b --- /dev/null +++ b/submodules/TelegramUI/Components/Settings/QuickReplyNameAlertController/BUILD @@ -0,0 +1,28 @@ +load("@build_bazel_rules_swift//swift:swift.bzl", "swift_library") + +swift_library( + name = "QuickReplyNameAlertController", + module_name = "QuickReplyNameAlertController", + srcs = glob([ + "Sources/**/*.swift", + ]), + copts = [ + "-warnings-as-errors", + ], + deps = [ + "//submodules/SSignalKit/SwiftSignalKit:SwiftSignalKit", + "//submodules/AsyncDisplayKit:AsyncDisplayKit", + "//submodules/Display:Display", + "//submodules/Postbox:Postbox", + "//submodules/TelegramCore:TelegramCore", + "//submodules/AccountContext:AccountContext", + "//submodules/TelegramPresentationData:TelegramPresentationData", + "//submodules/ComponentFlow", + "//submodules/Components/MultilineTextComponent", + "//submodules/Components/BalancedTextComponent", + "//submodules/TelegramUI/Components/EmojiStatusComponent", + ], + visibility = [ + "//visibility:public", + ], +) diff --git a/submodules/TelegramUI/Components/Settings/QuickReplyNameAlertController/Sources/QuickReplyNameAlertController.swift b/submodules/TelegramUI/Components/Settings/QuickReplyNameAlertController/Sources/QuickReplyNameAlertController.swift new file mode 100644 index 00000000000..9629f3f3bbe --- /dev/null +++ b/submodules/TelegramUI/Components/Settings/QuickReplyNameAlertController/Sources/QuickReplyNameAlertController.swift @@ -0,0 +1,543 @@ +import Foundation +import UIKit +import SwiftSignalKit +import AsyncDisplayKit +import Display +import Postbox +import TelegramCore +import TelegramPresentationData +import AccountContext +import ComponentFlow +import MultilineTextComponent +import BalancedTextComponent +import EmojiStatusComponent + +private final class PromptInputFieldNode: ASDisplayNode, ASEditableTextNodeDelegate { + private var theme: PresentationTheme + private let backgroundNode: ASImageNode + private let textInputNode: EditableTextNode + private let placeholderNode: ASTextNode + private let characterLimitView = ComponentView() + + private let characterLimit: Int + + var updateHeight: (() -> Void)? + var complete: (() -> Void)? + var textChanged: ((String) -> Void)? + + private let backgroundInsets = UIEdgeInsets(top: 8.0, left: 16.0, bottom: 15.0, right: 16.0) + private let inputInsets: UIEdgeInsets + + private let validCharacterSets: [CharacterSet] + + var text: String { + get { + return self.textInputNode.attributedText?.string ?? "" + } + set { + self.textInputNode.attributedText = NSAttributedString(string: newValue, font: Font.regular(13.0), textColor: self.theme.actionSheet.inputTextColor) + self.placeholderNode.isHidden = !newValue.isEmpty + } + } + + var placeholder: String = "" { + didSet { + self.placeholderNode.attributedText = NSAttributedString(string: self.placeholder, font: Font.regular(13.0), textColor: self.theme.actionSheet.inputPlaceholderColor) + } + } + + init(theme: PresentationTheme, placeholder: String, characterLimit: Int) { + self.theme = theme + self.characterLimit = characterLimit + + self.inputInsets = UIEdgeInsets(top: 9.0, left: 6.0, bottom: 9.0, right: 16.0) + + self.backgroundNode = ASImageNode() + self.backgroundNode.isLayerBacked = true + self.backgroundNode.displaysAsynchronously = false + self.backgroundNode.displayWithoutProcessing = true + self.backgroundNode.image = generateStretchableFilledCircleImage(diameter: 16.0, color: theme.actionSheet.inputHollowBackgroundColor, strokeColor: theme.actionSheet.inputBorderColor, strokeWidth: 1.0) + + self.textInputNode = EditableTextNode() + self.textInputNode.typingAttributes = [NSAttributedString.Key.font.rawValue: Font.regular(13.0), NSAttributedString.Key.foregroundColor.rawValue: theme.actionSheet.inputTextColor] + self.textInputNode.clipsToBounds = true + self.textInputNode.hitTestSlop = UIEdgeInsets(top: -5.0, left: -5.0, bottom: -5.0, right: -5.0) + self.textInputNode.textContainerInset = UIEdgeInsets(top: self.inputInsets.top, left: self.inputInsets.left, bottom: self.inputInsets.bottom, right: self.inputInsets.right) + self.textInputNode.keyboardAppearance = theme.rootController.keyboardColor.keyboardAppearance + self.textInputNode.keyboardType = .default + self.textInputNode.autocapitalizationType = .none + self.textInputNode.returnKeyType = .done + self.textInputNode.autocorrectionType = .no + self.textInputNode.tintColor = theme.actionSheet.controlAccentColor + + self.placeholderNode = ASTextNode() + self.placeholderNode.isUserInteractionEnabled = false + self.placeholderNode.displaysAsynchronously = false + self.placeholderNode.attributedText = NSAttributedString(string: placeholder, font: Font.regular(13.0), textColor: self.theme.actionSheet.inputPlaceholderColor) + + self.validCharacterSets = [ + CharacterSet.alphanumerics, + CharacterSet(charactersIn: "0123456789_"), + ] + + super.init() + + self.textInputNode.delegate = self + + self.addSubnode(self.backgroundNode) + self.addSubnode(self.textInputNode) + self.addSubnode(self.placeholderNode) + } + + func updateTheme(_ theme: PresentationTheme) { + self.theme = theme + + self.backgroundNode.image = generateStretchableFilledCircleImage(diameter: 16.0, color: self.theme.actionSheet.inputHollowBackgroundColor, strokeColor: self.theme.actionSheet.inputBorderColor, strokeWidth: 1.0) + self.textInputNode.keyboardAppearance = self.theme.rootController.keyboardColor.keyboardAppearance + self.placeholderNode.attributedText = NSAttributedString(string: self.placeholderNode.attributedText?.string ?? "", font: Font.regular(13.0), textColor: self.theme.actionSheet.inputPlaceholderColor) + self.textInputNode.tintColor = self.theme.actionSheet.controlAccentColor + } + + func updateLayout(width: CGFloat, transition: ContainedViewLayoutTransition) -> CGFloat { + let backgroundInsets = self.backgroundInsets + let inputInsets = self.inputInsets + + let textFieldHeight = self.calculateTextFieldMetrics(width: width) + let panelHeight = textFieldHeight + backgroundInsets.top + backgroundInsets.bottom + + let backgroundFrame = CGRect(origin: CGPoint(x: backgroundInsets.left, y: backgroundInsets.top), size: CGSize(width: width - backgroundInsets.left - backgroundInsets.right, height: panelHeight - backgroundInsets.top - backgroundInsets.bottom)) + transition.updateFrame(node: self.backgroundNode, frame: backgroundFrame) + + let placeholderSize = self.placeholderNode.measure(backgroundFrame.size) + transition.updateFrame(node: self.placeholderNode, frame: CGRect(origin: CGPoint(x: backgroundFrame.minX + inputInsets.left + 5.0, y: backgroundFrame.minY + floor((backgroundFrame.size.height - placeholderSize.height) / 2.0)), size: placeholderSize)) + + transition.updateFrame(node: self.textInputNode, frame: CGRect(origin: CGPoint(x: backgroundFrame.minX + inputInsets.left, y: backgroundFrame.minY), size: CGSize(width: backgroundFrame.size.width - inputInsets.left - inputInsets.right, height: backgroundFrame.size.height))) + + let characterLimitString: String + let characterLimitColor: UIColor + if self.text.count <= self.characterLimit { + let remaining = self.characterLimit - self.text.count + if remaining < 5 { + characterLimitString = "\(remaining)" + } else { + characterLimitString = " " + } + characterLimitColor = self.theme.list.itemPlaceholderTextColor + } else { + characterLimitString = "\(self.characterLimit - self.text.count)" + characterLimitColor = self.theme.list.itemDestructiveColor + } + + let characterLimitSize = self.characterLimitView.update( + transition: .immediate, + component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: characterLimitString, font: Font.regular(13.0), textColor: characterLimitColor)) + )), + environment: {}, + containerSize: CGSize(width: 100.0, height: 100.0) + ) + if let characterLimitComponentView = self.characterLimitView.view { + if characterLimitComponentView.superview == nil { + self.view.addSubview(characterLimitComponentView) + } + characterLimitComponentView.frame = CGRect(origin: CGPoint(x: width - 23.0 - characterLimitSize.width, y: 18.0), size: characterLimitSize) + } + + return panelHeight + } + + func activateInput() { + self.textInputNode.becomeFirstResponder() + } + + func deactivateInput() { + self.textInputNode.resignFirstResponder() + } + + @objc func editableTextNodeDidUpdateText(_ editableTextNode: ASEditableTextNode) { + self.updateTextNodeText(animated: true) + self.textChanged?(editableTextNode.textView.text) + self.placeholderNode.isHidden = !(editableTextNode.textView.text ?? "").isEmpty + } + + func editableTextNode(_ editableTextNode: ASEditableTextNode, shouldChangeTextIn range: NSRange, replacementText text: String) -> Bool { + if text == "\n" { + self.complete?() + return false + } + if text.unicodeScalars.contains(where: { c in + return !self.validCharacterSets.contains(where: { set in + return set.contains(c) + }) + }) { + return false + } + return true + } + + private func calculateTextFieldMetrics(width: CGFloat) -> CGFloat { + let backgroundInsets = self.backgroundInsets + let inputInsets = self.inputInsets + + let unboundTextFieldHeight = max(34.0, ceil(self.textInputNode.measure(CGSize(width: width - backgroundInsets.left - backgroundInsets.right - inputInsets.left - inputInsets.right, height: CGFloat.greatestFiniteMagnitude)).height)) + + return min(61.0, max(34.0, unboundTextFieldHeight)) + } + + private func updateTextNodeText(animated: Bool) { + let backgroundInsets = self.backgroundInsets + + let textFieldHeight = self.calculateTextFieldMetrics(width: self.bounds.size.width) + + let panelHeight = textFieldHeight + backgroundInsets.top + backgroundInsets.bottom + if !self.bounds.size.height.isEqual(to: panelHeight) { + self.updateHeight?() + } + } + + @objc func clearPressed() { + self.textInputNode.attributedText = nil + self.deactivateInput() + } +} + +public final class QuickReplyNameAlertContentNode: AlertContentNode { + private let context: AccountContext + private var theme: AlertControllerTheme + private let strings: PresentationStrings + private let text: String + private let subtext: String + private let titleFont: PromptControllerTitleFont + + private let textView = ComponentView() + private let subtextView = ComponentView() + + fileprivate let inputFieldNode: PromptInputFieldNode + + private let actionNodesSeparator: ASDisplayNode + private let actionNodes: [TextAlertContentActionNode] + private let actionVerticalSeparators: [ASDisplayNode] + + private let disposable = MetaDisposable() + + private var validLayout: CGSize? + private var errorText: String? + + private let hapticFeedback = HapticFeedback() + + var complete: (() -> Void)? + + override public var dismissOnOutsideTap: Bool { + return self.isUserInteractionEnabled + } + + init(context: AccountContext, theme: AlertControllerTheme, ptheme: PresentationTheme, strings: PresentationStrings, actions: [TextAlertAction], text: String, subtext: String, titleFont: PromptControllerTitleFont, value: String?, characterLimit: Int) { + self.context = context + self.theme = theme + self.strings = strings + self.text = text + self.subtext = subtext + self.titleFont = titleFont + + self.inputFieldNode = PromptInputFieldNode(theme: ptheme, placeholder: strings.QuickReply_ShortcutPlaceholder, characterLimit: characterLimit) + self.inputFieldNode.text = value ?? "" + + self.actionNodesSeparator = ASDisplayNode() + self.actionNodesSeparator.isLayerBacked = true + + self.actionNodes = actions.map { action -> TextAlertContentActionNode in + return TextAlertContentActionNode(theme: theme, action: action) + } + + var actionVerticalSeparators: [ASDisplayNode] = [] + if actions.count > 1 { + for _ in 0 ..< actions.count - 1 { + let separatorNode = ASDisplayNode() + separatorNode.isLayerBacked = true + actionVerticalSeparators.append(separatorNode) + } + } + self.actionVerticalSeparators = actionVerticalSeparators + + super.init() + + self.addSubnode(self.inputFieldNode) + + self.addSubnode(self.actionNodesSeparator) + + for actionNode in self.actionNodes { + self.addSubnode(actionNode) + } + self.actionNodes.last?.actionEnabled = true + + for separatorNode in self.actionVerticalSeparators { + self.addSubnode(separatorNode) + } + + self.inputFieldNode.updateHeight = { [weak self] in + if let strongSelf = self { + if let _ = strongSelf.validLayout { + strongSelf.requestLayout?(.immediate) + } + } + } + + self.inputFieldNode.textChanged = { [weak self] text in + if let strongSelf = self, let lastNode = strongSelf.actionNodes.last { + lastNode.actionEnabled = text.count <= characterLimit + strongSelf.requestLayout?(.immediate) + } + } + + self.updateTheme(theme) + + self.inputFieldNode.complete = { [weak self] in + guard let self else { + return + } + if let lastNode = self.actionNodes.last, lastNode.actionEnabled { + self.complete?() + } + } + } + + deinit { + self.disposable.dispose() + } + + var value: String { + return self.inputFieldNode.text + } + + public func setErrorText(errorText: String?) { + if self.errorText != errorText { + self.errorText = errorText + self.requestLayout?(.immediate) + } + + if errorText != nil { + HapticFeedback().error() + self.inputFieldNode.layer.addShakeAnimation() + } + } + + override public func updateTheme(_ theme: AlertControllerTheme) { + self.theme = theme + + self.actionNodesSeparator.backgroundColor = theme.separatorColor + for actionNode in self.actionNodes { + actionNode.updateTheme(theme) + } + for separatorNode in self.actionVerticalSeparators { + separatorNode.backgroundColor = theme.separatorColor + } + + if let size = self.validLayout { + _ = self.updateLayout(size: size, transition: .immediate) + } + } + + override public func updateLayout(size: CGSize, transition: ContainedViewLayoutTransition) -> CGSize { + var size = size + size.width = min(size.width, 270.0) + let measureSize = CGSize(width: size.width - 16.0 * 2.0, height: CGFloat.greatestFiniteMagnitude) + + let hadValidLayout = self.validLayout != nil + + self.validLayout = size + + var origin: CGPoint = CGPoint(x: 0.0, y: 16.0) + let spacing: CGFloat = 5.0 + let subtextSpacing: CGFloat = -1.0 + + let textSize = self.textView.update( + transition: .immediate, + component: AnyComponent(MultilineTextComponent( + text: .plain(NSAttributedString(string: self.text, font: Font.semibold(17.0), textColor: self.theme.primaryColor)), + horizontalAlignment: .center, + maximumNumberOfLines: 0 + )), + environment: {}, + containerSize: CGSize(width: measureSize.width, height: 1000.0) + ) + let textFrame = CGRect(origin: CGPoint(x: floor((size.width - textSize.width) * 0.5), y: origin.y), size: textSize) + if let textComponentView = self.textView.view { + if textComponentView.superview == nil { + textComponentView.layer.anchorPoint = CGPoint() + self.view.addSubview(textComponentView) + } + textComponentView.bounds = CGRect(origin: CGPoint(), size: textFrame.size) + transition.updatePosition(layer: textComponentView.layer, position: textFrame.origin) + } + origin.y += textSize.height + 6.0 + subtextSpacing + + let subtextSize = self.subtextView.update( + transition: .immediate, + component: AnyComponent(BalancedTextComponent( + text: .plain(NSAttributedString(string: self.errorText ?? self.subtext, font: Font.regular(13.0), textColor: self.errorText != nil ? self.theme.destructiveColor : self.theme.primaryColor)), + horizontalAlignment: .center, + maximumNumberOfLines: 0 + )), + environment: {}, + containerSize: CGSize(width: measureSize.width, height: 1000.0) + ) + let subtextFrame = CGRect(origin: CGPoint(x: floor((size.width - subtextSize.width) * 0.5), y: origin.y), size: subtextSize) + if let subtextComponentView = self.subtextView.view { + if subtextComponentView.superview == nil { + subtextComponentView.layer.anchorPoint = CGPoint() + self.view.addSubview(subtextComponentView) + } + subtextComponentView.bounds = CGRect(origin: CGPoint(), size: subtextFrame.size) + transition.updatePosition(layer: subtextComponentView.layer, position: subtextFrame.origin) + } + origin.y += subtextSize.height + 6.0 + spacing + + let actionButtonHeight: CGFloat = 44.0 + var minActionsWidth: CGFloat = 0.0 + let maxActionWidth: CGFloat = floor(size.width / CGFloat(self.actionNodes.count)) + let actionTitleInsets: CGFloat = 8.0 + + var effectiveActionLayout = TextAlertContentActionLayout.horizontal + for actionNode in self.actionNodes { + let actionTitleSize = actionNode.titleNode.updateLayout(CGSize(width: maxActionWidth, height: actionButtonHeight)) + if case .horizontal = effectiveActionLayout, actionTitleSize.height > actionButtonHeight * 0.6667 { + effectiveActionLayout = .vertical + } + switch effectiveActionLayout { + case .horizontal: + minActionsWidth += actionTitleSize.width + actionTitleInsets + case .vertical: + minActionsWidth = max(minActionsWidth, actionTitleSize.width + actionTitleInsets) + } + } + + let insets = UIEdgeInsets(top: 18.0, left: 18.0, bottom: 9.0, right: 18.0) + + var contentWidth = max(textSize.width, minActionsWidth) + contentWidth = max(subtextSize.width, minActionsWidth) + contentWidth = max(contentWidth, 234.0) + + var actionsHeight: CGFloat = 0.0 + switch effectiveActionLayout { + case .horizontal: + actionsHeight = actionButtonHeight + case .vertical: + actionsHeight = actionButtonHeight * CGFloat(self.actionNodes.count) + } + + let resultWidth = contentWidth + insets.left + insets.right + + let inputFieldWidth = resultWidth + let inputFieldHeight = self.inputFieldNode.updateLayout(width: inputFieldWidth, transition: transition) + let inputHeight = inputFieldHeight + let inputFieldFrame = CGRect(x: 0.0, y: origin.y, width: resultWidth, height: inputFieldHeight) + transition.updateFrame(node: self.inputFieldNode, frame: inputFieldFrame) + transition.updateAlpha(node: self.inputFieldNode, alpha: inputHeight > 0.0 ? 1.0 : 0.0) + + let resultSize = CGSize(width: resultWidth, height: textSize.height + subtextSpacing + subtextSize.height + spacing + inputHeight + actionsHeight + insets.top + insets.bottom) + + transition.updateFrame(node: self.actionNodesSeparator, frame: CGRect(origin: CGPoint(x: 0.0, y: resultSize.height - actionsHeight - UIScreenPixel), size: CGSize(width: resultSize.width, height: UIScreenPixel))) + + var actionOffset: CGFloat = 0.0 + let actionWidth: CGFloat = floor(resultSize.width / CGFloat(self.actionNodes.count)) + var separatorIndex = -1 + var nodeIndex = 0 + for actionNode in self.actionNodes { + if separatorIndex >= 0 { + let separatorNode = self.actionVerticalSeparators[separatorIndex] + switch effectiveActionLayout { + case .horizontal: + transition.updateFrame(node: separatorNode, frame: CGRect(origin: CGPoint(x: actionOffset - UIScreenPixel, y: resultSize.height - actionsHeight), size: CGSize(width: UIScreenPixel, height: actionsHeight - UIScreenPixel))) + case .vertical: + transition.updateFrame(node: separatorNode, frame: CGRect(origin: CGPoint(x: 0.0, y: resultSize.height - actionsHeight + actionOffset - UIScreenPixel), size: CGSize(width: resultSize.width, height: UIScreenPixel))) + } + } + separatorIndex += 1 + + let currentActionWidth: CGFloat + switch effectiveActionLayout { + case .horizontal: + if nodeIndex == self.actionNodes.count - 1 { + currentActionWidth = resultSize.width - actionOffset + } else { + currentActionWidth = actionWidth + } + case .vertical: + currentActionWidth = resultSize.width + } + + let actionNodeFrame: CGRect + switch effectiveActionLayout { + case .horizontal: + actionNodeFrame = CGRect(origin: CGPoint(x: actionOffset, y: resultSize.height - actionsHeight), size: CGSize(width: currentActionWidth, height: actionButtonHeight)) + actionOffset += currentActionWidth + case .vertical: + actionNodeFrame = CGRect(origin: CGPoint(x: 0.0, y: resultSize.height - actionsHeight + actionOffset), size: CGSize(width: currentActionWidth, height: actionButtonHeight)) + actionOffset += actionButtonHeight + } + + transition.updateFrame(node: actionNode, frame: actionNodeFrame) + + nodeIndex += 1 + } + + if !hadValidLayout { + self.inputFieldNode.activateInput() + } + + return resultSize + } + + func animateError() { + self.inputFieldNode.layer.addShakeAnimation() + self.hapticFeedback.error() + } +} + +public enum PromptControllerTitleFont { + case regular + case bold +} + +public func quickReplyNameAlertController(context: AccountContext, updatedPresentationData: (initial: PresentationData, signal: Signal)? = nil, text: String, subtext: String, titleFont: PromptControllerTitleFont = .regular, value: String?, characterLimit: Int = 1000, apply: @escaping (String?) -> Void) -> AlertController { + let presentationData = updatedPresentationData?.initial ?? context.sharedContext.currentPresentationData.with { $0 } + + var dismissImpl: ((Bool) -> Void)? + var applyImpl: (() -> Void)? + + let actions: [TextAlertAction] = [TextAlertAction(type: .genericAction, title: presentationData.strings.Common_Cancel, action: { + dismissImpl?(true) + apply(nil) + }), TextAlertAction(type: .defaultAction, title: presentationData.strings.Common_Done, action: { + applyImpl?() + })] + + let contentNode = QuickReplyNameAlertContentNode(context: context, theme: AlertControllerTheme(presentationData: presentationData), ptheme: presentationData.theme, strings: presentationData.strings, actions: actions, text: text, subtext: subtext, titleFont: titleFont, value: value, characterLimit: characterLimit) + contentNode.complete = { + applyImpl?() + } + applyImpl = { [weak contentNode] in + guard let contentNode = contentNode else { + return + } + apply(contentNode.value) + } + + let controller = AlertController(theme: AlertControllerTheme(presentationData: presentationData), contentNode: contentNode) + let presentationDataDisposable = (updatedPresentationData?.signal ?? context.sharedContext.presentationData).start(next: { [weak controller, weak contentNode] presentationData in + controller?.theme = AlertControllerTheme(presentationData: presentationData) + contentNode?.inputFieldNode.updateTheme(presentationData.theme) + }) + controller.dismissed = { _ in + presentationDataDisposable.dispose() + } + dismissImpl = { [weak controller] animated in + contentNode.inputFieldNode.deactivateInput() + if animated { + controller?.dismissAnimated() + } else { + controller?.dismiss() + } + } + return controller +} diff --git a/submodules/TelegramUI/Components/Settings/ThemeAccentColorScreen/Sources/ThemeAccentColorControllerNode.swift b/submodules/TelegramUI/Components/Settings/ThemeAccentColorScreen/Sources/ThemeAccentColorControllerNode.swift index 4944cbc942d..c18ef7ae63a 100644 --- a/submodules/TelegramUI/Components/Settings/ThemeAccentColorScreen/Sources/ThemeAccentColorControllerNode.swift +++ b/submodules/TelegramUI/Components/Settings/ThemeAccentColorScreen/Sources/ThemeAccentColorControllerNode.swift @@ -870,6 +870,7 @@ final class ThemeAccentColorControllerNode: ASDisplayNode, UIScrollViewDelegate }, openChatFolderUpdates: {}, hideChatFolderUpdates: { }, openStories: { _, _ in }, dismissNotice: { _ in + }, editPeer: { _ in }) let chatListPresentationData = ChatListPresentationData(theme: self.presentationData.theme, fontSize: self.presentationData.listsFontSize, strings: self.presentationData.strings, dateTimeFormat: self.presentationData.dateTimeFormat, nameSortOrder: self.presentationData.nameSortOrder, nameDisplayOrder: self.presentationData.nameDisplayOrder, disableAnimations: true) diff --git a/submodules/TelegramUI/Components/Settings/TimezoneSelectionScreen/BUILD b/submodules/TelegramUI/Components/Settings/TimezoneSelectionScreen/BUILD new file mode 100644 index 00000000000..38fdb23587f --- /dev/null +++ b/submodules/TelegramUI/Components/Settings/TimezoneSelectionScreen/BUILD @@ -0,0 +1,32 @@ +load("@build_bazel_rules_swift//swift:swift.bzl", "swift_library") + +swift_library( + name = "TimezoneSelectionScreen", + module_name = "TimezoneSelectionScreen", + srcs = glob([ + "Sources/**/*.swift", + ]), + copts = [ + "-warnings-as-errors", + ], + deps = [ + "//submodules/Display", + "//submodules/AsyncDisplayKit", + "//submodules/Postbox", + "//submodules/SSignalKit/SwiftSignalKit", + "//submodules/TelegramCore", + "//submodules/TelegramPresentationData", + "//submodules/AccountContext", + "//submodules/SearchUI", + "//submodules/MergeLists", + "//submodules/ItemListUI", + "//submodules/PresentationDataUtils", + "//submodules/SearchBarNode", + "//submodules/TelegramUIPreferences", + "//submodules/ComponentFlow", + "//submodules/Components/BalancedTextComponent", + ], + visibility = [ + "//visibility:public", + ], +) diff --git a/submodules/TelegramUI/Components/Settings/TimezoneSelectionScreen/Sources/TimezoneSelectionScreen.swift b/submodules/TelegramUI/Components/Settings/TimezoneSelectionScreen/Sources/TimezoneSelectionScreen.swift new file mode 100644 index 00000000000..d83ae3e0fbe --- /dev/null +++ b/submodules/TelegramUI/Components/Settings/TimezoneSelectionScreen/Sources/TimezoneSelectionScreen.swift @@ -0,0 +1,155 @@ +import Foundation +import UIKit +import Display +import AsyncDisplayKit +import Postbox +import SwiftSignalKit +import TelegramCore +import TelegramPresentationData +import AccountContext +import SearchUI + +public class TimezoneSelectionScreen: ViewController { + private let context: AccountContext + private let completed: (String) -> Void + + private var controllerNode: TimezoneSelectionScreenNode { + return self.displayNode as! TimezoneSelectionScreenNode + } + + private var _ready = Promise() + override public var ready: Promise { + return self._ready + } + + private var presentationData: PresentationData + private var presentationDataDisposable: Disposable? + + private var searchContentNode: NavigationBarSearchContentNode? + + private var previousContentOffset: ListViewVisibleContentOffset? + + public init(context: AccountContext, completed: @escaping (String) -> Void) { + self.context = context + self.completed = completed + + self.presentationData = context.sharedContext.currentPresentationData.with { $0 } + + super.init(navigationBarPresentationData: NavigationBarPresentationData(presentationData: self.presentationData)) + + self.statusBar.statusBarStyle = self.presentationData.theme.rootController.statusBarStyle.style + + self.title = self.presentationData.strings.TimeZoneSelection_Title + + self.navigationItem.backBarButtonItem = UIBarButtonItem(title: self.presentationData.strings.Common_Back, style: .plain, target: nil, action: nil) + + self.scrollToTop = { [weak self] in + if let strongSelf = self { + if let searchContentNode = strongSelf.searchContentNode { + searchContentNode.updateExpansionProgress(1.0, animated: true) + } + strongSelf.controllerNode.scrollToTop() + } + } + + self.presentationDataDisposable = (context.sharedContext.presentationData + |> deliverOnMainQueue).start(next: { [weak self] presentationData in + if let strongSelf = self { + let previousTheme = strongSelf.presentationData.theme + let previousStrings = strongSelf.presentationData.strings + + strongSelf.presentationData = presentationData + + if previousTheme !== presentationData.theme || previousStrings !== presentationData.strings { + strongSelf.updateThemeAndStrings() + } + } + }) + + self.searchContentNode = NavigationBarSearchContentNode(theme: self.presentationData.theme, placeholder: self.presentationData.strings.Common_Search, inline: true, activate: { [weak self] in + self?.activateSearch() + }) + self.navigationBar?.setContentNode(self.searchContentNode, animated: false) + } + + required public init(coder aDecoder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + deinit { + self.presentationDataDisposable?.dispose() + } + + private func updateThemeAndStrings() { + self.statusBar.statusBarStyle = self.presentationData.theme.rootController.statusBarStyle.style + self.navigationBar?.updatePresentationData(NavigationBarPresentationData(presentationData: self.presentationData)) + self.searchContentNode?.updateThemeAndPlaceholder(theme: self.presentationData.theme, placeholder: self.presentationData.strings.Common_Search) + self.title = self.presentationData.strings.Settings_AppLanguage + self.navigationItem.backBarButtonItem = UIBarButtonItem(title: self.presentationData.strings.Common_Back, style: .plain, target: nil, action: nil) + self.controllerNode.updatePresentationData(self.presentationData) + } + + override public func loadDisplayNode() { + self.displayNode = TimezoneSelectionScreenNode(context: self.context, presentationData: self.presentationData, navigationBar: self.navigationBar!, requestActivateSearch: { [weak self] in + self?.activateSearch() + }, requestDeactivateSearch: { [weak self] in + self?.deactivateSearch() + }, action: { [weak self] id in + guard let self else { + return + } + self.completed(id) + }, present: { [weak self] c, a in + self?.present(c, in: .window(.root), with: a) + }, push: { [weak self] c in + self?.push(c) + }) + + self.controllerNode.listNode.visibleContentOffsetChanged = { [weak self] offset in + if let strongSelf = self { + if let searchContentNode = strongSelf.searchContentNode { + searchContentNode.updateListVisibleContentOffset(offset) + } + + strongSelf.previousContentOffset = offset + } + } + + self.controllerNode.listNode.didEndScrolling = { [weak self] _ in + if let strongSelf = self, let searchContentNode = strongSelf.searchContentNode { + let _ = fixNavigationSearchableListNodeScrolling(strongSelf.controllerNode.listNode, searchNode: searchContentNode) + } + } + + self._ready.set(self.controllerNode._ready.get()) + + self.displayNodeDidLoad() + } + + override public func containerLayoutUpdated(_ layout: ContainerViewLayout, transition: ContainedViewLayoutTransition) { + super.containerLayoutUpdated(layout, transition: transition) + + self.controllerNode.containerLayoutUpdated(layout, navigationBarHeight: self.cleanNavigationHeight, transition: transition) + } + + private func activateSearch() { + if self.displayNavigationBar { + if let scrollToTop = self.scrollToTop { + scrollToTop() + } + if let searchContentNode = self.searchContentNode { + self.controllerNode.activateSearch(placeholderNode: searchContentNode.placeholderNode) + } + self.setDisplayNavigationBar(false, transition: .animated(duration: 0.5, curve: .spring)) + } + } + + private func deactivateSearch() { + if !self.displayNavigationBar { + self.setDisplayNavigationBar(true, transition: .animated(duration: 0.5, curve: .spring)) + if let searchContentNode = self.searchContentNode { + self.controllerNode.deactivateSearch(placeholderNode: searchContentNode.placeholderNode) + } + } + } +} diff --git a/submodules/TelegramUI/Components/Settings/TimezoneSelectionScreen/Sources/TimezoneSelectionScreenNode.swift b/submodules/TelegramUI/Components/Settings/TimezoneSelectionScreen/Sources/TimezoneSelectionScreenNode.swift new file mode 100644 index 00000000000..e20cc58d886 --- /dev/null +++ b/submodules/TelegramUI/Components/Settings/TimezoneSelectionScreen/Sources/TimezoneSelectionScreenNode.swift @@ -0,0 +1,550 @@ +import Foundation +import UIKit +import Display +import AsyncDisplayKit +import Postbox +import TelegramCore +import SwiftSignalKit +import TelegramPresentationData +import MergeLists +import ItemListUI +import PresentationDataUtils +import AccountContext +import SearchBarNode +import SearchUI +import TelegramUIPreferences +import ComponentFlow +import BalancedTextComponent + +private struct TimezoneListEntry: Comparable, Identifiable { + var id: String + var offset: Int + var title: String + + var stableId: String { + return self.id + } + + static func <(lhs: TimezoneListEntry, rhs: TimezoneListEntry) -> Bool { + if lhs.offset != rhs.offset { + return lhs.offset < rhs.offset + } + if lhs.title != rhs.title { + return lhs.title < rhs.title + } + return lhs.id < rhs.id + } + + func item(presentationData: PresentationData, searchMode: Bool, action: @escaping (String) -> Void) -> ListViewItem { + let hours = abs(self.offset / (60 * 60)) + let minutes = abs(self.offset % (60 * 60)) / 60 + let offsetString: String + if minutes == 0 { + offsetString = "UTC \(self.offset >= 0 ? "+" : "-")\(hours)" + } else { + let minutesString: String + if minutes < 10 { + minutesString = "0\(minutes)" + } else { + minutesString = "\(minutes)" + } + offsetString = "UTC \(self.offset >= 0 ? "+" : "-")\(hours):\(minutesString)" + } + + return ItemListDisclosureItem(presentationData: ItemListPresentationData(presentationData), title: self.title, label: offsetString, sectionId: 0, style: .plain, disclosureStyle: .none, action: { + action(self.id) + }) + } +} + +private struct TimezoneListSearchContainerTransition { + let deletions: [ListViewDeleteItem] + let insertions: [ListViewInsertItem] + let updates: [ListViewUpdateItem] + let isSearching: Bool + let isEmptyResult: Bool +} + +private func preparedLanguageListSearchContainerTransition(presentationData: PresentationData, from fromEntries: [TimezoneListEntry], to toEntries: [TimezoneListEntry], action: @escaping (String) -> Void, isSearching: Bool, forceUpdate: Bool) -> TimezoneListSearchContainerTransition { + let (deleteIndices, indicesAndItems, updateIndices) = mergeListsStableWithUpdates(leftList: fromEntries, rightList: toEntries, allUpdated: forceUpdate) + + let deletions = deleteIndices.map { ListViewDeleteItem(index: $0, directionHint: nil) } + let insertions = indicesAndItems.map { ListViewInsertItem(index: $0.0, previousIndex: $0.2, item: $0.1.item(presentationData: presentationData, searchMode: true, action: action), directionHint: nil) } + let updates = updateIndices.map { ListViewUpdateItem(index: $0.0, previousIndex: $0.2, item: $0.1.item(presentationData: presentationData, searchMode: true, action: action), directionHint: nil) } + + return TimezoneListSearchContainerTransition(deletions: deletions, insertions: insertions, updates: updates, isSearching: isSearching, isEmptyResult: isSearching && toEntries.isEmpty) +} + +private final class TimezoneListSearchContainerNode: SearchDisplayControllerContentNode { + private let timeZoneList: TimeZoneList + private let dimNode: ASDisplayNode + private let listNode: ListView + + private var notFoundText: ComponentView? + + private var enqueuedTransitions: [TimezoneListSearchContainerTransition] = [] + private var hasValidLayout = false + + private let searchQuery = Promise() + private let searchDisposable = MetaDisposable() + + private var presentationData: PresentationData + private var presentationDataDisposable: Disposable? + + private let presentationDataPromise: Promise + + private var isEmptyResult: Bool = false + private var currentLayout: (layout: ContainerViewLayout, navigationBarHeight: CGFloat)? + + public override var hasDim: Bool { + return true + } + + init(context: AccountContext, timeZoneList: TimeZoneList, action: @escaping (String) -> Void) { + self.timeZoneList = timeZoneList + + let presentationData = context.sharedContext.currentPresentationData.with { $0 } + self.presentationData = presentationData + + self.presentationDataPromise = Promise(self.presentationData) + + self.dimNode = ASDisplayNode() + self.dimNode.backgroundColor = UIColor.black.withAlphaComponent(0.5) + + self.listNode = ListView() + self.listNode.accessibilityPageScrolledString = { row, count in + return presentationData.strings.VoiceOver_ScrollStatus(row, count).string + } + + super.init() + + self.listNode.backgroundColor = self.presentationData.theme.chatList.backgroundColor + self.listNode.isHidden = true + + self.addSubnode(self.dimNode) + self.addSubnode(self.listNode) + + let querySplitCharacterSet: CharacterSet = CharacterSet(charactersIn: " /.+") + + let foundItems = self.searchQuery.get() + |> mapToSignal { query -> Signal<[TimeZoneList.Item]?, NoError> in + if let query, !query.isEmpty { + let query = query.lowercased() + + return .single(timeZoneList.items.filter { item in + if item.id.lowercased().hasPrefix(query) { + return true + } + if item.title.lowercased().components(separatedBy: querySplitCharacterSet).contains(where: { $0.hasPrefix(query) }) { + return true + } + + return false + }) + } else { + return .single(nil) + } + } + + let previousEntriesHolder = Atomic<([TimezoneListEntry], PresentationTheme, PresentationStrings)?>(value: nil) + self.searchDisposable.set(combineLatest(queue: .mainQueue(), foundItems, self.presentationDataPromise.get()).start(next: { [weak self] items, presentationData in + guard let strongSelf = self else { + return + } + var entries: [TimezoneListEntry] = [] + if let items { + for item in items { + entries.append(TimezoneListEntry( + id: item.id, + offset: Int(item.utcOffset), + title: item.title + )) + } + } + entries.sort() + let previousEntriesAndPresentationData = previousEntriesHolder.swap((entries, presentationData.theme, presentationData.strings)) + let transition = preparedLanguageListSearchContainerTransition(presentationData: presentationData, from: previousEntriesAndPresentationData?.0 ?? [], to: entries, action: action, isSearching: items != nil, forceUpdate: previousEntriesAndPresentationData?.1 !== presentationData.theme || previousEntriesAndPresentationData?.2 !== presentationData.strings) + strongSelf.enqueueTransition(transition) + })) + + self.presentationDataDisposable = (context.sharedContext.presentationData + |> deliverOnMainQueue).start(next: { [weak self] presentationData in + if let strongSelf = self { + let previousTheme = strongSelf.presentationData.theme + let previousStrings = strongSelf.presentationData.strings + + strongSelf.presentationData = presentationData + + if previousTheme !== presentationData.theme || previousStrings !== presentationData.strings { + strongSelf.updateThemeAndStrings(theme: presentationData.theme, strings: presentationData.strings) + strongSelf.presentationDataPromise.set(.single(presentationData)) + } + } + }) + + self.listNode.beganInteractiveDragging = { [weak self] _ in + self?.dismissInput?() + } + } + + deinit { + self.searchDisposable.dispose() + self.presentationDataDisposable?.dispose() + } + + override func didLoad() { + super.didLoad() + + self.dimNode.view.addGestureRecognizer(UITapGestureRecognizer(target: self, action: #selector(self.dimTapGesture(_:)))) + } + + func updateThemeAndStrings(theme: PresentationTheme, strings: PresentationStrings) { + self.listNode.backgroundColor = theme.chatList.backgroundColor + } + + override func searchTextUpdated(text: String) { + if text.isEmpty { + self.searchQuery.set(.single(nil)) + } else { + self.searchQuery.set(.single(text)) + } + } + + private func enqueueTransition(_ transition: TimezoneListSearchContainerTransition) { + self.enqueuedTransitions.append(transition) + + if self.hasValidLayout { + while !self.enqueuedTransitions.isEmpty { + self.dequeueTransitions() + } + } + } + + private func dequeueTransitions() { + if let transition = self.enqueuedTransitions.first { + self.enqueuedTransitions.remove(at: 0) + + var options = ListViewDeleteAndInsertOptions() + options.insert(.PreferSynchronousDrawing) + + let isSearching = transition.isSearching + let isEmptyResult = transition.isEmptyResult + + self.listNode.transaction(deleteIndices: transition.deletions, insertIndicesAndItems: transition.insertions, updateIndicesAndItems: transition.updates, options: options, updateSizeAndInsets: nil, updateOpaqueState: nil, completion: { [weak self] _ in + guard let self else { + return + } + self.listNode.isHidden = !isSearching + self.dimNode.isHidden = isSearching + self.isEmptyResult = isEmptyResult + + if let currentLayout = self.currentLayout { + self.containerLayoutUpdated(currentLayout.layout, navigationBarHeight: currentLayout.navigationBarHeight, transition: .immediate) + } + }) + } + } + + override func containerLayoutUpdated(_ layout: ContainerViewLayout, navigationBarHeight: CGFloat, transition: ContainedViewLayoutTransition) { + self.currentLayout = (layout, navigationBarHeight) + + super.containerLayoutUpdated(layout, navigationBarHeight: navigationBarHeight, transition: transition) + + let topInset = navigationBarHeight + transition.updateFrame(node: self.dimNode, frame: CGRect(origin: CGPoint(x: 0.0, y: topInset), size: CGSize(width: layout.size.width, height: layout.size.height - topInset))) + + let (duration, curve) = listViewAnimationDurationAndCurve(transition: transition) + + self.listNode.frame = CGRect(origin: CGPoint(), size: layout.size) + self.listNode.transaction(deleteIndices: [], insertIndicesAndItems: [], updateIndicesAndItems: [], options: [.Synchronous], scrollToItem: nil, updateSizeAndInsets: ListViewUpdateSizeAndInsets(size: layout.size, insets: UIEdgeInsets(top: navigationBarHeight, left: layout.safeInsets.left, bottom: layout.insets(options: [.input]).bottom, right: layout.safeInsets.right), duration: duration, curve: curve), stationaryItemRange: nil, updateOpaqueState: nil, completion: { _ in }) + + if !self.hasValidLayout { + self.hasValidLayout = true + while !self.enqueuedTransitions.isEmpty { + self.dequeueTransitions() + } + } + + if self.isEmptyResult { + let notFoundText: ComponentView + if let current = self.notFoundText { + notFoundText = current + } else { + notFoundText = ComponentView() + self.notFoundText = notFoundText + } + let notFoundTextSize = notFoundText.update( + transition: .immediate, + component: AnyComponent(BalancedTextComponent( + text: .plain(NSAttributedString(string: self.presentationData.strings.Conversation_SearchNoResults, font: Font.regular(17.0), textColor: self.presentationData.theme.list.freeTextColor, paragraphAlignment: .center)), + horizontalAlignment: .center, + maximumNumberOfLines: 0 + )), + environment: {}, + containerSize: CGSize(width: layout.size.width - 16.0 * 2.0, height: layout.size.height) + ) + let notFoundTextFrame = CGRect(origin: CGPoint(x: floor((layout.size.width - notFoundTextSize.width) * 0.5), y: navigationBarHeight + floor((layout.size.height - navigationBarHeight - notFoundTextSize.height) * 0.5)), size: notFoundTextSize) + if let notFoundTextView = notFoundText.view { + if notFoundTextView.superview == nil { + self.view.addSubview(notFoundTextView) + } + notFoundTextView.frame = notFoundTextFrame + } + } else if let notFoundText = self.notFoundText { + self.notFoundText = nil + notFoundText.view?.removeFromSuperview() + } + } + + @objc func dimTapGesture(_ recognizer: UITapGestureRecognizer) { + if case .ended = recognizer.state { + self.cancel?() + } + } +} + +private struct TimezoneListNodeTransition { + let deletions: [ListViewDeleteItem] + let insertions: [ListViewInsertItem] + let updates: [ListViewUpdateItem] + let firstTime: Bool + let isLoading: Bool + let animated: Bool + let crossfade: Bool +} + +private func preparedTimezoneListNodeTransition(presentationData: PresentationData, from fromEntries: [TimezoneListEntry], to toEntries: [TimezoneListEntry], action: @escaping (String) -> Void, firstTime: Bool, isLoading: Bool, forceUpdate: Bool, animated: Bool, crossfade: Bool) -> TimezoneListNodeTransition { + let (deleteIndices, indicesAndItems, updateIndices) = mergeListsStableWithUpdates(leftList: fromEntries, rightList: toEntries, allUpdated: forceUpdate) + + let deletions = deleteIndices.map { ListViewDeleteItem(index: $0, directionHint: nil) } + let insertions = indicesAndItems.map { ListViewInsertItem(index: $0.0, previousIndex: $0.2, item: $0.1.item(presentationData: presentationData, searchMode: false, action: action), directionHint: nil) } + let updates = updateIndices.map { ListViewUpdateItem(index: $0.0, previousIndex: $0.2, item: $0.1.item(presentationData: presentationData, searchMode: false, action: action), directionHint: nil) } + + return TimezoneListNodeTransition(deletions: deletions, insertions: insertions, updates: updates, firstTime: firstTime, isLoading: isLoading, animated: animated, crossfade: crossfade) +} + +private final class TimezoneData { + struct Item { + var id: String + var offset: Int + var title: String + + init(id: String, offset: Int, title: String) { + self.id = id + self.offset = offset + self.title = title + } + } + + let items: [Item] + + init() { + let locale = Locale.current + var items: [Item] = [] + for (key, value) in TimeZone.abbreviationDictionary { + guard let timezone = TimeZone(abbreviation: key) else { + continue + } + if items.contains(where: { $0.id == timezone.identifier }) { + continue + } + items.append(Item( + id: timezone.identifier, + offset: timezone.secondsFromGMT(), + title: timezone.localizedName(for: .standard, locale: locale) ?? value + )) + } + self.items = items + } +} + +final class TimezoneSelectionScreenNode: ViewControllerTracingNode { + private let context: AccountContext + private let action: (String) -> Void + private var presentationData: PresentationData + private weak var navigationBar: NavigationBar? + private let requestActivateSearch: () -> Void + private let requestDeactivateSearch: () -> Void + private let present: (ViewController, Any?) -> Void + private let push: (ViewController) -> Void + private var timeZoneList: TimeZoneList? + + private var didSetReady = false + let _ready = ValuePromise() + + private var containerLayout: (ContainerViewLayout, CGFloat)? + let listNode: ListView + private var queuedTransitions: [TimezoneListNodeTransition] = [] + private var searchDisplayController: SearchDisplayController? + + private let presentationDataValue = Promise() + + private var listDisposable: Disposable? + + init(context: AccountContext, presentationData: PresentationData, navigationBar: NavigationBar, requestActivateSearch: @escaping () -> Void, requestDeactivateSearch: @escaping () -> Void, action: @escaping (String) -> Void, present: @escaping (ViewController, Any?) -> Void, push: @escaping (ViewController) -> Void) { + self.context = context + self.action = action + self.presentationData = presentationData + self.presentationDataValue.set(.single(presentationData)) + self.navigationBar = navigationBar + self.requestActivateSearch = requestActivateSearch + self.requestDeactivateSearch = requestDeactivateSearch + self.present = present + self.push = push + + self.listNode = ListView() + self.listNode.accessibilityPageScrolledString = { row, count in + return presentationData.strings.VoiceOver_ScrollStatus(row, count).string + } + + super.init() + + self.backgroundColor = presentationData.theme.list.plainBackgroundColor + self.addSubnode(self.listNode) + + let previousEntriesHolder = Atomic<([TimezoneListEntry], PresentationTheme, PresentationStrings)?>(value: nil) + self.listDisposable = (combineLatest(queue: .mainQueue(), + self.presentationDataValue.get(), + context.engine.accountData.cachedTimeZoneList() + ) + |> deliverOnMainQueue).start(next: { [weak self] presentationData, timeZoneList in + guard let strongSelf = self else { + return + } + + strongSelf.timeZoneList = timeZoneList + + var entries: [TimezoneListEntry] = [] + if let timeZoneList { + for item in timeZoneList.items { + entries.append(TimezoneListEntry( + id: item.id, + offset: Int(item.utcOffset), + title: item.title + )) + } + } + entries.sort() + + let previousEntriesAndPresentationData = previousEntriesHolder.swap((entries, presentationData.theme, presentationData.strings)) + let transition = preparedTimezoneListNodeTransition(presentationData: presentationData, from: previousEntriesAndPresentationData?.0 ?? [], to: entries, action: action, firstTime: previousEntriesAndPresentationData == nil, isLoading: entries.isEmpty, forceUpdate: previousEntriesAndPresentationData?.1 !== presentationData.theme || previousEntriesAndPresentationData?.2 !== presentationData.strings, animated: false, crossfade: false) + strongSelf.enqueueTransition(transition) + }) + } + + deinit { + self.listDisposable?.dispose() + } + + func updatePresentationData(_ presentationData: PresentationData) { + let stringsUpdated = self.presentationData.strings !== presentationData.strings + self.presentationData = presentationData + + if stringsUpdated { + if let snapshotView = self.view.snapshotView(afterScreenUpdates: false) { + self.view.addSubview(snapshotView) + snapshotView.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.2, removeOnCompletion: false, completion: { [weak snapshotView] _ in + snapshotView?.removeFromSuperview() + }) + } + } + + self.presentationDataValue.set(.single(presentationData)) + self.backgroundColor = presentationData.theme.list.plainBackgroundColor + self.searchDisplayController?.updatePresentationData(presentationData) + } + + func containerLayoutUpdated(_ layout: ContainerViewLayout, navigationBarHeight: CGFloat, transition: ContainedViewLayoutTransition) { + let hadValidLayout = self.containerLayout != nil + self.containerLayout = (layout, navigationBarHeight) + + var listInsets = layout.insets(options: [.input]) + listInsets.top += navigationBarHeight + + if let searchDisplayController = self.searchDisplayController { + searchDisplayController.containerLayoutUpdated(layout, navigationBarHeight: navigationBarHeight, transition: transition) + } + + self.listNode.bounds = CGRect(x: 0.0, y: 0.0, width: layout.size.width, height: layout.size.height) + self.listNode.position = CGPoint(x: layout.size.width / 2.0, y: layout.size.height / 2.0) + + let (duration, curve) = listViewAnimationDurationAndCurve(transition: transition) + let updateSizeAndInsets = ListViewUpdateSizeAndInsets(size: layout.size, insets: listInsets, duration: duration, curve: curve) + + self.listNode.transaction(deleteIndices: [], insertIndicesAndItems: [], updateIndicesAndItems: [], options: [.Synchronous, .LowLatency], scrollToItem: nil, updateSizeAndInsets: updateSizeAndInsets, stationaryItemRange: nil, updateOpaqueState: nil, completion: { _ in }) + + if !hadValidLayout { + self.dequeueTransitions() + } + } + + private func enqueueTransition(_ transition: TimezoneListNodeTransition) { + self.queuedTransitions.append(transition) + + if self.containerLayout != nil { + self.dequeueTransitions() + } + } + + private func dequeueTransitions() { + guard let _ = self.containerLayout else { + return + } + while !self.queuedTransitions.isEmpty { + let transition = self.queuedTransitions.removeFirst() + + var options = ListViewDeleteAndInsertOptions() + if transition.firstTime { + options.insert(.Synchronous) + options.insert(.LowLatency) + } else if transition.crossfade { + options.insert(.AnimateCrossfade) + } else if transition.animated { + options.insert(.AnimateInsertion) + } + self.listNode.transaction(deleteIndices: transition.deletions, insertIndicesAndItems: transition.insertions, updateIndicesAndItems: transition.updates, options: options, updateOpaqueState: nil, completion: { [weak self] _ in + if let strongSelf = self { + if !strongSelf.didSetReady { + strongSelf.didSetReady = true + strongSelf._ready.set(true) + } + } + }) + } + } + + func activateSearch(placeholderNode: SearchBarPlaceholderNode) { + guard let (containerLayout, navigationBarHeight) = self.containerLayout, self.searchDisplayController == nil else { + return + } + guard let timeZoneList = self.timeZoneList else { + return + } + + self.searchDisplayController = SearchDisplayController(presentationData: self.presentationData, contentNode: TimezoneListSearchContainerNode(context: self.context, timeZoneList: timeZoneList, action: self.action), inline: true, cancel: { [weak self] in + self?.requestDeactivateSearch() + }) + + self.searchDisplayController?.containerLayoutUpdated(containerLayout, navigationBarHeight: navigationBarHeight, transition: .immediate) + self.searchDisplayController?.activate(insertSubnode: { [weak self, weak placeholderNode] subnode, isSearchBar in + if let strongSelf = self, let strongPlaceholderNode = placeholderNode { + if isSearchBar { + strongPlaceholderNode.supernode?.insertSubnode(subnode, aboveSubnode: strongPlaceholderNode) + } else if let navigationBar = strongSelf.navigationBar { + strongSelf.insertSubnode(subnode, belowSubnode: navigationBar) + } + } + }, placeholder: placeholderNode) + } + + func deactivateSearch(placeholderNode: SearchBarPlaceholderNode) { + if let searchDisplayController = self.searchDisplayController { + searchDisplayController.deactivate(placeholder: placeholderNode) + self.searchDisplayController = nil + } + } + + func scrollToTop() { + self.listNode.transaction(deleteIndices: [], insertIndicesAndItems: [], updateIndicesAndItems: [], options: [.Synchronous, .LowLatency], scrollToItem: ListViewScrollToItem(index: 0, position: .top(0.0), animated: true, curve: .Default(duration: nil), directionHint: .Up), updateSizeAndInsets: nil, stationaryItemRange: nil, updateOpaqueState: nil, completion: { _ in }) + } +} diff --git a/submodules/TelegramUI/Components/ShareWithPeersScreen/Sources/ShareWithPeersScreen.swift b/submodules/TelegramUI/Components/ShareWithPeersScreen/Sources/ShareWithPeersScreen.swift index f16f5ac5192..37f1d5c338a 100644 --- a/submodules/TelegramUI/Components/ShareWithPeersScreen/Sources/ShareWithPeersScreen.swift +++ b/submodules/TelegramUI/Components/ShareWithPeersScreen/Sources/ShareWithPeersScreen.swift @@ -427,6 +427,10 @@ final class ShareWithPeersScreenComponent: Component { } } + func scrollViewWillBeginDragging(_ scrollView: UIScrollView) { + self.endEditing(true) + } + func scrollViewWillEndDragging(_ scrollView: UIScrollView, withVelocity velocity: CGPoint, targetContentOffset: UnsafeMutablePointer) { guard let itemLayout = self.itemLayout, let topOffsetDistance = self.topOffsetDistance else { return @@ -613,15 +617,15 @@ final class ShareWithPeersScreenComponent: Component { controller.present(alertController, in: .window(.root)) } - if groupTooLarge { - showCountLimitAlert() - return - } - var append = false if let index = self.selectedGroups.firstIndex(of: peer.id) { self.selectedGroups.remove(at: index) } else { + if groupTooLarge { + showCountLimitAlert() + return + } + self.selectedGroups.append(peer.id) append = true } @@ -801,7 +805,7 @@ final class ShareWithPeersScreenComponent: Component { } private func updateModalOverlayTransition(transition: Transition) { - guard let component = self.component, let environment = self.environment, let itemLayout = self.itemLayout else { + guard let component = self.component, let environment = self.environment, let itemLayout = self.itemLayout, !self.isDismissed else { return } @@ -2071,6 +2075,14 @@ final class ShareWithPeersScreenComponent: Component { self.selectedCategories.insert(.everyone) } self.state?.updated(transition: Transition(animation: .curve(duration: 0.35, curve: .spring))) + }, + isFocusedUpdated: { [weak self] isFocused in + guard let self else { + return + } + if isFocused { + self.scrollView.setContentOffset(CGPoint(x: 0.0, y: -self.scrollView.contentInset.top), animated: true) + } } )), environment: {}, @@ -2122,11 +2134,11 @@ final class ShareWithPeersScreenComponent: Component { transition.setFrame(view: self.dimView, frame: CGRect(origin: CGPoint(), size: availableSize)) if case .members = component.stateContext.subject { - self.dimView .isHidden = true + self.dimView.isHidden = true } else if case .channels = component.stateContext.subject { - self.dimView .isHidden = true + self.dimView.isHidden = true } else { - self.dimView .isHidden = false + self.dimView.isHidden = false } let categoryItemSize = self.categoryTemplateItem.update( diff --git a/submodules/TelegramUI/Components/SliderComponent/BUILD b/submodules/TelegramUI/Components/SliderComponent/BUILD new file mode 100644 index 00000000000..6ed98b159f0 --- /dev/null +++ b/submodules/TelegramUI/Components/SliderComponent/BUILD @@ -0,0 +1,22 @@ +load("@build_bazel_rules_swift//swift:swift.bzl", "swift_library") + +swift_library( + name = "SliderComponent", + module_name = "SliderComponent", + srcs = glob([ + "Sources/**/*.swift", + ]), + copts = [ + "-warnings-as-errors", + ], + deps = [ + "//submodules/AsyncDisplayKit", + "//submodules/Display", + "//submodules/TelegramPresentationData", + "//submodules/LegacyComponents", + "//submodules/ComponentFlow", + ], + visibility = [ + "//visibility:public", + ], +) diff --git a/submodules/TelegramUI/Components/SliderComponent/Sources/SliderComponent.swift b/submodules/TelegramUI/Components/SliderComponent/Sources/SliderComponent.swift new file mode 100644 index 00000000000..2494d74a76b --- /dev/null +++ b/submodules/TelegramUI/Components/SliderComponent/Sources/SliderComponent.swift @@ -0,0 +1,152 @@ +import Foundation +import UIKit +import Display +import AsyncDisplayKit +import TelegramPresentationData +import LegacyComponents +import ComponentFlow + +public final class SliderComponent: Component { + public let valueCount: Int + public let value: Int + public let trackBackgroundColor: UIColor + public let trackForegroundColor: UIColor + public let valueUpdated: (Int) -> Void + public let isTrackingUpdated: ((Bool) -> Void)? + + public init( + valueCount: Int, + value: Int, + trackBackgroundColor: UIColor, + trackForegroundColor: UIColor, + valueUpdated: @escaping (Int) -> Void, + isTrackingUpdated: ((Bool) -> Void)? = nil + ) { + self.valueCount = valueCount + self.value = value + self.trackBackgroundColor = trackBackgroundColor + self.trackForegroundColor = trackForegroundColor + self.valueUpdated = valueUpdated + self.isTrackingUpdated = isTrackingUpdated + } + + public static func ==(lhs: SliderComponent, rhs: SliderComponent) -> Bool { + if lhs.valueCount != rhs.valueCount { + return false + } + if lhs.value != rhs.value { + return false + } + if lhs.trackBackgroundColor != rhs.trackBackgroundColor { + return false + } + if lhs.trackForegroundColor != rhs.trackForegroundColor { + return false + } + return true + } + + public final class View: UIView { + private var sliderView: TGPhotoEditorSliderView? + + private var component: SliderComponent? + private weak var state: EmptyComponentState? + + override public init(frame: CGRect) { + super.init(frame: frame) + } + + required public init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + func update(component: SliderComponent, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + self.component = component + self.state = state + + let size = CGSize(width: availableSize.width, height: 44.0) + + var internalIsTrackingUpdated: ((Bool) -> Void)? + if let isTrackingUpdated = component.isTrackingUpdated { + internalIsTrackingUpdated = { [weak self] isTracking in + if let self { + if isTracking { + self.sliderView?.bordered = true + } else { + DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + 0.1, execute: { [weak self] in + self?.sliderView?.bordered = false + }) + } + isTrackingUpdated(isTracking) + } + } + } + + let sliderView: TGPhotoEditorSliderView + if let current = self.sliderView { + sliderView = current + } else { + sliderView = TGPhotoEditorSliderView() + sliderView.enablePanHandling = true + sliderView.trackCornerRadius = 2.0 + sliderView.lineSize = 4.0 + sliderView.dotSize = 5.0 + sliderView.minimumValue = 0.0 + sliderView.startValue = 0.0 + sliderView.disablesInteractiveTransitionGestureRecognizer = true + sliderView.maximumValue = CGFloat(component.valueCount - 1) + sliderView.positionsCount = component.valueCount + sliderView.useLinesForPositions = true + + sliderView.backgroundColor = nil + sliderView.isOpaque = false + sliderView.backColor = component.trackBackgroundColor + sliderView.startColor = component.trackBackgroundColor + sliderView.trackColor = component.trackForegroundColor + sliderView.knobImage = 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: -3.0), blur: 12.0, color: UIColor(white: 0.0, alpha: 0.25).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))) + }) + + sliderView.frame = CGRect(origin: CGPoint(x: 0.0, y: 0.0), size: size) + sliderView.hitTestEdgeInsets = UIEdgeInsets(top: -sliderView.frame.minX, left: 0.0, bottom: 0.0, right: -sliderView.frame.minX) + + + sliderView.disablesInteractiveTransitionGestureRecognizer = true + sliderView.addTarget(self, action: #selector(self.sliderValueChanged), for: .valueChanged) + sliderView.layer.allowsGroupOpacity = true + self.sliderView = sliderView + self.addSubview(sliderView) + } + sliderView.value = CGFloat(component.value) + sliderView.interactionBegan = { + internalIsTrackingUpdated?(true) + } + sliderView.interactionEnded = { + internalIsTrackingUpdated?(false) + } + + transition.setFrame(view: sliderView, frame: CGRect(origin: CGPoint(x: 0.0, y: 0.0), size: CGSize(width: availableSize.width, height: 44.0))) + sliderView.hitTestEdgeInsets = UIEdgeInsets(top: 0.0, left: 0.0, bottom: 0.0, right: 0.0) + + return size + } + + @objc private func sliderValueChanged() { + guard let component = self.component, let sliderView = self.sliderView else { + return + } + component.valueUpdated(Int(sliderView.value)) + } + } + + public func makeView() -> View { + return View(frame: CGRect()) + } + + public func update(view: View, availableSize: CGSize, state: EmptyComponentState, environment: Environment, transition: Transition) -> CGSize { + return view.update(component: self, availableSize: availableSize, state: state, environment: environment, transition: transition) + } +} diff --git a/submodules/TelegramUI/Components/Stories/AvatarStoryIndicatorComponent/Sources/AvatarStoryIndicatorComponent.swift b/submodules/TelegramUI/Components/Stories/AvatarStoryIndicatorComponent/Sources/AvatarStoryIndicatorComponent.swift index ebe68f2d334..5cd33daa875 100644 --- a/submodules/TelegramUI/Components/Stories/AvatarStoryIndicatorComponent/Sources/AvatarStoryIndicatorComponent.swift +++ b/submodules/TelegramUI/Components/Stories/AvatarStoryIndicatorComponent/Sources/AvatarStoryIndicatorComponent.swift @@ -345,16 +345,17 @@ public final class AvatarStoryIndicatorComponent: Component { } let colorSpace = CGColorSpaceCreateDeviceRGB() - let gradient = CGGradient(colorsSpace: colorSpace, colors: colors as CFArray, locations: &locations)! - - context.drawLinearGradient(gradient, start: CGPoint(x: 0.0, y: 0.0), end: CGPoint(x: 0.0, y: size.height), options: CGGradientDrawingOptions()) + if let gradient = CGGradient(colorsSpace: colorSpace, colors: colors as CFArray, locations: &locations) { + context.drawLinearGradient(gradient, start: CGPoint(x: 0.0, y: 0.0), end: CGPoint(x: 0.0, y: size.height), options: CGGradientDrawingOptions()) + } } } } else { let lineWidth: CGFloat = component.hasUnseen ? component.activeLineWidth : component.inactiveLineWidth context.setLineWidth(lineWidth) if component.isRoundedRect { - context.addPath(UIBezierPath(roundedRect: CGRect(origin: CGPoint(x: size.width * 0.5 - diameter * 0.5, y: size.height * 0.5 - diameter * 0.5), size: size).insetBy(dx: lineWidth * 0.5, dy: lineWidth * 0.5), cornerRadius: floor(diameter * 0.25)).cgPath) + let path = UIBezierPath(roundedRect: CGRect(origin: CGPoint(x: size.width * 0.5 - diameter * 0.5, y: size.height * 0.5 - diameter * 0.5), size: size).insetBy(dx: lineWidth * 0.5, dy: lineWidth * 0.5), cornerRadius: floor(diameter * 0.27)) + context.addPath(path.cgPath) } else { context.addEllipse(in: CGRect(origin: CGPoint(x: size.width * 0.5 - diameter * 0.5, y: size.height * 0.5 - diameter * 0.5), size: size).insetBy(dx: lineWidth * 0.5, dy: lineWidth * 0.5)) } diff --git a/submodules/TelegramUI/Components/Stories/PeerListItemComponent/BUILD b/submodules/TelegramUI/Components/Stories/PeerListItemComponent/BUILD index 0ad1d287a5c..2841b2a498e 100644 --- a/submodules/TelegramUI/Components/Stories/PeerListItemComponent/BUILD +++ b/submodules/TelegramUI/Components/Stories/PeerListItemComponent/BUILD @@ -29,6 +29,8 @@ swift_library( "//submodules/ContextUI", "//submodules/TextFormat", "//submodules/PhotoResources", + "//submodules/TelegramUI/Components/ListSectionComponent", + "//submodules/TelegramUI/Components/ListItemSwipeOptionContainer", ], visibility = [ "//visibility:public", diff --git a/submodules/TelegramUI/Components/Stories/PeerListItemComponent/Sources/PeerListItemComponent.swift b/submodules/TelegramUI/Components/Stories/PeerListItemComponent/Sources/PeerListItemComponent.swift index c23e8cae70b..f23ef6eb2a6 100644 --- a/submodules/TelegramUI/Components/Stories/PeerListItemComponent/Sources/PeerListItemComponent.swift +++ b/submodules/TelegramUI/Components/Stories/PeerListItemComponent/Sources/PeerListItemComponent.swift @@ -1,4 +1,5 @@ import Foundation +import Foundation import UIKit import Display import AsyncDisplayKit @@ -19,6 +20,8 @@ import ContextUI import EmojiTextAttachmentView import TextFormat import PhotoResources +import ListSectionComponent +import ListItemSwipeOptionContainer private let avatarFont = avatarPlaceholderFont(size: 15.0) private let readIconImage: UIImage? = generateTintedImage(image: UIImage(bundleImageName: "Chat/Message/MenuReadIcon"), color: .white)?.withRenderingMode(.alwaysTemplate) @@ -64,6 +67,70 @@ public final class PeerListItemComponent: Component { case check } + public struct Avatar: Equatable { + public var icon: String + public var color: AvatarBackgroundColor + public var clipStyle: AvatarNodeClipStyle + + public init(icon: String, color: AvatarBackgroundColor, clipStyle: AvatarNodeClipStyle) { + self.icon = icon + self.color = color + self.clipStyle = clipStyle + } + } + + public final class InlineAction: Equatable { + public enum Color: Equatable { + case destructive + } + + public let id: AnyHashable + public let title: String + public let color: Color + public let action: () -> Void + + public init(id: AnyHashable, title: String, color: Color, action: @escaping () -> Void) { + self.id = id + self.title = title + self.color = color + self.action = action + } + + public static func ==(lhs: InlineAction, rhs: InlineAction) -> Bool { + if lhs === rhs { + return true + } + if lhs.id != rhs.id { + return false + } + if lhs.title != rhs.title { + return false + } + if lhs.color != rhs.color { + return false + } + return true + } + } + + public final class InlineActionsState: Equatable { + public let actions: [InlineAction] + + public init(actions: [InlineAction]) { + self.actions = actions + } + + public static func ==(lhs: InlineActionsState, rhs: InlineActionsState) -> Bool { + if lhs === rhs { + return true + } + if lhs.actions != rhs.actions { + return false + } + return true + } + } + public final class Reaction: Equatable { public let reaction: MessageReaction.Reaction public let file: TelegramMediaFile? @@ -103,6 +170,7 @@ public final class PeerListItemComponent: Component { let style: Style let sideInset: CGFloat let title: String + let avatar: Avatar? let peer: EnginePeer? let storyStats: PeerStoryStats? let subtitle: String? @@ -117,6 +185,7 @@ public final class PeerListItemComponent: Component { let isEnabled: Bool let hasNext: Bool let action: (EnginePeer, EngineMessage.Id?, UIView?) -> Void + let inlineActions: InlineActionsState? let contextAction: ((EnginePeer, ContextExtractedContentContainingView, ContextGesture) -> Void)? let openStories: ((EnginePeer, AvatarNode) -> Void)? @@ -127,6 +196,7 @@ public final class PeerListItemComponent: Component { style: Style, sideInset: CGFloat, title: String, + avatar: Avatar? = nil, peer: EnginePeer?, storyStats: PeerStoryStats? = nil, subtitle: String?, @@ -141,6 +211,7 @@ public final class PeerListItemComponent: Component { isEnabled: Bool = true, hasNext: Bool, action: @escaping (EnginePeer, EngineMessage.Id?, UIView?) -> Void, + inlineActions: InlineActionsState? = nil, contextAction: ((EnginePeer, ContextExtractedContentContainingView, ContextGesture) -> Void)? = nil, openStories: ((EnginePeer, AvatarNode) -> Void)? = nil ) { @@ -150,6 +221,7 @@ public final class PeerListItemComponent: Component { self.style = style self.sideInset = sideInset self.title = title + self.avatar = avatar self.peer = peer self.storyStats = storyStats self.subtitle = subtitle @@ -164,6 +236,7 @@ public final class PeerListItemComponent: Component { self.isEnabled = isEnabled self.hasNext = hasNext self.action = action + self.inlineActions = inlineActions self.contextAction = contextAction self.openStories = openStories } @@ -187,6 +260,9 @@ public final class PeerListItemComponent: Component { if lhs.title != rhs.title { return false } + if lhs.avatar != rhs.avatar { + return false + } if lhs.peer != rhs.peer { return false } @@ -226,17 +302,23 @@ public final class PeerListItemComponent: Component { if lhs.hasNext != rhs.hasNext { return false } + if lhs.inlineActions != rhs.inlineActions { + return false + } return true } - public final class View: ContextControllerSourceView { + public final class View: ContextControllerSourceView, ListSectionComponent.ChildView { private let extractedContainerView: ContextExtractedContentContainingView private let containerButton: HighlightTrackingButton + private let swipeOptionContainer: ListItemSwipeOptionContainer + private let title = ComponentView() private let label = ComponentView() private let separatorLayer: SimpleLayer private let avatarNode: AvatarNode + private var avatarImageView: UIImageView? private let avatarButtonView: HighlightTrackingButton private var avatarIcon: ComponentView? @@ -278,13 +360,19 @@ public final class PeerListItemComponent: Component { private var isExtractedToContextMenu: Bool = false + public var customUpdateIsHighlighted: ((Bool) -> Void)? + public private(set) var separatorInset: CGFloat = 0.0 + override init(frame: CGRect) { self.separatorLayer = SimpleLayer() self.extractedContainerView = ContextExtractedContentContainingView() self.containerButton = HighlightTrackingButton() + self.containerButton.layer.anchorPoint = CGPoint() self.containerButton.isExclusiveTouch = true + self.swipeOptionContainer = ListItemSwipeOptionContainer(frame: CGRect()) + self.avatarNode = AvatarNode(font: avatarFont) self.avatarNode.isLayerBacked = false self.avatarNode.isUserInteractionEnabled = false @@ -296,7 +384,9 @@ public final class PeerListItemComponent: Component { self.addSubview(self.extractedContainerView) self.targetViewForActivationProgress = self.extractedContainerView.contentView - self.extractedContainerView.contentView.addSubview(self.containerButton) + self.extractedContainerView.contentView.addSubview(self.swipeOptionContainer) + + self.swipeOptionContainer.addSubview(self.containerButton) self.layer.addSublayer(self.separatorLayer) self.containerButton.layer.addSublayer(self.avatarNode.layer) @@ -336,6 +426,34 @@ public final class PeerListItemComponent: Component { } component.contextAction?(peer, self.extractedContainerView, gesture) } + + self.containerButton.highligthedChanged = { [weak self] highlighted in + guard let self else { + return + } + if let customUpdateIsHighlighted = self.customUpdateIsHighlighted { + customUpdateIsHighlighted(highlighted) + } + } + + self.swipeOptionContainer.updateRevealOffset = { [weak self] offset, transition in + guard let self else { + return + } + transition.setBounds(view: self.containerButton, bounds: CGRect(origin: CGPoint(x: -offset, y: 0.0), size: self.containerButton.bounds.size)) + } + self.swipeOptionContainer.revealOptionSelected = { [weak self] option, animated in + guard let self, let component = self.component else { + return + } + guard let inlineActions = component.inlineActions else { + return + } + self.swipeOptionContainer.setRevealOptionsOpened(false, animated: animated) + if let inlineAction = inlineActions.actions.first(where: { $0.id == option.key }) { + inlineAction.action() + } + } } required init?(coder: NSCoder) { @@ -560,6 +678,27 @@ public final class PeerListItemComponent: Component { transition.setFrame(view: self.avatarButtonView, frame: avatarFrame) var statusIcon: EmojiStatusComponent.Content? + + if let avatar = component.avatar { + let avatarImageView: UIImageView + if let current = self.avatarImageView { + avatarImageView = current + } else { + avatarImageView = UIImageView() + self.avatarImageView = avatarImageView + self.containerButton.addSubview(avatarImageView) + } + if previousComponent?.avatar != avatar { + avatarImageView.image = generateAvatarImage(size: avatarFrame.size, icon: generateTintedImage(image: UIImage(bundleImageName: avatar.icon), color: .white), cornerRadius: 12.0, color: avatar.color) + } + transition.setFrame(view: avatarImageView, frame: avatarFrame) + } else { + if let avatarImageView = self.avatarImageView { + self.avatarImageView = nil + avatarImageView.removeFromSuperview() + } + } + if let peer = component.peer { let clipStyle: AvatarNodeClipStyle if case let .channel(channel) = peer, channel.flags.contains(.isForum) { @@ -596,6 +735,7 @@ public final class PeerListItemComponent: Component { lineWidth: 1.33, inactiveLineWidth: 1.33 ), transition: transition) + self.avatarNode.isHidden = false if peer.isScam { statusIcon = .text(color: component.theme.chat.message.incoming.scamColor, string: component.strings.Message_ScamAccount.uppercased()) @@ -608,6 +748,8 @@ public final class PeerListItemComponent: Component { } else if peer.isPremium { statusIcon = .premium(color: component.theme.list.itemAccentColor) } + } else { + self.avatarNode.isHidden = true } let previousTitleFrame = self.title.view?.frame @@ -951,7 +1093,38 @@ public final class PeerListItemComponent: Component { self.extractedContainerView.contentRect = resultBounds let containerFrame = CGRect(origin: CGPoint(x: contextInset, y: verticalInset), size: CGSize(width: availableSize.width - contextInset * 2.0, height: height - verticalInset * 2.0)) - transition.setFrame(view: self.containerButton, frame: containerFrame) + + let swipeOptionContainerFrame = CGRect(origin: CGPoint(x: 0.0, y: 0.0), size: CGSize(width: availableSize.width, height: height)) + transition.setFrame(view: self.swipeOptionContainer, frame: swipeOptionContainerFrame) + + transition.setPosition(view: self.containerButton, position: containerFrame.origin) + transition.setBounds(view: self.containerButton, bounds: CGRect(origin: self.containerButton.bounds.origin, size: containerFrame.size)) + + self.separatorInset = leftInset + + self.swipeOptionContainer.updateLayout(size: swipeOptionContainerFrame.size, leftInset: 0.0, rightInset: 0.0) + + var rightOptions: [ListItemSwipeOptionContainer.Option] = [] + if let inlineActions = component.inlineActions { + rightOptions = inlineActions.actions.map { action in + let color: UIColor + let textColor: UIColor + switch action.color { + case .destructive: + color = component.theme.list.itemDisclosureActions.destructive.fillColor + textColor = component.theme.list.itemDisclosureActions.destructive.foregroundColor + } + + return ListItemSwipeOptionContainer.Option( + key: action.id, + title: action.title, + icon: .none, + color: color, + textColor: textColor + ) + } + } + self.swipeOptionContainer.setRevealOptions(([], rightOptions)) return CGSize(width: availableSize.width, height: height) } diff --git a/submodules/TelegramUI/Components/Stories/StoryContainerScreen/Sources/StoryAuthorInfoComponent.swift b/submodules/TelegramUI/Components/Stories/StoryContainerScreen/Sources/StoryAuthorInfoComponent.swift index 810596bab85..3f3fe6dbcbe 100644 --- a/submodules/TelegramUI/Components/Stories/StoryContainerScreen/Sources/StoryAuthorInfoComponent.swift +++ b/submodules/TelegramUI/Components/Stories/StoryContainerScreen/Sources/StoryAuthorInfoComponent.swift @@ -134,6 +134,9 @@ final class StoryAuthorInfoComponent: Component { if timeString.count < 6 { combinedString.append(NSAttributedString(string: " • \(timeString)", font: Font.regular(11.0), textColor: subtitleColor)) } + if component.isEdited { + combinedString.append(NSAttributedString(string: " • \(component.strings.Story_HeaderEdited)", font: Font.regular(11.0), textColor: subtitleColor)) + } subtitle = combinedString subtitleTruncationType = .middle } else { diff --git a/submodules/TelegramUI/Components/Stories/StoryContainerScreen/Sources/StoryChatContent.swift b/submodules/TelegramUI/Components/Stories/StoryContainerScreen/Sources/StoryChatContent.swift index 1146bf3101c..dc8f056aabd 100644 --- a/submodules/TelegramUI/Components/Stories/StoryContainerScreen/Sources/StoryChatContent.swift +++ b/submodules/TelegramUI/Components/Stories/StoryContainerScreen/Sources/StoryChatContent.swift @@ -990,7 +990,7 @@ public final class StoryContentContextImpl: StoryContentContext { } var selectedMedia: EngineMedia - if let slice = stateValue.slice, let alternativeMedia = item.alternativeMedia, !slice.additionalPeerData.preferHighQualityStories { + if let slice = stateValue.slice, let alternativeMedia = item.alternativeMedia, (!slice.additionalPeerData.preferHighQualityStories && !item.isMy) { selectedMedia = alternativeMedia } else { selectedMedia = item.media @@ -1642,7 +1642,7 @@ public final class PeerStoryListContentContextImpl: StoryContentContext { } var selectedMedia: EngineMedia - if let alternativeMedia = item.alternativeMedia, !preferHighQualityStories { + if let alternativeMedia = item.alternativeMedia, (!preferHighQualityStories && !item.isMy) { selectedMedia = alternativeMedia } else { selectedMedia = item.media @@ -2880,7 +2880,7 @@ public final class RepostStoriesContentContextImpl: StoryContentContext { } var selectedMedia: EngineMedia - if let slice = stateValue.slice, let alternativeMedia = item.alternativeMedia, !slice.additionalPeerData.preferHighQualityStories { + if let slice = stateValue.slice, let alternativeMedia = item.alternativeMedia, (!slice.additionalPeerData.preferHighQualityStories && !item.isMy) { selectedMedia = alternativeMedia } else { selectedMedia = item.media diff --git a/submodules/TelegramUI/Components/Stories/StoryContainerScreen/Sources/StoryItemContentComponent.swift b/submodules/TelegramUI/Components/Stories/StoryContainerScreen/Sources/StoryItemContentComponent.swift index 76fb1778b63..fed8cb12203 100644 --- a/submodules/TelegramUI/Components/Stories/StoryContainerScreen/Sources/StoryItemContentComponent.swift +++ b/submodules/TelegramUI/Components/Stories/StoryContainerScreen/Sources/StoryItemContentComponent.swift @@ -593,7 +593,7 @@ final class StoryItemContentComponent: Component { let selectedMedia: EngineMedia var messageMedia: EngineMedia? - if !component.preferHighQuality, let alternativeMedia = component.item.alternativeMedia { + if !component.preferHighQuality, !component.item.isMy, let alternativeMedia = component.item.alternativeMedia { selectedMedia = alternativeMedia switch alternativeMedia { diff --git a/submodules/TelegramUI/Components/Stories/StoryContainerScreen/Sources/StoryItemSetContainerComponent.swift b/submodules/TelegramUI/Components/Stories/StoryContainerScreen/Sources/StoryItemSetContainerComponent.swift index 14aea4a93b9..412e4c50ff3 100644 --- a/submodules/TelegramUI/Components/Stories/StoryContainerScreen/Sources/StoryItemSetContainerComponent.swift +++ b/submodules/TelegramUI/Components/Stories/StoryContainerScreen/Sources/StoryItemSetContainerComponent.swift @@ -5420,6 +5420,7 @@ public final class StoryItemSetContainerComponent: Component { var updateProgressImpl: ((Float) -> Void)? let controller = MediaEditorScreen( context: context, + mode: .storyEditor, subject: subject, isEditing: !repost, forwardSource: repost ? (component.slice.peer, item) : nil, @@ -6857,7 +6858,7 @@ public final class StoryItemSetContainerComponent: Component { }))) } - if case let .file(file) = component.slice.item.storyItem.media, file.isVideo { + if !component.slice.item.storyItem.isMy, case let .file(file) = component.slice.item.storyItem.media, file.isVideo { let isHq = component.slice.additionalPeerData.preferHighQualityStories items.append(.action(ContextMenuActionItem(text: isHq ? component.strings.Story_ContextMenuSD : component.strings.Story_ContextMenuHD, icon: { theme in if isHq { diff --git a/submodules/TelegramUI/Components/Stories/StoryContainerScreen/Sources/StoryItemSetContainerViewSendMessage.swift b/submodules/TelegramUI/Components/Stories/StoryContainerScreen/Sources/StoryItemSetContainerViewSendMessage.swift index d374b63b525..d257ff5efa3 100644 --- a/submodules/TelegramUI/Components/Stories/StoryContainerScreen/Sources/StoryItemSetContainerViewSendMessage.swift +++ b/submodules/TelegramUI/Components/Stories/StoryContainerScreen/Sources/StoryItemSetContainerViewSendMessage.swift @@ -179,6 +179,7 @@ final class StoryItemSetContainerSendMessage { self.inputPanelExternalState?.deleteBackward() } }, + openStickerEditor: {}, presentController: { [weak self] c, a in if let self { self.view?.component?.controller()?.present(c, in: .window(.root), with: a) diff --git a/submodules/TelegramUI/Components/Stories/StoryPeerListComponent/Sources/StoryPeerListComponent.swift b/submodules/TelegramUI/Components/Stories/StoryPeerListComponent/Sources/StoryPeerListComponent.swift index d0af5a34fc6..0b79cb0ebb9 100644 --- a/submodules/TelegramUI/Components/Stories/StoryPeerListComponent/Sources/StoryPeerListComponent.swift +++ b/submodules/TelegramUI/Components/Stories/StoryPeerListComponent/Sources/StoryPeerListComponent.swift @@ -590,7 +590,7 @@ public final class StoryPeerListComponent: Component { case .premium: statusContent = .premium(color: component.theme.list.itemAccentColor) case let .emoji(emoji): - statusContent = .animation(content: .customEmoji(fileId: emoji.fileId), size: CGSize(width: 22.0, height: 22.0), placeholderColor: component.theme.list.mediaPlaceholderColor, themeColor: component.theme.list.itemAccentColor, loopMode: .count(2)) + statusContent = .animation(content: .customEmoji(fileId: emoji.fileId), size: CGSize(width: 44.0, height: 44.0), placeholderColor: component.theme.list.mediaPlaceholderColor, themeColor: component.theme.list.itemAccentColor, loopMode: .count(2)) } var animateStatusTransition = false diff --git a/submodules/TelegramUI/Components/TextFieldComponent/Sources/TextFieldComponent.swift b/submodules/TelegramUI/Components/TextFieldComponent/Sources/TextFieldComponent.swift index a4779139ada..7fa657812c9 100644 --- a/submodules/TelegramUI/Components/TextFieldComponent/Sources/TextFieldComponent.swift +++ b/submodules/TelegramUI/Components/TextFieldComponent/Sources/TextFieldComponent.swift @@ -96,6 +96,8 @@ public final class TextFieldComponent: Component { public let customInputView: UIView? public let resetText: NSAttributedString? public let isOneLineWhenUnfocused: Bool + public let characterLimit: Int? + public let allowEmptyLines: Bool public let formatMenuAvailability: FormatMenuAvailability public let lockedFormatAction: () -> Void public let present: (ViewController) -> Void @@ -112,6 +114,8 @@ public final class TextFieldComponent: Component { customInputView: UIView?, resetText: NSAttributedString?, isOneLineWhenUnfocused: Bool, + characterLimit: Int? = nil, + allowEmptyLines: Bool = true, formatMenuAvailability: FormatMenuAvailability, lockedFormatAction: @escaping () -> Void, present: @escaping (ViewController) -> Void, @@ -127,6 +131,8 @@ public final class TextFieldComponent: Component { self.customInputView = customInputView self.resetText = resetText self.isOneLineWhenUnfocused = isOneLineWhenUnfocused + self.characterLimit = characterLimit + self.allowEmptyLines = allowEmptyLines self.formatMenuAvailability = formatMenuAvailability self.lockedFormatAction = lockedFormatAction self.present = present @@ -161,6 +167,12 @@ public final class TextFieldComponent: Component { if lhs.isOneLineWhenUnfocused != rhs.isOneLineWhenUnfocused { return false } + if lhs.characterLimit != rhs.characterLimit { + return false + } + if lhs.allowEmptyLines != rhs.allowEmptyLines { + return false + } if lhs.formatMenuAvailability != rhs.formatMenuAvailability { return false } @@ -193,7 +205,7 @@ public final class TextFieldComponent: Component { private let ellipsisView = ComponentView() - private var inputState: InputState { + public var inputState: InputState { let selectionRange: Range = self.textView.selectedRange.location ..< (self.textView.selectedRange.location + self.textView.selectedRange.length) return InputState(inputText: stateAttributedStringForText(self.textView.attributedText ?? NSAttributedString()), selectionRange: selectionRange) } @@ -537,6 +549,24 @@ public final class TextFieldComponent: Component { } public func chatInputTextNode(shouldChangeTextIn range: NSRange, replacementText text: String) -> Bool { + guard let component = self.component else { + return true + } + if let characterLimit = component.characterLimit { + let string = self.inputState.inputText.string as NSString + let updatedString = string.replacingCharacters(in: range, with: text) + if (updatedString as NSString).length > characterLimit { + return false + } + } + if !component.allowEmptyLines { + let string = self.inputState.inputText.string as NSString + let updatedString = string.replacingCharacters(in: range, with: text) + if updatedString.range(of: "\n\n") != nil { + return false + } + } + return true } diff --git a/submodules/TelegramUI/Components/TimeSelectionActionSheet/BUILD b/submodules/TelegramUI/Components/TimeSelectionActionSheet/BUILD new file mode 100644 index 00000000000..b4680d882db --- /dev/null +++ b/submodules/TelegramUI/Components/TimeSelectionActionSheet/BUILD @@ -0,0 +1,25 @@ +load("@build_bazel_rules_swift//swift:swift.bzl", "swift_library") + +swift_library( + name = "TimeSelectionActionSheet", + module_name = "TimeSelectionActionSheet", + srcs = glob([ + "Sources/**/*.swift", + ]), + copts = [ + "-warnings-as-errors", + ], + deps = [ + "//submodules/Display", + "//submodules/AsyncDisplayKit", + "//submodules/SSignalKit/SwiftSignalKit", + "//submodules/TelegramCore", + "//submodules/TelegramPresentationData", + "//submodules/TelegramStringFormatting", + "//submodules/AccountContext", + "//submodules/UIKitRuntimeUtils", + ], + visibility = [ + "//visibility:public", + ], +) diff --git a/submodules/TelegramUI/Components/TimeSelectionActionSheet/Sources/TimeSelectionActionSheet.swift b/submodules/TelegramUI/Components/TimeSelectionActionSheet/Sources/TimeSelectionActionSheet.swift new file mode 100644 index 00000000000..8ec3ff48af3 --- /dev/null +++ b/submodules/TelegramUI/Components/TimeSelectionActionSheet/Sources/TimeSelectionActionSheet.swift @@ -0,0 +1,132 @@ +import Foundation +import Display +import AsyncDisplayKit +import UIKit +import SwiftSignalKit +import TelegramCore +import TelegramPresentationData +import TelegramStringFormatting +import AccountContext +import UIKitRuntimeUtils + +public final class TimeSelectionActionSheet: ActionSheetController { + private var presentationDisposable: Disposable? + + private let _ready = Promise() + override public var ready: Promise { + return self._ready + } + + public init(context: AccountContext, currentValue: Int32, emptyTitle: String? = nil, applyValue: @escaping (Int32?) -> Void) { + let presentationData = context.sharedContext.currentPresentationData.with { $0 } + let strings = presentationData.strings + + super.init(theme: ActionSheetControllerTheme(presentationData: presentationData)) + + self.presentationDisposable = context.sharedContext.presentationData.start(next: { [weak self] presentationData in + if let strongSelf = self { + strongSelf.theme = ActionSheetControllerTheme(presentationData: presentationData) + } + }) + + self._ready.set(.single(true)) + + var updatedValue = currentValue + var items: [ActionSheetItem] = [] + items.append(TimeSelectionActionSheetItem(strings: strings, currentValue: currentValue, valueChanged: { value in + updatedValue = value + })) + if let emptyTitle = emptyTitle { + items.append(ActionSheetButtonItem(title: emptyTitle, action: { [weak self] in + self?.dismissAnimated() + applyValue(nil) + })) + } + items.append(ActionSheetButtonItem(title: strings.Wallpaper_Set, action: { [weak self] in + self?.dismissAnimated() + applyValue(updatedValue) + })) + self.setItemGroups([ + ActionSheetItemGroup(items: items), + ActionSheetItemGroup(items: [ + ActionSheetButtonItem(title: strings.Common_Cancel, action: { [weak self] in + self?.dismissAnimated() + }), + ]) + ]) + } + + required public init(coder aDecoder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + deinit { + self.presentationDisposable?.dispose() + } +} + +private final class TimeSelectionActionSheetItem: ActionSheetItem { + let strings: PresentationStrings + + let currentValue: Int32 + let valueChanged: (Int32) -> Void + + init(strings: PresentationStrings, currentValue: Int32, valueChanged: @escaping (Int32) -> Void) { + self.strings = strings + self.currentValue = currentValue + self.valueChanged = valueChanged + } + + func node(theme: ActionSheetControllerTheme) -> ActionSheetItemNode { + return TimeSelectionActionSheetItemNode(theme: theme, strings: self.strings, currentValue: self.currentValue, valueChanged: self.valueChanged) + } + + func updateNode(_ node: ActionSheetItemNode) { + } +} + +private final class TimeSelectionActionSheetItemNode: ActionSheetItemNode { + private let theme: ActionSheetControllerTheme + private let strings: PresentationStrings + + private let valueChanged: (Int32) -> Void + private let pickerView: UIDatePicker + + init(theme: ActionSheetControllerTheme, strings: PresentationStrings, currentValue: Int32, valueChanged: @escaping (Int32) -> Void) { + self.theme = theme + self.strings = strings + self.valueChanged = valueChanged + + UILabel.setDateLabel(theme.primaryTextColor) + + self.pickerView = UIDatePicker() + self.pickerView.datePickerMode = .countDownTimer + self.pickerView.datePickerMode = .time + self.pickerView.timeZone = TimeZone(secondsFromGMT: 0) + self.pickerView.date = Date(timeIntervalSince1970: Double(currentValue)) + self.pickerView.locale = Locale.current + if #available(iOS 13.4, *) { + self.pickerView.preferredDatePickerStyle = .wheels + } + self.pickerView.setValue(theme.primaryTextColor, forKey: "textColor") + + super.init(theme: theme) + + self.view.addSubview(self.pickerView) + self.pickerView.addTarget(self, action: #selector(self.datePickerUpdated), for: .valueChanged) + } + + public override func updateLayout(constrainedSize: CGSize, transition: ContainedViewLayoutTransition) -> CGSize { + let size = CGSize(width: constrainedSize.width, height: 216.0) + + self.pickerView.frame = CGRect(origin: CGPoint(), size: size) + + self.updateInternalLayout(size, constrainedSize: constrainedSize) + return size + } + + @objc private func datePickerUpdated() { + self.valueChanged(Int32(self.pickerView.date.timeIntervalSince1970)) + } +} + diff --git a/submodules/TelegramUI/Components/TokenListTextField/Sources/TokenListTextField.swift b/submodules/TelegramUI/Components/TokenListTextField/Sources/TokenListTextField.swift index a5c33584308..d46862dac97 100644 --- a/submodules/TelegramUI/Components/TokenListTextField/Sources/TokenListTextField.swift +++ b/submodules/TelegramUI/Components/TokenListTextField/Sources/TokenListTextField.swift @@ -88,6 +88,7 @@ public final class TokenListTextField: Component { public let tokens: [Token] public let sideInset: CGFloat public let deleteToken: (AnyHashable) -> Void + public let isFocusedUpdated: (Bool) -> Void public init( externalState: ExternalState, @@ -96,7 +97,8 @@ public final class TokenListTextField: Component { placeholder: String, tokens: [Token], sideInset: CGFloat, - deleteToken: @escaping (AnyHashable) -> Void + deleteToken: @escaping (AnyHashable) -> Void, + isFocusedUpdated: @escaping (Bool) -> Void = { _ in } ) { self.externalState = externalState self.context = context @@ -105,6 +107,7 @@ public final class TokenListTextField: Component { self.tokens = tokens self.sideInset = sideInset self.deleteToken = deleteToken + self.isFocusedUpdated = isFocusedUpdated } public static func ==(lhs: TokenListTextField, rhs: TokenListTextField) -> Bool { @@ -191,6 +194,7 @@ public final class TokenListTextField: Component { guard let self else { return } + self.component?.isFocusedUpdated(self.tokenListNode?.isFocused ?? false) self.componentState?.updated(transition: Transition(animation: .curve(duration: 0.35, curve: .spring))) } diff --git a/submodules/TelegramUI/Components/VideoMessageCameraScreen/Sources/VideoMessageCameraScreen.swift b/submodules/TelegramUI/Components/VideoMessageCameraScreen/Sources/VideoMessageCameraScreen.swift index 1e6a8f738ce..1f0abf27446 100644 --- a/submodules/TelegramUI/Components/VideoMessageCameraScreen/Sources/VideoMessageCameraScreen.swift +++ b/submodules/TelegramUI/Components/VideoMessageCameraScreen/Sources/VideoMessageCameraScreen.swift @@ -263,23 +263,25 @@ private final class VideoMessageCameraScreenComponent: CombinedComponent { controller.updateCameraState({ $0.updatedRecording(pressing ? .holding : .handsFree).updatedDuration(initialDuration) }, transition: .spring(duration: 0.4)) controller.node.withReadyCamera(isFirstTime: !controller.node.cameraIsActive) { - self.resultDisposable.set((camera.startRecording() - |> deliverOnMainQueue).start(next: { [weak self] recordingData in - let duration = initialDuration + recordingData.duration - if let self, let controller = self.getController() { - controller.updateCameraState({ $0.updatedDuration(duration) }, transition: .easeInOut(duration: 0.1)) - if isFirstRecording { - controller.node.setupLiveUpload(filePath: recordingData.filePath) + Queue.mainQueue().after(0.15) { + self.resultDisposable.set((camera.startRecording() + |> deliverOnMainQueue).start(next: { [weak self] recordingData in + let duration = initialDuration + recordingData.duration + if let self, let controller = self.getController() { + controller.updateCameraState({ $0.updatedDuration(duration) }, transition: .easeInOut(duration: 0.1)) + if isFirstRecording { + controller.node.setupLiveUpload(filePath: recordingData.filePath) + } + if duration > 59.5 { + controller.onStop() + } } - if duration > 59.5 { - controller.onStop() + }, error: { [weak self] _ in + if let self, let controller = self.getController() { + controller.completion(nil, nil, nil) } - } - }, error: { [weak self] _ in - if let self, let controller = self.getController() { - controller.completion(nil, nil, nil) - } - })) + })) + } } if initialDuration > 0.0 { @@ -1104,6 +1106,7 @@ public class VideoMessageCameraScreen: ViewController { bottom: 44.0, right: layout.safeInsets.right ), + additionalInsets: layout.additionalInsets, inputHeight: layout.inputHeight ?? 0.0, metrics: layout.metrics, deviceMetrics: layout.deviceMetrics, @@ -1693,9 +1696,9 @@ public class VideoMessageCameraScreen: ViewController { private func requestAudioSession() { let audioSessionType: ManagedAudioSessionType if self.context.sharedContext.currentMediaInputSettings.with({ $0 }).pauseMusicOnRecording { - audioSessionType = .record(speaker: false, video: true, withOthers: false) + audioSessionType = .record(speaker: false, video: false, withOthers: false) } else { - audioSessionType = .record(speaker: false, video: true, withOthers: true) + audioSessionType = .record(speaker: false, video: false, withOthers: true) } self.audioSessionDisposable = self.context.sharedContext.mediaManager.audioSession.push(audioSessionType: audioSessionType, activate: { [weak self] _ in diff --git a/submodules/TelegramUI/Images.xcassets/Chat List/Filters/Chats.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Chat List/Filters/Chats.imageset/Contents.json new file mode 100644 index 00000000000..640ffa7fd91 --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Chat List/Filters/Chats.imageset/Contents.json @@ -0,0 +1,12 @@ +{ + "images" : [ + { + "filename" : "existing.pdf", + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Chat List/Filters/Chats.imageset/existing.pdf b/submodules/TelegramUI/Images.xcassets/Chat List/Filters/Chats.imageset/existing.pdf new file mode 100644 index 0000000000000000000000000000000000000000..455ed1a7c178f30d45247eeee47df7a53d9f31ec GIT binary patch literal 2451 zcmZvePmdEv5XJBFDf)s0k!W}I-)adVg(XB$5``?ep&T6V02{G(vNjRn)8|(+_Sist zShe3)cfG29_1X_!K7a98d>aOF!R`L|b8zm-6ZiD#Fm7+`>ku#T)lcL0?ctqEfY%na zKirS&yJ7Ww{O5KvUVs10UA?;ew%HGV58}i0apAOhysVc0sm}z(?7Ghynk2Wm8erkJd_Zr%~xWifTUO<0g@%QY%z2yR+;um ztSu-0sWhC0aA;1c*#Yd6%0e2HX#>nPNSyYiUb0x4oJ$MFs`k>XS<~eug>X%odjRPX zZpO3tRJB(%R!LKv3YDa4)xa&~N;^bSk-z}~mOs?5GAy;WbQs_(^w42g`?O9)^kxiU2<*|s@8R*HzBld zLX_htB(-cjm9k(lMa5QGs4uliZWi^0))r)AI;}O>r7ob!c-Ca|ZNBg`bg8uqldkaQG1z)#+XgxnCg7>lBeOI)eFQ{oGTZJ>{#L$ z|Ee`voQ6~yH6ZxX=^GGZKg-)XiM+WRzsFl+KgMj1NFk;9ii5LN+iHKX8efvBbxQm%P z$9_FRHd$}J8Mec(gT0@}$MN)%+M_wYu$WL-9@xCVJiM!Kx7*!e?_U4TL(FIO+rNK} zZuN5gc02+_ocH6_aC8NFb zC-CL^9?X>^bn$DzpygM>>c`s~s5-0Q+MV)T?LM$qzT&?Jxi|@lT%2SD?w5q;!mrkc z_50o1&$oWOAKwhP`Y@Moe_bsl6$gC$@RUp|w_PA!v(F*C^2;dc%+=>nktgR%NO)9E zQHT5W?fdb5u-p6LIqi73-`$Om+{4?OM{`qF*Sp=pS;13-SFdhAjj*47KCSN$Q#ldA M4G$iC_sfg_0B8*NNB{r; literal 0 HcmV?d00001 diff --git a/submodules/TelegramUI/Images.xcassets/Chat List/Filters/NewChats.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Chat List/Filters/NewChats.imageset/Contents.json new file mode 100644 index 00000000000..8cc615202a9 --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Chat List/Filters/NewChats.imageset/Contents.json @@ -0,0 +1,12 @@ +{ + "images" : [ + { + "filename" : "unread.pdf", + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Chat List/Filters/NewChats.imageset/unread.pdf b/submodules/TelegramUI/Images.xcassets/Chat List/Filters/NewChats.imageset/unread.pdf new file mode 100644 index 0000000000000000000000000000000000000000..7aa7e23a5bbd05fd389f28d9b0af5b98a8a77922 GIT binary patch literal 2015 zcma)7-;WzL5PtVx;g^$IqPiC zf_;c4pXcW{`!{cm20UGc>;St?Fr@+I4?A#x?>M82Y>6bfvCd7|vavUe1cMqn-E zjP)kEs7i?!icOEic}ap^%@7>Nm_x7&inF6kAs{Umf+nrK%^^~ZHaMXmRBUu4s-?g# zpo%Ct;jpBrp6j79(y;7Qy(r@-aVR;$rK%9=WCxlgPhq4)4p1*5;xn{-3O>Z@Ad-ql z#@UxB33luvVy9X~L~_W19S3HQ+vOPCAPjVr3Nip)Neb~~r%*}-m`~8AnkNQUq$1uE zdlbqdfg+GpG9yTD#1WL?9a)J)_!1|OasiS;h#Y|w1VlV?IcnirpE~BOsD|M{OAc0D zD2ndYge54dQI}%M@C1a56oFj^>xhRIqBkX#WDAN2EY*8Sp&;Je{08A}C9upnR{qX< z7>)BjFM)YYH0Mvc_=COT7DXw;PWK`|f8f zb$s1VfBrVu?tc9^oPqC$ht2x+_}RWYjT!ne3$Zn0mM)%F!(sfi?}vk}&&KNP?snWA zhaDo;H{$}{ulK-gIzx+>r9$n+>Au}OAXQr{SlLUQ+wl`@=_>yNkcB7Ql7**Q;GqSa z3%^|-*B{5nr>39x!+Ybcn$s-SUn?nvWC5Q)?7K;_y9VL@KZPjDHI(t<)UO#5R>(gg zUHCfg*PD;S-rx!DzPSS(&ko~ucy6C<9-f~_=~m--w6)-g;O_0_uL0RlPw{$xoXR0B M!Q9+@^~0OL0RpU{F#rGn literal 0 HcmV?d00001 diff --git a/submodules/TelegramUI/Images.xcassets/Chat/Attach Menu/Reply.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Chat/Attach Menu/Reply.imageset/Contents.json new file mode 100644 index 00000000000..358678e10dd --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Chat/Attach Menu/Reply.imageset/Contents.json @@ -0,0 +1,12 @@ +{ + "images" : [ + { + "filename" : "replies.pdf", + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Chat/Attach Menu/Reply.imageset/replies.pdf b/submodules/TelegramUI/Images.xcassets/Chat/Attach Menu/Reply.imageset/replies.pdf new file mode 100644 index 0000000000000000000000000000000000000000..6360561a0742baa4085638d7d73dc8365ca75050 GIT binary patch literal 1237 zcmZuxO>fgc5WVlOm`kJF_JQ_BY<4On_5G>bLXjD+_*^qG_G)2jz=3WV_EJ(vq z8f!f4r>LiWYm9^46b(FBgVKXUE?qF*fD#;0D;YpqVi_C?kbyf#n+OF{$h`=p10glS)ZoGc+89&+|6lRh!e)mpVQ&0gfHyr(dw-eguO1B?WX5 zlmkRA{|B<3u05^sloRBVQ^DLT)%%ETrnnCZx4U`X{>$o#=d=PhW)MnYvK}L{WR?F&i9r8u65Pv zd>l59W_3OMv)c~q@88+056l0yC-b)n&X14v^ULe@)_#i(sxDOu?Cti^H1PS$X}dSn z;OrgyEZN&f6UZ?RNMbwM7P7cdGc)p<*$}xv94Ojt4;aMx5+qdHk4sK6hy~l@HfU=o zy2L()WE)5cF>gd9^yS*;o!328|?iafZ2G71UC6+g4)He+Q}FC}EOK%s(%ug8q%6RYzTaj+W= zF~x4!)6eWo8FVKMYiCI^2 z&WTXm>|*vFt&J!saxFn~%u=~z%?DNWs47n9IW`zIo=;F*6X+go#wna#^5xc2=8l!qFuyp7km0kOOVy{t;-7qwEZj2iGpzS_+ zvDX+59K8r%A!qd>dI!EyFOJ$ZFREA7i{K8P7cT;vya??lFRFjli>L?Ot6o$^aoMUD zQ?L>BxbzViHJ(pU+$R?=c2px>yciMb(whrRb@#Yn5vil_|D-WEO(z1IZi^nfW8m5J z?C68ma`IwF)kEqyJ8@*T*8@)JIVDchz zY+lsQk9rZVZSDf2#&e0@n@``Ez4_JXyfFpJ#$_UjA#fW=@MY?Q%Ie$w{%}6o zkH2GZ!L$19-@gXCy4lyOM9D-S`Vkg=i_!bS)Gm6 z(|K2i{dw3Uqz*$3KRtzTuRn((<^p}% z#;6n0|DggQrd8+TX7?~04d%?%57&_6<>~M^ys|HMx34autk#Fa*=oY0z^f0tuLIbR U^XBGw9@B}5*t~f0-7oL|17*%LU;qFB literal 0 HcmV?d00001 diff --git a/submodules/TelegramUI/Images.xcassets/Chat/Empty Chat/GreetingShortcut.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Chat/Empty Chat/GreetingShortcut.imageset/Contents.json new file mode 100644 index 00000000000..747557322ca --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Chat/Empty Chat/GreetingShortcut.imageset/Contents.json @@ -0,0 +1,12 @@ +{ + "images" : [ + { + "filename" : "greetingdemo.pdf", + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Chat/Empty Chat/GreetingShortcut.imageset/greetingdemo.pdf b/submodules/TelegramUI/Images.xcassets/Chat/Empty Chat/GreetingShortcut.imageset/greetingdemo.pdf new file mode 100644 index 0000000000000000000000000000000000000000..b1a13e2cc83f96ab82d6467457b727df505de6d1 GIT binary patch literal 3042 zcmZXWU2jxJ42JK|ub3MosKi-&Jf9LmDxp+WTa||1pojSQu>0fBDTHUw!t>|TW_#&>PVu3A_0wki{_aCCjOV^< ze|NjNxS3WjH~(C%H|O8K2xo5|{+NKVxyYQPv?AeVQkxSOvXU#1&caReiD@T*eT6-t>1a7^Hg`{)oc(p)cy|vOpN^=yITm!}K$y)QH70je^ z;=Ud&WSyH?iwSr%=TaqSnA_Msw8W%_68qD#7!x4aCEe!5~ycvx$~uj zUaVU8vzoh7Vi9nDeHzV*RDnnfs+e2cUtS1JgGUw7VeojO? zQR8G=Pah`kS0Tw5qET$6qEI!KS$g4Lh(u3j)HFf{1$aQC29iwp>T?$elmfggP%wJ! zEdfm_LRDde^#ZF>oa3Ru?DQNtjm=t*sRg2l-O>R}YSL1d?6)_q1uC)=-`1gcoJ-EN z!>crNBvU&$*#J8hFGYsY)`n39UT(p{C;P!cOVZ+*;>YE$fJ)Nr~Pj)LE)?!y`+|zW`DS znZW|oI^3z*`DfHGO7Ek!KsQIvyGK*l&OQOQ&OR{pFr4^C%_|9UyK)*oth>i-tW*6+ zOTjE`5mao{b9BQvjUdpaAWfAe0~I?HN|SNmHf`d38aAp-1 z$8b90TU??N@(PrGJL(-l5`BSx5zuUKDR!hLATJ&K;Q4xrFbCm6y6DZOQi#{NZ`tL9 zj)EDHu6I6C;4a^f-|hHT4@yW^ta-#SxhL=!Gm<51xp`C)NS^XvLD>2@-UYM~^tNfhr6uy^UY54dM*d1}|E# z=n2`Sif|03LET)(J|BQXZ_yh|6~S|y3{Ige*AtOua0C-FwNAfPw=(r05o%8DkyV2u zXbTZ_XM6_}Dr&Bvmf$&_P#NW>Q70mSP1K&~k(S21BYNu42>>KG->VSGup%Bbm0QJX zL6g`iObbD$;7&Gle<)UEB*B*k$BLPGotPm$o!~MYWa`r6M31f`Szd92dn~&rkDp=a zyhpR(BRMnEd#TXb5v7HPvSzZATn4W}*;%&sJB{&o>^b^f#5I_%phY|;KxPhlm8B!w zqhm_}l4G36YNyID3#D>3flZ8f)ZBApJo6gyudk+qPmOoqOxx+#$zNl~r`G+u%%5O~ z_gZ9#FV8jR_URe7`gXhB-R;BM-+9*YTmAO$Uz@Oceer&CH2l1|yt;U?`xM@uo^qHH~Zi($MfTTXS?m)W{XRIHr^XwU)*8~`Dl9ZxOCI<=v@7H zbxEwjQ#cRze$I9uIjdhKu*dS?rt;ZMp5cCRc*^|j;_l*l_x|&vpKdqrCJA*qyafFJ z!04l|F&}>_VSI({VxnI^H$`UYm!=FCzt~m$A?3g2uE^+JOn0{zSJ#`{iC6gQhnK+d xc)z>ZJPD7lE}tAgS)K28cfk`L0I%L${kI|d@fyFly&LK9%8}{Oqwjus^*?%2Y+e8W literal 0 HcmV?d00001 diff --git a/submodules/TelegramUI/Images.xcassets/Chat/Empty Chat/QuickReplies.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Chat/Empty Chat/QuickReplies.imageset/Contents.json new file mode 100644 index 00000000000..5146f4aea05 --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Chat/Empty Chat/QuickReplies.imageset/Contents.json @@ -0,0 +1,12 @@ +{ + "images" : [ + { + "filename" : "quickrepliesdemo.pdf", + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Chat/Empty Chat/QuickReplies.imageset/quickrepliesdemo.pdf b/submodules/TelegramUI/Images.xcassets/Chat/Empty Chat/QuickReplies.imageset/quickrepliesdemo.pdf new file mode 100644 index 0000000000000000000000000000000000000000..7dd7ee40f28c6304eef7267f053c5b8076bc0beb GIT binary patch literal 3460 zcmZvfO>Y}V42JLfE9Md)DUi`{z6S^bG)_|#ZBf_hE$Bg&mBfWDwU$z({q=pgmS)yX zbue~clSA?)hvYuFeEsIR*fLFGoZ0{O`((_E7v|;5X}i1D*U2yO)eqaR0IyW) zaJ=8H@21u3?O!*W?bWxh%*ET~+vYI+If-@i&$#o$^ZwNTCmkwhsmUo89Eo zSvK>Ob573OO(IsBuQix~cKsA>Z4z7@-b8OhXrWZI83ui`rIwh28OKSpS-iMxhT(y2 zCSR;buCbbda*h=5V^XXp@im%&gCY4+jCVFTNugjGG!cV)Kx-~Mff0!hPPJPr9)E(j zIe1sRVj&ihq}sv88hpt{s!iF|lud}X#Ns1Zu|ya=Shh7pFU3f<-uvcf+YQ?;;hUi2 zOKRD~0FALGJcqMXf)9#S?`GSHSX;v0W{S~9L>qA@8;Up>Lb9bQGGmokPAA5cT2tF4 zxt2w>y-m1OW%e?)TAGQ1#o+bCY9%@46ik|P5ji-`CDpU%xw_np7n@UP1uvcTq529I z$yjvBE7o#qDQfN{=F)=N&gf*9;T$f-^I}r3m4GBid%wVz#WEyERfTG+D^$s%_(rvi zg4|3Wr=brYRBz6e@*OA4?TkgA$!h2s zVxZ*|MO{di9>vn5(xVu@mmbB4vGgdGY}6jb;BA3*&jvO!v6&WVkN0+gc4CHL=WqDa ziQY^L$IlrV@%6+FB6@~V?Tq!12C*GFWgsuQ8)@mzce&eXWVM5-J3pZA{KAz{PTl22 z)tv(y<=M}mI{V4p)_^*l1Co>L+GQ4Kzb&_VixM3xIAR8tmMK&&Q3n$pR06$%@Vs}v zB}0>{QUh0=;%YU}C_03lVry)PV>$>~p605D*b*FV&j2Nt=w?`rZN4%y(@Y)-Tq>G$ zu9-`9tNhdd9TyRlP8Mlaiesa2o@r875nGAv(fQFh$+~ zotI2&9&GD$q4~jGPe9L#wmdu!DMb8%WhcJqz#bs#mC;2sw5(Df2znU`c0}>9WvEg^OkvW+E!BZ5czbrIL>-AasnjVFZZoCjm^KR#i0lf6GZh6jr4L3MS~VjOmG#sZd~W#K2%zQQKS% z&PJQrj~vsgyUBg=>PEy(!tva(qUCTZo64S-sJ{iYAm^jTC@fLbi*Pp6Z(a%U$Tc=C z_KA*Sh?!9&WGVpjk!qW-D{;mYnXpUr|IXF!^Zu)8H~l>6zI*Ej1|`_JaxSXg@#Wz#w@z+(iR zkGk3(_Mh%I+k?^F{_1qz#eR3(?l7r~{TaMm--DTSf)?w)LjAK~_5IB?z8VeT%AD<7 z>_4F^Uj%RfS)7y~oum;Sy1^0o#rn9u-G6wj`uTqQexfL+hwlje%au+c8Q{~85}TjQ zu0sJo_17U!fBXUpu=7YNh$HI%K-`Wqbi7~R+-~nDzLQqpy@rmbhyC65nR$A1{p^91 l)zyA~G#cdlJ%F+{o>EygO zvZ#V=ITL~>#g1EsaKflU2*#wv&Q1hqi=HQ(9j92JrKKR24O^5F4}6@BQn9jTJGvSTN-7IH{Q!#9g*fwn$01gA`q>en>1%~hzTdx5E|O;K}L=xfjfkR z2&^+D%ZNN#2SX7(wIBivs#vcBN(;)HtkF$CKBoks1>7x`poo^E(KJ2)i;mh-Jdcxt zGrAqJaZqzq#bl!=SB-2IT{qF1I(cBxd+S}jgP@WWE!6_PlP(6_LKwErZPx=ZZ?eZ# zc7Zi+P)^^i#^VmXegB=*S-;F?~xo7Kzx{%}6YcfVuX!L$19pFc;ry4~E54R|x&Z8xtDpXIwn5Uc|C zFK=_C^>{jbIu7GW(mu6r;jRz+^SDPwTBc_3c5?(K@y}DCZsj6~SFzPk+dGsh=>%&z z=ea(7g0K0e{u7Wc68@&UNOZsxCtMDGy*Y1ohx<#{&&Tn-&@zch`}8Se2rfv#?V;6x z9=T@-*X|``E$->ZP=4%aJrgIT^$FU(tadq|mFR#A>q>?$5 literal 0 HcmV?d00001 diff --git a/submodules/TelegramUI/Images.xcassets/Item List/DownArrow.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Item List/DownArrow.imageset/Contents.json new file mode 100644 index 00000000000..ea04ba34971 --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Item List/DownArrow.imageset/Contents.json @@ -0,0 +1,12 @@ +{ + "images" : [ + { + "filename" : "arrow (1).pdf", + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Media Editor/Apply.imageset/check.pdf b/submodules/TelegramUI/Images.xcassets/Item List/DownArrow.imageset/arrow (1).pdf similarity index 53% rename from submodules/TelegramUI/Images.xcassets/Media Editor/Apply.imageset/check.pdf rename to submodules/TelegramUI/Images.xcassets/Item List/DownArrow.imageset/arrow (1).pdf index c4dfcbde1b34db23744bcfbe0b94dc13d8f8ee1c..2437bf015c4315723dc7584f2ec6f357e1207342 100644 GIT binary patch literal 2584 zcmds2QE!_t5PtWs@JprkA%G1ABUO>wq>ZMnYM1PxQnd$4+%#PR8IY;nuiqWm21>ei zuk%Ae2lEiBNiaO5JZSJ4M~w>-#5_R)E2Bzo=|gJyH%Y z&NVbE^8K!3P(L?kwM}EV0KOv0%3V^ZEPd&nkH?y7vA?fsr`MyY1i7xD$ z=jZt|y`!Gv)vMhwZXjtk-+>7X%c@Yxmbg>(Yn`h3?3_*(-KVr9KZzv-K{U19*fZae zUJy!2Lq|03(`{>MU$dJs&EPolBIZRja$=7MET(AUvrtH%N{7ch4gxwl2)(vU2}T!3 zq0fL$FeYO`TPSp7BqQHLHfZGa;vTO0|=z;Bxgpb$6ny4j62Z*8=&ui4T=sSHO_+R_Nun9n2-fR%|=p%hWv)FdE zB?o%grUP1b{0y|U-E=U^_}7YK$(o9D-d0>#6f3T0K;7hT8O1R%Z+r2@b2VF3Nu}!Z z3s(Ovu9EUcy=%q@KK7b^I~)2tcSiAZ)6=DYHh6MjtekCjkIDPgW^@cV5YV&EX7U!@ zc-*87Wae&!J*D(~jEq7b8agg$wknAW(oXZk_Yw)4Dg~i+EiH&eI?J-WD(M}LyEdhp zWTiB4rIzdDALk|#wG$Y>3N!X6c~+?m9HIaQyus;SQRuie)R76HCIO7~^Lh!9QcPe@ zO~8|U54LO&ehEm2iFrLLOt7>b+r%eHm2C1=-}bps55%L4k>=juFAat81qEC$bbyP( z)UFrvd`J>k2NDo`9}+&?rB{#`7)mjCD;~ZTM^Nz|7iR%z*oQcWtD>Q*NY)!w5R~`R x6vRC)^Q}6e$Lr-uBe6TrbI=#%tLx=1*1r{M0pgZbQdG4%oJ*e^9bH^s{sDUmFsJ|k literal 2635 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 a_y61DeZOzsJw5kuLP~Xc@#2ebzWhJbO8*N0 literal 0 HcmV?d00001 diff --git a/submodules/TelegramUI/Images.xcassets/Media Editor/CutoutUndo.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Media Editor/CutoutUndo.imageset/Contents.json new file mode 100644 index 00000000000..4a8e6450d0f --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Media Editor/CutoutUndo.imageset/Contents.json @@ -0,0 +1,12 @@ +{ + "images" : [ + { + "filename" : "undo2_30.pdf", + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Media Editor/CutoutUndo.imageset/undo2_30.pdf b/submodules/TelegramUI/Images.xcassets/Media Editor/CutoutUndo.imageset/undo2_30.pdf new file mode 100644 index 0000000000000000000000000000000000000000..e814c8183de81676450a48f9d3c51219fb8285ce GIT binary patch literal 2314 zcmZuzO>f&U487}D=u)6PR5tZ#14V(gU58;C)}^}@J9v)UW=QNUc86`hev~a!N@Jg# z7m<7%a7D@M9iwB%FMZuc$5I#v5`cRX zqrv8y6P62bK0YE)mNs|DBOsW$9_?zhBTN>u0E4H3r(BdUo+7e@)C(N8>^Y*K7N|uq zMpYBJA|WT2*hw+c(~%y)5x)o}1Y=asPzenRa^QskqZLT3of1wtsVsM*P&qiqu^Co6 zt5d`jwkXSJS2OJiY>b^$4|t*UeekziK}h2@2u*9#mxGwL-^0d@=(6YMdK z;9%>~5C_o^D8b}J$MM7(MKGo7TOM>71Nr#G6zqI<7gI!V%+c3oj_u~y(bUZ(eRV<& z!3k@m)P_@`Qz-NUAu3SxM^p~fMLOKELYVoIAuWFwwh^Pdu#L)G!pNVGLLQ@VcF{b7 ziO;CJB=|B&_u`=BB#!YSPeOau@L2}b$nBzzsC4Ag2Z|~pbKl$9XHsRlAF&^!-sYo; z3KyJ?UDEy*`N89aoW=~+k3h)%Z3CycFHmVpD|syHLh1-r zHmMARBvl|Hp7SGX*%#*5R@h1N0`t3ST5MpQqP>O#H150XP1|-u&mX?w{=+Bx`uAtU zv%Bi43Badjy{WFcKl}j)Zx|c&BJ~W#x7pLG>ARPG-SpgV*i~?M-L*s0Vnn}QXYj7t zBgS#_j8L+KJw(en`>3k7!$v)hHBeA9h&~x zH;)X(%%=25Wm{vE;RuHZ$NKojTY_*c975nM_QXGQ3S5E24k5e|uOJk1G#vKTX4~u; zeywEhZ?GJveYb1Q_~~YSHWejXb=|;y!efE6`^~=w*^gf>)qWV$k$EZE$;sQ#xBmff C8rxL> literal 0 HcmV?d00001 diff --git a/submodules/TelegramUI/Images.xcassets/Premium/Business/Away.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Premium/Business/Away.imageset/Contents.json new file mode 100644 index 00000000000..10d5d09d15e --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Premium/Business/Away.imageset/Contents.json @@ -0,0 +1,12 @@ +{ + "images" : [ + { + "filename" : "sleep_30.pdf", + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Premium/Business/Away.imageset/sleep_30.pdf b/submodules/TelegramUI/Images.xcassets/Premium/Business/Away.imageset/sleep_30.pdf new file mode 100644 index 0000000000000000000000000000000000000000..ea3295cd4c0941c8362a089181e321f52720409f GIT binary patch literal 6376 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 literal 0 HcmV?d00001 diff --git a/submodules/TelegramUI/Images.xcassets/Premium/Business/Contents.json b/submodules/TelegramUI/Images.xcassets/Premium/Business/Contents.json new file mode 100644 index 00000000000..6e965652df6 --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Premium/Business/Contents.json @@ -0,0 +1,9 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "provides-namespace" : true + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Premium/Business/Greetings.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Premium/Business/Greetings.imageset/Contents.json new file mode 100644 index 00000000000..3f0cc65f028 --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Premium/Business/Greetings.imageset/Contents.json @@ -0,0 +1,12 @@ +{ + "images" : [ + { + "filename" : "greeting_30.pdf", + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Premium/Business/Greetings.imageset/greeting_30.pdf b/submodules/TelegramUI/Images.xcassets/Premium/Business/Greetings.imageset/greeting_30.pdf new file mode 100644 index 0000000000000000000000000000000000000000..d2717c45d4d3fd9c798d2835eff6213b832d38c3 GIT binary patch literal 3542 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 literal 0 HcmV?d00001 diff --git a/submodules/TelegramUI/Images.xcassets/Premium/Business/Location.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Premium/Business/Location.imageset/Contents.json new file mode 100644 index 00000000000..904d28263a3 --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Premium/Business/Location.imageset/Contents.json @@ -0,0 +1,12 @@ +{ + "images" : [ + { + "filename" : "location_30.pdf", + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Premium/Business/Location.imageset/location_30.pdf b/submodules/TelegramUI/Images.xcassets/Premium/Business/Location.imageset/location_30.pdf new file mode 100644 index 0000000000000000000000000000000000000000..25df154e349bd0574d82be8a5c97704ceec20f1f GIT binary patch literal 2987 zcma)8O>f&q5WVlO*h_%qkk#z>E)WE0?4~H%qOQ_g(1S`djtWU^C8bFF>+^;phbzZG z%Y#8bb9ZLmdo!MAceig}T9cKsn)LAdA4=<2uk`EJsz2QGugbOf>c{@@aQ>)0z%_q$ zIv@MZUafBXzqeh#{_c&wdDs5doz!2-8X6x>^UF(llfNYslZ!;!nP6*mPIr4HpOFc< z1aGx`f9ej(CgZ5$(KZ@iUCc%AThy5t97V5fFvU8Ts;;YmEz3e9_+2MA4GAWPGkEXFxXDQVB<^siEY^1>}TKgitXOZsU@Y zwW$Dc zRexd3;WO{R1?5zkF>hRSHG3V5w>GkD!Prazsz#Gt@S*V0h-!ihkt_DDB1@a07|enb zP=c>Ro(m=_!2=X4JTRLEeT|l3 zX=i6;L)PbVWxx!AbB6!KrrE$sM3^+QmA^K8(*&X{d*BH&c$UlzRg+3|)}lDeou9Ph zK1L65?@gVyKur`ESZ(geo${Ddusyn4z=dyzfn_UCg9TfZ>6&C3xf=5>(J{{;kOyVH zNwaakRMTK`Df;Az?%6fyk}29YB**}}OGOJE0=0AqZOr-ybx^^MHza^pnh=_A-=umVldxadGn zv+0V4G3^);BOv>Nxh06?&@DL##)L=&i3%~1Tk%wuOvIOxzaVm3*R-;LZGTI480s)^ zZYq#0q|T;x$?U3aBt+9SJZ>#tEEQacfCdslkS0RGAQ+W`>E_%7Tbcud760-v5 z!zY}zU*&%S(%h7;ZsGz@5^yQ}W^>-`hKK8;pO5_qMLMfX`TSF-7_mqJo*ry({rT+|z8zsX literal 0 HcmV?d00001 diff --git a/submodules/TelegramUI/Images.xcassets/Premium/Business/Replies.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Premium/Business/Replies.imageset/Contents.json new file mode 100644 index 00000000000..5c235cd01de --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Premium/Business/Replies.imageset/Contents.json @@ -0,0 +1,12 @@ +{ + "images" : [ + { + "filename" : "arrowshape_30.pdf", + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Premium/Business/Replies.imageset/arrowshape_30.pdf b/submodules/TelegramUI/Images.xcassets/Premium/Business/Replies.imageset/arrowshape_30.pdf new file mode 100644 index 0000000000000000000000000000000000000000..75fe22b4f73055c9329ad41afcbc04903dc69f82 GIT binary patch literal 3326 zcmZu!O>Y}V488BKm`i}5FSpEqmVNoaa2+^^>N85r)ZLO zoMJ8p(G-^xi9%9C9LZ?V)bEKaaNFWl7qTH6gXa)In=%X23{0}P5|p&7b2VtTo+63I zQD36>MOkOW;NUw_B=}d7-oHLs4T!}xm!lCYM3hAhi1ku927rN+^VPuh$+>}ILu7Jc zW}u)LvPK0Dnt^t58Upg25N?)A!8ry5mTgGhl{^VzdbZkV8J$(%h$hRS1SERXAbk}%N$6^c$-n>Nvi0~TLYbClQ>UG1-X36vbU&vsF-2s5=x4+u|0dVX@HVl zv}uE;ZQ49jW*~o&xVC8t0TKXW&|Pqt7~ZAmOGc7yK$$@#QCfx2j(EO^FnfcV4~`{5 zM1qAQs2Z(_7=^)dvg%TTnjqv$??!Cu9(fy2ao;h&cOkUIOM_!$N&Px$e3d`vt(CJ2xR2+0tZ3r^qw#xz$qu{xFRK1OGyaoTki!2 z&D59@%%j$*$gboU{qNI#8;FJ=L45Ov9Dup8R`dN3+(7I6&1?#mGFcC3WiqcOk1+ih& zWcaE?aDj^_e^Np1SNG8cPTxud$wcaPrbdthBzoi&$YrZkx~`@TI`mp#suZ9~18Jxr zuL!<0b-)Ft1T7q)*7NI8b{jU4V)ZXGIv-cFO9^8PL1UWc(~@vG&d>+f?#;EbiEtff z5jBgp6c7Sx(Wqh@6TcFEH97Fqum*tD=?EPd#w|!v!ai6>|1c7> z((jPprthOamGtwH*1S~jKXu#gw@z2pVGlhot-7C{mRW`@v|SdBHSV~J@3!0B@nAmw zft?qh#qa<9-J8YT>Y*QiU;6vU)w|tS^AUwSY_RQ%f9BiS(y~A7zUC`04Q;t{RPCY0mH5?7rYGUHN|iGO>hiomkQU z5ADG*@SD|fwb?z)uYTS4pE_r9cM4yBniQjt2JrBKX4Y?J+aPGb9O8^k?-GLgUqGUY zrJrWYJFI1A=(t}!Zu)(PzfcxG-hz&ohuu^E%DjBMe{~{dvE1#BMgy({FFri}-^2U* Um&$5?tl>bbLiggu_rJdX1_e*1DF6Tf literal 0 HcmV?d00001 diff --git a/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Away.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Away.imageset/Contents.json new file mode 100644 index 00000000000..b9b3b2f4363 --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Away.imageset/Contents.json @@ -0,0 +1,12 @@ +{ + "images" : [ + { + "filename" : "bubble_30.pdf", + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Away.imageset/bubble_30.pdf b/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Away.imageset/bubble_30.pdf new file mode 100644 index 0000000000000000000000000000000000000000..23cc2b43c6d017e12675f03570b25d0420c7177f GIT binary patch literal 4561 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 D=Bw%1 literal 0 HcmV?d00001 diff --git a/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Contents.json b/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Contents.json new file mode 100644 index 00000000000..6e965652df6 --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Contents.json @@ -0,0 +1,9 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "provides-namespace" : true + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Greetings.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Greetings.imageset/Contents.json new file mode 100644 index 00000000000..93a1282c319 --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Greetings.imageset/Contents.json @@ -0,0 +1,12 @@ +{ + "images" : [ + { + "filename" : "hand_30.pdf", + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Greetings.imageset/hand_30.pdf b/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Greetings.imageset/hand_30.pdf new file mode 100644 index 0000000000000000000000000000000000000000..f0f8b105471119731868e3abdb269815ef6bd1f5 GIT binary patch literal 4059 zcmd^?Nsl8%6Nc~gEAlcD97yNBNN6Nx72-n--5ih*2al(iX>i+v?fEqD>v@+#b;LMVX(@m?dzWhr9S^$Ex!Bh{iqh-fAjk1adkk(v-sb* zd3$)L93NgyqL;hvhxcb2+VA7Lw&%F<8hyLvX8(S<8#k-p?!15ecv5x!;q{>zeX-r$ zPO__Uzx}XVje9ZCzkKp!`%(R%j5h6=7ru+;)sLuR=l-x8m$$>>+4%pP)p+&tiF$f{ z{%f@#ejV(E#?QJ&NSe$A?H$XkT-G%jpS)V#&U^cJeX(C{25Yp>-c+NU(>0`Et-2kG zHqlsPm9yI0oQzXR`(jGA%2w?{cF`-ZWAHImWs^=RS7+5~a5`bzWo4r-F_z3qqt?fu zY|yp(U<%ZPShGtaGP(F7TX8DeQX*2XZLJv(j7|-TvQ`HdLIi-$#aIwqr#=39{2YRI zC78rpXs{{NsMbTmN_H#(XcCHQ6V~b)KwhU%LQ+{Ln~g7mIpcN_5lT|QNYpM`?@BWg z&;mnZIa!OjH3TPDO^T|*zY`MH1E^haCaD4#>m?@IhU&9{$6Qi1!k&%A2al;@T=K#I zl$jdDDjGtnUH}&ztplL25<@mIH(|}V)HnrEMp(fPGlatvD_pb!$rfuHRV6x~GDeb) z*}xWW@i|wq;Q>q%R7HeBjCi{ylDR}{3`mPHtaur1sV?FTivVE}9zyVrZ&FT?7_Y!f z8U`CCT1fy%GP+u_2I|PDQW9#BvJOI!NGq9C@yBULautHX+iFv7t5S0&Qs~Kqz&Y%& z#v@yVK@Yb5=Cz)8!%Fmt=ZU=v6poSR!DY4D!S;5 z6i&plN>Q~aN;szyz z8<9qky%0wrlTB=v6*)uJv%ApJVgjHe7@@exmk<(-kl@%28VTA1eRegx8-5x#!?(jQ zZr)6jfR#+UL8jSp?Ahp}e9**n#D6vo!+&xb_Sx3wV_z@}`}%Srv#?7EDs7R6QP_pd z$4rNyh0dZNorQLZhz3w;e=cd8b8Rz;iACvBeGFswTGx{zfEIn9VD5Gf#YL|8BUD6Ze9LEWGhdU?R=~d6kGuOM)OEZ^HPKad` z_KmY|>%VGW(teKZi!|=tD|V@z<)>48JEninY>|#NpQfLGJiNRV(`WK{k!Ggt3Dn6FLa1tx~+6i{!HED z$49qOcIxx#4pY+&ht_js=pWr07I5}vE3xDp3I3wK?$qnV6C|rIe_osVbaZ|Yv zappdKb=WO$*5hsvdcS-Ixu5K}x8tYklbbi69w}a2ZMX0%^tE^`u5W%Hg$u~tAC|jA QW5;vI9UeV;{>_U&0Ksle1ONa4 literal 0 HcmV?d00001 diff --git a/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Hours.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Hours.imageset/Contents.json new file mode 100644 index 00000000000..4782f126a7c --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Hours.imageset/Contents.json @@ -0,0 +1,12 @@ +{ + "images" : [ + { + "filename" : "clock_30.pdf", + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Hours.imageset/clock_30.pdf b/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Hours.imageset/clock_30.pdf new file mode 100644 index 0000000000000000000000000000000000000000..0f46a28840b5e39d71dfbdfbaa0f4f03ee1b5ca8 GIT binary patch literal 3302 zcmd^BOK;pZ5We$Q@M0h-5K4TLKwuz#qG*G-yEz01dXQzs-gGUwl3Y0X^&Lvokjgr3 zdM+FV4=<)!D@>&MYCEmbAP6OelT*nilWBJT$cU@b3C^Q;&ebDt>L++wqPH zIJk+>W#2tMWD3*wyf^90id*38t9E#(`lhYFtoDzOBh~itdYmr3==%GJZJMEb?CWL- zSK!H$lkSOrq$Mk-l>~`Ul70s&&>Y6TsqRT}*8H`ro6Y43J>BMC^+5h4{DAQlGUA?D z4xv!rCrvzt!ce!wFFHsEDw26yY9o{V+%4x6D}`6m(gYz#O%j+$G6`8wB`MV88EUjD zksyQ|H9-t1P{1ubjz~@Jz<(!0t|(Vb2SL#L73j5--c!$15Dev-$)F(s3g?3=LN#$r zD{e}afsS!IpHnQdFq3VEh*%I zGe3iHCzrg`K*j|V(h6Bp@YOkQ90kZ4VYOW#FkG_I8Rs=XkpuW7T*6B!s9rc0$N&^M zW>sHG3KckW)G3&rNhO^UV97I4-$OD!{X|;w8zD`56JG^L;pLO?vgZ~6VF>#Z3I+OO zjY0jt8v|B63eXr(J`_S@KonjmR30pN7zhfe8&OJQQ;QIa8CPlzsS5SrwW7j6Nn0`G0YZ%n;DMjzzp#@X+FzjOm4_air7EZMY0}GP5mh*3&A}>+Yv+45y1k< z?N|&)bE5#owrpWdX09XY;WdY*Eak<)aa2scP3`M@-7B=8xqCqwWB1aChOs6NG}FZL zfzvSaY|>zH{xn{0$7*b*{c}G3Y~NPHAColCMB#Ha!K6WFEftqEw9Ysxt@jST3Yd}J z>>uu`f2UsNpaYHo#K8?c`5xWz@$5E`Eq&gsA{B={Xgza;erSLyAZE7%DIm0G-Shu3 zfpE@}GI+H^FeK?s+jiqXKZemQxpY~6qR8a&w0#p2WM5p?;1{R1Ew6E9a(!NUP>OrD=sW3}&Y z*Sb%A^NGMz#L3i-=bmh|mYM<{SGXW73jZ|Ig5d%Z5LO0h@Mr}Iq2XnG16e5MHKgsq zDRX$rczBp=MHl|j){s|;YwWAtzUfH_@6Trt+!sT4-@K$Rb~i8Q5Eq-SgZPHf*b5WVwP@M0h-5K1IPQ33)3@h6Hlh`ZZEfFK81R_sl8C0CLQC%?Wk5;eo! zO_1hX9(Jim@y*N|4mBq?7gw*e@m-#_ubQDv|)Ol*QP!5#x?NuRXaRZebd%o_SR2N1J(NJcASh}b^R`2t7hn) z`nnm=1ig87);-G)GF9m`cw!0S*+-y)ox|8S)lMWA%|DyESzVvW^L6}G58`j3j~HJO zBdvMmX0vX2yWa6mdW|MS-3r$GDsVkj-kjdf#jc4#u_9GOGc7SLTd4_2w-KoM1cZmmlw`6 zrqg-VN|+Uy)fvRaSma5BW;2Zgn!pG0gcJp5fRCVBxS6_)z>Mq?8$*#<2FX8SqsYi8 zA*jHoUqmbZ5Ta@C!dZkF)|kbE3NklBN{k>ivO z%1}sJ8D%mg7xTi3ky9)zLF_$2kQPc7fm^r$LsAJN*9oBm68o_@PVVwyTbAfdMxhhw zv6*967LPRqP8&J4EnZ*WyS?!GnQt#R#&ml*Lh<$rZN$_kyw?&tjTL{+FQ3Qj^;nI~ zw0^0lzxCT{_-mTYod`be^$px!?9Y;GiC1T6Q1;i3o)YLx-fSNqsvlD?-Nz9J0-;@^ zNWMpRdOC*v-?3LAnir>?QaYc&RNQGGF=deNqW<^ z-8je(*f4QOH`RR;fVa)vrux~|3z@i!!(*|uj^|xFHZ5301s3oIJH%0z0g7}&XpR7# z{cUpxu1btxB^Qs+yC>MIkNh71iI{Y{M}-j%WRW93uf}TI-S6*x?wd~{g=S04y(j+4 z+&Swc;9;Nv91^}X^C;j8GtC$(4U_>6f{fO1at|ONK7ca#G8{l?{t$8?D-Ndj=FkaK zq|7z;)n?oD0_FYn0>phWbi3xIe6hKEIZK?Zx(@V3`KEj&>&@2&wE%I4vFgW39cTxw KI61j|cl9q5;G9MP literal 0 HcmV?d00001 diff --git a/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Replies.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Replies.imageset/Contents.json new file mode 100644 index 00000000000..ac64efccdb7 --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Replies.imageset/Contents.json @@ -0,0 +1,12 @@ +{ + "images" : [ + { + "filename" : "stories_30.pdf", + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Replies.imageset/stories_30.pdf b/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Replies.imageset/stories_30.pdf new file mode 100644 index 0000000000000000000000000000000000000000..ba64f0a7547eabdd98a62eb72955e1bc7ff95aee GIT binary patch literal 3118 zcmds3OK;pZ5We$Q@M0h-5Q^^y5EzJ`DB2*dw}${h4)R*DH(g7vBo|J8eP{F-YQ0XH zT&si4{YcJy^W+>{oSeNmD%pir=gHroZJH z4qi^sdD}ePFD?wf$GzdscyR^t+OqEM%eJc5Uv~Bn4<}KphpT>2de*de6Lwj3%|pAc zI#fX?kB*v0{((y&hc7P-L3#BNM8VCjZ>#c-6(`j{n{{=0e#DPg%U|n`{msA;lgL*j zDX(Z#TakiM*6}E0NHH0{zGJk#zv$LAQ&M=Nw3A#Zp}bSlbGVdID{NFTSiW7NP8bKe z4Bqm=JMFo0A}XmoNYcV6r;oOJHet*us92St z(s=Gc0ctOK5S$@MB|UbpjDcEWpHZk?qMy;se}Rz!OFjz=hZvDUN4zLyu7J)MHEP_PKx zGU84cQ742FWXWSDPIPm!&0Ca{3)4yHzM6e1Gh=)Y=wh~Qi|y-M-3!~#*u5Z)se3s> zaXgp?gW=_W56=&Fto%TqKKAFUzU-@E|5OeCR&UGhuVFRL7Wi{#637F??y5NF*gB_? zV|Q=wR{^ueo9+E=`QM`Fd16ln0@DCQmV8Ux;o)dD2%GynygrLQKMhULdPWcZ&>&KP z&u%(afNM{h=ie9*9P0+$m2#ppX+21zTK$W8fmwaY?+&sWtwa32$WFbk(9Z_gu zM;f!`$7Nq`o15Lck8Smd;ZVlL(GUM}@0@WQ@aaPZ_;Gk?#t$Z}JqYHLVUte5+W=w0 zZvnx`@&qz4l|6y%{wbIkpPusZDG421mesfAW?Quk-TUhaIQK=@+*L36i_P`R(Z$7O i(|~``eM7v8)#hu3z5wTTecASdcXU*N9UPp#JNp+@jg|fY literal 0 HcmV?d00001 diff --git a/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Status.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Status.imageset/Contents.json new file mode 100644 index 00000000000..4cce7e34664 --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Status.imageset/Contents.json @@ -0,0 +1,12 @@ +{ + "images" : [ + { + "filename" : "work_30.pdf", + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Status.imageset/work_30.pdf b/submodules/TelegramUI/Images.xcassets/Premium/BusinessPerk/Status.imageset/work_30.pdf new file mode 100644 index 0000000000000000000000000000000000000000..0710349e369791108ddf8d6823862af996ac056c GIT binary patch literal 3904 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 literal 0 HcmV?d00001 diff --git a/submodules/TelegramUI/Images.xcassets/Settings/Menu/Business.imageset/Contents.json b/submodules/TelegramUI/Images.xcassets/Settings/Menu/Business.imageset/Contents.json new file mode 100644 index 00000000000..d1c4d41aef4 --- /dev/null +++ b/submodules/TelegramUI/Images.xcassets/Settings/Menu/Business.imageset/Contents.json @@ -0,0 +1,12 @@ +{ + "images" : [ + { + "filename" : "business_30.pdf", + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/submodules/TelegramUI/Images.xcassets/Settings/Menu/Business.imageset/business_30.pdf b/submodules/TelegramUI/Images.xcassets/Settings/Menu/Business.imageset/business_30.pdf new file mode 100644 index 0000000000000000000000000000000000000000..374eb7b7a72cdfaaa428c9e09a723725c3800e63 GIT binary patch literal 4653 zcmd^D%Wf1$6y5h%)CLJ6rKa8w30VRgtSCVcI}0R~#n5hy$#}+@8Ap(>&$-<-x2hQ< zl686kJ*TVg*e#V^D9 zdjC-vEW9Yt`F3-E*K8PnPv6Ep)5c54+b-9;yXAIR_g`k;Pfm)c%lr5HQR&Og_O@Ub z!)|lG?S~zy;FZsxZ$686LP<3~^~Ew&-+qTE?A-0Q!}7LUoDF~7^uxvZb8&jv{_1z# zpPfFE_#;c?Z%%5hl2gLVtgT8x^tW{V1lZ?SD=S2&dpki{O$JWETD5DSF(7I7t6go=lVuXQ3YYX)qKAWU<49&t7p)q-9 z+qJMkXJ?F}l|@T~CCTQ@&|-HZk$rA-GuA&u&Z_$eAEQ+^X`zF3)`d)*8JUywwgrte zQDtjs9pd3V?1j0`5wl+UY`n7IOiAaACU2ZH#(NvY3ZT~D4}@EpKmbBSAfhq)@V1m>>DuT3h+GL-Ek=nRX149%)Yh8*kkU=M{_Ot=g z;BM9t@jF0PhU|TaaHE2}&P$V{vaQx2y^02ARt0^BDC+2tuoX-YjKoCPSz3h}trg^T zS%#Q{YkklPB<2)NqX8KQ3Q(xI)eCA>$wC&`V2$Pmtt%Y}M+|2Y!9p~S@`X8tZsblv zejvb%A`4xxNcYC+25WUcv@#)&K;ldyxIo?^QL2`{GfKk9FtgIY3RD>-948I1XenC= zasAtegP-3Vl%oMA+Rpl~e;mzzNpL0VLj z*D%|`>TFT}9+dcu7e}o(TL;9&M9mB@TH6#WG6vrwN1=(Afr9`cMn_%2u`MZ(04OC? z8sbP@a)e+<)ylW9tdlAGZGD*T$ve&N~T_c0K%=5sr#0*kg#%wv{)0m zx^RW0aak!Bo81PD1BI&4Rf`TriYau&F$SBx&uHeW2{V37jwvV&*2IMOP9cHCWgTg4 zBEsvKzp*V5qm|`))-v%uj;9YlbnEWdt{c`@WwyW@%}*W8N7EpMs>-xrge}A0nXlac zd%gn8UyXH`uZ}@RXujef$kv$YK@Vck7#WR1E2xvAaVR{@3Wya3EgaNN7_>w*TnrMZ zHadb8h5>^UAp7wRO$a^F2$LixhKv~mBLk+R5QNE8`ZTTeoyVb_aX^SVf}&b*QbKTr za88)X%ZiOfUaYSuGMCM-Xtlq=*n!rb#sfCIy3l-OA6gC9MzJz!al08R`J!NFf@ z=M{Z`7^biKP-q^Hwdw+ggCkUemYYbH;8I*FV{B98Cei*;M(BprXua>(%#<%HeNk3;7!Fpeuzb(!Mwa#+moPRs| zwZK*L+3N1&^82C>*U2Lp1eE(Pd-5%DCnuBJ5O(PEVa8OsGo$rP4E?Kn=mMNw@45x3 zJ=;8d$dJG(N-nT!TvVDX{Im6Xv)_q#bOWv~y;)ul1$Z@F-7Np>U794-YdNLG;co$_ zoArKJLnAS80N>D+e=l-@8lDK6JfMvF`Q{3y3i9AW94tQF++(jkvVRApk%R%S6-6HI zSY!44bh%%yHrMmLpSQz@4&#^Zrc;Oh6&U6*IRh^bDWFH-q!}MWljitVA$PRO1`Jj6 zRhX$!U-a-so6Xm&oH|bAKv{cnZvtp|MBiu@}KYJKeos}{=av>iq;+8{k8q~?|+jo{pMeP z|GR(w?XUmvyD$FxuYdnPfgVSJN)CTAHMnfPw#$J4)1>afgk!e`R;$d`yM~O(hI!Tx47{C z%RkMjcrVZSkLlk(AN1q;SDW(mSNvQ4D)+ZR9*nn2My~uPFGZ`DBTk38CN z@-7@~NT*oi0pG))7oTybc(4gS<3{Q=68SjL5BM6Pdbplt5Eod zAOHHUk9cpeMsAFu*kOnq6|eGNeRxL4!teg+PwrG)J(yD!N&U$jtT~zF2zq*JK*;Owc|f*X5(O1|QA z##%lpoxF{QqjUJy|9bbUKY#VzkKg<}OW!ZvSf1x|mKRH|%Gd2F1$oVQDi$wsbminz zIMh=p^J_bQ3h8hxCujH}`4XNe-*S9)xu>%nR

{_g+f^Zxc7x8?uyH@*?a_J^N-`2OF%IWxMS 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 J#004Z0|3kCuF?Pi literal 0 HcmV?d00001 diff --git a/submodules/TelegramUI/Resources/Animations/HandWaveEmoji.tgs b/submodules/TelegramUI/Resources/Animations/HandWaveEmoji.tgs new file mode 100644 index 0000000000000000000000000000000000000000..dd1ab78d28fda166f300238af72de46bf5dc1636 GIT binary patch literal 11908 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 B53T?J literal 0 HcmV?d00001 diff --git a/submodules/TelegramUI/Resources/Animations/MapEmoji.tgs b/submodules/TelegramUI/Resources/Animations/MapEmoji.tgs new file mode 100644 index 0000000000000000000000000000000000000000..32ba54f16cd28f4f5143ac316b1eb66d1afbbcef GIT binary patch literal 64974 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 }a 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 literal 0 HcmV?d00001 diff --git a/submodules/TelegramUI/Resources/Animations/WriteEmoji.tgs b/submodules/TelegramUI/Resources/Animations/WriteEmoji.tgs new file mode 100644 index 0000000000000000000000000000000000000000..47caac05a2fc28b951f68bb336c11a332dea94e2 GIT binary patch literal 64148 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

Ct zGBrr)E(chPrDfAkXCUniP_p-D#Iur_QY*mFD!04;18N^QMnPVS+^|O?RlUGP&=S2q z0uwI5mF5UCQ8ujB={?wcy@E#eaW}Rpl4nZjR9)riO2_bC$oxE z%w&ROS5H+T|9!5AoU5j4TUdmfdALVd7K)^+o2%(^mviLE5&6##zyI@ZAO7H2<{lg!|zkV72@@@Rb|N8KU?8DQCf3bi6?XTmH{x|*a zKmOxC#$W#V&wu8h{P5dvKmI;W{QaMvKK%T{zyJ6v{`)_dU;p&C@$Yeh-~au?AEKS- zzx_Xd{Ns23^uPa?KmOP6{>vZ#IL`Ikue`8-9e?_tAAZ3-gtCo<972bL9Jw z<8yu*|9oUvwS&8w_du$8W#$=l}j!e#OD{MqO{R>;0#H z`Qh(Be)mV*+;@3g?%#j-^~b;c&fD5t`MgfAEBVh4KjFatH_rRq;yCBO+kcI#`epfz zFV%iC{%ibZoVO(2)%QDJ{c{TO$=Cjz6Yh8Xe*Ee8DL?mdd>8t&o5#nOeea$eAIfKc zbr18abLCf;u>Xc%mT$1H^M#*j<+v;M<{jO*GH@z_a;>?CeAO%<_?c<448=4fsrcuBqcpW!$k#Pu}A=KKDO9{NeXs z|L2eYc*zT_ZFB8A@17Gb%sqz~#<%+LT*H_>>ACc>T)#fGd}wE%+U-B4dkFQpHC%BD z2`4Mhp@gS#_p#Pn-N77!%hOL`JW3ZmFKk?5jd=WZ`}oI)cJovAq2)8Y_LiPQ9xu-O z_}mL#KrPQLjU#=G;+&5ze?5@xTEcT2k7?XR!rhOz-Etpinw)c<9S{lEP1^KU=$LNNpW z^79A&$$Dzxp|%og%C5I@*=2l}e)dn}|Nb=J#PSzo%^b(Hq;X}m@BKf1`uXQG-}^hJ z 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 R2G>sNIUJ}9kwRZ1`!7e_S!@6R literal 58585 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�e7_{MH50T#4_}HK$dI*_ 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 T^PTsc!6W|%JBRjt7eW94<*wVK literal 0 HcmV?d00001 diff --git a/submodules/PremiumUI/Resources/swirl.scn b/submodules/PremiumUI/Resources/swirl.scn deleted file mode 100644 index 6238c1d96d98c3440a950f8b70ee91e85b5dac26..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 16919 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 MXaC` zdw0CEHO!gwb!HWlbM3~)6->_RLtiJ+iKOr;Z&xun4<@`91_@X27I6q-wuC?YKZI

f}(`u3w- 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 literal 0 HcmV?d00001 diff --git a/submodules/TelegramUI/Sources/AccountContext.swift b/submodules/TelegramUI/Sources/AccountContext.swift index a7dce56b2ec..fd14a86226d 100644 --- a/submodules/TelegramUI/Sources/AccountContext.swift +++ b/submodules/TelegramUI/Sources/AccountContext.swift @@ -483,7 +483,7 @@ public final class AccountContextImpl: AccountContext { let context = chatLocationContext(holder: contextHolder, account: self.account, data: data) return .thread(peerId: data.peerId, threadId: data.threadId, data: context.state) } - case .feed: + case .customChatContents: preconditionFailure() } } @@ -509,7 +509,7 @@ public final class AccountContextImpl: AccountContext { } else { return .single(nil) } - case .feed: + case .customChatContents: return .single(nil) } } @@ -547,7 +547,7 @@ public final class AccountContextImpl: AccountContext { let context = chatLocationContext(holder: contextHolder, account: self.account, data: data) return context.unreadCount } - case .feed: + case .customChatContents: return .single(0) } } @@ -559,7 +559,7 @@ public final class AccountContextImpl: AccountContext { case let .replyThread(data): let context = chatLocationContext(holder: contextHolder, account: self.account, data: data) context.applyMaxReadIndex(messageIndex: messageIndex) - case .feed: + case .customChatContents: break } } diff --git a/submodules/TelegramUI/Sources/AppDelegate.swift b/submodules/TelegramUI/Sources/AppDelegate.swift index 93a5c63a840..f4eafc5bb7e 100644 --- a/submodules/TelegramUI/Sources/AppDelegate.swift +++ b/submodules/TelegramUI/Sources/AppDelegate.swift @@ -1,5 +1,6 @@ // MARK: Nicegram imports import AppLovinAdProvider +import FeatNicegramHub import FeatTasks import NGAiChat import NGAnalytics @@ -14,6 +15,7 @@ import NGOnboarding import NGRemoteConfig import NGRepoTg import NGRepoUser +import NGStats import NGStealthMode import NGStrings import SubscriptionAnalytics @@ -1210,34 +1212,36 @@ private class UserInterfaceStyleObserverWindow: UIWindow { }) // MARK: Nicegram - if #available(iOS 13.0, *) { - let _ = self.context.get().start(next: { context in - if let context = context { - TasksContainer.shared.channelSubscriptionChecker.register { - ChannelSubscriptionCheckerImpl(context: context.context) - } - } - }) - } - let _ = self.context.get().start(next: { context in if let context = context { + let accountContext = context.context + CoreContainer.shared.urlOpener.register { - UrlOpenerImpl(accountContext: context.context) + UrlOpenerImpl(accountContext: accountContext) } - } - }) - - if #available(iOS 13.0, *) { - let _ = self.context.get().start(next: { context in - if let context = context { - let accountContext = context.context + + if #available(iOS 13.0, *) { + NicegramHubContainer.shared.stickersDataProvider.register { + StickersDataProviderImpl(context: accountContext) + } + RepoTgHelper.setTelegramId( accountContext.account.peerId.id._internalGetInt64Value() ) + + TasksContainer.shared.channelSubscriptionChecker.register { + ChannelSubscriptionCheckerImpl(context: accountContext) + } } - }) - } + + if #available(iOS 13.0, *) { + Task { + let shareStickersUseCase = NicegramHubContainer.shared.shareStickersUseCase() + await shareStickersUseCase() + } + } + } + }) let _ = self.sharedContextPromise.get().start(next: { sharedContext in NGStealthMode.initialize( diff --git a/submodules/TelegramUI/Sources/Chat/ChatControllerNavigateToMessage.swift b/submodules/TelegramUI/Sources/Chat/ChatControllerNavigateToMessage.swift index 7ce0929b1ef..092847181da 100644 --- a/submodules/TelegramUI/Sources/Chat/ChatControllerNavigateToMessage.swift +++ b/submodules/TelegramUI/Sources/Chat/ChatControllerNavigateToMessage.swift @@ -104,6 +104,9 @@ extension ChatControllerImpl { if case let .peer(peerId) = self.chatLocation, messageLocation.peerId == peerId, !isPinnedMessages, !isScheduledMessages { forceInCurrentChat = true } + if case .customChatContents = self.chatLocation { + forceInCurrentChat = true + } if isPinnedMessages, let messageId = messageLocation.messageId { let _ = (combineLatest( @@ -139,7 +142,7 @@ extension ChatControllerImpl { completion?() }) - } else if case let .peer(peerId) = self.chatLocation, let messageId = messageLocation.messageId, (messageId.peerId != peerId && !forceInCurrentChat) || (isScheduledMessages && messageId.id != 0 && !Namespaces.Message.allScheduled.contains(messageId.namespace)) { + } else if case let .peer(peerId) = self.chatLocation, let messageId = messageLocation.messageId, (messageId.peerId != peerId && !forceInCurrentChat) || (isScheduledMessages && messageId.id != 0 && !Namespaces.Message.allNonRegular.contains(messageId.namespace)) { let _ = (self.context.engine.data.get( TelegramEngine.EngineData.Item.Peer.Peer(id: messageId.peerId), TelegramEngine.EngineData.Item.Messages.Message(id: messageId) @@ -205,6 +208,7 @@ extension ChatControllerImpl { if case let .id(_, params) = messageLocation { quote = params.quote.flatMap { quote in (string: quote.string, offset: quote.offset) } } + self.chatDisplayNode.historyNode.scrollToMessage(from: scrollFromIndex, to: message.index, animated: animated, quote: quote, scrollPosition: scrollPosition) if delayCompletion { diff --git a/submodules/TelegramUI/Sources/Chat/ChatControllerOpenMessageContextMenu.swift b/submodules/TelegramUI/Sources/Chat/ChatControllerOpenMessageContextMenu.swift index a396eef37f3..eacaa102761 100644 --- a/submodules/TelegramUI/Sources/Chat/ChatControllerOpenMessageContextMenu.swift +++ b/submodules/TelegramUI/Sources/Chat/ChatControllerOpenMessageContextMenu.swift @@ -290,7 +290,7 @@ extension ChatControllerImpl { if let location = location { source = .location(ChatMessageContextLocationContentSource(controller: self, location: node.view.convert(node.bounds, to: nil).origin.offsetBy(dx: location.x, dy: location.y))) } else { - source = .extracted(ChatMessageContextExtractedContentSource(chatNode: self.chatDisplayNode, engine: self.context.engine, message: message, selectAll: selectAll)) + source = .extracted(ChatMessageContextExtractedContentSource(chatController: self, chatNode: self.chatDisplayNode, engine: self.context.engine, message: message, selectAll: selectAll)) } self.canReadHistory.set(false) diff --git a/submodules/TelegramUI/Sources/ChatController.swift b/submodules/TelegramUI/Sources/ChatController.swift index 2be0fb4f156..cfd15f3af68 100644 --- a/submodules/TelegramUI/Sources/ChatController.swift +++ b/submodules/TelegramUI/Sources/ChatController.swift @@ -151,7 +151,7 @@ public final class ChatControllerOverlayPresentationData { enum ChatLocationInfoData { case peer(Promise) case replyThread(Promise) - case feed + case customChatContents } enum ChatRecordingActivity { @@ -661,10 +661,10 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G promise.set(.single(nil)) } self.chatLocationInfoData = .replyThread(promise) - case .feed: + case .customChatContents: locationBroadcastPanelSource = .none groupCallPanelSource = .none - self.chatLocationInfoData = .feed + self.chatLocationInfoData = .customChatContents } self.presentationData = context.sharedContext.currentPresentationData.with { $0 } @@ -733,6 +733,37 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G return false } + if case let .customChatContents(customChatContents) = strongSelf.presentationInterfaceState.subject { + switch customChatContents.kind { + case let .quickReplyMessageInput(_, shortcutType): + if let historyView = strongSelf.chatDisplayNode.historyNode.originalHistoryView, historyView.entries.isEmpty { + + let titleString: String + let textString: String + switch shortcutType { + case .generic: + titleString = strongSelf.presentationData.strings.QuickReply_ChatRemoveGeneric_Title + textString = strongSelf.presentationData.strings.QuickReply_ChatRemoveGeneric_Text + case .greeting: + titleString = strongSelf.presentationData.strings.QuickReply_ChatRemoveGreetingMessage_Title + textString = strongSelf.presentationData.strings.QuickReply_ChatRemoveGreetingMessage_Text + case .away: + titleString = strongSelf.presentationData.strings.QuickReply_ChatRemoveAwayMessage_Title + textString = strongSelf.presentationData.strings.QuickReply_ChatRemoveAwayMessage_Text + } + + strongSelf.present(standardTextAlertController(theme: AlertControllerTheme(presentationData: strongSelf.presentationData), title: titleString, text: textString, actions: [ + TextAlertAction(type: .genericAction, title: strongSelf.presentationData.strings.Common_Cancel, action: {}), + TextAlertAction(type: .defaultAction, title: strongSelf.presentationData.strings.QuickReply_ChatRemoveGeneric_DeleteAction, action: { [weak strongSelf] in + strongSelf?.dismiss() + }) + ]), in: .window(.root)) + + return false + } + } + } + return true } @@ -1129,7 +1160,12 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G chatFilterTag = value } - return context.sharedContext.openChatMessage(OpenChatMessageParams(context: context, updatedPresentationData: strongSelf.updatedPresentationData, chatLocation: openChatLocation, chatFilterTag: chatFilterTag, chatLocationContextHolder: strongSelf.chatLocationContextHolder, message: message, standalone: false, reverseMessageGalleryOrder: false, mode: mode, navigationController: strongSelf.effectiveNavigationController, dismissInput: { + var standalone = false + if case .customChatContents = strongSelf.chatLocation { + standalone = true + } + + return context.sharedContext.openChatMessage(OpenChatMessageParams(context: context, updatedPresentationData: strongSelf.updatedPresentationData, chatLocation: openChatLocation, chatFilterTag: chatFilterTag, chatLocationContextHolder: strongSelf.chatLocationContextHolder, message: message, standalone: standalone, reverseMessageGalleryOrder: false, mode: mode, navigationController: strongSelf.effectiveNavigationController, dismissInput: { self?.chatDisplayNode.dismissInput() }, present: { c, a in self?.present(c, in: .window(.root), with: a, blockInteraction: true) @@ -2320,7 +2356,7 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G } case .replyThread: postAsReply = true - case .feed: + case .customChatContents: postAsReply = true } @@ -2926,7 +2962,7 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G case let .replyThread(replyThreadMessage): let peerId = replyThreadMessage.peerId strongSelf.navigateToMessage(from: nil, to: .index(MessageIndex(id: MessageId(peerId: peerId, namespace: 0, id: 0), timestamp: timestamp - Int32(NSTimeZone.local.secondsFromGMT()))), scrollPosition: .bottom(0.0), rememberInStack: false, forceInCurrentChat: true, animated: true, completion: nil) - case .feed: + case .customChatContents: break } }, requestRedeliveryOfFailedMessages: { [weak self] id in @@ -2967,7 +3003,7 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G strongSelf.chatDisplayNode.messageTransitionNode.dismissMessageReactionContexts() - let controller = ContextController(presentationData: strongSelf.presentationData, source: .extracted(ChatMessageContextExtractedContentSource(chatNode: strongSelf.chatDisplayNode, engine: strongSelf.context.engine, message: message._asMessage(), selectAll: true)), items: .single(ContextController.Items(content: .list(actions))), recognizer: nil) + let controller = ContextController(presentationData: strongSelf.presentationData, source: .extracted(ChatMessageContextExtractedContentSource(chatController: strongSelf, chatNode: strongSelf.chatDisplayNode, engine: strongSelf.context.engine, message: message._asMessage(), selectAll: true)), items: .single(ContextController.Items(content: .list(actions))), recognizer: nil) strongSelf.currentContextController = controller strongSelf.forEachController({ controller in if let controller = controller as? TooltipScreen { @@ -3045,7 +3081,7 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G strongSelf.chatDisplayNode.messageTransitionNode.dismissMessageReactionContexts() - let controller = ContextController(presentationData: strongSelf.presentationData, source: .extracted(ChatMessageContextExtractedContentSource(chatNode: strongSelf.chatDisplayNode, engine: strongSelf.context.engine, message: topMessage, selectAll: true)), items: .single(ContextController.Items(content: .list(actions))), recognizer: nil) + let controller = ContextController(presentationData: strongSelf.presentationData, source: .extracted(ChatMessageContextExtractedContentSource(chatController: strongSelf, chatNode: strongSelf.chatDisplayNode, engine: strongSelf.context.engine, message: topMessage, selectAll: true)), items: .single(ContextController.Items(content: .list(actions))), recognizer: nil) strongSelf.currentContextController = controller strongSelf.forEachController({ controller in if let controller = controller as? TooltipScreen { @@ -4571,6 +4607,11 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G ) self.push(boostController) }) + }, openStickerEditor: { [weak self] in + guard let self else { + return + } + self.openStickerEditor() }, requestMessageUpdate: { [weak self] id, scroll in if let self { self.chatDisplayNode.historyNode.requestMessageUpdate(id, andScrollToItem: scroll) @@ -4820,7 +4861,7 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G chatInfoButtonItem = UIBarButtonItem(customDisplayNode: avatarNode)! self.avatarNode = avatarNode - case .feed: + case .customChatContents: chatInfoButtonItem = UIBarButtonItem(title: "", style: .plain, target: nil, action: nil) } chatInfoButtonItem.target = self @@ -5702,7 +5743,7 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G replyThreadType = .replies } } - case .feed: + case .customChatContents: replyThreadType = .replies } @@ -6135,6 +6176,13 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G } } + var appliedBoosts: Int32? + var boostsToUnrestrict: Int32? + if let cachedChannelData = peerView.cachedData as? CachedChannelData { + appliedBoosts = cachedChannelData.appliedBoosts + boostsToUnrestrict = cachedChannelData.boostsToUnrestrict + } + strongSelf.updateChatPresentationInterfaceState(animated: animated, interactive: false, { return $0.updatedPeer { _ in return renderedPeer @@ -6143,6 +6191,8 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G .updatedHasSearchTags(hasSearchTags) .updatedIsPremiumRequiredForMessaging(isPremiumRequiredForMessaging) .updatedHasSavedChats(hasSavedChats) + .updatedAppliedBoosts(appliedBoosts) + .updatedBoostsToUnrestrict(boostsToUnrestrict) .updatedInterfaceState { interfaceState in var interfaceState = interfaceState @@ -6182,11 +6232,25 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G } } })) - } else if case .feed = self.chatLocationInfoData { + } else if case .customChatContents = self.chatLocationInfoData { self.reportIrrelvantGeoNoticePromise.set(.single(nil)) self.titleDisposable.set(nil) - self.chatTitleView?.titleContent = .custom("Feed", nil, false) + if case let .customChatContents(customChatContents) = self.subject { + switch customChatContents.kind { + case let .quickReplyMessageInput(shortcut, shortcutType): + switch shortcutType { + case .generic: + self.chatTitleView?.titleContent = .custom("\(shortcut)", nil, false) + case .greeting: + self.chatTitleView?.titleContent = .custom(self.presentationData.strings.QuickReply_TitleGreetingMessage, nil, false) + case .away: + self.chatTitleView?.titleContent = .custom(self.presentationData.strings.QuickReply_TitleAwayMessage, nil, false) + } + } + } else { + self.chatTitleView?.titleContent = .custom(" ", nil, false) + } if !self.didSetChatLocationInfoReady { self.didSetChatLocationInfoReady = true @@ -6340,7 +6404,7 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G activitySpace = PeerActivitySpace(peerId: peerId, category: .global) case let .replyThread(replyThreadMessage): activitySpace = PeerActivitySpace(peerId: replyThreadMessage.peerId, category: .thread(replyThreadMessage.threadId)) - case .feed: + case .customChatContents: activitySpace = nil } @@ -7796,7 +7860,7 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G case .peer: pinnedMessageId = topPinnedMessage?.message.id pinnedMessage = topPinnedMessage - case .feed: + case .customChatContents: pinnedMessageId = nil pinnedMessage = nil } @@ -7967,7 +8031,7 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G $0.updatedChatHistoryState(state) }) - if let botStart = strongSelf.botStart, case let .loaded(isEmpty) = state { + if let botStart = strongSelf.botStart, case let .loaded(isEmpty, _) = state { strongSelf.botStart = nil if !isEmpty { strongSelf.startBot(botStart.payload) @@ -8204,20 +8268,24 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G } self.chatDisplayNode.sendMessages = { [weak self] messages, silentPosting, scheduleTime, isAnyMessageTextPartitioned in - if let strongSelf = self, let peerId = strongSelf.chatLocation.peerId { - var correlationIds: [Int64] = [] - for message in messages { - switch message { - case let .message(_, _, _, _, _, _, _, _, correlationId, _): - if let correlationId = correlationId { - correlationIds.append(correlationId) - } - default: - break + guard let strongSelf = self else { + return + } + + var correlationIds: [Int64] = [] + for message in messages { + switch message { + case let .message(_, _, _, _, _, _, _, _, correlationId, _): + if let correlationId = correlationId { + correlationIds.append(correlationId) } + default: + break } - strongSelf.commitPurposefulAction() - + } + strongSelf.commitPurposefulAction() + + if let peerId = strongSelf.chatLocation.peerId { var hasDisabledContent = false if "".isEmpty { hasDisabledContent = false @@ -8304,9 +8372,12 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G }) donateSendMessageIntent(account: strongSelf.context.account, sharedContext: strongSelf.context.sharedContext, intentContext: .chat, peerIds: [peerId]) - - strongSelf.updateChatPresentationInterfaceState(interactive: true, { $0.updatedShowCommands(false) }) + } else if case let .customChatContents(customChatContents) = strongSelf.subject { + customChatContents.enqueueMessages(messages: messages) + strongSelf.chatDisplayNode.historyNode.scrollToEndOfHistory() } + + strongSelf.updateChatPresentationInterfaceState(interactive: true, { $0.updatedShowCommands(false) }) } self.chatDisplayNode.requestUpdateChatInterfaceState = { [weak self] transition, saveInterfaceState, f in @@ -9183,7 +9254,10 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G return } - let _ = (strongSelf.context.engine.data.get(TelegramEngine.EngineData.Item.Messages.Message(id: editMessage.messageId)) + let sourceMessage: Signal + sourceMessage = strongSelf.context.engine.data.get(TelegramEngine.EngineData.Item.Messages.Message(id: editMessage.messageId)) + + let _ = (sourceMessage |> deliverOnMainQueue).start(next: { [weak strongSelf] message in guard let strongSelf, let message else { return @@ -9279,8 +9353,7 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G } let _ = (strongSelf.context.account.postbox.messageAtId(editMessage.messageId) - |> deliverOnMainQueue) - .startStandalone(next: { [weak self] currentMessage in + |> deliverOnMainQueue).startStandalone(next: { [weak self] currentMessage in if let strongSelf = self { if let currentMessage = currentMessage { let currentEntities = currentMessage.textEntitiesAttribute?.entities ?? [] @@ -9430,7 +9503,7 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G strongSelf.updateItemNodesSearchTextHighlightStates() if let navigateIndex = navigateIndex { switch strongSelf.chatLocation { - case .peer, .replyThread, .feed: + case .peer, .replyThread, .customChatContents: strongSelf.navigateToMessage(from: nil, to: .index(navigateIndex), forceInCurrentChat: true) } } @@ -9537,6 +9610,74 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G } } } + }, sendShortcut: { [weak self] shortcutId in + guard let self else { + return + } + guard let peerId = self.chatLocation.peerId else { + return + } + + self.updateChatPresentationInterfaceState(animated: true, interactive: false, { + $0.updatedInterfaceState { $0.withUpdatedReplyMessageSubject(nil).withUpdatedComposeInputState(ChatTextInputState(inputText: NSAttributedString(string: ""))).withUpdatedComposeDisableUrlPreviews([]) } + }) + + if !self.presentationInterfaceState.isPremium { + let controller = PremiumIntroScreen(context: self.context, source: .settings) + self.push(controller) + return + } + + self.context.engine.accountData.sendMessageShortcut(peerId: peerId, id: shortcutId) + + /*self.chatDisplayNode.setupSendActionOnViewUpdate({ [weak self] in + guard let self else { + return + } + self.updateChatPresentationInterfaceState(animated: true, interactive: false, { + $0.updatedInterfaceState { $0.withUpdatedReplyMessageSubject(nil).withUpdatedComposeInputState(ChatTextInputState(inputText: NSAttributedString(string: ""))).withUpdatedComposeDisableUrlPreviews([]) } + }) + }, nil) + + var messages: [EnqueueMessage] = [] + do { + let message = shortcut.topMessage + var attributes: [MessageAttribute] = [] + let entities = generateTextEntities(message.text, enabledTypes: .all) + if !entities.isEmpty { + attributes.append(TextEntitiesMessageAttribute(entities: entities)) + } + + messages.append(.message( + text: message.text, + attributes: attributes, + inlineStickers: [:], + mediaReference: message.media.first.flatMap { AnyMediaReference.standalone(media: $0) }, + threadId: self.chatLocation.threadId, + replyToMessageId: nil, + replyToStoryId: nil, + localGroupingKey: nil, + correlationId: nil, + bubbleUpEmojiOrStickersets: [] + )) + } + + self.sendMessages(messages)*/ + }, openEditShortcuts: { [weak self] in + guard let self else { + return + } + let _ = (self.context.sharedContext.makeQuickReplySetupScreenInitialData(context: self.context) + |> take(1) + |> deliverOnMainQueue).start(next: { [weak self] initialData in + guard let self else { + return + } + + let controller = self.context.sharedContext.makeQuickReplySetupScreen(context: self.context, initialData: initialData) + controller.navigationPresentation = .modal + self.push(controller) + }) }, sendBotStart: { [weak self] payload in if let strongSelf = self, canSendMessagesToChat(strongSelf.presentationInterfaceState) { strongSelf.startBot(payload) @@ -9558,9 +9699,6 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G guard let strongSelf = self else { return } - guard let peer = strongSelf.presentationInterfaceState.renderedPeer?.peer else { - return - } strongSelf.dismissAllTooltips() @@ -9570,32 +9708,34 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G } var bannedMediaInput = false - if let channel = peer as? TelegramChannel { - if channel.hasBannedPermission(.banSendVoice) != nil && channel.hasBannedPermission(.banSendInstantVideos) != nil { - bannedMediaInput = true - } else if channel.hasBannedPermission(.banSendVoice) != nil { - if !isVideo { - strongSelf.controllerInteraction?.displayUndo(.info(title: nil, text: strongSelf.restrictedSendingContentsText(), timeout: nil, customUndoText: nil)) - return - } - } else if channel.hasBannedPermission(.banSendInstantVideos) != nil { - if isVideo { - strongSelf.controllerInteraction?.displayUndo(.info(title: nil, text: strongSelf.restrictedSendingContentsText(), timeout: nil, customUndoText: nil)) - return - } - } - } else if let group = peer as? TelegramGroup { - if group.hasBannedPermission(.banSendVoice) && group.hasBannedPermission(.banSendInstantVideos) { - bannedMediaInput = true - } else if group.hasBannedPermission(.banSendVoice) { - if !isVideo { - strongSelf.controllerInteraction?.displayUndo(.info(title: nil, text: strongSelf.restrictedSendingContentsText(), timeout: nil, customUndoText: nil)) - return + if let peer = strongSelf.presentationInterfaceState.renderedPeer?.peer { + if let channel = peer as? TelegramChannel { + if channel.hasBannedPermission(.banSendVoice) != nil && channel.hasBannedPermission(.banSendInstantVideos) != nil { + bannedMediaInput = true + } else if channel.hasBannedPermission(.banSendVoice) != nil { + if !isVideo { + strongSelf.controllerInteraction?.displayUndo(.info(title: nil, text: strongSelf.restrictedSendingContentsText(), timeout: nil, customUndoText: nil)) + return + } + } else if channel.hasBannedPermission(.banSendInstantVideos) != nil { + if isVideo { + strongSelf.controllerInteraction?.displayUndo(.info(title: nil, text: strongSelf.restrictedSendingContentsText(), timeout: nil, customUndoText: nil)) + return + } } - } else if group.hasBannedPermission(.banSendInstantVideos) { - if isVideo { - strongSelf.controllerInteraction?.displayUndo(.info(title: nil, text: strongSelf.restrictedSendingContentsText(), timeout: nil, customUndoText: nil)) - return + } else if let group = peer as? TelegramGroup { + if group.hasBannedPermission(.banSendVoice) && group.hasBannedPermission(.banSendInstantVideos) { + bannedMediaInput = true + } else if group.hasBannedPermission(.banSendVoice) { + if !isVideo { + strongSelf.controllerInteraction?.displayUndo(.info(title: nil, text: strongSelf.restrictedSendingContentsText(), timeout: nil, customUndoText: nil)) + return + } + } else if group.hasBannedPermission(.banSendInstantVideos) { + if isVideo { + strongSelf.controllerInteraction?.displayUndo(.info(title: nil, text: strongSelf.restrictedSendingContentsText(), timeout: nil, customUndoText: nil)) + return + } } } } @@ -9891,37 +10031,36 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G guard let strongSelf = self else { return } - guard let peer = strongSelf.presentationInterfaceState.renderedPeer?.peer else { - return - } var bannedMediaInput = false - if let channel = peer as? TelegramChannel { - if channel.hasBannedPermission(.banSendVoice) != nil && channel.hasBannedPermission(.banSendInstantVideos) != nil { - bannedMediaInput = true - } else if channel.hasBannedPermission(.banSendVoice) != nil { - if channel.hasBannedPermission(.banSendInstantVideos) == nil { - strongSelf.displayMediaRecordingTooltip() - return - } - } else if channel.hasBannedPermission(.banSendInstantVideos) != nil { - if channel.hasBannedPermission(.banSendVoice) == nil { - strongSelf.displayMediaRecordingTooltip() - return - } - } - } else if let group = peer as? TelegramGroup { - if group.hasBannedPermission(.banSendVoice) && group.hasBannedPermission(.banSendInstantVideos) { - bannedMediaInput = true - } else if group.hasBannedPermission(.banSendVoice) { - if !group.hasBannedPermission(.banSendInstantVideos) { - strongSelf.displayMediaRecordingTooltip() - return + if let peer = strongSelf.presentationInterfaceState.renderedPeer?.peer { + if let channel = peer as? TelegramChannel { + if channel.hasBannedPermission(.banSendVoice) != nil && channel.hasBannedPermission(.banSendInstantVideos) != nil { + bannedMediaInput = true + } else if channel.hasBannedPermission(.banSendVoice) != nil { + if channel.hasBannedPermission(.banSendInstantVideos) == nil { + strongSelf.displayMediaRecordingTooltip() + return + } + } else if channel.hasBannedPermission(.banSendInstantVideos) != nil { + if channel.hasBannedPermission(.banSendVoice) == nil { + strongSelf.displayMediaRecordingTooltip() + return + } } - } else if group.hasBannedPermission(.banSendInstantVideos) { - if !group.hasBannedPermission(.banSendVoice) { - strongSelf.displayMediaRecordingTooltip() - return + } else if let group = peer as? TelegramGroup { + if group.hasBannedPermission(.banSendVoice) && group.hasBannedPermission(.banSendInstantVideos) { + bannedMediaInput = true + } else if group.hasBannedPermission(.banSendVoice) { + if !group.hasBannedPermission(.banSendInstantVideos) { + strongSelf.displayMediaRecordingTooltip() + return + } + } else if group.hasBannedPermission(.banSendInstantVideos) { + if !group.hasBannedPermission(.banSendVoice) { + strongSelf.displayMediaRecordingTooltip() + return + } } } } @@ -11376,7 +11515,7 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G activitySpace = PeerActivitySpace(peerId: peerId, category: .global) case let .replyThread(replyThreadMessage): activitySpace = PeerActivitySpace(peerId: replyThreadMessage.peerId, category: .thread(replyThreadMessage.threadId)) - case .feed: + case .customChatContents: activitySpace = nil } @@ -11789,9 +11928,11 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G // // MARK: Nicegram NGStats - if !self.didAppear { - if let peerId = self.chatLocation.peerId { - shareChannelInfo(peerId: peerId, context: self.context) + if #available(iOS 13.0, *) { + if !self.didAppear { + if let peerId = self.chatLocation.peerId { + sharePeerData(peerId: peerId, context: self.context) + } } } // @@ -12336,7 +12477,7 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G peerId = replyThreadMessage.peerId threadId = replyThreadMessage.threadId } - case .feed: + case .customChatContents: return } @@ -12903,14 +13044,16 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G self.effectiveNavigationController?.pushViewController(infoController) } } - case .feed: + case .customChatContents: break } }) case .search: self.interfaceInteraction?.beginMessageSearch(.everything, "") case .dismiss: - self.dismiss() + if self.attemptNavigation({}) { + self.dismiss() + } case .clearCache: let controller = OverlayStatusController(theme: self.presentationData.theme, type: .loading(cancelled: nil)) self.present(controller, in: .window(.root)) @@ -13115,9 +13258,11 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G })) case .replyThread: break - case .feed: + case .customChatContents: break } + case .edit: + self.editChat() } } @@ -13847,7 +13992,7 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G if let effectiveMessageId = replyThreadMessage.effectiveMessageId { defaultReplyMessageSubject = EngineMessageReplySubject(messageId: effectiveMessageId, quote: nil) } - case .feed: + case .customChatContents: break } @@ -13902,6 +14047,11 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G } func sendMessages(_ messages: [EnqueueMessage], media: Bool = false, commit: Bool = false) { + if case let .customChatContents(customChatContents) = self.subject { + customChatContents.enqueueMessages(messages: messages) + return + } + guard let peerId = self.chatLocation.peerId else { return } @@ -14025,6 +14175,15 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G }) } } + + if case let .customChatContents(customChatContents) = strongSelf.presentationInterfaceState.subject, let messageLimit = customChatContents.messageLimit { + if let originalHistoryView = strongSelf.chatDisplayNode.historyNode.originalHistoryView, originalHistoryView.entries.count + mappedMessages.count > messageLimit { + strongSelf.present(standardTextAlertController(theme: AlertControllerTheme(presentationData: strongSelf.presentationData), title: nil, text: strongSelf.presentationData.strings.Chat_QuickReplyMediaMessageLimitReachedText(Int32(messageLimit)), actions: [ + TextAlertAction(type: .genericAction, title: strongSelf.presentationData.strings.Common_OK, action: {}) + ]), in: .window(.root)) + return + } + } let messages = strongSelf.transformEnqueueMessages(mappedMessages, silentPosting: silentPosting, scheduleTime: scheduleTime) let replyMessageSubject = strongSelf.presentationInterfaceState.interfaceState.replyMessageSubject @@ -14250,10 +14409,6 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G } func requestVideoRecorder() { - guard let peerId = self.chatLocation.peerId else { - return - } - if self.videoRecorderValue == nil { if let currentInputPanelFrame = self.chatDisplayNode.currentInputPanelFrame() { if self.recorderFeedback == nil { @@ -14267,6 +14422,16 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G } var isBot = false + + var allowLiveUpload = false + var viewOnceAvailable = false + if let peerId = self.chatLocation.peerId { + allowLiveUpload = peerId.namespace != Namespaces.Peer.SecretChat + viewOnceAvailable = !isScheduledMessages && peerId.namespace == Namespaces.Peer.CloudUser && peerId != self.context.account.peerId && !isBot + } else if case .customChatContents = self.chatLocation { + allowLiveUpload = true + } + if let user = self.presentationInterfaceState.renderedPeer?.peer as? TelegramUser, user.botInfo != nil { isBot = true } @@ -14274,8 +14439,8 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G let controller = VideoMessageCameraScreen( context: self.context, updatedPresentationData: self.updatedPresentationData, - allowLiveUpload: peerId.namespace != Namespaces.Peer.SecretChat, - viewOnceAvailable: !isScheduledMessages && peerId.namespace == Namespaces.Peer.CloudUser && peerId != self.context.account.peerId && !isBot, + allowLiveUpload: allowLiveUpload, + viewOnceAvailable: viewOnceAvailable, inputPanelFrame: (currentInputPanelFrame, self.chatDisplayNode.inputNode != nil), chatNode: self.chatDisplayNode.historyNode, completion: { [weak self] message, silentPosting, scheduleTime in @@ -15585,14 +15750,20 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G let _ = strongSelf.context.engine.messages.deleteMessagesInteractively(messageIds: Array(messageIds), type: .forEveryone).startStandalone() } if let giveaway { - Queue.mainQueue().after(0.2) { - let dateString = stringForDate(timestamp: giveaway.untilDate, timeZone: .current, strings: strongSelf.presentationData.strings) - strongSelf.present(textAlertController(context: strongSelf.context, updatedPresentationData: strongSelf.updatedPresentationData, title: strongSelf.presentationData.strings.Chat_Giveaway_DeleteConfirmation_Title, text: strongSelf.presentationData.strings.Chat_Giveaway_DeleteConfirmation_Text(dateString).string, actions: [TextAlertAction(type: .destructiveAction, title: strongSelf.presentationData.strings.Common_Delete, action: { - commit() - }), TextAlertAction(type: .defaultAction, title: strongSelf.presentationData.strings.Common_Cancel, action: { - })], parseMarkdown: true), in: .window(.root)) + let currentTime = Int32(CFAbsoluteTimeGetCurrent() + kCFAbsoluteTimeIntervalSince1970) + if currentTime < giveaway.untilDate { + Queue.mainQueue().after(0.2) { + let dateString = stringForDate(timestamp: giveaway.untilDate, timeZone: .current, strings: strongSelf.presentationData.strings) + strongSelf.present(textAlertController(context: strongSelf.context, updatedPresentationData: strongSelf.updatedPresentationData, title: strongSelf.presentationData.strings.Chat_Giveaway_DeleteConfirmation_Title, text: strongSelf.presentationData.strings.Chat_Giveaway_DeleteConfirmation_Text(dateString).string, actions: [TextAlertAction(type: .destructiveAction, title: strongSelf.presentationData.strings.Common_Delete, action: { + commit() + }), TextAlertAction(type: .defaultAction, title: strongSelf.presentationData.strings.Common_Cancel, action: { + })], parseMarkdown: true), in: .window(.root)) + } + f(.default) + } else { + f(.dismissWithoutContent) + commit() } - f(.default) } else { if "".isEmpty { f(.dismissWithoutContent) @@ -16954,6 +17125,7 @@ public final class ChatControllerImpl: TelegramBaseController, ChatController, G let controller = MediaEditorScreen( context: context, + mode: .storyEditor, subject: subject, transitionIn: nil, transitionOut: { _, _ in diff --git a/submodules/TelegramUI/Sources/ChatControllerEditChat.swift b/submodules/TelegramUI/Sources/ChatControllerEditChat.swift new file mode 100644 index 00000000000..57a7666b2ac --- /dev/null +++ b/submodules/TelegramUI/Sources/ChatControllerEditChat.swift @@ -0,0 +1,64 @@ +import Foundation +import TelegramPresentationData +import AccountContext +import Postbox +import TelegramCore +import SwiftSignalKit +import Display +import TelegramPresentationData +import PresentationDataUtils +import QuickReplyNameAlertController + +extension ChatControllerImpl { + func editChat() { + if case let .customChatContents(customChatContents) = self.subject, case let .quickReplyMessageInput(currentValue, shortcutType) = customChatContents.kind, case .generic = shortcutType { + var completion: ((String?) -> Void)? + let alertController = quickReplyNameAlertController( + context: self.context, + text: self.presentationData.strings.QuickReply_EditShortcutTitle, + subtext: self.presentationData.strings.QuickReply_EditShortcutText, + value: currentValue, + characterLimit: 32, + apply: { value in + completion?(value) + } + ) + completion = { [weak self, weak alertController] value in + guard let self else { + alertController?.dismissAnimated() + return + } + if let value, !value.isEmpty { + if value == currentValue { + alertController?.dismissAnimated() + return + } + + let _ = (self.context.engine.accountData.shortcutMessageList(onlyRemote: false) + |> take(1) + |> deliverOnMainQueue).start(next: { [weak self] shortcutMessageList in + guard let self else { + alertController?.dismissAnimated() + return + } + + if shortcutMessageList.items.contains(where: { $0.shortcut.lowercased() == value.lowercased() }) { + if let contentNode = alertController?.contentNode as? QuickReplyNameAlertContentNode { + contentNode.setErrorText(errorText: self.presentationData.strings.QuickReply_ShortcutExistsInlineError) + } + } else { + self.chatTitleView?.titleContent = .custom("\(value)", nil, false) + alertController?.view.endEditing(true) + alertController?.dismissAnimated() + + if case let .customChatContents(customChatContents) = self.subject { + customChatContents.quickReplyUpdateShortcut(value: value) + } + } + }) + } + } + self.present(alertController, in: .window(.root)) + } + } +} diff --git a/submodules/TelegramUI/Sources/ChatControllerNode.swift b/submodules/TelegramUI/Sources/ChatControllerNode.swift index bab49d8e450..f160c304112 100644 --- a/submodules/TelegramUI/Sources/ChatControllerNode.swift +++ b/submodules/TelegramUI/Sources/ChatControllerNode.swift @@ -338,6 +338,8 @@ class ChatControllerNode: ASDisplayNode, UIScrollViewDelegate { private var derivedLayoutState: ChatControllerNodeDerivedLayoutState? + private var loadMoreSearchResultsDisposable: Disposable? + private var isLoadingValue: Bool = false private var isLoadingEarlier: Bool = false private func updateIsLoading(isLoading: Bool, earlier: Bool, animated: Bool) { @@ -659,6 +661,12 @@ class ChatControllerNode: ASDisplayNode, UIScrollViewDelegate { } source = .custom(messages: messages, messageId: MessageId(peerId: PeerId(0), namespace: 0, id: 0), quote: nil, loadMore: nil) } + } else if case .customChatContents = chatLocation { + if case let .customChatContents(customChatContents) = subject { + source = .customView(historyView: customChatContents.historyView) + } else { + source = .custom(messages: .single(([], 0, false)), messageId: nil, quote: nil, loadMore: nil) + } } else { source = .default } @@ -973,6 +981,7 @@ class ChatControllerNode: ASDisplayNode, UIScrollViewDelegate { self.displayVideoUnmuteTipDisposable?.dispose() self.inputMediaNodeDataDisposable?.dispose() self.inlineSearchResultsReadyDisposable?.dispose() + self.loadMoreSearchResultsDisposable?.dispose() } override func didLoad() { @@ -1765,6 +1774,8 @@ class ChatControllerNode: ASDisplayNode, UIScrollViewDelegate { isSelectionEnabled = false } else if self.chatPresentationInterfaceState.inputTextPanelState.mediaRecordingState != nil { isSelectionEnabled = false + } else if case .customChatContents = self.chatLocation { + isSelectionEnabled = false } self.historyNode.isSelectionGestureEnabled = isSelectionEnabled @@ -2012,7 +2023,7 @@ class ChatControllerNode: ASDisplayNode, UIScrollViewDelegate { overlayNavigationBar.updateLayout(size: barFrame.size, transition: transition) } - var listInsets = UIEdgeInsets(top: containerInsets.bottom + contentBottomInset, left: containerInsets.right, bottom: containerInsets.top, right: containerInsets.left) + var listInsets = UIEdgeInsets(top: containerInsets.bottom + contentBottomInset, left: containerInsets.right, bottom: containerInsets.top + 6.0, right: containerInsets.left) let listScrollIndicatorInsets = UIEdgeInsets(top: containerInsets.bottom + inputPanelsHeight, left: containerInsets.right, bottom: containerInsets.top, right: containerInsets.left) var childContentInsets: UIEdgeInsets = containerInsets @@ -2852,6 +2863,51 @@ class ChatControllerNode: ASDisplayNode, UIScrollViewDelegate { } } return foundLocalPeers + }, + loadMoreSearchResults: { [weak self] in + guard let self, let controller = self.controller else { + return + } + guard let currentSearchState = controller.searchState, let currentResultsState = controller.presentationInterfaceState.search?.resultsState else { + return + } + + self.loadMoreSearchResultsDisposable?.dispose() + self.loadMoreSearchResultsDisposable = (self.context.engine.messages.searchMessages(location: currentSearchState.location, query: currentSearchState.query, state: currentResultsState.state) + |> deliverOnMainQueue).startStrict(next: { [weak self] results, updatedState in + guard let self, let controller = self.controller else { + return + } + + controller.searchResult.set(.single((results, updatedState, currentSearchState.location))) + + var navigateIndex: MessageIndex? + controller.updateChatPresentationInterfaceState(animated: true, interactive: true, { current in + if let data = current.search { + let messageIndices = results.messages.map({ $0.index }).sorted() + var currentIndex = messageIndices.last + if let previousResultId = data.resultsState?.currentId { + for index in messageIndices { + if index.id >= previousResultId { + currentIndex = index + break + } + } + } + navigateIndex = currentIndex + return current.updatedSearch(data.withUpdatedResultsState(ChatSearchResultsState(messageIndices: messageIndices, currentId: currentIndex?.id, state: updatedState, totalCount: results.totalCount, completed: results.completed))) + } else { + return current + } + }) + if let navigateIndex = navigateIndex { + switch controller.chatLocation { + case .peer, .replyThread, .customChatContents: + controller.navigateToMessage(from: nil, to: .index(navigateIndex), forceInCurrentChat: true) + } + } + controller.updateItemNodesSearchTextHighlightStates() + }) } )), environment: {}, diff --git a/submodules/TelegramUI/Sources/ChatControllerOpenAttachmentMenu.swift b/submodules/TelegramUI/Sources/ChatControllerOpenAttachmentMenu.swift index f519a120989..8772e663cc5 100644 --- a/submodules/TelegramUI/Sources/ChatControllerOpenAttachmentMenu.swift +++ b/submodules/TelegramUI/Sources/ChatControllerOpenAttachmentMenu.swift @@ -29,6 +29,8 @@ import ChatEntityKeyboardInputNode import PremiumUI import PremiumGiftAttachmentScreen import TelegramCallsUI +import AutomaticBusinessMessageSetupScreen +import MediaEditorScreen extension ChatControllerImpl { enum AttachMenuSubject { @@ -39,10 +41,6 @@ extension ChatControllerImpl { } func presentAttachmentMenu(subject: AttachMenuSubject) { - guard let peer = self.presentationInterfaceState.renderedPeer?.peer else { - return - } - let context = self.context let inputIsActive = self.presentationInterfaceState.inputMode == .text @@ -56,42 +54,46 @@ extension ChatControllerImpl { var bannedSendFiles: (Int32, Bool)? var canSendPolls = true - if let peer = peer as? TelegramUser, peer.botInfo == nil { - canSendPolls = false - } else if peer is TelegramSecretChat { - canSendPolls = false - } else if let channel = peer as? TelegramChannel { - if let value = channel.hasBannedPermission(.banSendPhotos, ignoreDefault: canByPassRestrictions) { - bannedSendPhotos = value - } - if let value = channel.hasBannedPermission(.banSendVideos, ignoreDefault: canByPassRestrictions) { - bannedSendVideos = value - } - if let value = channel.hasBannedPermission(.banSendFiles, ignoreDefault: canByPassRestrictions) { - bannedSendFiles = value - } - if let value = channel.hasBannedPermission(.banSendText, ignoreDefault: canByPassRestrictions) { - banSendText = value - } - if channel.hasBannedPermission(.banSendPolls, ignoreDefault: canByPassRestrictions) != nil { + if let peer = self.presentationInterfaceState.renderedPeer?.peer { + if let peer = peer as? TelegramUser, peer.botInfo == nil { canSendPolls = false - } - } else if let group = peer as? TelegramGroup { - if group.hasBannedPermission(.banSendPhotos) { - bannedSendPhotos = (Int32.max, false) - } - if group.hasBannedPermission(.banSendVideos) { - bannedSendVideos = (Int32.max, false) - } - if group.hasBannedPermission(.banSendFiles) { - bannedSendFiles = (Int32.max, false) - } - if group.hasBannedPermission(.banSendText) { - banSendText = (Int32.max, false) - } - if group.hasBannedPermission(.banSendPolls) { + } else if peer is TelegramSecretChat { canSendPolls = false + } else if let channel = peer as? TelegramChannel { + if let value = channel.hasBannedPermission(.banSendPhotos, ignoreDefault: canByPassRestrictions) { + bannedSendPhotos = value + } + if let value = channel.hasBannedPermission(.banSendVideos, ignoreDefault: canByPassRestrictions) { + bannedSendVideos = value + } + if let value = channel.hasBannedPermission(.banSendFiles, ignoreDefault: canByPassRestrictions) { + bannedSendFiles = value + } + if let value = channel.hasBannedPermission(.banSendText, ignoreDefault: canByPassRestrictions) { + banSendText = value + } + if channel.hasBannedPermission(.banSendPolls, ignoreDefault: canByPassRestrictions) != nil { + canSendPolls = false + } + } else if let group = peer as? TelegramGroup { + if group.hasBannedPermission(.banSendPhotos) { + bannedSendPhotos = (Int32.max, false) + } + if group.hasBannedPermission(.banSendVideos) { + bannedSendVideos = (Int32.max, false) + } + if group.hasBannedPermission(.banSendFiles) { + bannedSendFiles = (Int32.max, false) + } + if group.hasBannedPermission(.banSendText) { + banSendText = (Int32.max, false) + } + if group.hasBannedPermission(.banSendPolls) { + canSendPolls = false + } } + } else { + canSendPolls = false } var availableButtons: [AttachmentButtonType] = [.gallery, .file] @@ -111,26 +113,31 @@ extension ChatControllerImpl { } var peerType: AttachMenuBots.Bot.PeerFlags = [] - if let user = peer as? TelegramUser { - if let _ = user.botInfo { - peerType.insert(.bot) - } else { - peerType.insert(.user) - } - } else if let _ = peer as? TelegramGroup { - peerType = .group - } else if let channel = peer as? TelegramChannel { - if case .broadcast = channel.info { - peerType = .channel - } else { + if let peer = self.presentationInterfaceState.renderedPeer?.peer { + if let user = peer as? TelegramUser { + if let _ = user.botInfo { + peerType.insert(.bot) + } else { + peerType.insert(.user) + } + } else if let _ = peer as? TelegramGroup { peerType = .group + } else if let channel = peer as? TelegramChannel { + if case .broadcast = channel.info { + peerType = .channel + } else { + peerType = .group + } } } let buttons: Signal<([AttachmentButtonType], [AttachmentButtonType], AttachmentButtonType?), NoError> - if !isScheduledMessages && !peer.isDeleted { - buttons = self.context.engine.messages.attachMenuBots() - |> map { attachMenuBots in + if let peer = self.presentationInterfaceState.renderedPeer?.peer, !isScheduledMessages, !peer.isDeleted { + buttons = combineLatest( + self.context.engine.messages.attachMenuBots(), + self.context.engine.accountData.shortcutMessageList(onlyRemote: true) |> take(1) + ) + |> map { attachMenuBots, shortcutMessageList in var buttons = availableButtons var allButtons = availableButtons var initialButton: AttachmentButtonType? @@ -164,6 +171,19 @@ extension ChatControllerImpl { allButtons.insert(button, at: 1) } + if let user = peer as? TelegramUser, user.botInfo == nil { + if let index = buttons.firstIndex(where: { $0 == .location }) { + buttons.insert(.quickReply, at: index + 1) + } else { + buttons.append(.quickReply) + } + if let index = allButtons.firstIndex(where: { $0 == .location }) { + allButtons.insert(.quickReply, at: index + 1) + } else { + allButtons.append(.quickReply) + } + } + return (buttons, allButtons, initialButton) } } else { @@ -177,7 +197,7 @@ extension ChatControllerImpl { let premiumConfiguration = PremiumConfiguration.with(appConfiguration: self.context.currentAppConfiguration.with { $0 }) let premiumGiftOptions: [CachedPremiumGiftOption] - if !premiumConfiguration.isPremiumDisabled && premiumConfiguration.showPremiumGiftInAttachMenu, let user = peer as? TelegramUser, !user.isPremium && !user.isDeleted && user.botInfo == nil && !user.flags.contains(.isSupport) { + if let peer = self.presentationInterfaceState.renderedPeer?.peer, !premiumConfiguration.isPremiumDisabled, premiumConfiguration.showPremiumGiftInAttachMenu, let user = peer as? TelegramUser, !user.isPremium && !user.isDeleted && user.botInfo == nil && !user.flags.contains(.isSupport) { premiumGiftOptions = self.presentationInterfaceState.premiumGiftOptions } else { premiumGiftOptions = [] @@ -300,16 +320,12 @@ extension ChatControllerImpl { attachmentController?.dismiss(animated: true) self?.presentICloudFileGallery() }, send: { [weak self] mediaReference in - guard let strongSelf = self, let peerId = strongSelf.chatLocation.peerId else { + guard let strongSelf = self else { return } + let message: EnqueueMessage = .message(text: "", attributes: [], inlineStickers: [:], mediaReference: mediaReference, threadId: strongSelf.chatLocation.threadId, replyToMessageId: nil, replyToStoryId: nil, localGroupingKey: nil, correlationId: nil, bubbleUpEmojiOrStickersets: []) - let _ = (enqueueMessages(account: strongSelf.context.account, peerId: peerId, messages: strongSelf.transformEnqueueMessages([message])) - |> deliverOnMainQueue).startStandalone(next: { [weak self] _ in - if let strongSelf = self, strongSelf.presentationInterfaceState.subject != .scheduledMessages { - strongSelf.chatDisplayNode.historyNode.scrollToEndOfHistory() - } - }) + strongSelf.sendMessages([message], media: true) }) if let controller = controller as? AttachmentFileControllerImpl { let _ = currentFilesController.swap(controller) @@ -324,20 +340,30 @@ extension ChatControllerImpl { return } let selfPeerId: PeerId - if let peer = peer as? TelegramChannel, case .broadcast = peer.info { - selfPeerId = peer.id - } else if let peer = peer as? TelegramChannel, case .group = peer.info, peer.hasPermission(.canBeAnonymous) { - selfPeerId = peer.id + if let peer = strongSelf.presentationInterfaceState.renderedPeer?.peer { + if let peer = peer as? TelegramChannel, case .broadcast = peer.info { + selfPeerId = peer.id + } else if let peer = peer as? TelegramChannel, case .group = peer.info, peer.hasPermission(.canBeAnonymous) { + selfPeerId = peer.id + } else { + selfPeerId = strongSelf.context.account.peerId + } } else { selfPeerId = strongSelf.context.account.peerId } let _ = (strongSelf.context.engine.data.get(TelegramEngine.EngineData.Item.Peer.Peer(id: selfPeerId)) - |> deliverOnMainQueue).startStandalone(next: { [weak self] selfPeer in + |> deliverOnMainQueue).startStandalone(next: { selfPeer in guard let strongSelf = self, let selfPeer = selfPeer else { return } - let hasLiveLocation = peer.id.namespace != Namespaces.Peer.SecretChat && peer.id != strongSelf.context.account.peerId && strongSelf.presentationInterfaceState.subject != .scheduledMessages - let controller = LocationPickerController(context: strongSelf.context, updatedPresentationData: strongSelf.updatedPresentationData, mode: .share(peer: EnginePeer(peer), selfPeer: selfPeer, hasLiveLocation: hasLiveLocation), completion: { [weak self] location, _, _, _, _ in + let hasLiveLocation: Bool + if let peer = strongSelf.presentationInterfaceState.renderedPeer?.peer { + hasLiveLocation = peer.id.namespace != Namespaces.Peer.SecretChat && peer.id != strongSelf.context.account.peerId && strongSelf.presentationInterfaceState.subject != .scheduledMessages + } else { + hasLiveLocation = false + } + let sharePeer = (strongSelf.presentationInterfaceState.renderedPeer?.peer).flatMap(EnginePeer.init) + let controller = LocationPickerController(context: strongSelf.context, updatedPresentationData: strongSelf.updatedPresentationData, mode: .share(peer: sharePeer, selfPeer: selfPeer, hasLiveLocation: hasLiveLocation), completion: { location, _, _, _, _ in guard let strongSelf = self else { return } @@ -523,69 +549,91 @@ extension ChatControllerImpl { completion(controller, controller?.mediaPickerContext) strongSelf.controllerNavigationDisposable.set(nil) case .gift: - let premiumGiftOptions = strongSelf.presentationInterfaceState.premiumGiftOptions - if !premiumGiftOptions.isEmpty { - let controller = PremiumGiftAttachmentScreen(context: context, peerIds: [peer.id], options: premiumGiftOptions, source: .attachMenu, pushController: { [weak self] c in - if let strongSelf = self { - strongSelf.push(c) - } - }, completion: { [weak self] in + if let peer = strongSelf.presentationInterfaceState.renderedPeer?.peer { + let premiumGiftOptions = strongSelf.presentationInterfaceState.premiumGiftOptions + if !premiumGiftOptions.isEmpty { + let controller = PremiumGiftAttachmentScreen(context: context, peerIds: [peer.id], options: premiumGiftOptions, source: .attachMenu, pushController: { [weak self] c in + if let strongSelf = self { + strongSelf.push(c) + } + }, completion: { [weak self] in + if let strongSelf = self { + strongSelf.hintPlayNextOutgoingGift() + strongSelf.attachmentController?.dismiss(animated: true) + } + }) + completion(controller, controller.mediaPickerContext) + strongSelf.controllerNavigationDisposable.set(nil) + + let _ = ApplicationSpecificNotice.incrementDismissedPremiumGiftSuggestion(accountManager: context.sharedContext.accountManager, peerId: peer.id).startStandalone() + } + } + case let .app(bot): + if let peer = strongSelf.presentationInterfaceState.renderedPeer?.peer { + var payload: String? + var fromAttachMenu = true + if case let .bot(_, botPayload, _) = subject { + payload = botPayload + fromAttachMenu = false + } + let params = WebAppParameters(source: fromAttachMenu ? .attachMenu : .generic, peerId: peer.id, botId: bot.peer.id, botName: bot.shortName, url: nil, queryId: nil, payload: payload, buttonText: nil, keepAliveSignal: nil, forceHasSettings: false) + let replyMessageSubject = strongSelf.presentationInterfaceState.interfaceState.replyMessageSubject + let controller = WebAppController(context: strongSelf.context, updatedPresentationData: strongSelf.updatedPresentationData, params: params, replyToMessageId: replyMessageSubject?.messageId, threadId: strongSelf.chatLocation.threadId) + controller.openUrl = { [weak self] url, concealed, commit in + self?.openUrl(url, concealed: concealed, forceExternal: true, commit: commit) + } + controller.getNavigationController = { [weak self] in + return self?.effectiveNavigationController + } + controller.completion = { [weak self] in if let strongSelf = self { - strongSelf.hintPlayNextOutgoingGift() - strongSelf.attachmentController?.dismiss(animated: true) + strongSelf.updateChatPresentationInterfaceState(animated: true, interactive: false, { + $0.updatedInterfaceState { $0.withUpdatedReplyMessageSubject(nil) } + }) + strongSelf.chatDisplayNode.historyNode.scrollToEndOfHistory() } - }) + } completion(controller, controller.mediaPickerContext) strongSelf.controllerNavigationDisposable.set(nil) - let _ = ApplicationSpecificNotice.incrementDismissedPremiumGiftSuggestion(accountManager: context.sharedContext.accountManager, peerId: peer.id).startStandalone() - } - case let .app(bot): - var payload: String? - var fromAttachMenu = true - if case let .bot(_, botPayload, _) = subject { - payload = botPayload - fromAttachMenu = false - } - let params = WebAppParameters(source: fromAttachMenu ? .attachMenu : .generic, peerId: peer.id, botId: bot.peer.id, botName: bot.shortName, url: nil, queryId: nil, payload: payload, buttonText: nil, keepAliveSignal: nil, forceHasSettings: false) - let replyMessageSubject = strongSelf.presentationInterfaceState.interfaceState.replyMessageSubject - let controller = WebAppController(context: strongSelf.context, updatedPresentationData: strongSelf.updatedPresentationData, params: params, replyToMessageId: replyMessageSubject?.messageId, threadId: strongSelf.chatLocation.threadId) - controller.openUrl = { [weak self] url, concealed, commit in - self?.openUrl(url, concealed: concealed, forceExternal: true, commit: commit) - } - controller.getNavigationController = { [weak self] in - return self?.effectiveNavigationController - } - controller.completion = { [weak self] in - if let strongSelf = self { - strongSelf.updateChatPresentationInterfaceState(animated: true, interactive: false, { - $0.updatedInterfaceState { $0.withUpdatedReplyMessageSubject(nil) } + if bot.flags.contains(.notActivated) { + let alertController = webAppTermsAlertController(context: strongSelf.context, updatedPresentationData: strongSelf.updatedPresentationData, bot: bot, completion: { [weak self] allowWrite in + guard let self else { + return + } + if bot.flags.contains(.showInSettingsDisclaimer) { + let _ = self.context.engine.messages.acceptAttachMenuBotDisclaimer(botId: bot.peer.id).startStandalone() + } + let _ = (self.context.engine.messages.addBotToAttachMenu(botId: bot.peer.id, allowWrite: allowWrite) + |> deliverOnMainQueue).startStandalone(error: { _ in + }, completed: { [weak controller] in + controller?.refresh() + }) + }, + dismissed: { + strongSelf.attachmentController?.dismiss(animated: true) }) - strongSelf.chatDisplayNode.historyNode.scrollToEndOfHistory() + strongSelf.present(alertController, in: .window(.root)) } } - completion(controller, controller.mediaPickerContext) - strongSelf.controllerNavigationDisposable.set(nil) - - if bot.flags.contains(.notActivated) { - let alertController = webAppTermsAlertController(context: strongSelf.context, updatedPresentationData: strongSelf.updatedPresentationData, bot: bot, completion: { [weak self] allowWrite in - guard let self else { + case .quickReply: + let _ = (strongSelf.context.sharedContext.makeQuickReplySetupScreenInitialData(context: strongSelf.context) + |> take(1) + |> deliverOnMainQueue).start(next: { [weak strongSelf] initialData in + guard let strongSelf else { + return + } + + let controller = QuickReplySetupScreen(context: strongSelf.context, initialData: initialData as! QuickReplySetupScreen.InitialData, mode: .select(completion: { [weak strongSelf] shortcutId in + guard let strongSelf else { return } - if bot.flags.contains(.showInSettingsDisclaimer) { - let _ = self.context.engine.messages.acceptAttachMenuBotDisclaimer(botId: bot.peer.id).startStandalone() - } - let _ = (self.context.engine.messages.addBotToAttachMenu(botId: bot.peer.id, allowWrite: allowWrite) - |> deliverOnMainQueue).startStandalone(error: { _ in - }, completed: { [weak controller] in - controller?.refresh() - }) - }, - dismissed: { strongSelf.attachmentController?.dismiss(animated: true) - }) - strongSelf.present(alertController, in: .window(.root)) - } + strongSelf.interfaceInteraction?.sendShortcut(shortcutId) + })) + completion(controller, controller.mediaPickerContext) + strongSelf.controllerNavigationDisposable.set(nil) + }) default: break } @@ -632,26 +680,28 @@ extension ChatControllerImpl { return entry ?? GeneratedMediaStoreSettings.defaultSettings } |> deliverOnMainQueue).startStandalone(next: { [weak self] settings in - guard let strongSelf = self, let peer = strongSelf.presentationInterfaceState.renderedPeer?.peer else { + guard let strongSelf = self else { return } strongSelf.chatDisplayNode.dismissInput() var bannedSendMedia: (Int32, Bool)? var canSendPolls = true - if let channel = peer as? TelegramChannel { - if let value = channel.hasBannedPermission(.banSendMedia) { - bannedSendMedia = value - } - if channel.hasBannedPermission(.banSendPolls) != nil { - canSendPolls = false - } - } else if let group = peer as? TelegramGroup { - if group.hasBannedPermission(.banSendMedia) { - bannedSendMedia = (Int32.max, false) - } - if group.hasBannedPermission(.banSendPolls) { - canSendPolls = false + if let peer = strongSelf.presentationInterfaceState.renderedPeer?.peer { + if let channel = peer as? TelegramChannel { + if let value = channel.hasBannedPermission(.banSendMedia) { + bannedSendMedia = value + } + if channel.hasBannedPermission(.banSendPolls) != nil { + canSendPolls = false + } + } else if let group = peer as? TelegramGroup { + if group.hasBannedPermission(.banSendMedia) { + bannedSendMedia = (Int32.max, false) + } + if group.hasBannedPermission(.banSendPolls) { + canSendPolls = false + } } } @@ -719,11 +769,15 @@ extension ChatControllerImpl { } var slowModeEnabled = false - if let channel = peer as? TelegramChannel, channel.isRestrictedBySlowmode { - slowModeEnabled = true + var hasSchedule = false + if let peer = strongSelf.presentationInterfaceState.renderedPeer?.peer { + if let channel = peer as? TelegramChannel, channel.isRestrictedBySlowmode { + slowModeEnabled = true + } + hasSchedule = strongSelf.presentationInterfaceState.subject != .scheduledMessages && peer.id.namespace != Namespaces.Peer.SecretChat } - let controller = legacyAttachmentMenu(context: strongSelf.context, peer: peer, threadTitle: strongSelf.threadInfo?.title, chatLocation: strongSelf.chatLocation, editMediaOptions: menuEditMediaOptions, saveEditedPhotos: settings.storeEditedPhotos, allowGrouping: true, hasSchedule: strongSelf.presentationInterfaceState.subject != .scheduledMessages && peer.id.namespace != Namespaces.Peer.SecretChat, canSendPolls: canSendPolls, updatedPresentationData: strongSelf.updatedPresentationData, parentController: legacyController, recentlyUsedInlineBots: strongSelf.recentlyUsedInlineBotsValue, initialCaption: inputText, openGallery: { + let controller = legacyAttachmentMenu(context: strongSelf.context, peer: strongSelf.presentationInterfaceState.renderedPeer?.peer, threadTitle: strongSelf.threadInfo?.title, chatLocation: strongSelf.chatLocation, editMediaOptions: menuEditMediaOptions, saveEditedPhotos: settings.storeEditedPhotos, allowGrouping: true, hasSchedule: hasSchedule, canSendPolls: canSendPolls, updatedPresentationData: strongSelf.updatedPresentationData, parentController: legacyController, recentlyUsedInlineBots: strongSelf.recentlyUsedInlineBotsValue, initialCaption: inputText, openGallery: { self?.presentOldMediaPicker(fileMode: false, editingMedia: editMediaOptions != nil, completion: { signals, silentPosting, scheduleTime in if !inputText.string.isEmpty { strongSelf.clearInputText() @@ -735,7 +789,7 @@ extension ChatControllerImpl { } }) }, openCamera: { [weak self] cameraView, menuController in - if let strongSelf = self, let peer = strongSelf.presentationInterfaceState.renderedPeer?.peer { + if let strongSelf = self { var enablePhoto = true var enableVideo = true @@ -746,19 +800,21 @@ extension ChatControllerImpl { var bannedSendPhotos: (Int32, Bool)? var bannedSendVideos: (Int32, Bool)? - if let channel = peer as? TelegramChannel { - if let value = channel.hasBannedPermission(.banSendPhotos) { - bannedSendPhotos = value - } - if let value = channel.hasBannedPermission(.banSendVideos) { - bannedSendVideos = value - } - } else if let group = peer as? TelegramGroup { - if group.hasBannedPermission(.banSendPhotos) { - bannedSendPhotos = (Int32.max, false) - } - if group.hasBannedPermission(.banSendVideos) { - bannedSendVideos = (Int32.max, false) + if let peer = strongSelf.presentationInterfaceState.renderedPeer?.peer { + if let channel = peer as? TelegramChannel { + if let value = channel.hasBannedPermission(.banSendPhotos) { + bannedSendPhotos = value + } + if let value = channel.hasBannedPermission(.banSendVideos) { + bannedSendVideos = value + } + } else if let group = peer as? TelegramGroup { + if group.hasBannedPermission(.banSendPhotos) { + bannedSendPhotos = (Int32.max, false) + } + if group.hasBannedPermission(.banSendVideos) { + bannedSendVideos = (Int32.max, false) + } } } @@ -769,7 +825,15 @@ extension ChatControllerImpl { enableVideo = false } - presentedLegacyCamera(context: strongSelf.context, peer: peer, chatLocation: strongSelf.chatLocation, cameraView: cameraView, menuController: menuController, parentController: strongSelf, editingMedia: editMediaOptions != nil, saveCapturedPhotos: peer.id.namespace != Namespaces.Peer.SecretChat, mediaGrouping: true, initialCaption: inputText, hasSchedule: strongSelf.presentationInterfaceState.subject != .scheduledMessages && peer.id.namespace != Namespaces.Peer.SecretChat, enablePhoto: enablePhoto, enableVideo: enableVideo, sendMessagesWithSignals: { [weak self] signals, silentPosting, scheduleTime in + var storeCapturedPhotos = false + var hasSchedule = false + if let peer = strongSelf.presentationInterfaceState.renderedPeer?.peer { + storeCapturedPhotos = peer.id.namespace != Namespaces.Peer.SecretChat + + hasSchedule = strongSelf.presentationInterfaceState.subject != .scheduledMessages && peer.id.namespace != Namespaces.Peer.SecretChat + } + + presentedLegacyCamera(context: strongSelf.context, peer: strongSelf.presentationInterfaceState.renderedPeer?.peer, chatLocation: strongSelf.chatLocation, cameraView: cameraView, menuController: menuController, parentController: strongSelf, editingMedia: editMediaOptions != nil, saveCapturedPhotos: storeCapturedPhotos, mediaGrouping: true, initialCaption: inputText, hasSchedule: hasSchedule, enablePhoto: enablePhoto, enableVideo: enableVideo, sendMessagesWithSignals: { [weak self] signals, silentPosting, scheduleTime in if let strongSelf = self { if editMediaOptions != nil { strongSelf.editMessageMediaWithLegacySignals(signals!) @@ -927,7 +991,7 @@ extension ChatControllerImpl { func presentICloudFileGallery(editingMessage: Bool = false) { let _ = (self.context.engine.data.get( - TelegramEngine.EngineData.Item.Peer.Peer(id: context.account.peerId), + TelegramEngine.EngineData.Item.Peer.Peer(id: self.context.account.peerId), TelegramEngine.EngineData.Item.Configuration.UserLimits(isPremium: false), TelegramEngine.EngineData.Item.Configuration.UserLimits(isPremium: true) ) @@ -1063,9 +1127,6 @@ extension ChatControllerImpl { } func presentMediaPicker(subject: MediaPickerScreen.Subject = .assets(nil, .default), saveEditedPhotos: Bool, bannedSendPhotos: (Int32, Bool)?, bannedSendVideos: (Int32, Bool)?, present: @escaping (MediaPickerScreen, AttachmentMediaPickerContext?) -> Void, updateMediaPickerContext: @escaping (AttachmentMediaPickerContext?) -> Void, completion: @escaping ([Any], Bool, Int32?, @escaping (String) -> UIView?, @escaping () -> Void) -> Void) { - guard let peer = self.presentationInterfaceState.renderedPeer?.peer else { - return - } var isScheduledMessages = false if case .scheduledMessages = self.presentationInterfaceState.subject { isScheduledMessages = true @@ -1073,7 +1134,7 @@ extension ChatControllerImpl { let controller = MediaPickerScreen( context: self.context, updatedPresentationData: self.updatedPresentationData, - peer: EnginePeer(peer), + peer: (self.presentationInterfaceState.renderedPeer?.peer).flatMap(EnginePeer.init), threadTitle: self.threadInfo?.title, chatLocation: self.chatLocation, isScheduledMessages: isScheduledMessages, @@ -1561,17 +1622,12 @@ extension ChatControllerImpl { } func openCamera(cameraView: TGAttachmentCameraView? = nil) { - guard let peer = self.presentationInterfaceState.renderedPeer?.peer else { - return - } - let _ = peer - let _ = (self.context.sharedContext.accountManager.transaction { transaction -> GeneratedMediaStoreSettings in let entry = transaction.getSharedData(ApplicationSpecificSharedDataKeys.generatedMediaStoreSettings)?.get(GeneratedMediaStoreSettings.self) return entry ?? GeneratedMediaStoreSettings.defaultSettings } |> deliverOnMainQueue).startStandalone(next: { [weak self] settings in - guard let strongSelf = self, let peer = strongSelf.presentationInterfaceState.renderedPeer?.peer else { + guard let strongSelf = self else { return } @@ -1585,19 +1641,21 @@ extension ChatControllerImpl { var bannedSendPhotos: (Int32, Bool)? var bannedSendVideos: (Int32, Bool)? - if let channel = peer as? TelegramChannel { - if let value = channel.hasBannedPermission(.banSendPhotos) { - bannedSendPhotos = value - } - if let value = channel.hasBannedPermission(.banSendVideos) { - bannedSendVideos = value - } - } else if let group = peer as? TelegramGroup { - if group.hasBannedPermission(.banSendPhotos) { - bannedSendPhotos = (Int32.max, false) - } - if group.hasBannedPermission(.banSendVideos) { - bannedSendVideos = (Int32.max, false) + if let peer = strongSelf.presentationInterfaceState.renderedPeer?.peer { + if let channel = peer as? TelegramChannel { + if let value = channel.hasBannedPermission(.banSendPhotos) { + bannedSendPhotos = value + } + if let value = channel.hasBannedPermission(.banSendVideos) { + bannedSendVideos = value + } + } else if let group = peer as? TelegramGroup { + if group.hasBannedPermission(.banSendPhotos) { + bannedSendPhotos = (Int32.max, false) + } + if group.hasBannedPermission(.banSendVideos) { + bannedSendVideos = (Int32.max, false) + } } } @@ -1608,10 +1666,15 @@ extension ChatControllerImpl { enableVideo = false } - let storeCapturedMedia = peer.id.namespace != Namespaces.Peer.SecretChat + var storeCapturedMedia = false + var hasSchedule = false + if let peer = strongSelf.presentationInterfaceState.renderedPeer?.peer { + storeCapturedMedia = peer.id.namespace != Namespaces.Peer.SecretChat + hasSchedule = strongSelf.presentationInterfaceState.subject != .scheduledMessages && peer.id.namespace != Namespaces.Peer.SecretChat + } let inputText = strongSelf.presentationInterfaceState.interfaceState.effectiveInputState.inputText - presentedLegacyCamera(context: strongSelf.context, peer: peer, chatLocation: strongSelf.chatLocation, cameraView: cameraView, menuController: nil, parentController: strongSelf, attachmentController: self?.attachmentController, editingMedia: false, saveCapturedPhotos: storeCapturedMedia, mediaGrouping: true, initialCaption: inputText, hasSchedule: strongSelf.presentationInterfaceState.subject != .scheduledMessages && peer.id.namespace != Namespaces.Peer.SecretChat, enablePhoto: enablePhoto, enableVideo: enableVideo, sendMessagesWithSignals: { [weak self] signals, silentPosting, scheduleTime in + presentedLegacyCamera(context: strongSelf.context, peer: strongSelf.presentationInterfaceState.renderedPeer?.peer, chatLocation: strongSelf.chatLocation, cameraView: cameraView, menuController: nil, parentController: strongSelf, attachmentController: self?.attachmentController, editingMedia: false, saveCapturedPhotos: storeCapturedMedia, mediaGrouping: true, initialCaption: inputText, hasSchedule: hasSchedule, enablePhoto: enablePhoto, enableVideo: enableVideo, sendMessagesWithSignals: { [weak self] signals, silentPosting, scheduleTime in if let strongSelf = self { strongSelf.enqueueMediaMessages(signals: signals, silentPosting: silentPosting, scheduleTime: scheduleTime > 0 ? scheduleTime : nil) if !inputText.string.isEmpty { @@ -1650,4 +1713,71 @@ extension ChatControllerImpl { }) }) } + + func openStickerEditor() { + let mainController = AttachmentController(context: self.context, updatedPresentationData: self.updatedPresentationData, chatLocation: nil, buttons: [.standalone], initialButton: .standalone, fromMenu: false, hasTextInput: false, makeEntityInputView: { + return nil + }) +// controller.forceSourceRect = true +// controller.getSourceRect = getSourceRect + mainController.requestController = { [weak self, weak mainController] _, present in + guard let self else { + return + } + let mediaPickerController = MediaPickerScreen(context: self.context, updatedPresentationData: self.updatedPresentationData, peer: nil, threadTitle: nil, chatLocation: nil, subject: .assets(nil, .createSticker)) + mediaPickerController.customSelection = { [weak self, weak mainController] controller, result in + guard let self else { + return + } + if let result = result as? PHAsset { + controller.updateHiddenMediaId(result.localIdentifier) + if let transitionView = controller.transitionView(for: result.localIdentifier, snapshot: false) { + let editorController = MediaEditorScreen( + context: self.context, + mode: .stickerEditor, + subject: .single(.asset(result)), + transitionIn: .gallery( + MediaEditorScreen.TransitionIn.GalleryTransitionIn( + sourceView: transitionView, + sourceRect: transitionView.bounds, + sourceImage: controller.transitionImage(for: result.localIdentifier) + ) + ), + transitionOut: { finished, isNew in + if !finished { + return MediaEditorScreen.TransitionOut( + destinationView: transitionView, + destinationRect: transitionView.bounds, + destinationCornerRadius: 0.0 + ) + } + return nil + }, completion: { [weak self, weak mainController] result, commit in + mainController?.dismiss() + + Queue.mainQueue().after(0.1) { + commit({}) + if let mediaResult = result.media, case let .image(image, _) = mediaResult { + self?.enqueueStickerImage(image, isMemoji: false) + } + } + } as (MediaEditorScreen.Result, @escaping (@escaping () -> Void) -> Void) -> Void + ) + editorController.dismissed = { [weak controller] in + controller?.updateHiddenMediaId(nil) + } + self.push(editorController) + +// completion(result, transitionView, transitionView.bounds, controller.transitionImage(for: result.localIdentifier), transitionOut, { [weak controller] in +// controller?.updateHiddenMediaId(nil) +// }) + } + } + } + present(mediaPickerController, mediaPickerController.mediaPickerContext) + } + mainController.navigationPresentation = .flatModal + mainController.supportedOrientations = ViewControllerSupportedOrientations(regularSize: .all, compactSize: .portrait) + self.push(mainController) + } } diff --git a/submodules/TelegramUI/Sources/ChatControllerOpenCalendarSearch.swift b/submodules/TelegramUI/Sources/ChatControllerOpenCalendarSearch.swift index b29df6f9416..62ae3df62d2 100644 --- a/submodules/TelegramUI/Sources/ChatControllerOpenCalendarSearch.swift +++ b/submodules/TelegramUI/Sources/ChatControllerOpenCalendarSearch.swift @@ -59,7 +59,7 @@ extension ChatControllerImpl { case let .replyThread(replyThreadMessage): peerId = replyThreadMessage.peerId threadId = replyThreadMessage.threadId - case .feed: + case .customChatContents: return } @@ -96,7 +96,7 @@ extension ChatControllerImpl { case let .replyThread(replyThreadMessage): peerId = replyThreadMessage.peerId threadId = replyThreadMessage.threadId - case .feed: + case .customChatContents: return } diff --git a/submodules/TelegramUI/Sources/ChatControllerOpenMessageShareMenu.swift b/submodules/TelegramUI/Sources/ChatControllerOpenMessageShareMenu.swift index 1e6576268b2..d1445389c0e 100644 --- a/submodules/TelegramUI/Sources/ChatControllerOpenMessageShareMenu.swift +++ b/submodules/TelegramUI/Sources/ChatControllerOpenMessageShareMenu.swift @@ -31,7 +31,7 @@ func chatShareToSavedMessagesAdditionalView(_ chatController: ChatControllerImpl return } - let _ = (chatController.context.account.postbox.aroundMessageHistoryViewForLocation(.peer(peerId: chatController.context.account.peerId, threadId: nil), anchor: .upperBound, ignoreMessagesInTimestampRange: nil, count: 45, fixedCombinedReadStates: nil, topTaggedMessageIdNamespaces: Set(), tag: nil, appendMessagesFromTheSameGroup: false, namespaces: .not(Namespaces.Message.allScheduled), orderStatistics: []) + let _ = (chatController.context.account.postbox.aroundMessageHistoryViewForLocation(.peer(peerId: chatController.context.account.peerId, threadId: nil), anchor: .upperBound, ignoreMessagesInTimestampRange: nil, count: 45, fixedCombinedReadStates: nil, topTaggedMessageIdNamespaces: Set(), tag: nil, appendMessagesFromTheSameGroup: false, namespaces: .not(Namespaces.Message.allNonRegular), orderStatistics: []) |> map { view, _, _ -> [EngineMessage.Id] in let messageIds = correlationIds.compactMap { correlationId in return chatController.context.engine.messages.synchronouslyLookupCorrelationId(correlationId: correlationId) diff --git a/submodules/TelegramUI/Sources/ChatControllerUpdateSearch.swift b/submodules/TelegramUI/Sources/ChatControllerUpdateSearch.swift index 75d9bbc33f9..59f3755db38 100644 --- a/submodules/TelegramUI/Sources/ChatControllerUpdateSearch.swift +++ b/submodules/TelegramUI/Sources/ChatControllerUpdateSearch.swift @@ -33,7 +33,7 @@ extension ChatControllerImpl { break case let .replyThread(replyThreadMessage): threadId = replyThreadMessage.threadId - case .feed: + case .customChatContents: break } @@ -125,7 +125,7 @@ extension ChatControllerImpl { }) if let navigateIndex = navigateIndex { switch strongSelf.chatLocation { - case .peer, .replyThread, .feed: + case .peer, .replyThread, .customChatContents: strongSelf.navigateToMessage(from: nil, to: .index(navigateIndex), forceInCurrentChat: true) } } diff --git a/submodules/TelegramUI/Sources/ChatEmptyNode.swift b/submodules/TelegramUI/Sources/ChatEmptyNode.swift index 5983a2c37a9..b81dd01bc04 100644 --- a/submodules/TelegramUI/Sources/ChatEmptyNode.swift +++ b/submodules/TelegramUI/Sources/ChatEmptyNode.swift @@ -24,8 +24,8 @@ private protocol ChatEmptyNodeContent { func updateLayout(interfaceState: ChatPresentationInterfaceState, subject: ChatEmptyNode.Subject, size: CGSize, transition: ContainedViewLayoutTransition) -> CGSize } -private let titleFont = Font.medium(15.0) -private let messageFont = Font.regular(14.0) +private let titleFont = Font.semibold(15.0) +private let messageFont = Font.regular(13.0) private final class ChatEmptyNodeRegularChatContent: ASDisplayNode, ChatEmptyNodeContent { private let textNode: ImmediateTextNode @@ -53,7 +53,7 @@ private final class ChatEmptyNodeRegularChatContent: ASDisplayNode, ChatEmptyNod text = interfaceState.strings.ChatList_StartMessaging } else { switch interfaceState.chatLocation { - case .peer, .replyThread, .feed: + case .peer, .replyThread, .customChatContents: if case .scheduledMessages = interfaceState.subject { text = interfaceState.strings.ScheduledMessages_EmptyPlaceholder } else { @@ -701,25 +701,87 @@ private final class ChatEmptyNodeCloudChatContent: ASDisplayNode, ChatEmptyNodeC } func updateLayout(interfaceState: ChatPresentationInterfaceState, subject: ChatEmptyNode.Subject, size: CGSize, transition: ContainedViewLayoutTransition) -> CGSize { + var maxWidth: CGFloat = size.width + var centerText = false + + var insets = UIEdgeInsets(top: 15.0, left: 15.0, bottom: 15.0, right: 15.0) + var imageSpacing: CGFloat = 12.0 + var titleSpacing: CGFloat = 4.0 + + if case let .customChatContents(customChatContents) = interfaceState.subject { + maxWidth = min(240.0, maxWidth) + + switch customChatContents.kind { + case .quickReplyMessageInput: + insets.top = 10.0 + imageSpacing = 5.0 + titleSpacing = 5.0 + } + } + if self.currentTheme !== interfaceState.theme || self.currentStrings !== interfaceState.strings { self.currentTheme = interfaceState.theme self.currentStrings = interfaceState.strings let serviceColor = serviceMessageColorComponents(theme: interfaceState.theme, wallpaper: interfaceState.chatWallpaper) - self.iconNode.image = generateTintedImage(image: UIImage(bundleImageName: "Chat/Empty Chat/Cloud"), color: serviceColor.primaryText) + var iconName = "Chat/Empty Chat/Cloud" - let titleString = interfaceState.strings.Conversation_CloudStorageInfo_Title - self.titleNode.attributedText = NSAttributedString(string: titleString, font: titleFont, textColor: serviceColor.primaryText) + let titleString: String + let strings: [String] - let strings: [String] = [ - interfaceState.strings.Conversation_ClousStorageInfo_Description1, - interfaceState.strings.Conversation_ClousStorageInfo_Description2, - interfaceState.strings.Conversation_ClousStorageInfo_Description3, - interfaceState.strings.Conversation_ClousStorageInfo_Description4 - ] + if case let .customChatContents(customChatContents) = interfaceState.subject { + switch customChatContents.kind { + case let .quickReplyMessageInput(shortcut, shortcutType): + switch shortcutType { + case .generic: + iconName = "Chat/Empty Chat/QuickReplies" + centerText = false + titleString = interfaceState.strings.Chat_EmptyState_QuickReply_Title + strings = [ + interfaceState.strings.Chat_EmptyState_QuickReply_Text1(shortcut).string, + interfaceState.strings.Chat_EmptyState_QuickReply_Text2 + ] + case .greeting: + iconName = "Chat/Empty Chat/GreetingShortcut" + centerText = true + titleString = interfaceState.strings.EmptyState_GreetingMessage_Title + strings = [ + interfaceState.strings.EmptyState_GreetingMessage_Text + ] + case .away: + iconName = "Chat/Empty Chat/AwayShortcut" + centerText = true + titleString = interfaceState.strings.EmptyState_AwayMessage_Title + strings = [ + interfaceState.strings.EmptyState_AwayMessage_Text + ] + } + } + } else { + titleString = interfaceState.strings.Conversation_CloudStorageInfo_Title + strings = [ + interfaceState.strings.Conversation_ClousStorageInfo_Description1, + interfaceState.strings.Conversation_ClousStorageInfo_Description2, + interfaceState.strings.Conversation_ClousStorageInfo_Description3, + interfaceState.strings.Conversation_ClousStorageInfo_Description4 + ] + } - let lines: [NSAttributedString] = strings.map { NSAttributedString(string: $0, font: messageFont, textColor: serviceColor.primaryText) } + self.iconNode.image = generateTintedImage(image: UIImage(bundleImageName: iconName), color: serviceColor.primaryText) + + self.titleNode.attributedText = NSAttributedString(string: titleString, font: titleFont, textColor: serviceColor.primaryText) + + let lines: [NSAttributedString] = strings.map { + return parseMarkdownIntoAttributedString($0, attributes: MarkdownAttributes( + body: MarkdownAttributeSet(font: Font.regular(14.0), textColor: serviceColor.primaryText), + bold: MarkdownAttributeSet(font: Font.semibold(14.0), textColor: serviceColor.primaryText), + link: MarkdownAttributeSet(font: Font.regular(14.0), textColor: serviceColor.primaryText), + linkAttribute: { url in + return ("URL", url) + } + ), textAlignment: centerText ? .center : .natural) + } for i in 0 ..< lines.count { if i >= self.lineNodes.count { @@ -727,6 +789,7 @@ private final class ChatEmptyNodeCloudChatContent: ASDisplayNode, ChatEmptyNodeC textNode.maximumNumberOfLines = 0 textNode.isUserInteractionEnabled = false textNode.displaysAsynchronously = false + textNode.textAlignment = centerText ? .center : .natural self.addSubnode(textNode) self.lineNodes.append(textNode) } @@ -735,11 +798,6 @@ private final class ChatEmptyNodeCloudChatContent: ASDisplayNode, ChatEmptyNodeC } } - let insets = UIEdgeInsets(top: 15.0, left: 15.0, bottom: 15.0, right: 15.0) - - let imageSpacing: CGFloat = 12.0 - let titleSpacing: CGFloat = 4.0 - var contentWidth: CGFloat = 100.0 var contentHeight: CGFloat = 0.0 @@ -751,7 +809,7 @@ private final class ChatEmptyNodeCloudChatContent: ASDisplayNode, ChatEmptyNodeC var lineNodes: [(CGSize, ImmediateTextNode)] = [] for textNode in self.lineNodes { - let textSize = textNode.updateLayout(CGSize(width: size.width - insets.left - insets.right - 10.0, height: CGFloat.greatestFiniteMagnitude)) + let textSize = textNode.updateLayout(CGSize(width: maxWidth - insets.left - insets.right - 10.0, height: CGFloat.greatestFiniteMagnitude)) contentWidth = max(contentWidth, textSize.width) contentHeight += textSize.height + titleSpacing lineNodes.append((textSize, textNode)) @@ -1166,7 +1224,9 @@ final class ChatEmptyNode: ASDisplayNode { case .detailsPlaceholder: contentType = .regular case let .emptyChat(emptyType): - if case .replyThread = interfaceState.chatLocation { + if case .customChatContents = interfaceState.subject { + contentType = .cloud + } else if case .replyThread = interfaceState.chatLocation { if case .topic = emptyType { contentType = .topic } else { diff --git a/submodules/TelegramUI/Sources/ChatHistoryEntriesForView.swift b/submodules/TelegramUI/Sources/ChatHistoryEntriesForView.swift index 3af4fe35df3..29924427880 100644 --- a/submodules/TelegramUI/Sources/ChatHistoryEntriesForView.swift +++ b/submodules/TelegramUI/Sources/ChatHistoryEntriesForView.swift @@ -516,6 +516,64 @@ func chatHistoryEntriesForView( } } + if let subject = associatedData.subject, case let .customChatContents(customChatContents) = subject, case let .quickReplyMessageInput(_, shortcutType) = customChatContents.kind, case .generic = shortcutType { + if !view.isLoading && view.laterId == nil && !view.entries.isEmpty { + for i in 0 ..< 2 { + let string = i == 1 ? presentationData.strings.Chat_QuickReply_ServiceHeader1 : presentationData.strings.Chat_QuickReply_ServiceHeader2 + let formattedString = parseMarkdownIntoAttributedString( + string, + attributes: MarkdownAttributes( + body: MarkdownAttributeSet(font: Font.regular(15.0), textColor: .black), + bold: MarkdownAttributeSet(font: Font.regular(15.0), textColor: .black), + link: MarkdownAttributeSet(font: Font.regular(15.0), textColor: .white), + linkAttribute: { url in + return ("URL", url) + } + ) + ) + var entities: [MessageTextEntity] = [] + formattedString.enumerateAttribute(.foregroundColor, in: NSRange(location: 0, length: formattedString.length), options: [], using: { value, range, _ in + if let value = value as? UIColor, value == .white { + entities.append(MessageTextEntity(range: range.lowerBound ..< range.upperBound, type: .Bold)) + } + }) + formattedString.enumerateAttribute(NSAttributedString.Key(rawValue: "URL"), in: NSRange(location: 0, length: formattedString.length), options: [], using: { value, range, _ in + if value != nil { + entities.append(MessageTextEntity(range: range.lowerBound ..< range.upperBound, type: .TextMention(peerId: context.account.peerId))) + } + }) + + let message = Message( + stableId: UInt32.max - 1001 - UInt32(i), + stableVersion: 0, + id: MessageId(peerId: context.account.peerId, namespace: Namespaces.Message.Local, id: Int32.max - 100 - Int32(i)), + globallyUniqueId: nil, + groupingKey: nil, + groupInfo: nil, + threadId: nil, + timestamp: -Int32(i), + flags: [.Incoming], + tags: [], + globalTags: [], + localTags: [], + customTags: [], + forwardInfo: nil, + author: nil, + text: "", + attributes: [], + media: [TelegramMediaAction(action: .customText(text: formattedString.string, entities: entities, additionalAttributes: nil))], + peers: SimpleDictionary(), + associatedMessages: SimpleDictionary(), + associatedMessageIds: [], + associatedMedia: [:], + associatedThreadInfo: nil, + associatedStories: [:] + ) + entries.insert(.MessageEntry(message, presentationData, false, nil, .none, ChatMessageEntryAttributes(rank: nil, isContact: false, contentTypeHint: .generic, updatingMedia: nil, isPlaying: false, isCentered: false, authorStoryStats: nil)), at: 0) + } + } + } + if reverse { return entries.reversed() } else { diff --git a/submodules/TelegramUI/Sources/ChatHistoryListNode.swift b/submodules/TelegramUI/Sources/ChatHistoryListNode.swift index 4d926a9e311..f042200c0b1 100644 --- a/submodules/TelegramUI/Sources/ChatHistoryListNode.swift +++ b/submodules/TelegramUI/Sources/ChatHistoryListNode.swift @@ -227,6 +227,11 @@ extension ListMessageItemInteraction { } private func mappedInsertEntries(context: AccountContext, chatLocation: ChatLocation, associatedData: ChatMessageItemAssociatedData, controllerInteraction: ChatControllerInteraction, mode: ChatHistoryListMode, lastHeaderId: Int64, entries: [ChatHistoryViewTransitionInsertEntry], wantTrButton: [(Bool, [String])]) -> [ListViewInsertItem] { + var disableFloatingDateHeaders = false + if case .customChatContents = chatLocation { + disableFloatingDateHeaders = true + } + return entries.map { entry -> ListViewInsertItem in switch entry.entry { case let .MessageEntry(message, presentationData, read, location, selection, attributes): @@ -234,7 +239,7 @@ private func mappedInsertEntries(context: AccountContext, chatLocation: ChatLoca switch mode { case .bubbles: // MARK: Nicegram, wantTrButton - item = ChatMessageItemImpl(presentationData: presentationData, context: context, chatLocation: chatLocation, associatedData: associatedData, controllerInteraction: controllerInteraction, content: .message(message: message, read: read, selection: selection, attributes: attributes, location: location), wantTrButton: wantTrButton) + item = ChatMessageItemImpl(presentationData: presentationData, context: context, chatLocation: chatLocation, associatedData: associatedData, controllerInteraction: controllerInteraction, content: .message(message: message, read: read, selection: selection, attributes: attributes, location: location), disableDate: disableFloatingDateHeaders, wantTrButton: wantTrButton) case let .list(_, _, _, displayHeaders, hintLinks, isGlobalSearch): let displayHeader: Bool switch displayHeaders { @@ -253,7 +258,7 @@ private func mappedInsertEntries(context: AccountContext, chatLocation: ChatLoca switch mode { case .bubbles: // MARK: Nicegram, wantTrButton - item = ChatMessageItemImpl(presentationData: presentationData, context: context, chatLocation: chatLocation, associatedData: associatedData, controllerInteraction: controllerInteraction, content: .group(messages: messages), wantTrButton: wantTrButton) + item = ChatMessageItemImpl(presentationData: presentationData, context: context, chatLocation: chatLocation, associatedData: associatedData, controllerInteraction: controllerInteraction, content: .group(messages: messages), disableDate: disableFloatingDateHeaders, wantTrButton: wantTrButton) case .list: assertionFailure() item = ListMessageItem(presentationData: presentationData, context: context, chatLocation: chatLocation, interaction: ListMessageItemInteraction(controllerInteraction: controllerInteraction), message: messages[0].0, selection: .none, displayHeader: false) @@ -274,6 +279,11 @@ private func mappedInsertEntries(context: AccountContext, chatLocation: ChatLoca } private func mappedUpdateEntries(context: AccountContext, chatLocation: ChatLocation, associatedData: ChatMessageItemAssociatedData, controllerInteraction: ChatControllerInteraction, mode: ChatHistoryListMode, lastHeaderId: Int64, entries: [ChatHistoryViewTransitionUpdateEntry], wantTrButton: [(Bool, [String])]) -> [ListViewUpdateItem] { + var disableFloatingDateHeaders = false + if case .customChatContents = chatLocation { + disableFloatingDateHeaders = true + } + return entries.map { entry -> ListViewUpdateItem in switch entry.entry { case let .MessageEntry(message, presentationData, read, location, selection, attributes): @@ -281,7 +291,7 @@ private func mappedUpdateEntries(context: AccountContext, chatLocation: ChatLoca switch mode { case .bubbles: // MARK: Nicegram, wantTrButton - item = ChatMessageItemImpl(presentationData: presentationData, context: context, chatLocation: chatLocation, associatedData: associatedData, controllerInteraction: controllerInteraction, content: .message(message: message, read: read, selection: selection, attributes: attributes, location: location), wantTrButton: wantTrButton) + item = ChatMessageItemImpl(presentationData: presentationData, context: context, chatLocation: chatLocation, associatedData: associatedData, controllerInteraction: controllerInteraction, content: .message(message: message, read: read, selection: selection, attributes: attributes, location: location), disableDate: disableFloatingDateHeaders, wantTrButton: wantTrButton) case let .list(_, _, _, displayHeaders, hintLinks, isGlobalSearch): let displayHeader: Bool switch displayHeaders { @@ -300,7 +310,7 @@ private func mappedUpdateEntries(context: AccountContext, chatLocation: ChatLoca switch mode { case .bubbles: // MARK: Nicegram, wantTrButton - item = ChatMessageItemImpl(presentationData: presentationData, context: context, chatLocation: chatLocation, associatedData: associatedData, controllerInteraction: controllerInteraction, content: .group(messages: messages), wantTrButton: wantTrButton) + item = ChatMessageItemImpl(presentationData: presentationData, context: context, chatLocation: chatLocation, associatedData: associatedData, controllerInteraction: controllerInteraction, content: .group(messages: messages), disableDate: disableFloatingDateHeaders, wantTrButton: wantTrButton) case .list: assertionFailure() item = ListMessageItem(presentationData: presentationData, context: context, chatLocation: chatLocation, interaction: ListMessageItemInteraction(controllerInteraction: controllerInteraction), message: messages[0].0, selection: .none, displayHeader: false) @@ -475,6 +485,9 @@ public final class ChatHistoryListNodeImpl: ListView, ChatHistoryNode, ChatHisto private var enableUnreadAlignment: Bool = true private var historyView: ChatHistoryView? + public var originalHistoryView: MessageHistoryView? { + return self.historyView?.originalView + } private let historyDisposable = MetaDisposable() private let readHistoryDisposable = MetaDisposable() @@ -811,35 +824,35 @@ public final class ChatHistoryListNodeImpl: ListView, ChatHistoryNode, ChatHisto //self.debugInfo = true self.messageProcessingManager.process = { [weak context] messageIds in - context?.account.viewTracker.updateViewCountForMessageIds(messageIds: messageIds, clientId: clientId.with { $0 }) + context?.account.viewTracker.updateViewCountForMessageIds(messageIds: Set(messageIds.map(\.messageId)), clientId: clientId.with { $0 }) } self.messageWithReactionsProcessingManager.process = { [weak context] messageIds in - context?.account.viewTracker.updateReactionsForMessageIds(messageIds: messageIds) + context?.account.viewTracker.updateReactionsForMessageIds(messageIds: Set(messageIds.map(\.messageId))) } self.seenLiveLocationProcessingManager.process = { [weak context] messageIds in - context?.account.viewTracker.updateSeenLiveLocationForMessageIds(messageIds: messageIds) + context?.account.viewTracker.updateSeenLiveLocationForMessageIds(messageIds: Set(messageIds.map(\.messageId))) } self.unsupportedMessageProcessingManager.process = { [weak context] messageIds in context?.account.viewTracker.updateUnsupportedMediaForMessageIds(messageIds: messageIds) } self.refreshMediaProcessingManager.process = { [weak context] messageIds in - context?.account.viewTracker.refreshSecretMediaMediaForMessageIds(messageIds: messageIds) + context?.account.viewTracker.refreshSecretMediaMediaForMessageIds(messageIds: Set(messageIds.map(\.messageId))) } self.refreshStoriesProcessingManager.process = { [weak context] messageIds in - context?.account.viewTracker.refreshStoriesForMessageIds(messageIds: messageIds) + context?.account.viewTracker.refreshStoriesForMessageIds(messageIds: Set(messageIds.map(\.messageId))) } self.translationProcessingManager.process = { [weak self, weak context] messageIds in if let context = context, let toLang = self?.toLang { - let _ = translateMessageIds(context: context, messageIds: Array(messageIds), toLang: toLang).startStandalone() + let _ = translateMessageIds(context: context, messageIds: Array(messageIds.map(\.messageId)), toLang: toLang).startStandalone() } } self.messageMentionProcessingManager.process = { [weak self, weak context] messageIds in if let strongSelf = self { if strongSelf.canReadHistoryValue { - context?.account.viewTracker.updateMarkMentionsSeenForMessageIds(messageIds: messageIds) + context?.account.viewTracker.updateMarkMentionsSeenForMessageIds(messageIds: Set(messageIds.map(\.messageId))) } else { - strongSelf.messageIdsScheduledForMarkAsSeen.formUnion(messageIds) + strongSelf.messageIdsScheduledForMarkAsSeen.formUnion(messageIds.map(\.messageId)) } } } @@ -849,9 +862,9 @@ public final class ChatHistoryListNodeImpl: ListView, ChatHistoryNode, ChatHisto return } if strongSelf.canReadHistoryValue && !strongSelf.suspendReadingReactions && !strongSelf.context.sharedContext.immediateExperimentalUISettings.skipReadHistory { - strongSelf.context.account.viewTracker.updateMarkReactionsSeenForMessageIds(messageIds: messageIds) + strongSelf.context.account.viewTracker.updateMarkReactionsSeenForMessageIds(messageIds: Set(messageIds.map(\.messageId))) } else { - strongSelf.messageIdsWithReactionsScheduledForMarkAsSeen.formUnion(messageIds) + strongSelf.messageIdsWithReactionsScheduledForMarkAsSeen.formUnion(messageIds.map(\.messageId)) } } @@ -859,7 +872,7 @@ public final class ChatHistoryListNodeImpl: ListView, ChatHistoryNode, ChatHisto guard let strongSelf = self else { return } - strongSelf.context.account.viewTracker.updatedExtendedMediaForMessageIds(messageIds: messageIds) + strongSelf.context.account.viewTracker.updatedExtendedMediaForMessageIds(messageIds: Set(messageIds.map(\.messageId))) } self.preloadPages = false @@ -1286,6 +1299,68 @@ public final class ChatHistoryListNodeImpl: ListView, ChatHistoryNode, ChatHisto return (ChatHistoryViewUpdate.HistoryView(view: MessageHistoryView(tag: nil, namespaces: .all, entries: messages.reversed().map { MessageHistoryEntry(message: $0, isRead: false, location: nil, monthLocation: nil, attributes: MutableMessageHistoryEntryAttributes(authorIsContact: false)) }, holeEarlier: hasMore, holeLater: false, isLoading: false), type: .Generic(type: version > 0 ? ViewUpdateType.Generic : ViewUpdateType.Initial), scrollPosition: scrollPosition, flashIndicators: false, originalScrollPosition: nil, initialData: ChatHistoryCombinedInitialData(initialData: nil, buttonKeyboardMessage: nil, cachedData: nil, cachedDataMessages: nil, readStateData: nil), id: 0), version, nil, nil) } + } else if case let .customView(historyView) = self.source { + historyViewUpdate = combineLatest(queue: .mainQueue(), + self.chatHistoryLocationPromise.get(), + self.ignoreMessagesInTimestampRangePromise.get() + ) + |> distinctUntilChanged(isEqual: { lhs, rhs in + if lhs.0 != rhs.0 { + return false + } + if lhs.1 != rhs.1 { + return false + } + return true + }) + |> mapToSignal { location, _ -> Signal<((MessageHistoryView, ViewUpdateType), ChatHistoryLocationInput?), NoError> in + return historyView + |> map { historyView in + return (historyView, location) + } + } + |> map { viewAndUpdate, location in + let (view, update) = viewAndUpdate + + let version = currentViewVersion.modify({ value in + if let value = value { + return value + 1 + } else { + return 0 + } + })! + + var scrollPositionValue: ChatHistoryViewScrollPosition? + if let location { + switch location.content { + case let .Scroll(subject, _, _, scrollPosition, animated, highlight): + scrollPositionValue = .index(subject: subject, position: scrollPosition, directionHint: .Up, animated: animated, highlight: highlight, displayLink: false) + default: + break + } + } + + return ( + ChatHistoryViewUpdate.HistoryView( + view: view, + type: .Generic(type: update), + scrollPosition: scrollPositionValue, + flashIndicators: false, + originalScrollPosition: nil, + initialData: ChatHistoryCombinedInitialData( + initialData: nil, + buttonKeyboardMessage: nil, + cachedData: nil, + cachedDataMessages: nil, + readStateData: nil + ), + id: location?.id ?? 0 + ), + version, + location, + nil + ) + } } else { historyViewUpdate = combineLatest(queue: .mainQueue(), self.chatHistoryLocationPromise.get(), @@ -2043,10 +2118,12 @@ public final class ChatHistoryListNodeImpl: ListView, ChatHistoryNode, ChatHisto } if apply { switch strongSelf.chatLocation { - case .peer, .replyThread, .feed: + case .peer, .replyThread: if !strongSelf.context.sharedContext.immediateExperimentalUISettings.skipReadHistory { strongSelf.context.applyMaxReadIndex(for: strongSelf.chatLocation, contextHolder: strongSelf.chatLocationContextHolder, messageIndex: messageIndex) } + case .customChatContents: + break } } }).strict()) @@ -2432,7 +2509,7 @@ public final class ChatHistoryListNodeImpl: ListView, ChatHistoryNode, ChatHisto var messageIdsWithViewCount: [MessageId] = [] var messageIdsWithLiveLocation: [MessageId] = [] - var messageIdsWithUnsupportedMedia: [MessageId] = [] + var messageIdsWithUnsupportedMedia: [MessageAndThreadId] = [] var messageIdsWithRefreshMedia: [MessageId] = [] var messageIdsWithRefreshStories: [MessageId] = [] var messageIdsWithUnseenPersonalMention: [MessageId] = [] @@ -2533,7 +2610,7 @@ public final class ChatHistoryListNodeImpl: ListView, ChatHistoryNode, ChatHisto } } if contentRequiredValidation { - messageIdsWithUnsupportedMedia.append(message.id) + messageIdsWithUnsupportedMedia.append(MessageAndThreadId(messageId: message.id, threadId: message.threadId)) } if mediaRequiredValidation { messageIdsWithRefreshMedia.append(message.id) @@ -2729,41 +2806,41 @@ public final class ChatHistoryListNodeImpl: ListView, ChatHistoryNode, ChatHisto } if !messageIdsWithViewCount.isEmpty { - self.messageProcessingManager.add(messageIdsWithViewCount) + self.messageProcessingManager.add(messageIdsWithViewCount.map { MessageAndThreadId(messageId: $0, threadId: nil) }) } if !messageIdsWithLiveLocation.isEmpty { - self.seenLiveLocationProcessingManager.add(messageIdsWithLiveLocation) + self.seenLiveLocationProcessingManager.add(messageIdsWithLiveLocation.map { MessageAndThreadId(messageId: $0, threadId: nil) }) } if !messageIdsWithUnsupportedMedia.isEmpty { self.unsupportedMessageProcessingManager.add(messageIdsWithUnsupportedMedia) } if !messageIdsWithRefreshMedia.isEmpty { - self.refreshMediaProcessingManager.add(messageIdsWithRefreshMedia) + self.refreshMediaProcessingManager.add(messageIdsWithRefreshMedia.map { MessageAndThreadId(messageId: $0, threadId: nil) }) } if !messageIdsWithRefreshStories.isEmpty { - self.refreshStoriesProcessingManager.add(messageIdsWithRefreshStories) + self.refreshStoriesProcessingManager.add(messageIdsWithRefreshStories.map { MessageAndThreadId(messageId: $0, threadId: nil) }) } if !messageIdsWithUnseenPersonalMention.isEmpty { - self.messageMentionProcessingManager.add(messageIdsWithUnseenPersonalMention) + self.messageMentionProcessingManager.add(messageIdsWithUnseenPersonalMention.map { MessageAndThreadId(messageId: $0, threadId: nil) }) } if !messageIdsWithUnseenReactions.isEmpty { - self.unseenReactionsProcessingManager.add(messageIdsWithUnseenReactions) + self.unseenReactionsProcessingManager.add(messageIdsWithUnseenReactions.map { MessageAndThreadId(messageId: $0, threadId: nil) }) if self.canReadHistoryValue && !self.context.sharedContext.immediateExperimentalUISettings.skipReadHistory { let _ = self.displayUnseenReactionAnimations(messageIds: messageIdsWithUnseenReactions) } } if !messageIdsWithPossibleReactions.isEmpty { - self.messageWithReactionsProcessingManager.add(messageIdsWithPossibleReactions) + self.messageWithReactionsProcessingManager.add(messageIdsWithPossibleReactions.map { MessageAndThreadId(messageId: $0, threadId: nil) }) } if !downloadableResourceIds.isEmpty { let _ = markRecentDownloadItemsAsSeen(postbox: self.context.account.postbox, items: downloadableResourceIds).startStandalone() } if !messageIdsWithInactiveExtendedMedia.isEmpty { - self.extendedMediaProcessingManager.update(messageIdsWithInactiveExtendedMedia) + self.extendedMediaProcessingManager.update(Set(messageIdsWithInactiveExtendedMedia.map { MessageAndThreadId(messageId: $0, threadId: nil) })) } if !messageIdsToTranslate.isEmpty { - self.translationProcessingManager.add(messageIdsToTranslate) + self.translationProcessingManager.add(messageIdsToTranslate.map { MessageAndThreadId(messageId: $0, threadId: nil) }) } if !visibleAdOpaqueIds.isEmpty { for opaqueId in visibleAdOpaqueIds { @@ -2789,7 +2866,7 @@ public final class ChatHistoryListNodeImpl: ListView, ChatHistoryNode, ChatHisto switch self.chatLocation { case .peer: messageIndex = maxIncomingIndex - case .replyThread, .feed: + case .replyThread, .customChatContents: messageIndex = maxOverallIndex } @@ -3174,7 +3251,13 @@ public final class ChatHistoryListNodeImpl: ListView, ChatHistoryNode, ChatHisto } let isEmpty = transition.historyView.originalView.entries.isEmpty || loadState == .empty(.botInfo) - let historyState: ChatHistoryNodeHistoryState = .loaded(isEmpty: isEmpty) + + var hasReachedLimits = false + if case let .customChatContents(customChatContents) = self.subject, let messageLimit = customChatContents.messageLimit { + hasReachedLimits = transition.historyView.originalView.entries.count >= messageLimit + } + + let historyState: ChatHistoryNodeHistoryState = .loaded(isEmpty: isEmpty, hasReachedLimits: hasReachedLimits) if self.currentHistoryState != historyState { self.currentHistoryState = historyState self.historyState.set(historyState) @@ -3435,7 +3518,14 @@ public final class ChatHistoryListNodeImpl: ListView, ChatHistoryNode, ChatHisto strongSelf.historyView = transition.historyView let loadState: ChatHistoryNodeLoadState + var alwaysHasMessages = false if case .custom = strongSelf.source { + if case .customChatContents = strongSelf.chatLocation { + } else { + alwaysHasMessages = true + } + } + if alwaysHasMessages { loadState = .messages } else if let historyView = strongSelf.historyView { if historyView.filteredEntries.isEmpty { @@ -3545,7 +3635,7 @@ public final class ChatHistoryListNodeImpl: ListView, ChatHistoryNode, ChatHisto switch strongSelf.chatLocation { case .peer: messageIndex = incomingIndex - case .replyThread, .feed: + case .replyThread, .customChatContents: messageIndex = overallIndex } @@ -3566,7 +3656,11 @@ public final class ChatHistoryListNodeImpl: ListView, ChatHistoryNode, ChatHisto } strongSelf._cachedPeerDataAndMessages.set(.single((transition.cachedData, transition.cachedDataMessages))) let isEmpty = transition.historyView.originalView.entries.isEmpty || loadState == .empty(.botInfo) - let historyState: ChatHistoryNodeHistoryState = .loaded(isEmpty: isEmpty) + var hasReachedLimits = false + if case let .customChatContents(customChatContents) = strongSelf.subject, let messageLimit = customChatContents.messageLimit { + hasReachedLimits = transition.historyView.originalView.entries.count >= messageLimit + } + let historyState: ChatHistoryNodeHistoryState = .loaded(isEmpty: isEmpty, hasReachedLimits: hasReachedLimits) if strongSelf.currentHistoryState != historyState { strongSelf.currentHistoryState = historyState strongSelf.historyState.set(historyState) @@ -3650,7 +3744,7 @@ public final class ChatHistoryListNodeImpl: ListView, ChatHistoryNode, ChatHisto let visibleNewIncomingReactionMessageIds = strongSelf.displayUnseenReactionAnimations(messageIds: messageIds) if !visibleNewIncomingReactionMessageIds.isEmpty { - strongSelf.unseenReactionsProcessingManager.add(visibleNewIncomingReactionMessageIds) + strongSelf.unseenReactionsProcessingManager.add(visibleNewIncomingReactionMessageIds.map { MessageAndThreadId(messageId: $0, threadId: nil) }) } } @@ -4074,6 +4168,9 @@ public final class ChatHistoryListNodeImpl: ListView, ChatHistoryNode, ChatHisto if let messageItem = messageItem { let associatedData = messageItem.associatedData let wantTrButton = usetrButton() + + let disableFloatingDateHeaders = messageItem.disableDate + loop: for i in 0 ..< historyView.filteredEntries.count { switch historyView.filteredEntries[i] { case let .MessageEntry(message, presentationData, read, location, selection, attributes): @@ -4083,7 +4180,7 @@ public final class ChatHistoryListNodeImpl: ListView, ChatHistoryNode, ChatHisto switch self.mode { case .bubbles: // MARK: Nicegram, wantTrButton - item = ChatMessageItemImpl(presentationData: presentationData, context: self.context, chatLocation: self.chatLocation, associatedData: associatedData, controllerInteraction: self.controllerInteraction, content: .message(message: message, read: read, selection: selection, attributes: attributes, location: location), wantTrButton: wantTrButton) + item = ChatMessageItemImpl(presentationData: presentationData, context: self.context, chatLocation: self.chatLocation, associatedData: associatedData, controllerInteraction: self.controllerInteraction, content: .message(message: message, read: read, selection: selection, attributes: attributes, location: location), disableDate: disableFloatingDateHeaders, wantTrButton: wantTrButton) case let .list(_, _, _, displayHeaders, hintLinks, isGlobalSearch): let displayHeader: Bool switch displayHeaders { @@ -4130,6 +4227,7 @@ public final class ChatHistoryListNodeImpl: ListView, ChatHistoryNode, ChatHisto if let messageItem = messageItem { let associatedData = messageItem.associatedData + let disableFloatingDateHeaders = messageItem.disableDate loop: for i in 0 ..< historyView.filteredEntries.count { switch historyView.filteredEntries[i] { @@ -4139,7 +4237,7 @@ public final class ChatHistoryListNodeImpl: ListView, ChatHistoryNode, ChatHisto let item: ListViewItem switch self.mode { case .bubbles: - item = ChatMessageItemImpl(presentationData: presentationData, context: self.context, chatLocation: self.chatLocation, associatedData: associatedData, controllerInteraction: self.controllerInteraction, content: .message(message: message, read: read, selection: selection, attributes: attributes, location: location)) + item = ChatMessageItemImpl(presentationData: presentationData, context: self.context, chatLocation: self.chatLocation, associatedData: associatedData, controllerInteraction: self.controllerInteraction, content: .message(message: message, read: read, selection: selection, attributes: attributes, location: location), disableDate: disableFloatingDateHeaders) case let .list(_, _, _, displayHeaders, hintLinks, isGlobalSearch): let displayHeader: Bool switch displayHeaders { diff --git a/submodules/TelegramUI/Sources/ChatHistoryViewForLocation.swift b/submodules/TelegramUI/Sources/ChatHistoryViewForLocation.swift index 47c47ed6c3d..47f937eb456 100644 --- a/submodules/TelegramUI/Sources/ChatHistoryViewForLocation.swift +++ b/submodules/TelegramUI/Sources/ChatHistoryViewForLocation.swift @@ -333,7 +333,7 @@ private func extractAdditionalData(view: MessageHistoryView, chatLocation: ChatL readStateData[peerId] = ChatHistoryCombinedInitialReadStateData(unreadCount: readState.count, totalState: totalUnreadState, notificationSettings: notificationSettings) } } - case .replyThread, .feed: + case .replyThread, .customChatContents: break } default: diff --git a/submodules/TelegramUI/Sources/ChatInterfaceInputContextPanels.swift b/submodules/TelegramUI/Sources/ChatInterfaceInputContextPanels.swift index f9fd936ea06..60ad36b3211 100644 --- a/submodules/TelegramUI/Sources/ChatInterfaceInputContextPanels.swift +++ b/submodules/TelegramUI/Sources/ChatInterfaceInputContextPanels.swift @@ -15,7 +15,7 @@ private func inputQueryResultPriority(_ result: ChatPresentationInputQueryResult case let .mentions(items): return (2, !items.isEmpty) case let .commands(items): - return (3, !items.isEmpty) + return (3, !items.commands.isEmpty || items.hasShortcuts) case let .contextRequestResult(_, result): var nonEmpty = false if let result = result, !result.results.isEmpty { @@ -32,10 +32,6 @@ private func inputQueryResultPriority(_ result: ChatPresentationInputQueryResult } func inputContextPanelForChatPresentationIntefaceState(_ chatPresentationInterfaceState: ChatPresentationInterfaceState, context: AccountContext, currentPanel: ChatInputContextPanelNode?, controllerInteraction: ChatControllerInteraction, interfaceInteraction: ChatPanelInterfaceInteraction?, chatPresentationContext: ChatPresentationContext) -> ChatInputContextPanelNode? { - guard let _ = chatPresentationInterfaceState.renderedPeer?.peer else { - return nil - } - if chatPresentationInterfaceState.showCommands, let renderedPeer = chatPresentationInterfaceState.renderedPeer { if let currentPanel = currentPanel as? CommandMenuChatInputContextPanelNode { return currentPanel @@ -159,14 +155,14 @@ func inputContextPanelForChatPresentationIntefaceState(_ chatPresentationInterfa return nil } case let .commands(commands): - if !commands.isEmpty { + if !commands.commands.isEmpty || commands.hasShortcuts { if let currentPanel = currentPanel as? CommandChatInputContextPanelNode { - currentPanel.updateResults(commands) + currentPanel.updateResults(commands.commands, accountPeer: commands.accountPeer, hasShortcuts: commands.hasShortcuts, query: commands.query) return currentPanel } else { let panel = CommandChatInputContextPanelNode(context: context, theme: chatPresentationInterfaceState.theme, strings: chatPresentationInterfaceState.strings, fontSize: chatPresentationInterfaceState.fontSize, chatPresentationContext: controllerInteraction.presentationContext) panel.interfaceInteraction = interfaceInteraction - panel.updateResults(commands) + panel.updateResults(commands.commands, accountPeer: commands.accountPeer, hasShortcuts: commands.hasShortcuts, query: commands.query) return panel } } else { diff --git a/submodules/TelegramUI/Sources/ChatInterfaceStateAccessoryPanels.swift b/submodules/TelegramUI/Sources/ChatInterfaceStateAccessoryPanels.swift index fe7d3204546..ac201532fef 100644 --- a/submodules/TelegramUI/Sources/ChatInterfaceStateAccessoryPanels.swift +++ b/submodules/TelegramUI/Sources/ChatInterfaceStateAccessoryPanels.swift @@ -76,12 +76,21 @@ func accessoryPanelForChatPresentationIntefaceState(_ chatPresentationInterfaceS replyPanelNode.interfaceInteraction = interfaceInteraction replyPanelNode.updateThemeAndStrings(theme: chatPresentationInterfaceState.theme, strings: chatPresentationInterfaceState.strings) return replyPanelNode - } else if let peerId = chatPresentationInterfaceState.chatLocation.peerId { - let panelNode = ReplyAccessoryPanelNode(context: context, chatPeerId: peerId, messageId: replyMessageSubject.messageId, quote: replyMessageSubject.quote, theme: chatPresentationInterfaceState.theme, strings: chatPresentationInterfaceState.strings, nameDisplayOrder: chatPresentationInterfaceState.nameDisplayOrder, dateTimeFormat: chatPresentationInterfaceState.dateTimeFormat, animationCache: chatControllerInteraction?.presentationContext.animationCache, animationRenderer: chatControllerInteraction?.presentationContext.animationRenderer) - panelNode.interfaceInteraction = interfaceInteraction - return panelNode } else { - return nil + var chatPeerId: EnginePeer.Id? + if let peerId = chatPresentationInterfaceState.chatLocation.peerId { + chatPeerId = peerId + } else if case .customChatContents = chatPresentationInterfaceState.chatLocation { + chatPeerId = context.account.peerId + } + + if let chatPeerId { + let panelNode = ReplyAccessoryPanelNode(context: context, chatPeerId: chatPeerId, messageId: replyMessageSubject.messageId, quote: replyMessageSubject.quote, theme: chatPresentationInterfaceState.theme, strings: chatPresentationInterfaceState.strings, nameDisplayOrder: chatPresentationInterfaceState.nameDisplayOrder, dateTimeFormat: chatPresentationInterfaceState.dateTimeFormat, animationCache: chatControllerInteraction?.presentationContext.animationCache, animationRenderer: chatControllerInteraction?.presentationContext.animationRenderer) + panelNode.interfaceInteraction = interfaceInteraction + return panelNode + } else { + return nil + } } } else { return nil diff --git a/submodules/TelegramUI/Sources/ChatInterfaceStateContextMenus.swift b/submodules/TelegramUI/Sources/ChatInterfaceStateContextMenus.swift index 52e43d8871a..f2935d90359 100644 --- a/submodules/TelegramUI/Sources/ChatInterfaceStateContextMenus.swift +++ b/submodules/TelegramUI/Sources/ChatInterfaceStateContextMenus.swift @@ -90,6 +90,8 @@ private func canEditMessage(accountPeerId: PeerId, limitsConfiguration: EngineCo } else { hasEditRights = true } + } else if message.id.namespace == Namespaces.Message.QuickReplyCloud { + hasEditRights = true } else if message.id.peerId.namespace == Namespaces.Peer.SecretChat || message.id.namespace != Namespaces.Message.Cloud { hasEditRights = false } else if let author = message.author, author.id == accountPeerId, let peer = message.peers[message.id.peerId] { @@ -293,6 +295,10 @@ private func canViewReadStats(message: Message, participantCount: Int?, isMessag } func canReplyInChat(_ chatPresentationInterfaceState: ChatPresentationInterfaceState, accountPeerId: PeerId) -> Bool { + if case .customChatContents = chatPresentationInterfaceState.chatLocation { + return true + } + guard let peer = chatPresentationInterfaceState.renderedPeer?.peer else { return false } @@ -359,8 +365,8 @@ func canReplyInChat(_ chatPresentationInterfaceState: ChatPresentationInterfaceS } case .replyThread: canReply = true - case .feed: - canReply = false + case .customChatContents: + canReply = true } return canReply } @@ -671,12 +677,12 @@ func contextMenuForChatPresentationInterfaceState(chatPresentationInterfaceState } } - if Namespaces.Message.allScheduled.contains(message.id.namespace) || message.id.peerId.isReplies { + if Namespaces.Message.allNonRegular.contains(message.id.namespace) || message.id.peerId.isReplies { canReply = false canPin = false } else if messages[0].flags.intersection([.Failed, .Unsent]).isEmpty { switch chatPresentationInterfaceState.chatLocation { - case .peer, .replyThread, .feed: + case .peer, .replyThread, .customChatContents: if let channel = messages[0].peers[messages[0].id.peerId] as? TelegramChannel { if !isAction { canPin = channel.hasPermission(.pinMessages) @@ -1134,8 +1140,6 @@ func contextMenuForChatPresentationInterfaceState(chatPresentationInterfaceState }))) } - - var messageText: String = "" for message in messages { if !message.text.isEmpty { @@ -1158,6 +1162,16 @@ func contextMenuForChatPresentationInterfaceState(chatPresentationInterfaceState } } + for attribute in message.attributes { + if hasExpandedAudioTranscription, let attribute = attribute as? AudioTranscriptionMessageAttribute { + if !messageText.isEmpty { + messageText.append("\n") + } + messageText.append(attribute.text) + break + } + } + var isPoll = false if messageText.isEmpty { for media in message.media { @@ -2229,6 +2243,76 @@ func contextMenuForChatPresentationInterfaceState(chatPresentationInterfaceState actions.removeFirst() } + if let message = messages.first, case let .customChatContents(customChatContents) = chatPresentationInterfaceState.subject { + actions.removeAll() + + switch customChatContents.kind { + case .quickReplyMessageInput: + if !messageText.isEmpty || (resourceAvailable && isImage) || diceEmoji != nil { + actions.append(.action(ContextMenuActionItem(text: chatPresentationInterfaceState.strings.Conversation_ContextMenuCopy, icon: { theme in + return generateTintedImage(image: UIImage(bundleImageName: "Chat/Context Menu/Copy"), color: theme.actionSheet.primaryTextColor) + }, action: { _, f in + var messageEntities: [MessageTextEntity]? + var restrictedText: String? + for attribute in message.attributes { + if let attribute = attribute as? TextEntitiesMessageAttribute { + messageEntities = attribute.entities + } + if let attribute = attribute as? RestrictedContentMessageAttribute { + restrictedText = attribute.platformText(platform: "ios", contentSettings: context.currentContentSettings.with { $0 }) ?? "" + } + } + + if let restrictedText = restrictedText { + storeMessageTextInPasteboard(restrictedText, entities: nil) + } else { + if let translationState = chatPresentationInterfaceState.translationState, translationState.isEnabled, + let translation = message.attributes.first(where: { ($0 as? TranslationMessageAttribute)?.toLang == translationState.toLang }) as? TranslationMessageAttribute, !translation.text.isEmpty { + storeMessageTextInPasteboard(translation.text, entities: translation.entities) + } else { + storeMessageTextInPasteboard(message.text, entities: messageEntities) + } + } + + Queue.mainQueue().after(0.2, { + let content: UndoOverlayContent = .copy(text: chatPresentationInterfaceState.strings.Conversation_MessageCopied) + controllerInteraction.displayUndo(content) + }) + + f(.default) + }))) + } + + if message.id.namespace == Namespaces.Message.QuickReplyCloud { + if data.canEdit { + 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 + interfaceInteraction.setupEditMessage(messages[0].id, { transition in + f(.custom(transition)) + }) + }))) + } + } + + if message.id.id < Int32.max - 1000 { + if !actions.isEmpty { + actions.append(.separator) + } + actions.append(.action(ContextMenuActionItem(text: chatPresentationInterfaceState.strings.Conversation_ContextMenuDelete, textColor: .destructive, icon: { theme in + return generateTintedImage(image: UIImage(bundleImageName: "Chat/Context Menu/Delete"), color: theme.actionSheet.destructiveActionTextColor) + }, action: { [weak customChatContents] _, f in + f(.dismissWithoutContent) + + guard let customChatContents else { + return + } + customChatContents.deleteMessages(ids: messages.map(\.id)) + }))) + } + } + } + return ContextController.Items(content: .list(actions), tip: nil) } } diff --git a/submodules/TelegramUI/Sources/ChatInterfaceStateContextQueries.swift b/submodules/TelegramUI/Sources/ChatInterfaceStateContextQueries.swift index d55673ba542..521035310f6 100644 --- a/submodules/TelegramUI/Sources/ChatInterfaceStateContextQueries.swift +++ b/submodules/TelegramUI/Sources/ChatInterfaceStateContextQueries.swift @@ -18,9 +18,6 @@ import ChatPresentationInterfaceState import ChatContextQuery func contextQueryResultStateForChatInterfacePresentationState(_ chatPresentationInterfaceState: ChatPresentationInterfaceState, context: AccountContext, currentQueryStates: inout [ChatPresentationInputQueryKind: (ChatPresentationInputQuery, Disposable)], requestBotLocationStatus: @escaping (PeerId) -> Void) -> [ChatPresentationInputQueryKind: ChatContextQueryUpdate] { - guard let peer = chatPresentationInterfaceState.renderedPeer?.peer else { - return [:] - } let inputQueries = inputContextQueriesForChatPresentationIntefaceState(chatPresentationInterfaceState).filter({ query in if chatPresentationInterfaceState.editMessageState != nil { switch query { @@ -39,7 +36,7 @@ func contextQueryResultStateForChatInterfacePresentationState(_ chatPresentation for query in inputQueries { let previousQuery = currentQueryStates[query.kind]?.0 if previousQuery != query { - let signal = updatedContextQueryResultStateForQuery(context: context, peer: peer, chatLocation: chatPresentationInterfaceState.chatLocation, inputQuery: query, previousQuery: previousQuery, requestBotLocationStatus: requestBotLocationStatus) + let signal = updatedContextQueryResultStateForQuery(context: context, peer: chatPresentationInterfaceState.renderedPeer?.peer, chatLocation: chatPresentationInterfaceState.chatLocation, inputQuery: query, previousQuery: previousQuery, requestBotLocationStatus: requestBotLocationStatus) updates[query.kind] = .update(query, signal) } } @@ -60,7 +57,7 @@ func contextQueryResultStateForChatInterfacePresentationState(_ chatPresentation return updates } -private func updatedContextQueryResultStateForQuery(context: AccountContext, peer: Peer, chatLocation: ChatLocation, inputQuery: ChatPresentationInputQuery, previousQuery: ChatPresentationInputQuery?, requestBotLocationStatus: @escaping (PeerId) -> Void) -> Signal<(ChatPresentationInputQueryResult?) -> ChatPresentationInputQueryResult?, ChatContextQueryError> { +private func updatedContextQueryResultStateForQuery(context: AccountContext, peer: Peer?, chatLocation: ChatLocation, inputQuery: ChatPresentationInputQuery, previousQuery: ChatPresentationInputQuery?, requestBotLocationStatus: @escaping (PeerId) -> Void) -> Signal<(ChatPresentationInputQueryResult?) -> ChatPresentationInputQueryResult?, ChatContextQueryError> { switch inputQuery { case let .emoji(query): var signal: Signal<(ChatPresentationInputQueryResult?) -> ChatPresentationInputQueryResult?, ChatContextQueryError> = .complete() @@ -182,74 +179,133 @@ private func updatedContextQueryResultStateForQuery(context: AccountContext, pee let inlineBots: Signal<[(EnginePeer, Double)], NoError> = types.contains(.contextBots) ? context.engine.peers.recentlyUsedInlineBots() : .single([]) let strings = context.sharedContext.currentPresentationData.with({ $0 }).strings - let participants = combineLatest(inlineBots, searchPeerMembers(context: context, peerId: peer.id, chatLocation: chatLocation, query: query, scope: .mention)) - |> map { inlineBots, peers -> (ChatPresentationInputQueryResult?) -> ChatPresentationInputQueryResult? in - let filteredInlineBots = inlineBots.sorted(by: { $0.1 > $1.1 }).filter { peer, rating in - if rating < 0.14 { + + let participants: Signal<(ChatPresentationInputQueryResult?) -> ChatPresentationInputQueryResult?, ChatContextQueryError> + if let peer { + participants = combineLatest( + inlineBots, + searchPeerMembers(context: context, peerId: peer.id, chatLocation: chatLocation, query: query, scope: .mention) + ) + |> map { inlineBots, peers -> (ChatPresentationInputQueryResult?) -> ChatPresentationInputQueryResult? in + let filteredInlineBots = inlineBots.sorted(by: { $0.1 > $1.1 }).filter { peer, rating in + if rating < 0.14 { + return false + } + if peer.indexName.matchesByTokens(normalizedQuery) { + return true + } + if let addressName = peer.addressName, addressName.lowercased().hasPrefix(normalizedQuery) { + return true + } return false - } - if peer.indexName.matchesByTokens(normalizedQuery) { - return true - } - if let addressName = peer.addressName, addressName.lowercased().hasPrefix(normalizedQuery) { + }.map { $0.0 } + + let inlineBotPeerIds = Set(filteredInlineBots.map { $0.id }) + + let filteredPeers = peers.filter { peer in + if inlineBotPeerIds.contains(peer.id) { + return false + } + if !types.contains(.accountPeer) && peer.id == context.account.peerId { + return false + } return true } - return false - }.map { $0.0 } - - let inlineBotPeerIds = Set(filteredInlineBots.map { $0.id }) - - let filteredPeers = peers.filter { peer in - if inlineBotPeerIds.contains(peer.id) { - return false - } - if !types.contains(.accountPeer) && peer.id == context.account.peerId { - return false + var sortedPeers = filteredInlineBots + sortedPeers.append(contentsOf: filteredPeers.sorted(by: { lhs, rhs in + let result = lhs.indexName.stringRepresentation(lastNameFirst: true).compare(rhs.indexName.stringRepresentation(lastNameFirst: true)) + return result == .orderedAscending + })) + sortedPeers = sortedPeers.filter { peer in + return !peer.displayTitle(strings: strings, displayOrder: .firstLast).isEmpty } - return true - } - var sortedPeers = filteredInlineBots - sortedPeers.append(contentsOf: filteredPeers.sorted(by: { lhs, rhs in - let result = lhs.indexName.stringRepresentation(lastNameFirst: true).compare(rhs.indexName.stringRepresentation(lastNameFirst: true)) - return result == .orderedAscending - })) - sortedPeers = sortedPeers.filter { peer in - return !peer.displayTitle(strings: strings, displayOrder: .firstLast).isEmpty + return { _ in return .mentions(sortedPeers) } } - return { _ in return .mentions(sortedPeers) } + |> castError(ChatContextQueryError.self) + } else { + participants = .single({ _ in return nil }) } - |> castError(ChatContextQueryError.self) return signal |> then(participants) case let .command(query): + guard let peer else { + return .single({ _ in return .commands(ChatInputQueryCommandsResult( + commands: [], + accountPeer: nil, + hasShortcuts: false, + query: "" + )) }) + } + let normalizedQuery = query.lowercased() var signal: Signal<(ChatPresentationInputQueryResult?) -> ChatPresentationInputQueryResult?, ChatContextQueryError> = .complete() if let previousQuery = previousQuery { switch previousQuery { - case .command: - break - default: - signal = .single({ _ in return .commands([]) }) + case .command: + break + default: + signal = .single({ _ in return .commands(ChatInputQueryCommandsResult( + commands: [], + accountPeer: nil, + hasShortcuts: false, + query: "" + )) }) } } else { - signal = .single({ _ in return .commands([]) }) + signal = .single({ _ in return .commands(ChatInputQueryCommandsResult( + commands: [], + accountPeer: nil, + hasShortcuts: false, + query: "" + )) }) + } + + var shortcuts: Signal<[ShortcutMessageList.Item], NoError> = .single([]) + if let user = peer as? TelegramUser, user.botInfo == nil { + context.account.viewTracker.keepQuickRepliesApproximatelyUpdated() + + shortcuts = context.engine.accountData.shortcutMessageList(onlyRemote: true) + |> map { shortcutMessageList -> [ShortcutMessageList.Item] in + return shortcutMessageList.items.filter { item in + return item.shortcut.hasPrefix(normalizedQuery) + } + } } - let commands = context.engine.peers.peerCommands(id: peer.id) - |> map { commands -> (ChatPresentationInputQueryResult?) -> ChatPresentationInputQueryResult? in + let commands = combineLatest( + context.engine.peers.peerCommands(id: peer.id), + shortcuts, + context.engine.data.subscribe( + TelegramEngine.EngineData.Item.Peer.Peer(id: context.account.peerId) + ) + ) + |> map { commands, shortcuts, accountPeer -> (ChatPresentationInputQueryResult?) -> ChatPresentationInputQueryResult? in let filteredCommands = commands.commands.filter { command in if command.command.text.hasPrefix(normalizedQuery) { return true } return false } - let sortedCommands = filteredCommands - return { _ in return .commands(sortedCommands) } + var sortedCommands = filteredCommands.map(ChatInputTextCommand.command) + if !shortcuts.isEmpty && sortedCommands.isEmpty { + for shortcut in shortcuts { + sortedCommands.append(.shortcut(shortcut)) + } + } + return { _ in return .commands(ChatInputQueryCommandsResult( + commands: sortedCommands, + accountPeer: accountPeer, + hasShortcuts: !shortcuts.isEmpty, + query: normalizedQuery + )) } } |> castError(ChatContextQueryError.self) return signal |> then(commands) case let .contextRequest(addressName, query): + guard let peer else { + return .single({ _ in return .contextRequestResult(nil, nil) }) + } var delayRequest = true var signal: Signal<(ChatPresentationInputQueryResult?) -> ChatPresentationInputQueryResult?, ChatContextQueryError> = .complete() if let previousQuery = previousQuery { diff --git a/submodules/TelegramUI/Sources/ChatInterfaceStateInputPanels.swift b/submodules/TelegramUI/Sources/ChatInterfaceStateInputPanels.swift index 893cc5ecca6..d5f3f2d97ef 100644 --- a/submodules/TelegramUI/Sources/ChatInterfaceStateInputPanels.swift +++ b/submodules/TelegramUI/Sources/ChatInterfaceStateInputPanels.swift @@ -377,7 +377,7 @@ func inputPanelForChatPresentationIntefaceState(_ chatPresentationInterfaceState if let user = chatPresentationInterfaceState.renderedPeer?.peer as? TelegramUser, user.botInfo != nil { displayBotStartPanel = true } - } else if let chatHistoryState = chatPresentationInterfaceState.chatHistoryState, case .loaded(true) = chatHistoryState { + } else if let chatHistoryState = chatPresentationInterfaceState.chatHistoryState, case .loaded(true, _) = chatHistoryState { if let user = chatPresentationInterfaceState.renderedPeer?.peer as? TelegramUser, user.botInfo != nil { displayBotStartPanel = true } @@ -410,6 +410,24 @@ func inputPanelForChatPresentationIntefaceState(_ chatPresentationInterfaceState } } + if case let .customChatContents(customChatContents) = chatPresentationInterfaceState.subject { + switch customChatContents.kind { + case .quickReplyMessageInput: + displayInputTextPanel = true + } + + if let chatHistoryState = chatPresentationInterfaceState.chatHistoryState, case .loaded(_, true) = chatHistoryState { + if let currentPanel = (currentPanel as? ChatRestrictedInputPanelNode) ?? (currentSecondaryPanel as? ChatRestrictedInputPanelNode) { + return (currentPanel, nil) + } else { + let panel = ChatRestrictedInputPanelNode() + panel.context = context + panel.interfaceInteraction = interfaceInteraction + return (panel, nil) + } + } + } + if case .inline = chatPresentationInterfaceState.mode { displayInputTextPanel = false } diff --git a/submodules/TelegramUI/Sources/ChatInterfaceStateNavigationButtons.swift b/submodules/TelegramUI/Sources/ChatInterfaceStateNavigationButtons.swift index a003c03e133..689cc49425a 100644 --- a/submodules/TelegramUI/Sources/ChatInterfaceStateNavigationButtons.swift +++ b/submodules/TelegramUI/Sources/ChatInterfaceStateNavigationButtons.swift @@ -54,6 +54,20 @@ func leftNavigationButtonForChatInterfaceState(_ presentationInterfaceState: Cha } } } + + if case let .customChatContents(customChatContents) = presentationInterfaceState.subject { + switch customChatContents.kind { + case .quickReplyMessageInput: + if let currentButton = currentButton, currentButton.action == .dismiss { + return currentButton + } else { + let buttonItem = UIBarButtonItem(title: strings.Common_Close, style: .plain, target: target, action: selector) + buttonItem.accessibilityLabel = strings.Common_Close + return ChatNavigationButton(action: .dismiss, buttonItem: buttonItem) + } + } + } + return nil } @@ -95,7 +109,7 @@ func rightNavigationButtonForChatInterfaceState(context: AccountContext, present var hasMessages = false if let chatHistoryState = presentationInterfaceState.chatHistoryState { - if case .loaded(false) = chatHistoryState { + if case .loaded(false, _) = chatHistoryState { hasMessages = true } } @@ -108,6 +122,24 @@ func rightNavigationButtonForChatInterfaceState(context: AccountContext, present return nil } + if case let .customChatContents(customChatContents) = presentationInterfaceState.subject { + switch customChatContents.kind { + case let .quickReplyMessageInput(_, shortcutType): + switch shortcutType { + case .generic: + if let currentButton = currentButton, currentButton.action == .edit { + return currentButton + } else { + let buttonItem = UIBarButtonItem(title: strings.Common_Edit, style: .plain, target: target, action: selector) + buttonItem.accessibilityLabel = strings.Common_Done + return ChatNavigationButton(action: .edit, buttonItem: buttonItem) + } + case .greeting, .away: + return nil + } + } + } + if case .replyThread = presentationInterfaceState.chatLocation { if let channel = presentationInterfaceState.renderedPeer?.peer as? TelegramChannel, channel.flags.contains(.isForum) { } else if hasMessages { diff --git a/submodules/TelegramUI/Sources/ChatMessageContextControllerContentSource.swift b/submodules/TelegramUI/Sources/ChatMessageContextControllerContentSource.swift index c14268f35a1..0bce425e9be 100644 --- a/submodules/TelegramUI/Sources/ChatMessageContextControllerContentSource.swift +++ b/submodules/TelegramUI/Sources/ChatMessageContextControllerContentSource.swift @@ -34,6 +34,7 @@ final class ChatMessageContextExtractedContentSource: ContextExtractedContentSou let blurBackground: Bool = true let centerVertically: Bool + private weak var chatController: ChatControllerImpl? private weak var chatNode: ChatControllerNode? private let engine: TelegramEngine private let message: Message @@ -43,6 +44,9 @@ final class ChatMessageContextExtractedContentSource: ContextExtractedContentSou if self.message.adAttribute != nil { return .single(false) } + if let chatController = self.chatController, case .customChatContents = chatController.subject { + return .single(false) + } return self.engine.data.subscribe(TelegramEngine.EngineData.Item.Messages.Message(id: self.message.id)) |> map { message -> Bool in @@ -55,7 +59,8 @@ final class ChatMessageContextExtractedContentSource: ContextExtractedContentSou |> distinctUntilChanged } - init(chatNode: ChatControllerNode, engine: TelegramEngine, message: Message, selectAll: Bool, centerVertically: Bool = false) { + init(chatController: ChatControllerImpl, chatNode: ChatControllerNode, engine: TelegramEngine, message: Message, selectAll: Bool, centerVertically: Bool = false) { + self.chatController = chatController self.chatNode = chatNode self.engine = engine self.message = message diff --git a/submodules/TelegramUI/Sources/ChatMessageThrottledProcessingManager.swift b/submodules/TelegramUI/Sources/ChatMessageThrottledProcessingManager.swift index a7eaa0f5b26..a29c5cb6f1a 100644 --- a/submodules/TelegramUI/Sources/ChatMessageThrottledProcessingManager.swift +++ b/submodules/TelegramUI/Sources/ChatMessageThrottledProcessingManager.swift @@ -4,30 +4,30 @@ import Postbox import SwiftSignalKit final class ChatMessageThrottledProcessingManager { - private let queue = Queue() + private let queue = Queue.mainQueue() private let delay: Double private let submitInterval: Double? - var process: ((Set) -> Void)? + var process: ((Set) -> Void)? private var timer: SwiftSignalKit.Timer? - private var processedList: [MessageId] = [] - private var processed: [MessageId: Double] = [:] - private var buffer = Set() + private var processedList: [MessageAndThreadId] = [] + private var processed: [MessageAndThreadId: Double] = [:] + private var buffer = Set() init(delay: Double = 1.0, submitInterval: Double? = nil) { self.delay = delay self.submitInterval = submitInterval } - func setProcess(process: @escaping (Set) -> Void) { + func setProcess(process: @escaping (Set) -> Void) { self.queue.async { self.process = process } } - func add(_ messageIds: [MessageId]) { + func add(_ messageIds: [MessageAndThreadId]) { self.queue.async { let timestamp = CFAbsoluteTimeGetCurrent() @@ -76,13 +76,13 @@ final class ChatMessageThrottledProcessingManager { final class ChatMessageVisibleThrottledProcessingManager { - private let queue = Queue() + private let queue = Queue.mainQueue() private let interval: Double - private var currentIds = Set() + private var currentIds = Set() - var process: ((Set) -> Void)? + var process: ((Set) -> Void)? private let timer: SwiftSignalKit.Timer @@ -107,13 +107,13 @@ final class ChatMessageVisibleThrottledProcessingManager { self.timer.invalidate() } - func setProcess(process: @escaping (Set) -> Void) { + func setProcess(process: @escaping (Set) -> Void) { self.queue.async { self.process = process } } - func update(_ ids: Set) { + func update(_ ids: Set) { self.queue.async { if self.currentIds != ids { self.currentIds = ids diff --git a/submodules/TelegramUI/Sources/ChatRestrictedInputPanelNode.swift b/submodules/TelegramUI/Sources/ChatRestrictedInputPanelNode.swift index 2325486c84d..265336934cf 100644 --- a/submodules/TelegramUI/Sources/ChatRestrictedInputPanelNode.swift +++ b/submodules/TelegramUI/Sources/ChatRestrictedInputPanelNode.swift @@ -91,16 +91,22 @@ final class ChatRestrictedInputPanelNode: ChatInputPanelNode { } else if personal { self.textNode.attributedText = NSAttributedString(string: interfaceState.strings.Conversation_RestrictedText, font: Font.regular(13.0), textColor: interfaceState.theme.chat.inputPanel.secondaryTextColor) } else { - if "".isEmpty { - //TODO:localize + if (self.presentationInterfaceState?.boostsToUnrestrict ?? 0) > 0 { iconSpacing = 0.0 iconImage = PresentationResourcesChat.chatPanelBoostIcon(interfaceState.theme) - self.textNode.attributedText = NSAttributedString(string: "Boost this group to send messages", font: Font.regular(15.0), textColor: interfaceState.theme.chat.inputPanel.panelControlAccentColor) + self.textNode.attributedText = NSAttributedString(string: interfaceState.strings.Conversation_BoostToUnrestrictText, font: Font.regular(15.0), textColor: interfaceState.theme.chat.inputPanel.panelControlAccentColor) isUserInteractionEnabled = true } else { self.textNode.attributedText = NSAttributedString(string: interfaceState.strings.Conversation_DefaultRestrictedText, font: Font.regular(13.0), textColor: interfaceState.theme.chat.inputPanel.secondaryTextColor) } } + } else if case let .customChatContents(customChatContents) = interfaceState.subject { + let displayCount: Int + switch customChatContents.kind { + case .quickReplyMessageInput: + displayCount = 20 + } + self.textNode.attributedText = NSAttributedString(string: interfaceState.strings.Chat_QuickReplyMessageLimitReachedText(Int32(displayCount)), font: Font.regular(13.0), textColor: interfaceState.theme.chat.inputPanel.secondaryTextColor) } self.buttonNode.isUserInteractionEnabled = isUserInteractionEnabled diff --git a/submodules/TelegramUI/Sources/ChatSearchNavigationContentNode.swift b/submodules/TelegramUI/Sources/ChatSearchNavigationContentNode.swift index 8ad90ba599f..73423965214 100644 --- a/submodules/TelegramUI/Sources/ChatSearchNavigationContentNode.swift +++ b/submodules/TelegramUI/Sources/ChatSearchNavigationContentNode.swift @@ -34,7 +34,7 @@ final class ChatSearchNavigationContentNode: NavigationBarContentNode { self.searchBar = SearchBarNode(theme: SearchBarNodeTheme(theme: theme, hasBackground: false, hasSeparator: false), strings: strings, fieldStyle: .modern) let placeholderText: String switch chatLocation { - case .peer, .replyThread, .feed: + case .peer, .replyThread, .customChatContents: if chatLocation.peerId == context.account.peerId, presentationInterfaceState.hasSearchTags { if case .standard(.embedded(false)) = presentationInterfaceState.mode { placeholderText = strings.Common_Search @@ -114,7 +114,7 @@ final class ChatSearchNavigationContentNode: NavigationBarContentNode { self.searchBar.prefixString = nil let placeholderText: String switch self.chatLocation { - case .peer, .replyThread, .feed: + case .peer, .replyThread, .customChatContents: if presentationInterfaceState.historyFilter != nil { placeholderText = self.strings.Common_Search } else if self.chatLocation.peerId == self.context.account.peerId, presentationInterfaceState.hasSearchTags { diff --git a/submodules/TelegramUI/Sources/ChatSearchResultsContollerNode.swift b/submodules/TelegramUI/Sources/ChatSearchResultsContollerNode.swift index 757c4e17b43..d55b4b900ba 100644 --- a/submodules/TelegramUI/Sources/ChatSearchResultsContollerNode.swift +++ b/submodules/TelegramUI/Sources/ChatSearchResultsContollerNode.swift @@ -289,6 +289,7 @@ class ChatSearchResultsControllerNode: ViewControllerTracingNode, UIScrollViewDe }, hideChatFolderUpdates: { }, openStories: { _, _ in }, dismissNotice: { _ in + }, editPeer: { _ in }) interaction.searchTextHighightState = searchQuery self.interaction = interaction diff --git a/submodules/TelegramUI/Sources/ChatTextInputPanelNode.swift b/submodules/TelegramUI/Sources/ChatTextInputPanelNode.swift index b8cc9b8c209..02d4a717607 100644 --- a/submodules/TelegramUI/Sources/ChatTextInputPanelNode.swift +++ b/submodules/TelegramUI/Sources/ChatTextInputPanelNode.swift @@ -1560,7 +1560,7 @@ class ChatTextInputPanelNode: ChatInputPanelNode, ASEditableTextNodeDelegate, Ch } else { if let user = interfaceState.renderedPeer?.peer as? TelegramUser, user.botInfo != nil { - if let chatHistoryState = interfaceState.chatHistoryState, case .loaded(true) = chatHistoryState { + if let chatHistoryState = interfaceState.chatHistoryState, case .loaded(true, _) = chatHistoryState { displayBotStartButton = true } else if interfaceState.peerIsBlocked { displayBotStartButton = true @@ -1844,37 +1844,57 @@ class ChatTextInputPanelNode: ChatInputPanelNode, ASEditableTextNodeDelegate, Ch let dismissedButtonMessageUpdated = interfaceState.interfaceState.messageActionsState.dismissedButtonKeyboardMessageId != previousState?.interfaceState.messageActionsState.dismissedButtonKeyboardMessageId let replyMessageUpdated = interfaceState.interfaceState.replyMessageSubject != previousState?.interfaceState.replyMessageSubject - if let peer = interfaceState.renderedPeer?.peer, previousState?.renderedPeer?.peer == nil || !peer.isEqual(previousState!.renderedPeer!.peer!) || previousState?.interfaceState.silentPosting != interfaceState.interfaceState.silentPosting || themeUpdated || !self.initializedPlaceholder || previousState?.keyboardButtonsMessage?.id != interfaceState.keyboardButtonsMessage?.id || previousState?.keyboardButtonsMessage?.visibleReplyMarkupPlaceholder != interfaceState.keyboardButtonsMessage?.visibleReplyMarkupPlaceholder || dismissedButtonMessageUpdated || replyMessageUpdated || (previousState?.interfaceState.editMessage == nil) != (interfaceState.interfaceState.editMessage == nil) || previousState?.forumTopicData != interfaceState.forumTopicData || previousState?.replyMessage?.id != interfaceState.replyMessage?.id { + var peerUpdated = false + if let peer = interfaceState.renderedPeer?.peer, previousState?.renderedPeer?.peer == nil || !peer.isEqual(previousState!.renderedPeer!.peer!) { + peerUpdated = true + } + + if peerUpdated || previousState?.interfaceState.silentPosting != interfaceState.interfaceState.silentPosting || themeUpdated || !self.initializedPlaceholder || previousState?.keyboardButtonsMessage?.id != interfaceState.keyboardButtonsMessage?.id || previousState?.keyboardButtonsMessage?.visibleReplyMarkupPlaceholder != interfaceState.keyboardButtonsMessage?.visibleReplyMarkupPlaceholder || dismissedButtonMessageUpdated || replyMessageUpdated || (previousState?.interfaceState.editMessage == nil) != (interfaceState.interfaceState.editMessage == nil) || previousState?.forumTopicData != interfaceState.forumTopicData || previousState?.replyMessage?.id != interfaceState.replyMessage?.id { self.initializedPlaceholder = true - var placeholder: String + var placeholder: String = "" - if let channel = peer as? TelegramChannel, case .broadcast = channel.info { - if interfaceState.interfaceState.silentPosting { - placeholder = interfaceState.strings.Conversation_InputTextSilentBroadcastPlaceholder - } else { - placeholder = interfaceState.strings.Conversation_InputTextBroadcastPlaceholder - } - } else { - if sendingTextDisabled { - placeholder = interfaceState.strings.Chat_PlaceholderTextNotAllowed + if let peer = interfaceState.renderedPeer?.peer { + if let channel = peer as? TelegramChannel, case .broadcast = channel.info { + if interfaceState.interfaceState.silentPosting { + placeholder = interfaceState.strings.Conversation_InputTextSilentBroadcastPlaceholder + } else { + placeholder = interfaceState.strings.Conversation_InputTextBroadcastPlaceholder + } } else { - if let channel = peer as? TelegramChannel, case .group = channel.info, channel.hasPermission(.canBeAnonymous) { - placeholder = interfaceState.strings.Conversation_InputTextAnonymousPlaceholder - } else if case let .replyThread(replyThreadMessage) = interfaceState.chatLocation, !replyThreadMessage.isForumPost, replyThreadMessage.peerId != self.context?.account.peerId { - if replyThreadMessage.isChannelPost { - placeholder = interfaceState.strings.Conversation_InputTextPlaceholderComment - } else { - placeholder = interfaceState.strings.Conversation_InputTextPlaceholderReply - } - } else if let channel = peer as? TelegramChannel, channel.isForum, let forumTopicData = interfaceState.forumTopicData { - if let replyMessage = interfaceState.replyMessage, let threadInfo = replyMessage.associatedThreadInfo { - placeholder = interfaceState.strings.Chat_InputPlaceholderReplyInTopic(threadInfo.title).string + if sendingTextDisabled { + placeholder = interfaceState.strings.Chat_PlaceholderTextNotAllowed + } else { + if let channel = peer as? TelegramChannel, case .group = channel.info, channel.hasPermission(.canBeAnonymous) { + placeholder = interfaceState.strings.Conversation_InputTextAnonymousPlaceholder + } else if case let .replyThread(replyThreadMessage) = interfaceState.chatLocation, !replyThreadMessage.isForumPost, replyThreadMessage.peerId != self.context?.account.peerId { + if replyThreadMessage.isChannelPost { + placeholder = interfaceState.strings.Conversation_InputTextPlaceholderComment + } else { + placeholder = interfaceState.strings.Conversation_InputTextPlaceholderReply + } + } else if let channel = peer as? TelegramChannel, channel.isForum, let forumTopicData = interfaceState.forumTopicData { + if let replyMessage = interfaceState.replyMessage, let threadInfo = replyMessage.associatedThreadInfo { + placeholder = interfaceState.strings.Chat_InputPlaceholderReplyInTopic(threadInfo.title).string + } else { + placeholder = interfaceState.strings.Chat_InputPlaceholderMessageInTopic(forumTopicData.title).string + } } else { - placeholder = interfaceState.strings.Chat_InputPlaceholderMessageInTopic(forumTopicData.title).string + placeholder = interfaceState.strings.Conversation_InputTextPlaceholder } - } else { - placeholder = interfaceState.strings.Conversation_InputTextPlaceholder + } + } + } + if case let .customChatContents(customChatContents) = interfaceState.subject { + switch customChatContents.kind { + case let .quickReplyMessageInput(_, shortcutType): + switch shortcutType { + case .generic: + placeholder = interfaceState.strings.Chat_Placeholder_QuickReply + case .greeting: + placeholder = interfaceState.strings.Chat_Placeholder_GreetingMessage + case .away: + placeholder = interfaceState.strings.Chat_Placeholder_AwayMessage } } } diff --git a/submodules/TelegramUI/Sources/CommandChatInputContextPanelNode.swift b/submodules/TelegramUI/Sources/CommandChatInputContextPanelNode.swift index 1e5fd7f7f6b..18b7c7bed8d 100644 --- a/submodules/TelegramUI/Sources/CommandChatInputContextPanelNode.swift +++ b/submodules/TelegramUI/Sources/CommandChatInputContextPanelNode.swift @@ -13,34 +13,230 @@ import ChatControllerInteraction import ItemListUI import ChatContextQuery import ChatInputContextPanelNode +import ChatListUI +import ComponentFlow +import ComponentDisplayAdapters -private struct CommandChatInputContextPanelEntryStableId: Hashable { - let command: PeerCommand +private enum CommandChatInputContextPanelEntryStableId: Hashable { + case editShortcuts + case command(PeerCommand) + case shortcut(Int32) } private struct CommandChatInputContextPanelEntry: Comparable, Identifiable { - let index: Int - let command: PeerCommand - let theme: PresentationTheme + struct Command: Equatable { + let command: ChatInputTextCommand + let accountPeer: EnginePeer? + let searchQuery: String? + + static func ==(lhs: Command, rhs: Command) -> Bool { + return lhs.command == rhs.command && lhs.accountPeer == rhs.accountPeer && lhs.searchQuery == rhs.searchQuery + } + } - var stableId: CommandChatInputContextPanelEntryStableId { - return CommandChatInputContextPanelEntryStableId(command: self.command) + enum Content: Equatable { + case editShortcuts + case command(Command) } - func withUpdatedTheme(_ theme: PresentationTheme) -> CommandChatInputContextPanelEntry { - return CommandChatInputContextPanelEntry(index: self.index, command: self.command, theme: theme) + let content: Content + let index: Int + let theme: PresentationTheme + + init(index: Int, content: Content, theme: PresentationTheme) { + self.content = content + self.index = index + self.theme = theme } static func ==(lhs: CommandChatInputContextPanelEntry, rhs: CommandChatInputContextPanelEntry) -> Bool { - return lhs.index == rhs.index && lhs.command == rhs.command && lhs.theme === rhs.theme + return lhs.index == rhs.index && lhs.content == rhs.content && lhs.theme === rhs.theme } static func <(lhs: CommandChatInputContextPanelEntry, rhs: CommandChatInputContextPanelEntry) -> Bool { return lhs.index < rhs.index } - func item(context: AccountContext, presentationData: PresentationData, commandSelected: @escaping (PeerCommand, Bool) -> Void) -> ListViewItem { - return CommandChatInputPanelItem(context: context, presentationData: ItemListPresentationData(presentationData), command: self.command, commandSelected: commandSelected) + var stableId: CommandChatInputContextPanelEntryStableId { + switch self.content { + case .editShortcuts: + return .editShortcuts + case let .command(command): + switch command.command { + case let .command(command): + return .command(command) + case let .shortcut(shortcut): + if let shortcutId = shortcut.id { + return .shortcut(shortcutId) + } else { + return .shortcut(0) + } + } + } + } + + func withUpdatedTheme(_ theme: PresentationTheme) -> CommandChatInputContextPanelEntry { + return CommandChatInputContextPanelEntry(index: self.index, content: self.content, theme: theme) + } + + func item(context: AccountContext, presentationData: PresentationData, commandSelected: @escaping (ChatInputTextCommand, Bool) -> Void, openEditShortcuts: @escaping () -> Void) -> ListViewItem { + switch self.content { + case .editShortcuts: + return VerticalListContextResultsChatInputPanelButtonItem(theme: presentationData.theme, style: .round, title: presentationData.strings.Chat_CommandList_EditQuickReplies, pressed: { + openEditShortcuts() + }) + case let .command(command): + switch command.command { + case let .command(command): + return CommandChatInputPanelItem(context: context, presentationData: ItemListPresentationData(presentationData), command: command, commandSelected: { value, sendImmediately in + commandSelected(.command(value), sendImmediately) + }) + case let .shortcut(shortcut): + let chatListNodeInteraction = ChatListNodeInteraction( + context: context, + animationCache: context.animationCache, + animationRenderer: context.animationRenderer, + activateSearch: { + }, + peerSelected: { _, _, _, _ in + commandSelected(.shortcut(shortcut), true) + }, + disabledPeerSelected: { _, _, _ in + }, + togglePeerSelected: { _, _ in + }, + togglePeersSelection: { _, _ in + }, + additionalCategorySelected: { _ in + }, + messageSelected: { _, _, _, _ in + commandSelected(.shortcut(shortcut), true) + }, + groupSelected: { _ in + }, + addContact: { _ in + }, + setPeerIdWithRevealedOptions: { _, _ in + }, + setItemPinned: { _, _ in + }, + setPeerMuted: { _, _ in + }, + setPeerThreadMuted: { _, _, _ in + }, + deletePeer: { _, _ in + }, + deletePeerThread: { _, _ in + }, + setPeerThreadStopped: { _, _, _ in + }, + setPeerThreadPinned: { _, _, _ in + }, + setPeerThreadHidden: { _, _, _ in + }, + updatePeerGrouping: { _, _ in + }, + togglePeerMarkedUnread: { _, _ in + }, + toggleArchivedFolderHiddenByDefault: { + }, + toggleThreadsSelection: { _, _ in + }, + hidePsa: { _ in + }, + activateChatPreview: { _, _, _, _, _ in + }, + present: { _ in + }, + openForumThread: { _, _ in + }, + openStorageManagement: { + }, + openPasswordSetup: { + }, + openPremiumIntro: { + }, + openPremiumGift: { + }, + openActiveSessions: { + }, + performActiveSessionAction: { _, _ in + }, + openChatFolderUpdates: { + }, + hideChatFolderUpdates: { + }, + openStories: { _, _ in + }, + dismissNotice: { _ in + }, + editPeer: { _ in + } + ) + + let chatListPresentationData = ChatListPresentationData( + theme: presentationData.theme, + fontSize: presentationData.listsFontSize, + strings: presentationData.strings, + dateTimeFormat: presentationData.dateTimeFormat, + nameSortOrder: presentationData.nameSortOrder, + nameDisplayOrder: presentationData.nameDisplayOrder, + disableAnimations: false + ) + + let renderedPeer: EngineRenderedPeer + if let accountPeer = command.accountPeer { + renderedPeer = EngineRenderedPeer(peer: accountPeer) + } else { + renderedPeer = EngineRenderedPeer(peerId: context.account.peerId, peers: [:], associatedMedia: [:]) + } + + return ChatListItem( + presentationData: chatListPresentationData, + context: context, + chatListLocation: .chatList(groupId: .root), + filterData: nil, + index: EngineChatList.Item.Index.chatList(ChatListIndex(pinningIndex: nil, messageIndex: MessageIndex(id: MessageId(peerId: context.account.peerId, namespace: 0, id: 0), timestamp: 0))), + content: .peer(ChatListItemContent.PeerData( + messages: [shortcut.topMessage], + peer: renderedPeer, + threadInfo: nil, + combinedReadState: nil, + isRemovedFromTotalUnreadCount: false, + presence: nil, + hasUnseenMentions: false, + hasUnseenReactions: false, + draftState: nil, + mediaDraftContentType: nil, + inputActivities: nil, + promoInfo: nil, + ignoreUnreadBadge: false, + displayAsMessage: false, + hasFailedMessages: false, + forumTopicData: nil, + topForumTopicItems: [], + autoremoveTimeout: nil, + storyState: nil, + requiresPremiumForMessaging: false, + displayAsTopicList: false, + tags: [], + customMessageListData: ChatListItemContent.CustomMessageListData( + commandPrefix: "/\(shortcut.shortcut)", + searchQuery: command.searchQuery.flatMap { "/\($0)"}, + messageCount: shortcut.totalCount, + hideSeparator: false + ) + )), + editing: false, + hasActiveRevealControls: false, + selected: false, + header: nil, + enableContextActions: false, + hiddenOffset: false, + interaction: chatListNodeInteraction + ) + } + } } } @@ -48,21 +244,33 @@ private struct CommandChatInputContextPanelTransition { let deletions: [ListViewDeleteItem] let insertions: [ListViewInsertItem] let updates: [ListViewUpdateItem] + let hasShortcuts: Bool + let itemCountChanged: Bool } -private func preparedTransition(from fromEntries: [CommandChatInputContextPanelEntry], to toEntries: [CommandChatInputContextPanelEntry], context: AccountContext, presentationData: PresentationData, commandSelected: @escaping (PeerCommand, Bool) -> Void) -> CommandChatInputContextPanelTransition { +private func preparedTransition(from fromEntries: [CommandChatInputContextPanelEntry], to toEntries: [CommandChatInputContextPanelEntry], context: AccountContext, presentationData: PresentationData, commandSelected: @escaping (ChatInputTextCommand, Bool) -> Void, openEditShortcuts: @escaping () -> Void) -> CommandChatInputContextPanelTransition { 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(context: context, presentationData: presentationData, commandSelected: commandSelected), directionHint: nil) } - let updates = updateIndices.map { ListViewUpdateItem(index: $0.0, previousIndex: $0.2, item: $0.1.item(context: context, presentationData: presentationData, commandSelected: commandSelected), directionHint: nil) } + let insertions = indicesAndItems.map { ListViewInsertItem(index: $0.0, previousIndex: $0.2, item: $0.1.item(context: context, presentationData: presentationData, commandSelected: commandSelected, openEditShortcuts: openEditShortcuts), directionHint: nil) } + let updates = updateIndices.map { ListViewUpdateItem(index: $0.0, previousIndex: $0.2, item: $0.1.item(context: context, presentationData: presentationData, commandSelected: commandSelected, openEditShortcuts: openEditShortcuts), directionHint: nil) } - return CommandChatInputContextPanelTransition(deletions: deletions, insertions: insertions, updates: updates) + let itemCountChanged = fromEntries.count != toEntries.count + + return CommandChatInputContextPanelTransition(deletions: deletions, insertions: insertions, updates: updates, hasShortcuts: toEntries.contains(where: { entry in + if case .editShortcuts = entry.content { + return true + } + return false + }), itemCountChanged: itemCountChanged) } final class CommandChatInputContextPanelNode: ChatInputContextPanelNode { private let listView: ListView + private let listBackgroundView: UIView private var currentEntries: [CommandChatInputContextPanelEntry]? + private var contentOffsetChangeTransition: Transition? + private var isAnimatingOut: Bool = false private var enqueuedTransitions: [(CommandChatInputContextPanelTransition, Bool)] = [] private var validLayout: (CGSize, CGFloat, CGFloat, CGFloat)? @@ -78,20 +286,54 @@ final class CommandChatInputContextPanelNode: ChatInputContextPanelNode { return strings.VoiceOver_ScrollStatus(row, count).string } + self.listBackgroundView = UIView() + self.listBackgroundView.backgroundColor = theme.list.plainBackgroundColor + self.listBackgroundView.layer.maskedCorners = [.layerMinXMinYCorner, .layerMaxXMinYCorner] + self.listBackgroundView.layer.cornerRadius = 10.0 + super.init(context: context, theme: theme, strings: strings, fontSize: fontSize, chatPresentationContext: chatPresentationContext) self.isOpaque = false self.clipsToBounds = true + self.view.addSubview(self.listBackgroundView) self.addSubnode(self.listView) + + self.listView.visibleContentOffsetChanged = { [weak self] offset in + guard let self else { + return + } + + if self.isAnimatingOut { + return + } + + var topItemOffset: CGFloat = self.listView.bounds.height + var isFirst = true + self.listView.forEachItemNode { itemNode in + if isFirst { + isFirst = false + topItemOffset = itemNode.frame.minY + } + } + + let transition: Transition = self.contentOffsetChangeTransition ?? .immediate + transition.setFrame(view: self.listBackgroundView, frame: CGRect(origin: CGPoint(x: 0.0, y: topItemOffset), size: CGSize(width: self.listView.bounds.width, height: self.listView.bounds.height + 1000.0))) + } } - func updateResults(_ results: [PeerCommand]) { + func updateResults(_ results: [ChatInputTextCommand], accountPeer: EnginePeer?, hasShortcuts: Bool, query: String?) { var entries: [CommandChatInputContextPanelEntry] = [] var index = 0 var stableIds = Set() + if hasShortcuts { + let entry = CommandChatInputContextPanelEntry(index: index, content: .editShortcuts, theme: self.theme) + stableIds.insert(entry.stableId) + entries.append(entry) + index += 1 + } for command in results { - let entry = CommandChatInputContextPanelEntry(index: index, command: command, theme: self.theme) + let entry = CommandChatInputContextPanelEntry(index: index, content: .command(CommandChatInputContextPanelEntry.Command(command: command, accountPeer: accountPeer, searchQuery: query)), theme: self.theme) if stableIds.contains(entry.stableId) { continue } @@ -106,7 +348,11 @@ final class CommandChatInputContextPanelNode: ChatInputContextPanelNode { let firstTime = self.currentEntries == nil let presentationData = self.context.sharedContext.currentPresentationData.with { $0 } let transition = preparedTransition(from: from ?? [], to: to, context: self.context, presentationData: presentationData, commandSelected: { [weak self] command, sendImmediately in - if let strongSelf = self, let interfaceInteraction = strongSelf.interfaceInteraction { + guard let strongSelf = self, let interfaceInteraction = strongSelf.interfaceInteraction else { + return + } + switch command { + case let .command(command): if sendImmediately { interfaceInteraction.sendBotCommand(command.peer, "/" + command.command.text) } else { @@ -132,7 +378,16 @@ final class CommandChatInputContextPanelNode: ChatInputContextPanelNode { return (textInputState, inputMode) } } + case let .shortcut(shortcut): + if let shortcutId = shortcut.id { + interfaceInteraction.sendShortcut(shortcutId) + } + } + }, openEditShortcuts: { [weak self] in + guard let self, let interfaceInteraction = self.interfaceInteraction else { + return } + interfaceInteraction.openEditShortcuts() }) self.currentEntries = to self.enqueueTransition(transition, firstTime: firstTime) @@ -153,12 +408,20 @@ final class CommandChatInputContextPanelNode: ChatInputContextPanelNode { self.enqueuedTransitions.remove(at: 0) var options = ListViewDeleteAndInsertOptions() + options.insert(.Synchronous) + options.insert(.LowLatency) + options.insert(.PreferSynchronousResourceLoading) if firstTime { - //options.insert(.Synchronous) - //options.insert(.LowLatency) + self.contentOffsetChangeTransition = .immediate + + self.listBackgroundView.frame = CGRect(origin: CGPoint(x: 0.0, y: self.listView.bounds.height), size: CGSize(width: self.listView.bounds.width, height: self.listView.bounds.height + 1000.0)) } else { - options.insert(.AnimateTopItemPosition) - options.insert(.AnimateCrossfade) + if transition.itemCountChanged { + options.insert(.AnimateTopItemPosition) + options.insert(.AnimateCrossfade) + } + + self.contentOffsetChangeTransition = .spring(duration: 0.4) } var insets = UIEdgeInsets() @@ -178,19 +441,45 @@ final class CommandChatInputContextPanelNode: ChatInputContextPanelNode { } if let topItemOffset = topItemOffset { - let position = strongSelf.listView.layer.position - strongSelf.listView.position = CGPoint(x: position.x, y: position.y + (strongSelf.listView.bounds.size.height - topItemOffset)) - ContainedViewLayoutTransition.animated(duration: 0.3, curve: .spring).animateView { - strongSelf.listView.position = position - } + let transition: ContainedViewLayoutTransition = .animated(duration: 0.4, curve: .spring) + + transition.animatePositionAdditive(layer: strongSelf.listView.layer, offset: CGPoint(x: 0.0, y: strongSelf.listView.bounds.size.height - topItemOffset)) + transition.animatePositionAdditive(layer: strongSelf.listBackgroundView.layer, offset: CGPoint(x: 0.0, y: strongSelf.listView.bounds.size.height - topItemOffset)) } } }) + + self.contentOffsetChangeTransition = nil } } private func topInsetForLayout(size: CGSize) -> CGFloat { - let minimumItemHeights: CGFloat = floor(MentionChatInputPanelItemNode.itemHeight * 3.5) + var minimumItemHeights: CGFloat = 0.0 + if let currentEntries = self.currentEntries, !currentEntries.isEmpty { + let indexLimit = min(4, currentEntries.count - 1) + for i in 0 ... indexLimit { + var itemHeight: CGFloat + switch currentEntries[i].content { + case .editShortcuts: + itemHeight = VerticalListContextResultsChatInputPanelButtonItemNode.itemHeight(style: .round) + case let .command(command): + switch command.command { + case .command: + itemHeight = MentionChatInputPanelItemNode.itemHeight + case .shortcut: + itemHeight = 58.0 + } + } + if indexLimit >= 4 && i == indexLimit { + minimumItemHeights += floor(itemHeight * 0.5) + } else { + minimumItemHeights += itemHeight + } + } + } else { + minimumItemHeights = floor(MentionChatInputPanelItemNode.itemHeight * 3.5) + } + return max(size.height - minimumItemHeights, 0.0) } @@ -208,8 +497,12 @@ final class CommandChatInputContextPanelNode: ChatInputContextPanelNode { let (duration, curve) = listViewAnimationDurationAndCurve(transition: transition) let updateSizeAndInsets = ListViewUpdateSizeAndInsets(size: size, insets: insets, duration: duration, curve: curve) + self.contentOffsetChangeTransition = Transition(transition) + self.listView.transaction(deleteIndices: [], insertIndicesAndItems: [], updateIndicesAndItems: [], options: [.Synchronous, .LowLatency], scrollToItem: nil, updateSizeAndInsets: updateSizeAndInsets, stationaryItemRange: nil, updateOpaqueState: nil, completion: { _ in }) + self.contentOffsetChangeTransition = nil + if !hadValidLayout { while !self.enqueuedTransitions.isEmpty { self.dequeueTransition() @@ -226,6 +519,8 @@ final class CommandChatInputContextPanelNode: ChatInputContextPanelNode { } override func animateOut(completion: @escaping () -> Void) { + self.isAnimatingOut = true + var topItemOffset: CGFloat? self.listView.forEachItemNode { itemNode in if topItemOffset == nil { @@ -235,9 +530,12 @@ final class CommandChatInputContextPanelNode: ChatInputContextPanelNode { if let topItemOffset = topItemOffset { let position = self.listView.layer.position - self.listView.layer.animatePosition(from: position, to: CGPoint(x: position.x, y: position.y + (self.listView.bounds.size.height - topItemOffset)), duration: 0.3, timingFunction: kCAMediaTimingFunctionSpring, removeOnCompletion: false, completion: { _ in + let offset = (self.listView.bounds.size.height - topItemOffset) + self.listView.layer.animatePosition(from: position, to: CGPoint(x: position.x, y: position.y + offset), duration: 0.3, timingFunction: kCAMediaTimingFunctionSpring, removeOnCompletion: false, completion: { _ in completion() }) + self.listBackgroundView.layer.animatePosition(from: self.listBackgroundView.layer.position, to: CGPoint(x: self.listBackgroundView.layer.position.x, y: self.listBackgroundView.layer.position.y + offset), duration: 0.3, timingFunction: kCAMediaTimingFunctionSpring, removeOnCompletion: false, completion: { _ in + }) } else { completion() } diff --git a/submodules/TelegramUI/Sources/ContactMultiselectionControllerNode.swift b/submodules/TelegramUI/Sources/ContactMultiselectionControllerNode.swift index a26be36efb7..177d376c497 100644 --- a/submodules/TelegramUI/Sources/ContactMultiselectionControllerNode.swift +++ b/submodules/TelegramUI/Sources/ContactMultiselectionControllerNode.swift @@ -127,8 +127,23 @@ final class ContactMultiselectionControllerNode: ASDisplayNode { let additionalCategories = chatSelection.additionalCategories let chatListFilters = chatSelection.chatListFilters + var chatListFilter: ChatListFilter? + if chatSelection.onlyUsers { + chatListFilter = .filter(id: Int32.max, title: "", emoticon: nil, data: ChatListFilterData( + isShared: false, + hasSharedLinks: false, + categories: [.contacts, .nonContacts], + excludeMuted: false, + excludeRead: false, + excludeArchived: false, + includePeers: ChatListFilterIncludePeers(), + excludePeers: [], + color: nil + )) + } + placeholder = placeholderValue - let chatListNode = ChatListNode(context: context, location: .chatList(groupId: .root), previewing: false, fillPreloadItems: false, mode: .peers(filter: [.excludeSecretChats], isSelecting: true, additionalCategories: additionalCategories?.categories ?? [], chatListFilters: chatListFilters, displayAutoremoveTimeout: chatSelection.displayAutoremoveTimeout, displayPresence: chatSelection.displayPresence), isPeerEnabled: isPeerEnabled, theme: self.presentationData.theme, fontSize: self.presentationData.listsFontSize, strings: self.presentationData.strings, dateTimeFormat: self.presentationData.dateTimeFormat, nameSortOrder: self.presentationData.nameSortOrder, nameDisplayOrder: self.presentationData.nameDisplayOrder, animationCache: self.animationCache, animationRenderer: self.animationRenderer, disableAnimations: true, isInlineMode: false, autoSetReady: true, isMainTab: false) + let chatListNode = ChatListNode(context: context, location: .chatList(groupId: .root), chatListFilter: chatListFilter, previewing: false, fillPreloadItems: false, mode: .peers(filter: [.excludeSecretChats], isSelecting: true, additionalCategories: additionalCategories?.categories ?? [], chatListFilters: chatListFilters, displayAutoremoveTimeout: chatSelection.displayAutoremoveTimeout, displayPresence: chatSelection.displayPresence), isPeerEnabled: isPeerEnabled, theme: self.presentationData.theme, fontSize: self.presentationData.listsFontSize, strings: self.presentationData.strings, dateTimeFormat: self.presentationData.dateTimeFormat, nameSortOrder: self.presentationData.nameSortOrder, nameDisplayOrder: self.presentationData.nameDisplayOrder, animationCache: self.animationCache, animationRenderer: self.animationRenderer, disableAnimations: true, isInlineMode: false, autoSetReady: true, isMainTab: false) chatListNode.passthroughPeerSelection = true chatListNode.disabledPeerSelected = { peer, _, reason in attemptDisabledItemSelection?(peer, reason) @@ -237,6 +252,8 @@ final class ContactMultiselectionControllerNode: ASDisplayNode { var searchGroups = false var searchChannels = false var globalSearch = false + var displaySavedMessages = true + var filters = filters switch mode { case .groupCreation, .channelCreation: globalSearch = true @@ -245,15 +262,31 @@ final class ContactMultiselectionControllerNode: ASDisplayNode { searchGroups = searchGroupsValue searchChannels = searchChannelsValue globalSearch = true - case .chatSelection: - searchChatList = true - searchGroups = true - searchChannels = true + case let .chatSelection(chatSelection): + if chatSelection.onlyUsers { + searchChatList = true + searchGroups = false + searchChannels = false + displaySavedMessages = false + filters.append(.excludeSelf) + } else { + searchChatList = true + searchGroups = true + searchChannels = true + } globalSearch = false case .premiumGifting, .requestedUsersSelection: searchChatList = true } - let searchResultsNode = ContactListNode(context: context, presentation: .single(.search(signal: searchText.get(), searchChatList: searchChatList, searchDeviceContacts: false, searchGroups: searchGroups, searchChannels: searchChannels, globalSearch: globalSearch)), filters: filters, onlyWriteable: strongSelf.onlyWriteable, isPeerEnabled: strongSelf.isPeerEnabled, selectionState: selectionState, isSearch: true) + let searchResultsNode = ContactListNode(context: context, presentation: .single(.search(ContactListPresentation.Search( + signal: searchText.get(), + searchChatList: searchChatList, + searchDeviceContacts: false, + searchGroups: searchGroups, + searchChannels: searchChannels, + globalSearch: globalSearch, + displaySavedMessages: displaySavedMessages + ))), filters: filters, onlyWriteable: strongSelf.onlyWriteable, isPeerEnabled: strongSelf.isPeerEnabled, selectionState: selectionState, isSearch: true) searchResultsNode.openPeer = { peer, _ in self?.tokenListNode.setText("") self?.openPeer?(peer) diff --git a/submodules/TelegramUI/Sources/EditAccessoryPanelNode.swift b/submodules/TelegramUI/Sources/EditAccessoryPanelNode.swift index c6e029ff09b..b844464a372 100644 --- a/submodules/TelegramUI/Sources/EditAccessoryPanelNode.swift +++ b/submodules/TelegramUI/Sources/EditAccessoryPanelNode.swift @@ -284,7 +284,9 @@ final class EditAccessoryPanelNode: AccessoryPanelNode { } let titleString: String - if canEditMedia { + if let message, message.id.namespace == Namespaces.Message.QuickReplyCloud { + titleString = self.strings.Conversation_EditingQuickReplyPanelTitle + } else if canEditMedia { titleString = isPhoto ? self.strings.Conversation_EditingPhotoPanelTitle : self.strings.Conversation_EditingCaptionPanelTitle } else { titleString = self.strings.Conversation_EditingMessagePanelTitle diff --git a/submodules/TelegramUI/Sources/Nicegram/NGDeeplinkHandler.swift b/submodules/TelegramUI/Sources/Nicegram/NGDeeplinkHandler.swift index 81c909ed355..5381d7975cf 100644 --- a/submodules/TelegramUI/Sources/Nicegram/NGDeeplinkHandler.swift +++ b/submodules/TelegramUI/Sources/Nicegram/NGDeeplinkHandler.swift @@ -1,6 +1,7 @@ import Foundation import AccountContext import Display +import FeatAvatarGeneratorUI import FeatCardUI import FeatImagesHubUI import FeatPremiumUI @@ -63,6 +64,25 @@ class NGDeeplinkHandler { } else { return false } + case "avatarGenerator": + if #available(iOS 15.0, *) { + Task { @MainActor in + guard let topController = UIApplication.topViewController else { + return + } + AvatarGeneratorUIHelper().navigateToGenerationFlow( + from: topController + ) + } + } + return true + case "avatarMyGenerations": + if #available(iOS 15.0, *) { + Task { @MainActor in + AvatarGeneratorUIHelper().navigateToGenerator() + } + } + return true case "generateImage": return handleGenerateImage(url: url) case "nicegramPremium": @@ -100,7 +120,8 @@ class NGDeeplinkHandler { } return true default: - return false + showUpdateAppAlert() + return true } } } @@ -222,6 +243,34 @@ private extension NGDeeplinkHandler { } return false } + + func showUpdateAppAlert() { + let alert = UIAlertController( + title: "Update the app", + message: "Please update the app to use the newest features!", + preferredStyle: .alert + ) + + alert.addAction( + UIAlertAction( + title: "Close", + style: .cancel + ) + ) + + alert.addAction( + UIAlertAction( + title: "Update", + style: .default, + handler: { _ in + let urlOpener = CoreContainer.shared.urlOpener() + urlOpener.open(.appStoreAppUrl) + } + ) + ) + + UIApplication.topViewController?.present(alert, animated: true) + } } // MARK: - Helpers diff --git a/submodules/TelegramUI/Sources/OpenResolvedUrl.swift b/submodules/TelegramUI/Sources/OpenResolvedUrl.swift index 2a5d09ecf77..c820caf77ab 100644 --- a/submodules/TelegramUI/Sources/OpenResolvedUrl.swift +++ b/submodules/TelegramUI/Sources/OpenResolvedUrl.swift @@ -302,7 +302,7 @@ func openResolvedUrlImpl( }) dismissInput() case let .share(url, text, to): - let continueWithPeer: (PeerId) -> Void = { peerId in + let continueWithPeer: (PeerId, Int64?) -> Void = { peerId, threadId in let textInputState: ChatTextInputState? if let text = text, !text.isEmpty { if let url = url, !url.isEmpty { @@ -320,15 +320,42 @@ func openResolvedUrlImpl( textInputState = nil } + let updateControllers = { [weak navigationController] in + guard let navigationController else { + return + } + let chatController: Signal + if let threadId { + chatController = chatControllerForForumThreadImpl(context: context, peerId: peerId, threadId: threadId) + } else { + chatController = .single(ChatControllerImpl(context: context, chatLocation: .peer(id: peerId))) + } + + let _ = (chatController + |> deliverOnMainQueue).start(next: { [weak navigationController] chatController in + guard let navigationController else { + return + } + var controllers = navigationController.viewControllers.filter { controller in + if controller is PeerSelectionController { + return false + } + return true + } + controllers.append(chatController) + navigationController.setViewControllers(controllers, animated: true) + }) + } + if let textInputState = textInputState { - let _ = (ChatInterfaceState.update(engine: context.engine, peerId: peerId, threadId: nil, { currentState in + let _ = (ChatInterfaceState.update(engine: context.engine, peerId: peerId, threadId: threadId, { currentState in return currentState.withUpdatedComposeInputState(textInputState) }) |> deliverOnMainQueue).startStandalone(completed: { - navigationController?.pushViewController(ChatControllerImpl(context: context, chatLocation: .peer(id: peerId))) + updateControllers() }) } else { - navigationController?.pushViewController(ChatControllerImpl(context: context, chatLocation: .peer(id: peerId))) + updateControllers() } } @@ -344,7 +371,7 @@ func openResolvedUrlImpl( |> deliverOnMainQueue).startStandalone(next: { peer in if let peer = peer { context.sharedContext.applicationBindings.dismissNativeController() - continueWithPeer(peer.id) + continueWithPeer(peer.id, nil) } }) } else { @@ -352,7 +379,7 @@ func openResolvedUrlImpl( |> deliverOnMainQueue).startStandalone(next: { peer in if let peer = peer { context.sharedContext.applicationBindings.dismissNativeController() - continueWithPeer(peer.id) + continueWithPeer(peer.id, nil) } }) /*let query = to.trimmingCharacters(in: CharacterSet(charactersIn: "0123456789").inverted) @@ -377,13 +404,8 @@ func openResolvedUrlImpl( context.sharedContext.applicationBindings.dismissNativeController() } else { let controller = context.sharedContext.makePeerSelectionController(PeerSelectionControllerParams(context: context, filter: [.onlyWriteable, .excludeDisabled], selectForumThreads: true)) - controller.peerSelected = { [weak controller] peer, _ in - let peerId = peer.id - - if let strongController = controller { - strongController.dismiss() - continueWithPeer(peerId) - } + controller.peerSelected = { peer, threadId in + continueWithPeer(peer.id, threadId) } context.sharedContext.applicationBindings.dismissNativeController() navigationController?.pushViewController(controller) diff --git a/submodules/TelegramUI/Sources/OverlayAudioPlayerControllerNode.swift b/submodules/TelegramUI/Sources/OverlayAudioPlayerControllerNode.swift index ba27451adaf..e7be8e9ccdb 100644 --- a/submodules/TelegramUI/Sources/OverlayAudioPlayerControllerNode.swift +++ b/submodules/TelegramUI/Sources/OverlayAudioPlayerControllerNode.swift @@ -173,6 +173,7 @@ final class OverlayAudioPlayerControllerNode: ViewControllerTracingNode, UIGestu }, openPremiumStatusInfo: { _, _, _, _ in }, openRecommendedChannelContextMenu: { _, _, _ in }, openGroupBoostInfo: { _, _ in + }, openStickerEditor: { }, requestMessageUpdate: { _, _ in }, cancelInteractiveKeyboardGestures: { }, dismissTextInput: { diff --git a/submodules/TelegramUI/Sources/PeerMessagesMediaPlaylist.swift b/submodules/TelegramUI/Sources/PeerMessagesMediaPlaylist.swift index f0b44c24c07..6830e72ed31 100644 --- a/submodules/TelegramUI/Sources/PeerMessagesMediaPlaylist.swift +++ b/submodules/TelegramUI/Sources/PeerMessagesMediaPlaylist.swift @@ -549,8 +549,10 @@ final class PeerMessagesMediaPlaylist: SharedMediaPlaylist { let namespaces: MessageIdNamespaces if Namespaces.Message.allScheduled.contains(anchor.id.namespace) { namespaces = .just(Namespaces.Message.allScheduled) + } else if Namespaces.Message.allQuickReply.contains(anchor.id.namespace) { + namespaces = .just(Namespaces.Message.allQuickReply) } else { - namespaces = .not(Namespaces.Message.allScheduled) + namespaces = .not(Namespaces.Message.allNonRegular) } switch anchor { diff --git a/submodules/TelegramUI/Sources/PrefetchManager.swift b/submodules/TelegramUI/Sources/PrefetchManager.swift index 0fd47eb4482..43717a36b4e 100644 --- a/submodules/TelegramUI/Sources/PrefetchManager.swift +++ b/submodules/TelegramUI/Sources/PrefetchManager.swift @@ -174,17 +174,17 @@ private final class PrefetchManagerInnerImpl { if case .full = automaticDownload { if let image = media as? TelegramMediaImage { - context.fetchDisposable.set(messageMediaImageInteractiveFetched(fetchManager: self.fetchManager, messageId: mediaItem.media.index.id, messageReference: MessageReference(peer: mediaItem.media.peer, author: nil, id: mediaItem.media.index.id, timestamp: mediaItem.media.index.timestamp, incoming: true, secret: false), image: image, resource: resource._asResource(), userInitiated: false, priority: priority, storeToDownloadsPeerId: nil).startStrict()) + context.fetchDisposable.set(messageMediaImageInteractiveFetched(fetchManager: self.fetchManager, messageId: mediaItem.media.index.id, messageReference: MessageReference(peer: mediaItem.media.peer, author: nil, id: mediaItem.media.index.id, timestamp: mediaItem.media.index.timestamp, incoming: true, secret: false, threadId: nil), image: image, resource: resource._asResource(), userInitiated: false, priority: priority, storeToDownloadsPeerId: nil).startStrict()) } else if let _ = media as? TelegramMediaWebFile { //strongSelf.fetchDisposable.set(chatMessageWebFileInteractiveFetched(account: context.account, image: image).startStrict()) } else if let file = media as? TelegramMediaFile { // MARK: Nicegram downloading feature, accountContext added - let fetchSignal = messageMediaFileInteractiveFetched(fetchManager: self.fetchManager, messageId: mediaItem.media.index.id, messageReference: MessageReference(peer: mediaItem.media.peer, author: nil, id: mediaItem.media.index.id, timestamp: mediaItem.media.index.timestamp, incoming: true, secret: false), file: file, userInitiated: false, priority: priority, accountContext: nil) + let fetchSignal = messageMediaFileInteractiveFetched(fetchManager: self.fetchManager, messageId: mediaItem.media.index.id, messageReference: MessageReference(peer: mediaItem.media.peer, author: nil, id: mediaItem.media.index.id, timestamp: mediaItem.media.index.timestamp, incoming: true, secret: false, threadId: nil), file: file, userInitiated: false, priority: priority, accountContext: nil) context.fetchDisposable.set(fetchSignal.startStrict()) } } else if case .prefetch = automaticDownload, mediaItem.media.peer.id.namespace != Namespaces.Peer.SecretChat { if let file = media as? TelegramMediaFile, let _ = file.size { - context.fetchDisposable.set(preloadVideoResource(postbox: self.account.postbox, userLocation: .peer(mediaItem.media.index.id.peerId), userContentType: MediaResourceUserContentType(file: file), resourceReference: FileMediaReference.message(message: MessageReference(peer: mediaItem.media.peer, author: nil, id: mediaItem.media.index.id, timestamp: mediaItem.media.index.timestamp, incoming: true, secret: false), media: file).resourceReference(file.resource), duration: 4.0).startStrict()) + context.fetchDisposable.set(preloadVideoResource(postbox: self.account.postbox, userLocation: .peer(mediaItem.media.index.id.peerId), userContentType: MediaResourceUserContentType(file: file), resourceReference: FileMediaReference.message(message: MessageReference(peer: mediaItem.media.peer, author: nil, id: mediaItem.media.index.id, timestamp: mediaItem.media.index.timestamp, incoming: true, secret: false, threadId: nil), media: file).resourceReference(file.resource), duration: 4.0).startStrict()) } } } diff --git a/submodules/TelegramUI/Sources/SharedAccountContext.swift b/submodules/TelegramUI/Sources/SharedAccountContext.swift index f84bbe96159..b761456f892 100644 --- a/submodules/TelegramUI/Sources/SharedAccountContext.swift +++ b/submodules/TelegramUI/Sources/SharedAccountContext.swift @@ -52,8 +52,10 @@ import PeerInfoScreen import ChatQrCodeScreen import UndoUI import ChatMessageNotificationItem -import BusinessSetupScreen import ChatbotSetupScreen +import BusinessLocationSetupScreen +import BusinessHoursSetupScreen +import AutomaticBusinessMessageSetupScreen import NGCore import NGData @@ -1885,6 +1887,7 @@ public final class SharedAccountContextImpl: SharedAccountContext { }, openPremiumStatusInfo: { _, _, _, _ in }, openRecommendedChannelContextMenu: { _, _, _ in }, openGroupBoostInfo: { _, _ in + }, openStickerEditor: { }, requestMessageUpdate: { _, _ in }, cancelInteractiveKeyboardGestures: { }, dismissTextInput: { @@ -2013,12 +2016,44 @@ public final class SharedAccountContextImpl: SharedAccountContext { return archiveSettingsController(context: context) } + public func makeFilterSettingsController(context: AccountContext, modal: Bool, scrollToTags: Bool, dismissed: (() -> Void)?) -> ViewController { + return chatListFilterPresetListController(context: context, mode: modal ? .modal : .default, scrollToTags: scrollToTags, dismissed: dismissed) + } + public func makeBusinessSetupScreen(context: AccountContext) -> ViewController { - return BusinessSetupScreen(context: context) + return PremiumIntroScreen(context: context, mode: .business, source: .settings, modal: false, forceDark: false) + } + + public func makeChatbotSetupScreen(context: AccountContext, initialData: ChatbotSetupScreenInitialData) -> ViewController { + return ChatbotSetupScreen(context: context, initialData: initialData as! ChatbotSetupScreen.InitialData) + } + + public func makeChatbotSetupScreenInitialData(context: AccountContext) -> Signal { + return ChatbotSetupScreen.initialData(context: context) + } + + public func makeBusinessLocationSetupScreen(context: AccountContext, initialValue: TelegramBusinessLocation?, completion: @escaping (TelegramBusinessLocation?) -> Void) -> ViewController { + return BusinessLocationSetupScreen(context: context, initialValue: initialValue, completion: completion) + } + + public func makeBusinessHoursSetupScreen(context: AccountContext, initialValue: TelegramBusinessHours?, completion: @escaping (TelegramBusinessHours?) -> Void) -> ViewController { + return BusinessHoursSetupScreen(context: context, initialValue: initialValue, completion: completion) + } + + public func makeAutomaticBusinessMessageSetupScreen(context: AccountContext, initialData: AutomaticBusinessMessageSetupScreenInitialData, isAwayMode: Bool) -> ViewController { + return AutomaticBusinessMessageSetupScreen(context: context, initialData: initialData as! AutomaticBusinessMessageSetupScreen.InitialData, mode: isAwayMode ? .away : .greeting) + } + + public func makeAutomaticBusinessMessageSetupScreenInitialData(context: AccountContext) -> Signal { + return AutomaticBusinessMessageSetupScreen.initialData(context: context) } - public func makeChatbotSetupScreen(context: AccountContext) -> ViewController { - return ChatbotSetupScreen(context: context) + public func makeQuickReplySetupScreen(context: AccountContext, initialData: QuickReplySetupScreenInitialData) -> ViewController { + return QuickReplySetupScreen(context: context, initialData: initialData as! QuickReplySetupScreen.InitialData, mode: .manage) + } + + public func makeQuickReplySetupScreenInitialData(context: AccountContext) -> Signal { + return QuickReplySetupScreen.initialData(context: context) } public func makePremiumIntroController(context: AccountContext, source: PremiumIntroSource, forceDark: Bool, dismissed: (() -> Void)?) -> ViewController { @@ -2098,8 +2133,10 @@ public final class SharedAccountContextImpl: SharedAccountContext { mappedSource = .readTime case .messageTags: mappedSource = .messageTags + case .folderTags: + mappedSource = .folderTags } - let controller = PremiumIntroScreen(context: context, modal: modal, source: mappedSource, forceDark: forceDark) + let controller = PremiumIntroScreen(context: context, source: mappedSource, modal: modal, forceDark: forceDark) controller.wasDismissed = dismissed return controller } @@ -2147,6 +2184,10 @@ public final class SharedAccountContextImpl: SharedAccountContext { mappedSubject = .lastSeen case .messagePrivacy: mappedSubject = .messagePrivacy + case .folderTags: + mappedSubject = .folderTags + default: + mappedSubject = .doubleLimits } return PremiumDemoScreen(context: context, subject: mappedSubject, action: action) } diff --git a/submodules/TelegramUI/Sources/TelegramRootController.swift b/submodules/TelegramUI/Sources/TelegramRootController.swift index d0f14394d02..faf9ede9b22 100644 --- a/submodules/TelegramUI/Sources/TelegramRootController.swift +++ b/submodules/TelegramUI/Sources/TelegramRootController.swift @@ -42,6 +42,8 @@ import ImageCompression import TextFormat import MediaEditor import PeerInfoScreen +import PeerInfoStoryGridScreen +import ShareWithPeersScreen private class DetailsChatPlaceholderNode: ASDisplayNode, NavigationDetailsPlaceholderNode { private var presentationData: PresentationData @@ -508,6 +510,7 @@ public final class TelegramRootController: NavigationController, TelegramRootCon let controller = MediaEditorScreen( context: context, + mode: .storyEditor, subject: subject, customTarget: customTarget, transitionIn: transitionIn, @@ -663,6 +666,19 @@ public final class TelegramRootController: NavigationController, TelegramRootCon viewControllers.removeSubrange(range) self.setViewControllers(viewControllers, animated: false) } + } else if self.viewControllers.contains(where: { $0 is PeerInfoStoryGridScreen }) { + var viewControllers: [UIViewController] = [] + for i in (0 ..< self.viewControllers.count) { + let controller = self.viewControllers[i] + if i == 0 { + viewControllers.append(controller) + } else if controller is MediaEditorScreen { + viewControllers.append(controller) + } else if controller is ShareWithPeersScreen { + viewControllers.append(controller) + } + } + self.setViewControllers(viewControllers, animated: false) } } diff --git a/submodules/TelegramUI/Sources/VerticalListContextResultsChatInputContextPanelNode.swift b/submodules/TelegramUI/Sources/VerticalListContextResultsChatInputContextPanelNode.swift index d26978cc54c..e8dba3fc2f9 100644 --- a/submodules/TelegramUI/Sources/VerticalListContextResultsChatInputContextPanelNode.swift +++ b/submodules/TelegramUI/Sources/VerticalListContextResultsChatInputContextPanelNode.swift @@ -287,7 +287,7 @@ final class VerticalListContextResultsChatInputContextPanelNode: ChatInputContex private func topInsetForLayout(size: CGSize, hasSwitchPeer: Bool) -> CGFloat { var minimumItemHeights: CGFloat = floor(VerticalListContextResultsChatInputPanelItemNode.itemHeight * 3.5) if hasSwitchPeer { - minimumItemHeights += VerticalListContextResultsChatInputPanelButtonItemNode.itemHeight + minimumItemHeights += VerticalListContextResultsChatInputPanelButtonItemNode.itemHeight(style: .regular) } return max(size.height - minimumItemHeights, 0.0) diff --git a/submodules/TelegramUI/Sources/VerticalListContextResultsChatInputPanelButtonItem.swift b/submodules/TelegramUI/Sources/VerticalListContextResultsChatInputPanelButtonItem.swift index 89b9d2cb20a..51c63e48bb3 100644 --- a/submodules/TelegramUI/Sources/VerticalListContextResultsChatInputPanelButtonItem.swift +++ b/submodules/TelegramUI/Sources/VerticalListContextResultsChatInputPanelButtonItem.swift @@ -7,12 +7,19 @@ import SwiftSignalKit import TelegramPresentationData final class VerticalListContextResultsChatInputPanelButtonItem: ListViewItem { + enum Style { + case regular + case round + } + fileprivate let theme: PresentationTheme + fileprivate let style: Style fileprivate let title: String fileprivate let pressed: () -> Void - public init(theme: PresentationTheme, title: String, pressed: @escaping () -> Void) { + public init(theme: PresentationTheme, style: Style = .regular, title: String, pressed: @escaping () -> Void) { self.theme = theme + self.style = style self.title = title self.pressed = pressed } @@ -65,10 +72,15 @@ final class VerticalListContextResultsChatInputPanelButtonItem: ListViewItem { } } -private let titleFont = Font.regular(15.0) - final class VerticalListContextResultsChatInputPanelButtonItemNode: ListViewItemNode { - static let itemHeight: CGFloat = 32.0 + static func itemHeight(style: VerticalListContextResultsChatInputPanelButtonItem.Style) -> CGFloat { + switch style { + case .regular: + return 32.0 + case .round: + return 42.0 + } + } private let buttonNode: HighlightTrackingButtonNode private let titleNode: TextNode @@ -125,11 +137,19 @@ final class VerticalListContextResultsChatInputPanelButtonItemNode: ListViewItem let makeTitleLayout = TextNode.asyncLayout(self.titleNode) return { [weak self] item, params, mergedTop, mergedBottom in + let titleFont: UIFont + switch item.style { + case .regular: + titleFont = Font.regular(15.0) + case .round: + titleFont = Font.regular(17.0) + } + let titleString = NSAttributedString(string: item.title, font: titleFont, textColor: item.theme.list.itemAccentColor) let (titleLayout, titleApply) = makeTitleLayout(TextNodeLayoutArguments(attributedString: titleString, backgroundColor: nil, maximumNumberOfLines: 1, truncationType: .end, constrainedSize: CGSize(width: params.width - params.leftInset - params.rightInset - 16.0, height: 100.0), alignment: .natural, cutout: nil, insets: UIEdgeInsets())) - let nodeLayout = ListViewItemNodeLayout(contentSize: CGSize(width: params.width, height: VerticalListContextResultsChatInputPanelButtonItemNode.itemHeight), insets: UIEdgeInsets()) + let nodeLayout = ListViewItemNodeLayout(contentSize: CGSize(width: params.width, height: VerticalListContextResultsChatInputPanelButtonItemNode.itemHeight(style: item.style)), insets: UIEdgeInsets()) return (nodeLayout, { _ in if let strongSelf = self { @@ -137,14 +157,24 @@ final class VerticalListContextResultsChatInputPanelButtonItemNode: ListViewItem strongSelf.separatorNode.backgroundColor = item.theme.list.itemPlainSeparatorColor strongSelf.topSeparatorNode.backgroundColor = item.theme.list.itemPlainSeparatorColor - strongSelf.backgroundColor = item.theme.list.plainBackgroundColor - let _ = titleApply() + let titleOffsetY: CGFloat + switch item.style { + case .regular: + strongSelf.backgroundColor = item.theme.list.plainBackgroundColor + strongSelf.topSeparatorNode.isHidden = mergedTop + strongSelf.separatorNode.isHidden = !mergedBottom + titleOffsetY = 2.0 + case .round: + strongSelf.backgroundColor = nil + strongSelf.topSeparatorNode.isHidden = true + strongSelf.separatorNode.isHidden = !mergedBottom + titleOffsetY = 1.0 + } - strongSelf.titleNode.frame = CGRect(origin: CGPoint(x: floor((params.width - titleLayout.size.width) / 2.0), y: floor((nodeLayout.contentSize.height - titleLayout.size.height) / 2.0) + 2.0), size: titleLayout.size) + let _ = titleApply() - strongSelf.topSeparatorNode.isHidden = mergedTop - strongSelf.separatorNode.isHidden = !mergedBottom + strongSelf.titleNode.frame = CGRect(origin: CGPoint(x: floor((params.width - titleLayout.size.width) / 2.0), y: floor((nodeLayout.contentSize.height - titleLayout.size.height) / 2.0) + titleOffsetY), size: titleLayout.size) strongSelf.topSeparatorNode.frame = CGRect(origin: CGPoint(x: 0.0, y: 0.0), size: CGSize(width: params.width, height: UIScreenPixel)) strongSelf.separatorNode.frame = CGRect(origin: CGPoint(x: 0.0, y: nodeLayout.contentSize.height - UIScreenPixel), size: CGSize(width: params.width, height: UIScreenPixel)) diff --git a/submodules/TelegramVoip/Sources/GroupCallContext.swift b/submodules/TelegramVoip/Sources/GroupCallContext.swift index eaeda817df1..daefa3fa480 100644 --- a/submodules/TelegramVoip/Sources/GroupCallContext.swift +++ b/submodules/TelegramVoip/Sources/GroupCallContext.swift @@ -392,7 +392,7 @@ public final class OngoingGroupCallContext { public let mirrorHorizontally: Bool public let mirrorVertically: Bool - init(frameData: CallVideoFrameData) { + public init(frameData: CallVideoFrameData) { if let nativeBuffer = frameData.buffer as? CallVideoFrameNativePixelBuffer { if CVPixelBufferGetPixelFormatType(nativeBuffer.pixelBuffer) == kCVPixelFormatType_32ARGB { self.buffer = .argb(NativeBuffer(pixelBuffer: nativeBuffer.pixelBuffer)) diff --git a/submodules/TelegramVoip/Sources/macOS/OngoingCallVideoCapturer.swift b/submodules/TelegramVoip/Sources/macOS/OngoingCallVideoCapturer.swift index 2bf532a9db2..a16c539778e 100644 --- a/submodules/TelegramVoip/Sources/macOS/OngoingCallVideoCapturer.swift +++ b/submodules/TelegramVoip/Sources/macOS/OngoingCallVideoCapturer.swift @@ -8,7 +8,7 @@ import Foundation import Cocoa import TgVoipWebrtc - +import SwiftSignalKit public enum OngoingCallVideoOrientation { @@ -79,6 +79,25 @@ public final class OngoingCallVideoCapturer { self.impl = OngoingCallThreadLocalContextVideoCapturer(deviceId: deviceId, keepLandscape: keepLandscape) } + public func video() -> Signal { + return Signal { [weak self] subscriber in + let disposable = MetaDisposable() + + guard let strongSelf = self else { + return disposable + } + let innerDisposable = strongSelf.impl.addVideoOutput({ videoFrameData in + subscriber.putNext(OngoingGroupCallContext.VideoFrameData(frameData: videoFrameData)) + }) + disposable.set(ActionDisposable { + innerDisposable.dispose() + }) + + return disposable + } + } + + public func makeOutgoingVideoView(completion: @escaping (OngoingCallContextPresentationCallVideoView?) -> Void) { self.impl.makeOutgoingVideoView(false, completion: { view, _ in if let view = view { diff --git a/submodules/TgVoipWebrtc/tgcalls b/submodules/TgVoipWebrtc/tgcalls index 6b73742cdc1..564c632f936 160000 --- a/submodules/TgVoipWebrtc/tgcalls +++ b/submodules/TgVoipWebrtc/tgcalls @@ -1 +1 @@ -Subproject commit 6b73742cdc140c46a1ab1b8e3390354a9738e429 +Subproject commit 564c632f9368409870631d3cef75a7fc4070d45b diff --git a/submodules/TranslateUI/Sources/TranslateScreen.swift b/submodules/TranslateUI/Sources/TranslateScreen.swift index ba0f4d1ba7c..94f61b42ae3 100644 --- a/submodules/TranslateUI/Sources/TranslateScreen.swift +++ b/submodules/TranslateUI/Sources/TranslateScreen.swift @@ -720,6 +720,7 @@ public class TranslateScreen: ViewController { statusBarHeight: 0.0, navigationHeight: navigationHeight, safeInsets: UIEdgeInsets(top: layout.intrinsicInsets.top + layout.safeInsets.top, left: layout.safeInsets.left, bottom: layout.intrinsicInsets.bottom + layout.safeInsets.bottom, right: layout.safeInsets.right), + additionalInsets: layout.additionalInsets, inputHeight: layout.inputHeight ?? 0.0, metrics: layout.metrics, deviceMetrics: layout.deviceMetrics, diff --git a/submodules/WebUI/Sources/WebAppWebView.swift b/submodules/WebUI/Sources/WebAppWebView.swift index 169383d6f7d..28c66b03616 100644 --- a/submodules/WebUI/Sources/WebAppWebView.swift +++ b/submodules/WebUI/Sources/WebAppWebView.swift @@ -159,7 +159,7 @@ final class WebAppWebView: WKWebView { self.interactiveTransitionGestureRecognizerTest = { point -> Bool in return point.x > 30.0 } - self.allowsBackForwardNavigationGestures = false + self.allowsBackForwardNavigationGestures = true if #available(iOS 16.4, *) { self.isInspectable = true } diff --git a/swift_deps.bzl b/swift_deps.bzl index 3fdfdb43d4a..3947adc4e8c 100644 --- a/swift_deps.bzl +++ b/swift_deps.bzl @@ -12,7 +12,7 @@ def swift_dependencies(): # version: 2.6.1 swift_package( name = "swiftpkg_floatingpanel", - commit = "5b33d3d5ff1f50f4a2d64158ccfe8c07b5a3e649", + commit = "8f2be39bf49b4d5e22bbf7bdde69d5b76d0ecd2a", dependencies_index = "@//:swift_deps_index.json", remote = "https://github.com/scenee/FloatingPanel", ) @@ -33,10 +33,18 @@ def swift_dependencies(): remote = "https://github.com/LeoNatan/LNExtensionExecutor", ) - # branch: develop + # branch: main + swift_package( + name = "swiftpkg_navigation_stack_backport", + commit = "66716ce9c31198931c2275a0b69de2fdaa687e74", + dependencies_index = "@//:swift_deps_index.json", + remote = "https://github.com/denis15yo/navigation-stack-backport.git", + ) + + # branch: avatar-generator swift_package( name = "swiftpkg_nicegram_assistant_ios", - commit = "123eb001dcc9477f9087e40ff1b0e7f012182d45", + commit = "9d2bc90674d3fa38fdd2e6f8f069c67d296d2bde", dependencies_index = "@//:swift_deps_index.json", remote = "git@bitbucket.org:mobyrix/nicegram-assistant-ios.git", ) @@ -52,7 +60,7 @@ def swift_dependencies(): # version: 5.15.5 swift_package( name = "swiftpkg_sdwebimage", - commit = "b11493f76481dff17ac8f45274a6b698ba0d3af5", + commit = "73b9397cfbd902f606572964055464903b1d84c6", dependencies_index = "@//:swift_deps_index.json", remote = "https://github.com/SDWebImage/SDWebImage.git", ) diff --git a/swift_deps_index.json b/swift_deps_index.json index 910153ab308..991eed44722 100644 --- a/swift_deps_index.json +++ b/swift_deps_index.json @@ -47,6 +47,16 @@ "LNExtensionExecutor-Static" ] }, + { + "name": "NavigationStackBackport", + "c99name": "NavigationStackBackport", + "src_type": "swift", + "label": "@swiftpkg_navigation_stack_backport//:NavigationStackBackport.rspm", + "package_identity": "navigation-stack-backport", + "product_memberships": [ + "NavigationStackBackport" + ] + }, { "name": "CoreSwiftUI", "c99name": "CoreSwiftUI", @@ -54,8 +64,10 @@ "label": "@swiftpkg_nicegram_assistant_ios//:CoreSwiftUI.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatChatBanner", "FeatImagesHubUI", + "FeatNicegramHub", "FeatPhoneEntryBanner", "FeatPremiumUI", "NGAssistantUI", @@ -69,9 +81,30 @@ "label": "@swiftpkg_nicegram_assistant_ios//:FeatAmbassadors.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "NGAssistantUI" ] }, + { + "name": "FeatAvatarGenerator", + "c99name": "FeatAvatarGenerator", + "src_type": "swift", + "label": "@swiftpkg_nicegram_assistant_ios//:FeatAvatarGenerator.rspm", + "package_identity": "nicegram-assistant-ios", + "product_memberships": [ + "FeatAvatarGeneratorUI" + ] + }, + { + "name": "FeatAvatarGeneratorUI", + "c99name": "FeatAvatarGeneratorUI", + "src_type": "swift", + "label": "@swiftpkg_nicegram_assistant_ios//:FeatAvatarGeneratorUI.rspm", + "package_identity": "nicegram-assistant-ios", + "product_memberships": [ + "FeatAvatarGeneratorUI" + ] + }, { "name": "FeatBilling", "c99name": "FeatBilling", @@ -79,8 +112,10 @@ "label": "@swiftpkg_nicegram_assistant_ios//:FeatBilling.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatCardUI", "FeatImagesHubUI", + "FeatNicegramHub", "FeatPremiumUI", "FeatSpeechToText", "NGAiChatUI", @@ -96,8 +131,10 @@ "label": "@swiftpkg_nicegram_assistant_ios//:FeatCard.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatCardUI", "FeatImagesHubUI", + "FeatNicegramHub", "FeatPremiumUI", "NGAiChatUI", "NGAssistantUI", @@ -112,6 +149,7 @@ "label": "@swiftpkg_nicegram_assistant_ios//:FeatCardUI.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatCardUI", "NGAssistantUI" ] @@ -133,6 +171,7 @@ "label": "@swiftpkg_nicegram_assistant_ios//:FeatChatListBanner.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "NGAssistantUI" ] }, @@ -153,6 +192,7 @@ "label": "@swiftpkg_nicegram_assistant_ios//:FeatHiddenChats.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatHiddenChats", "NGAssistantUI" ] @@ -164,6 +204,7 @@ "label": "@swiftpkg_nicegram_assistant_ios//:FeatImagesHub.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatImagesHubUI", "NGAssistantUI" ] @@ -175,10 +216,24 @@ "label": "@swiftpkg_nicegram_assistant_ios//:FeatImagesHubUI.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatImagesHubUI", "NGAssistantUI" ] }, + { + "name": "FeatNicegramHub", + "c99name": "FeatNicegramHub", + "src_type": "swift", + "label": "@swiftpkg_nicegram_assistant_ios//:FeatNicegramHub.rspm", + "package_identity": "nicegram-assistant-ios", + "product_memberships": [ + "FeatAvatarGeneratorUI", + "FeatNicegramHub", + "NGAssistantUI", + "NGEntryPoint" + ] + }, { "name": "FeatPersistentStorage", "c99name": "FeatPersistentStorage", @@ -186,8 +241,10 @@ "label": "@swiftpkg_nicegram_assistant_ios//:FeatPersistentStorage.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatCardUI", "FeatImagesHubUI", + "FeatNicegramHub", "FeatPremiumUI", "NGAiChat", "NGAiChatUI", @@ -223,9 +280,11 @@ "label": "@swiftpkg_nicegram_assistant_ios//:FeatPremium.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatCardUI", "FeatChatBanner", "FeatImagesHubUI", + "FeatNicegramHub", "FeatPinnedChats", "FeatPremium", "FeatPremiumUI", @@ -248,6 +307,7 @@ "label": "@swiftpkg_nicegram_assistant_ios//:FeatPremiumUI.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatImagesHubUI", "FeatPremiumUI", "NGAssistantUI", @@ -261,8 +321,10 @@ "label": "@swiftpkg_nicegram_assistant_ios//:FeatRewards.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatCardUI", "FeatImagesHubUI", + "FeatNicegramHub", "FeatPremiumUI", "FeatTasks", "NGAiChatUI", @@ -278,6 +340,7 @@ "label": "@swiftpkg_nicegram_assistant_ios//:FeatRewardsUI.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "NGAssistantUI" ] }, @@ -319,8 +382,10 @@ "label": "@swiftpkg_nicegram_assistant_ios//:NGAiChat.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatCardUI", "FeatImagesHubUI", + "FeatNicegramHub", "FeatPremiumUI", "NGAiChat", "NGAiChatUI", @@ -336,6 +401,7 @@ "label": "@swiftpkg_nicegram_assistant_ios//:NGAiChatUI.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatImagesHubUI", "NGAiChatUI", "NGAssistantUI" @@ -348,9 +414,11 @@ "label": "@swiftpkg_nicegram_assistant_ios//:NGAnalytics.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatCardUI", "FeatChatBanner", "FeatImagesHubUI", + "FeatNicegramHub", "FeatPinnedChats", "FeatPremiumUI", "FeatTasks", @@ -370,9 +438,11 @@ "label": "@swiftpkg_nicegram_assistant_ios//:NGApi.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatCardUI", "FeatChatBanner", "FeatImagesHubUI", + "FeatNicegramHub", "FeatPinnedChats", "FeatPremiumUI", "FeatSpeechToText", @@ -395,6 +465,7 @@ "label": "@swiftpkg_nicegram_assistant_ios//:NGAssistant.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatImagesHubUI", "NGAiChatUI", "NGAssistantUI", @@ -408,6 +479,7 @@ "label": "@swiftpkg_nicegram_assistant_ios//:NGAssistantUI.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "NGAssistantUI" ] }, @@ -418,8 +490,10 @@ "label": "@swiftpkg_nicegram_assistant_ios//:NGAuth.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatCardUI", "FeatImagesHubUI", + "FeatNicegramHub", "FeatPremiumUI", "NGAiChatUI", "NGAssistantUI", @@ -434,10 +508,12 @@ "label": "@swiftpkg_nicegram_assistant_ios//:NGCore.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatCardUI", "FeatChatBanner", "FeatHiddenChats", "FeatImagesHubUI", + "FeatNicegramHub", "FeatPhoneEntryBanner", "FeatPinnedChats", "FeatPremium", @@ -467,9 +543,11 @@ "modulemap_label": "@swiftpkg_nicegram_assistant_ios//:NGCoreUI.rspm_modulemap", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatCardUI", "FeatChatBanner", "FeatImagesHubUI", + "FeatNicegramHub", "FeatPhoneEntryBanner", "FeatPinnedChats", "FeatPremiumUI", @@ -497,6 +575,7 @@ "label": "@swiftpkg_nicegram_assistant_ios//:NGGrumUI.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "NGAssistantUI" ] }, @@ -507,10 +586,12 @@ "label": "@swiftpkg_nicegram_assistant_ios//:NGLocalization.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatCardUI", "FeatChatBanner", "FeatHiddenChats", "FeatImagesHubUI", + "FeatNicegramHub", "FeatPhoneEntryBanner", "FeatPinnedChats", "FeatPremium", @@ -540,8 +621,10 @@ "label": "@swiftpkg_nicegram_assistant_ios//:NGRepoTg.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatCardUI", "FeatImagesHubUI", + "FeatNicegramHub", "FeatPremiumUI", "NGAiChatUI", "NGAssistantUI", @@ -557,9 +640,11 @@ "label": "@swiftpkg_nicegram_assistant_ios//:NGRepoUser.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatCardUI", "FeatChatBanner", "FeatImagesHubUI", + "FeatNicegramHub", "FeatPinnedChats", "FeatPremiumUI", "FeatSpeechToText", @@ -581,6 +666,7 @@ "label": "@swiftpkg_nicegram_assistant_ios//:NGSpecialOffer.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "NGAssistantUI", "NGSpecialOffer" ] @@ -592,8 +678,10 @@ "label": "@swiftpkg_nicegram_assistant_ios//:Tapjoy.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatCardUI", "FeatImagesHubUI", + "FeatNicegramHub", "FeatPremiumUI", "FeatTasks", "NGAiChatUI", @@ -609,9 +697,11 @@ "label": "@swiftpkg_nicegram_assistant_ios//:_NGRemoteConfig.rspm", "package_identity": "nicegram-assistant-ios", "product_memberships": [ + "FeatAvatarGeneratorUI", "FeatCardUI", "FeatChatBanner", "FeatImagesHubUI", + "FeatNicegramHub", "FeatPinnedChats", "FeatPremiumUI", "FeatSpeechToText", @@ -865,6 +955,18 @@ "type": "library", "label": "@swiftpkg_lnextensionexecutor//:LNExtensionExecutor-Static" }, + { + "identity": "navigation-stack-backport", + "name": "NavigationStackBackport", + "type": "library", + "label": "@swiftpkg_navigation_stack_backport//:NavigationStackBackport" + }, + { + "identity": "nicegram-assistant-ios", + "name": "FeatAvatarGeneratorUI", + "type": "library", + "label": "@swiftpkg_nicegram_assistant_ios//:FeatAvatarGeneratorUI" + }, { "identity": "nicegram-assistant-ios", "name": "FeatCardUI", @@ -889,6 +991,12 @@ "type": "library", "label": "@swiftpkg_nicegram_assistant_ios//:FeatImagesHubUI" }, + { + "identity": "nicegram-assistant-ios", + "name": "FeatNicegramHub", + "type": "library", + "label": "@swiftpkg_nicegram_assistant_ios//:FeatNicegramHub" + }, { "identity": "nicegram-assistant-ios", "name": "FeatPhoneEntryBanner", @@ -1114,9 +1222,9 @@ "name": "swiftpkg_floatingpanel", "identity": "floatingpanel", "remote": { - "commit": "5b33d3d5ff1f50f4a2d64158ccfe8c07b5a3e649", + "commit": "8f2be39bf49b4d5e22bbf7bdde69d5b76d0ecd2a", "remote": "https://github.com/scenee/FloatingPanel", - "version": "2.8.1" + "version": "2.8.2" } }, { @@ -1137,11 +1245,20 @@ "version": "1.2.0" } }, + { + "name": "swiftpkg_navigation_stack_backport", + "identity": "navigation-stack-backport", + "remote": { + "commit": "66716ce9c31198931c2275a0b69de2fdaa687e74", + "remote": "https://github.com/denis15yo/navigation-stack-backport.git", + "branch": "main" + } + }, { "name": "swiftpkg_nicegram_assistant_ios", "identity": "nicegram-assistant-ios", "remote": { - "commit": "123eb001dcc9477f9087e40ff1b0e7f012182d45", + "commit": "9d2bc90674d3fa38fdd2e6f8f069c67d296d2bde", "remote": "git@bitbucket.org:mobyrix/nicegram-assistant-ios.git", "branch": "develop" } @@ -1159,9 +1276,9 @@ "name": "swiftpkg_sdwebimage", "identity": "sdwebimage", "remote": { - "commit": "b11493f76481dff17ac8f45274a6b698ba0d3af5", + "commit": "73b9397cfbd902f606572964055464903b1d84c6", "remote": "https://github.com/SDWebImage/SDWebImage.git", - "version": "5.18.11" + "version": "5.19.0" } }, { diff --git a/versions.json b/versions.json index 3f573437c30..398ac3d2d61 100644 --- a/versions.json +++ b/versions.json @@ -1,5 +1,5 @@ { - "app": "1.5.6", + "app": "1.5.7", "bazel": "7.0.2", "xcode": "15.2", "macos": "13.0"