From a746a072dce383cc1122fa15244bbcffb33a59d2 Mon Sep 17 00:00:00 2001 From: dkaraush Date: Sun, 31 Dec 2023 16:05:27 +0400 Subject: [PATCH] update to 10.5.0 (4228) --- TMessagesProj/build.gradle | 16 + TMessagesProj/jni/image.cpp | 8 + TMessagesProj/jni/tgnet/Connection.cpp | 30 +- .../jni/voip/tgcalls/v2/InstanceV2Impl.cpp | 63 +- TMessagesProj/src/main/AndroidManifest.xml | 9 +- .../src/main/assets/darkblue.attheme | 2 +- TMessagesProj/src/main/assets/night.attheme | 2 +- .../widget/ChatListItemAnimator.java | 178 +- .../telegram/messenger/AndroidUtilities.java | 18 +- .../telegram/messenger/ApplicationLoader.java | 20 +- .../telegram/messenger/BillingController.java | 2 +- .../org/telegram/messenger/BuildVars.java | 3 +- .../org/telegram/messenger/ChatObject.java | 6 +- .../messenger/ChatThemeController.java | 244 +- .../org/telegram/messenger/DialogObject.java | 22 + .../telegram/messenger/FileLoadOperation.java | 2 +- .../org/telegram/messenger/FileLoader.java | 2 +- .../telegram/messenger/FileRefController.java | 5 +- .../org/telegram/messenger/ImageLoader.java | 6 +- .../org/telegram/messenger/ImageReceiver.java | 2 +- .../java/org/telegram/messenger/LiteMode.java | 30 +- .../telegram/messenger/LocaleController.java | 40 +- .../telegram/messenger/MediaController.java | 44 +- .../messenger/MediaDataController.java | 77 +- .../org/telegram/messenger/MessageObject.java | 448 ++- .../messenger/MessagePreviewParams.java | 34 +- .../messenger/MessagesController.java | 293 +- .../telegram/messenger/MessagesStorage.java | 32 +- .../messenger/NotificationCenter.java | 22 +- .../messenger/NotificationsController.java | 6 + .../messenger/SavedMessagesController.java | 76 + .../org/telegram/messenger/SharedConfig.java | 38 +- .../telegram/messenger/TopicsController.java | 6 +- .../org/telegram/messenger/UserObject.java | 6 +- .../telegram/messenger/VideoEditedInfo.java | 105 +- .../video/MediaCodecVideoConvertor.java | 54 +- .../messenger/video/OutputSurface.java | 4 +- .../messenger/video/TextureRenderer.java | 325 ++- .../telegram/messenger/voip/VoIPService.java | 82 +- .../messenger/voip/VoipAudioManager.java | 11 + .../telegram/tgnet/ConnectionsManager.java | 2 +- .../main/java/org/telegram/tgnet/TLRPC.java | 2520 +++++++++++++++-- .../org/telegram/tgnet/tl/TL_stories.java | 482 +++- .../org/telegram/ui/ActionBar/ActionBar.java | 79 +- .../ui/ActionBar/ActionBarLayout.java | 22 +- .../telegram/ui/ActionBar/AlertDialog.java | 26 +- .../telegram/ui/ActionBar/BottomSheet.java | 45 +- .../telegram/ui/ActionBar/EmojiThemes.java | 14 +- .../ui/ActionBar/FloatingToolbar.java | 2 +- .../ui/ActionBar/INavigationLayout.java | 1 + .../telegram/ui/ActionBar/SimpleTextView.java | 101 +- .../java/org/telegram/ui/ActionBar/Theme.java | 19 +- .../telegram/ui/ActionBar/ThemeColors.java | 4 +- .../ui/Cells/AppIconsSelectorCell.java | 2 +- .../java/org/telegram/ui/Cells/BaseCell.java | 11 + .../org/telegram/ui/Cells/ChatActionCell.java | 158 +- .../telegram/ui/Cells/ChatMessageCell.java | 568 +++- .../org/telegram/ui/Cells/DialogCell.java | 48 +- .../telegram/ui/Cells/DialogsHintCell.java | 38 +- .../telegram/ui/Cells/GroupCallUserCell.java | 9 +- .../telegram/ui/Cells/ManageChatUserCell.java | 2 + .../telegram/ui/Cells/ProfileSearchCell.java | 20 +- .../ui/Cells/ReactedUserHolderView.java | 72 +- .../telegram/ui/Cells/ShareDialogCell.java | 60 +- .../java/org/telegram/ui/Cells/TextCell.java | 107 +- .../org/telegram/ui/Cells/TextCheckCell.java | 11 +- .../ui/Cells/ThemePreviewMessagesCell.java | 44 +- .../org/telegram/ui/Cells/WallpaperCell.java | 2 +- .../telegram/ui/ChannelAdminLogActivity.java | 6 +- .../org/telegram/ui/ChannelColorActivity.java | 2185 ++++++++++++++ .../telegram/ui/ChannelWallpaperActivity.java | 491 ++++ .../view_data/ChartHorizontalLinesData.java | 6 +- .../java/org/telegram/ui/ChatActivity.java | 410 +-- .../telegram/ui/ChatBackgroundDrawable.java | 2 +- .../org/telegram/ui/ChatEditActivity.java | 17 +- .../org/telegram/ui/ChatEditTypeActivity.java | 36 +- .../telegram/ui/Components/AlertsCreator.java | 20 +- .../ui/Components/AnimatedEmojiDrawable.java | 15 +- .../ui/Components/BitmapShaderTools.java | 15 + .../telegram/ui/Components/BlobDrawable.java | 20 +- .../ui/Components/BlurringShader.java | 38 +- .../ui/Components/BotWebViewSheet.java | 6 +- .../BottomSheetWithRecyclerListView.java | 3 + .../org/telegram/ui/Components/Bulletin.java | 14 +- .../ui/Components/BulletinFactory.java | 15 + .../ui/Components/CaptionPhotoViewer.java | 2 +- .../ui/Components/ChatActivityEnterView.java | 404 ++- .../ui/Components/ChatAttachAlert.java | 4 +- .../ui/Components/ChatAvatarContainer.java | 50 +- .../ui/Components/ChatThemeBottomSheet.java | 342 ++- .../ui/Components/DotDividerSpan.java | 4 +- .../telegram/ui/Components/EditTextEmoji.java | 9 +- .../org/telegram/ui/Components/EmojiView.java | 2 +- .../ui/Components/FilterTabsView.java | 4 +- .../ui/Components/FireworksOverlay.java | 71 +- .../ui/Components/FolderBottomSheet.java | 3 + .../ui/Components/InstantCameraView.java | 40 +- .../ui/Components/MediaActionDrawable.java | 6 +- .../ui/Components/MenuToItemOptions.java | 2 +- .../ui/Components/MessagePreviewView.java | 4 +- .../Components/MotionBackgroundDrawable.java | 20 +- .../ui/Components/Paint/Views/EntityView.java | 17 +- .../Paint/Views/LPhotoPaintView.java | 280 +- .../Components/Paint/Views/LocationView.java | 2 +- .../Paint/Views/MessageEntityView.java | 1361 +++++++++ .../ui/Components/Paint/Views/PhotoView.java | 337 ++- .../Paint/Views/ReactionWidgetEntityView.java | 2 +- .../ui/Components/Paint/Views/RoundView.java | 2 +- .../Components/Paint/Views/StickerView.java | 2 +- .../Components/Paint/Views/TextPaintView.java | 2 +- .../PhotoViewerCaptionEnterView.java | 4 +- .../Premium/LimitReachedBottomSheet.java | 655 ++++- .../Components/Premium/PremiumGradient.java | 12 +- .../Premium/PremiumPreviewBottomSheet.java | 155 +- .../Components/Premium/StarParticlesView.java | 4 + .../Premium/boosts/BoostCounterView.java | 11 +- .../Premium/boosts/BoostDialogs.java | 118 +- .../Premium/boosts/BoostRepository.java | 104 +- .../boosts/BoostViaGiftsBottomSheet.java | 113 +- .../Premium/boosts/DiscountSpan.java | 84 + .../Premium/boosts/GiftInfoBottomSheet.java | 21 +- .../boosts/GradientButtonWithCounterView.java | 53 + .../PremiumPreviewGiftLinkBottomSheet.java | 170 ++ .../PremiumPreviewGiftSentBottomSheet.java | 158 ++ .../PremiumPreviewGiftToUsersBottomSheet.java | 402 +++ .../boosts/ReassignBoostBottomSheet.java | 76 +- .../Premium/boosts/SelectorBottomSheet.java | 2 +- .../boosts/UserSelectorBottomSheet.java | 593 ++++ .../Premium/boosts/adapters/BoostAdapter.java | 75 +- .../boosts/adapters/GiftInfoAdapter.java | 49 +- .../boosts/adapters/SelectorAdapter.java | 46 +- .../Premium/boosts/cells/ActionBtnCell.java | 12 + .../Premium/boosts/cells/DurationCell.java | 2 +- .../cells/DurationWithDiscountCell.java | 83 + .../Premium/boosts/cells/EnterPrizeCell.java | 123 + .../Premium/boosts/cells/HeaderCell.java | 49 +- .../Premium/boosts/cells/SwitcherCell.java | 41 + .../Premium/boosts/cells/TableCell.java | 5 +- .../boosts/cells/msg/GiveawayMessageCell.java | 156 +- .../cells/msg/GiveawayResultsMessageCell.java | 652 +++++ .../cells/selector/SelectorHeaderCell.java | 8 +- .../cells/selector/SelectorLetterCell.java | 2 +- .../cells/selector/SelectorSearchCell.java | 4 + .../ui/Components/RadialProgress2.java | 19 +- .../ui/Components/ReactedUsersListView.java | 6 +- .../ChatCustomReactionsEditActivity.java | 2 + .../Components/ReactionsContainerLayout.java | 6 +- .../ui/Components/RecyclerListView.java | 3 + .../ui/Components/ReplyMessageLine.java | 53 +- .../ui/Components/SearchViewPager.java | 2 +- .../org/telegram/ui/Components/SeekBar.java | 9 +- .../ui/Components/SeekBarWaveform.java | 177 +- .../telegram/ui/Components/ShareAlert.java | 22 +- .../ui/Components/SharedMediaLayout.java | 121 +- .../ui/Components/StatusBadgeComponent.java | 15 +- .../Components/StickerSetBulletinLayout.java | 4 +- .../telegram/ui/Components/ThanosEffect.java | 1115 ++++++++ .../ui/Components/ThemeSmallPreviewView.java | 91 +- .../Components/spoilers/SpoilerEffect2.java | 26 +- .../ui/Components/voip/AcceptDeclineView.java | 201 +- .../Components/voip/EmojiRationalLayout.java | 30 + .../ui/Components/voip/EndCloseLayout.java | 252 ++ .../voip/FabBackgroundDrawable.java | 1 - .../ui/Components/voip/HideEmojiTextView.java | 40 + .../Components/voip/ImageWithWavesView.java | 267 ++ .../voip/PrivateVideoPreviewDialogNew.java | 797 ++++++ .../ui/Components/voip/RateCallLayout.java | 274 ++ .../voip/VoIPBackgroundProvider.java | 205 ++ .../ui/Components/voip/VoIPEllipsizeSpan.java | 54 + .../Components/voip/VoIPFloatingLayout.java | 15 +- .../ui/Components/voip/VoIPHelper.java | 17 + .../voip/VoIPNotificationsLayout.java | 160 +- .../Components/voip/VoIPStatusTextView.java | 128 +- .../ui/Components/voip/VoIPTimerView.java | 27 +- .../ui/Components/voip/VoIPToggleButton.java | 21 +- .../ui/Components/voip/VoIPWindowView.java | 31 +- .../Components/voip/VoIpBitmapTextView.java | 70 + .../Components/voip/VoIpGradientLayout.java | 402 +++ .../ui/Components/voip/VoIpHintView.java | 44 + .../ui/Components/voip/VoIpSnowView.java | 39 + .../ui/Components/voip/VoIpSwitchLayout.java | 482 ++++ .../ui/Components/voip/VoipBlobDrawable.java | 42 + .../org/telegram/ui/ContactsActivity.java | 4 +- .../telegram/ui/DefaultThemesPreviewCell.java | 37 +- .../java/org/telegram/ui/DialogsActivity.java | 61 +- .../telegram/ui/EmojiAnimationsOverlay.java | 2 +- .../telegram/ui/LanguageSelectActivity.java | 14 +- .../java/org/telegram/ui/LaunchActivity.java | 110 +- .../telegram/ui/LauncherIconController.java | 2 +- .../telegram/ui/LiteModeSettingsActivity.java | 7 + .../telegram/ui/MessageStatisticActivity.java | 36 +- .../ui/MultiContactsSelectorBottomSheet.java | 497 ++++ .../org/telegram/ui/PaymentFormActivity.java | 20 +- .../org/telegram/ui/PeerColorActivity.java | 584 ++-- .../java/org/telegram/ui/PhotoViewer.java | 8 +- .../ui/PopupNotificationActivity.java | 2 +- .../telegram/ui/PremiumPreviewFragment.java | 2 +- .../telegram/ui/PrivacySettingsActivity.java | 2 +- .../java/org/telegram/ui/ProfileActivity.java | 239 +- .../org/telegram/ui/SecretMediaViewer.java | 5 +- .../org/telegram/ui/SecretVoicePlayer.java | 623 ++++ .../ui/SelectAnimatedEmojiDialog.java | 96 +- .../ui/Stories/DialogStoriesCell.java | 6 +- .../org/telegram/ui/Stories/HwLayouts.java | 1 + .../telegram/ui/Stories/PeerStoriesView.java | 171 +- .../ui/Stories/SelfStoryViewsPage.java | 482 +++- .../ui/Stories/StoriesController.java | 24 +- .../ui/Stories/StoriesListPlaceProvider.java | 34 +- .../telegram/ui/Stories/StoriesStorage.java | 7 + .../telegram/ui/Stories/StoryCaptionView.java | 96 +- .../ui/Stories/StoryMediaAreasView.java | 121 +- .../org/telegram/ui/Stories/StoryViewer.java | 10 +- .../recorder/ButtonWithCounterView.java | 15 +- .../recorder/CaptionContainerView.java | 94 +- .../ui/Stories/recorder/CaptionStory.java | 1 - .../ui/Stories/recorder/DraftsController.java | 37 +- .../ui/Stories/recorder/EmojiBottomSheet.java | 8 +- .../ui/Stories/recorder/HintView2.java | 34 +- .../ui/Stories/recorder/IStoryPart.java | 11 - .../ui/Stories/recorder/PaintView.java | 227 +- .../ui/Stories/recorder/PreviewButtons.java | 37 +- .../ui/Stories/recorder/PreviewView.java | 577 ++-- .../ui/Stories/recorder/StoryEntry.java | 479 ++-- .../ui/Stories/recorder/StoryRecorder.java | 927 +++++- .../ui/Stories/recorder/StoryThemeSheet.java | 154 + .../ui/Stories/recorder/TimelineView.java | 27 +- .../java/org/telegram/ui/ThemeActivity.java | 2 +- .../org/telegram/ui/ThemePreviewActivity.java | 622 +++- .../java/org/telegram/ui/VoIPFragment.java | 1182 ++++++-- .../ui/VoiceMessageEnterTransition.java | 29 +- .../telegram/ui/WallpapersListActivity.java | 57 +- .../drawable-hdpi/menu_feature_color_name.png | Bin 0 -> 1116 bytes .../menu_feature_color_profile.png | Bin 0 -> 1241 bytes .../res/drawable-hdpi/menu_feature_cover.png | Bin 0 -> 1051 bytes .../drawable-hdpi/menu_feature_custombg.png | Bin 0 -> 1050 bytes .../res/drawable-hdpi/menu_feature_links.png | Bin 0 -> 919 bytes .../res/drawable-hdpi/menu_feature_links2.png | Bin 0 -> 1019 bytes .../drawable-hdpi/menu_feature_reactions.png | Bin 0 -> 995 bytes .../res/drawable-hdpi/menu_feature_status.png | Bin 0 -> 1136 bytes .../drawable-hdpi/menu_feature_stories.png | Bin 0 -> 1136 bytes .../drawable-hdpi/menu_feature_wallpaper.png | Bin 0 -> 950 bytes .../src/main/res/drawable-hdpi/menu_gift.png | Bin 0 -> 861 bytes .../res/drawable-hdpi/menu_views_reposts.png | Bin 0 -> 691 bytes .../res/drawable-hdpi/menu_views_reposts3.png | Bin 0 -> 669 bytes .../res/drawable-hdpi/mini_forward_story.png | Bin 0 -> 609 bytes .../res/drawable-hdpi/msg_call_bluetooth.png | Bin 0 -> 1044 bytes .../res/drawable-hdpi/msg_call_earpiece.png | Bin 0 -> 1211 bytes .../res/drawable-hdpi/msg_call_minimize.png | Bin 0 -> 584 bytes .../msg_call_minimize_shadow.png | Bin 0 -> 861 bytes .../res/drawable-hdpi/msg_call_speaker.png | Bin 0 -> 949 bytes .../drawable-mdpi/menu_feature_color_name.png | Bin 0 -> 762 bytes .../menu_feature_color_profile.png | Bin 0 -> 802 bytes .../res/drawable-mdpi/menu_feature_cover.png | Bin 0 -> 737 bytes .../drawable-mdpi/menu_feature_custombg.png | Bin 0 -> 741 bytes .../res/drawable-mdpi/menu_feature_links.png | Bin 0 -> 620 bytes .../res/drawable-mdpi/menu_feature_links2.png | Bin 0 -> 682 bytes .../drawable-mdpi/menu_feature_reactions.png | Bin 0 -> 672 bytes .../res/drawable-mdpi/menu_feature_status.png | Bin 0 -> 737 bytes .../drawable-mdpi/menu_feature_stories.png | Bin 0 -> 787 bytes .../drawable-mdpi/menu_feature_wallpaper.png | Bin 0 -> 637 bytes .../src/main/res/drawable-mdpi/menu_gift.png | Bin 0 -> 601 bytes .../res/drawable-mdpi/menu_views_reposts.png | Bin 0 -> 557 bytes .../res/drawable-mdpi/menu_views_reposts3.png | Bin 0 -> 506 bytes .../res/drawable-mdpi/mini_forward_story.png | Bin 0 -> 416 bytes .../res/drawable-mdpi/msg_call_bluetooth.png | Bin 0 -> 682 bytes .../res/drawable-mdpi/msg_call_earpiece.png | Bin 0 -> 795 bytes .../res/drawable-mdpi/msg_call_minimize.png | Bin 0 -> 466 bytes .../msg_call_minimize_shadow.png | Bin 0 -> 622 bytes .../res/drawable-mdpi/msg_call_speaker.png | Bin 0 -> 708 bytes .../menu_feature_color_name.png | Bin 0 -> 1511 bytes .../menu_feature_color_profile.png | Bin 0 -> 1680 bytes .../res/drawable-xhdpi/menu_feature_cover.png | Bin 0 -> 1520 bytes .../drawable-xhdpi/menu_feature_custombg.png | Bin 0 -> 1451 bytes .../res/drawable-xhdpi/menu_feature_links.png | Bin 0 -> 1201 bytes .../drawable-xhdpi/menu_feature_links2.png | Bin 0 -> 1350 bytes .../drawable-xhdpi/menu_feature_reactions.png | Bin 0 -> 1352 bytes .../drawable-xhdpi/menu_feature_status.png | Bin 0 -> 1563 bytes .../drawable-xhdpi/menu_feature_stories.png | Bin 0 -> 1567 bytes .../drawable-xhdpi/menu_feature_wallpaper.png | Bin 0 -> 1263 bytes .../src/main/res/drawable-xhdpi/menu_gift.png | Bin 0 -> 1099 bytes .../res/drawable-xhdpi/menu_views_reposts.png | Bin 0 -> 911 bytes .../drawable-xhdpi/menu_views_reposts3.png | Bin 0 -> 881 bytes .../res/drawable-xhdpi/mini_forward_story.png | Bin 0 -> 803 bytes .../res/drawable-xhdpi/msg_call_bluetooth.png | Bin 0 -> 1399 bytes .../res/drawable-xhdpi/msg_call_earpiece.png | Bin 0 -> 1654 bytes .../res/drawable-xhdpi/msg_call_minimize.png | Bin 0 -> 741 bytes .../msg_call_minimize_shadow.png | Bin 0 -> 1189 bytes .../res/drawable-xhdpi/msg_call_speaker.png | Bin 0 -> 1363 bytes .../menu_feature_color_name.png | Bin 0 -> 2211 bytes .../menu_feature_color_profile.png | Bin 0 -> 2573 bytes .../drawable-xxhdpi/menu_feature_cover.png | Bin 0 -> 2126 bytes .../drawable-xxhdpi/menu_feature_custombg.png | Bin 0 -> 1988 bytes .../drawable-xxhdpi/menu_feature_links.png | Bin 0 -> 1808 bytes .../drawable-xxhdpi/menu_feature_links2.png | Bin 0 -> 1975 bytes .../menu_feature_reactions.png | Bin 0 -> 1945 bytes .../drawable-xxhdpi/menu_feature_status.png | Bin 0 -> 2438 bytes .../drawable-xxhdpi/menu_feature_stories.png | Bin 0 -> 2358 bytes .../menu_feature_wallpaper.png | Bin 0 -> 1739 bytes .../main/res/drawable-xxhdpi/menu_gift.png | Bin 0 -> 1592 bytes .../drawable-xxhdpi/menu_views_reposts.png | Bin 0 -> 1142 bytes .../drawable-xxhdpi/menu_views_reposts3.png | Bin 0 -> 1255 bytes .../drawable-xxhdpi/mini_forward_story.png | Bin 0 -> 1095 bytes .../drawable-xxhdpi/msg_call_bluetooth.png | Bin 0 -> 2124 bytes .../res/drawable-xxhdpi/msg_call_earpiece.png | Bin 0 -> 2445 bytes .../res/drawable-xxhdpi/msg_call_minimize.png | Bin 0 -> 979 bytes .../msg_call_minimize_shadow.png | Bin 0 -> 1595 bytes .../res/drawable-xxhdpi/msg_call_speaker.png | Bin 0 -> 1883 bytes .../main/res/drawable/icon_2_background.xml | 8 +- .../res/drawable/icon_2_background_round.xml | 8 +- .../res/drawable/icon_2_background_sa.png | Bin 48746 -> 0 bytes .../main/res/drawable/icon_3_background.xml | 8 +- .../res/drawable/icon_3_background_round.xml | 8 +- .../res/drawable/icon_3_background_sa.png | Bin 21353 -> 0 bytes .../res/drawable/icon_3_background_sa.xml | 35 + .../main/res/drawable/icon_4_background.xml | 8 +- .../res/drawable/icon_4_background_round.xml | 8 +- .../res/drawable/icon_4_background_sa.png | Bin 70958 -> 0 bytes .../res/drawable/icon_4_background_sa.xml | 9 + .../main/res/drawable/icon_5_background.xml | 8 +- .../res/drawable/icon_5_background_round.xml | 8 +- .../res/drawable/icon_5_background_sa.png | Bin 61301 -> 0 bytes .../res/drawable/icon_5_background_sa.xml | 9 + .../main/res/drawable/icon_6_background.xml | 8 +- .../res/drawable/icon_6_background_round.xml | 8 +- .../res/drawable/icon_6_background_sa.png | Bin 7328 -> 0 bytes .../res/drawable/icon_6_background_sa.xml | 8 + .../src/main/res/drawable/icon_background.xml | 8 +- .../res/drawable/icon_background_round.xml | 8 +- .../main/res/drawable/icon_background_sa.png | Bin 7149 -> 0 bytes .../main/res/drawable/icon_background_sa.xml | 8 + .../src/main/res/drawable/icon_plane.xml | 10 +- .../res/mipmap-hdpi/icon_2_background_sa.png | Bin 0 -> 15309 bytes .../res/mipmap-hdpi/icon_background_clip.png | Bin 0 -> 2578 bytes .../icon_background_clip_round.png | Bin 0 -> 2205 bytes .../res/mipmap-mdpi/icon_2_background_sa.png | Bin 0 -> 7007 bytes .../res/mipmap-mdpi/icon_background_clip.png | Bin 0 -> 1203 bytes .../icon_background_clip_round.png | Bin 0 -> 1297 bytes .../res/mipmap-xhdpi/icon_2_background_sa.png | Bin 0 -> 22932 bytes .../res/mipmap-xhdpi/icon_background_clip.png | Bin 0 -> 2578 bytes .../icon_background_clip_round.png | Bin 0 -> 2716 bytes .../mipmap-xxhdpi/icon_2_background_sa.png | Bin 0 -> 60781 bytes .../mipmap-xxhdpi/icon_background_clip.png | Bin 0 -> 4523 bytes .../icon_background_clip_round.png | Bin 0 -> 4821 bytes .../mipmap-xxxhdpi/icon_2_background_sa.png | Bin 0 -> 126332 bytes .../mipmap-xxxhdpi/icon_background_clip.png | Bin 0 -> 5492 bytes .../icon_background_clip_round.png | Bin 0 -> 5846 bytes .../src/main/res/raw/bt_to_speaker.json | 1 + .../src/main/res/raw/call_accept.json | 1 + TMessagesProj/src/main/res/raw/call_mute.json | 1 + .../src/main/res/raw/call_unmute.json | 1 + .../src/main/res/raw/camera_flip2.json | 1 + TMessagesProj/src/main/res/raw/fire_once.json | 1 + .../src/main/res/raw/giveaway_results.json | 1 + .../src/main/res/raw/ic_boosts_replace.json | 1 + .../src/main/res/raw/metadatascript.js | 11 + TMessagesProj/src/main/res/raw/rate.json | 1 + .../src/main/res/raw/speaker_to_bt.json | 1 + .../src/main/res/raw/spoiler_compute.glsl | 39 - TMessagesProj/src/main/res/raw/star_fill.json | 1 + .../src/main/res/raw/star_stroke.json | 1 + .../src/main/res/raw/thanos_fragment.glsl | 18 + .../src/main/res/raw/thanos_vertex.glsl | 145 + .../src/main/res/raw/video_start.json | 1 + .../src/main/res/raw/video_stop.json | 1 + TMessagesProj/src/main/res/values/strings.xml | 192 +- .../src/main/res/xml/automotive_app_desc.xml | 5 - .../mipmap-anydpi-v26/icon_2_launcher_sa.xml | 2 +- gradle.properties | 4 +- 368 files changed, 26782 insertions(+), 4071 deletions(-) create mode 100644 TMessagesProj/src/main/java/org/telegram/messenger/SavedMessagesController.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/ChannelColorActivity.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/ChannelWallpaperActivity.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/MessageEntityView.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/DiscountSpan.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/GradientButtonWithCounterView.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/PremiumPreviewGiftLinkBottomSheet.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/PremiumPreviewGiftSentBottomSheet.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/PremiumPreviewGiftToUsersBottomSheet.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/UserSelectorBottomSheet.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/DurationWithDiscountCell.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/EnterPrizeCell.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/SwitcherCell.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/msg/GiveawayResultsMessageCell.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/ThanosEffect.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/voip/EmojiRationalLayout.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/voip/EndCloseLayout.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/voip/HideEmojiTextView.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/voip/ImageWithWavesView.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/voip/PrivateVideoPreviewDialogNew.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/voip/RateCallLayout.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPBackgroundProvider.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPEllipsizeSpan.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIpBitmapTextView.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIpGradientLayout.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIpHintView.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIpSnowView.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIpSwitchLayout.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoipBlobDrawable.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/MultiContactsSelectorBottomSheet.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/SecretVoicePlayer.java delete mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/IStoryPart.java create mode 100644 TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/StoryThemeSheet.java create mode 100644 TMessagesProj/src/main/res/drawable-hdpi/menu_feature_color_name.png create mode 100644 TMessagesProj/src/main/res/drawable-hdpi/menu_feature_color_profile.png create mode 100644 TMessagesProj/src/main/res/drawable-hdpi/menu_feature_cover.png create mode 100644 TMessagesProj/src/main/res/drawable-hdpi/menu_feature_custombg.png create mode 100644 TMessagesProj/src/main/res/drawable-hdpi/menu_feature_links.png create mode 100644 TMessagesProj/src/main/res/drawable-hdpi/menu_feature_links2.png create mode 100644 TMessagesProj/src/main/res/drawable-hdpi/menu_feature_reactions.png create mode 100644 TMessagesProj/src/main/res/drawable-hdpi/menu_feature_status.png create mode 100644 TMessagesProj/src/main/res/drawable-hdpi/menu_feature_stories.png create mode 100644 TMessagesProj/src/main/res/drawable-hdpi/menu_feature_wallpaper.png create mode 100644 TMessagesProj/src/main/res/drawable-hdpi/menu_gift.png create mode 100644 TMessagesProj/src/main/res/drawable-hdpi/menu_views_reposts.png create mode 100644 TMessagesProj/src/main/res/drawable-hdpi/menu_views_reposts3.png create mode 100644 TMessagesProj/src/main/res/drawable-hdpi/mini_forward_story.png create mode 100644 TMessagesProj/src/main/res/drawable-hdpi/msg_call_bluetooth.png create mode 100644 TMessagesProj/src/main/res/drawable-hdpi/msg_call_earpiece.png create mode 100644 TMessagesProj/src/main/res/drawable-hdpi/msg_call_minimize.png create mode 100644 TMessagesProj/src/main/res/drawable-hdpi/msg_call_minimize_shadow.png create mode 100644 TMessagesProj/src/main/res/drawable-hdpi/msg_call_speaker.png create mode 100644 TMessagesProj/src/main/res/drawable-mdpi/menu_feature_color_name.png create mode 100644 TMessagesProj/src/main/res/drawable-mdpi/menu_feature_color_profile.png create mode 100644 TMessagesProj/src/main/res/drawable-mdpi/menu_feature_cover.png create mode 100644 TMessagesProj/src/main/res/drawable-mdpi/menu_feature_custombg.png create mode 100644 TMessagesProj/src/main/res/drawable-mdpi/menu_feature_links.png create mode 100644 TMessagesProj/src/main/res/drawable-mdpi/menu_feature_links2.png create mode 100644 TMessagesProj/src/main/res/drawable-mdpi/menu_feature_reactions.png create mode 100644 TMessagesProj/src/main/res/drawable-mdpi/menu_feature_status.png create mode 100644 TMessagesProj/src/main/res/drawable-mdpi/menu_feature_stories.png create mode 100644 TMessagesProj/src/main/res/drawable-mdpi/menu_feature_wallpaper.png create mode 100644 TMessagesProj/src/main/res/drawable-mdpi/menu_gift.png create mode 100644 TMessagesProj/src/main/res/drawable-mdpi/menu_views_reposts.png create mode 100644 TMessagesProj/src/main/res/drawable-mdpi/menu_views_reposts3.png create mode 100644 TMessagesProj/src/main/res/drawable-mdpi/mini_forward_story.png create mode 100644 TMessagesProj/src/main/res/drawable-mdpi/msg_call_bluetooth.png create mode 100644 TMessagesProj/src/main/res/drawable-mdpi/msg_call_earpiece.png create mode 100644 TMessagesProj/src/main/res/drawable-mdpi/msg_call_minimize.png create mode 100644 TMessagesProj/src/main/res/drawable-mdpi/msg_call_minimize_shadow.png create mode 100644 TMessagesProj/src/main/res/drawable-mdpi/msg_call_speaker.png create mode 100644 TMessagesProj/src/main/res/drawable-xhdpi/menu_feature_color_name.png create mode 100644 TMessagesProj/src/main/res/drawable-xhdpi/menu_feature_color_profile.png create mode 100644 TMessagesProj/src/main/res/drawable-xhdpi/menu_feature_cover.png create mode 100644 TMessagesProj/src/main/res/drawable-xhdpi/menu_feature_custombg.png create mode 100644 TMessagesProj/src/main/res/drawable-xhdpi/menu_feature_links.png create mode 100644 TMessagesProj/src/main/res/drawable-xhdpi/menu_feature_links2.png create mode 100644 TMessagesProj/src/main/res/drawable-xhdpi/menu_feature_reactions.png create mode 100644 TMessagesProj/src/main/res/drawable-xhdpi/menu_feature_status.png create mode 100644 TMessagesProj/src/main/res/drawable-xhdpi/menu_feature_stories.png create mode 100644 TMessagesProj/src/main/res/drawable-xhdpi/menu_feature_wallpaper.png create mode 100644 TMessagesProj/src/main/res/drawable-xhdpi/menu_gift.png create mode 100644 TMessagesProj/src/main/res/drawable-xhdpi/menu_views_reposts.png create mode 100644 TMessagesProj/src/main/res/drawable-xhdpi/menu_views_reposts3.png create mode 100644 TMessagesProj/src/main/res/drawable-xhdpi/mini_forward_story.png create mode 100644 TMessagesProj/src/main/res/drawable-xhdpi/msg_call_bluetooth.png create mode 100644 TMessagesProj/src/main/res/drawable-xhdpi/msg_call_earpiece.png create mode 100644 TMessagesProj/src/main/res/drawable-xhdpi/msg_call_minimize.png create mode 100644 TMessagesProj/src/main/res/drawable-xhdpi/msg_call_minimize_shadow.png create mode 100644 TMessagesProj/src/main/res/drawable-xhdpi/msg_call_speaker.png create mode 100644 TMessagesProj/src/main/res/drawable-xxhdpi/menu_feature_color_name.png create mode 100644 TMessagesProj/src/main/res/drawable-xxhdpi/menu_feature_color_profile.png create mode 100644 TMessagesProj/src/main/res/drawable-xxhdpi/menu_feature_cover.png create mode 100644 TMessagesProj/src/main/res/drawable-xxhdpi/menu_feature_custombg.png create mode 100644 TMessagesProj/src/main/res/drawable-xxhdpi/menu_feature_links.png create mode 100644 TMessagesProj/src/main/res/drawable-xxhdpi/menu_feature_links2.png create mode 100644 TMessagesProj/src/main/res/drawable-xxhdpi/menu_feature_reactions.png create mode 100644 TMessagesProj/src/main/res/drawable-xxhdpi/menu_feature_status.png create mode 100644 TMessagesProj/src/main/res/drawable-xxhdpi/menu_feature_stories.png create mode 100644 TMessagesProj/src/main/res/drawable-xxhdpi/menu_feature_wallpaper.png create mode 100644 TMessagesProj/src/main/res/drawable-xxhdpi/menu_gift.png create mode 100644 TMessagesProj/src/main/res/drawable-xxhdpi/menu_views_reposts.png create mode 100644 TMessagesProj/src/main/res/drawable-xxhdpi/menu_views_reposts3.png create mode 100644 TMessagesProj/src/main/res/drawable-xxhdpi/mini_forward_story.png create mode 100644 TMessagesProj/src/main/res/drawable-xxhdpi/msg_call_bluetooth.png create mode 100644 TMessagesProj/src/main/res/drawable-xxhdpi/msg_call_earpiece.png create mode 100644 TMessagesProj/src/main/res/drawable-xxhdpi/msg_call_minimize.png create mode 100644 TMessagesProj/src/main/res/drawable-xxhdpi/msg_call_minimize_shadow.png create mode 100644 TMessagesProj/src/main/res/drawable-xxhdpi/msg_call_speaker.png delete mode 100644 TMessagesProj/src/main/res/drawable/icon_2_background_sa.png delete mode 100644 TMessagesProj/src/main/res/drawable/icon_3_background_sa.png create mode 100644 TMessagesProj/src/main/res/drawable/icon_3_background_sa.xml delete mode 100644 TMessagesProj/src/main/res/drawable/icon_4_background_sa.png create mode 100644 TMessagesProj/src/main/res/drawable/icon_4_background_sa.xml delete mode 100644 TMessagesProj/src/main/res/drawable/icon_5_background_sa.png create mode 100644 TMessagesProj/src/main/res/drawable/icon_5_background_sa.xml delete mode 100644 TMessagesProj/src/main/res/drawable/icon_6_background_sa.png create mode 100644 TMessagesProj/src/main/res/drawable/icon_6_background_sa.xml delete mode 100644 TMessagesProj/src/main/res/drawable/icon_background_sa.png create mode 100644 TMessagesProj/src/main/res/drawable/icon_background_sa.xml create mode 100644 TMessagesProj/src/main/res/mipmap-hdpi/icon_2_background_sa.png create mode 100644 TMessagesProj/src/main/res/mipmap-hdpi/icon_background_clip.png create mode 100644 TMessagesProj/src/main/res/mipmap-hdpi/icon_background_clip_round.png create mode 100644 TMessagesProj/src/main/res/mipmap-mdpi/icon_2_background_sa.png create mode 100644 TMessagesProj/src/main/res/mipmap-mdpi/icon_background_clip.png create mode 100644 TMessagesProj/src/main/res/mipmap-mdpi/icon_background_clip_round.png create mode 100644 TMessagesProj/src/main/res/mipmap-xhdpi/icon_2_background_sa.png create mode 100644 TMessagesProj/src/main/res/mipmap-xhdpi/icon_background_clip.png create mode 100644 TMessagesProj/src/main/res/mipmap-xhdpi/icon_background_clip_round.png create mode 100644 TMessagesProj/src/main/res/mipmap-xxhdpi/icon_2_background_sa.png create mode 100644 TMessagesProj/src/main/res/mipmap-xxhdpi/icon_background_clip.png create mode 100644 TMessagesProj/src/main/res/mipmap-xxhdpi/icon_background_clip_round.png create mode 100644 TMessagesProj/src/main/res/mipmap-xxxhdpi/icon_2_background_sa.png create mode 100644 TMessagesProj/src/main/res/mipmap-xxxhdpi/icon_background_clip.png create mode 100644 TMessagesProj/src/main/res/mipmap-xxxhdpi/icon_background_clip_round.png create mode 100644 TMessagesProj/src/main/res/raw/bt_to_speaker.json create mode 100644 TMessagesProj/src/main/res/raw/call_accept.json create mode 100644 TMessagesProj/src/main/res/raw/call_mute.json create mode 100644 TMessagesProj/src/main/res/raw/call_unmute.json create mode 100644 TMessagesProj/src/main/res/raw/camera_flip2.json create mode 100644 TMessagesProj/src/main/res/raw/fire_once.json create mode 100644 TMessagesProj/src/main/res/raw/giveaway_results.json create mode 100644 TMessagesProj/src/main/res/raw/ic_boosts_replace.json create mode 100644 TMessagesProj/src/main/res/raw/metadatascript.js create mode 100644 TMessagesProj/src/main/res/raw/rate.json create mode 100644 TMessagesProj/src/main/res/raw/speaker_to_bt.json delete mode 100644 TMessagesProj/src/main/res/raw/spoiler_compute.glsl create mode 100644 TMessagesProj/src/main/res/raw/star_fill.json create mode 100644 TMessagesProj/src/main/res/raw/star_stroke.json create mode 100644 TMessagesProj/src/main/res/raw/thanos_fragment.glsl create mode 100644 TMessagesProj/src/main/res/raw/thanos_vertex.glsl create mode 100644 TMessagesProj/src/main/res/raw/video_start.json create mode 100644 TMessagesProj/src/main/res/raw/video_stop.json delete mode 100644 TMessagesProj/src/main/res/xml/automotive_app_desc.xml diff --git a/TMessagesProj/build.gradle b/TMessagesProj/build.gradle index 04241d3b6e9..db6643c9195 100644 --- a/TMessagesProj/build.gradle +++ b/TMessagesProj/build.gradle @@ -46,6 +46,16 @@ dependencies { implementation 'com.google.code.gson:gson:2.10' implementation 'com.google.guava:guava:31.1-android' + implementation 'com.google.android.gms:play-services-mlkit-subject-segmentation:16.0.0-beta1' + constraints { + implementation("org.jetbrains.kotlin:kotlin-stdlib-jdk7:1.8.0") { + because("kotlin-stdlib-jdk7 is now a part of kotlin-stdlib") + } + implementation("org.jetbrains.kotlin:kotlin-stdlib-jdk8:1.8.0") { + because("kotlin-stdlib-jdk8 is now a part of kotlin-stdlib") + } + } + coreLibraryDesugaring 'com.android.tools:desugar_jdk_libs:1.1.5' } @@ -108,6 +118,7 @@ android { multiDexEnabled true proguardFiles getDefaultProguardFile('proguard-android.txt'), '../TMessagesProj/proguard-rules.pro' ndk.debugSymbolLevel = 'FULL' + buildConfigField "String", "BUILD_VERSION_STRING", "\"" + APP_VERSION_NAME + "\"" buildConfigField "String", "APP_CENTER_HASH", "\"\"" buildConfigField "boolean", "DEBUG_VERSION", "true" buildConfigField "boolean", "DEBUG_PRIVATE_VERSION", "true" @@ -121,6 +132,7 @@ android { multiDexEnabled true proguardFiles getDefaultProguardFile('proguard-android.txt'), '../TMessagesProj/proguard-rules.pro' ndk.debugSymbolLevel = 'FULL' + buildConfigField "String", "BUILD_VERSION_STRING", "\"" + APP_VERSION_NAME + "\"" buildConfigField "String", "APP_CENTER_HASH", "\"" + getProps("APP_CENTER_HASH_PRIVATE") + "\"" buildConfigField "boolean", "DEBUG_VERSION", "true" buildConfigField "boolean", "DEBUG_PRIVATE_VERSION", "true" @@ -134,6 +146,7 @@ android { multiDexEnabled true proguardFiles getDefaultProguardFile('proguard-android.txt'), '../TMessagesProj/proguard-rules.pro' ndk.debugSymbolLevel = 'FULL' + buildConfigField "String", "BUILD_VERSION_STRING", "\"" + APP_VERSION_NAME + "\"" buildConfigField "String", "APP_CENTER_HASH", "\"" + getProps("APP_CENTER_HASH_PUBLIC") + "\"" buildConfigField "boolean", "DEBUG_VERSION", "true" buildConfigField "boolean", "DEBUG_PRIVATE_VERSION", "false" @@ -147,6 +160,7 @@ android { multiDexEnabled true proguardFiles getDefaultProguardFile('proguard-android.txt'), '../TMessagesProj/proguard-rules.pro' ndk.debugSymbolLevel = 'FULL' + buildConfigField "String", "BUILD_VERSION_STRING", "\"" + APP_VERSION_NAME + "\"" buildConfigField "String", "APP_CENTER_HASH", "\"" + getProps("APP_CENTER_HASH_HARDCORE") + "\"" buildConfigField "boolean", "DEBUG_VERSION", "true" buildConfigField "boolean", "DEBUG_PRIVATE_VERSION", "true" @@ -160,6 +174,7 @@ android { multiDexEnabled true proguardFiles getDefaultProguardFile('proguard-android.txt'), '../TMessagesProj/proguard-rules.pro' ndk.debugSymbolLevel = 'FULL' + buildConfigField "String", "BUILD_VERSION_STRING", "\"" + APP_VERSION_NAME + "\"" buildConfigField "String", "APP_CENTER_HASH", "\"\"" buildConfigField "boolean", "DEBUG_VERSION", "false" buildConfigField "boolean", "DEBUG_PRIVATE_VERSION", "false" @@ -174,6 +189,7 @@ android { multiDexEnabled true proguardFiles getDefaultProguardFile('proguard-android.txt'), '../TMessagesProj/proguard-rules.pro' ndk.debugSymbolLevel = 'FULL' + buildConfigField "String", "BUILD_VERSION_STRING", "\"" + APP_VERSION_NAME + "\"" buildConfigField "String", "APP_CENTER_HASH", "\"\"" buildConfigField "boolean", "DEBUG_VERSION", "false" buildConfigField "boolean", "DEBUG_PRIVATE_VERSION", "false" diff --git a/TMessagesProj/jni/image.cpp b/TMessagesProj/jni/image.cpp index 6169b000a69..8fb78c18e0a 100644 --- a/TMessagesProj/jni/image.cpp +++ b/TMessagesProj/jni/image.cpp @@ -1186,6 +1186,7 @@ std::vector> gatherPositions(std::vectorGetIntArrayElements(colors, nullptr); float *newPixelCache = nullptr; + + if (width * height != pixelCacheSize && pixelCache != nullptr) { + delete[] pixelCache; + pixelCache = nullptr; + } + pixelCacheSize = width * height; + if (pixelCache == nullptr) { newPixelCache = new float[width * height * 2]; } diff --git a/TMessagesProj/jni/tgnet/Connection.cpp b/TMessagesProj/jni/tgnet/Connection.cpp index fa53030ed58..6457d8957e3 100644 --- a/TMessagesProj/jni/tgnet/Connection.cpp +++ b/TMessagesProj/jni/tgnet/Connection.cpp @@ -122,6 +122,7 @@ void Connection::onReceivedData(NativeByteBuffer *buffer) { buffer->rewind(); + NativeByteBuffer *reuseLater = nullptr; while (buffer->hasRemaining()) { if (!hasSomeDataSinceLastConnect) { currentDatacenter->storeCurrentAddressAndPortNum(); @@ -154,14 +155,11 @@ void Connection::onReceivedData(NativeByteBuffer *buffer) { if ((fByte & (1 << 7)) != 0) { buffer->position(mark); if (buffer->remaining() < 4) { - NativeByteBuffer *reuseLater = restOfTheData; + reuseLater = restOfTheData; restOfTheData = BuffersStorage::getInstance().getFreeBuffer(16384); restOfTheData->writeBytes(buffer); restOfTheData->limit(restOfTheData->position()); lastPacketLength = 0; - if (reuseLater != nullptr) { - reuseLater->reuse(); - } break; } int32_t ackId = buffer->readBigInt32(nullptr) & (~(1 << 31)); @@ -175,14 +173,11 @@ void Connection::onReceivedData(NativeByteBuffer *buffer) { buffer->position(mark); if (buffer->remaining() < 4) { if (restOfTheData == nullptr || (restOfTheData != nullptr && restOfTheData->position() != 0)) { - NativeByteBuffer *reuseLater = restOfTheData; + reuseLater = restOfTheData; restOfTheData = BuffersStorage::getInstance().getFreeBuffer(16384); restOfTheData->writeBytes(buffer); restOfTheData->limit(restOfTheData->position()); lastPacketLength = 0; - if (reuseLater != nullptr) { - reuseLater->reuse(); - } } else { restOfTheData->position(restOfTheData->limit()); } @@ -195,14 +190,11 @@ void Connection::onReceivedData(NativeByteBuffer *buffer) { } else { if (buffer->remaining() < 4) { if (restOfTheData == nullptr || (restOfTheData != nullptr && restOfTheData->position() != 0)) { - NativeByteBuffer *reuseLater = restOfTheData; + reuseLater = restOfTheData; restOfTheData = BuffersStorage::getInstance().getFreeBuffer(16384); restOfTheData->writeBytes(buffer); restOfTheData->limit(restOfTheData->position()); lastPacketLength = 0; - if (reuseLater != nullptr) { - reuseLater->reuse(); - } } else { restOfTheData->position(restOfTheData->limit()); } @@ -223,7 +215,7 @@ void Connection::onReceivedData(NativeByteBuffer *buffer) { if (currentProtocolType != ProtocolTypeDD && currentProtocolType != ProtocolTypeTLS && currentPacketLength % 4 != 0 || currentPacketLength > 2 * 1024 * 1024) { if (LOGS_ENABLED) DEBUG_D("connection(%p, account%u, dc%u, type %d) received invalid packet length", this, currentDatacenter->instanceNum, currentDatacenter->getDatacenterId(), connectionType); reconnect(); - return; + break; } if (currentPacketLength < buffer->remaining()) { @@ -233,8 +225,6 @@ void Connection::onReceivedData(NativeByteBuffer *buffer) { } else { if (LOGS_ENABLED) DEBUG_D("connection(%p, account%u, dc%u, type %d) received packet size less(%u) then message size(%u)", this, currentDatacenter->instanceNum, currentDatacenter->getDatacenterId(), connectionType, buffer->remaining(), currentPacketLength); - NativeByteBuffer *reuseLater = nullptr; - if (restOfTheData != nullptr && restOfTheData->capacity() < len) { reuseLater = restOfTheData; restOfTheData = nullptr; @@ -248,10 +238,7 @@ void Connection::onReceivedData(NativeByteBuffer *buffer) { restOfTheData->limit(len); } lastPacketLength = len; - if (reuseLater != nullptr) { - reuseLater->reuse(); - } - return; + break; } uint32_t old = buffer->limit(); @@ -262,7 +249,7 @@ void Connection::onReceivedData(NativeByteBuffer *buffer) { if (restOfTheData != nullptr) { if ((lastPacketLength != 0 && restOfTheData->position() == lastPacketLength) || (lastPacketLength == 0 && !restOfTheData->hasRemaining())) { - restOfTheData->reuse(); + reuseLater = restOfTheData; restOfTheData = nullptr; } else { restOfTheData->compact(); @@ -276,6 +263,9 @@ void Connection::onReceivedData(NativeByteBuffer *buffer) { parseLaterBuffer = nullptr; } } + if (reuseLater != nullptr) { + reuseLater->reuse(); + } } void Connection::connect() { diff --git a/TMessagesProj/jni/voip/tgcalls/v2/InstanceV2Impl.cpp b/TMessagesProj/jni/voip/tgcalls/v2/InstanceV2Impl.cpp index c3913cd4db6..9da0069447e 100644 --- a/TMessagesProj/jni/voip/tgcalls/v2/InstanceV2Impl.cpp +++ b/TMessagesProj/jni/voip/tgcalls/v2/InstanceV2Impl.cpp @@ -253,6 +253,54 @@ class OutgoingAudioChannel : public sigslot::has_slots<> { bool _isMuted = true; }; +namespace { +class AudioSinkImpl: public webrtc::AudioSinkInterface { +public: + AudioSinkImpl(std::function update) : + _update(update) { + } + + virtual ~AudioSinkImpl() { + } + + virtual void OnData(const Data& audio) override { + if (_update && audio.channels == 1) { + const int16_t *samples = (const int16_t *)audio.data; + int numberOfSamplesInFrame = (int)audio.samples_per_channel; + + int16_t currentPeak = 0; + for (int i = 0; i < numberOfSamplesInFrame; i++) { + int16_t sample = samples[i]; + if (sample < 0) { + sample = -sample; + } + if (_peak < sample) { + _peak = sample; + } + if (currentPeak < sample) { + currentPeak = sample; + } + _peakCount += 1; + } + + if (_peakCount >= 4400) { + float level = ((float)(_peak)) / 8000.0f; + _peak = 0; + _peakCount = 0; + _update(0, level); + } + } + } + +private: + std::function _update; + + int _peakCount = 0; + uint16_t _peak = 0; +}; + +} + class IncomingV2AudioChannel : public sigslot::has_slots<> { public: IncomingV2AudioChannel( @@ -261,6 +309,7 @@ class IncomingV2AudioChannel : public sigslot::has_slots<> { webrtc::RtpTransport *rtpTransport, rtc::UniqueRandomIdGenerator *randomIdGenerator, signaling::MediaContent const &mediaContent, + std::function &&onAudioLevelUpdated, std::shared_ptr threads) : _threads(threads), _ssrc(mediaContent.ssrc), @@ -278,9 +327,7 @@ class IncomingV2AudioChannel : public sigslot::has_slots<> { _threads->getNetworkThread()->BlockingCall([&]() { _audioChannel->SetRtpTransport(rtpTransport); }); - - - + std::vector codecs; for (const auto &payloadType : mediaContent.payloadTypes) { cricket::AudioCodec codec(payloadType.id, payloadType.name, payloadType.clockrate, 0, payloadType.channels); @@ -317,11 +364,14 @@ class IncomingV2AudioChannel : public sigslot::has_slots<> { streamParams.set_stream_ids({ streamId }); incomingAudioDescription->AddStream(streamParams); - threads->getWorkerThread()->BlockingCall([&]() { + threads->getWorkerThread()->BlockingCall([this, &outgoingAudioDescription, &incomingAudioDescription, onAudioLevelUpdated = std::move(onAudioLevelUpdated), ssrc = mediaContent.ssrc]() { _audioChannel->SetPayloadTypeDemuxingEnabled(false); std::string errorDesc; _audioChannel->SetLocalContent(outgoingAudioDescription.get(), webrtc::SdpType::kOffer, errorDesc); _audioChannel->SetRemoteContent(incomingAudioDescription.get(), webrtc::SdpType::kAnswer, errorDesc); + + std::unique_ptr audioLevelSink(new AudioSinkImpl(std::move(onAudioLevelUpdated))); + _audioChannel->media_channel()->SetRawAudioSink(ssrc, std::move(audioLevelSink)); }); outgoingAudioDescription.reset(); @@ -1101,7 +1151,7 @@ class InstanceV2ImplInternal : public std::enable_shared_from_thisGetSupportedFormats(); @@ -1468,6 +1518,9 @@ class InstanceV2ImplInternal : public std::enable_shared_from_this + + @@ -266,6 +268,9 @@ + - - + + diff --git a/TMessagesProj/src/main/assets/darkblue.attheme b/TMessagesProj/src/main/assets/darkblue.attheme index bb945993aa4..8c3eda0baf5 100644 --- a/TMessagesProj/src/main/assets/darkblue.attheme +++ b/TMessagesProj/src/main/assets/darkblue.attheme @@ -429,7 +429,7 @@ chat_searchPanelText=-8796932 chat_inContactIcon=-1 code_comment=-2130706433 chat_outCodeBackground=857487708 -chat_inCodeBackground=856033549 +chat_inCodeBackground=-1 code_keyword=-27776 code_constant=-27776 code_function=-27776 diff --git a/TMessagesProj/src/main/assets/night.attheme b/TMessagesProj/src/main/assets/night.attheme index 1a394741d47..46217ff1c25 100644 --- a/TMessagesProj/src/main/assets/night.attheme +++ b/TMessagesProj/src/main/assets/night.attheme @@ -454,7 +454,7 @@ chat_searchPanelText=-10767620 chat_inContactIcon=-1 code_comment=-2130706433 chat_outCodeBackground=859062986 -chat_inCodeBackground=855638016 +chat_inCodeBackground=-1 code_keyword=-27776 code_constant=-27776 code_function=-27776 diff --git a/TMessagesProj/src/main/java/androidx/recyclerview/widget/ChatListItemAnimator.java b/TMessagesProj/src/main/java/androidx/recyclerview/widget/ChatListItemAnimator.java index 01f4a439fab..66516c7b06a 100644 --- a/TMessagesProj/src/main/java/androidx/recyclerview/widget/ChatListItemAnimator.java +++ b/TMessagesProj/src/main/java/androidx/recyclerview/widget/ChatListItemAnimator.java @@ -5,6 +5,7 @@ import android.animation.AnimatorSet; import android.animation.ObjectAnimator; import android.animation.ValueAnimator; +import android.util.LongSparseArray; import android.view.View; import android.view.ViewPropertyAnimator; import android.view.animation.Interpolator; @@ -14,11 +15,13 @@ import androidx.annotation.Nullable; import androidx.core.view.ViewCompat; +import org.telegram.messenger.AndroidUtilities; import org.telegram.messenger.BuildVars; import org.telegram.messenger.FileLog; import org.telegram.messenger.ImageReceiver; import org.telegram.messenger.MessageObject; import org.telegram.messenger.SharedConfig; +import org.telegram.messenger.Utilities; import org.telegram.ui.ActionBar.Theme; import org.telegram.ui.Cells.BotHelpCell; import org.telegram.ui.Cells.ChatActionCell; @@ -27,6 +30,7 @@ import org.telegram.ui.Components.ChatGreetingsView; import org.telegram.ui.Components.CubicBezierInterpolator; import org.telegram.ui.Components.RecyclerListView; +import org.telegram.ui.Components.ThanosEffect; import org.telegram.ui.TextMessageEnterTransition; import org.telegram.ui.VoiceMessageEnterTransition; @@ -125,9 +129,60 @@ private void runAlphaEnterTransition() { return; } // First, remove stuff + boolean hadThanos = false; + final boolean supportsThanos = getThanosEffectContainer != null && supportsThanosEffectContainer != null && supportsThanosEffectContainer.run(); + if (supportsThanos) { + LongSparseArray> groupsToRemoveWithThanos = null; + for (int i = 0; i < mPendingRemovals.size(); ++i) { + RecyclerView.ViewHolder holder = mPendingRemovals.get(i); + if (toBeSnapped.contains(holder) && holder.itemView instanceof ChatMessageCell && ((ChatMessageCell) holder.itemView).getCurrentMessagesGroup() != null) { + MessageObject msg = ((ChatMessageCell) holder.itemView).getMessageObject(); + if (msg != null && msg.getGroupId() != 0) { + if (groupsToRemoveWithThanos == null) { + groupsToRemoveWithThanos = new LongSparseArray<>(); + } + ArrayList holders = groupsToRemoveWithThanos.get(msg.getGroupId()); + if (holders == null) { + groupsToRemoveWithThanos.put(msg.getGroupId(), holders = new ArrayList<>()); + } + toBeSnapped.remove(holder); + mPendingRemovals.remove(i); + i--; + holders.add(holder); + } + } + } + if (groupsToRemoveWithThanos != null) { + for (int i = 0; i < groupsToRemoveWithThanos.size(); ++i) { + // check whether we remove the whole group + ArrayList holders = groupsToRemoveWithThanos.valueAt(i); + if (holders.size() <= 0) continue; + boolean wholeGroup = true; + RecyclerView.ViewHolder firstHolder = holders.get(0); + if (firstHolder.itemView instanceof ChatMessageCell) { + MessageObject.GroupedMessages group = ((ChatMessageCell) firstHolder.itemView).getCurrentMessagesGroup(); + if (group != null) { + wholeGroup = group.messages.size() <= holders.size(); + } + } + if (!wholeGroup) { + // not whole group, fallback to prev animation + mPendingRemovals.addAll(holders); + } else { + animateRemoveGroupImpl(holders); + hadThanos = true; + } + } + } + } for (RecyclerView.ViewHolder holder : mPendingRemovals) { - animateRemoveImpl(holder); + boolean thanos = toBeSnapped.remove(holder) && supportsThanos; + animateRemoveImpl(holder, thanos); + if (thanos) { + hadThanos = true; + } } + final boolean finalThanos = hadThanos; mPendingRemovals.clear(); // Next, move stuff if (movesPending) { @@ -139,7 +194,7 @@ private void runAlphaEnterTransition() { @Override public void run() { for (MoveInfo moveInfo : moves) { - animateMoveImpl(moveInfo.holder, moveInfo); + animateMoveImpl(moveInfo.holder, moveInfo, finalThanos); } moves.clear(); mMovesList.remove(moves); @@ -147,7 +202,7 @@ public void run() { }; if (delayAnimations && removalsPending) { View view = moves.get(0).holder.itemView; - ViewCompat.postOnAnimationDelayed(view, mover, getMoveAnimationDelay()); + ViewCompat.postOnAnimationDelayed(view, mover, hadThanos ? 0 : getMoveAnimationDelay()); } else { mover.run(); } @@ -661,6 +716,9 @@ public boolean animateMove(RecyclerView.ViewHolder holder, ItemHolderInfo info, @Override protected void animateMoveImpl(RecyclerView.ViewHolder holder, MoveInfo moveInfo) { + animateMoveImpl(holder, moveInfo, false); + } + protected void animateMoveImpl(RecyclerView.ViewHolder holder, MoveInfo moveInfo, boolean withThanos) { int fromX = moveInfo.fromX; int fromY = moveInfo.fromY; int toX = moveInfo.toX; @@ -843,10 +901,12 @@ public void onAnimationEnd(Animator animation) { } } - if (translationInterpolator != null) { + if (withThanos) { + animatorSet.setInterpolator(CubicBezierInterpolator.EASE_OUT); + } else if (translationInterpolator != null) { animatorSet.setInterpolator(translationInterpolator); } - animatorSet.setDuration(getMoveDuration()); + animatorSet.setDuration((long) (getMoveDuration() * (withThanos ? 1.9f : 1f))); animatorSet.addListener(new AnimatorListenerAdapter() { @Override public void onAnimationStart(Animator animator) { @@ -1372,36 +1432,72 @@ public void onAnimationEnd(Animator animator) { animatorSet.start(); } - protected void animateRemoveImpl(final RecyclerView.ViewHolder holder) { + protected void animateRemoveImpl(final RecyclerView.ViewHolder holder, boolean thanos) { if (BuildVars.LOGS_ENABLED) { - FileLog.d("animate remove impl"); + FileLog.d("animate remove impl " + (thanos ? " with thanos" : "")); } final View view = holder.itemView; mRemoveAnimations.add(holder); - ObjectAnimator animator = ObjectAnimator.ofFloat(view, View.ALPHA, view.getAlpha(), 0f); - - dispatchRemoveStarting(holder); - - animator.setDuration(getRemoveDuration()); - animator.addListener( - new AnimatorListenerAdapter() { - - @Override - public void onAnimationEnd(Animator animator) { - animator.removeAllListeners(); - view.setAlpha(1); - view.setScaleX(1f); - view.setScaleY(1f); - view.setTranslationX(0); - view.setTranslationY(0); - if (mRemoveAnimations.remove(holder)) { - dispatchRemoveFinished(holder); - dispatchFinishedWhenDone(); + if (thanos && getThanosEffectContainer != null) { + ThanosEffect thanosEffect = getThanosEffectContainer.run(); + dispatchRemoveStarting(holder); + thanosEffect.animate(view, () -> { + view.setVisibility(View.VISIBLE); + if (mRemoveAnimations.remove(holder)) { + dispatchRemoveFinished(holder); + dispatchFinishedWhenDone(); + } + }); + } else { + ObjectAnimator animator = ObjectAnimator.ofFloat(view, View.ALPHA, view.getAlpha(), 0f); + dispatchRemoveStarting(holder); + animator.setDuration(getRemoveDuration()); + animator.addListener( + new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animator) { + animator.removeAllListeners(); + view.setAlpha(1); + view.setScaleX(1f); + view.setScaleY(1f); + view.setTranslationX(0); + view.setTranslationY(0); + if (mRemoveAnimations.remove(holder)) { + dispatchRemoveFinished(holder); + dispatchFinishedWhenDone(); + } } - } - }); - animators.put(holder, animator); - animator.start(); + }); + animators.put(holder, animator); + animator.start(); + } + recyclerListView.stopScroll(); + } + + private void animateRemoveGroupImpl(final ArrayList holders) { + if (BuildVars.LOGS_ENABLED) { + FileLog.d("animate remove group impl with thanos"); + } + mRemoveAnimations.addAll(holders); + ThanosEffect thanosEffect = getThanosEffectContainer.run(); + for (int i = 0; i < holders.size(); ++i) { + dispatchRemoveStarting(holders.get(i)); + } + final ArrayList views = new ArrayList<>(); + for (int i = 0; i < holders.size(); ++i) { + views.add(holders.get(i).itemView); + } + thanosEffect.animateGroup(views, () -> { + for (int i = 0; i < views.size(); ++i) { + views.get(i).setVisibility(View.VISIBLE); + } + if (mRemoveAnimations.removeAll(holders)) { + for (int i = 0; i < holders.size(); ++i) { + dispatchRemoveFinished(holders.get(i)); + } + dispatchFinishedWhenDone(); + } + }); recyclerListView.stopScroll(); } @@ -1502,5 +1598,27 @@ class ItemHolderInfoExtended extends ItemHolderInfo { int captionX; int captionY; } + + private final ArrayList toBeSnapped = new ArrayList<>(); + public void prepareThanos(RecyclerView.ViewHolder viewHolder) { + if (viewHolder == null) return; + toBeSnapped.add(viewHolder); + if (viewHolder.itemView instanceof ChatMessageCell) { + MessageObject msg = ((ChatMessageCell) viewHolder.itemView).getMessageObject(); + if (msg != null) { + msg.deletedByThanos = true; + } + } + } + + private Utilities.Callback0Return supportsThanosEffectContainer; + private Utilities.Callback0Return getThanosEffectContainer; + public void setOnSnapMessage( + Utilities.Callback0Return supportsThanosEffectContainer, + Utilities.Callback0Return getThanosEffectContainer + ) { + this.supportsThanosEffectContainer = supportsThanosEffectContainer; + this.getThanosEffectContainer = getThanosEffectContainer; + } } diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/AndroidUtilities.java b/TMessagesProj/src/main/java/org/telegram/messenger/AndroidUtilities.java index c7e4b586385..1b0801312dc 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/AndroidUtilities.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/AndroidUtilities.java @@ -4893,6 +4893,10 @@ public static void updateViewVisibilityAnimated(View view, boolean show, float s } public static void updateViewVisibilityAnimated(View view, boolean show, float scaleFactor, boolean goneOnHide, boolean animated) { + updateViewVisibilityAnimated(view, show, scaleFactor, goneOnHide, 1f, animated); + } + + public static void updateViewVisibilityAnimated(View view, boolean show, float scaleFactor, boolean goneOnHide, float maxAlpha, boolean animated) { if (view == null) { return; } @@ -4904,7 +4908,7 @@ public static void updateViewVisibilityAnimated(View view, boolean show, float s view.animate().setListener(null).cancel(); view.setVisibility(show ? View.VISIBLE : (goneOnHide ? View.GONE : View.INVISIBLE)); view.setTag(show ? 1 : null); - view.setAlpha(1f); + view.setAlpha(maxAlpha); view.setScaleX(1f); view.setScaleY(1f); } else if (show && view.getTag() == null) { @@ -4915,7 +4919,7 @@ public static void updateViewVisibilityAnimated(View view, boolean show, float s view.setScaleX(scaleFactor); view.setScaleY(scaleFactor); } - view.animate().alpha(1f).scaleY(1f).scaleX(1f).setDuration(150).start(); + view.animate().alpha(maxAlpha).scaleY(1f).scaleX(1f).setDuration(150).start(); view.setTag(1); } else if (!show && view.getTag() != null) { view.animate().setListener(null).cancel(); @@ -5112,7 +5116,7 @@ public static boolean isEROFS(Exception e) { ); } - public static CharSequence replaceCharSequence(String what, CharSequence from, CharSequence obj) { + public static SpannableStringBuilder replaceCharSequence(String what, CharSequence from, CharSequence obj) { SpannableStringBuilder spannableStringBuilder; if (from instanceof SpannableStringBuilder) { spannableStringBuilder = (SpannableStringBuilder) from; @@ -5241,14 +5245,14 @@ public static void makeGlobalBlurBitmap(Utilities.Callback onBitmapDone, canvas.restore(); } Utilities.stackBlurBitmap(bitmap, Math.max((int) amount, Math.max(w, h) / 180)); - AndroidUtilities.runOnUIThread(() -> { +// AndroidUtilities.runOnUIThread(() -> { onBitmapDone.run(bitmap); - }); +// }); } catch (Exception e) { FileLog.e(e); - AndroidUtilities.runOnUIThread(() -> { +// AndroidUtilities.runOnUIThread(() -> { onBitmapDone.run(null); - }); +// }); } // }); } diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/ApplicationLoader.java b/TMessagesProj/src/main/java/org/telegram/messenger/ApplicationLoader.java index a577695a5a9..f427f8ab3f9 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/ApplicationLoader.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/ApplicationLoader.java @@ -17,6 +17,8 @@ import android.content.IntentFilter; import android.content.SharedPreferences; import android.content.pm.ApplicationInfo; +import android.content.pm.PackageInfo; +import android.content.pm.PackageManager; import android.content.res.Configuration; import android.net.ConnectivityManager; import android.net.Network; @@ -260,7 +262,11 @@ public void onCreate() { if (BuildVars.LOGS_ENABLED) { FileLog.d("app start time = " + (startTime = SystemClock.elapsedRealtime())); - FileLog.d("buildVersion = " + BuildVars.BUILD_VERSION); + try { + FileLog.d("buildVersion = " + ApplicationLoader.applicationContext.getPackageManager().getPackageInfo(ApplicationLoader.applicationContext.getPackageName(), 0).versionCode); + } catch (Exception e) { + FileLog.e(e); + } } if (applicationContext == null) { applicationContext = getApplicationContext(); @@ -350,6 +356,13 @@ private boolean checkPlayServices() { return true; } + private static long lastNetworkCheck = -1; + private static void ensureCurrentNetworkGet() { + final long now = System.currentTimeMillis(); + ensureCurrentNetworkGet(now - lastNetworkCheck > 5000); + lastNetworkCheck = now; + } + private static void ensureCurrentNetworkGet(boolean force) { if (force || currentNetworkInfo == null) { try { @@ -416,6 +429,11 @@ public static boolean isConnectedToWiFi() { return false; } + public static boolean useLessData() { + ensureCurrentNetworkGet(); + return BuildVars.DEBUG_PRIVATE_VERSION && (SharedConfig.forceLessData || isConnectionSlow()); + } + public static boolean isConnectionSlow() { try { ensureCurrentNetworkGet(false); diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/BillingController.java b/TMessagesProj/src/main/java/org/telegram/messenger/BillingController.java index 6731ac65a6c..c55c4741145 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/BillingController.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/BillingController.java @@ -96,7 +96,7 @@ public String formatCurrency(long amount, String currency, int exp) { } public String formatCurrency(long amount, String currency, int exp, boolean rounded) { - if (currency.isEmpty()) { + if (currency == null || currency.isEmpty()) { return String.valueOf(amount); } Currency cur = Currency.getInstance(currency); diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/BuildVars.java b/TMessagesProj/src/main/java/org/telegram/messenger/BuildVars.java index 4c5454d0d63..ea16016083e 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/BuildVars.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/BuildVars.java @@ -24,8 +24,7 @@ public class BuildVars { public static boolean USE_CLOUD_STRINGS = true; public static boolean CHECK_UPDATES = true; public static boolean NO_SCOPED_STORAGE = Build.VERSION.SDK_INT <= 29; - public static int BUILD_VERSION = 4139; - public static String BUILD_VERSION_STRING = "10.3.2"; + public static String BUILD_VERSION_STRING = BuildConfig.BUILD_VERSION_STRING; public static int APP_ID = 4; public static String APP_HASH = "014b35b6184100b085b0d0572f9b5103"; diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/ChatObject.java b/TMessagesProj/src/main/java/org/telegram/messenger/ChatObject.java index 7cb7b1dd196..d3891b27022 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/ChatObject.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/ChatObject.java @@ -2107,13 +2107,13 @@ public static long getEmojiId(TLRPC.Chat chat) { public static int getProfileColorId(TLRPC.Chat chat) { if (chat == null) return 0; -// if (chat.profile_color != null && (chat.profile_color.flags & 1) != 0) return chat.profile_color.color; + if (chat.profile_color != null && (chat.profile_color.flags & 1) != 0) return chat.profile_color.color; return -1; } public static long getProfileEmojiId(TLRPC.Chat chat) { -// if (chat != null && chat.profile_color != null && (chat.profile_color.flags & 2) != 0) return chat.profile_color.background_emoji_id; - return -1; + if (chat != null && chat.profile_color != null && (chat.profile_color.flags & 2) != 0) return chat.profile_color.background_emoji_id; + return 0; } } diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/ChatThemeController.java b/TMessagesProj/src/main/java/org/telegram/messenger/ChatThemeController.java index 29cb2c552fe..e71d7e30b4e 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/ChatThemeController.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/ChatThemeController.java @@ -214,7 +214,7 @@ public static boolean equals(TLRPC.WallPaper wallPaper, TLRPC.WallPaper oldWallp if (wallPaper.uploadingImage != null) { return TextUtils.equals(oldWallpaper.uploadingImage, wallPaper.uploadingImage); } - return wallPaper.id == oldWallpaper.id && TextUtils.equals(ChatBackgroundDrawable.hash(wallPaper.settings), ChatBackgroundDrawable.hash(oldWallpaper.settings)); + return wallPaper.id == oldWallpaper.id && TextUtils.equals(ChatBackgroundDrawable.hash(wallPaper.settings), ChatBackgroundDrawable.hash(oldWallpaper.settings)) && TextUtils.equals(ChatThemeController.getWallpaperEmoticon(wallPaper), ChatThemeController.getWallpaperEmoticon(oldWallpaper)); } return false; } @@ -263,6 +263,10 @@ public EmojiThemes getDialogTheme(long dialogId) { emoticon = getEmojiSharedPreferences().getString("chatTheme_" + currentAccount + "_" + dialogId, null); dialogEmoticonsMap.put(dialogId, emoticon); } + return getTheme(emoticon); + } + + public EmojiThemes getTheme(String emoticon) { if (emoticon != null) { for (EmojiThemes theme : allChatThemes) { if (emoticon.equals(theme.getEmoticon())) { @@ -274,10 +278,10 @@ public EmojiThemes getDialogTheme(long dialogId) { } public void saveChatWallpaper(long dialogId, TLRPC.WallPaper wallPaper) { - if (dialogId < 0) { - return; - } if (wallPaper != null) { + if (wallPaper.document == null) { + return; + } SerializedData data = new SerializedData(wallPaper.getObjectSize()); wallPaper.serializeToStream(data); String wallpaperString = Utilities.bytesToHex(data.toByteArray()); @@ -293,12 +297,16 @@ public void saveChatWallpaper(long dialogId, TLRPC.WallPaper wallPaper) { } public TLRPC.WallPaper getDialogWallpaper(long dialogId) { - if (dialogId < 0) { - return null; - } - TLRPC.UserFull userFull = getMessagesController().getUserFull(dialogId); - if (userFull != null) { - return userFull.wallpaper; + if (dialogId >= 0) { + TLRPC.UserFull userFull = getMessagesController().getUserFull(dialogId); + if (userFull != null) { + return userFull.wallpaper; + } + } else { + TLRPC.ChatFull chatFull = getMessagesController().getChatFull(-dialogId); + if (chatFull != null) { + return chatFull.wallpaper; + } } String wallpaperString = getEmojiSharedPreferences().getString("chatWallpaper_" + currentAccount + "_" + dialogId, null); if (wallpaperString != null) { @@ -411,30 +419,75 @@ public void processUpdate(TLRPC.TL_updatePeerWallpaper update) { return; } final long dialogId = userFull.id; - userFull.wallpaper_overridden = update.wallpaper_overridden; - userFull.wallpaper = update.wallpaper; - userFull.flags |= 16777216; + if ((update.flags & 1) != 0) { + userFull.wallpaper_overridden = update.wallpaper_overridden; + userFull.wallpaper = update.wallpaper; + userFull.flags |= 16777216; + } else { + userFull.wallpaper_overridden = false; + userFull.wallpaper = null; + userFull.flags &=~ 16777216; + } getMessagesStorage().updateUserInfo(userFull, false); - saveChatWallpaper(dialogId, null); + saveChatWallpaper(dialogId, userFull.wallpaper); AndroidUtilities.runOnUIThread(() -> { NotificationCenter.getInstance(currentAccount).postNotificationName(NotificationCenter.userInfoDidLoad, dialogId, userFull); }); } } else { - // todo + TLRPC.ChatFull chatFull = getMessagesController().getChatFull(-DialogObject.getPeerDialogId(update.peer)); + if (chatFull != null) { + if (wallpaperEquals(chatFull.wallpaper, update.wallpaper)) { + return; + } + final long dialogId = -chatFull.id; + if ((update.flags & 1) != 0) { + chatFull.wallpaper = update.wallpaper; + chatFull.flags2 |= 128; + } else { + chatFull.wallpaper = null; + chatFull.flags2 &=~ 128; + } + getMessagesStorage().updateChatInfo(chatFull, false); + saveChatWallpaper(dialogId, chatFull.wallpaper); + AndroidUtilities.runOnUIThread(() -> { + NotificationCenter.getInstance(currentAccount).postNotificationName(NotificationCenter.chatInfoDidLoad, chatFull, 0, false, false); + }); + } } } public static boolean wallpaperEquals(TLRPC.WallPaper a, TLRPC.WallPaper b) { - if ((a == null || a instanceof TLRPC.TL_wallPaperNoFile) && (b == null || b instanceof TLRPC.TL_wallPaperNoFile)) { + if (a == null && b == null) { return true; } if (a instanceof TLRPC.TL_wallPaper && b instanceof TLRPC.TL_wallPaper) { return a.id == b.id; } + if (a instanceof TLRPC.TL_wallPaperNoFile && b instanceof TLRPC.TL_wallPaperNoFile) { + if (a.settings != null && b.settings != null) { + return TextUtils.equals(getWallpaperEmoticon(a), getWallpaperEmoticon(b)); + } + return a.id == b.id; + } return false; } + public static String getWallpaperEmoticon(TLRPC.WallPaper a) { + if (a != null) { + if (a.settings != null && !TextUtils.isEmpty(a.settings.emoticon)) { + return a.settings.emoticon; + } + return ""; + } + return null; + } + + public static boolean isNotEmoticonWallpaper(TLRPC.WallPaper a) { + String emoticon = getWallpaperEmoticon(a); + return emoticon != null && emoticon.length() == 0; + } + public void clearWallpaper(long dialogId, boolean notify) { clearWallpaper(dialogId, notify, false); } @@ -460,6 +513,16 @@ public void clearWallpaper(long dialogId, boolean notify, boolean onlyRevert) { } else { TLRPC.Chat chat = MessagesController.getInstance(currentAccount).getChat(-dialogId); req.peer = MessagesController.getInputPeer(chat); + TLRPC.ChatFull chatFull = getMessagesController().getChatFull(-dialogId); + if (chatFull != null) { + chatFull.wallpaper = null; + chatFull.flags2 &= ~128; + getMessagesStorage().updateChatInfo(chatFull, false); + } + saveChatWallpaper(dialogId, null); + if (notify) { + NotificationCenter.getInstance(currentAccount).postNotificationName(NotificationCenter.chatInfoDidLoad, chatFull, 0, false, false); + } } getConnectionsManager().sendRequest(req, (response, error) -> { @@ -467,13 +530,12 @@ public void clearWallpaper(long dialogId, boolean notify, boolean onlyRevert) { }); } - public int setWallpaperToUser(long dialogId, String wallpaperLocalPath, Theme.OverrideWallpaperInfo wallpaperInfo, MessageObject serverWallpaper, Runnable callback) { + public int setWallpaperToPeer(long dialogId, String wallpaperLocalPath, Theme.OverrideWallpaperInfo wallpaperInfo, MessageObject serverWallpaper, Runnable callback) { TLRPC.TL_messages_setChatWallPaper req = new TLRPC.TL_messages_setChatWallPaper(); - if (dialogId > 0) { + if (dialogId >= 0) { TLRPC.User user = MessagesController.getInstance(currentAccount).getUser(dialogId); req.peer = MessagesController.getInputPeer(user); } else { - //chat not supported yet TLRPC.Chat chat = MessagesController.getInstance(currentAccount).getChat(-dialogId); req.peer = MessagesController.getInputPeer(chat); } @@ -484,52 +546,71 @@ public int setWallpaperToUser(long dialogId, String wallpaperLocalPath, Theme.Ov req.flags |= 2; req.id = serverWallpaper.getId(); - TLRPC.UserFull userFull = MessagesController.getInstance(currentAccount).getUserFull(dialogId); + TLRPC.UserFull userFull = null; + TLRPC.ChatFull chatFull = null; + if (dialogId >= 0) { + userFull = MessagesController.getInstance(currentAccount).getUserFull(dialogId); + } else { + chatFull = MessagesController.getInstance(currentAccount).getChatFull(-dialogId); + } + + TLRPC.TL_messageActionSetChatWallPaper action = (TLRPC.TL_messageActionSetChatWallPaper) serverWallpaper.messageOwner.action; + TLRPC.WallPaper wallPaper = new TLRPC.TL_wallPaper(); + wallPaper.id = action.wallpaper.id; + wallPaper.document = action.wallpaper.document; + wallPaper.settings = new TLRPC.TL_wallPaperSettings(); + wallPaper.settings.intensity = (int) (wallpaperInfo.intensity * 100); + wallPaper.settings.motion = wallpaperInfo.isMotion; + wallPaper.settings.blur = wallpaperInfo.isBlurred; + wallPaper.settings.background_color = wallpaperInfo.color; + wallPaper.settings.second_background_color = wallpaperInfo.gradientColor1; + wallPaper.settings.third_background_color = wallpaperInfo.gradientColor2; + wallPaper.settings.fourth_background_color = wallpaperInfo.gradientColor3; + wallPaper.settings.rotation = wallpaperInfo.rotation; + wallPaper.uploadingImage = wallpaperLocalPath; + TLRPC.WallPaper pastWallpaper = null; if (userFull != null) { - TLRPC.TL_messageActionSetChatWallPaper action = (TLRPC.TL_messageActionSetChatWallPaper) serverWallpaper.messageOwner.action; - TLRPC.WallPaper wallPaper = new TLRPC.TL_wallPaper(); - wallPaper.id = action.wallpaper.id; - wallPaper.document = action.wallpaper.document; - wallPaper.settings = new TLRPC.TL_wallPaperSettings(); - wallPaper.settings.intensity = (int) (wallpaperInfo.intensity * 100); - wallPaper.settings.motion = wallpaperInfo.isMotion; - wallPaper.settings.blur = wallpaperInfo.isBlurred; - wallPaper.settings.background_color = wallpaperInfo.color; - wallPaper.settings.second_background_color = wallpaperInfo.gradientColor1; - wallPaper.settings.third_background_color = wallpaperInfo.gradientColor2; - wallPaper.settings.fourth_background_color = wallpaperInfo.gradientColor3; - wallPaper.settings.rotation = wallpaperInfo.rotation; - wallPaper.uploadingImage = wallpaperLocalPath; - if (userFull.wallpaper != null && userFull.wallpaper.uploadingImage != null && userFull.wallpaper.uploadingImage.equals(wallPaper.uploadingImage)) { - wallPaper.stripedThumb = userFull.wallpaper.stripedThumb; - } + pastWallpaper = userFull.wallpaper; + } else if (chatFull != null) { + pastWallpaper = chatFull.wallpaper; + } + if (pastWallpaper != null && pastWallpaper.uploadingImage != null && pastWallpaper.uploadingImage.equals(wallPaper.uploadingImage)) { + wallPaper.stripedThumb = pastWallpaper.stripedThumb; + } - wallPaper.settings.flags |= 1; - wallPaper.settings.flags |= 8; - wallPaper.settings.flags |= 16; - wallPaper.settings.flags |= 32; - wallPaper.settings.flags |= 64; - - userFull.wallpaper = new TLRPC.TL_wallPaper(); - userFull.wallpaper.pattern = action.wallpaper.pattern; - userFull.wallpaper.id = action.wallpaper.id; - userFull.wallpaper.document = action.wallpaper.document; - userFull.wallpaper.flags = action.wallpaper.flags; - userFull.wallpaper.creator = action.wallpaper.creator; - userFull.wallpaper.dark = action.wallpaper.dark; - userFull.wallpaper.isDefault = action.wallpaper.isDefault; - userFull.wallpaper.slug = action.wallpaper.slug; - userFull.wallpaper.access_hash = action.wallpaper.access_hash; - userFull.wallpaper.stripedThumb = action.wallpaper.stripedThumb; - userFull.wallpaper.settings = wallPaper.settings; - userFull.wallpaper.flags |= 4; + wallPaper.settings.flags |= 1; + wallPaper.settings.flags |= 8; + wallPaper.settings.flags |= 16; + wallPaper.settings.flags |= 32; + wallPaper.settings.flags |= 64; + + TLRPC.TL_wallPaper wallpaper = new TLRPC.TL_wallPaper(); + wallpaper.pattern = action.wallpaper.pattern; + wallpaper.id = action.wallpaper.id; + wallpaper.document = action.wallpaper.document; + wallpaper.flags = action.wallpaper.flags; + wallpaper.creator = action.wallpaper.creator; + wallpaper.dark = action.wallpaper.dark; + wallpaper.isDefault = action.wallpaper.isDefault; + wallpaper.slug = action.wallpaper.slug; + wallpaper.access_hash = action.wallpaper.access_hash; + wallpaper.stripedThumb = action.wallpaper.stripedThumb; + wallpaper.settings = wallPaper.settings; + wallpaper.flags |= 4; + if (userFull != null) { + userFull.wallpaper = wallpaper; userFull.flags |= 16777216; - getMessagesStorage().updateUserInfo(userFull, false); NotificationCenter.getInstance(currentAccount).postNotificationName(NotificationCenter.userInfoDidLoad, dialogId, userFull); - if (callback != null) { - callback.run(); - } + } else if (chatFull != null) { + chatFull.wallpaper = wallpaper; + chatFull.flags2 |= 128; + getMessagesStorage().updateChatInfo(chatFull, false); + NotificationCenter.getInstance(currentAccount).postNotificationName(NotificationCenter.chatInfoDidLoad, chatFull, 0, false, false); + } + + if (callback != null) { + callback.run(); } } else { req.flags |= 1; @@ -543,27 +624,44 @@ public int setWallpaperToUser(long dialogId, String wallpaperLocalPath, Theme.Ov return ConnectionsManager.getInstance(currentAccount).sendRequest(req, (response, error) -> AndroidUtilities.runOnUIThread(() -> { if (response instanceof TLRPC.Updates) { TLRPC.Updates res = (TLRPC.Updates) response; - TLRPC.UserFull userFull = MessagesController.getInstance(currentAccount).getUserFull(dialogId); + TLRPC.UserFull userFull = null; + TLRPC.ChatFull chatFull = null; + if (dialogId >= 0) { + userFull = MessagesController.getInstance(currentAccount).getUserFull(dialogId); + } else { + chatFull = MessagesController.getInstance(currentAccount).getChatFull(-dialogId); + } + TLRPC.WallPaper pastWallpaper = null; if (userFull != null) { - for (int i = 0; i < res.updates.size(); i++) { - if (res.updates.get(i) instanceof TLRPC.TL_updateNewMessage) { - TLRPC.Message message = ((TLRPC.TL_updateNewMessage) res.updates.get(i)).message; - if (message.action instanceof TLRPC.TL_messageActionSetChatWallPaper) { - if (finalApplyOnRequest) { - TLRPC.TL_messageActionSetChatWallPaper actionSetChatWallPaper = (TLRPC.TL_messageActionSetChatWallPaper) message.action; - actionSetChatWallPaper.wallpaper.uploadingImage = wallpaperLocalPath; - if (userFull.wallpaper != null && userFull.wallpaper.uploadingImage != null && userFull.wallpaper.uploadingImage.equals(actionSetChatWallPaper.wallpaper.uploadingImage)) { - actionSetChatWallPaper.wallpaper.stripedThumb = userFull.wallpaper.stripedThumb; - } + pastWallpaper = userFull.wallpaper; + } else if (chatFull != null) { + pastWallpaper = chatFull.wallpaper; + } + for (int i = 0; i < res.updates.size(); i++) { + if (res.updates.get(i) instanceof TLRPC.TL_updateNewMessage) { + TLRPC.Message message = ((TLRPC.TL_updateNewMessage) res.updates.get(i)).message; + if (message.action instanceof TLRPC.TL_messageActionSetChatWallPaper) { + if (finalApplyOnRequest) { + TLRPC.TL_messageActionSetChatWallPaper actionSetChatWallPaper = (TLRPC.TL_messageActionSetChatWallPaper) message.action; + actionSetChatWallPaper.wallpaper.uploadingImage = wallpaperLocalPath; + if (pastWallpaper != null && pastWallpaper.uploadingImage != null && pastWallpaper.uploadingImage.equals(actionSetChatWallPaper.wallpaper.uploadingImage)) { + actionSetChatWallPaper.wallpaper.stripedThumb = pastWallpaper.stripedThumb; + } + if (userFull != null) { userFull.wallpaper = actionSetChatWallPaper.wallpaper; userFull.flags |= 16777216; - saveChatWallpaper(dialogId, userFull.wallpaper); getMessagesStorage().updateUserInfo(userFull, false); NotificationCenter.getInstance(currentAccount).postNotificationName(NotificationCenter.userInfoDidLoad, dialogId, userFull); + } else if (chatFull != null) { + chatFull.wallpaper = actionSetChatWallPaper.wallpaper; + chatFull.flags2 |= 128; + saveChatWallpaper(dialogId, chatFull.wallpaper); + getMessagesStorage().updateChatInfo(chatFull, false); + NotificationCenter.getInstance(currentAccount).postNotificationName(NotificationCenter.chatInfoDidLoad, chatFull, 0, false, false); } - break; } + break; } } } diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/DialogObject.java b/TMessagesProj/src/main/java/org/telegram/messenger/DialogObject.java index 1bf846e3519..8b269fdcffb 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/DialogObject.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/DialogObject.java @@ -163,4 +163,26 @@ public static String getPublicUsername(TLObject dialog) { } return null; } + + public static long getEmojiStatusDocumentId(TLRPC.EmojiStatus emojiStatus) { + if (emojiStatus instanceof TLRPC.TL_emojiStatus) { + return ((TLRPC.TL_emojiStatus) emojiStatus).document_id; + } else if (emojiStatus instanceof TLRPC.TL_emojiStatusUntil && ((TLRPC.TL_emojiStatusUntil) emojiStatus).until > (int) (System.currentTimeMillis() / 1000)) { + return ((TLRPC.TL_emojiStatusUntil) emojiStatus).document_id; + } else { + return 0; + } + } + + public static int getEmojiStatusUntil(TLRPC.EmojiStatus emojiStatus) { + if (emojiStatus instanceof TLRPC.TL_emojiStatusUntil && ((TLRPC.TL_emojiStatusUntil) emojiStatus).until > (int) (System.currentTimeMillis() / 1000)) { + return ((TLRPC.TL_emojiStatusUntil) emojiStatus).until; + } + return 0; + } + + public static boolean emojiStatusesEqual(TLRPC.EmojiStatus a, TLRPC.EmojiStatus b) { + return getEmojiStatusDocumentId(a) == getEmojiStatusDocumentId(b) && getEmojiStatusUntil(a) == getEmojiStatusUntil(b); + } + } diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/FileLoadOperation.java b/TMessagesProj/src/main/java/org/telegram/messenger/FileLoadOperation.java index bb4bdedc304..d1de0978127 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/FileLoadOperation.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/FileLoadOperation.java @@ -2102,7 +2102,7 @@ private void requestReference(RequestInfo requestInfo) { requestingReference = true; if (parentObject instanceof MessageObject) { MessageObject messageObject = (MessageObject) parentObject; - if (messageObject.getId() < 0 && messageObject.messageOwner.media.webpage != null) { + if (messageObject.getId() < 0 && messageObject.messageOwner != null && messageObject.messageOwner.media != null && messageObject.messageOwner.media.webpage != null) { parentObject = messageObject.messageOwner.media.webpage; } } diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/FileLoader.java b/TMessagesProj/src/main/java/org/telegram/messenger/FileLoader.java index 9f4fa3c0860..0ba4d9047df 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/FileLoader.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/FileLoader.java @@ -887,7 +887,7 @@ private FileLoadOperation loadFileInternal(final TLRPC.Document document, final } else { storeDir = getDirectory(type); } - } else if (cacheType == 2) { + } else if (cacheType == ImageLoader.CACHE_TYPE_ENCRYPTED) { operation.setEncryptFile(true); } operation.setPaths(currentAccount, fileName, loaderQueue, storeDir, tempDir, storeFileName); diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/FileRefController.java b/TMessagesProj/src/main/java/org/telegram/messenger/FileRefController.java index 570f956d252..32682a310eb 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/FileRefController.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/FileRefController.java @@ -248,7 +248,7 @@ public void requestReference(Object parentObject, Object... args) { } if (parentObject instanceof MessageObject) { MessageObject messageObject = (MessageObject) parentObject; - if (messageObject.getRealId() < 0 && messageObject.messageOwner.media.webpage != null) { + if (messageObject.getRealId() < 0 && messageObject.messageOwner != null && messageObject.messageOwner.media != null && messageObject.messageOwner.media.webpage != null) { parentObject = messageObject.messageOwner.media.webpage; } } @@ -1087,6 +1087,9 @@ private boolean onRequestComplete(String locationKey, String parentKey, TLObject if (storyItem.media.document != null) { result = getFileReference(storyItem.media.document, requester.location, needReplacement, locationReplacement); } + if (storyItem.media.alt_document != null) { + result = getFileReference(storyItem.media.alt_document, requester.location, needReplacement, locationReplacement); + } } } Object arg = requester.args[1]; diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/ImageLoader.java b/TMessagesProj/src/main/java/org/telegram/messenger/ImageLoader.java index bdec42be099..653409b0b3b 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/ImageLoader.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/ImageLoader.java @@ -103,6 +103,10 @@ */ public class ImageLoader { + public static final int CACHE_TYPE_NONE = 0; + public static final int CACHE_TYPE_CACHE = 1; + public static final int CACHE_TYPE_ENCRYPTED = 2; + private static final boolean DEBUG_MODE = false; private HashMap bitmapUseCounts = new HashMap<>(); @@ -4225,7 +4229,7 @@ public static void saveMessageThumbs(TLRPC.Message message) { } else { File file = FileLoader.getInstance(UserConfig.selectedAccount).getPathToAttach(photoSize, true); boolean isEncrypted = false; - if (MessageObject.shouldEncryptPhotoOrVideo(message)) { + if (MessageObject.shouldEncryptPhotoOrVideo(UserConfig.selectedAccount, message)) { file = new File(file.getAbsolutePath() + ".enc"); isEncrypted = true; } diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/ImageReceiver.java b/TMessagesProj/src/main/java/org/telegram/messenger/ImageReceiver.java index 26d7d02cf2c..bfa725edf21 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/ImageReceiver.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/ImageReceiver.java @@ -1017,7 +1017,7 @@ private void updateDrawableRadius(Drawable drawable) { } else if (bitmapDrawable instanceof AnimatedFileDrawable) { AnimatedFileDrawable animatedFileDrawable = (AnimatedFileDrawable) drawable; animatedFileDrawable.setRoundRadius(roundRadius); - } else if (bitmapDrawable.getBitmap() != null) { + } else if (bitmapDrawable.getBitmap() != null && !bitmapDrawable.getBitmap().isRecycled()) { setDrawableShader(drawable, new BitmapShader(bitmapDrawable.getBitmap(), Shader.TileMode.CLAMP, Shader.TileMode.CLAMP)); } } diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/LiteMode.java b/TMessagesProj/src/main/java/org/telegram/messenger/LiteMode.java index c48221982b2..f299997c90d 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/LiteMode.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/LiteMode.java @@ -39,7 +39,8 @@ public class LiteMode { public static final int FLAG_CHAT_SPOILER = 128; public static final int FLAG_CHAT_BLUR = 256; public static final int FLAG_CHAT_SCALE = 32768; - public static final int FLAGS_CHAT = FLAG_CHAT_BACKGROUND | FLAG_CHAT_FORUM_TWOCOLUMN | FLAG_CHAT_SPOILER | FLAG_CHAT_BLUR | FLAG_CHAT_SCALE; + public static final int FLAG_CHAT_THANOS = 65536; + public static final int FLAGS_CHAT = FLAG_CHAT_BACKGROUND | FLAG_CHAT_FORUM_TWOCOLUMN | FLAG_CHAT_SPOILER | FLAG_CHAT_BLUR | FLAG_CHAT_SCALE | FLAG_CHAT_THANOS; public static final int FLAG_CALLS_ANIMATIONS = 512; public static final int FLAG_AUTOPLAY_VIDEOS = 1024; @@ -49,8 +50,9 @@ public class LiteMode { FLAG_ANIMATED_EMOJI_CHAT_PREMIUM | FLAG_ANIMATED_EMOJI_KEYBOARD_PREMIUM | FLAG_ANIMATED_EMOJI_REACTIONS_PREMIUM | - FLAG_AUTOPLAY_GIFS - ); // 2076 + FLAG_AUTOPLAY_GIFS | + FLAG_CHAT_THANOS + ); // 67612 public static int PRESET_MEDIUM = ( FLAGS_ANIMATED_STICKERS | FLAG_ANIMATED_EMOJI_KEYBOARD_PREMIUM | @@ -59,16 +61,18 @@ public class LiteMode { FLAG_CHAT_FORUM_TWOCOLUMN | FLAG_CALLS_ANIMATIONS | FLAG_AUTOPLAY_VIDEOS | - FLAG_AUTOPLAY_GIFS - ); // 7775 + FLAG_AUTOPLAY_GIFS | + FLAG_CHAT_THANOS + ); // 73311 public static int PRESET_HIGH = ( FLAGS_ANIMATED_STICKERS | FLAGS_ANIMATED_EMOJI | FLAGS_CHAT | FLAG_CALLS_ANIMATIONS | FLAG_AUTOPLAY_VIDEOS | - FLAG_AUTOPLAY_GIFS - ); // 65535 + FLAG_AUTOPLAY_GIFS | + FLAG_CHAT_THANOS + ); // 131071 public static int PRESET_POWER_SAVER = 0; private static int BATTERY_LOW = 10; @@ -196,8 +200,12 @@ public static void loadPreference() { } final SharedPreferences preferences = MessagesController.getGlobalMainSettings(); - if (!preferences.contains("lite_mode2")) { - if (preferences.contains("lite_mode")) { + if (!preferences.contains("lite_mode3")) { + if (preferences.contains("lite_mode2")) { + defaultValue = preferences.getInt("lite_mode2", defaultValue); + defaultValue |= FLAG_CHAT_THANOS; + preferences.edit().putInt("lite_mode3", defaultValue).apply(); + } else if (preferences.contains("lite_mode")) { defaultValue = preferences.getInt("lite_mode", defaultValue); if (defaultValue == 4095) { defaultValue = PRESET_HIGH; @@ -248,7 +256,7 @@ public static void loadPreference() { } int prevValue = value; - value = preferences.getInt("lite_mode2", defaultValue); + value = preferences.getInt("lite_mode3", defaultValue); if (loaded) { onFlagsUpdate(prevValue, value); } @@ -257,7 +265,7 @@ public static void loadPreference() { } public static void savePreference() { - MessagesController.getGlobalMainSettings().edit().putInt("lite_mode2", value).putInt("lite_mode_battery_level", powerSaverLevel).apply(); + MessagesController.getGlobalMainSettings().edit().putInt("lite_mode3", value).putInt("lite_mode_battery_level", powerSaverLevel).apply(); } public static int getPowerSaverLevel() { diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/LocaleController.java b/TMessagesProj/src/main/java/org/telegram/messenger/LocaleController.java index 8593d8d63f1..50c45f8c01d 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/LocaleController.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/LocaleController.java @@ -1061,10 +1061,10 @@ public static String getCurrentLanguageName() { } private String getStringInternal(String key, int res) { - return getStringInternal(key, null, res); + return getStringInternal(key, null, 0, res); } - private String getStringInternal(String key, String fallback, int res) { + private String getStringInternal(String key, String fallback, int fallbackRes, int res) { String value = BuildVars.USE_CLOUD_STRINGS ? localeValues.get(key) : null; if (value == null) { if (BuildVars.USE_CLOUD_STRINGS && fallback != null) { @@ -1074,6 +1074,11 @@ private String getStringInternal(String key, String fallback, int res) { try { value = ApplicationLoader.applicationContext.getString(res); } catch (Exception e) { + if (fallbackRes != 0) { + try { + value = ApplicationLoader.applicationContext.getString(fallbackRes); + } catch (Exception ignored) {} + } FileLog.e(e); } } @@ -1107,8 +1112,12 @@ public static String getString(String key, int res) { return getInstance().getStringInternal(key, res); } + public static String getString(String key, String fallback, int fallbackRes, int res) { + return getInstance().getStringInternal(key, fallback, fallbackRes, res); + } + public static String getString(String key, String fallback, int res) { - return getInstance().getStringInternal(key, fallback, res); + return getInstance().getStringInternal(key, fallback, 0, res); } public static String getString(String key) { @@ -1129,7 +1138,8 @@ public static String getPluralString(String key, int plural) { String param = getInstance().stringForQuantity(getInstance().currentPluralRules.quantityForNumber(plural)); param = key + "_" + param; int resourceId = ApplicationLoader.applicationContext.getResources().getIdentifier(param, "string", ApplicationLoader.applicationContext.getPackageName()); - return getString(param, key + "_other", resourceId); + int fallbackResourceId = ApplicationLoader.applicationContext.getResources().getIdentifier(key + "_other", "string", ApplicationLoader.applicationContext.getPackageName()); + return getString(param, key + "_other", resourceId, fallbackResourceId); } public static String formatPluralString(String key, int plural, Object... args) { @@ -1146,6 +1156,10 @@ public static String formatPluralString(String key, int plural, Object... args) return formatString(param, key + "_other", resourceId, fallbackResourceId, argsWithPlural); } + public static String getStringParamForNumber(int number) { + return getInstance().stringForQuantity(getInstance().currentPluralRules.quantityForNumber(number)); + } + public static String formatPluralStringComma(String key, int plural) { return formatPluralStringComma(key, plural, ','); } @@ -1611,6 +1625,24 @@ public static String formatDateChat(long date, boolean checkYear) { return "LOC_ERR: formatDateChat"; } + public static String formatSmallDateChat(long date) { + try { + Calendar calendar = Calendar.getInstance(); + calendar.setTimeInMillis(System.currentTimeMillis()); + int currentYear = calendar.get(Calendar.YEAR); + date *= 1000; + + calendar.setTimeInMillis(date); + if (currentYear == calendar.get(Calendar.YEAR)) { + return getInstance().formatterDayMonth.format(date); + } + return getInstance().formatterDayMonth.format(date) + ", " + calendar.get(Calendar.YEAR); + } catch (Exception e) { + FileLog.e(e); + } + return "LOC_ERR: formatDateChat"; + } + public static String formatDate(long date) { try { date *= 1000; diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/MediaController.java b/TMessagesProj/src/main/java/org/telegram/messenger/MediaController.java index bf26ad48b65..5911e370468 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/MediaController.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/MediaController.java @@ -779,6 +779,7 @@ public void run() { private int sendAfterDone; private boolean sendAfterDoneNotify; private int sendAfterDoneScheduleDate; + private boolean sendAfterDoneOnce; private Runnable recordStartRunnable; private DispatchQueue recordQueue; @@ -867,7 +868,7 @@ public void run() { } else { recordBuffers.add(buffer); if (sendAfterDone != 3) { - stopRecordingInternal(sendAfterDone, sendAfterDoneNotify, sendAfterDoneScheduleDate); + stopRecordingInternal(sendAfterDone, sendAfterDoneNotify, sendAfterDoneScheduleDate, sendAfterDoneOnce); } } } @@ -1133,7 +1134,7 @@ public void onCallStateChanged(final int state, String incomingNumber) { if (isPlayingMessage(playingMessageObject) && !isMessagePaused()) { pauseMessage(playingMessageObject); } else if (recordStartRunnable != null || recordingAudio != null) { - stopRecording(2, false, 0); + stopRecording(2, false, 0, false); } EmbedBottomSheet embedBottomSheet = EmbedBottomSheet.getInstance(); if (embedBottomSheet != null) { @@ -1585,6 +1586,7 @@ public void didReceivedNotification(int id, int account, Object... args) { int addedCount = 0; for (int a = 0; a < arr.size(); a++) { MessageObject message = arr.get(a); + if (message.isVoiceOnce()) continue; if (playlistMap.containsKey(message.getId())) { continue; } @@ -1618,7 +1620,7 @@ public void didReceivedNotification(int id, int account, Object... args) { ArrayList arr = (ArrayList) args[1]; for (int a = 0; a < arr.size(); a++) { messageObject = arr.get(a); - if ((messageObject.isVoice() || messageObject.isRoundVideo()) && (!voiceMessagesPlaylistUnread || messageObject.isContentUnread() && !messageObject.isOut())) { + if ((messageObject.isVoice() || messageObject.isRoundVideo()) && !messageObject.isVoiceOnce() && !messageObject.isRoundOnce() && (!voiceMessagesPlaylistUnread || messageObject.isContentUnread() && !messageObject.isOut())) { voiceMessagesPlaylist.add(messageObject); voiceMessagesPlaylistMap.put(messageObject.getId(), messageObject); } @@ -1913,7 +1915,7 @@ public void onSensorChanged(SensorEvent event) { if (BuildVars.LOGS_ENABLED) { FileLog.d("stop record"); } - stopRecording(2, false, 0); + stopRecording(2, false, 0, false); raiseToEarRecord = false; ignoreOnPause = false; // if (!ignoreAccelerometerGestures() && proximityHasDifferentValues && proximityWakeLock != null && proximityWakeLock.isHeld()) { @@ -2051,7 +2053,7 @@ public void stopRaiseToEarSensors(ChatActivity chatActivity, boolean fromChat, b return; } if (stopRecording) { - stopRecording(fromChat ? 2 : 0, false, 0); + stopRecording(fromChat ? 2 : 0, false, 0, false); } if (!sensorsStarted || ignoreOnPause || accelerometerSensor == null && (gravitySensor == null || linearAcceleration == null) || proximitySensor == null || raiseChat != chatActivity) { return; @@ -2361,6 +2363,7 @@ public void loadMoreMusic() { int addedCount = 0; for (int i = 0; i < n; i++) { MessageObject messageObject = new MessageObject(currentAccount, res.messages.get(i), false, true); + if (messageObject.isVoiceOnce()) continue; if (playlistMap.containsKey(messageObject.getId())) { continue; } @@ -3761,6 +3764,9 @@ public ArrayList getPlaylist() { } public boolean isPlayingMessage(MessageObject messageObject) { + if (messageObject != null && messageObject.isRepostPreview) { + return false; + } if (audioPlayer == null && videoPlayer == null || messageObject == null || playingMessageObject == null) { return false; } @@ -3954,7 +3960,7 @@ public void generateWaveform(MessageObject messageObject) { }); } - private void stopRecordingInternal(final int send, boolean notify, int scheduleDate) { + private void stopRecordingInternal(final int send, boolean notify, int scheduleDate, boolean once) { if (send != 0) { final TLRPC.TL_document audioToSend = recordingAudio; final File recordingAudioFileToSend = recordingAudioFile; @@ -3987,7 +3993,7 @@ private void stopRecordingInternal(final int send, boolean notify, int scheduleD audioToSend.attributes.add(attributeAudio); if (duration > 700) { if (send == 1) { - SendMessagesHelper.SendMessageParams params = SendMessagesHelper.SendMessageParams.of(audioToSend, null, recordingAudioFileToSend.getAbsolutePath(), recordDialogId, recordReplyingMsg, recordReplyingTopMsg, null, null, null, null, notify, scheduleDate, 0, null, null, false); + SendMessagesHelper.SendMessageParams params = SendMessagesHelper.SendMessageParams.of(audioToSend, null, recordingAudioFileToSend.getAbsolutePath(), recordDialogId, recordReplyingMsg, recordReplyingTopMsg, null, null, null, null, notify, scheduleDate, once ? 0x7FFFFFFF : 0, null, null, false); params.replyToStoryItem = recordReplyingStory; SendMessagesHelper.getInstance(recordingCurrentAccount).sendMessage(params); } @@ -4020,7 +4026,7 @@ private void stopRecordingInternal(final int send, boolean notify, int scheduleD manualRecording = false; } - public void stopRecording(final int send, boolean notify, int scheduleDate) { + public void stopRecording(final int send, boolean notify, int scheduleDate, boolean once) { if (recordStartRunnable != null) { recordQueue.cancelRunnable(recordStartRunnable); recordStartRunnable = null; @@ -4028,7 +4034,7 @@ public void stopRecording(final int send, boolean notify, int scheduleDate) { recordQueue.postRunnable(() -> { if (sendAfterDone == 3) { sendAfterDone = 0; - stopRecordingInternal(send, notify, scheduleDate); + stopRecordingInternal(send, notify, scheduleDate, once); return; } if (audioRecorder == null) { @@ -4038,6 +4044,7 @@ public void stopRecording(final int send, boolean notify, int scheduleDate) { sendAfterDone = send; sendAfterDoneNotify = notify; sendAfterDoneScheduleDate = scheduleDate; + sendAfterDoneOnce = once; audioRecorder.stop(); setBluetoothScoOn(false); } catch (Exception e) { @@ -4050,7 +4057,7 @@ public void stopRecording(final int send, boolean notify, int scheduleDate) { } } if (send == 0) { - stopRecordingInternal(0, false, 0); + stopRecordingInternal(0, false, 0, false); } try { feedbackView.performHapticFeedback(HapticFeedbackConstants.KEYBOARD_TAP, HapticFeedbackConstants.FLAG_IGNORE_GLOBAL_SETTING); @@ -5018,12 +5025,13 @@ public static void loadGalleryPhotosAlbums(final int guid) { thread.start(); } + public static boolean forceBroadcastNewPhotos; private static void broadcastNewPhotos(final int guid, final ArrayList mediaAlbumsSorted, final ArrayList photoAlbumsSorted, final Integer cameraAlbumIdFinal, final AlbumEntry allMediaAlbumFinal, final AlbumEntry allPhotosAlbumFinal, final AlbumEntry allVideosAlbumFinal, int delay) { if (broadcastPhotosRunnable != null) { AndroidUtilities.cancelRunOnUIThread(broadcastPhotosRunnable); } AndroidUtilities.runOnUIThread(broadcastPhotosRunnable = () -> { - if (PhotoViewer.getInstance().isVisible()) { + if (PhotoViewer.getInstance().isVisible() && !forceBroadcastNewPhotos) { broadcastNewPhotos(guid, mediaAlbumsSorted, photoAlbumsSorted, cameraAlbumIdFinal, allMediaAlbumFinal, allPhotosAlbumFinal, allVideosAlbumFinal, 1000); return; } @@ -5376,20 +5384,8 @@ public void didWriteData(long availableSize, float progress) { framerate, bitrate, originalBitrate, startTime, endTime, avatarStartTime, needCompress, duration, - info.filterState, - info.paintPath, - info.blurPath, - info.mediaEntities, - info.isPhoto, - info.cropState, - info.roundVideo, callback, - info.gradientTopColor, - info.gradientBottomColor, - info.muted, - info.isStory, - info.hdrInfo, - info.parts); + info); convertVideoParams.soundInfos.addAll(info.mixedSoundInfos); boolean error = videoConvertor.convertVideo(convertVideoParams); diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/MediaDataController.java b/TMessagesProj/src/main/java/org/telegram/messenger/MediaDataController.java index c577da7e100..7b597bcab06 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/MediaDataController.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/MediaDataController.java @@ -260,6 +260,7 @@ private void loadRepliesOfDraftReplies(final ArrayList messages) private LongSparseArray groupStickerSets = new LongSparseArray<>(); private ConcurrentHashMap stickerSetsByName = new ConcurrentHashMap<>(100, 1.0f, 1); private TLRPC.TL_messages_stickerSet stickerSetDefaultStatuses = null; + private TLRPC.TL_messages_stickerSet stickerSetDefaultChannelStatuses = null; private HashMap diceStickerSetsByEmoji = new HashMap<>(); private LongSparseArray diceEmojiStickerSetsById = new LongSparseArray<>(); private HashSet loadingDiceStickerSets = new HashSet<>(); @@ -317,11 +318,11 @@ private void loadRepliesOfDraftReplies(final ArrayList messages) public final ArrayList premiumPreviewStickers = new ArrayList<>(); boolean previewStickersLoading; - private long[] emojiStatusesHash = new long[2]; - private ArrayList[] emojiStatuses = new ArrayList[2]; - private Long[] emojiStatusesFetchDate = new Long[2]; - private boolean[] emojiStatusesFromCacheFetched = new boolean[2]; - private boolean[] emojiStatusesFetching = new boolean[2]; + private long[] emojiStatusesHash = new long[4]; + private ArrayList[] emojiStatuses = new ArrayList[4]; + private Long[] emojiStatusesFetchDate = new Long[4]; + private boolean[] emojiStatusesFromCacheFetched = new boolean[4]; + private boolean[] emojiStatusesFetching = new boolean[4]; public void cleanup() { for (int a = 0; a < recentStickers.length; a++) { @@ -1217,6 +1218,8 @@ public TLRPC.TL_messages_stickerSet getStickerSet(TLRPC.InputStickerSet inputSti cacheSet = stickerSetsByName.get(inputStickerSet.short_name.toLowerCase()); } else if (inputStickerSet instanceof TLRPC.TL_inputStickerSetEmojiDefaultStatuses && stickerSetDefaultStatuses != null) { cacheSet = stickerSetDefaultStatuses; + } else if (inputStickerSet instanceof TLRPC.TL_inputStickerSetEmojiChannelDefaultStatuses && stickerSetDefaultChannelStatuses != null) { + cacheSet = stickerSetDefaultChannelStatuses; } if (cacheSet != null) { if (onResponse != null) { @@ -1264,6 +1267,9 @@ public TLRPC.TL_messages_stickerSet getStickerSet(TLRPC.InputStickerSet inputSti if (inputStickerSet instanceof TLRPC.TL_inputStickerSetEmojiDefaultStatuses) { stickerSetDefaultStatuses = set; } + if (inputStickerSet instanceof TLRPC.TL_inputStickerSetEmojiDefaultStatuses) { + stickerSetDefaultChannelStatuses = set; + } } saveStickerSetIntoCache(set); getNotificationCenter().postNotificationName(NotificationCenter.groupStickersDidLoad, set.set.id, set); @@ -5567,6 +5573,13 @@ public void loadReplyMessagesForMessages(ArrayList messages, long } } + if (channelId != 0 && messageObject.getDialogId() != -channelId) { + TLRPC.Chat chat = MessagesController.getInstance(currentAccount).getChat(channelId); + if (chat != null && !ChatObject.isPublic(chat)) { + continue; + } + } + SparseArray> sparseArray = replyMessageOwners.get(dialogId); ArrayList ids = dialogReplyMessagesIds.get(channelId); if (sparseArray == null) { @@ -8180,6 +8193,16 @@ public ArrayList getDefaultEmojiStatuses() { return emojiStatuses[type]; } + public ArrayList getDefaultChannelEmojiStatuses() { + final int type = 2; // default channel + if (!emojiStatusesFromCacheFetched[type]) { + fetchEmojiStatuses(type, true); + } else if (emojiStatuses[type] == null || emojiStatusesFetchDate[type] != null && (System.currentTimeMillis() / 1000 - emojiStatusesFetchDate[type]) > 60 * 30) { + fetchEmojiStatuses(type, false); + } + return emojiStatuses[type]; + } + public ArrayList getRecentEmojiStatuses() { final int type = 0; // recent if (!emojiStatusesFromCacheFetched[type]) { @@ -8271,10 +8294,14 @@ public void fetchEmojiStatuses(int type, boolean cache) { TLRPC.TL_account_getRecentEmojiStatuses recentReq = new TLRPC.TL_account_getRecentEmojiStatuses(); recentReq.hash = emojiStatusesHash[type]; req = recentReq; - } else { + } else if (type == 1) { TLRPC.TL_account_getDefaultEmojiStatuses defaultReq = new TLRPC.TL_account_getDefaultEmojiStatuses(); defaultReq.hash = emojiStatusesHash[type]; req = defaultReq; + } else { + TLRPC.TL_account_getChannelDefaultEmojiStatuses defaultReq = new TLRPC.TL_account_getChannelDefaultEmojiStatuses(); + defaultReq.hash = emojiStatusesHash[type]; + req = defaultReq; } ConnectionsManager.getInstance(currentAccount).sendRequest(req, (res, err) -> { emojiStatusesFetchDate[type] = System.currentTimeMillis() / 1000; @@ -8518,4 +8545,42 @@ public void loadReplyIcons() { })); } } + + public TLRPC.TL_emojiList restrictedStatusEmojis; + public void loadRestrictedStatusEmojis() { + SharedPreferences preferences = ApplicationLoader.applicationContext.getSharedPreferences("restrictedstatuses_" + currentAccount, Context.MODE_PRIVATE); + + String value = preferences.getString("restrictedstatuses", null); + long lastCheckTime = preferences.getLong("restrictedstatuses_last_check", 0); + + TLRPC.TL_emojiList emojiList = null; + if (value != null) { + SerializedData serializedData = new SerializedData(Utilities.hexToBytes(value)); + try { + emojiList = (TLRPC.TL_emojiList) TLRPC.TL_emojiList.TLdeserialize(serializedData, serializedData.readInt32(true), true); + restrictedStatusEmojis = emojiList; + } catch (Throwable e) { + FileLog.e(e); + } + } + + if (emojiList == null || (System.currentTimeMillis() - lastCheckTime) > 24 * 60 * 60 * 1000) { + TLRPC.TL_account_getChannelRestrictedStatusEmojis req = new TLRPC.TL_account_getChannelRestrictedStatusEmojis(); + if (emojiList != null) { + req.hash = emojiList.hash; + } + getConnectionsManager().sendRequest(req, (response, error) -> AndroidUtilities.runOnUIThread(() -> { + if (response instanceof TLRPC.TL_emojiList) { + SerializedData data = new SerializedData(response.getObjectSize()); + response.serializeToStream(data); + SharedPreferences.Editor editor = preferences.edit(); + restrictedStatusEmojis = (TLRPC.TL_emojiList) response; + editor.putString("restrictedstatuses", Utilities.bytesToHex(data.toByteArray())); + editor.putLong("restrictedstatuses_last_check", System.currentTimeMillis()); + + editor.apply(); + } + })); + } + } } diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/MessageObject.java b/TMessagesProj/src/main/java/org/telegram/messenger/MessageObject.java index 24c2fa93f27..5f4165cc890 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/MessageObject.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/MessageObject.java @@ -69,6 +69,7 @@ import org.telegram.ui.Components.URLSpanReplacement; import org.telegram.ui.Components.URLSpanUserMention; import org.telegram.ui.Components.spoilers.SpoilerEffect; +import org.telegram.ui.PeerColorActivity; import java.io.BufferedReader; import java.io.File; @@ -122,6 +123,7 @@ public class MessageObject { public static final int TYPE_GIFT_PREMIUM_CHANNEL = 25; public static final int TYPE_GIVEAWAY = 26; public static final int TYPE_JOINED_CHANNEL = 27; // recommendations list + public static final int TYPE_GIVEAWAY_RESULTS = 28; public int localType; public String localName; @@ -165,6 +167,7 @@ public class MessageObject { public String dateKey; public String monthKey; public boolean deleted; + public boolean deletedByThanos; public float audioProgress; public float forceSeekTo = -1; public int audioProgressMs; @@ -212,7 +215,6 @@ public class MessageObject { public boolean replyTextRevealed; public int overrideLinkColor = -1; public long overrideLinkEmoji = -1; - public MessagesController.PeerColor overrideProfilePeerColor; private boolean channelJoined; public boolean channelJoinedExpanded; @@ -305,6 +307,8 @@ public class MessageObject { " & ", " . " }; + public boolean isRepostPreview; + public boolean isRepostVideoPreview; public boolean forceAvatar; public Drawable customAvatarDrawable; @@ -441,7 +445,14 @@ public int getEmojiOnlyCount() { } public boolean hasMediaSpoilers() { - return messageOwner.media != null && messageOwner.media.spoiler || needDrawBluredPreview(); + return !isRepostPreview && (messageOwner.media != null && messageOwner.media.spoiler || needDrawBluredPreview()); + } + + public boolean shouldDrawReactions() { + if (isRepostPreview) { + return false; + } + return true; } public boolean shouldDrawReactionsInLayout() { @@ -1487,8 +1498,15 @@ public MessageObject(int accountNum, TLRPC.Message message, AbstractMap users, AbstractMap chats, LongSparseArray sUsers, LongSparseArray sChats, boolean generateLayout, boolean checkMediaExists, long eid) { + this(accountNum, message, replyToMessage, users, chats, sUsers, sChats, generateLayout, checkMediaExists, eid, false, false); + } + + public MessageObject(int accountNum, TLRPC.Message message, MessageObject replyToMessage, AbstractMap users, AbstractMap chats, LongSparseArray sUsers, LongSparseArray sChats, boolean generateLayout, boolean checkMediaExists, long eid, boolean isRepostPreview, boolean isRepostVideoPreview) { Theme.createCommonMessageResources(); + this.isRepostPreview = isRepostPreview; + this.isRepostVideoPreview = isRepostVideoPreview; + currentAccount = accountNum; messageOwner = message; replyMessageObject = replyToMessage; @@ -2696,6 +2714,133 @@ public MessageObject(int accountNum, TLRPC.TL_channelAdminLogEvent event, ArrayL } else if (event.action instanceof TLRPC.TL_channelAdminLogEventActionChangeColor) { TLRPC.TL_channelAdminLogEventActionChangeColor action = (TLRPC.TL_channelAdminLogEventActionChangeColor) event.action; messageText = replaceWithLink(LocaleController.formatString(R.string.EventLogChangedColor, AvatarDrawable.colorName(action.prev_value).toLowerCase(), AvatarDrawable.colorName(action.new_value).toLowerCase()), "un1", fromUser); + } else if (event.action instanceof TLRPC.TL_channelAdminLogEventActionChangePeerColor) { + TLRPC.TL_channelAdminLogEventActionChangePeerColor action = (TLRPC.TL_channelAdminLogEventActionChangePeerColor) event.action; + SpannableStringBuilder ssb = new SpannableStringBuilder(LocaleController.getString(R.string.EventLogChangedPeerColorIcon)); + + SpannableStringBuilder prev = new SpannableStringBuilder(); + if ((action.prev_value.flags & 1) != 0) { + prev.append("c"); + prev.setSpan(new PeerColorActivity.PeerColorSpan(false, currentAccount, action.prev_value.color).setSize(dp(18)), prev.length() - 1, prev.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + } + if ((action.prev_value.flags & 2) != 0) { + if (prev.length() > 0) + prev.append(", "); + prev.append("e"); + prev.setSpan(new AnimatedEmojiSpan(action.prev_value.background_emoji_id, Theme.chat_actionTextPaint.getFontMetricsInt()), prev.length() - 1, prev.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + } + if (prev.length() == 0) { + prev.append(LocaleController.getString(R.string.EventLogEmojiNone)); + } + + SpannableStringBuilder next = new SpannableStringBuilder(); + if ((action.new_value.flags & 1) != 0) { + next.append("c"); + next.setSpan(new PeerColorActivity.PeerColorSpan(false, currentAccount, action.new_value.color).setSize(dp(18)), next.length() - 1, next.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + } + if ((action.new_value.flags & 2) != 0) { + if (next.length() > 0) + next.append(", "); + next.append("e"); + next.setSpan(new AnimatedEmojiSpan(action.new_value.background_emoji_id, Theme.chat_actionTextPaint.getFontMetricsInt()), next.length() - 1, next.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + } + if (next.length() == 0) { + next.append(LocaleController.getString(R.string.EventLogEmojiNone)); + } + + ssb = AndroidUtilities.replaceCharSequence("%1$s", ssb, prev); + ssb = AndroidUtilities.replaceCharSequence("%2$s", ssb, next); + + messageText = replaceWithLink(ssb, "un1", fromUser); + } else if (event.action instanceof TLRPC.TL_channelAdminLogEventActionChangeProfilePeerColor) { + TLRPC.TL_channelAdminLogEventActionChangeProfilePeerColor action = (TLRPC.TL_channelAdminLogEventActionChangeProfilePeerColor) event.action; + SpannableStringBuilder ssb = new SpannableStringBuilder(LocaleController.getString(R.string.EventLogChangedProfileColorIcon)); + + SpannableStringBuilder prev = new SpannableStringBuilder(); + if ((action.prev_value.flags & 1) != 0) { + prev.append("c"); + prev.setSpan(new PeerColorActivity.PeerColorSpan(true, currentAccount, action.prev_value.color).setSize(dp(18)), prev.length() - 1, prev.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + } + if ((action.prev_value.flags & 2) != 0) { + if (prev.length() > 0) + prev.append(", "); + prev.append("e"); + prev.setSpan(new AnimatedEmojiSpan(action.prev_value.background_emoji_id, Theme.chat_actionTextPaint.getFontMetricsInt()), prev.length() - 1, prev.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + } + if (prev.length() == 0) { + prev.append(LocaleController.getString(R.string.EventLogEmojiNone)); + } + + SpannableStringBuilder next = new SpannableStringBuilder(); + if ((action.new_value.flags & 1) != 0) { + next.append("c"); + next.setSpan(new PeerColorActivity.PeerColorSpan(true, currentAccount, action.new_value.color).setSize(dp(18)), next.length() - 1, next.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + } + if ((action.new_value.flags & 2) != 0) { + if (next.length() > 0) + next.append(", "); + next.append("e"); + next.setSpan(new AnimatedEmojiSpan(action.new_value.background_emoji_id, Theme.chat_actionTextPaint.getFontMetricsInt()), next.length() - 1, next.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + } + if (next.length() == 0) { + next.append(LocaleController.getString(R.string.EventLogEmojiNone)); + } + + ssb = AndroidUtilities.replaceCharSequence("%1$s", ssb, prev); + ssb = AndroidUtilities.replaceCharSequence("%2$s", ssb, next); + + messageText = replaceWithLink(ssb, "un1", fromUser); + } else if (event.action instanceof TLRPC.TL_channelAdminLogEventActionChangeEmojiStatus) { + TLRPC.TL_channelAdminLogEventActionChangeEmojiStatus action = (TLRPC.TL_channelAdminLogEventActionChangeEmojiStatus) event.action; + + boolean prevNone = false; + SpannableString prev; + if (action.prev_value instanceof TLRPC.TL_emojiStatusEmpty) { + prev = new SpannableString(LocaleController.getString(R.string.EventLogEmojiNone)); + prevNone = true; + } else { + prev = new SpannableString("e"); + prev.setSpan(new AnimatedEmojiSpan(DialogObject.getEmojiStatusDocumentId(action.prev_value), Theme.chat_actionTextPaint.getFontMetricsInt()), 0, 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + } + + boolean hasUntil = action.new_value instanceof TLRPC.TL_emojiStatusUntil; + + SpannableString next; + if (action.new_value instanceof TLRPC.TL_emojiStatusEmpty) { + next = new SpannableString(LocaleController.getString(R.string.EventLogEmojiNone)); + } else { + next = new SpannableString("e"); + next.setSpan(new AnimatedEmojiSpan(DialogObject.getEmojiStatusDocumentId(action.new_value), Theme.chat_actionTextPaint.getFontMetricsInt()), 0, 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + } + + SpannableStringBuilder ssb = new SpannableStringBuilder(LocaleController.getString( + prevNone ? ( + hasUntil ? R.string.EventLogChangedEmojiStatusFor : R.string.EventLogChangedEmojiStatus + ) : ( + hasUntil ? R.string.EventLogChangedEmojiStatusFromFor : R.string.EventLogChangedEmojiStatusFrom + ) + )); + + ssb = AndroidUtilities.replaceCharSequence("%1$s", ssb, prev); + ssb = AndroidUtilities.replaceCharSequence("%2$s", ssb, next); + if (hasUntil) { + String until = LocaleController.formatTTLString((int) ((DialogObject.getEmojiStatusUntil(action.new_value) - event.date) * 1.05f)); + ssb = AndroidUtilities.replaceCharSequence("%3$s", ssb, until); + } + + messageText = replaceWithLink(ssb, "un1", fromUser); + } else if (event.action instanceof TLRPC.TL_channelAdminLogEventActionChangeWallpaper) { + TLRPC.TL_channelAdminLogEventActionChangeWallpaper action = (TLRPC.TL_channelAdminLogEventActionChangeWallpaper) event.action; + if (action.new_value instanceof TLRPC.TL_wallPaperNoFile && action.new_value.id == 0 && action.new_value.settings == null) { + messageText = replaceWithLink(LocaleController.getString(R.string.EventLogRemovedWallpaper), "un1", fromUser); + } else { + photoThumbs = new ArrayList<>(); + if (action.new_value.document != null) { + photoThumbs.addAll(action.new_value.document.thumbs); + photoThumbsObject = action.new_value.document; + } + messageText = replaceWithLink(LocaleController.getString(R.string.EventLogChangedWallpaper), "un1", fromUser); + } } else if (event.action instanceof TLRPC.TL_channelAdminLogEventActionChangeBackgroundEmoji) { TLRPC.TL_channelAdminLogEventActionChangeBackgroundEmoji action = (TLRPC.TL_channelAdminLogEventActionChangeBackgroundEmoji) event.action; messageText = replaceWithLink(LocaleController.getString(R.string.EventLogChangedEmoji), "un1", fromUser); @@ -3441,7 +3586,7 @@ public void createMessageSendInfo() { } public boolean hasInlineBotButtons() { - return !isRestrictedMessage && messageOwner != null && messageOwner.reply_markup instanceof TLRPC.TL_replyInlineMarkup && !messageOwner.reply_markup.rows.isEmpty(); + return !isRestrictedMessage && !isRepostPreview && messageOwner != null && messageOwner.reply_markup instanceof TLRPC.TL_replyInlineMarkup && !messageOwner.reply_markup.rows.isEmpty(); } public void measureInlineBotButtons() { @@ -3555,6 +3700,8 @@ private void updateMessageText(AbstractMap users, AbstractMap< } else { messageText = LocaleController.formatString(R.string.ActionSetSameWallpaperForThisChat, user.first_name); } + } else if (fromChat != null) { + messageText = LocaleController.getString(R.string.ActionSetWallpaperForThisChannel); } } else if (messageOwner.action instanceof TLRPC.TL_messageActionSetChatWallPaper) { contentType = 1; @@ -3593,6 +3740,8 @@ private void updateMessageText(AbstractMap users, AbstractMap< } messageText = AndroidUtilities.replaceCharSequence("%s", messageText, userName); } + } else if (fromChat != null) { + messageText = LocaleController.getString(R.string.ActionSetWallpaperForThisChannel); } } else if (messageOwner.action instanceof TLRPC.TL_messageActionGroupCallScheduled) { TLRPC.TL_messageActionGroupCallScheduled action = (TLRPC.TL_messageActionGroupCallScheduled) messageOwner.action; @@ -3790,9 +3939,9 @@ private void updateMessageText(AbstractMap users, AbstractMap< stringBuilder.append(LocaleController.formatPluralString("BoostingGiveawayServiceUndistributed", giveawayResults.unclaimed_count)); } messageText = stringBuilder; - } else if (messageOwner.action instanceof TLRPC.TL_messageActionGiftCode) { + } else if (messageOwner.action instanceof TLRPC.TL_messageActionGiftCode && ((TLRPC.TL_messageActionGiftCode) messageOwner.action).boost_peer != null) { messageText = LocaleController.getString("BoostingReceivedGiftNoName", R.string.BoostingReceivedGiftNoName); - } else if (messageOwner.action instanceof TLRPC.TL_messageActionGiftPremium) { + } else if (messageOwner.action instanceof TLRPC.TL_messageActionGiftPremium || messageOwner.action instanceof TLRPC.TL_messageActionGiftCode) { if (fromObject instanceof TLRPC.User && ((TLRPC.User) fromObject).self) { TLRPC.User user = getUser(users, sUsers, messageOwner.peer_id.user_id); messageText = replaceWithLink(AndroidUtilities.replaceTags(LocaleController.getString(R.string.ActionGiftOutbound)), "un1", user); @@ -3874,39 +4023,61 @@ private void updateMessageText(AbstractMap users, AbstractMap< } } } else if (messageOwner.action instanceof TLRPC.TL_messageActionRequestedPeer) { - TLRPC.Peer peer = ((TLRPC.TL_messageActionRequestedPeer) messageOwner.action).peer; - TLObject peerObject = null; - if (peer instanceof TLRPC.TL_peerUser) { - peerObject = MessagesController.getInstance(currentAccount).getUser(peer.user_id); - if (peerObject == null) { - peerObject = getUser(users, sUsers, peer.user_id); + List peerObjects = new ArrayList<>(); + int sharedUsers = 0; + int sharedChannels = 0; + int sharedChats = 0; + List peers = ((TLRPC.TL_messageActionRequestedPeer) messageOwner.action).peers; + for (TLRPC.Peer peer : peers) { + TLObject peerObject = null; + if (peer instanceof TLRPC.TL_peerUser) { + peerObject = MessagesController.getInstance(currentAccount).getUser(peer.user_id); + if (peerObject == null) { + peerObject = getUser(users, sUsers, peer.user_id); + } + } else if (peer instanceof TLRPC.TL_peerChat) { + peerObject = MessagesController.getInstance(currentAccount).getChat(peer.chat_id); + if (peerObject == null) { + peerObject = getChat(chats, sChats, peer.chat_id); + } + } else if (peer instanceof TLRPC.TL_peerChannel) { + peerObject = MessagesController.getInstance(currentAccount).getChat(peer.channel_id); + if (peerObject == null) { + peerObject = getChat(chats, sChats, peer.channel_id); + } } - } else if (peer instanceof TLRPC.TL_peerChat) { - peerObject = MessagesController.getInstance(currentAccount).getChat(peer.chat_id); - if (peerObject == null) { - peerObject = getChat(chats, sChats, peer.chat_id); + if (peer instanceof TLRPC.TL_peerUser) { + sharedUsers++; + } else if (peer instanceof TLRPC.TL_peerChat) { + sharedChats++; + } else { + sharedChannels++; } - } else if (peer instanceof TLRPC.TL_peerChannel) { - peerObject = MessagesController.getInstance(currentAccount).getChat(peer.channel_id); - if (peerObject == null) { - peerObject = getChat(chats, sChats, peer.channel_id); + if (peerObject != null) { + peerObjects.add(peerObject); } } + if (sharedUsers > 0 && sharedUsers != peerObjects.size()) { + messageText = LocaleController.getPluralString("ActionRequestedPeerUserPlural", sharedUsers); + } else if (sharedChannels > 0 && sharedChannels != peerObjects.size()) { + messageText = LocaleController.getPluralString("ActionRequestedPeerChannelPlural", sharedChannels); + } else if (sharedChats > 0 && sharedChats != peerObjects.size()) { + messageText = LocaleController.getPluralString("ActionRequestedPeerChatPlural", sharedChats); + } else { + String separator = ", "; + SpannableStringBuilder names = new SpannableStringBuilder(); + for (int i = 0; i < peerObjects.size(); i++) { + names.append(replaceWithLink("un1", "un1", peerObjects.get(i))); + if (i < peerObjects.size() - 1) { + names.append(separator); + } + } + messageText = AndroidUtilities.replaceCharSequence("un1", LocaleController.getString(R.string.ActionRequestedPeer), names); + } TLRPC.User bot = MessagesController.getInstance(currentAccount).getUser(getDialogId()); if (bot == null) { bot = getUser(users, sUsers, getDialogId()); } - if (peerObject == null) { - if (peer instanceof TLRPC.TL_peerUser) { - messageText = LocaleController.getString(R.string.ActionRequestedPeerUser); - } else if (peer instanceof TLRPC.TL_peerChat) { - messageText = LocaleController.getString(R.string.ActionRequestedPeerChat); - } else { - messageText = LocaleController.getString(R.string.ActionRequestedPeerChannel); - } - } else { - messageText = replaceWithLink(LocaleController.getString(R.string.ActionRequestedPeer), "un1", peerObject); - } messageText = replaceWithLink(messageText, "un2", bot); } else if (messageOwner.action instanceof TLRPC.TL_messageActionSetMessagesTTL) { TLRPC.TL_messageActionSetMessagesTTL action = (TLRPC.TL_messageActionSetMessagesTTL) messageOwner.action; @@ -4241,15 +4412,19 @@ private void updateMessageText(AbstractMap users, AbstractMap< } else if (messageOwner.action instanceof TLRPC.TL_messageActionSetChatTheme) { String emoticon = ((TLRPC.TL_messageActionSetChatTheme) messageOwner.action).emoticon; String userName = UserObject.getFirstName(fromUser); + boolean isChannel = fromUser == null && fromChat != null; + if (isChannel) { + userName = fromChat.title; + } boolean isUserSelf = UserObject.isUserSelf(fromUser); if (TextUtils.isEmpty(emoticon)) { messageText = isUserSelf ? LocaleController.formatString("ChatThemeDisabledYou", R.string.ChatThemeDisabledYou) - : LocaleController.formatString("ChatThemeDisabled", R.string.ChatThemeDisabled, userName, emoticon); + : LocaleController.formatString(isChannel ? R.string.ChannelThemeDisabled : R.string.ChatThemeDisabled, userName, emoticon); } else { messageText = isUserSelf ? LocaleController.formatString("ChatThemeChangedYou", R.string.ChatThemeChangedYou, emoticon) - : LocaleController.formatString("ChatThemeChangedTo", R.string.ChatThemeChangedTo, userName, emoticon); + : LocaleController.formatString(isChannel ? R.string.ChannelThemeChangedTo : R.string.ChatThemeChangedTo, userName, emoticon); } } else if (messageOwner.action instanceof TLRPC.TL_messageActionChatJoinedByRequest) { if (UserObject.isUserSelf(fromUser)) { @@ -4272,6 +4447,8 @@ private void updateMessageText(AbstractMap users, AbstractMap< // messageText = getMediaTitle(getMedia(messageOwner)); // I'm afraid doing this if (getMedia(messageOwner) instanceof TLRPC.TL_messageMediaGiveaway) { messageText = LocaleController.getString("BoostingGiveawayChannelStarted", R.string.BoostingGiveawayChannelStarted); + } else if (getMedia(messageOwner) instanceof TLRPC.TL_messageMediaGiveawayResults) { + messageText = LocaleController.getString("BoostingGiveawayResults", R.string.BoostingGiveawayResults); } else if (getMedia(messageOwner) instanceof TLRPC.TL_messageMediaStory) { if (getMedia(messageOwner).via_mention) { TLRPC.User user = MessagesController.getInstance(currentAccount).getUser(getMedia(messageOwner).user_id); @@ -4375,6 +4552,8 @@ private void updateMessageText(AbstractMap users, AbstractMap< public CharSequence getMediaTitle(TLRPC.MessageMedia media) { if (media instanceof TLRPC.TL_messageMediaGiveaway) { return LocaleController.getString("BoostingGiveaway", R.string.BoostingGiveaway); + } else if (media instanceof TLRPC.TL_messageMediaGiveawayResults) { + return LocaleController.getString("BoostingGiveawayResults", R.string.BoostingGiveawayResults); } else if (media instanceof TLRPC.TL_messageMediaStory) { if (media.via_mention) { TLRPC.User user = MessagesController.getInstance(currentAccount).getUser(media.user_id); @@ -4522,6 +4701,8 @@ public void setType() { type = TYPE_DATE; } else if (getMedia(messageOwner) instanceof TLRPC.TL_messageMediaGiveaway) { type = TYPE_GIVEAWAY; + } else if (getMedia(messageOwner) instanceof TLRPC.TL_messageMediaGiveawayResults) { + type = TYPE_GIVEAWAY_RESULTS; } else if (getMedia(messageOwner) instanceof TLRPC.TL_messageMediaDice) { type = TYPE_ANIMATED_STICKER; if (getMedia(messageOwner).document == null) { @@ -4579,6 +4760,19 @@ public void setType() { contentType = 1; } } + } else if (currentEvent != null && currentEvent.action instanceof TLRPC.TL_channelAdminLogEventActionChangeWallpaper) { + TLRPC.TL_channelAdminLogEventActionChangeWallpaper wallPaper = (TLRPC.TL_channelAdminLogEventActionChangeWallpaper) currentEvent.action; + contentType = 1; + if (wallPaper.new_value instanceof TLRPC.TL_wallPaperNoFile && wallPaper.new_value.id == 0 && wallPaper.new_value.settings == null) { + type = TYPE_DATE; + } else { + type = TYPE_ACTION_WALLPAPER; + photoThumbs = new ArrayList<>(); + if (wallPaper.new_value.document != null) { + photoThumbs.addAll(wallPaper.new_value.document.thumbs); + photoThumbsObject = wallPaper.new_value.document; + } + } } else if (messageOwner instanceof TLRPC.TL_messageService) { if (messageOwner.action instanceof TLRPC.TL_messageActionSetSameChatWallPaper) { contentType = 1; @@ -4600,10 +4794,10 @@ public void setType() { photoThumbsObject = messageOwner.action.photo; } else if (messageOwner.action instanceof TLRPC.TL_messageActionLoginUnknownLocation) { type = TYPE_TEXT; - } else if (messageOwner.action instanceof TLRPC.TL_messageActionGiftCode) { + } else if (messageOwner.action instanceof TLRPC.TL_messageActionGiftCode && ((TLRPC.TL_messageActionGiftCode) messageOwner.action).boost_peer != null) { contentType = 1; type = TYPE_GIFT_PREMIUM_CHANNEL; - } else if (messageOwner.action instanceof TLRPC.TL_messageActionGiftPremium) { + } else if (messageOwner.action instanceof TLRPC.TL_messageActionGiftPremium || messageOwner.action instanceof TLRPC.TL_messageActionGiftCode) { contentType = 1; type = TYPE_GIFT_PREMIUM; } else if (messageOwner.action instanceof TLRPC.TL_messageActionChatEditPhoto || messageOwner.action instanceof TLRPC.TL_messageActionUserUpdatedPhoto) { @@ -5937,6 +6131,9 @@ public static boolean addEntitiesToText(CharSequence text, ArrayList= 4 ? -1 : 0; Layout.Alignment align = Layout.Alignment.ALIGN_NORMAL; //type == TYPE_EMOJIS && isOut() ? Layout.Alignment.ALIGN_OPPOSITE : Layout.Alignment.ALIGN_NORMAL; + CharSequence text = messageText; try { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { StaticLayout.Builder builder = - StaticLayout.Builder.obtain(messageText, 0, messageText.length(), paint, maxWidth) + StaticLayout.Builder.obtain(text, 0, text.length(), paint, maxWidth) .setLineSpacing(lineAdd, lineSpacing) .setBreakStrategy(StaticLayout.BREAK_STRATEGY_HIGH_QUALITY) .setHyphenationFrequency(StaticLayout.HYPHENATION_FREQUENCY_NONE) @@ -6174,13 +6372,76 @@ public void generateLayout(TLRPC.User fromUser) { } textLayout = builder.build(); } else { - textLayout = new StaticLayout(messageText, paint, maxWidth, align, lineSpacing, lineAdd, false); + textLayout = new StaticLayout(text, paint, maxWidth, align, lineSpacing, lineAdd, false); } } catch (Exception e) { FileLog.e(e); return; } + if (isRepostPreview) { + int maxLines = 22; + if (type != MessageObject.TYPE_TEXT) { + maxLines = hasValidGroupId() ? 7 : 12; + } + if (isWebpage()) { + maxLines -= 8; + } + if (textLayout.getLineCount() > maxLines) { + String readMore = LocaleController.getString(R.string.ReadMore); + int readMoreWidth = (int) Math.ceil(paint.measureText("… " + readMore) + AndroidUtilities.dp(1)); + + float maxRight = 0; + for (int i = 0; i < maxLines; ++i) { + maxRight = Math.max(maxRight, textLayout.getLineRight(i)); + } + + int start = textLayout.getLineStart(maxLines - 1); + int end = textLayout.getLineEnd(maxLines - 1) - 1; + int offset = end; + for (; offset >= start; --offset) { + if (textLayout.getPrimaryHorizontal(offset) < maxRight - readMoreWidth) { + break; + } + } + for (; offset >= start; --offset) { + if (Character.isWhitespace(text.charAt(offset))) { + break; + } + } + text = new SpannableStringBuilder(text.subSequence(0, offset)).append("… ").append(readMore); + ((SpannableStringBuilder) text).setSpan(new CharacterStyle() { + @Override + public void updateDrawState(TextPaint tp) { + tp.setColor(Theme.chat_msgTextPaint.linkColor); + } + }, text.length() - readMore.length(), text.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + + try { + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { + StaticLayout.Builder builder = + StaticLayout.Builder.obtain(text, 0, text.length(), paint, maxWidth) + .setLineSpacing(lineAdd, lineSpacing) + .setBreakStrategy(StaticLayout.BREAK_STRATEGY_HIGH_QUALITY) + .setHyphenationFrequency(StaticLayout.HYPHENATION_FREQUENCY_NONE) + .setAlignment(align); + if (emojiOnlyCount > 0) { + builder.setIncludePad(false); + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) { + builder.setUseLineSpacingFromFallbacks(false); + } + } + textLayout = builder.build(); + } else { + textLayout = new StaticLayout(text, paint, maxWidth, align, lineSpacing, lineAdd, false); + } + } catch (Exception e) { + FileLog.e(e); + return; + } + } + } + if (hasSingleQuote) { maxWidth += AndroidUtilities.dp(32); } else if (hasSingleCode) { @@ -6202,9 +6463,9 @@ public void generateLayout(TLRPC.User fromUser) { float prevOffset = 0; ArrayList textRanges = new ArrayList<>(); - if (messageText instanceof Spanned && (hasQuote || hasCode)) { + if (text instanceof Spanned && (hasQuote || hasCode)) { singleLayout = false; - cutIntoRanges(messageText, textRanges); + cutIntoRanges(text, textRanges); } else if (singleLayout || blocksCount == 1) { textRanges.add(new TextRange(0, textLayout.getText().length())); } else { @@ -6281,7 +6542,7 @@ public void generateLayout(TLRPC.User fromUser) { } } - CharSequence blockText = messageText.subSequence(range.start, range.end); + CharSequence blockText = text.subSequence(range.start, range.end); int blockMaxWidth = maxWidth; if (block.quote) { blockMaxWidth -= dp(24); @@ -6506,6 +6767,9 @@ public void generateLayout(TLRPC.User fromUser) { textWidth = Math.max(textWidth, Math.min(maxWidth, linesMaxWidth)); } + if (block.languageLayout != null) { + textWidth = (int) Math.max(textWidth, Math.min(block.languageLayout.getCurrentWidth() + dp(15), block.textLayout == null ? 0 : block.textLayout.getWidth())); + } linesOffset += currentBlockLinesCount; @@ -6589,6 +6853,62 @@ public TextLayoutBlocks(MessageObject messageObject, @NonNull CharSequence text, FileLog.e(e); return; } + if (messageObject != null && messageObject.isRepostPreview) { + int maxLines = 22; + if (messageObject.type != MessageObject.TYPE_TEXT) { + maxLines = messageObject.hasValidGroupId() ? 7 : 12; + } + if (messageObject.isWebpage()) { + maxLines -= 8; + } + if (textLayout.getLineCount() > maxLines) { + String readMore = LocaleController.getString(R.string.ReadMore); + int readMoreWidth = (int) Math.ceil(textPaint.measureText("… " + readMore) + AndroidUtilities.dp(1)); + + float maxRight = 0; + for (int i = 0; i < maxLines; ++i) { + maxRight = Math.max(maxRight, textLayout.getLineRight(i)); + } + + int start = textLayout.getLineStart(maxLines - 1); + int end = textLayout.getLineEnd(maxLines - 1) - 1; + int offset = end; + for (; offset >= start; --offset) { + if (textLayout.getPrimaryHorizontal(offset) < maxRight - readMoreWidth) { + break; + } + } + for (; offset >= start; --offset) { + if (Character.isWhitespace(text.charAt(offset))) { + break; + } + } + text = new SpannableStringBuilder(text.subSequence(0, offset)).append("… ").append(readMore); + ((SpannableStringBuilder) text).setSpan(new CharacterStyle() { + @Override + public void updateDrawState(TextPaint tp) { + tp.setColor(Theme.chat_msgTextPaint.linkColor); + } + }, text.length() - readMore.length(), text.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + + try { + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { + StaticLayout.Builder builder = + StaticLayout.Builder.obtain(text, 0, text.length(), textPaint, width) + .setLineSpacing(lineAdd, lineSpacing) + .setBreakStrategy(StaticLayout.BREAK_STRATEGY_HIGH_QUALITY) + .setHyphenationFrequency(StaticLayout.HYPHENATION_FREQUENCY_NONE) + .setAlignment(align); + textLayout = builder.build(); + } else { + textLayout = new StaticLayout(text, textPaint, width, align, lineSpacing, lineAdd, false); + } + } catch (Exception e) { + FileLog.e(e); + return; + } + } + } if (hasSingleQuote) { width += AndroidUtilities.dp(32); @@ -6874,6 +7194,9 @@ public TextLayoutBlocks(MessageObject messageObject, @NonNull CharSequence text, textWidth = Math.max(textWidth, Math.min(width, linesMaxWidth)); } + if (block.languageLayout != null) { + textWidth = (int) Math.max(textWidth, Math.min(block.languageLayout.getCurrentWidth() + dp(15), block.textLayout == null ? 0 : block.textLayout.getWidth())); + } linesOffset += currentBlockLinesCount; if (messageObject != null && !messageObject.isSpoilersRevealed && !messageObject.spoiledLoginCode) { @@ -6918,6 +7241,9 @@ public boolean isOutOwner() { } public boolean needDrawAvatar() { + if (isRepostPreview) { + return true; + } if (forceAvatar || customAvatarDrawable != null) { return true; } @@ -6925,6 +7251,9 @@ public boolean needDrawAvatar() { } private boolean needDrawAvatarInternal() { + if (isRepostPreview) { + return true; + } if (forceAvatar || customAvatarDrawable != null) { return true; } @@ -7199,7 +7528,22 @@ public static long getChannelId(TLRPC.Message message) { return 0; } - public static boolean shouldEncryptPhotoOrVideo(TLRPC.Message message) { + public static long getChatId(TLRPC.Message message) { + if (message == null) { + return 0; + } + if (message.peer_id instanceof TLRPC.TL_peerChat) { + return message.peer_id.chat_id; + } else if (message.peer_id instanceof TLRPC.TL_peerChannel) { + return message.peer_id.channel_id; + } + return 0; + } + + public static boolean shouldEncryptPhotoOrVideo(int currentAccount, TLRPC.Message message) { + if (MessagesController.getInstance(currentAccount).isChatNoForwards(getChatId(message)) || message != null && message.noforwards) { + return true; + } if (message instanceof TLRPC.TL_message_secret) { return (getMedia(message) instanceof TLRPC.TL_messageMediaPhoto || isVideoMessage(message)) && message.ttl > 0 && message.ttl <= 60; } else { @@ -7208,7 +7552,7 @@ public static boolean shouldEncryptPhotoOrVideo(TLRPC.Message message) { } public boolean shouldEncryptPhotoOrVideo() { - return shouldEncryptPhotoOrVideo(messageOwner); + return shouldEncryptPhotoOrVideo(currentAccount, messageOwner); } public static boolean isSecretPhotoOrVideo(TLRPC.Message message) { @@ -7230,6 +7574,9 @@ public static boolean isSecretMedia(TLRPC.Message message) { } public boolean needDrawBluredPreview() { + if (isRepostPreview) { + return false; + } if (hasExtendedMediaPreview()) { return true; } else if (messageOwner instanceof TLRPC.TL_message_secret) { @@ -8063,6 +8410,14 @@ public boolean isVoice() { return isVoiceMessage(messageOwner); } + public boolean isVoiceOnce() { + return isVoice() && messageOwner != null && messageOwner.media != null && messageOwner.media.ttl_seconds == 0x7FFFFFFF; + } + + public boolean isRoundOnce() { + return isRoundVideo() && messageOwner != null && messageOwner.media != null && messageOwner.media.ttl_seconds == 0x7FFFFFFF; + } + public boolean isVideo() { return isVideoMessage(messageOwner); } @@ -9095,6 +9450,7 @@ public boolean selectReaction(ReactionsLayoutInBubble.VisibleReaction visibleRea } public boolean probablyRingtone() { + if (isVoiceOnce()) return false; if (getDocument() != null && RingtoneDataStore.ringtoneSupportedMimeType.contains(getDocument().mime_type) && getDocument().size < MessagesController.getInstance(currentAccount).ringtoneSizeMax * 2) { for (int a = 0; a < getDocument().attributes.size(); a++) { TLRPC.DocumentAttribute attribute = getDocument().attributes.get(a); @@ -9166,6 +9522,14 @@ public boolean isGiveaway() { return type == MessageObject.TYPE_GIVEAWAY; } + public boolean isGiveawayOrGiveawayResults() { + return isGiveaway() || isGiveawayResults(); + } + + public boolean isGiveawayResults() { + return type == MessageObject.TYPE_GIVEAWAY_RESULTS; + } + public boolean isAnyGift() { return type == MessageObject.TYPE_GIFT_PREMIUM || type == MessageObject.TYPE_GIFT_PREMIUM_CHANNEL; } @@ -9332,7 +9696,7 @@ public boolean hasLinkMediaToMakeSmall() { final boolean hasLinkPreview = !isRestrictedMessage && MessageObject.getMedia(messageOwner) instanceof TLRPC.TL_messageMediaWebPage && MessageObject.getMedia(messageOwner).webpage instanceof TLRPC.TL_webPage; final TLRPC.WebPage webpage = hasLinkPreview ? MessageObject.getMedia(messageOwner).webpage : null; final String webpageType = webpage != null ? webpage.type : null; - return hasLinkPreview && !isGiveaway() && + return hasLinkPreview && !isGiveawayOrGiveawayResults() && webpage != null && (webpage.photo != null || isVideoDocument(webpage.document)) && !(webpage != null && TextUtils.isEmpty(webpage.description) && TextUtils.isEmpty(webpage.title)) && !(isSponsored() && sponsoredWebPage == null && sponsoredChannelPost == 0) && // drawInstantViewType = 1 diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/MessagePreviewParams.java b/TMessagesProj/src/main/java/org/telegram/messenger/MessagePreviewParams.java index ebdd5433079..b500c29ec93 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/MessagePreviewParams.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/MessagePreviewParams.java @@ -196,6 +196,7 @@ public Messages checkEdits(ArrayList replaceMessageObjects) { public boolean webpagePhoto; public boolean noforwards; + public boolean hasSecretMessages; public TLRPC.WebPage webpage; public CharacterStyle currentLink; @@ -206,10 +207,13 @@ public MessagePreviewParams(boolean secret, boolean noforwards) { } public void updateReply(MessageObject replyMessageObject, MessageObject.GroupedMessages group, long dialogId, ChatActivity.ReplyQuote replyQuote) { - if (isSecret || replyMessageObject == null || replyMessageObject.type == MessageObject.TYPE_DATE || replyMessageObject.type == MessageObject.TYPE_ACTION_PHOTO || replyMessageObject.type == MessageObject.TYPE_ACTION_WALLPAPER || replyMessageObject.type == MessageObject.TYPE_SUGGEST_PHOTO) { + if (isSecret || replyMessageObject == null || replyMessageObject.type == MessageObject.TYPE_DATE || replyMessageObject.type == MessageObject.TYPE_ACTION_PHOTO + || replyMessageObject.type == MessageObject.TYPE_ACTION_WALLPAPER || replyMessageObject.type == MessageObject.TYPE_SUGGEST_PHOTO + || replyMessageObject.type == MessageObject.TYPE_GIFT_PREMIUM || replyMessageObject.type == MessageObject.TYPE_GIFT_PREMIUM_CHANNEL || replyMessageObject.type == MessageObject.TYPE_PHONE_CALL) { replyMessageObject = null; replyQuote = null; } + hasSecretMessages = replyMessageObject != null && (replyMessageObject.isVoiceOnce() || replyMessageObject.isRoundOnce()); if (replyMessageObject != null || replyQuote != null) { if (group != null) { replyMessage = new Messages(null, 1, group.messages, dialogId, null); @@ -341,18 +345,22 @@ public void checkCurrentLink(MessageObject msg) { public boolean hasLink(CharSequence text, String url) { if (url != null) { - Spannable spanned = SpannableString.valueOf(text); try { - AndroidUtilities.addLinks(spanned, Linkify.WEB_URLS); - } catch (Exception e) { - FileLog.e(e); - } - URLSpan[] urlSpans = spanned.getSpans(0, spanned.length(), URLSpan.class); + Spannable spanned = SpannableString.valueOf(text); + try { + AndroidUtilities.addLinks(spanned, Linkify.WEB_URLS); + } catch (Exception e2) { + FileLog.e(e2); + } + URLSpan[] urlSpans = spanned.getSpans(0, spanned.length(), URLSpan.class); - for (int i = 0; i < urlSpans.length; ++i) { - if (areUrlsEqual(urlSpans[i].getURL(), url)) { - return true; + for (int i = 0; i < urlSpans.length; ++i) { + if (areUrlsEqual(urlSpans[i].getURL(), url)) { + return true; + } } + } catch (Exception e) { + FileLog.e(e); } } return false; @@ -449,6 +457,10 @@ private MessageObject toPreviewMessage(MessageObject messageObject, Boolean out, } message.out = out == null ? messageObject.messageOwner.out : out; + if (message.out) { + message.from_id = new TLRPC.TL_peerUser(); + message.from_id.user_id = UserConfig.getInstance(messageObject.currentAccount).getClientUserId(); + } message.unread = false; message.via_bot_id = messageObject.messageOwner.via_bot_id; message.reply_markup = messageObject.messageOwner.reply_markup; @@ -464,7 +476,7 @@ private MessageObject toPreviewMessage(MessageObject messageObject, Boolean out, if (msgtype == 0) { TLRPC.MessageFwdHeader header = null; - long clientUserId = UserConfig.getInstance(messageObject.currentAccount).clientUserId; + long clientUserId = UserConfig.getInstance(messageObject.currentAccount).getClientUserId(); if (!isSecret) { if (messageObject.messageOwner.fwd_from != null) { header = messageObject.messageOwner.fwd_from; diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/MessagesController.java b/TMessagesProj/src/main/java/org/telegram/messenger/MessagesController.java index abba21a22c0..2ecc7192405 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/MessagesController.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/MessagesController.java @@ -515,7 +515,6 @@ protected boolean useCache(Integer arguments) { public int ringtoneSizeMax; public boolean storiesExportNopublicLink; public int authorizationAutoconfirmPeriod; - public int channelColorLevelMin; public int quoteLengthMax; public boolean giveawayGiftsPurchaseAvailable; public PeerColors peerColors; @@ -565,6 +564,13 @@ protected boolean useCache(Integer arguments) { public int storiesSentWeeklyLimitPremium; public int storiesSentMonthlyLimitDefault; public int storiesSentMonthlyLimitPremium; + public int storiesSuggestedReactionsLimitDefault; + public int storiesSuggestedReactionsLimitPremium; + public int channelBgIconLevelMin; + public int channelProfileIconLevelMin; + public int channelEmojiStatusLevelMin; + public int channelWallpaperLevelMin; + public int channelCustomWallpaperLevelMin; public int uploadMaxFileParts; public int uploadMaxFilePartsPremium; @@ -580,6 +586,13 @@ protected boolean useCache(Integer arguments) { public boolean premiumLocked; public int transcribeButtonPressed; + public boolean premiumFeaturesBlocked() { + return premiumLocked && !getUserConfig().isPremium(); + } + public boolean premiumPurchaseBlocked() { + return premiumLocked; + } + public List directPaymentsCurrency = new ArrayList<>(); public NewMessageCallback newMessageCallback; @@ -715,7 +728,7 @@ public int getChatReactionsCount() { } public boolean isPremiumUser(TLRPC.User currentUser) { - return !premiumLocked && currentUser.premium; + return !premiumFeaturesBlocked() && currentUser.premium; } public boolean didPressTranscribeButtonEnough() { @@ -733,7 +746,7 @@ public void pressTranscribeButton() { public ArrayList filterPremiumStickers(ArrayList stickerSets) { - if (!premiumLocked) { + if (!premiumFeaturesBlocked()) { return stickerSets; } for (int i = 0; i < stickerSets.size(); i++) { @@ -749,7 +762,7 @@ public ArrayList filterPremiumStickers(ArrayList colors = new ArrayList<>(); private final LongSparseArray colorsById = new LongSparseArray<>(); + public boolean needUpdate() { + boolean noLevels = true; + boolean hasStandardColors = false; + for (int i = 0; i < colors.size(); ++i) { + if (colors.get(i).lvl > 0) { + noLevels = false; + } + if (colors.get(i).id < 7) { + hasStandardColors = true; + } + } + return noLevels || type == TYPE_NAME && !hasStandardColors; + } + + public int colorsAvailable(int lvl) { + int count = 0; + for (int i = 0; i < colors.size(); ++i) { + if (!colors.get(i).hidden && lvl >= colors.get(i).lvl) { + count++; + } + } + return count; + } + + public int maxLevel() { + int maxLvl = 0; + for (int i = 0; i < colors.size(); ++i) { + if (!colors.get(i).hidden) { + maxLvl = Math.max(maxLvl, colors.get(i).lvl); + } + } + return maxLvl; + } + + public int minLevel() { + int minLvl = maxLevel(); + for (int i = 0; i < colors.size(); ++i) { + if (!colors.get(i).hidden) { + minLvl = Math.min(minLvl, colors.get(i).lvl); + } + } + return minLvl; + } + private PeerColors(int type, int hash) { this.type = type; this.hash = hash; @@ -3692,6 +3821,7 @@ public static PeerColors fromString(int type, String str) { PeerColor peerColor = PeerColor.fromString(colorParts[i]); if (peerColor == null) continue; + peerColor.isDefaultName = peerColor.id < 7 && type == TYPE_NAME; if (!peerColor.hidden) peerColors.colors.add(peerColor); peerColors.colorsById.put(peerColor.id, peerColor); @@ -3702,7 +3832,7 @@ public static PeerColors fromString(int type, String str) { private static int color(String str) { return Integer.parseUnsignedInt("ff" + str, 16); } - + public static PeerColors fromTL(int type, TLRPC.TL_help_peerColors tl) { if (tl == null) return null; try { @@ -3710,7 +3840,7 @@ public static PeerColors fromTL(int type, TLRPC.TL_help_peerColors tl) { for (int i = 0; i < tl.colors.size(); ++i) { PeerColor peerColor = PeerColor.fromTL(tl.colors.get(i)); if (peerColor == null) continue; - if (peerColor.id < 7 && type == TYPE_NAME) continue; + peerColor.isDefaultName = peerColor.id < 7 && type == TYPE_NAME; if (!peerColor.hidden) peerColors.colors.add(peerColor); peerColors.colorsById.put(peerColor.id, peerColor); @@ -3748,7 +3878,7 @@ public static PeerColors fromJSON( FileLog.e(e2); continue; } - if (type == TYPE_NAME && peerColor.id < 7) continue; + peerColor.isDefaultName = peerColor.id < 7 && type == TYPE_NAME; peerColors.colorsById.put(id, peerColor); } } @@ -3794,10 +3924,20 @@ public static PeerColors fromJSON( } public static class PeerColor { + public boolean isDefaultName; public int id; public boolean hidden; + public int lvl; private final int[] colors = new int[6]; private final int[] darkColors = new int[6]; + public int getColor(int i, Theme.ResourcesProvider resourcesProvider) { + if (i < 0 || i > 5) return 0; + if (isDefaultName && id >= 0 && id < 7) { + return Theme.getColor(Theme.keys_avatar_nameInMessage[id], resourcesProvider); + } + final boolean isDark = resourcesProvider != null ? resourcesProvider.isDark() : Theme.isCurrentThemeDark(); + return (isDark ? darkColors : colors)[i]; + } public int getColor1(boolean isDark) { return (isDark ? darkColors : colors)[0]; } @@ -3831,9 +3971,6 @@ public int getColor4() { public int getColor5() { return (Theme.isCurrentThemeDark() ? darkColors : colors)[4]; } - public int getColor6() { - return (Theme.isCurrentThemeDark() ? darkColors : colors)[5]; - } public boolean hasColor2() { return getColor2() != getColor1(); } @@ -3870,6 +4007,9 @@ public int getAvatarColor2() { public void appendString(StringBuilder sb) { sb.append("#"); if (hidden) sb.append("H"); + if (lvl > 0) { + sb.append("[").append(lvl).append("]"); + } sb.append(id); sb.append("{"); sb.append(colors[0]); @@ -3918,6 +4058,9 @@ public static PeerColor fromTL(TLRPC.TL_help_peerColorOption tl) { final PeerColor peerColor = new PeerColor(); peerColor.id = tl.color_id; peerColor.hidden = tl.hidden; + if ((tl.flags & 8) != 0) { + peerColor.lvl = tl.channel_min_level; + } System.arraycopy(optionToColors(tl.colors), 0, peerColor.colors, 0, 6); System.arraycopy(optionToColors(tl.dark_colors), 0, peerColor.darkColors, 0, 6); @@ -3962,16 +4105,25 @@ public static PeerColor fromString(String string) { if (string == null || string.isEmpty() || string.charAt(0) != '#') return null; int startIndex = 1; - boolean hidden = string.length() > 1 && string.charAt(1) == 'H'; + boolean hidden = string.length() > 1 && string.charAt(startIndex) == 'H'; if (hidden) { startIndex++; } + int lvl = 0; + if (string.length() > startIndex && string.charAt(startIndex) == '[') { + int eindex = string.indexOf(']'); + if (eindex > startIndex) { + lvl = Utilities.parseInt(string.substring(startIndex + 1, eindex)); + startIndex = eindex + 1; + } + } int index = string.indexOf('{'); if (index < 0) return null; try { final PeerColor peerColor = new PeerColor(); peerColor.id = Utilities.parseInt(string.substring(startIndex, index)); peerColor.hidden = hidden; + peerColor.lvl = lvl; final String[] parts = string.substring(index + 1, string.length() - 1).split("@"); String[] colorsString = parts[0].split(","); for (int i = 0; i < 6; ++i) @@ -4458,7 +4610,7 @@ public void didReceivedNotification(int id, int account, Object... args) { } NotificationCenter.getGlobalInstance().postNotificationName(NotificationCenter.wallpapersNeedReload, wallPaper.slug); if (uploadingWallpaperInfo.requestIds != null && overrideWallpaperInfo.dialogId != 0) { - uploadingWallpaperInfo.requestIds.add(ChatThemeController.getInstance(currentAccount).setWallpaperToUser(overrideWallpaperInfo.dialogId, uploadingWallpaperFinal, overrideWallpaperInfo, null, null)); + uploadingWallpaperInfo.requestIds.add(ChatThemeController.getInstance(currentAccount).setWallpaperToPeer(overrideWallpaperInfo.dialogId, uploadingWallpaperFinal, overrideWallpaperInfo, null, null)); } } }); @@ -4853,7 +5005,6 @@ public void cleanup() { loadingSuggestedFilters = false; loadingRemoteFilters = false; suggestedFilters.clear(); - gettingAppChangelog = false; dialogFiltersLoaded = false; ignoreSetOnline = false; @@ -5230,6 +5381,7 @@ public void putChat(final TLRPC.Chat chat, boolean fromCache) { } } } + updateEmojiStatusUntilUpdate(-chat.id, chat.emoji_status); if (chat.min) { if (oldChat != null) { if (!fromCache) { @@ -5311,7 +5463,7 @@ public void putChat(final TLRPC.Chat chat, boolean fromCache) { } else { oldChat.flags |= 16384; } - if (!chat.stories_hidden_min) { + if (chat.stories_hidden_min) { chat.stories_hidden = oldChat.stories_hidden; } if (oldFlags != newFlags || oldFlags2 != newFlags2) { @@ -5699,6 +5851,7 @@ public void loadFullChat(long chatId, int classGuid, boolean force) { getMessagesStorage().putUsersAndChats(res.users, res.chats, true, true); getMessagesStorage().updateChatInfo(res.full_chat, false); getStoriesController().updateStoriesFromFullPeer(dialogId, res.full_chat.stories); + ChatThemeController.getInstance(currentAccount).saveChatWallpaper(-chatId, res.full_chat.wallpaper); if (ChatObject.isChannel(chat)) { Integer value = dialogs_read_inbox_max.get(dialogId); if (value == null) { @@ -5780,7 +5933,10 @@ public void loadFullChat(long chatId, int classGuid, boolean force) { dialog.ttl_period = res.full_chat.ttl_period; getNotificationCenter().postNotificationName(NotificationCenter.dialogsNeedReload); } - dialog.view_forum_as_messages = res.full_chat.view_forum_as_messages; + if (dialog.view_forum_as_messages != res.full_chat.view_forum_as_messages) { + dialog.view_forum_as_messages = res.full_chat.view_forum_as_messages; + getMessagesStorage().setDialogViewThreadAsMessages(dialogId, res.full_chat.view_forum_as_messages); + } } }); } else { @@ -10975,7 +11131,6 @@ public void processLoadedDialogs(final TLRPC.messages_Dialogs dialogsRes, ArrayL migratingDialogs = false; getNotificationCenter().postNotificationName(NotificationCenter.needReloadRecentDialogsSearch); } else { - generateUpdateMessage(); if (!added && loadType == DIALOGS_LOAD_TYPE_CACHE && dialogsEndReached.get(folderId)) { loadDialogs(folderId, 0, count, false); } @@ -12117,13 +12272,28 @@ public void convertToMegaGroup(Context context, long chatId, BaseFragment fragme processUpdates((TLRPC.Updates) response, false); AndroidUtilities.runOnUIThread(() -> { if (convertRunnable != null) { + TLRPC.Chat prevChat = null; for (int a = 0; a < updates.chats.size(); a++) { TLRPC.Chat chat = updates.chats.get(a); - if (ChatObject.isChannel(chat)) { - convertRunnable.run(chat.id); + if (chatId == chat.id) { + prevChat = chat; break; } } + if (prevChat != null && prevChat.migrated_to != null) { + long newChatId = prevChat.migrated_to.channel_id; + TLRPC.Chat newChat = null; + for (int a = 0; a < updates.chats.size(); a++) { + TLRPC.Chat chat = updates.chats.get(a); + if (newChatId == chat.id) { + newChat = chat; + break; + } + } + if (newChat != null) { + convertRunnable.run(newChatId); + } + } } }); } else { @@ -12960,27 +13130,6 @@ public void performLogout(int type) { getContactsController().deleteUnknownAppAccounts(); } - - private boolean gettingAppChangelog; - - public void generateUpdateMessage() { - if (gettingAppChangelog || BuildVars.DEBUG_VERSION || SharedConfig.lastUpdateVersion == null || SharedConfig.lastUpdateVersion.equals(BuildVars.BUILD_VERSION_STRING)) { - return; - } - gettingAppChangelog = true; - TLRPC.TL_help_getAppChangelog req = new TLRPC.TL_help_getAppChangelog(); - req.prev_app_version = SharedConfig.lastUpdateVersion; - getConnectionsManager().sendRequest(req, (response, error) -> { - if (error == null) { - SharedConfig.lastUpdateVersion = BuildVars.BUILD_VERSION_STRING; - SharedConfig.saveConfig(); - } - if (response instanceof TLRPC.Updates) { - processUpdates((TLRPC.Updates) response, false); - } - }); - } - public void registerForPush(@PushListenerController.PushType int pushType, String regid) { if (TextUtils.isEmpty(regid) || registeringForPush || getUserConfig().getClientUserId() == 0) { return; @@ -17256,9 +17405,13 @@ public boolean processUpdateArray(ArrayList updates, ArrayList= 0) { + TLRPC.UserFull userFull = getUserFull(uploadingWallpaperInfo.dialogId); + if (userFull != null) { + userFull.wallpaper = uploadingWallpaperInfo.prevUserWallpaper; + NotificationCenter.getInstance(currentAccount).postNotificationName(NotificationCenter.userInfoDidLoad, uploadingWallpaperInfo.dialogId, userFull); + } + } else { + TLRPC.ChatFull chatFull = getChatFull(-uploadingWallpaperInfo.dialogId); + if (chatFull != null) { + chatFull.wallpaper = uploadingWallpaperInfo.prevUserWallpaper; + NotificationCenter.getInstance(currentAccount).postNotificationName(NotificationCenter.chatInfoDidLoad, chatFull, 0, false, false); + } } } uploadingWallpaperInfo = null; @@ -19544,9 +19705,12 @@ public ChannelRecommendations getChannelRecommendations(long chatId) { } public void checkPeerColors(boolean force) { - if (peerColors == null || force) { + if (peerColors == null || peerColors.needUpdate() || force) { TLRPC.TL_help_getPeerColors req = new TLRPC.TL_help_getPeerColors(); req.hash = peerColors != null ? peerColors.hash : 0; + if (peerColors != null && peerColors.needUpdate()) { + req.hash = 0; + } getConnectionsManager().sendRequest(req, (res, err) -> { if (res instanceof TLRPC.TL_help_peerColors) { AndroidUtilities.runOnUIThread(() -> { @@ -19556,9 +19720,12 @@ public void checkPeerColors(boolean force) { } }); } - if (profilePeerColors == null || force) { + if (profilePeerColors == null || profilePeerColors.needUpdate() || force) { TLRPC.TL_help_getPeerProfileColors req = new TLRPC.TL_help_getPeerProfileColors(); req.hash = profilePeerColors != null ? profilePeerColors.hash : 0; + if (profilePeerColors != null && profilePeerColors.needUpdate()) { + req.hash = 0; + } getConnectionsManager().sendRequest(req, (res, err) -> { if (res instanceof TLRPC.TL_help_peerColors) { AndroidUtilities.runOnUIThread(() -> { diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/MessagesStorage.java b/TMessagesProj/src/main/java/org/telegram/messenger/MessagesStorage.java index 18015e67c3a..ace1a6c2421 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/MessagesStorage.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/MessagesStorage.java @@ -14160,6 +14160,14 @@ public static void addUsersAndChatsFromMessage(TLRPC.Message message, ArrayList< } } } + if (message.media instanceof TLRPC.TL_messageMediaGiveawayResults) { + TLRPC.TL_messageMediaGiveawayResults giveaway = (TLRPC.TL_messageMediaGiveawayResults) message.media; + for (Long uid : giveaway.winners) { + if (!usersToLoad.contains(uid)) { + usersToLoad.add(uid); + } + } + } if (message.media instanceof TLRPC.TL_messageMediaPoll) { TLRPC.TL_messageMediaPoll messageMediaPoll = (TLRPC.TL_messageMediaPoll) message.media; if (!messageMediaPoll.results.recent_voters.isEmpty()) { @@ -14168,8 +14176,19 @@ public static void addUsersAndChatsFromMessage(TLRPC.Message message, ArrayList< } } } - if (message.media instanceof TLRPC.TL_messageMediaStory && message.media.storyItem != null && message.media.storyItem.fwd_from != null) { - addLoadPeerInfo(message.media.storyItem.fwd_from.from, usersToLoad, chatsToLoad); + if (message.media instanceof TLRPC.TL_messageMediaStory && message.media.storyItem != null) { + if (message.media.storyItem.fwd_from != null) { + addLoadPeerInfo(message.media.storyItem.fwd_from.from, usersToLoad, chatsToLoad); + } + if (message.media.storyItem != null && message.media.storyItem.media_areas != null) { + for (int j = 0; j < message.media.storyItem.media_areas.size(); ++j) { + if (message.media.storyItem.media_areas.get(j) instanceof TL_stories.TL_mediaAreaChannelPost) { + long channelId = ((TL_stories.TL_mediaAreaChannelPost) message.media.storyItem.media_areas.get(j)).channel_id; + if (!chatsToLoad.contains(channelId)) + chatsToLoad.add(channelId); + } + } + } } if (message.media instanceof TLRPC.TL_messageMediaWebPage && message.media.webpage != null && message.media.webpage.attributes != null) { for (int i = 0; i < message.media.webpage.attributes.size(); ++i) { @@ -14178,6 +14197,15 @@ public static void addUsersAndChatsFromMessage(TLRPC.Message message, ArrayList< if (attr.storyItem != null && attr.storyItem.fwd_from != null) { addLoadPeerInfo(attr.storyItem.fwd_from.from, usersToLoad, chatsToLoad); } + if (attr.storyItem != null && attr.storyItem.media_areas != null) { + for (int j = 0; j < attr.storyItem.media_areas.size(); ++j) { + if (attr.storyItem.media_areas.get(j) instanceof TL_stories.TL_mediaAreaChannelPost) { + long channelId = ((TL_stories.TL_mediaAreaChannelPost) attr.storyItem.media_areas.get(j)).channel_id; + if (!chatsToLoad.contains(channelId)) + chatsToLoad.add(channelId); + } + } + } } } } diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/NotificationCenter.java b/TMessagesProj/src/main/java/org/telegram/messenger/NotificationCenter.java index 8df45b8416a..fa191a30871 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/NotificationCenter.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/NotificationCenter.java @@ -210,6 +210,8 @@ public class NotificationCenter { public static final int updateBotMenuButton = totalEvents++; + public static final int giftsToUserSent = totalEvents++; + public static final int didStartedMultiGiftsSelector = totalEvents++; public static final int boostedChannelByUser = totalEvents++; public static final int boostByChannelCreated = totalEvents++; public static final int didUpdatePremiumGiftStickers = totalEvents++; @@ -221,6 +223,7 @@ public class NotificationCenter { public static final int unconfirmedAuthUpdate = totalEvents++; public static final int dialogPhotosUpdate = totalEvents++; public static final int channelRecommendationsLoaded = totalEvents++; + public static final int savedMessagesUpdate = totalEvents++; //global public static final int pushMessagesUpdated = totalEvents++; @@ -285,21 +288,22 @@ public class NotificationCenter { public static final int userEmojiStatusUpdated = totalEvents++; public static final int requestPermissions = totalEvents++; public static final int permissionsGranted = totalEvents++; - public static int topicsDidLoaded = totalEvents++; - public static int chatSwithcedToForum = totalEvents++; - public static int didUpdateGlobalAutoDeleteTimer = totalEvents++; - public static int onDatabaseReset = totalEvents++; - public static int wallpaperSettedToUser = totalEvents++; - public static int storiesUpdated = totalEvents++; - public static int storiesListUpdated = totalEvents++; - public static int storiesDraftsUpdated = totalEvents++; - public static int chatlistFolderUpdate = totalEvents++; + public static final int topicsDidLoaded = totalEvents++; + public static final int chatSwithcedToForum = totalEvents++; + public static final int didUpdateGlobalAutoDeleteTimer = totalEvents++; + public static final int onDatabaseReset = totalEvents++; + public static final int wallpaperSettedToUser = totalEvents++; + public static final int storiesUpdated = totalEvents++; + public static final int storiesListUpdated = totalEvents++; + public static final int storiesDraftsUpdated = totalEvents++; + public static final int chatlistFolderUpdate = totalEvents++; public static final int uploadStoryProgress = totalEvents++; public static final int uploadStoryEnd = totalEvents++; public static final int customTypefacesLoaded = totalEvents++; public static final int stealthModeChanged = totalEvents++; public static final int onReceivedChannelDifference = totalEvents++; public static final int storiesReadUpdated = totalEvents++; + public static final int nearEarEvent = totalEvents++; public static boolean alreadyLogged; diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/NotificationsController.java b/TMessagesProj/src/main/java/org/telegram/messenger/NotificationsController.java index c56f701f7e8..535974bb9ba 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/NotificationsController.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/NotificationsController.java @@ -2120,6 +2120,8 @@ private String getShortStringForMessage(MessageObject messageObject, String[] us } } else if (messageObject.messageOwner.media instanceof TLRPC.TL_messageMediaGiveaway) { return LocaleController.getString("BoostingGiveaway", R.string.BoostingGiveaway); + } else if (messageObject.messageOwner.media instanceof TLRPC.TL_messageMediaGiveawayResults) { + return LocaleController.getString("BoostingGiveawayResults", R.string.BoostingGiveawayResults); } else if (messageObject.messageOwner.media instanceof TLRPC.TL_messageMediaGeo || messageObject.messageOwner.media instanceof TLRPC.TL_messageMediaVenue) { return LocaleController.getString("AttachLocation", R.string.AttachLocation); } else if (messageObject.messageOwner.media instanceof TLRPC.TL_messageMediaGeoLive) { @@ -2369,6 +2371,8 @@ private String getStringForMessage(MessageObject messageObject, boolean shortMes } else if (messageObject.messageOwner.media instanceof TLRPC.TL_messageMediaGiveaway) { TLRPC.TL_messageMediaGiveaway giveaway = (TLRPC.TL_messageMediaGiveaway) messageObject.messageOwner.media; msg = LocaleController.formatString("NotificationMessageChannelGiveaway", R.string.NotificationMessageChannelGiveaway, name, giveaway.quantity, giveaway.months); + } else if (messageObject.messageOwner.media instanceof TLRPC.TL_messageMediaGiveawayResults) { + msg = LocaleController.formatString("BoostingGiveawayResults", R.string.BoostingGiveawayResults); } else if (messageObject.messageOwner.media instanceof TLRPC.TL_messageMediaPoll) { TLRPC.TL_messageMediaPoll mediaPoll = (TLRPC.TL_messageMediaPoll) messageObject.messageOwner.media; if (mediaPoll.poll.quiz) { @@ -2823,6 +2827,8 @@ private String getStringForMessage(MessageObject messageObject, boolean shortMes } else if (messageObject.messageOwner.media instanceof TLRPC.TL_messageMediaGiveaway) { TLRPC.TL_messageMediaGiveaway giveaway = (TLRPC.TL_messageMediaGiveaway) messageObject.messageOwner.media; msg = LocaleController.formatString("NotificationMessageChannelGiveaway", R.string.NotificationMessageChannelGiveaway, chat.title, giveaway.quantity, giveaway.months); + } else if (messageObject.messageOwner.media instanceof TLRPC.TL_messageMediaGiveawayResults) { + msg = LocaleController.formatString("BoostingGiveawayResults", R.string.BoostingGiveawayResults); } else if (messageObject.messageOwner.media instanceof TLRPC.TL_messageMediaGeo || messageObject.messageOwner.media instanceof TLRPC.TL_messageMediaVenue) { msg = LocaleController.formatString("NotificationMessageGroupMap", R.string.NotificationMessageGroupMap, name, chat.title); } else if (messageObject.messageOwner.media instanceof TLRPC.TL_messageMediaGeoLive) { diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/SavedMessagesController.java b/TMessagesProj/src/main/java/org/telegram/messenger/SavedMessagesController.java new file mode 100644 index 00000000000..a3896ed82cb --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/messenger/SavedMessagesController.java @@ -0,0 +1,76 @@ +package org.telegram.messenger; + +import androidx.collection.LongSparseArray; + +import org.telegram.SQLite.SQLiteCursor; +import org.telegram.SQLite.SQLiteDatabase; +import org.telegram.SQLite.SQLiteException; +import org.telegram.tgnet.NativeByteBuffer; +import org.telegram.tgnet.TLRPC; + +import java.util.ArrayList; +import java.util.HashMap; + +public class SavedMessagesController { + + private final int currentAccount; + + public boolean loading, loaded; + public LongSparseArray> messages = new LongSparseArray>(); + + public SavedMessagesController(int account) { + this.currentAccount = account; + } + + public void getSavedMessagesDialogs() { + if (loaded || loading) { + return; + } + loading = true; + final long myself = UserConfig.getInstance(currentAccount).getClientUserId(); + MessagesStorage storage = MessagesStorage.getInstance(currentAccount); + storage.getStorageQueue().postRunnable(() -> { + SQLiteDatabase database = storage.getDatabase(); + SQLiteCursor cursor = null; + final LongSparseArray> messages = new LongSparseArray<>(); + try { + cursor = database.queryFinalized("SELECT data, mid, date, send_state, read_state, custom_params FROM messages_v2 WHERE out = 0 AND uid = ?", myself); + while (cursor.next()) { + NativeByteBuffer data = cursor.byteBufferValue(0); + if (data != null) { + TLRPC.Message message = TLRPC.Message.TLdeserialize(data, data.readInt32(false), false); + if (message.fwd_from == null || message.fwd_from.saved_from_peer == null) { + continue; + } + long did = DialogObject.getPeerDialogId(message.fwd_from.saved_from_peer); + + message.id = cursor.intValue(1); + message.date = cursor.intValue(2); + message.send_state = cursor.intValue(3); + MessageObject.setUnreadFlags(message, cursor.intValue(4)); + + MessageObject messageObject = new MessageObject(currentAccount, message, true, true); + ArrayList messageObjects = messages.get(did); + if (messageObjects == null) { + messages.put(did, messageObjects = new ArrayList<>()); + } + messageObjects.add(messageObject); + } + } + + AndroidUtilities.runOnUIThread(() -> { + SavedMessagesController.this.messages.clear(); + SavedMessagesController.this.messages.putAll(messages); + loading = false; + }); + } catch (SQLiteException e) { + e.printStackTrace(); + } finally { + if (cursor != null) { + cursor.dispose(); + cursor = null; + } + } + }); + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/SharedConfig.java b/TMessagesProj/src/main/java/org/telegram/messenger/SharedConfig.java index fab279e0ea3..3fbbfce0e44 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/SharedConfig.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/SharedConfig.java @@ -21,7 +21,6 @@ import android.os.SystemClock; import android.text.TextUtils; import android.util.Base64; -import android.util.Log; import android.webkit.WebView; import androidx.annotation.IntDef; @@ -234,7 +233,6 @@ private static boolean isWhitelisted(MediaCodecInfo codecInfo) { public static int lastPauseTime; public static boolean isWaitingForPasscodeEnter; public static boolean useFingerprint = true; - public static String lastUpdateVersion; public static int suggestStickers; public static boolean suggestAnimatedEmoji; public static int keepMedia = CacheByChatsController.KEEP_MEDIA_ONE_MONTH; //deprecated @@ -312,6 +310,8 @@ private static boolean isWhitelisted(MediaCodecInfo codecInfo) { public static int messageSeenHintCount; public static int emojiInteractionsHintCount; public static int dayNightThemeSwitchHintCount; + public static boolean forceLessData; + public static int callEncryptionHintDisplayedCount; public static TLRPC.TL_help_appUpdate pendingAppUpdate; public static int pendingAppUpdateBuildVersion; @@ -431,7 +431,6 @@ public static void saveConfig() { editor.putInt("badPasscodeTries", badPasscodeTries); editor.putInt("autoLockIn", autoLockIn); editor.putInt("lastPauseTime", lastPauseTime); - editor.putString("lastUpdateVersion2", lastUpdateVersion); editor.putBoolean("useFingerprint", useFingerprint); editor.putBoolean("allowScreenCapture", allowScreenCapture); editor.putString("pushString2", pushString); @@ -511,7 +510,6 @@ public static void loadConfig() { autoLockIn = preferences.getInt("autoLockIn", 60 * 60); lastPauseTime = preferences.getInt("lastPauseTime", 0); useFingerprint = preferences.getBoolean("useFingerprint", true); - lastUpdateVersion = preferences.getString("lastUpdateVersion2", "3.5"); allowScreenCapture = preferences.getBoolean("allowScreenCapture", false); lastLocalId = preferences.getInt("lastLocalId", -210000); pushString = preferences.getString("pushString2", ""); @@ -541,7 +539,7 @@ public static void loadConfig() { try { String update = preferences.getString("appUpdate", null); if (update != null) { - pendingAppUpdateBuildVersion = preferences.getInt("appUpdateBuild", BuildVars.BUILD_VERSION); + pendingAppUpdateBuildVersion = preferences.getInt("appUpdateBuild", buildVersion()); byte[] arr = Base64.decode(update, Base64.DEFAULT); if (arr != null) { SerializedData data = new SerializedData(arr); @@ -561,7 +559,7 @@ public static void loadConfig() { FileLog.e(e); } if (updateVersion == 0) { - updateVersion = BuildVars.BUILD_VERSION; + updateVersion = buildVersion(); } if (updateVersionString == null) { updateVersionString = BuildVars.BUILD_VERSION_STRING; @@ -651,6 +649,8 @@ public static void loadConfig() { payByInvoice = preferences.getBoolean("payByInvoice", false); photoViewerBlur = preferences.getBoolean("photoViewerBlur", true); multipleReactionsPromoShowed = preferences.getBoolean("multipleReactionsPromoShowed", false); + forceLessData = preferences.getBoolean("forceLessData", false); + callEncryptionHintDisplayedCount = preferences.getInt("callEncryptionHintDisplayedCount", 0); loadDebugConfig(preferences); @@ -669,6 +669,15 @@ public static void loadConfig() { } } + public static int buildVersion() { + try { + return ApplicationLoader.applicationContext.getPackageManager().getPackageInfo(ApplicationLoader.applicationContext.getPackageName(), 0).versionCode; + } catch (Exception e) { + FileLog.e(e); + return 0; + } + } + public static void updateTabletConfig() { if (fontSizeIsDefault) { SharedPreferences preferences = ApplicationLoader.applicationContext.getSharedPreferences("mainconfig", Activity.MODE_PRIVATE); @@ -752,7 +761,7 @@ public static boolean isAppUpdateAvailable() { currentVersion = pInfo.versionCode; } catch (Exception e) { FileLog.e(e); - currentVersion = BuildVars.BUILD_VERSION; + currentVersion = buildVersion(); } return pendingAppUpdateBuildVersion == currentVersion; } @@ -768,7 +777,7 @@ public static boolean setNewAppVersionAvailable(TLRPC.TL_help_appUpdate update) FileLog.e(e); } if (versionCode == 0) { - versionCode = BuildVars.BUILD_VERSION; + versionCode = buildVersion(); } if (updateVersionString == null) { updateVersionString = BuildVars.BUILD_VERSION_STRING; @@ -847,7 +856,6 @@ public static void clearConfig() { useFingerprint = true; isWaitingForPasscodeEnter = false; allowScreenCapture = false; - lastUpdateVersion = BuildVars.BUILD_VERSION_STRING; textSelectionHintShows = 0; scheduledOrNoSoundHintShows = 0; scheduledOrNoSoundHintSeenAt = 0; @@ -1071,6 +1079,14 @@ public static void toggleDebugWebView() { editor.apply(); } + public static void incrementCallEncryptionHintDisplayed(int count) { + callEncryptionHintDisplayedCount += count; + SharedPreferences preferences = MessagesController.getGlobalMainSettings(); + SharedPreferences.Editor editor = preferences.edit(); + editor.putInt("callEncryptionHintDisplayedCount", callEncryptionHintDisplayedCount); + editor.apply(); + } + public static void toggleLoopStickers() { LiteMode.toggleFlag(LiteMode.FLAG_ANIMATED_STICKERS_CHAT); } @@ -1529,6 +1545,10 @@ public static void updateEmojiInteractionsHintCount(int count) { preferences.edit().putInt("emojiInteractionsHintCount", emojiInteractionsHintCount).apply(); } + public static void setForceLessData(boolean value) { + SharedPreferences preferences = MessagesController.getGlobalMainSettings(); + preferences.edit().putBoolean("forceLessData", forceLessData = value).apply(); + } public static void updateDayNightThemeSwitchHintCount(int count) { dayNightThemeSwitchHintCount = count; diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/TopicsController.java b/TMessagesProj/src/main/java/org/telegram/messenger/TopicsController.java index 5822315adb9..4b2bd265733 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/TopicsController.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/TopicsController.java @@ -662,7 +662,11 @@ public void toggleViewForumAsMessages(long channelId, boolean enabled) { TLRPC.TL_channels_toggleViewForumAsMessages request = new TLRPC.TL_channels_toggleViewForumAsMessages(); request.channel_id = getMessagesController().getInputChannel(channelId); request.enabled = enabled; - getConnectionsManager().sendRequest(request, null); + getConnectionsManager().sendRequest(request, (res, err) -> { + if (res != null) { + getMessagesController().processUpdates((TLRPC.Updates) res, false); + } + }); } public void pinTopic(long chatId, int topicId, boolean pin, BaseFragment fragment) { diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/UserObject.java b/TMessagesProj/src/main/java/org/telegram/messenger/UserObject.java index e2f259b7efe..6af2c53cb9a 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/UserObject.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/UserObject.java @@ -144,9 +144,9 @@ public static boolean isService(long user_id) { } public static MessagesController.PeerColor getPeerColorForAvatar(int currentAccount, TLRPC.User user) { - if (user != null && user.profile_color != null && user.profile_color.color >= 0 && MessagesController.getInstance(currentAccount).profilePeerColors != null) { - return MessagesController.getInstance(currentAccount).profilePeerColors.getColor(user.profile_color.color); - } +// if (user != null && user.profile_color != null && user.profile_color.color >= 0 && MessagesController.getInstance(currentAccount).profilePeerColors != null) { +// return MessagesController.getInstance(currentAccount).profilePeerColors.getColor(user.profile_color.color); +// } return null; } diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/VideoEditedInfo.java b/TMessagesProj/src/main/java/org/telegram/messenger/VideoEditedInfo.java index b8a7e7d38a1..5dc4868b3ce 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/VideoEditedInfo.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/VideoEditedInfo.java @@ -55,15 +55,17 @@ public class VideoEditedInfo { public byte[] key; public byte[] iv; public MediaController.SavedFilterState filterState; - public String paintPath, blurPath; + public String paintPath, blurPath, messagePath, messageVideoMaskPath, backgroundPath; public ArrayList mediaEntities; public MediaController.CropState cropState; public boolean isPhoto; public boolean isStory; public StoryEntry.HDRInfo hdrInfo; - public ArrayList parts; public Integer gradientTopColor, gradientBottomColor; + public int account; + public boolean isDark; + public long wallpaperPeerId = Long.MIN_VALUE; public boolean forceFragmenting; public boolean alreadyScheduledConverting; @@ -115,6 +117,7 @@ public static class MediaEntity { public static final byte TYPE_LOCATION = 3; public static final byte TYPE_REACTION = 4; public static final byte TYPE_ROUND = 5; + public static final byte TYPE_MESSAGE = 6; public byte type; public byte subType; @@ -135,7 +138,9 @@ public static class MediaEntity { public int viewHeight; public float roundRadius; - public float scale; + public String segmentedPath = ""; + + public float scale = 1.0f; public float textViewWidth; public float textViewHeight; public float textViewX; @@ -172,50 +177,53 @@ public static class MediaEntity { public MediaEntity() { } - public MediaEntity(AbstractSerializedData data, boolean full) { - type = data.readByte(false); - subType = data.readByte(false); - x = data.readFloat(false); - y = data.readFloat(false); - rotation = data.readFloat(false); - width = data.readFloat(false); - height = data.readFloat(false); - text = data.readString(false); - int count = data.readInt32(false); + this(data, full, false); + } + + public MediaEntity(AbstractSerializedData data, boolean full, boolean exception) { + type = data.readByte(exception); + subType = data.readByte(exception); + x = data.readFloat(exception); + y = data.readFloat(exception); + rotation = data.readFloat(exception); + width = data.readFloat(exception); + height = data.readFloat(exception); + text = data.readString(exception); + int count = data.readInt32(exception); for (int i = 0; i < count; ++i) { EmojiEntity entity = new EmojiEntity(); - data.readInt32(false); - entity.readParams(data, false); + data.readInt32(exception); + entity.readParams(data, exception); entities.add(entity); } - color = data.readInt32(false); - fontSize = data.readInt32(false); - viewWidth = data.readInt32(false); - viewHeight = data.readInt32(false); - textAlign = data.readInt32(false); - textTypeface = PaintTypeface.find(textTypefaceKey = data.readString(false)); - scale = data.readFloat(false); - textViewWidth = data.readFloat(false); - textViewHeight = data.readFloat(false); - textViewX = data.readFloat(false); - textViewY = data.readFloat(false); + color = data.readInt32(exception); + fontSize = data.readInt32(exception); + viewWidth = data.readInt32(exception); + viewHeight = data.readInt32(exception); + textAlign = data.readInt32(exception); + textTypeface = PaintTypeface.find(textTypefaceKey = data.readString(exception)); + scale = data.readFloat(exception); + textViewWidth = data.readFloat(exception); + textViewHeight = data.readFloat(exception); + textViewX = data.readFloat(exception); + textViewY = data.readFloat(exception); if (full) { - int magic = data.readInt32(false); + int magic = data.readInt32(exception); if (magic == TLRPC.TL_null.constructor) { document = null; } else { - document = TLRPC.Document.TLdeserialize(data, magic, false); + document = TLRPC.Document.TLdeserialize(data, magic, exception); } } if (type == TYPE_LOCATION) { - density = data.readFloat(false); - mediaArea = TL_stories.MediaArea.TLdeserialize(data, data.readInt32(false), false); - mediaGeo = TLRPC.MessageMedia.TLdeserialize(data, data.readInt32(false), false); + density = data.readFloat(exception); + mediaArea = TL_stories.MediaArea.TLdeserialize(data, data.readInt32(exception), exception); + mediaGeo = TLRPC.MessageMedia.TLdeserialize(data, data.readInt32(exception), exception); if (data.remaining() > 0) { - int magic = data.readInt32(false); + int magic = data.readInt32(exception); if (magic == 0xdeadbeef) { - String emoji = data.readString(false); + String emoji = data.readString(exception); if (mediaGeo instanceof TLRPC.TL_messageMediaVenue) { ((TLRPC.TL_messageMediaVenue) mediaGeo).emoji = emoji; } @@ -223,13 +231,16 @@ public MediaEntity(AbstractSerializedData data, boolean full) { } } if (type == TYPE_REACTION) { - mediaArea = TL_stories.MediaArea.TLdeserialize(data, data.readInt32(false), false); + mediaArea = TL_stories.MediaArea.TLdeserialize(data, data.readInt32(exception), exception); } if (type == TYPE_ROUND) { - roundOffset = data.readInt64(false); - roundLeft = data.readInt64(false); - roundRight = data.readInt64(false); - roundDuration = data.readInt64(false); + roundOffset = data.readInt64(exception); + roundLeft = data.readInt64(exception); + roundRight = data.readInt64(exception); + roundDuration = data.readInt64(exception); + } + if (type == TYPE_PHOTO) { + segmentedPath = data.readString(exception); } } @@ -293,6 +304,9 @@ public void serializeTo(AbstractSerializedData data, boolean full) { data.writeInt64(roundRight); data.writeInt64(roundDuration); } + if (type == TYPE_PHOTO) { + data.writeString(segmentedPath); + } } public MediaEntity copy() { @@ -452,14 +466,7 @@ public String getString() { } else { serializedData.writeByte(0); } - if (parts != null && !parts.isEmpty()) { - serializedData.writeInt32(parts.size()); - for (StoryEntry.Part part : parts) { - part.serializeToStream(serializedData); - } - } else { - serializedData.writeInt32(0); - } + serializedData.writeInt32(0); serializedData.writeBool(isStory); serializedData.writeBool(fromCamera); if (blurPathBytes != null) { @@ -582,11 +589,7 @@ public boolean parseString(String string) { } } if (version >= 6) { - int count = serializedData.readInt32(false); - for (int i = 0; i < count; ++i) { - StoryEntry.Part part = new StoryEntry.Part(); - part.readParams(serializedData, false); - } + serializedData.readInt32(false); } if (version >= 7) { isStory = serializedData.readBool(false); diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/video/MediaCodecVideoConvertor.java b/TMessagesProj/src/main/java/org/telegram/messenger/video/MediaCodecVideoConvertor.java index 22d8ec1c270..d8b614a008f 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/video/MediaCodecVideoConvertor.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/video/MediaCodecVideoConvertor.java @@ -89,7 +89,6 @@ private boolean convertVideoInternal(ConvertVideoParams convertVideoParams, boolean muted = convertVideoParams.muted; boolean isStory = convertVideoParams.isStory; StoryEntry.HDRInfo hdrInfo = convertVideoParams.hdrInfo; - ArrayList parts = convertVideoParams.parts; FileLog.d("convertVideoInternal original=" + originalWidth + "x" + originalHeight + " result=" + resultWidth + "x" + resultHeight + " " + avatarStartTime); long time = System.currentTimeMillis(); @@ -178,7 +177,7 @@ private boolean convertVideoInternal(ConvertVideoParams convertVideoParams, inputSurface.makeCurrent(); encoder.start(); - outputSurface = new OutputSurface(savedFilterState, videoPath, paintPath, blurPath, mediaEntities, cropState != null && cropState.useMatrix != null ? cropState : null, resultWidth, resultHeight, originalWidth, originalHeight, rotationValue, framerate, true, gradientTopColor, gradientBottomColor, null, parts); + outputSurface = new OutputSurface(savedFilterState, videoPath, paintPath, blurPath, mediaEntities, cropState != null && cropState.useMatrix != null ? cropState : null, resultWidth, resultHeight, originalWidth, originalHeight, rotationValue, framerate, true, gradientTopColor, gradientBottomColor, null, convertVideoParams); ByteBuffer[] encoderOutputBuffers = null; ByteBuffer[] encoderInputBuffers = null; @@ -493,7 +492,7 @@ private boolean convertVideoInternal(ConvertVideoParams convertVideoParams, inputSurface.makeCurrent(); encoder.start(); - outputSurface = new OutputSurface(savedFilterState, null, paintPath, blurPath, mediaEntities, cropState, resultWidth, resultHeight, originalWidth, originalHeight, rotationValue, framerate, false, gradientTopColor, gradientBottomColor, hdrInfo, parts); + outputSurface = new OutputSurface(savedFilterState, null, paintPath, blurPath, mediaEntities, cropState, resultWidth, resultHeight, originalWidth, originalHeight, rotationValue, framerate, false, gradientTopColor, gradientBottomColor, hdrInfo, convertVideoParams); if (hdrInfo == null && outputSurface.supportsEXTYUV() && hasHDR) { hdrInfo = new StoryEntry.HDRInfo(); hdrInfo.colorTransfer = colorTransfer; @@ -1333,6 +1332,9 @@ public static class ConvertVideoParams { MediaController.SavedFilterState savedFilterState; String paintPath; String blurPath; + String messagePath; + String messageVideoMaskPath; + String backgroundPath; ArrayList mediaEntities; boolean isPhoto; MediaController.CropState cropState; @@ -1343,8 +1345,10 @@ public static class ConvertVideoParams { boolean muted; boolean isStory; StoryEntry.HDRInfo hdrInfo; - ArrayList parts; public ArrayList soundInfos = new ArrayList(); + int account; + boolean isDark; + long wallpaperPeerId; private ConvertVideoParams() { @@ -1357,16 +1361,8 @@ public static ConvertVideoParams of(String videoPath, File cacheFile, int framerate, int bitrate, int originalBitrate, long startTime, long endTime, long avatarStartTime, boolean needCompress, long duration, - MediaController.SavedFilterState savedFilterState, - String paintPath, String blurPath, - ArrayList mediaEntities, - boolean isPhoto, - MediaController.CropState cropState, - boolean isRound, MediaController.VideoConvertorListener callback, - Integer gradientTopColor, Integer gradientBottomColor, - boolean muted, boolean isStory, StoryEntry.HDRInfo hdrInfo, - ArrayList parts) { + VideoEditedInfo info) { ConvertVideoParams params = new ConvertVideoParams(); params.videoPath = videoPath; params.cacheFile = cacheFile; @@ -1384,21 +1380,25 @@ public static ConvertVideoParams of(String videoPath, File cacheFile, params.avatarStartTime = avatarStartTime; params.needCompress = needCompress; params.duration = duration; - params.savedFilterState = savedFilterState; - params.paintPath = paintPath; - params.blurPath = blurPath; - params.mediaEntities = mediaEntities; - params.isPhoto = isPhoto; - params.cropState = cropState; - params.isRound = isRound; + params.savedFilterState = info.filterState; + params.paintPath = info.paintPath; + params.blurPath = info.blurPath; + params.mediaEntities = info.mediaEntities; + params.isPhoto = info.isPhoto; + params.cropState = info.cropState; + params.isRound = info.roundVideo; params.callback = callback; - params.gradientTopColor = gradientTopColor; - params.gradientBottomColor = gradientBottomColor; - params.muted = muted; - params.isStory = isStory; - params.hdrInfo = hdrInfo; - params.parts = parts; - + params.gradientTopColor = info.gradientTopColor; + params.gradientBottomColor = info.gradientBottomColor; + params.muted = info.muted; + params.isStory = info.isStory; + params.hdrInfo = info.hdrInfo; + params.isDark = info.isDark; + params.wallpaperPeerId = info.wallpaperPeerId; + params.account = info.account; + params.messagePath = info.messagePath; + params.messageVideoMaskPath = info.messageVideoMaskPath; + params.backgroundPath = info.backgroundPath; return params; } } diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/video/OutputSurface.java b/TMessagesProj/src/main/java/org/telegram/messenger/video/OutputSurface.java index 140b797cfdc..6793076ba32 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/video/OutputSurface.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/video/OutputSurface.java @@ -39,8 +39,8 @@ public class OutputSurface implements SurfaceTexture.OnFrameAvailableListener { private boolean mFrameAvailable; private TextureRenderer mTextureRender; - public OutputSurface(MediaController.SavedFilterState savedFilterState, String imagePath, String paintPath, String blurPath, ArrayList mediaEntities, MediaController.CropState cropState, int w, int h, int originalW, int originalH, int rotation, float fps, boolean photo, Integer gradientTopColor, Integer gradientBottomColor, StoryEntry.HDRInfo hdrInfo, ArrayList parts) { - mTextureRender = new TextureRenderer(savedFilterState, imagePath, paintPath, blurPath, mediaEntities, cropState, w, h, originalW, originalH, rotation, fps, photo, gradientTopColor, gradientBottomColor, hdrInfo, parts); + public OutputSurface(MediaController.SavedFilterState savedFilterState, String imagePath, String paintPath, String blurPath, ArrayList mediaEntities, MediaController.CropState cropState, int w, int h, int originalW, int originalH, int rotation, float fps, boolean photo, Integer gradientTopColor, Integer gradientBottomColor, StoryEntry.HDRInfo hdrInfo, MediaCodecVideoConvertor.ConvertVideoParams params) { + mTextureRender = new TextureRenderer(savedFilterState, imagePath, paintPath, blurPath, mediaEntities, cropState, w, h, originalW, originalH, rotation, fps, photo, gradientTopColor, gradientBottomColor, hdrInfo, params); mTextureRender.surfaceCreated(); mSurfaceTexture = new SurfaceTexture(mTextureRender.getTextureId()); mSurfaceTexture.setOnFrameAvailableListener(this); diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/video/TextureRenderer.java b/TMessagesProj/src/main/java/org/telegram/messenger/video/TextureRenderer.java index 57f8bff8276..fcd99b71759 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/video/TextureRenderer.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/video/TextureRenderer.java @@ -21,6 +21,7 @@ import android.graphics.RectF; import android.graphics.SurfaceTexture; import android.graphics.Typeface; +import android.graphics.drawable.Drawable; import android.opengl.GLES11Ext; import android.opengl.GLES20; import android.opengl.GLES30; @@ -30,6 +31,7 @@ import android.text.Layout; import android.text.SpannableString; import android.text.Spanned; +import android.text.TextUtils; import android.text.style.ReplacementSpan; import android.util.Log; import android.util.Pair; @@ -70,6 +72,7 @@ import org.telegram.ui.Components.Paint.Views.PaintTextOptionsView; import org.telegram.ui.Components.RLottieDrawable; import org.telegram.ui.Components.Rect; +import org.telegram.ui.Stories.recorder.PreviewView; import org.telegram.ui.Stories.recorder.StoryEntry; import java.io.File; @@ -89,15 +92,11 @@ public class TextureRenderer { private FloatBuffer gradientTextureBuffer; private FloatBuffer textureBuffer; private FloatBuffer renderTextureBuffer; + private FloatBuffer maskTextureBuffer; private FloatBuffer bitmapVerticesBuffer; private FloatBuffer blurVerticesBuffer; - private FloatBuffer partsVerticesBuffer[]; - private FloatBuffer partsTextureBuffer; - private ArrayList parts; - private int[] partsTexture; - private boolean useMatrixForImagePath; float[] bitmapData = { @@ -109,6 +108,9 @@ public class TextureRenderer { private FilterShaders filterShaders; private String paintPath; + private String messagePath; + private String messageVideoMaskPath; + private String backgroundPath; private String blurPath; private String imagePath; private int imageWidth, imageHeight; @@ -118,6 +120,7 @@ public class TextureRenderer { private int originalHeight; private int transformedWidth; private int transformedHeight; + private Drawable backgroundDrawable; private BlurringShader blur; @@ -144,6 +147,35 @@ public class TextureRenderer { " vTextureCoord = (uSTMatrix * aTextureCoord).xy;\n" + "}\n"; + private static final String VERTEX_SHADER_MASK = + "uniform mat4 uMVPMatrix;\n" + + "uniform mat4 uSTMatrix;\n" + + "attribute vec4 aPosition;\n" + + "attribute vec4 aTextureCoord;\n" + + "attribute vec4 mTextureCoord;\n" + + "varying vec2 vTextureCoord;\n" + + "varying vec2 MTextureCoord;\n" + + "void main() {\n" + + " gl_Position = uMVPMatrix * aPosition;\n" + + " vTextureCoord = (uSTMatrix * aTextureCoord).xy;\n" + + " MTextureCoord = (uSTMatrix * mTextureCoord).xy;\n" + + "}\n"; + + private static final String VERTEX_SHADER_MASK_300 = + "#version 320 es\n" + + "uniform mat4 uMVPMatrix;\n" + + "uniform mat4 uSTMatrix;\n" + + "in vec4 aPosition;\n" + + "in vec4 aTextureCoord;\n" + + "in vec4 mTextureCoord;\n" + + "out vec2 vTextureCoord;\n" + + "out vec2 MTextureCoord;\n" + + "void main() {\n" + + " gl_Position = uMVPMatrix * aPosition;\n" + + " vTextureCoord = (uSTMatrix * aTextureCoord).xy;\n" + + " MTextureCoord = (uSTMatrix * mTextureCoord).xy;\n" + + "}\n"; + private static final String FRAGMENT_EXTERNAL_SHADER = "#extension GL_OES_EGL_image_external : require\n" + "precision highp float;\n" + @@ -153,6 +185,17 @@ public class TextureRenderer { " gl_FragColor = texture2D(sTexture, vTextureCoord);" + "}\n"; + private static final String FRAGMENT_EXTERNAL_MASK_SHADER = + "#extension GL_OES_EGL_image_external : require\n" + + "precision highp float;\n" + + "varying vec2 vTextureCoord;\n" + + "varying vec2 MTextureCoord;\n" + + "uniform samplerExternalOES sTexture;\n" + + "uniform sampler2D sMask;\n" + + "void main() {\n" + + " gl_FragColor = texture2D(sTexture, vTextureCoord) * texture2D(sMask, MTextureCoord).a;\n" + + "}\n"; + private static final String FRAGMENT_SHADER = "precision highp float;\n" + "varying vec2 vTextureCoord;\n" + @@ -161,6 +204,16 @@ public class TextureRenderer { " gl_FragColor = texture2D(sTexture, vTextureCoord);\n" + "}\n"; + private static final String FRAGMENT_MASK_SHADER = + "precision highp float;\n" + + "varying vec2 vTextureCoord;\n" + + "varying vec2 MTextureCoord;\n" + + "uniform sampler2D sTexture;\n" + + "uniform sampler2D sMask;\n" + + "void main() {\n" + + " gl_FragColor = texture2D(sTexture, vTextureCoord) * texture2D(sMask, MTextureCoord).a;\n" + + "}\n"; + private static final String GRADIENT_FRAGMENT_SHADER = "precision highp float;\n" + "varying vec2 vTextureCoord;\n" + @@ -181,11 +234,14 @@ public class TextureRenderer { private float[] mSTMatrix = new float[16]; private float[] mSTMatrixIdentity = new float[16]; private int mTextureID; + private int videoMaskTexture; private int[] mProgram; private int[] muMVPMatrixHandle; private int[] muSTMatrixHandle; private int[] maPositionHandle; private int[] maTextureHandle; + private int[] mmTextureHandle; + private int[] maskTextureHandle; private int gradientTopColorHandle, gradientBottomColorHandle; private int texSizeHandle; // todo: HDR handles @@ -207,6 +263,11 @@ public class TextureRenderer { private Canvas stickerCanvas; private float videoFps; + private int imagePathIndex = -1; + private int paintPathIndex = -1; + private int messagePathIndex = -1; + private int backgroundPathIndex = -1; + private Bitmap roundBitmap; private Canvas roundCanvas; private final android.graphics.Rect roundSrc = new android.graphics.Rect(); @@ -244,10 +305,9 @@ public TextureRenderer( Integer gradientTopColor, Integer gradientBottomColor, StoryEntry.HDRInfo hdrInfo, - ArrayList parts + MediaCodecVideoConvertor.ConvertVideoParams params ) { isPhoto = photo; - this.parts = parts; float[] texData = { 0.f, 0.f, @@ -285,6 +345,9 @@ public TextureRenderer( this.originalHeight = originalHeight; imagePath = image; paintPath = paint; + messagePath = params.messagePath; + messageVideoMaskPath = params.messageVideoMaskPath; + backgroundPath = params.backgroundPath; blurPath = blurtex; mediaEntities = entities; videoFps = fps == 0 ? 30 : fps; @@ -292,21 +355,12 @@ public TextureRenderer( int count = 0; NUM_EXTERNAL_SHADER = count++; - if (gradientBottomColor != null && gradientTopColor != null) { - NUM_GRADIENT_SHADER = count++; - } - if (filterShaders != null) { - NUM_FILTER_SHADER = count++; - } - mProgram = new int[count]; - muMVPMatrixHandle = new int[count]; - muSTMatrixHandle = new int[count]; - maPositionHandle = new int[count]; - maTextureHandle = new int[count]; Matrix.setIdentityM(mMVPMatrix, 0); int textureRotation = 0; - if (gradientBottomColor != null && gradientTopColor != null) { + if (params != null && params.wallpaperPeerId != Long.MIN_VALUE) { + backgroundDrawable = PreviewView.getBackgroundDrawable(null, params.account, params.wallpaperPeerId, params.isDark); + } else if (gradientBottomColor != null && gradientTopColor != null) { final float[] verticesData = { -1.0f, -1.0f, 1.0f, -1.0f, @@ -325,7 +379,18 @@ public TextureRenderer( gradientTextureBuffer.put(textureData).position(0); this.gradientTopColor = gradientTopColor; this.gradientBottomColor = gradientBottomColor; + NUM_GRADIENT_SHADER = count++; } + if (filterShaders != null) { + NUM_FILTER_SHADER = count++; + } + mProgram = new int[count]; + muMVPMatrixHandle = new int[count]; + muSTMatrixHandle = new int[count]; + maPositionHandle = new int[count]; + maTextureHandle = new int[count]; + mmTextureHandle = new int[count]; + maskTextureHandle = new int[count]; if (cropState != null) { if (cropState.useMatrix != null) { useMatrixForImagePath = true; @@ -455,35 +520,53 @@ public TextureRenderer( } renderTextureBuffer = ByteBuffer.allocateDirect(textureData.length * 4).order(ByteOrder.nativeOrder()).asFloatBuffer(); renderTextureBuffer.put(textureData).position(0); + + textureData = new float[]{ + 0.f, 0.f, + 1.f, 0.f, + 0.f, 1.f, + 1.f, 1.f + }; + maskTextureBuffer = ByteBuffer.allocateDirect(textureData.length * 4).order(ByteOrder.nativeOrder()).asFloatBuffer(); + maskTextureBuffer.put(textureData).position(0); } public int getTextureId() { return mTextureID; } - private void drawGradient() { - if (NUM_GRADIENT_SHADER < 0) { - return; - } - GLES20.glUseProgram(mProgram[NUM_GRADIENT_SHADER]); + private void drawBackground() { + if (NUM_GRADIENT_SHADER >= 0) { + GLES20.glUseProgram(mProgram[NUM_GRADIENT_SHADER]); - GLES20.glVertexAttribPointer(maPositionHandle[NUM_GRADIENT_SHADER], 2, GLES20.GL_FLOAT, false, 8, gradientVerticesBuffer); - GLES20.glEnableVertexAttribArray(maPositionHandle[NUM_GRADIENT_SHADER]); - GLES20.glVertexAttribPointer(maTextureHandle[NUM_GRADIENT_SHADER], 2, GLES20.GL_FLOAT, false, 8, gradientTextureBuffer); - GLES20.glEnableVertexAttribArray(maTextureHandle[NUM_GRADIENT_SHADER]); + GLES20.glVertexAttribPointer(maPositionHandle[NUM_GRADIENT_SHADER], 2, GLES20.GL_FLOAT, false, 8, gradientVerticesBuffer); + GLES20.glEnableVertexAttribArray(maPositionHandle[NUM_GRADIENT_SHADER]); + GLES20.glVertexAttribPointer(maTextureHandle[NUM_GRADIENT_SHADER], 2, GLES20.GL_FLOAT, false, 8, gradientTextureBuffer); + GLES20.glEnableVertexAttribArray(maTextureHandle[NUM_GRADIENT_SHADER]); - GLES20.glUniformMatrix4fv(muSTMatrixHandle[NUM_GRADIENT_SHADER], 1, false, mSTMatrix, 0); - GLES20.glUniformMatrix4fv(muMVPMatrixHandle[NUM_GRADIENT_SHADER], 1, false, mMVPMatrix, 0); + GLES20.glUniformMatrix4fv(muSTMatrixHandle[NUM_GRADIENT_SHADER], 1, false, mSTMatrix, 0); + GLES20.glUniformMatrix4fv(muMVPMatrixHandle[NUM_GRADIENT_SHADER], 1, false, mMVPMatrix, 0); - GLES20.glUniform4f(gradientTopColorHandle, Color.red(gradientTopColor) / 255f, Color.green(gradientTopColor) / 255f, Color.blue(gradientTopColor) / 255f, Color.alpha(gradientTopColor) / 255f); - GLES20.glUniform4f(gradientBottomColorHandle, Color.red(gradientBottomColor) / 255f, Color.green(gradientBottomColor) / 255f, Color.blue(gradientBottomColor) / 255f, Color.alpha(gradientBottomColor) / 255f); - GLES20.glDrawArrays(GLES20.GL_TRIANGLE_STRIP, 0, 4); + GLES20.glUniform4f(gradientTopColorHandle, Color.red(gradientTopColor) / 255f, Color.green(gradientTopColor) / 255f, Color.blue(gradientTopColor) / 255f, Color.alpha(gradientTopColor) / 255f); + GLES20.glUniform4f(gradientBottomColorHandle, Color.red(gradientBottomColor) / 255f, Color.green(gradientBottomColor) / 255f, Color.blue(gradientBottomColor) / 255f, Color.alpha(gradientBottomColor) / 255f); + GLES20.glDrawArrays(GLES20.GL_TRIANGLE_STRIP, 0, 4); + } else if (backgroundPathIndex >= 0) { + GLES20.glUseProgram(simpleShaderProgram); + GLES20.glActiveTexture(GLES20.GL_TEXTURE0); + + GLES20.glUniform1i(simpleSourceImageHandle, 0); + GLES20.glEnableVertexAttribArray(simpleInputTexCoordHandle); + GLES20.glVertexAttribPointer(simpleInputTexCoordHandle, 2, GLES20.GL_FLOAT, false, 8, textureBuffer); + GLES20.glEnableVertexAttribArray(simplePositionHandle); + + drawTexture(true, paintTexture[backgroundPathIndex], -10000, -10000, -10000, -10000, 0, false, false, -1); + } } public void drawFrame(SurfaceTexture st, long time) { boolean blurred = false; if (isPhoto) { - drawGradient(); + drawBackground(); } else { st.getTransformMatrix(mSTMatrix); if (BuildVars.LOGS_ENABLED && firstFrame) { @@ -530,16 +613,26 @@ public void drawFrame(SurfaceTexture st, long time) { stMatrix = mSTMatrix; } - drawGradient(); + drawBackground(); GLES20.glUseProgram(mProgram[index]); GLES20.glActiveTexture(GLES20.GL_TEXTURE0); GLES20.glBindTexture(target, texture); + if (messageVideoMaskPath != null && videoMaskTexture != -1) { + GLES20.glActiveTexture(GLES20.GL_TEXTURE1); + GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, videoMaskTexture); + GLES20.glUniform1i(maskTextureHandle[index], 1); + } + GLES20.glVertexAttribPointer(maPositionHandle[index], 2, GLES20.GL_FLOAT, false, 8, verticesBuffer); GLES20.glEnableVertexAttribArray(maPositionHandle[index]); GLES20.glVertexAttribPointer(maTextureHandle[index], 2, GLES20.GL_FLOAT, false, 8, renderTextureBuffer); GLES20.glEnableVertexAttribArray(maTextureHandle[index]); + if (messageVideoMaskPath != null && videoMaskTexture != -1) { + GLES20.glVertexAttribPointer(mmTextureHandle[index], 2, GLES20.GL_FLOAT, false, 8, maskTextureBuffer); + GLES20.glEnableVertexAttribArray(mmTextureHandle[index]); + } if (texSizeHandle != 0) { GLES20.glUniform2f(texSizeHandle, transformedWidth, transformedHeight); @@ -590,7 +683,7 @@ public void drawFrame(SurfaceTexture st, long time) { GLES20.glDrawArrays(GLES20.GL_TRIANGLE_STRIP, 0, 4); } } - if (isPhoto || paintTexture != null || stickerTexture != null || partsTexture != null) { + if (isPhoto || paintTexture != null || stickerTexture != null) { GLES20.glUseProgram(simpleShaderProgram); GLES20.glActiveTexture(GLES20.GL_TEXTURE0); @@ -599,20 +692,14 @@ public void drawFrame(SurfaceTexture st, long time) { GLES20.glVertexAttribPointer(simpleInputTexCoordHandle, 2, GLES20.GL_FLOAT, false, 8, textureBuffer); GLES20.glEnableVertexAttribArray(simplePositionHandle); } - if (paintTexture != null && imagePath != null) { - for (int a = 0; a < 1; a++) { - drawTexture(true, paintTexture[a], -10000, -10000, -10000, -10000, 0, false, useMatrixForImagePath && isPhoto && a == 0, -1); - } + if (imagePathIndex >= 0) { + drawTexture(true, paintTexture[imagePathIndex], -10000, -10000, -10000, -10000, 0, false, useMatrixForImagePath && isPhoto, -1); } - if (partsTexture != null) { - for (int a = 0; a < partsTexture.length; a++) { - drawTexture(true, partsTexture[a], -10000, -10000, -10000, -10000, 0, false, false, a); - } + if (paintPathIndex >= 0) { + drawTexture(true, paintTexture[paintPathIndex], -10000, -10000, -10000, -10000, 0, false, false, -1); } - if (paintTexture != null) { - for (int a = (imagePath != null ? 1 : 0); a < paintTexture.length; a++) { - drawTexture(true, paintTexture[a], -10000, -10000, -10000, -10000, 0, false, useMatrixForImagePath && isPhoto && a == 0, -1); - } + if (messagePathIndex >= 0) { + drawTexture(true, paintTexture[messagePathIndex], -10000, -10000, -10000, -10000, 0, false, false, -1); } if (stickerTexture != null) { for (int a = 0, N = mediaEntities.size(); a < N; a++) { @@ -857,9 +944,9 @@ private void drawTexture(boolean bind, int texture, float x, float y, float w, f } } bitmapVerticesBuffer.put(bitmapData).position(0); - GLES20.glVertexAttribPointer(simplePositionHandle, 2, GLES20.GL_FLOAT, false, 8, matrixIndex >= 0 ? partsVerticesBuffer[matrixIndex] : (useCropMatrix ? verticesBuffer : bitmapVerticesBuffer)); + GLES20.glVertexAttribPointer(simplePositionHandle, 2, GLES20.GL_FLOAT, false, 8, useCropMatrix ? verticesBuffer : bitmapVerticesBuffer); GLES20.glEnableVertexAttribArray(simpleInputTexCoordHandle); - GLES20.glVertexAttribPointer(simpleInputTexCoordHandle, 2, GLES20.GL_FLOAT, false, 8, matrixIndex >= 0 ? partsTextureBuffer : (useCropMatrix ? renderTextureBuffer : textureBuffer)); + GLES20.glVertexAttribPointer(simpleInputTexCoordHandle, 2, GLES20.GL_FLOAT, false, 8, useCropMatrix ? renderTextureBuffer : textureBuffer); if (bind) { GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, texture); } @@ -874,22 +961,27 @@ public void setBreakStrategy(EditTextOutline editText) { @SuppressLint("WrongConstant") public void surfaceCreated() { for (int a = 0; a < mProgram.length; a++) { - String shader = null; + String fragSshader = null; + String vertexShader = VERTEX_SHADER; if (a == NUM_EXTERNAL_SHADER) { - shader = FRAGMENT_EXTERNAL_SHADER; + fragSshader = messageVideoMaskPath != null ? FRAGMENT_EXTERNAL_MASK_SHADER : FRAGMENT_EXTERNAL_SHADER; + vertexShader = messageVideoMaskPath != null ? VERTEX_SHADER_MASK : VERTEX_SHADER; } else if (a == NUM_FILTER_SHADER) { - shader = FRAGMENT_SHADER; + fragSshader = messageVideoMaskPath != null ? FRAGMENT_MASK_SHADER : FRAGMENT_SHADER; + vertexShader = messageVideoMaskPath != null ? VERTEX_SHADER_MASK : VERTEX_SHADER; } else if (a == NUM_GRADIENT_SHADER) { - shader = GRADIENT_FRAGMENT_SHADER; + fragSshader = GRADIENT_FRAGMENT_SHADER; } - if (shader == null) { + if (vertexShader == null || fragSshader == null) { continue; } - mProgram[a] = createProgram(VERTEX_SHADER, shader, false); + mProgram[a] = createProgram(vertexShader, fragSshader, false); maPositionHandle[a] = GLES20.glGetAttribLocation(mProgram[a], "aPosition"); maTextureHandle[a] = GLES20.glGetAttribLocation(mProgram[a], "aTextureCoord"); + mmTextureHandle[a] = GLES20.glGetAttribLocation(mProgram[a], "mTextureCoord"); muMVPMatrixHandle[a] = GLES20.glGetUniformLocation(mProgram[a], "uMVPMatrix"); muSTMatrixHandle[a] = GLES20.glGetUniformLocation(mProgram[a], "uSTMatrix"); + maskTextureHandle[a] = GLES20.glGetUniformLocation(mProgram[a], "sMask"); if (a == NUM_GRADIENT_SHADER) { gradientTopColorHandle = GLES20.glGetUniformLocation(mProgram[a], "gradientTopColor"); gradientBottomColorHandle = GLES20.glGetUniformLocation(mProgram[a], "gradientBottomColor"); @@ -904,6 +996,23 @@ public void surfaceCreated() { GLES20.glTexParameteri(GLES11Ext.GL_TEXTURE_EXTERNAL_OES, GLES20.GL_TEXTURE_WRAP_S, GLES20.GL_CLAMP_TO_EDGE); GLES20.glTexParameteri(GLES11Ext.GL_TEXTURE_EXTERNAL_OES, GLES20.GL_TEXTURE_WRAP_T, GLES20.GL_CLAMP_TO_EDGE); + if (messageVideoMaskPath != null) { + try { + GLES20.glGenTextures(1, textures, 0); + GLES20.glBindTexture(GL10.GL_TEXTURE_2D, videoMaskTexture = textures[0]); + GLES20.glTexParameteri(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_MIN_FILTER, GL10.GL_LINEAR); + GLES20.glTexParameteri(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_MAG_FILTER, GL10.GL_LINEAR); + GLES20.glTexParameteri(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_WRAP_S, GL10.GL_CLAMP_TO_EDGE); + GLES20.glTexParameteri(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_WRAP_T, GL10.GL_CLAMP_TO_EDGE); + Bitmap bitmap = BitmapFactory.decodeFile(messageVideoMaskPath); + GLUtils.texImage2D(GL10.GL_TEXTURE_2D, 0, bitmap, 0); + bitmap.recycle(); + } catch (Exception e) { + FileLog.e(e); + videoMaskTexture = -1; + } + } + if (blurPath != null && cropState != null && cropState.useMatrix != null) { blur = new BlurringShader(); if (!blur.setup(transformedWidth / (float) transformedHeight, true, 0)) { @@ -980,7 +1089,7 @@ public void surfaceCreated() { } } } - if (filterShaders != null || imagePath != null || paintPath != null || mediaEntities != null || parts != null) { + if (filterShaders != null || imagePath != null || paintPath != null || messagePath != null || mediaEntities != null) { int vertexShader = FilterShaders.loadShader(GLES20.GL_VERTEX_SHADER, FilterShaders.simpleVertexShaderCode); int fragmentShader = FilterShaders.loadShader(GLES20.GL_FRAGMENT_SHADER, FilterShaders.simpleFragmentShaderCode); if (vertexShader != 0 && fragmentShader != 0) { @@ -1008,24 +1117,41 @@ public void surfaceCreated() { filterShaders.create(); filterShaders.setRenderData(null, 0, mTextureID, originalWidth, originalHeight); } - if (imagePath != null || paintPath != null) { - paintTexture = new int[(imagePath != null ? 1 : 0) + (paintPath != null ? 1 : 0)]; + if (imagePath != null || paintPath != null || messagePath != null) { + int texturePathesCount = 0; + if (imagePath != null) { + imagePathIndex = texturePathesCount++; + } + if (paintPath != null) { + paintPathIndex = texturePathesCount++; + } + if (messagePath != null) { + messagePathIndex = texturePathesCount++; + } + if (backgroundPath != null) { + backgroundPathIndex = texturePathesCount++; + } + paintTexture = new int[texturePathesCount]; GLES20.glGenTextures(paintTexture.length, paintTexture, 0); try { for (int a = 0; a < paintTexture.length; a++) { String path; int angle = 0, invert = 0; - if (a == 0 && imagePath != null) { + if (a == imagePathIndex) { path = imagePath; Pair orientation = AndroidUtilities.getImageOrientation(path); angle = orientation.first; invert = orientation.second; - } else { + } else if (a == paintPathIndex) { path = paintPath; + } else if (a == backgroundPathIndex) { + path = backgroundPath; + } else { // messagePathIndex + path = messagePath; } Bitmap bitmap = BitmapFactory.decodeFile(path); if (bitmap != null) { - if (a == 0 && imagePath != null && !useMatrixForImagePath) { + if (a == imagePathIndex && !useMatrixForImagePath) { Bitmap newBitmap = Bitmap.createBitmap(transformedWidth, transformedHeight, Bitmap.Config.ARGB_8888); newBitmap.eraseColor(0xff000000); Canvas canvas = new Canvas(newBitmap); @@ -1045,7 +1171,7 @@ public void surfaceCreated() { bitmap = newBitmap; } - if (a == 0 && imagePath != null) { + if (a == imagePathIndex) { imageWidth = bitmap.getWidth(); imageHeight = bitmap.getHeight(); } @@ -1062,56 +1188,7 @@ public void surfaceCreated() { FileLog.e(e); } } - if (parts != null && !parts.isEmpty()) { - partsTexture = new int[parts.size()]; - partsVerticesBuffer = new FloatBuffer[parts.size()]; - GLES20.glGenTextures(partsTexture.length, partsTexture, 0); - try { - for (int a = 0; a < partsTexture.length; a++) { - StoryEntry.Part part = parts.get(a); - String path = part.file.getAbsolutePath(); - - BitmapFactory.Options opts = new BitmapFactory.Options(); - opts.inJustDecodeBounds = true; - BitmapFactory.decodeFile(path, opts); - opts.inJustDecodeBounds = false; - opts.inSampleSize = StoryEntry.calculateInSampleSize(opts, transformedWidth, transformedHeight); - Bitmap bitmap = BitmapFactory.decodeFile(path, opts); - GLES20.glBindTexture(GL10.GL_TEXTURE_2D, partsTexture[a]); - GLES20.glTexParameteri(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_MIN_FILTER, GL10.GL_LINEAR); - GLES20.glTexParameteri(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_MAG_FILTER, GL10.GL_LINEAR); - GLES20.glTexParameteri(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_WRAP_S, GL10.GL_CLAMP_TO_EDGE); - GLES20.glTexParameteri(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_WRAP_T, GL10.GL_CLAMP_TO_EDGE); - GLUtils.texImage2D(GL10.GL_TEXTURE_2D, 0, bitmap, 0); - - final float[] verticesData = { - 0, 0, - part.width, 0, - 0, part.height, - part.width, part.height - }; - part.matrix.mapPoints(verticesData); - for (int i = 0; i < 4; i++) { - verticesData[i * 2] = verticesData[i * 2] / transformedWidth * 2f - 1f; - verticesData[i * 2 + 1] = 1f - verticesData[i * 2 + 1] / transformedHeight * 2f; - } - partsVerticesBuffer[a] = ByteBuffer.allocateDirect(verticesData.length * 4).order(ByteOrder.nativeOrder()).asFloatBuffer(); - partsVerticesBuffer[a].put(verticesData).position(0); - } - } catch (Throwable e2) { - FileLog.e(e2); - } - - final float[] textureData = { - 0, 0, - 1f, 0, - 0, 1f, - 1f, 1f - }; - partsTextureBuffer = ByteBuffer.allocateDirect(textureData.length * 4).order(ByteOrder.nativeOrder()).asFloatBuffer(); - partsTextureBuffer.put(textureData).position(0); - } - if (mediaEntities != null) { + if (mediaEntities != null || backgroundDrawable != null) { try { stickerBitmap = Bitmap.createBitmap(512, 512, Bitmap.Config.ARGB_8888); stickerTexture = new int[1]; @@ -1335,17 +1412,21 @@ private void initStickerEntity(VideoEditedInfo.MediaEntity entity) { entity.firstSeek = true; } } else { + String path = entity.text; + if (!TextUtils.isEmpty(entity.segmentedPath) && (entity.subType & 16) != 0) { + path = entity.segmentedPath; + } if (Build.VERSION.SDK_INT >= 19) { BitmapFactory.Options opts = new BitmapFactory.Options(); if (entity.type == VideoEditedInfo.MediaEntity.TYPE_PHOTO) { opts.inMutable = true; } - entity.bitmap = BitmapFactory.decodeFile(entity.text, opts); + entity.bitmap = BitmapFactory.decodeFile(path, opts); } else { try { - File path = new File(entity.text); - RandomAccessFile file = new RandomAccessFile(path, "r"); - ByteBuffer buffer = file.getChannel().map(FileChannel.MapMode.READ_ONLY, 0, path.length()); + File filePath = new File(path); + RandomAccessFile file = new RandomAccessFile(filePath, "r"); + ByteBuffer buffer = file.getChannel().map(FileChannel.MapMode.READ_ONLY, 0, filePath.length()); BitmapFactory.Options bmOptions = new BitmapFactory.Options(); bmOptions.inJustDecodeBounds = true; Utilities.loadWebpImage(null, buffer, buffer.limit(), bmOptions, true); @@ -1461,8 +1542,14 @@ public void release() { } public void changeFragmentShader(String fragmentExternalShader, String fragmentShader, boolean is300) { + String vertexCode; + if (messageVideoMaskPath != null) { + vertexCode = is300 ? VERTEX_SHADER_MASK_300 : VERTEX_SHADER_MASK; + } else { + vertexCode = is300 ? VERTEX_SHADER_300 : VERTEX_SHADER; + } if (NUM_EXTERNAL_SHADER >= 0 && NUM_EXTERNAL_SHADER < mProgram.length) { - int newProgram = createProgram(is300 ? VERTEX_SHADER_300 : VERTEX_SHADER, fragmentExternalShader, is300); + int newProgram = createProgram(vertexCode, fragmentExternalShader, is300); if (newProgram != 0) { GLES20.glDeleteProgram(mProgram[NUM_EXTERNAL_SHADER]); mProgram[NUM_EXTERNAL_SHADER] = newProgram; @@ -1471,7 +1558,7 @@ public void changeFragmentShader(String fragmentExternalShader, String fragmentS } } if (NUM_FILTER_SHADER >= 0 && NUM_FILTER_SHADER < mProgram.length) { - int newProgram = createProgram(is300 ? VERTEX_SHADER_300 : VERTEX_SHADER, fragmentShader, is300); + int newProgram = createProgram(vertexCode, fragmentShader, is300); if (newProgram != 0) { GLES20.glDeleteProgram(mProgram[NUM_FILTER_SHADER]); mProgram[NUM_FILTER_SHADER] = newProgram; diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/voip/VoIPService.java b/TMessagesProj/src/main/java/org/telegram/messenger/voip/VoIPService.java index 84bf20c0ba9..ebbf0456b58 100755 --- a/TMessagesProj/src/main/java/org/telegram/messenger/voip/VoIPService.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/voip/VoIPService.java @@ -77,8 +77,8 @@ import android.text.SpannableString; import android.text.TextUtils; import android.text.style.ForegroundColorSpan; -import android.util.Log; import android.util.LruCache; +import android.view.HapticFeedbackConstants; import android.view.KeyEvent; import android.view.View; import android.view.WindowManager; @@ -236,6 +236,7 @@ public class VoIPService extends Service implements SensorEventListener, AudioMa private BluetoothAdapter btAdapter; private Instance.TrafficStats prevTrafficStats; private boolean isBtHeadsetConnected; + private volatile boolean isCallEnded; private Runnable updateNotificationRunnable; @@ -1115,6 +1116,10 @@ public void switchCamera() { tgVoip[CAPTURE_DEVICE_CAMERA].switchCamera(!isFrontFaceCamera); } + public boolean isSwitchingCamera() { + return switchingCamera; + } + public void createCaptureDevice(boolean screencast) { int index = screencast ? CAPTURE_DEVICE_SCREEN : CAPTURE_DEVICE_CAMERA; int deviceType; @@ -1598,6 +1603,10 @@ private void startRatingActivity() { } } + public void sendCallRating(int rating) { + VoIPHelper.sendCallRating(privateCall.id, privateCall.access_hash, currentAccount, rating); + } + public byte[] getEncryptionKey() { return authKey; } @@ -2711,16 +2720,38 @@ public boolean isMicMute() { } public void toggleSpeakerphoneOrShowRouteSheet(Context context, boolean fromOverlayWindow) { + toggleSpeakerphoneOrShowRouteSheet(context, fromOverlayWindow, null); + } + + public void switchToSpeaker() { + AndroidUtilities.runOnUIThread(() -> { + VoipAudioManager vam = VoipAudioManager.get(); + if ((isBluetoothHeadsetConnected() && hasEarpiece()) || isHeadsetPlugged || isSpeakerphoneOn()) { + return; + } + vam.setSpeakerphoneOn(true); + vam.isBluetoothAndSpeakerOnAsync((isBluetoothOn, isSpeakerOn) -> { + updateOutputGainControlState(); + for (StateListener l : stateListeners) { + l.onAudioSettingsChanged(); + } + }); + }, 500); + } + + public void toggleSpeakerphoneOrShowRouteSheet(Context context, boolean fromOverlayWindow, Integer selectedPos) { if (isBluetoothHeadsetConnected() && hasEarpiece()) { BottomSheet.Builder builder = new BottomSheet.Builder(context) .setTitle(LocaleController.getString("VoipOutputDevices", R.string.VoipOutputDevices), true) + .selectedPos(selectedPos) + .setCellType(selectedPos != null ? BottomSheet.Builder.CELL_TYPE_CALL : 0) .setItems(new CharSequence[]{ LocaleController.getString("VoipAudioRoutingSpeaker", R.string.VoipAudioRoutingSpeaker), isHeadsetPlugged ? LocaleController.getString("VoipAudioRoutingHeadset", R.string.VoipAudioRoutingHeadset) : LocaleController.getString("VoipAudioRoutingEarpiece", R.string.VoipAudioRoutingEarpiece), currentBluetoothDeviceName != null ? currentBluetoothDeviceName : LocaleController.getString("VoipAudioRoutingBluetooth", R.string.VoipAudioRoutingBluetooth)}, - new int[]{R.drawable.calls_menu_speaker, - isHeadsetPlugged ? R.drawable.calls_menu_headset : R.drawable.calls_menu_phone, - R.drawable.calls_menu_bluetooth}, (dialog, which) -> { + new int[]{R.drawable.msg_call_speaker, + isHeadsetPlugged ? R.drawable.calls_menu_headset : R.drawable.msg_call_earpiece, + R.drawable.msg_call_bluetooth}, (dialog, which) -> { if (getSharedInstance() == null) { return; } @@ -2728,6 +2759,15 @@ public void toggleSpeakerphoneOrShowRouteSheet(Context context, boolean fromOver }); BottomSheet bottomSheet = builder.create(); + bottomSheet.setOnShowListener(dialog -> { + for (int i = 0; i < bottomSheet.getItemViews().size(); i++) { + bottomSheet.setItemColor(i, Theme.getColor(Theme.key_dialogTextBlack), Theme.getColor(Theme.key_dialogTextBlack)); + } + if (selectedPos != null) { + int selectedColor = Theme.getColor(Theme.key_dialogTextLink); + bottomSheet.setItemColor(selectedPos, selectedColor, selectedColor); + } + }); if (fromOverlayWindow) { if (Build.VERSION.SDK_INT >= 26) { bottomSheet.getWindow().setType(WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY); @@ -2752,7 +2792,13 @@ public void toggleSpeakerphoneOrShowRouteSheet(Context context, boolean fromOver } else { am.setBluetoothScoOn(!am.isBluetoothScoOn()); } - updateOutputGainControlState(); + vam.isBluetoothAndSpeakerOnAsync((isBluetoothOn, isSpeakerOn) -> { + updateOutputGainControlState(); + for (StateListener l : stateListeners) { + l.onAudioSettingsChanged(); + } + }); + return; } else { speakerphoneStateToSet = !speakerphoneStateToSet; } @@ -3482,6 +3528,10 @@ public static String getStringFromFile(String filePath) throws Exception { return ret; } + public boolean hasRate() { + return needRateCall || forceRating; + } + private void onTgVoipStop(Instance.FinalState finalState) { if (user == null) { return; @@ -3493,11 +3543,6 @@ private void onTgVoipStop(Instance.FinalState finalState) { e.printStackTrace(); } } - - if (needRateCall || forceRating || finalState.isRatingSuggested) { - startRatingActivity(); - needRateCall = false; - } if (needSendDebugLog && finalState.debugLog != null) { TLRPC.TL_phone_saveCallDebug req = new TLRPC.TL_phone_saveCallDebug(); req.debug = new TLRPC.TL_dataJSON(); @@ -3753,6 +3798,10 @@ private void configureDeviceForCall() { } else { audioRouteToSet = AUDIO_ROUTE_EARPIECE; } + if (lastSensorEvent != null) { + //For the case when the phone was put to the ear before configureDeviceForCall. + onSensorChanged(lastSensorEvent); + } } updateOutputGainControlState(); audioConfigured = true; @@ -3787,9 +3836,12 @@ private void fetchBluetoothDeviceName() { } } + private SensorEvent lastSensorEvent; + @SuppressLint("NewApi") @Override public void onSensorChanged(SensorEvent event) { + lastSensorEvent = event; if (unmutedByHold || remoteVideoState == Instance.VIDEO_STATE_ACTIVE || videoState[CAPTURE_DEVICE_CAMERA] == Instance.VIDEO_STATE_ACTIVE) { return; } @@ -3801,6 +3853,7 @@ public void onSensorChanged(SensorEvent event) { } boolean newIsNear = event.values[0] < Math.min(event.sensor.getMaximumRange(), 3); checkIsNear(newIsNear); + NotificationCenter.getGlobalInstance().postNotificationName(NotificationCenter.nearEarEvent, newIsNear); } } @@ -4255,9 +4308,9 @@ public void onConnectionStateChanged(int newState, boolean inTransition) { if (groupCall == null && !wasEstablished) { wasEstablished = true; if (!isProximityNear && !privateCall.video) { - Vibrator vibrator = (Vibrator) getSystemService(VIBRATOR_SERVICE); - if (vibrator.hasVibrator()) { - vibrator.vibrate(100); + try { + LaunchActivity.getLastFragment().getFragmentView().performHapticFeedback(HapticFeedbackConstants.KEYBOARD_TAP, HapticFeedbackConstants.FLAG_IGNORE_GLOBAL_SETTING); + } catch (Exception ignore) { } } AndroidUtilities.runOnUIThread(new Runnable() { @@ -4276,7 +4329,7 @@ public void run() { } } } - if (newState == STATE_RECONNECTING) { + if (newState == STATE_RECONNECTING && !isCallEnded) { Utilities.globalQueue.postRunnable(() -> { if (spPlayId != 0) { soundPool.stop(spPlayId); @@ -4324,6 +4377,7 @@ private void callEnded() { if (BuildVars.LOGS_ENABLED) { FileLog.d("Call " + getCallID() + " ended"); } + isCallEnded = true; if (groupCall != null && (!playedConnectedSound || onDestroyRunnable != null)) { needPlayEndSound = false; } diff --git a/TMessagesProj/src/main/java/org/telegram/messenger/voip/VoipAudioManager.java b/TMessagesProj/src/main/java/org/telegram/messenger/voip/VoipAudioManager.java index e96edb73edc..b2b93bea179 100644 --- a/TMessagesProj/src/main/java/org/telegram/messenger/voip/VoipAudioManager.java +++ b/TMessagesProj/src/main/java/org/telegram/messenger/voip/VoipAudioManager.java @@ -4,6 +4,8 @@ import android.media.AudioManager; + +import org.telegram.messenger.AndroidUtilities; import org.telegram.messenger.ApplicationLoader; import org.telegram.messenger.Utilities; @@ -47,6 +49,15 @@ public boolean isSpeakerphoneOn() { return isSpeakerphoneOn; } + public void isBluetoothAndSpeakerOnAsync(Utilities.Callback2 onDone) { + Utilities.globalQueue.postRunnable(() -> { + AudioManager audioManager = getAudioManager(); + boolean isBluetoothScoOn = audioManager.isBluetoothScoOn(); + boolean isSpeakerphoneOn = audioManager.isSpeakerphoneOn(); + AndroidUtilities.runOnUIThread(() -> onDone.run(isBluetoothScoOn, isSpeakerphoneOn)); + }); + } + private AudioManager getAudioManager() { return (AudioManager) ApplicationLoader.applicationContext.getSystemService(AUDIO_SERVICE); } diff --git a/TMessagesProj/src/main/java/org/telegram/tgnet/ConnectionsManager.java b/TMessagesProj/src/main/java/org/telegram/tgnet/ConnectionsManager.java index 8a50fb73759..5f373304c0b 100644 --- a/TMessagesProj/src/main/java/org/telegram/tgnet/ConnectionsManager.java +++ b/TMessagesProj/src/main/java/org/telegram/tgnet/ConnectionsManager.java @@ -245,7 +245,7 @@ public ConnectionsManager(int instance) { if (getUserConfig().getCurrentUser() != null) { userPremium = getUserConfig().getCurrentUser().premium; } - init(BuildVars.BUILD_VERSION, TLRPC.LAYER, BuildVars.APP_ID, deviceModel, systemVersion, appVersion, langCode, systemLangCode, configPath, FileLog.getNetworkLogPath(), pushString, fingerprint, timezoneOffset, getUserConfig().getClientUserId(), userPremium, enablePushConnection); + init(SharedConfig.buildVersion(), TLRPC.LAYER, BuildVars.APP_ID, deviceModel, systemVersion, appVersion, langCode, systemLangCode, configPath, FileLog.getNetworkLogPath(), pushString, fingerprint, timezoneOffset, getUserConfig().getClientUserId(), userPremium, enablePushConnection); } private String getRegId() { diff --git a/TMessagesProj/src/main/java/org/telegram/tgnet/TLRPC.java b/TMessagesProj/src/main/java/org/telegram/tgnet/TLRPC.java index f4a1d7273ac..b6fdda8a0e2 100644 --- a/TMessagesProj/src/main/java/org/telegram/tgnet/TLRPC.java +++ b/TMessagesProj/src/main/java/org/telegram/tgnet/TLRPC.java @@ -17,6 +17,7 @@ import androidx.annotation.Nullable; +import org.telegram.messenger.ApplicationLoader; import org.telegram.messenger.BuildVars; import org.telegram.messenger.FileLoader; import org.telegram.messenger.FileLog; @@ -75,7 +76,7 @@ public class TLRPC { public static final int MESSAGE_FLAG_HAS_BOT_ID = 0x00000800; public static final int MESSAGE_FLAG_EDITED = 0x00008000; - public static final int LAYER = 167; + public static final int LAYER = 170; public static class TL_stats_megagroupStats extends TLObject { public static final int constructor = 0xef7ff916; @@ -8651,6 +8652,9 @@ public void readParams(AbstractSerializedData stream, boolean exception) { flags = stream.readInt32(exception); nopremium = (flags & 8) != 0; spoiler = (flags & 16) != 0; + video = (flags & 64) != 0; + round = (flags & 128) != 0; + voice = (flags & 256) != 0; if ((flags & 1) != 0) { document = Document.TLdeserialize(stream, stream.readInt32(exception), exception); } @@ -8666,6 +8670,9 @@ public void serializeToStream(AbstractSerializedData stream) { stream.writeInt32(constructor); flags = nopremium ? (flags | 8) : (flags &~ 8); flags = spoiler ? (flags | 16) : (flags &~ 16); + flags = video ? (flags | 64) : (flags &~ 64); + flags = round ? (flags | 128) : (flags &~ 128); + flags = voice ? (flags | 256) : (flags &~ 256); stream.writeInt32(flags); if ((flags & 1) != 0) { document.serializeToStream(stream); @@ -12303,7 +12310,7 @@ public void serializeToStream(AbstractSerializedData stream) { } } } - + public static class WebPageAttribute extends TLObject { public int flags; @@ -13088,6 +13095,7 @@ public static abstract class ChatFull extends TLObject { public boolean view_forum_as_messages; public ChatReactions available_reactions; public TL_stories.PeerStories stories; + public WallPaper wallpaper; public long inviterId; //custom public int invitesCount; //custom @@ -13098,9 +13106,12 @@ public static ChatFull TLdeserialize(AbstractSerializedData stream, int construc case 0xc9d31138: result = new TL_chatFull(); break; - case 0x723027bd: + case TL_channelFull.constructor: result = new TL_channelFull(); break; + case TL_channelFull_layer167.constructor: + result = new TL_channelFull_layer167(); + break; case 0xf2355507: result = new TL_channelFull_layer162(); break; @@ -15549,7 +15560,7 @@ public void serializeToStream(AbstractSerializedData stream) { } public static class TL_channelFull extends ChatFull { - public static final int constructor = 0x723027bd; + public static final int constructor = 0xf2bcb6f; public void readParams(AbstractSerializedData stream, boolean exception) { flags = stream.readInt32(exception); @@ -15692,6 +15703,9 @@ public void readParams(AbstractSerializedData stream, boolean exception) { if ((flags2 & 16) != 0) { stories = TL_stories.PeerStories.TLdeserialize(stream, stream.readInt32(exception), exception); } + if ((flags2 & 128) != 0) { + wallpaper = WallPaper.TLdeserialize(stream, stream.readInt32(exception), exception); + } } public void serializeToStream(AbstractSerializedData stream) { @@ -15817,11 +15831,14 @@ public void serializeToStream(AbstractSerializedData stream) { if ((flags2 & 16) != 0) { stories.serializeToStream(stream); } + if ((flags2 & 128) != 0) { + wallpaper.serializeToStream(stream); + } } } - public static class TL_channelFull_layer162 extends TL_channelFull { - public static final int constructor = 0xf2355507; + public static class TL_channelFull_layer167 extends TL_channelFull { + public static final int constructor = 0x723027bd; public void readParams(AbstractSerializedData stream, boolean exception) { flags = stream.readInt32(exception); @@ -15838,6 +15855,8 @@ public void readParams(AbstractSerializedData stream, boolean exception) { antispam = (flags2 & 2) != 0; participants_hidden = (flags2 & 4) != 0; translations_disabled = (flags2 & 8) != 0; + stories_pinned_available = (flags2 & 32) != 0; + view_forum_as_messages = (flags2 & 64) != 0; id = stream.readInt64(exception); about = stream.readString(exception); if ((flags & 1) != 0) { @@ -15959,6 +15978,9 @@ public void readParams(AbstractSerializedData stream, boolean exception) { if ((flags & 1073741824) != 0) { available_reactions = ChatReactions.TLdeserialize(stream, stream.readInt32(exception), exception); } + if ((flags2 & 16) != 0) { + stories = TL_stories.PeerStories.TLdeserialize(stream, stream.readInt32(exception), exception); + } } public void serializeToStream(AbstractSerializedData stream) { @@ -15976,6 +15998,8 @@ public void serializeToStream(AbstractSerializedData stream) { flags2 = antispam ? (flags2 | 2) : (flags2 &~ 2); flags2 = participants_hidden ? (flags2 | 4) : (flags2 &~ 4); flags2 = translations_disabled ? (flags2 | 8) : (flags2 &~ 8); + flags2 = stories_pinned_available ? (flags2 | 32) : (flags2 &~ 32); + flags2 = view_forum_as_messages ? (flags2 | 64) : (flags2 &~ 64); stream.writeInt32(flags2); stream.writeInt64(id); stream.writeString(about); @@ -16079,11 +16103,14 @@ public void serializeToStream(AbstractSerializedData stream) { if ((flags & 1073741824) != 0) { available_reactions.serializeToStream(stream); } + if ((flags2 & 16) != 0) { + stories.serializeToStream(stream); + } } } - public static class TL_channelFull_layer144 extends ChatFull { - public static final int constructor = 0xea68a619; + public static class TL_channelFull_layer162 extends TL_channelFull { + public static final int constructor = 0xf2355507; public void readParams(AbstractSerializedData stream, boolean exception) { flags = stream.readInt32(exception); @@ -16097,6 +16124,9 @@ public void readParams(AbstractSerializedData stream, boolean exception) { blocked = (flags & 4194304) != 0; flags2 = stream.readInt32(exception); can_delete_channel = (flags2 & 1) != 0; + antispam = (flags2 & 2) != 0; + participants_hidden = (flags2 & 4) != 0; + translations_disabled = (flags2 & 8) != 0; id = stream.readInt64(exception); about = stream.readString(exception); if ((flags & 1) != 0) { @@ -16216,17 +16246,7 @@ public void readParams(AbstractSerializedData stream, boolean exception) { default_send_as = Peer.TLdeserialize(stream, stream.readInt32(exception), exception); } if ((flags & 1073741824) != 0) { - magic = stream.readInt32(exception); - if (magic != 0x1cb5c415) { - if (exception) { - throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); - } - return; - } - count = stream.readInt32(exception); - for (int a = 0; a < count; a++) { - available_reactions_legacy.add(stream.readString(exception)); - } + available_reactions = ChatReactions.TLdeserialize(stream, stream.readInt32(exception), exception); } } @@ -16242,6 +16262,9 @@ public void serializeToStream(AbstractSerializedData stream) { flags = blocked ? (flags | 4194304) : (flags &~ 4194304); stream.writeInt32(flags); flags2 = can_delete_channel ? (flags2 | 1) : (flags2 &~ 1); + flags2 = antispam ? (flags2 | 2) : (flags2 &~ 2); + flags2 = participants_hidden ? (flags2 | 4) : (flags2 &~ 4); + flags2 = translations_disabled ? (flags2 | 8) : (flags2 &~ 8); stream.writeInt32(flags2); stream.writeInt64(id); stream.writeString(about); @@ -16343,18 +16366,13 @@ public void serializeToStream(AbstractSerializedData stream) { default_send_as.serializeToStream(stream); } if ((flags & 1073741824) != 0) { - stream.writeInt32(0x1cb5c415); - count = available_reactions_legacy.size(); - stream.writeInt32(count); - for (int a = 0; a < count; a++) { - stream.writeString(available_reactions_legacy.get(a)); - } + available_reactions.serializeToStream(stream); } } } - public static class TL_channelFull_layer139 extends ChatFull { - public static final int constructor = 0xe13c3d20; + public static class TL_channelFull_layer144 extends ChatFull { + public static final int constructor = 0xea68a619; public void readParams(AbstractSerializedData stream, boolean exception) { flags = stream.readInt32(exception); @@ -16366,6 +16384,8 @@ public void readParams(AbstractSerializedData stream, boolean exception) { has_scheduled = (flags & 524288) != 0; can_view_stats = (flags & 1048576) != 0; blocked = (flags & 4194304) != 0; + flags2 = stream.readInt32(exception); + can_delete_channel = (flags2 & 1) != 0; id = stream.readInt64(exception); about = stream.readString(exception); if ((flags & 1) != 0) { @@ -16510,6 +16530,8 @@ public void serializeToStream(AbstractSerializedData stream) { flags = can_view_stats ? (flags | 1048576) : (flags &~ 1048576); flags = blocked ? (flags | 4194304) : (flags &~ 4194304); stream.writeInt32(flags); + flags2 = can_delete_channel ? (flags2 | 1) : (flags2 &~ 1); + stream.writeInt32(flags2); stream.writeInt64(id); stream.writeString(about); if ((flags & 1) != 0) { @@ -16620,9 +16642,8 @@ public void serializeToStream(AbstractSerializedData stream) { } } - public static class TL_channelFull_layer131 extends TL_channelFull { - public static final int constructor = 0x548c3f93; - + public static class TL_channelFull_layer139 extends ChatFull { + public static final int constructor = 0xe13c3d20; public void readParams(AbstractSerializedData stream, boolean exception) { flags = stream.readInt32(exception); @@ -16657,7 +16678,7 @@ public void readParams(AbstractSerializedData stream, boolean exception) { chat_photo = Photo.TLdeserialize(stream, stream.readInt32(exception), exception); notify_settings = PeerNotifySettings.TLdeserialize(stream, stream.readInt32(exception), exception); if ((flags & 8388608) != 0) { - exported_invite = (TL_chatInviteExported) ExportedChatInvite.TLdeserialize(stream, stream.readInt32(exception), exception); + exported_invite = ExportedChatInvite.TLdeserialize(stream, stream.readInt32(exception), exception); } int magic = stream.readInt32(exception); if (magic != 0x1cb5c415) { @@ -16730,6 +16751,41 @@ public void readParams(AbstractSerializedData stream, boolean exception) { if ((flags & 67108864) != 0) { groupcall_default_join_as = Peer.TLdeserialize(stream, stream.readInt32(exception), exception); } + if ((flags & 134217728) != 0) { + theme_emoticon = stream.readString(exception); + } + if ((flags & 268435456) != 0) { + requests_pending = stream.readInt32(exception); + } + if ((flags & 268435456) != 0) { + magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + recent_requesters.add(stream.readInt64(exception)); + } + } + if ((flags & 536870912) != 0) { + default_send_as = Peer.TLdeserialize(stream, stream.readInt32(exception), exception); + } + if ((flags & 1073741824) != 0) { + magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + available_reactions_legacy.add(stream.readString(exception)); + } + } } public void serializeToStream(AbstractSerializedData stream) { @@ -16825,11 +16881,36 @@ public void serializeToStream(AbstractSerializedData stream) { if ((flags & 67108864) != 0) { groupcall_default_join_as.serializeToStream(stream); } + if ((flags & 134217728) != 0) { + stream.writeString(theme_emoticon); + } + if ((flags & 268435456) != 0) { + stream.writeInt32(requests_pending); + } + if ((flags & 268435456) != 0) { + stream.writeInt32(0x1cb5c415); + count = recent_requesters.size(); + stream.writeInt32(count); + for (int a = 0; a < count; a++) { + stream.writeInt64(recent_requesters.get(a)); + } + } + if ((flags & 536870912) != 0) { + default_send_as.serializeToStream(stream); + } + if ((flags & 1073741824) != 0) { + stream.writeInt32(0x1cb5c415); + count = available_reactions_legacy.size(); + stream.writeInt32(count); + for (int a = 0; a < count; a++) { + stream.writeString(available_reactions_legacy.get(a)); + } + } } } - public static class TL_channelFull_layer122 extends TL_channelFull { - public static final int constructor = 0xef3a6acd; + public static class TL_channelFull_layer131 extends TL_channelFull { + public static final int constructor = 0x548c3f93; public void readParams(AbstractSerializedData stream, boolean exception) { @@ -16842,7 +16923,7 @@ public void readParams(AbstractSerializedData stream, boolean exception) { has_scheduled = (flags & 524288) != 0; can_view_stats = (flags & 1048576) != 0; blocked = (flags & 4194304) != 0; - id = stream.readInt32(exception); + id = stream.readInt64(exception); about = stream.readString(exception); if ((flags & 1) != 0) { participants_count = stream.readInt32(exception); @@ -16864,9 +16945,8 @@ public void readParams(AbstractSerializedData stream, boolean exception) { unread_count = stream.readInt32(exception); chat_photo = Photo.TLdeserialize(stream, stream.readInt32(exception), exception); notify_settings = PeerNotifySettings.TLdeserialize(stream, stream.readInt32(exception), exception); - ExportedChatInvite invite = ExportedChatInvite.TLdeserialize(stream, stream.readInt32(exception), exception); - if (invite instanceof TL_chatInviteExported) { - exported_invite = (TL_chatInviteExported) invite; + if ((flags & 8388608) != 0) { + exported_invite = (TL_chatInviteExported) ExportedChatInvite.TLdeserialize(stream, stream.readInt32(exception), exception); } int magic = stream.readInt32(exception); if (magic != 0x1cb5c415) { @@ -16884,7 +16964,7 @@ public void readParams(AbstractSerializedData stream, boolean exception) { bot_info.add(object); } if ((flags & 16) != 0) { - migrated_from_chat_id = stream.readInt32(exception); + migrated_from_chat_id = stream.readInt64(exception); } if ((flags & 16) != 0) { migrated_from_max_id = stream.readInt32(exception); @@ -16902,7 +16982,7 @@ public void readParams(AbstractSerializedData stream, boolean exception) { folder_id = stream.readInt32(exception); } if ((flags & 16384) != 0) { - linked_chat_id = stream.readInt32(exception); + linked_chat_id = stream.readInt64(exception); } if ((flags & 32768) != 0) { location = ChannelLocation.TLdeserialize(stream, stream.readInt32(exception), exception); @@ -16920,6 +17000,25 @@ public void readParams(AbstractSerializedData stream, boolean exception) { if ((flags & 2097152) != 0) { call = TL_inputGroupCall.TLdeserialize(stream, stream.readInt32(exception), exception); } + if ((flags & 16777216) != 0) { + ttl_period = stream.readInt32(exception); + } + if ((flags & 33554432) != 0) { + magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + pending_suggestions.add(stream.readString(exception)); + } + } + if ((flags & 67108864) != 0) { + groupcall_default_join_as = Peer.TLdeserialize(stream, stream.readInt32(exception), exception); + } } public void serializeToStream(AbstractSerializedData stream) { @@ -16933,7 +17032,7 @@ public void serializeToStream(AbstractSerializedData stream) { flags = can_view_stats ? (flags | 1048576) : (flags &~ 1048576); flags = blocked ? (flags | 4194304) : (flags &~ 4194304); stream.writeInt32(flags); - stream.writeInt32((int) id); + stream.writeInt64(id); stream.writeString(about); if ((flags & 1) != 0) { stream.writeInt32(participants_count); @@ -16955,10 +17054,8 @@ public void serializeToStream(AbstractSerializedData stream) { stream.writeInt32(unread_count); chat_photo.serializeToStream(stream); notify_settings.serializeToStream(stream); - if (exported_invite != null) { + if ((flags & 8388608) != 0) { exported_invite.serializeToStream(stream); - } else { - new TLRPC.TL_chatInviteEmpty_layer122().serializeToStream(stream); } stream.writeInt32(0x1cb5c415); int count = bot_info.size(); @@ -16967,7 +17064,7 @@ public void serializeToStream(AbstractSerializedData stream) { bot_info.get(a).serializeToStream(stream); } if ((flags & 16) != 0) { - stream.writeInt32((int) migrated_from_chat_id); + stream.writeInt64(migrated_from_chat_id); } if ((flags & 16) != 0) { stream.writeInt32(migrated_from_max_id); @@ -16985,7 +17082,7 @@ public void serializeToStream(AbstractSerializedData stream) { stream.writeInt32(folder_id); } if ((flags & 16384) != 0) { - stream.writeInt32((int) linked_chat_id); + stream.writeInt64(linked_chat_id); } if ((flags & 32768) != 0) { location.serializeToStream(stream); @@ -17003,11 +17100,25 @@ public void serializeToStream(AbstractSerializedData stream) { if ((flags & 2097152) != 0) { call.serializeToStream(stream); } + if ((flags & 16777216) != 0) { + stream.writeInt32(ttl_period); + } + if ((flags & 33554432) != 0) { + stream.writeInt32(0x1cb5c415); + count = pending_suggestions.size(); + stream.writeInt32(count); + for (int a = 0; a < count; a++) { + stream.writeString(pending_suggestions.get(a)); + } + } + if ((flags & 67108864) != 0) { + groupcall_default_join_as.serializeToStream(stream); + } } } - public static class TL_channelFull_layer121 extends TL_channelFull { - public static final int constructor = 0xf0e6672a; + public static class TL_channelFull_layer122 extends TL_channelFull { + public static final int constructor = 0xef3a6acd; public void readParams(AbstractSerializedData stream, boolean exception) { @@ -17095,6 +17206,9 @@ public void readParams(AbstractSerializedData stream, boolean exception) { stats_dc = stream.readInt32(exception); } pts = stream.readInt32(exception); + if ((flags & 2097152) != 0) { + call = TL_inputGroupCall.TLdeserialize(stream, stream.readInt32(exception), exception); + } } public void serializeToStream(AbstractSerializedData stream) { @@ -17130,7 +17244,11 @@ public void serializeToStream(AbstractSerializedData stream) { stream.writeInt32(unread_count); chat_photo.serializeToStream(stream); notify_settings.serializeToStream(stream); - exported_invite.serializeToStream(stream); + if (exported_invite != null) { + exported_invite.serializeToStream(stream); + } else { + new TLRPC.TL_chatInviteEmpty_layer122().serializeToStream(stream); + } stream.writeInt32(0x1cb5c415); int count = bot_info.size(); stream.writeInt32(count); @@ -17171,11 +17289,14 @@ public void serializeToStream(AbstractSerializedData stream) { stream.writeInt32(stats_dc); } stream.writeInt32(pts); + if ((flags & 2097152) != 0) { + call.serializeToStream(stream); + } } } - public static class TL_channelFull_layer103 extends TL_channelFull { - public static final int constructor = 0x10916653; + public static class TL_channelFull_layer121 extends TL_channelFull { + public static final int constructor = 0xf0e6672a; public void readParams(AbstractSerializedData stream, boolean exception) { @@ -17184,8 +17305,10 @@ public void readParams(AbstractSerializedData stream, boolean exception) { can_set_username = (flags & 64) != 0; can_set_stickers = (flags & 128) != 0; hidden_prehistory = (flags & 1024) != 0; - can_view_stats = (flags & 4096) != 0; can_set_location = (flags & 65536) != 0; + has_scheduled = (flags & 524288) != 0; + can_view_stats = (flags & 1048576) != 0; + blocked = (flags & 4194304) != 0; id = stream.readInt32(exception); about = stream.readString(exception); if ((flags & 1) != 0) { @@ -17251,6 +17374,15 @@ public void readParams(AbstractSerializedData stream, boolean exception) { if ((flags & 32768) != 0) { location = ChannelLocation.TLdeserialize(stream, stream.readInt32(exception), exception); } + if ((flags & 131072) != 0) { + slowmode_seconds = stream.readInt32(exception); + } + if ((flags & 262144) != 0) { + slowmode_next_send_date = stream.readInt32(exception); + } + if ((flags & 4096) != 0) { + stats_dc = stream.readInt32(exception); + } pts = stream.readInt32(exception); } @@ -17260,8 +17392,10 @@ public void serializeToStream(AbstractSerializedData stream) { flags = can_set_username ? (flags | 64) : (flags &~ 64); flags = can_set_stickers ? (flags | 128) : (flags &~ 128); flags = hidden_prehistory ? (flags | 1024) : (flags &~ 1024); - flags = can_view_stats ? (flags | 4096) : (flags &~ 4096); flags = can_set_location ? (flags | 65536) : (flags &~ 65536); + flags = has_scheduled ? (flags | 524288) : (flags &~ 524288); + flags = can_view_stats ? (flags | 1048576) : (flags &~ 1048576); + flags = blocked ? (flags | 4194304) : (flags &~ 4194304); stream.writeInt32(flags); stream.writeInt32((int) id); stream.writeString(about); @@ -17316,12 +17450,21 @@ public void serializeToStream(AbstractSerializedData stream) { if ((flags & 32768) != 0) { location.serializeToStream(stream); } + if ((flags & 131072) != 0) { + stream.writeInt32(slowmode_seconds); + } + if ((flags & 262144) != 0) { + stream.writeInt32(slowmode_next_send_date); + } + if ((flags & 4096) != 0) { + stream.writeInt32(stats_dc); + } stream.writeInt32(pts); } } - public static class TL_channelFull_layer101 extends TL_channelFull { - public static final int constructor = 0x9882e516; + public static class TL_channelFull_layer103 extends TL_channelFull { + public static final int constructor = 0x10916653; public void readParams(AbstractSerializedData stream, boolean exception) { @@ -17331,6 +17474,7 @@ public void readParams(AbstractSerializedData stream, boolean exception) { can_set_stickers = (flags & 128) != 0; hidden_prehistory = (flags & 1024) != 0; can_view_stats = (flags & 4096) != 0; + can_set_location = (flags & 65536) != 0; id = stream.readInt32(exception); about = stream.readString(exception); if ((flags & 1) != 0) { @@ -17390,9 +17534,12 @@ public void readParams(AbstractSerializedData stream, boolean exception) { if ((flags & 2048) != 0) { folder_id = stream.readInt32(exception); } - if ((flags & 8192) != 0) { + if ((flags & 16384) != 0) { linked_chat_id = stream.readInt32(exception); } + if ((flags & 32768) != 0) { + location = ChannelLocation.TLdeserialize(stream, stream.readInt32(exception), exception); + } pts = stream.readInt32(exception); } @@ -17403,6 +17550,7 @@ public void serializeToStream(AbstractSerializedData stream) { flags = can_set_stickers ? (flags | 128) : (flags &~ 128); flags = hidden_prehistory ? (flags | 1024) : (flags &~ 1024); flags = can_view_stats ? (flags | 4096) : (flags &~ 4096); + flags = can_set_location ? (flags | 65536) : (flags &~ 65536); stream.writeInt32(flags); stream.writeInt32((int) id); stream.writeString(about); @@ -17451,15 +17599,18 @@ public void serializeToStream(AbstractSerializedData stream) { if ((flags & 2048) != 0) { stream.writeInt32(folder_id); } - if ((flags & 8192) != 0) { + if ((flags & 16384) != 0) { stream.writeInt32((int) linked_chat_id); } + if ((flags & 32768) != 0) { + location.serializeToStream(stream); + } stream.writeInt32(pts); } } - public static class TL_channelFull_layer99 extends TL_channelFull { - public static final int constructor = 0x3648977; + public static class TL_channelFull_layer101 extends TL_channelFull { + public static final int constructor = 0x9882e516; public void readParams(AbstractSerializedData stream, boolean exception) { @@ -17528,6 +17679,9 @@ public void readParams(AbstractSerializedData stream, boolean exception) { if ((flags & 2048) != 0) { folder_id = stream.readInt32(exception); } + if ((flags & 8192) != 0) { + linked_chat_id = stream.readInt32(exception); + } pts = stream.readInt32(exception); } @@ -17586,12 +17740,147 @@ public void serializeToStream(AbstractSerializedData stream) { if ((flags & 2048) != 0) { stream.writeInt32(folder_id); } + if ((flags & 8192) != 0) { + stream.writeInt32((int) linked_chat_id); + } stream.writeInt32(pts); } } - public static class TL_channelFull_layer98 extends TL_channelFull { - public static final int constructor = 0x1c87a71a; + public static class TL_channelFull_layer99 extends TL_channelFull { + public static final int constructor = 0x3648977; + + + public void readParams(AbstractSerializedData stream, boolean exception) { + flags = stream.readInt32(exception); + can_view_participants = (flags & 8) != 0; + can_set_username = (flags & 64) != 0; + can_set_stickers = (flags & 128) != 0; + hidden_prehistory = (flags & 1024) != 0; + can_view_stats = (flags & 4096) != 0; + id = stream.readInt32(exception); + about = stream.readString(exception); + if ((flags & 1) != 0) { + participants_count = stream.readInt32(exception); + } + if ((flags & 2) != 0) { + admins_count = stream.readInt32(exception); + } + if ((flags & 4) != 0) { + kicked_count = stream.readInt32(exception); + } + if ((flags & 4) != 0) { + banned_count = stream.readInt32(exception); + } + if ((flags & 8192) != 0) { + online_count = stream.readInt32(exception); + } + read_inbox_max_id = stream.readInt32(exception); + read_outbox_max_id = stream.readInt32(exception); + unread_count = stream.readInt32(exception); + chat_photo = Photo.TLdeserialize(stream, stream.readInt32(exception), exception); + notify_settings = PeerNotifySettings.TLdeserialize(stream, stream.readInt32(exception), exception); + ExportedChatInvite invite = ExportedChatInvite.TLdeserialize(stream, stream.readInt32(exception), exception); + if (invite instanceof TL_chatInviteExported) { + exported_invite = (TL_chatInviteExported) invite; + } + int magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + int count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + BotInfo object = BotInfo.TLdeserialize(stream, stream.readInt32(exception), exception); + if (object == null) { + return; + } + bot_info.add(object); + } + if ((flags & 16) != 0) { + migrated_from_chat_id = stream.readInt32(exception); + } + if ((flags & 16) != 0) { + migrated_from_max_id = stream.readInt32(exception); + } + if ((flags & 32) != 0) { + pinned_msg_id = stream.readInt32(exception); + } + if ((flags & 256) != 0) { + stickerset = StickerSet.TLdeserialize(stream, stream.readInt32(exception), exception); + } + if ((flags & 512) != 0) { + available_min_id = stream.readInt32(exception); + } + if ((flags & 2048) != 0) { + folder_id = stream.readInt32(exception); + } + pts = stream.readInt32(exception); + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + flags = can_view_participants ? (flags | 8) : (flags &~ 8); + flags = can_set_username ? (flags | 64) : (flags &~ 64); + flags = can_set_stickers ? (flags | 128) : (flags &~ 128); + flags = hidden_prehistory ? (flags | 1024) : (flags &~ 1024); + flags = can_view_stats ? (flags | 4096) : (flags &~ 4096); + stream.writeInt32(flags); + stream.writeInt32((int) id); + stream.writeString(about); + if ((flags & 1) != 0) { + stream.writeInt32(participants_count); + } + if ((flags & 2) != 0) { + stream.writeInt32(admins_count); + } + if ((flags & 4) != 0) { + stream.writeInt32(kicked_count); + } + if ((flags & 4) != 0) { + stream.writeInt32(banned_count); + } + if ((flags & 8192) != 0) { + stream.writeInt32(online_count); + } + stream.writeInt32(read_inbox_max_id); + stream.writeInt32(read_outbox_max_id); + stream.writeInt32(unread_count); + chat_photo.serializeToStream(stream); + notify_settings.serializeToStream(stream); + exported_invite.serializeToStream(stream); + stream.writeInt32(0x1cb5c415); + int count = bot_info.size(); + stream.writeInt32(count); + for (int a = 0; a < count; a++) { + bot_info.get(a).serializeToStream(stream); + } + if ((flags & 16) != 0) { + stream.writeInt32((int) migrated_from_chat_id); + } + if ((flags & 16) != 0) { + stream.writeInt32(migrated_from_max_id); + } + if ((flags & 32) != 0) { + stream.writeInt32(pinned_msg_id); + } + if ((flags & 256) != 0) { + stickerset.serializeToStream(stream); + } + if ((flags & 512) != 0) { + stream.writeInt32(available_min_id); + } + if ((flags & 2048) != 0) { + stream.writeInt32(folder_id); + } + stream.writeInt32(pts); + } + } + + public static class TL_channelFull_layer98 extends TL_channelFull { + public static final int constructor = 0x1c87a71a; public void readParams(AbstractSerializedData stream, boolean exception) { @@ -19971,7 +20260,10 @@ public static KeyboardButton TLdeserialize(AbstractSerializedData stream, int co case 0x13767230: result = new TL_keyboardButtonWebView(); break; - case 0xd0b468c: + case TL_keyboardButtonRequestPeer_layer168.constructor: + result = new TL_keyboardButtonRequestPeer_layer168(); + break; + case TL_keyboardButtonRequestPeer.constructor: result = new TL_keyboardButtonRequestPeer(); break; } @@ -24156,9 +24448,15 @@ public void serializeToStream(AbstractSerializedData stream) { stream.writeInt32(stories_max_id); } if ((flags2 & 256) != 0) { + if (color == null) { + color = new TL_peerColor(); + } color.serializeToStream(stream); } if ((flags2 & 512) != 0) { + if (profile_color == null) { + profile_color = new TL_peerColor(); + } profile_color.serializeToStream(stream); } } @@ -26321,9 +26619,12 @@ public static MessageAction TLdeserialize(AbstractSerializedData stream, int con case 0x55555550: result = new TL_messageActionUserJoined(); break; - case 0xd2cfdb0e: + case TL_messageActionGiftCode.constructor: result = new TL_messageActionGiftCode(); break; + case TL_messageActionGiftCode_layer167.constructor: + result = new TL_messageActionGiftCode_layer167(); + break; case 0xb18a431c: result = new TL_messageActionTopicEdit_layer149(); break; @@ -26420,7 +26721,10 @@ public static MessageAction TLdeserialize(AbstractSerializedData stream, int con case 0xc83d6aec: result = new TL_messageActionGiftPremium(); break; - case 0xfe77345d: + case TL_messageActionRequestedPeer_layer168.constructor: + result = new TL_messageActionRequestedPeer_layer168(); + break; + case TL_messageActionRequestedPeer.constructor: result = new TL_messageActionRequestedPeer(); break; } @@ -33055,6 +33359,9 @@ public static InputStickerSet TLdeserialize(AbstractSerializedData stream, int c case 0x44c1f8e9: result = new TL_inputStickerSetEmojiDefaultTopicIcons(); break; + case TL_inputStickerSetEmojiChannelDefaultStatuses.constructor: + result = new TL_inputStickerSetEmojiChannelDefaultStatuses(); + break; } if (result == null && exception) { throw new RuntimeException(String.format("can't parse magic %x in InputStickerSet", constructor)); @@ -33075,6 +33382,15 @@ public void serializeToStream(AbstractSerializedData stream) { } } + public static class TL_inputStickerSetEmojiChannelDefaultStatuses extends InputStickerSet { + public static final int constructor = 0x49748553; + + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + } + } + public static class TL_inputStickerSetEmojiDefaultStatuses extends InputStickerSet { public static final int constructor = 0x29d0f5ee; @@ -34068,6 +34384,12 @@ public static Update TLdeserialize(AbstractSerializedData stream, int constructo case TL_updatePeerWallpaper.constructor: result = new TL_updatePeerWallpaper(); break; + case TL_updateSavedDialogPinned.constructor: + result = new TL_updateSavedDialogPinned(); + break; + case TL_updatePinnedSavedDialogs.constructor: + result = new TL_updatePinnedSavedDialogs(); + break; } if (result == null && exception) { throw new RuntimeException(String.format("can't parse magic %x in Update", constructor)); @@ -35252,7 +35574,7 @@ public static class TL_updateUserEmojiStatus extends Update { public long user_id; public EmojiStatus emoji_status; - + public void readParams(AbstractSerializedData stream, boolean exception) { user_id = stream.readInt64(exception); emoji_status = EmojiStatus.TLdeserialize(stream, stream.readInt32(exception), exception); @@ -35459,10 +35781,10 @@ public void serializeToStream(AbstractSerializedData stream) { draft.serializeToStream(stream); } } - + public static class TL_updateNewAuthorization extends Update { public static final int constructor = 0x8951abef; - + public int flags; public boolean unconfirmed; public long hash; @@ -36084,7 +36406,7 @@ public void serializeToStream(AbstractSerializedData stream) { stream.writeInt32(constructor); } } - + public static class TL_updateRecentEmojiStatuses extends Update { public static final int constructor = 0x30f443db; @@ -42648,17 +42970,21 @@ public static abstract class WallPaperSettings extends TLObject { public int fourth_background_color; public int intensity; public int rotation; + public String emoticon; public static WallPaperSettings TLdeserialize(AbstractSerializedData stream, int constructor, boolean exception) { WallPaperSettings result = null; switch (constructor) { - case 0xa12f40b8: + case TL_wallPaperSettings_layer106.constructor: result = new TL_wallPaperSettings_layer106(); break; - case 0x5086cf8: + case TL_wallPaperSettings_layer128.constructor: result = new TL_wallPaperSettings_layer128(); break; - case 0x1dc1bca4: + case TL_wallPaperSettings_layer167.constructor: + result = new TL_wallPaperSettings_layer167(); + break; + case TL_wallPaperSettings.constructor: result = new TL_wallPaperSettings(); break; } @@ -42745,6 +43071,66 @@ public void serializeToStream(AbstractSerializedData stream) { } public static class TL_wallPaperSettings extends WallPaperSettings { + public static final int constructor = 0x372efcd0; + + + public void readParams(AbstractSerializedData stream, boolean exception) { + flags = stream.readInt32(exception); + blur = (flags & 2) != 0; + motion = (flags & 4) != 0; + if ((flags & 1) != 0) { + background_color = stream.readInt32(exception); + } + if ((flags & 16) != 0) { + second_background_color = stream.readInt32(exception); + } + if ((flags & 32) != 0) { + third_background_color = stream.readInt32(exception); + } + if ((flags & 64) != 0) { + fourth_background_color = stream.readInt32(exception); + } + if ((flags & 8) != 0) { + intensity = stream.readInt32(exception); + } + if ((flags & 16) != 0) { + rotation = stream.readInt32(exception); + } + if ((flags & 128) != 0) { + emoticon = stream.readString(exception); + } + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + flags = blur ? (flags | 2) : (flags &~ 2); + flags = motion ? (flags | 4) : (flags &~ 4); + stream.writeInt32(flags); + if ((flags & 1) != 0) { + stream.writeInt32(background_color); + } + if ((flags & 16) != 0) { + stream.writeInt32(second_background_color); + } + if ((flags & 32) != 0) { + stream.writeInt32(third_background_color); + } + if ((flags & 64) != 0) { + stream.writeInt32(fourth_background_color); + } + if ((flags & 8) != 0) { + stream.writeInt32(intensity); + } + if ((flags & 16) != 0) { + stream.writeInt32(rotation); + } + if ((flags & 128) != 0) { + stream.writeString(emoticon); + } + } + } + + public static class TL_wallPaperSettings_layer167 extends TL_wallPaperSettings { public static final int constructor = 0x1dc1bca4; @@ -44476,6 +44862,18 @@ public static ChannelAdminLogEventAction TLdeserialize(AbstractSerializedData st case 0x445fc434: result = new TL_channelAdminLogEventActionChangeBackgroundEmoji(); break; + case TL_channelAdminLogEventActionChangePeerColor.constructor: + result = new TL_channelAdminLogEventActionChangePeerColor(); + break; + case TL_channelAdminLogEventActionChangeProfilePeerColor.constructor: + result = new TL_channelAdminLogEventActionChangeProfilePeerColor(); + break; + case TL_channelAdminLogEventActionChangeWallpaper.constructor: + result = new TL_channelAdminLogEventActionChangeWallpaper(); + break; + case TL_channelAdminLogEventActionChangeEmojiStatus.constructor: + result = new TL_channelAdminLogEventActionChangeEmojiStatus(); + break; } if (result == null && exception) { throw new RuntimeException(String.format("can't parse magic %x in ChannelAdminLogEventAction", constructor)); @@ -45087,6 +45485,24 @@ public void serializeToStream(AbstractSerializedData stream) { } } + public static class TL_channelAdminLogEventActionChangeEmojiStatus extends ChannelAdminLogEventAction { + public static final int constructor = 0x3ea9feb1; + + public EmojiStatus prev_value; + public EmojiStatus new_value; + + public void readParams(AbstractSerializedData stream, boolean exception) { + prev_value = EmojiStatus.TLdeserialize(stream, stream.readInt32(exception), exception); + new_value = EmojiStatus.TLdeserialize(stream, stream.readInt32(exception), exception); + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + prev_value.serializeToStream(stream); + new_value.serializeToStream(stream); + } + } + public static class TL_channelAdminLogEventActionPinTopic extends ChannelAdminLogEventAction { public static final int constructor = 0x5d8d353b; @@ -45228,10 +45644,10 @@ public void serializeToStream(AbstractSerializedData stream) { public static class TL_channelAdminLogEventActionChangeColor extends ChannelAdminLogEventAction { public static final int constructor = 0x3c2b247b; - + public int prev_value; public int new_value; - + public void readParams(AbstractSerializedData stream, boolean exception) { prev_value = stream.readInt32(exception); new_value = stream.readInt32(exception); @@ -45262,6 +45678,60 @@ public void serializeToStream(AbstractSerializedData stream) { } } + public static class TL_channelAdminLogEventActionChangePeerColor extends ChannelAdminLogEventAction { + public static final int constructor = 0x5796e780; + + public TL_peerColor prev_value; + public TL_peerColor new_value; + + public void readParams(AbstractSerializedData stream, boolean exception) { + prev_value = TL_peerColor.TLdeserialize(stream, stream.readInt32(exception), exception); + new_value = TL_peerColor.TLdeserialize(stream, stream.readInt32(exception), exception); + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + prev_value.serializeToStream(stream); + new_value.serializeToStream(stream); + } + } + + public static class TL_channelAdminLogEventActionChangeProfilePeerColor extends ChannelAdminLogEventAction { + public static final int constructor = 0x5e477b25; + + public TL_peerColor prev_value; + public TL_peerColor new_value; + + public void readParams(AbstractSerializedData stream, boolean exception) { + prev_value = TL_peerColor.TLdeserialize(stream, stream.readInt32(exception), exception); + new_value = TL_peerColor.TLdeserialize(stream, stream.readInt32(exception), exception); + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + prev_value.serializeToStream(stream); + new_value.serializeToStream(stream); + } + } + + public static class TL_channelAdminLogEventActionChangeWallpaper extends ChannelAdminLogEventAction { + public static final int constructor = 0x31bb5d52; + + public WallPaper prev_value; + public WallPaper new_value; + + public void readParams(AbstractSerializedData stream, boolean exception) { + prev_value = WallPaper.TLdeserialize(stream, stream.readInt32(exception), exception); + new_value = WallPaper.TLdeserialize(stream, stream.readInt32(exception), exception); + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + prev_value.serializeToStream(stream); + new_value.serializeToStream(stream); + } + } + public static class TL_channelAdminLogEventActionToggleAntiSpam extends ChannelAdminLogEventAction { public static final int constructor = 0x64f36dfc; @@ -45715,6 +46185,9 @@ public static abstract class Chat extends TLObject { public boolean stories_unavailable; public int stories_max_id; public TL_peerColor color; + public TL_peerColor profile_color; + public EmojiStatus emoji_status; + public int level; public ArrayList usernames = new ArrayList<>(); @@ -45740,6 +46213,15 @@ public static Chat TLdeserialize(AbstractSerializedData stream, int constructor, case TL_channel.constructor: result = new TL_channel(); break; + case TL_channel_layer167_3.constructor: + result = new TL_channel_layer167_3(); + break; + case TL_channel_layer167_2.constructor: + result = new TL_channel_layer167_2(); + break; + case TL_channel_layer167.constructor: + result = new TL_channel_layer167(); + break; case TL_channel_layer166.constructor: result = new TL_channel_layer166(); break; @@ -46183,8 +46665,8 @@ public void serializeToStream(AbstractSerializedData stream) { } } - public static class TL_channel extends Chat { - public static final int constructor = 0x8e87ccd8; + public static class TL_channel_layer167_2 extends TL_channel { + public static final int constructor = 0xa636a3e2; public void readParams(AbstractSerializedData stream, boolean exception) { flags = stream.readInt32(exception); @@ -46274,6 +46756,9 @@ public void readParams(AbstractSerializedData stream, boolean exception) { if ((flags2 & 128) != 0) { color = TL_peerColor.TLdeserialize(stream, stream.readInt32(exception), exception); } + if ((flags2 & 256) != 0) { + profile_color = TL_peerColor.TLdeserialize(stream, stream.readInt32(exception), exception); + } } public void serializeToStream(AbstractSerializedData stream) { @@ -46347,11 +46832,14 @@ public void serializeToStream(AbstractSerializedData stream) { if ((flags2 & 128) != 0) { color.serializeToStream(stream); } + if ((flags2 & 256) != 0) { + profile_color.serializeToStream(stream); + } } } - public static class TL_channel_layer166 extends TL_channel { - public static final int constructor = 0x1981ea7e; + public static class TL_channel_layer167_3 extends TL_channel { + public static final int constructor = 0xdb12acb; public void readParams(AbstractSerializedData stream, boolean exception) { flags = stream.readInt32(exception); @@ -46438,15 +46926,14 @@ public void readParams(AbstractSerializedData stream, boolean exception) { if ((flags2 & 16) != 0) { stories_max_id = stream.readInt32(exception); } - if ((flags2 & 64) != 0) { - color = new TL_peerColor(); - color.color = stream.readInt32(exception); + if ((flags2 & 128) != 0) { + color = TL_peerColor.TLdeserialize(stream, stream.readInt32(exception), exception); } - if ((flags2 & 32) != 0) { - if (color == null) { - color = new TL_peerColor(); - } - color.background_emoji_id = stream.readInt64(exception); + if ((flags2 & 256) != 0) { + profile_color = TL_peerColor.TLdeserialize(stream, stream.readInt32(exception), exception); + } + if ((flags2 & 512) != 0) { + emoji_status = EmojiStatus.TLdeserialize(stream, stream.readInt32(exception), exception); } } @@ -46518,17 +47005,20 @@ public void serializeToStream(AbstractSerializedData stream) { if ((flags2 & 16) != 0) { stream.writeInt32(stories_max_id); } - if ((flags2 & 64) != 0) { - stream.writeInt32(color.color); + if ((flags2 & 128) != 0) { + color.serializeToStream(stream); } - if ((flags2 & 32) != 0) { - stream.writeInt64(color.background_emoji_id); + if ((flags2 & 256) != 0) { + profile_color.serializeToStream(stream); + } + if ((flags2 & 512) != 0) { + emoji_status.serializeToStream(stream); } } } - public static class TL_channel_layer165 extends TL_channel { - public static final int constructor = 0x94f592db; + public static class TL_channel extends Chat { + public static final int constructor = 0xaadfc8f; public void readParams(AbstractSerializedData stream, boolean exception) { flags = stream.readInt32(exception); @@ -46615,10 +47105,17 @@ public void readParams(AbstractSerializedData stream, boolean exception) { if ((flags2 & 16) != 0) { stories_max_id = stream.readInt32(exception); } - color = new TL_peerColor(); - color.color = stream.readInt32(exception); - if ((flags2 & 32) != 0) { - color.background_emoji_id = stream.readInt64(exception); + if ((flags2 & 128) != 0) { + color = TL_peerColor.TLdeserialize(stream, stream.readInt32(exception), exception); + } + if ((flags2 & 256) != 0) { + profile_color = TL_peerColor.TLdeserialize(stream, stream.readInt32(exception), exception); + } + if ((flags2 & 512) != 0) { + emoji_status = EmojiStatus.TLdeserialize(stream, stream.readInt32(exception), exception); + } + if ((flags2 & 1024) != 0) { + level = stream.readInt32(exception); } } @@ -46690,21 +47187,192 @@ public void serializeToStream(AbstractSerializedData stream) { if ((flags2 & 16) != 0) { stream.writeInt32(stories_max_id); } - stream.writeInt32(color == null ? (int) (id % 7) : color.color); - if ((flags2 & 32) != 0) { - stream.writeInt64(color == null ? 0 : color.background_emoji_id); + if ((flags2 & 128) != 0) { + color.serializeToStream(stream); + } + if ((flags2 & 256) != 0) { + profile_color.serializeToStream(stream); + } + if ((flags2 & 512) != 0) { + emoji_status.serializeToStream(stream); + } + if ((flags2 & 1024) != 0) { + stream.writeInt32(level); } } } - public static class TL_channel_layer161 extends TL_channel { - public static final int constructor = 0x83259464; + public static class TL_channel_layer167 extends TL_channel { + public static final int constructor = 0x8e87ccd8; public void readParams(AbstractSerializedData stream, boolean exception) { - readParams(stream, exception, true); + flags = stream.readInt32(exception); + creator = (flags & 1) != 0; + left = (flags & 4) != 0; + broadcast = (flags & 32) != 0; + verified = (flags & 128) != 0; + megagroup = (flags & 256) != 0; + restricted = (flags & 512) != 0; + signatures = (flags & 2048) != 0; + min = (flags & 4096) != 0; + scam = (flags & 524288) != 0; + has_link = (flags & 1048576) != 0; + has_geo = (flags & 2097152) != 0; + slowmode_enabled = (flags & 4194304) != 0; + call_active = (flags & 8388608) != 0; + call_not_empty = (flags & 16777216) != 0; + fake = (flags & 33554432) != 0; + gigagroup = (flags & 67108864) != 0; + noforwards = (flags & 134217728) != 0; + join_to_send = (flags & 268435456) != 0; + join_request = (flags & 536870912) != 0; + forum = (flags & 1073741824) != 0; + flags2 = stream.readInt32(exception); + stories_hidden = (flags2 & 2) != 0; + stories_hidden_min = (flags2 & 4) != 0; + stories_unavailable = (flags2 & 8) != 0; + id = stream.readInt64(exception); + if ((flags & 8192) != 0) { + access_hash = stream.readInt64(exception); + } + title = stream.readString(exception); + if ((flags & 64) != 0) { + username = stream.readString(exception); + } + photo = ChatPhoto.TLdeserialize(stream, stream.readInt32(exception), exception); + date = stream.readInt32(exception); + if ((flags & 512) != 0) { + int magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + int count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + TL_restrictionReason object = TL_restrictionReason.TLdeserialize(stream, stream.readInt32(exception), exception); + if (object == null) { + return; + } + restriction_reason.add(object); + } + } + if ((flags & 16384) != 0) { + admin_rights = TL_chatAdminRights.TLdeserialize(stream, stream.readInt32(exception), exception); + } + if ((flags & 32768) != 0) { + banned_rights = TL_chatBannedRights.TLdeserialize(stream, stream.readInt32(exception), exception); + } + if ((flags & 262144) != 0) { + default_banned_rights = TL_chatBannedRights.TLdeserialize(stream, stream.readInt32(exception), exception); + } + if ((flags & 131072) != 0) { + participants_count = stream.readInt32(exception); + } + if ((flags2 & 1) != 0) { + int magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + int count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + TL_username object = TL_username.TLdeserialize(stream, stream.readInt32(exception), exception); + if (object == null) { + return; + } + usernames.add(object); + } + } + if ((flags2 & 16) != 0) { + stories_max_id = stream.readInt32(exception); + } + if ((flags2 & 128) != 0) { + color = TL_peerColor.TLdeserialize(stream, stream.readInt32(exception), exception); + } } - public void readParams(AbstractSerializedData stream, boolean exception, boolean allowStrippedThumb) { + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + flags = creator ? (flags | 1) : (flags &~ 1); + flags = left ? (flags | 4) : (flags &~ 4); + flags = broadcast ? (flags | 32) : (flags &~ 32); + flags = verified ? (flags | 128) : (flags &~ 128); + flags = megagroup ? (flags | 256) : (flags &~ 256); + flags = restricted ? (flags | 512) : (flags &~ 512); + flags = signatures ? (flags | 2048) : (flags &~ 2048); + flags = min ? (flags | 4096) : (flags &~ 4096); + flags = scam ? (flags | 524288) : (flags &~ 524288); + flags = has_link ? (flags | 1048576) : (flags &~ 1048576); + flags = has_geo ? (flags | 2097152) : (flags &~ 2097152); + flags = slowmode_enabled ? (flags | 4194304) : (flags &~ 4194304); + flags = call_active ? (flags | 8388608) : (flags &~ 8388608); + flags = call_not_empty ? (flags | 16777216) : (flags &~ 16777216); + flags = fake ? (flags | 33554432) : (flags &~ 33554432); + flags = gigagroup ? (flags | 67108864) : (flags &~ 67108864); + flags = noforwards ? (flags | 134217728) : (flags &~ 134217728); + flags = join_to_send ? (flags | 268435456) : (flags &~ 268435456); + flags = join_request ? (flags | 536870912) : (flags &~ 536870912); + flags = forum ? (flags | 1073741824) : (flags &~ 1073741824); + stream.writeInt32(flags); + flags2 = stories_hidden ? (flags2 | 2) : (flags2 &~ 2); + flags2 = stories_hidden_min ? (flags2 | 4) : (flags2 &~ 4); + flags2 = stories_unavailable ? (flags2 | 8) : (flags2 &~ 8); + stream.writeInt32(flags2); + stream.writeInt64(id); + if ((flags & 8192) != 0) { + stream.writeInt64(access_hash); + } + stream.writeString(title); + if ((flags & 64) != 0) { + stream.writeString(username); + } + photo.serializeToStream(stream); + stream.writeInt32(date); + if ((flags & 512) != 0) { + stream.writeInt32(0x1cb5c415); + int count = restriction_reason.size(); + stream.writeInt32(count); + for (int a = 0; a < count; a++) { + restriction_reason.get(a).serializeToStream(stream); + } + } + if ((flags & 16384) != 0) { + admin_rights.serializeToStream(stream); + } + if ((flags & 32768) != 0) { + banned_rights.serializeToStream(stream); + } + if ((flags & 262144) != 0) { + default_banned_rights.serializeToStream(stream); + } + if ((flags & 131072) != 0) { + stream.writeInt32(participants_count); + } + if ((flags2 & 1) != 0) { + stream.writeInt32(0x1cb5c415); + int count = usernames.size(); + stream.writeInt32(count); + for (int a = 0; a < count; a++) { + usernames.get(a).serializeToStream(stream); + } + } + if ((flags2 & 16) != 0) { + stream.writeInt32(stories_max_id); + } + if ((flags2 & 128) != 0) { + color.serializeToStream(stream); + } + } + } + + public static class TL_channel_layer166 extends TL_channel { + public static final int constructor = 0x1981ea7e; + + public void readParams(AbstractSerializedData stream, boolean exception) { flags = stream.readInt32(exception); creator = (flags & 1) != 0; left = (flags & 4) != 0; @@ -46729,6 +47397,7 @@ public void readParams(AbstractSerializedData stream, boolean exception, boolean flags2 = stream.readInt32(exception); stories_hidden = (flags2 & 2) != 0; stories_hidden_min = (flags2 & 4) != 0; + stories_unavailable = (flags2 & 8) != 0; id = stream.readInt64(exception); if ((flags & 8192) != 0) { access_hash = stream.readInt64(exception); @@ -46737,7 +47406,7 @@ public void readParams(AbstractSerializedData stream, boolean exception, boolean if ((flags & 64) != 0) { username = stream.readString(exception); } - photo = ChatPhoto.TLdeserialize(stream, stream.readInt32(exception), exception, allowStrippedThumb); + photo = ChatPhoto.TLdeserialize(stream, stream.readInt32(exception), exception); date = stream.readInt32(exception); if ((flags & 512) != 0) { int magic = stream.readInt32(exception); @@ -46785,6 +47454,19 @@ public void readParams(AbstractSerializedData stream, boolean exception, boolean usernames.add(object); } } + if ((flags2 & 16) != 0) { + stories_max_id = stream.readInt32(exception); + } + if ((flags2 & 64) != 0) { + color = new TL_peerColor(); + color.color = stream.readInt32(exception); + } + if ((flags2 & 32) != 0) { + if (color == null) { + color = new TL_peerColor(); + } + color.background_emoji_id = stream.readInt64(exception); + } } public void serializeToStream(AbstractSerializedData stream) { @@ -46812,6 +47494,7 @@ public void serializeToStream(AbstractSerializedData stream) { stream.writeInt32(flags); flags2 = stories_hidden ? (flags2 | 2) : (flags2 &~ 2); flags2 = stories_hidden_min ? (flags2 | 4) : (flags2 &~ 4); + flags2 = stories_unavailable ? (flags2 | 8) : (flags2 &~ 8); stream.writeInt32(flags2); stream.writeInt64(id); if ((flags & 8192) != 0) { @@ -46851,12 +47534,348 @@ public void serializeToStream(AbstractSerializedData stream) { usernames.get(a).serializeToStream(stream); } } + if ((flags2 & 16) != 0) { + stream.writeInt32(stories_max_id); + } + if ((flags2 & 64) != 0) { + stream.writeInt32(color == null ? (int) (id % 7) : color.color); + } + if ((flags2 & 32) != 0) { + stream.writeInt64(color == null ? 0 : color.background_emoji_id); + } } } - public static class TL_channel_layer147 extends TL_channel { - public static final int constructor = 0x8261ac61; - + public static class TL_channel_layer165 extends TL_channel { + public static final int constructor = 0x94f592db; + + public void readParams(AbstractSerializedData stream, boolean exception) { + flags = stream.readInt32(exception); + creator = (flags & 1) != 0; + left = (flags & 4) != 0; + broadcast = (flags & 32) != 0; + verified = (flags & 128) != 0; + megagroup = (flags & 256) != 0; + restricted = (flags & 512) != 0; + signatures = (flags & 2048) != 0; + min = (flags & 4096) != 0; + scam = (flags & 524288) != 0; + has_link = (flags & 1048576) != 0; + has_geo = (flags & 2097152) != 0; + slowmode_enabled = (flags & 4194304) != 0; + call_active = (flags & 8388608) != 0; + call_not_empty = (flags & 16777216) != 0; + fake = (flags & 33554432) != 0; + gigagroup = (flags & 67108864) != 0; + noforwards = (flags & 134217728) != 0; + join_to_send = (flags & 268435456) != 0; + join_request = (flags & 536870912) != 0; + forum = (flags & 1073741824) != 0; + flags2 = stream.readInt32(exception); + stories_hidden = (flags2 & 2) != 0; + stories_hidden_min = (flags2 & 4) != 0; + stories_unavailable = (flags2 & 8) != 0; + id = stream.readInt64(exception); + if ((flags & 8192) != 0) { + access_hash = stream.readInt64(exception); + } + title = stream.readString(exception); + if ((flags & 64) != 0) { + username = stream.readString(exception); + } + photo = ChatPhoto.TLdeserialize(stream, stream.readInt32(exception), exception); + date = stream.readInt32(exception); + if ((flags & 512) != 0) { + int magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + int count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + TL_restrictionReason object = TL_restrictionReason.TLdeserialize(stream, stream.readInt32(exception), exception); + if (object == null) { + return; + } + restriction_reason.add(object); + } + } + if ((flags & 16384) != 0) { + admin_rights = TL_chatAdminRights.TLdeserialize(stream, stream.readInt32(exception), exception); + } + if ((flags & 32768) != 0) { + banned_rights = TL_chatBannedRights.TLdeserialize(stream, stream.readInt32(exception), exception); + } + if ((flags & 262144) != 0) { + default_banned_rights = TL_chatBannedRights.TLdeserialize(stream, stream.readInt32(exception), exception); + } + if ((flags & 131072) != 0) { + participants_count = stream.readInt32(exception); + } + if ((flags2 & 1) != 0) { + int magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + int count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + TL_username object = TL_username.TLdeserialize(stream, stream.readInt32(exception), exception); + if (object == null) { + return; + } + usernames.add(object); + } + } + if ((flags2 & 16) != 0) { + stories_max_id = stream.readInt32(exception); + } +// color = new TL_peerColor(); +// color.color = stream.readInt32(exception); +// if ((flags2 & 32) != 0) { +// color.background_emoji_id = stream.readInt64(exception); +// } + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + flags = creator ? (flags | 1) : (flags &~ 1); + flags = left ? (flags | 4) : (flags &~ 4); + flags = broadcast ? (flags | 32) : (flags &~ 32); + flags = verified ? (flags | 128) : (flags &~ 128); + flags = megagroup ? (flags | 256) : (flags &~ 256); + flags = restricted ? (flags | 512) : (flags &~ 512); + flags = signatures ? (flags | 2048) : (flags &~ 2048); + flags = min ? (flags | 4096) : (flags &~ 4096); + flags = scam ? (flags | 524288) : (flags &~ 524288); + flags = has_link ? (flags | 1048576) : (flags &~ 1048576); + flags = has_geo ? (flags | 2097152) : (flags &~ 2097152); + flags = slowmode_enabled ? (flags | 4194304) : (flags &~ 4194304); + flags = call_active ? (flags | 8388608) : (flags &~ 8388608); + flags = call_not_empty ? (flags | 16777216) : (flags &~ 16777216); + flags = fake ? (flags | 33554432) : (flags &~ 33554432); + flags = gigagroup ? (flags | 67108864) : (flags &~ 67108864); + flags = noforwards ? (flags | 134217728) : (flags &~ 134217728); + flags = join_to_send ? (flags | 268435456) : (flags &~ 268435456); + flags = join_request ? (flags | 536870912) : (flags &~ 536870912); + flags = forum ? (flags | 1073741824) : (flags &~ 1073741824); + stream.writeInt32(flags); + flags2 = stories_hidden ? (flags2 | 2) : (flags2 &~ 2); + flags2 = stories_hidden_min ? (flags2 | 4) : (flags2 &~ 4); + flags2 = stories_unavailable ? (flags2 | 8) : (flags2 &~ 8); + stream.writeInt32(flags2); + stream.writeInt64(id); + if ((flags & 8192) != 0) { + stream.writeInt64(access_hash); + } + stream.writeString(title); + if ((flags & 64) != 0) { + stream.writeString(username); + } + photo.serializeToStream(stream); + stream.writeInt32(date); + if ((flags & 512) != 0) { + stream.writeInt32(0x1cb5c415); + int count = restriction_reason.size(); + stream.writeInt32(count); + for (int a = 0; a < count; a++) { + restriction_reason.get(a).serializeToStream(stream); + } + } + if ((flags & 16384) != 0) { + admin_rights.serializeToStream(stream); + } + if ((flags & 32768) != 0) { + banned_rights.serializeToStream(stream); + } + if ((flags & 262144) != 0) { + default_banned_rights.serializeToStream(stream); + } + if ((flags & 131072) != 0) { + stream.writeInt32(participants_count); + } + if ((flags2 & 1) != 0) { + stream.writeInt32(0x1cb5c415); + int count = usernames.size(); + stream.writeInt32(count); + for (int a = 0; a < count; a++) { + usernames.get(a).serializeToStream(stream); + } + } + if ((flags2 & 16) != 0) { + stream.writeInt32(stories_max_id); + } +// stream.writeInt32(color == null ? (int) (id % 7) : color.color); +// if ((flags2 & 32) != 0) { +// stream.writeInt64(color == null ? 0 : color.background_emoji_id); +// } + } + } + + public static class TL_channel_layer161 extends TL_channel { + public static final int constructor = 0x83259464; + + public void readParams(AbstractSerializedData stream, boolean exception) { + readParams(stream, exception, true); + } + + public void readParams(AbstractSerializedData stream, boolean exception, boolean allowStrippedThumb) { + flags = stream.readInt32(exception); + creator = (flags & 1) != 0; + left = (flags & 4) != 0; + broadcast = (flags & 32) != 0; + verified = (flags & 128) != 0; + megagroup = (flags & 256) != 0; + restricted = (flags & 512) != 0; + signatures = (flags & 2048) != 0; + min = (flags & 4096) != 0; + scam = (flags & 524288) != 0; + has_link = (flags & 1048576) != 0; + has_geo = (flags & 2097152) != 0; + slowmode_enabled = (flags & 4194304) != 0; + call_active = (flags & 8388608) != 0; + call_not_empty = (flags & 16777216) != 0; + fake = (flags & 33554432) != 0; + gigagroup = (flags & 67108864) != 0; + noforwards = (flags & 134217728) != 0; + join_to_send = (flags & 268435456) != 0; + join_request = (flags & 536870912) != 0; + forum = (flags & 1073741824) != 0; + flags2 = stream.readInt32(exception); + stories_hidden = (flags2 & 2) != 0; + stories_hidden_min = (flags2 & 4) != 0; + id = stream.readInt64(exception); + if ((flags & 8192) != 0) { + access_hash = stream.readInt64(exception); + } + title = stream.readString(exception); + if ((flags & 64) != 0) { + username = stream.readString(exception); + } + photo = ChatPhoto.TLdeserialize(stream, stream.readInt32(exception), exception, allowStrippedThumb); + date = stream.readInt32(exception); + if ((flags & 512) != 0) { + int magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + int count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + TL_restrictionReason object = TL_restrictionReason.TLdeserialize(stream, stream.readInt32(exception), exception); + if (object == null) { + return; + } + restriction_reason.add(object); + } + } + if ((flags & 16384) != 0) { + admin_rights = TL_chatAdminRights.TLdeserialize(stream, stream.readInt32(exception), exception); + } + if ((flags & 32768) != 0) { + banned_rights = TL_chatBannedRights.TLdeserialize(stream, stream.readInt32(exception), exception); + } + if ((flags & 262144) != 0) { + default_banned_rights = TL_chatBannedRights.TLdeserialize(stream, stream.readInt32(exception), exception); + } + if ((flags & 131072) != 0) { + participants_count = stream.readInt32(exception); + } + if ((flags2 & 1) != 0) { + int magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + int count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + TL_username object = TL_username.TLdeserialize(stream, stream.readInt32(exception), exception); + if (object == null) { + return; + } + usernames.add(object); + } + } + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + flags = creator ? (flags | 1) : (flags &~ 1); + flags = left ? (flags | 4) : (flags &~ 4); + flags = broadcast ? (flags | 32) : (flags &~ 32); + flags = verified ? (flags | 128) : (flags &~ 128); + flags = megagroup ? (flags | 256) : (flags &~ 256); + flags = restricted ? (flags | 512) : (flags &~ 512); + flags = signatures ? (flags | 2048) : (flags &~ 2048); + flags = min ? (flags | 4096) : (flags &~ 4096); + flags = scam ? (flags | 524288) : (flags &~ 524288); + flags = has_link ? (flags | 1048576) : (flags &~ 1048576); + flags = has_geo ? (flags | 2097152) : (flags &~ 2097152); + flags = slowmode_enabled ? (flags | 4194304) : (flags &~ 4194304); + flags = call_active ? (flags | 8388608) : (flags &~ 8388608); + flags = call_not_empty ? (flags | 16777216) : (flags &~ 16777216); + flags = fake ? (flags | 33554432) : (flags &~ 33554432); + flags = gigagroup ? (flags | 67108864) : (flags &~ 67108864); + flags = noforwards ? (flags | 134217728) : (flags &~ 134217728); + flags = join_to_send ? (flags | 268435456) : (flags &~ 268435456); + flags = join_request ? (flags | 536870912) : (flags &~ 536870912); + flags = forum ? (flags | 1073741824) : (flags &~ 1073741824); + stream.writeInt32(flags); + flags2 = stories_hidden ? (flags2 | 2) : (flags2 &~ 2); + flags2 = stories_hidden_min ? (flags2 | 4) : (flags2 &~ 4); + stream.writeInt32(flags2); + stream.writeInt64(id); + if ((flags & 8192) != 0) { + stream.writeInt64(access_hash); + } + stream.writeString(title); + if ((flags & 64) != 0) { + stream.writeString(username); + } + photo.serializeToStream(stream); + stream.writeInt32(date); + if ((flags & 512) != 0) { + stream.writeInt32(0x1cb5c415); + int count = restriction_reason.size(); + stream.writeInt32(count); + for (int a = 0; a < count; a++) { + restriction_reason.get(a).serializeToStream(stream); + } + } + if ((flags & 16384) != 0) { + admin_rights.serializeToStream(stream); + } + if ((flags & 32768) != 0) { + banned_rights.serializeToStream(stream); + } + if ((flags & 262144) != 0) { + default_banned_rights.serializeToStream(stream); + } + if ((flags & 131072) != 0) { + stream.writeInt32(participants_count); + } + if ((flags2 & 1) != 0) { + stream.writeInt32(0x1cb5c415); + int count = usernames.size(); + stream.writeInt32(count); + for (int a = 0; a < count; a++) { + usernames.get(a).serializeToStream(stream); + } + } + } + } + + public static class TL_channel_layer147 extends TL_channel { + public static final int constructor = 0x8261ac61; + public void readParams(AbstractSerializedData stream, boolean exception) { flags = stream.readInt32(exception); @@ -47685,6 +48704,7 @@ public static abstract class StickerSet extends TLObject { public boolean videos; public boolean emojis; public boolean text_color; + public boolean channel_emoji_status; public long id; public long access_hash; public String title; @@ -47768,6 +48788,7 @@ public void readParams(AbstractSerializedData stream, boolean exception) { videos = (flags & 64) != 0; emojis = (flags & 128) != 0; text_color = (flags & 512) != 0; + channel_emoji_status = (flags & 1024) != 0; if ((flags & 1) != 0) { installed_date = stream.readInt32(exception); } @@ -47814,6 +48835,7 @@ public void serializeToStream(AbstractSerializedData stream) { flags = videos ? (flags | 64) : (flags &~ 64); flags = emojis ? (flags | 128) : (flags &~ 128); flags = text_color ? (flags | 512) : (flags &~ 512); + flags = channel_emoji_status ? (flags | 1024) : (flags &~ 1024); stream.writeInt32(flags); if ((flags & 1) != 0) { stream.writeInt32(installed_date); @@ -48912,6 +49934,7 @@ public static abstract class MessageFwdHeader extends TLObject { public int flags; public boolean imported; + public boolean saved_out; public Peer from_id; public String from_name; public int date; @@ -48919,6 +49942,9 @@ public static abstract class MessageFwdHeader extends TLObject { public String post_author; public Peer saved_from_peer; public int saved_from_msg_id; + public Peer saved_from_id; + public String saved_from_name; + public int saved_date; public String psa_type; public static MessageFwdHeader TLdeserialize(AbstractSerializedData stream, int constructor, boolean exception) { @@ -48936,9 +49962,12 @@ public static MessageFwdHeader TLdeserialize(AbstractSerializedData stream, int case 0xfadff4ac: result = new TL_messageFwdHeader_layer72(); break; - case 0x5f777dce: + case TL_messageFwdHeader.constructor: result = new TL_messageFwdHeader(); break; + case TL_messageFwdHeader_layer169.constructor: + result = new TL_messageFwdHeader_layer169(); + break; case 0x559ebe6d: result = new TL_messageFwdHeader_layer96(); break; @@ -48990,6 +50019,85 @@ public void serializeToStream(AbstractSerializedData stream) { } public static class TL_messageFwdHeader extends MessageFwdHeader { + public static final int constructor = 0x4e4df4bb; + + public void readParams(AbstractSerializedData stream, boolean exception) { + flags = stream.readInt32(exception); + imported = (flags & 128) != 0; + saved_out = (flags & 2048) != 0; + if ((flags & 1) != 0) { + from_id = Peer.TLdeserialize(stream, stream.readInt32(exception), exception); + } + if ((flags & 32) != 0) { + from_name = stream.readString(exception); + } + date = stream.readInt32(exception); + if ((flags & 4) != 0) { + channel_post = stream.readInt32(exception); + } + if ((flags & 8) != 0) { + post_author = stream.readString(exception); + } + if ((flags & 16) != 0) { + saved_from_peer = Peer.TLdeserialize(stream, stream.readInt32(exception), exception); + } + if ((flags & 16) != 0) { + saved_from_msg_id = stream.readInt32(exception); + } + if ((flags & 256) != 0) { + saved_from_id = Peer.TLdeserialize(stream, stream.readInt32(exception), exception); + } + if ((flags & 512) != 0) { + saved_from_name = stream.readString(exception); + } + if ((flags & 1024) != 0) { + saved_date = stream.readInt32(exception); + } + if ((flags & 64) != 0) { + psa_type = stream.readString(exception); + } + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + flags = imported ? (flags | 128) : (flags &~ 128); + flags = saved_out ? (flags | 2048) : (flags &~ 2048); + stream.writeInt32(flags); + if ((flags & 1) != 0) { + from_id.serializeToStream(stream); + } + if ((flags & 32) != 0) { + stream.writeString(from_name); + } + stream.writeInt32(date); + if ((flags & 4) != 0) { + stream.writeInt32(channel_post); + } + if ((flags & 8) != 0) { + stream.writeString(post_author); + } + if ((flags & 16) != 0) { + saved_from_peer.serializeToStream(stream); + } + if ((flags & 16) != 0) { + stream.writeInt32(saved_from_msg_id); + } + if ((flags & 256) != 0) { + saved_from_id.serializeToStream(stream); + } + if ((flags & 512) != 0) { + stream.writeString(saved_from_name); + } + if ((flags & 1024) != 0) { + stream.writeInt32(saved_date); + } + if ((flags & 64) != 0) { + stream.writeString(psa_type); + } + } + } + + public static class TL_messageFwdHeader_layer169 extends TL_messageFwdHeader { public static final int constructor = 0x5f777dce; @@ -50193,7 +51301,7 @@ public void serializeToStream(AbstractSerializedData stream) { } public static abstract class ReactionCount extends TLObject { - + public int flags; public int chosen_order; public boolean chosen; //custom @@ -55479,12 +56587,13 @@ public void serializeToStream(AbstractSerializedData stream) { } public static class TL_messages_search extends TLObject { - public static final int constructor = 0xa0fda762; + public static final int constructor = 0xa7b4e929; public int flags; public InputPeer peer; public String q; public InputPeer from_id; + public InputPeer saved_peer_id; public int top_msg_id; public MessagesFilter filter; public int min_date; @@ -55508,6 +56617,9 @@ public void serializeToStream(AbstractSerializedData stream) { if ((flags & 1) != 0) { from_id.serializeToStream(stream); } + if ((flags & 4) != 0) { + saved_peer_id.serializeToStream(stream); + } if ((flags & 2) != 0) { stream.writeInt32(top_msg_id); } @@ -55741,6 +56853,33 @@ public void serializeToStream(AbstractSerializedData stream) { } } + public static class TL_messages_deleteSavedHistory extends TLObject { + public static final int constructor = 0x6e98102b; + + public int flags; + public InputPeer peer; + public int max_id; + public int min_date; + public int max_date; + + public TLObject deserializeResponse(AbstractSerializedData stream, int constructor, boolean exception) { + return TL_messages_affectedHistory.TLdeserialize(stream, constructor, exception); + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + stream.writeInt32(flags); + peer.serializeToStream(stream); + stream.writeInt32(max_id); + if ((flags & 4) != 0) { + stream.writeInt32(min_date); + } + if ((flags & 8) != 0) { + stream.writeInt32(max_date); + } + } + } + public static class TL_channels_togglePreHistoryHidden extends TLObject { public static final int constructor = 0xeabbb94c; @@ -56368,9 +57507,11 @@ public void serializeToStream(AbstractSerializedData stream) { } public static class TL_messages_getSearchResultsCalendar extends TLObject { - public static final int constructor = 0x49f0bde9; + public static final int constructor = 0x6aa3f6bd; + public int flags; public InputPeer peer; + public InputPeer saved_peer_id; public MessagesFilter filter; public int offset_id; public int offset_date; @@ -56381,8 +57522,12 @@ public TLObject deserializeResponse(AbstractSerializedData stream, int construct public void serializeToStream(AbstractSerializedData stream) { stream.writeInt32(constructor); + stream.writeInt32(flags); peer.serializeToStream(stream); filter.serializeToStream(stream); + if ((flags & 4) != 0) { + saved_peer_id.serializeToStream(stream); + } stream.writeInt32(offset_id); stream.writeInt32(offset_date); } @@ -57825,6 +58970,21 @@ public void serializeToStream(AbstractSerializedData stream) { } } + public static class TL_account_getChannelDefaultEmojiStatuses extends TLObject { + public static final int constructor = 0x7727a7d5; + + public long hash; + + public TLObject deserializeResponse(AbstractSerializedData stream, int constructor, boolean exception) { + return account_EmojiStatuses.TLdeserialize(stream, constructor, exception); + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + stream.writeInt64(hash); + } + } + public static class TL_account_getDefaultEmojiStatuses extends TLObject { public static final int constructor = 0xd6753386; @@ -59342,10 +60502,11 @@ public void serializeToStream(AbstractSerializedData stream) { } public static class TL_messages_getSearchCounters extends TLObject { - public static final int constructor = 0xae7cc1; + public static final int constructor = 0x1bbcf300; public int flags; public InputPeer peer; + public InputPeer saved_peer_id; public int top_msg_id; public ArrayList filters = new ArrayList<>(); @@ -59366,6 +60527,9 @@ public void serializeToStream(AbstractSerializedData stream) { stream.writeInt32(constructor); stream.writeInt32(flags); peer.serializeToStream(stream); + if ((flags & 2) != 0) { + saved_peer_id.serializeToStream(stream); + } if ((flags & 1) != 0) { stream.writeInt32(top_msg_id); } @@ -62492,16 +63656,16 @@ public static abstract class InputStorePaymentPurpose extends TLObject { public static InputStorePaymentPurpose TLdeserialize(AbstractSerializedData stream, int constructor, boolean exception) { InputStorePaymentPurpose result = null; switch (constructor) { - case 0x616f7fe8: + case TL_inputStorePaymentGiftPremium.constructor: result = new TL_inputStorePaymentGiftPremium(); break; - case 0xa3805f3f: + case TL_inputStorePaymentPremiumGiftCode.constructor: result = new TL_inputStorePaymentPremiumGiftCode(); break; - case 0xa6751e66: + case TL_inputStorePaymentPremiumSubscription.constructor: result = new TL_inputStorePaymentPremiumSubscription(); break; - case 0x7c9375e6: + case TL_inputStorePaymentPremiumGiveaway.constructor: result = new TL_inputStorePaymentPremiumGiveaway(); break; } @@ -62803,26 +63967,22 @@ public void serializeToStream(AbstractSerializedData stream) { } public static class TL_stats_getMessagePublicForwards extends TLObject { - public static final int constructor = 0x5630281b; + public static final int constructor = 0x5f150144; public InputChannel channel; public int msg_id; - public int offset_rate; - public InputPeer offset_peer; - public int offset_id; + public String offset; public int limit; public TLObject deserializeResponse(AbstractSerializedData stream, int constructor, boolean exception) { - return messages_Messages.TLdeserialize(stream, constructor, exception); + return TL_stats_publicForwards.TLdeserialize(stream, constructor, exception); } public void serializeToStream(AbstractSerializedData stream) { stream.writeInt32(constructor); channel.serializeToStream(stream); stream.writeInt32(msg_id); - stream.writeInt32(offset_rate); - offset_peer.serializeToStream(stream); - stream.writeInt32(offset_id); + stream.writeString(offset); stream.writeInt32(limit); } } @@ -63309,6 +64469,9 @@ public static abstract class MessageMedia extends TLObject { public boolean force_small_media; public boolean manual; public boolean safe; + public boolean video; + public boolean round; + public boolean voice; public static MessageMedia TLdeserialize(AbstractSerializedData stream, int constructor, boolean exception) { MessageMedia result = null; @@ -63382,9 +64545,15 @@ public static MessageMedia TLdeserialize(AbstractSerializedData stream, int cons case 0x4bd6e798: result = new TL_messageMediaPoll(); break; - case 0x58260664: + case TL_messageMediaGiveawayResults.constructor: + result = new TL_messageMediaGiveawayResults(); + break; + case TL_messageMediaGiveaway.constructor: result = new TL_messageMediaGiveaway(); break; + case TL_messageMediaGiveaway_layer167.constructor: + result = new TL_messageMediaGiveaway_layer167(); + break; case 0xb5223b0f: result = new TL_messageMediaPhoto_layer74(); break; @@ -63496,6 +64665,14 @@ public static MessageMedia TLdeserialize(AbstractSerializedData stream, int cons } return result; } + + public Boolean documentExists; + public Document getDocument() { + if (alt_document != null && ApplicationLoader.useLessData()) { + return alt_document; + } + return document; + } } //MessageMedia end @@ -63831,6 +65008,7 @@ public static class Message extends TLObject { public int id; public Peer from_id; public Peer peer_id; + public Peer saved_peer_id; public int date; public int expire_date; public MessageAction action; @@ -63979,9 +65157,12 @@ public static Message TLdeserialize(AbstractSerializedData stream, int construct case 0x85d6cbe2: result = new TL_message_layer135(); break; - case 0x38116ee0: + case TL_message.constructor: result = new TL_message(); break; + case TL_message_layer169.constructor: + result = new TL_message_layer169(); + break; case 0x9e19a1f6: result = new TL_messageService_layer118(); break; @@ -64682,7 +65863,7 @@ public void serializeToStream(AbstractSerializedData stream) { } public static class TL_message extends Message { - public static final int constructor = 0x38116ee0; + public static final int constructor = 0x76bec211; public void readParams(AbstractSerializedData stream, boolean exception) { flags = stream.readInt32(exception); @@ -64702,6 +65883,9 @@ public void readParams(AbstractSerializedData stream, boolean exception) { from_id = Peer.TLdeserialize(stream, stream.readInt32(exception), exception); } peer_id = Peer.TLdeserialize(stream, stream.readInt32(exception), exception); + if ((flags & 268435456) != 0) { + saved_peer_id = Peer.TLdeserialize(stream, stream.readInt32(exception), exception); + } if ((flags & 4) != 0) { fwd_from = MessageFwdHeader.TLdeserialize(stream, stream.readInt32(exception), exception); } @@ -64804,6 +65988,9 @@ public void serializeToStream(AbstractSerializedData stream) { from_id.serializeToStream(stream); } peer_id.serializeToStream(stream); + if ((flags & 268435456) != 0) { + saved_peer_id.serializeToStream(stream); + } if ((flags & 4) != 0) { fwd_from.serializeToStream(stream); } @@ -64865,9 +66052,8 @@ public void serializeToStream(AbstractSerializedData stream) { } } - public static class TL_message_layer135 extends TL_message { - public static final int constructor = 0x85d6cbe2; - + public static class TL_message_layer169 extends TL_message { + public static final int constructor = 0x38116ee0; public void readParams(AbstractSerializedData stream, boolean exception) { flags = stream.readInt32(exception); @@ -64881,6 +66067,7 @@ public void readParams(AbstractSerializedData stream, boolean exception) { edit_hide = (flags & 2097152) != 0; pinned = (flags & 16777216) != 0; noforwards = (flags & 67108864) != 0; + invert_media = (flags & 134217728) != 0; id = stream.readInt32(exception); if ((flags & 256) != 0) { from_id = Peer.TLdeserialize(stream, stream.readInt32(exception), exception); @@ -64944,6 +66131,9 @@ public void readParams(AbstractSerializedData stream, boolean exception) { if ((flags & 131072) != 0) { grouped_id = stream.readInt64(exception); } + if ((flags & 1048576) != 0) { + reactions = MessageReactions.TLdeserialize(stream, stream.readInt32(exception), exception); + } if ((flags & 4194304) != 0) { int magic = stream.readInt32(exception); if (magic != 0x1cb5c415) { @@ -64978,6 +66168,7 @@ public void serializeToStream(AbstractSerializedData stream) { flags = edit_hide ? (flags | 2097152) : (flags &~ 2097152); flags = pinned ? (flags | 16777216) : (flags &~ 16777216); flags = noforwards ? (flags | 67108864) : (flags &~ 67108864); + flags = invert_media ? (flags | 134217728) : (flags &~ 134217728); stream.writeInt32(flags); stream.writeInt32(id); if ((flags & 256) != 0) { @@ -65027,6 +66218,9 @@ public void serializeToStream(AbstractSerializedData stream) { if ((flags & 131072) != 0) { stream.writeInt64(grouped_id); } + if ((flags & 1048576) != 0) { + reactions.serializeToStream(stream); + } if ((flags & 4194304) != 0) { stream.writeInt32(0x1cb5c415); int count = restriction_reason.size(); @@ -65042,8 +66236,8 @@ public void serializeToStream(AbstractSerializedData stream) { } } - public static class TL_message_layer131 extends TL_message { - public static final int constructor = 0xbce383d2; + public static class TL_message_layer135 extends TL_message { + public static final int constructor = 0x85d6cbe2; public void readParams(AbstractSerializedData stream, boolean exception) { @@ -65057,6 +66251,7 @@ public void readParams(AbstractSerializedData stream, boolean exception) { legacy = (flags & 524288) != 0; edit_hide = (flags & 2097152) != 0; pinned = (flags & 16777216) != 0; + noforwards = (flags & 67108864) != 0; id = stream.readInt32(exception); if ((flags & 256) != 0) { from_id = Peer.TLdeserialize(stream, stream.readInt32(exception), exception); @@ -65066,7 +66261,7 @@ public void readParams(AbstractSerializedData stream, boolean exception) { fwd_from = MessageFwdHeader.TLdeserialize(stream, stream.readInt32(exception), exception); } if ((flags & 2048) != 0) { - via_bot_id = stream.readInt32(exception); + via_bot_id = stream.readInt64(exception); } if ((flags & 8) != 0) { reply_to = MessageReplyHeader.TLdeserialize(stream, stream.readInt32(exception), exception); @@ -65076,7 +66271,7 @@ public void readParams(AbstractSerializedData stream, boolean exception) { if ((flags & 512) != 0) { media = MessageMedia.TLdeserialize(stream, stream.readInt32(exception), exception); if (media != null) { - ttl = media.ttl_seconds; //custom + ttl = media.ttl_seconds; } if (media != null && !TextUtils.isEmpty(media.captionLegacy)) { message = media.captionLegacy; @@ -65153,6 +66348,7 @@ public void serializeToStream(AbstractSerializedData stream) { flags = legacy ? (flags | 524288) : (flags &~ 524288); flags = edit_hide ? (flags | 2097152) : (flags &~ 2097152); flags = pinned ? (flags | 16777216) : (flags &~ 16777216); + flags = noforwards ? (flags | 67108864) : (flags &~ 67108864); stream.writeInt32(flags); stream.writeInt32(id); if ((flags & 256) != 0) { @@ -65163,7 +66359,7 @@ public void serializeToStream(AbstractSerializedData stream) { fwd_from.serializeToStream(stream); } if ((flags & 2048) != 0) { - stream.writeInt32((int) via_bot_id); + stream.writeInt64(via_bot_id); } if ((flags & 8) != 0) { reply_to.serializeToStream(stream); @@ -65213,12 +66409,12 @@ public void serializeToStream(AbstractSerializedData stream) { if ((flags & 33554432) != 0) { stream.writeInt32(ttl_period); } - writeAttachPath(stream); //custom + writeAttachPath(stream); } } - public static class TL_message_layer123 extends TL_message { - public static final int constructor = 0x58ae39c9; + public static class TL_message_layer131 extends TL_message { + public static final int constructor = 0xbce383d2; public void readParams(AbstractSerializedData stream, boolean exception) { @@ -65312,6 +66508,9 @@ public void readParams(AbstractSerializedData stream, boolean exception) { restriction_reason.add(object); } } + if ((flags & 33554432) != 0) { + ttl_period = stream.readInt32(exception); + } } public void serializeToStream(AbstractSerializedData stream) { @@ -65382,12 +66581,15 @@ public void serializeToStream(AbstractSerializedData stream) { restriction_reason.get(a).serializeToStream(stream); } } - writeAttachPath(stream); + if ((flags & 33554432) != 0) { + stream.writeInt32(ttl_period); + } + writeAttachPath(stream); //custom } } - public static class TL_message_layer118 extends TL_message { - public static final int constructor = 0xf52e6b7f; + public static class TL_message_layer123 extends TL_message { + public static final int constructor = 0x58ae39c9; public void readParams(AbstractSerializedData stream, boolean exception) { @@ -65400,10 +66602,10 @@ public void readParams(AbstractSerializedData stream, boolean exception) { from_scheduled = (flags & 262144) != 0; legacy = (flags & 524288) != 0; edit_hide = (flags & 2097152) != 0; + pinned = (flags & 16777216) != 0; id = stream.readInt32(exception); if ((flags & 256) != 0) { - from_id = new TLRPC.TL_peerUser(); - from_id.user_id = stream.readInt32(exception); + from_id = Peer.TLdeserialize(stream, stream.readInt32(exception), exception); } peer_id = Peer.TLdeserialize(stream, stream.readInt32(exception), exception); if ((flags & 4) != 0) { @@ -65413,9 +66615,7 @@ public void readParams(AbstractSerializedData stream, boolean exception) { via_bot_id = stream.readInt32(exception); } if ((flags & 8) != 0) { - reply_to = new TLRPC.TL_messageReplyHeader(); - reply_to.flags |= 16; - reply_to.reply_to_msg_id = stream.readInt32(exception); + reply_to = MessageReplyHeader.TLdeserialize(stream, stream.readInt32(exception), exception); } date = stream.readInt32(exception); message = stream.readString(exception); @@ -65454,6 +66654,9 @@ public void readParams(AbstractSerializedData stream, boolean exception) { if ((flags & 1024) != 0) { forwards = stream.readInt32(exception); } + if ((flags & 8388608) != 0) { + replies = MessageReplies.TLdeserialize(stream, stream.readInt32(exception), exception); + } if ((flags & 32768) != 0) { edit_date = stream.readInt32(exception); } @@ -65492,10 +66695,11 @@ public void serializeToStream(AbstractSerializedData stream) { flags = from_scheduled ? (flags | 262144) : (flags &~ 262144); flags = legacy ? (flags | 524288) : (flags &~ 524288); flags = edit_hide ? (flags | 2097152) : (flags &~ 2097152); + flags = pinned ? (flags | 16777216) : (flags &~ 16777216); stream.writeInt32(flags); stream.writeInt32(id); if ((flags & 256) != 0) { - stream.writeInt32((int) from_id.user_id); + from_id.serializeToStream(stream); } peer_id.serializeToStream(stream); if ((flags & 4) != 0) { @@ -65505,7 +66709,7 @@ public void serializeToStream(AbstractSerializedData stream) { stream.writeInt32((int) via_bot_id); } if ((flags & 8) != 0) { - stream.writeInt32(reply_to.reply_to_msg_id); + reply_to.serializeToStream(stream); } stream.writeInt32(date); stream.writeString(message); @@ -65529,6 +66733,9 @@ public void serializeToStream(AbstractSerializedData stream) { if ((flags & 1024) != 0) { stream.writeInt32(forwards); } + if ((flags & 8388608) != 0) { + replies.serializeToStream(stream); + } if ((flags & 32768) != 0) { stream.writeInt32(edit_date); } @@ -65550,8 +66757,8 @@ public void serializeToStream(AbstractSerializedData stream) { } } - public static class TL_message_layer117 extends TL_message { - public static final int constructor = 0x452c0e65; + public static class TL_message_layer118 extends TL_message { + public static final int constructor = 0xf52e6b7f; public void readParams(AbstractSerializedData stream, boolean exception) { @@ -65615,6 +66822,9 @@ public void readParams(AbstractSerializedData stream, boolean exception) { if ((flags & 1024) != 0) { views = stream.readInt32(exception); } + if ((flags & 1024) != 0) { + forwards = stream.readInt32(exception); + } if ((flags & 32768) != 0) { edit_date = stream.readInt32(exception); } @@ -65687,6 +66897,9 @@ public void serializeToStream(AbstractSerializedData stream) { if ((flags & 1024) != 0) { stream.writeInt32(views); } + if ((flags & 1024) != 0) { + stream.writeInt32(forwards); + } if ((flags & 32768) != 0) { stream.writeInt32(edit_date); } @@ -65708,8 +66921,8 @@ public void serializeToStream(AbstractSerializedData stream) { } } - public static class TL_message_layer104_3 extends TL_message { - public static final int constructor = 0x9789dac4; + public static class TL_message_layer117 extends TL_message { + public static final int constructor = 0x452c0e65; public void readParams(AbstractSerializedData stream, boolean exception) { @@ -65782,9 +66995,6 @@ public void readParams(AbstractSerializedData stream, boolean exception) { if ((flags & 131072) != 0) { grouped_id = stream.readInt64(exception); } - if ((flags & 1048576) != 0) { - reactions = MessageReactions.TLdeserialize(stream, stream.readInt32(exception), exception); - } if ((flags & 4194304) != 0) { int magic = stream.readInt32(exception); if (magic != 0x1cb5c415) { @@ -65857,9 +67067,6 @@ public void serializeToStream(AbstractSerializedData stream) { if ((flags & 131072) != 0) { stream.writeInt64(grouped_id); } - if ((flags & 1048576) != 0) { - reactions.serializeToStream(stream); - } if ((flags & 4194304) != 0) { stream.writeInt32(0x1cb5c415); int count = restriction_reason.size(); @@ -65872,8 +67079,8 @@ public void serializeToStream(AbstractSerializedData stream) { } } - public static class TL_message_layer104_2 extends TL_message { - public static final int constructor = 0x1c9b1027; + public static class TL_message_layer104_3 extends TL_message { + public static final int constructor = 0x9789dac4; public void readParams(AbstractSerializedData stream, boolean exception) { @@ -65950,7 +67157,21 @@ public void readParams(AbstractSerializedData stream, boolean exception) { reactions = MessageReactions.TLdeserialize(stream, stream.readInt32(exception), exception); } if ((flags & 4194304) != 0) { - stream.readString(exception); + int magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + int count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + TL_restrictionReason object = TL_restrictionReason.TLdeserialize(stream, stream.readInt32(exception), exception); + if (object == null) { + return; + } + restriction_reason.add(object); + } } } @@ -66011,14 +67232,164 @@ public void serializeToStream(AbstractSerializedData stream) { reactions.serializeToStream(stream); } if ((flags & 4194304) != 0) { - stream.writeString(""); + stream.writeInt32(0x1cb5c415); + int count = restriction_reason.size(); + stream.writeInt32(count); + for (int a = 0; a < count; a++) { + restriction_reason.get(a).serializeToStream(stream); + } } writeAttachPath(stream); } } - public static class TL_message_layer104 extends TL_message { - public static final int constructor = 0x44f9b43d; + public static class TL_message_layer104_2 extends TL_message { + public static final int constructor = 0x1c9b1027; + + + public void readParams(AbstractSerializedData stream, boolean exception) { + flags = stream.readInt32(exception); + out = (flags & 2) != 0; + mentioned = (flags & 16) != 0; + media_unread = (flags & 32) != 0; + silent = (flags & 8192) != 0; + post = (flags & 16384) != 0; + from_scheduled = (flags & 262144) != 0; + legacy = (flags & 524288) != 0; + edit_hide = (flags & 2097152) != 0; + id = stream.readInt32(exception); + if ((flags & 256) != 0) { + from_id = new TLRPC.TL_peerUser(); + from_id.user_id = stream.readInt32(exception); + } + peer_id = Peer.TLdeserialize(stream, stream.readInt32(exception), exception); + if ((flags & 4) != 0) { + fwd_from = MessageFwdHeader.TLdeserialize(stream, stream.readInt32(exception), exception); + } + if ((flags & 2048) != 0) { + via_bot_id = stream.readInt32(exception); + } + if ((flags & 8) != 0) { + reply_to = new TLRPC.TL_messageReplyHeader(); + reply_to.flags |= 16; + reply_to.reply_to_msg_id = stream.readInt32(exception); + } + date = stream.readInt32(exception); + message = stream.readString(exception); + if ((flags & 512) != 0) { + media = MessageMedia.TLdeserialize(stream, stream.readInt32(exception), exception); + if (media != null) { + ttl = media.ttl_seconds; //custom + } + if (media != null && !TextUtils.isEmpty(media.captionLegacy)) { + message = media.captionLegacy; + } + } + if ((flags & 64) != 0) { + reply_markup = ReplyMarkup.TLdeserialize(stream, stream.readInt32(exception), exception); + } + if ((flags & 128) != 0) { + int magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + int count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + MessageEntity object = MessageEntity.TLdeserialize(stream, stream.readInt32(exception), exception); + if (object == null) { + return; + } + entities.add(object); + } + } + if ((flags & 1024) != 0) { + views = stream.readInt32(exception); + } + if ((flags & 32768) != 0) { + edit_date = stream.readInt32(exception); + } + if ((flags & 65536) != 0) { + post_author = stream.readString(exception); + } + if ((flags & 131072) != 0) { + grouped_id = stream.readInt64(exception); + } + if ((flags & 1048576) != 0) { + reactions = MessageReactions.TLdeserialize(stream, stream.readInt32(exception), exception); + } + if ((flags & 4194304) != 0) { + stream.readString(exception); + } + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + flags = out ? (flags | 2) : (flags &~ 2); + flags = mentioned ? (flags | 16) : (flags &~ 16); + flags = media_unread ? (flags | 32) : (flags &~ 32); + flags = silent ? (flags | 8192) : (flags &~ 8192); + flags = post ? (flags | 16384) : (flags &~ 16384); + flags = from_scheduled ? (flags | 262144) : (flags &~ 262144); + flags = legacy ? (flags | 524288) : (flags &~ 524288); + flags = edit_hide ? (flags | 2097152) : (flags &~ 2097152); + stream.writeInt32(flags); + stream.writeInt32(id); + if ((flags & 256) != 0) { + stream.writeInt32((int) from_id.user_id); + } + peer_id.serializeToStream(stream); + if ((flags & 4) != 0) { + fwd_from.serializeToStream(stream); + } + if ((flags & 2048) != 0) { + stream.writeInt32((int) via_bot_id); + } + if ((flags & 8) != 0) { + stream.writeInt32(reply_to.reply_to_msg_id); + } + stream.writeInt32(date); + stream.writeString(message); + if ((flags & 512) != 0) { + media.serializeToStream(stream); + } + if ((flags & 64) != 0) { + reply_markup.serializeToStream(stream); + } + if ((flags & 128) != 0) { + stream.writeInt32(0x1cb5c415); + int count = entities.size(); + stream.writeInt32(count); + for (int a = 0; a < count; a++) { + entities.get(a).serializeToStream(stream); + } + } + if ((flags & 1024) != 0) { + stream.writeInt32(views); + } + if ((flags & 32768) != 0) { + stream.writeInt32(edit_date); + } + if ((flags & 65536) != 0) { + stream.writeString(post_author); + } + if ((flags & 131072) != 0) { + stream.writeInt64(grouped_id); + } + if ((flags & 1048576) != 0) { + reactions.serializeToStream(stream); + } + if ((flags & 4194304) != 0) { + stream.writeString(""); + } + writeAttachPath(stream); + } + } + + public static class TL_message_layer104 extends TL_message { + public static final int constructor = 0x44f9b43d; public void readParams(AbstractSerializedData stream, boolean exception) { flags = stream.readInt32(exception); @@ -67006,7 +68377,7 @@ public static abstract class Dialog extends TLObject { public int pinnedNum; //custom public boolean isFolder; //custom - public static Dialog TLdeserialize(AbstractSerializedData stream, int constructor, boolean exception) { + public static Dialog TLdeserialize(AbstractSerializedData stream, int constructor, boolean exception) { Dialog result = null; switch (constructor) { case 0xd58a08c6: @@ -67850,9 +69221,11 @@ public void freeResources() { } public static class TL_messages_getSearchResultsPositions extends TLObject { - public static final int constructor = 0x6e9583a3; + public static final int constructor = 0x9c7f2f10; + public int flags; public InputPeer peer; + public InputPeer saved_peer_id; public MessagesFilter filter; public int offset_id; public int limit; @@ -67863,7 +69236,11 @@ public TLObject deserializeResponse(AbstractSerializedData stream, int construct public void serializeToStream(AbstractSerializedData stream) { stream.writeInt32(constructor); + stream.writeInt32(flags); peer.serializeToStream(stream); + if ((flags & 4) != 0) { + saved_peer_id.serializeToStream(stream); + } filter.serializeToStream(stream); stream.writeInt32(offset_id); stream.writeInt32(limit); @@ -68405,15 +69782,34 @@ public void serializeToStream(AbstractSerializedData stream) { } } - public static class TL_keyboardButtonRequestPeer extends KeyboardButton { + public static class TL_keyboardButtonRequestPeer_layer168 extends TL_keyboardButtonRequestPeer { public static final int constructor = 0xd0b468c; + public void readParams(AbstractSerializedData stream, boolean exception) { + text = stream.readString(exception); + button_id = stream.readInt32(exception); + peer_type = RequestPeerType.TLdeserialize(stream, stream.readInt32(exception), exception); + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + stream.writeString(text); + stream.writeInt32(button_id); + peer_type.serializeToStream(stream); + } + } + + public static class TL_keyboardButtonRequestPeer extends KeyboardButton { + public static final int constructor = 0x53d7bfd8; + public RequestPeerType peer_type; + public int max_quantity; public void readParams(AbstractSerializedData stream, boolean exception) { text = stream.readString(exception); button_id = stream.readInt32(exception); peer_type = RequestPeerType.TLdeserialize(stream, stream.readInt32(exception), exception); + max_quantity = stream.readInt32(exception); } public void serializeToStream(AbstractSerializedData stream) { @@ -68421,6 +69817,7 @@ public void serializeToStream(AbstractSerializedData stream) { stream.writeString(text); stream.writeInt32(button_id); peer_type.serializeToStream(stream); + stream.writeInt32(max_quantity); } } @@ -70781,10 +72178,10 @@ public void serializeToStream(AbstractSerializedData stream) { stream.writeBool(enabled); } } - + public static class TL_channels_clickSponsoredMessage extends TLObject { public static final int constructor = 0x18afbc93; - + public InputChannel channel; public byte[] random_id; @@ -70856,12 +72253,12 @@ public void serializeToStream(AbstractSerializedData stream) { } public static class TL_messages_sendBotRequestedPeer extends TLObject { - public static final int constructor = 0xfe38d01b; + public static final int constructor = 0x91b2d060; public InputPeer peer; public int msg_id; public int button_id; - public InputPeer requested_peer; + public ArrayList requested_peers = new ArrayList<>(); public TLObject deserializeResponse(AbstractSerializedData stream, int constructor, boolean exception) { return Updates.TLdeserialize(stream, constructor, exception); @@ -70872,7 +72269,12 @@ public void serializeToStream(AbstractSerializedData stream) { peer.serializeToStream(stream); stream.writeInt32(msg_id); stream.writeInt32(button_id); - requested_peer.serializeToStream(stream); + stream.writeInt32(0x1cb5c415); + int count = requested_peers.size(); + stream.writeInt32(count); + for (int a = 0; a < count; a++) { + requested_peers.get(a).serializeToStream(stream); + } } } @@ -70929,15 +72331,18 @@ public void serializeToStream(AbstractSerializedData stream) { } } - public static class TL_messageActionRequestedPeer extends MessageAction { + public static class TL_messageActionRequestedPeer_layer168 extends TL_messageActionRequestedPeer { public static final int constructor = 0xfe77345d; - public int button_id; public TLRPC.Peer peer; public void readParams(AbstractSerializedData stream, boolean exception) { button_id = stream.readInt32(exception); peer = TLRPC.Peer.TLdeserialize(stream, stream.readInt32(exception), exception); + if (peer == null) { + return; + } + peers.add(peer); } public void serializeToStream(AbstractSerializedData stream) { @@ -70947,6 +72352,43 @@ public void serializeToStream(AbstractSerializedData stream) { } } + public static class TL_messageActionRequestedPeer extends MessageAction { + public static final int constructor = 0x31518e9b; + + public int button_id; + public ArrayList peers = new ArrayList<>(); + + public void readParams(AbstractSerializedData stream, boolean exception) { + button_id = stream.readInt32(exception); + int magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + int count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + Peer object = Peer.TLdeserialize(stream, stream.readInt32(exception), exception); + if (object == null) { + return; + } + peers.add(object); + } + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + stream.writeInt32(button_id); + stream.writeInt32(0x1cb5c415); + int count = peers.size(); + stream.writeInt32(count); + for (int a = 0; a < count; a++) { + peers.get(a).serializeToStream(stream); + } + } + } + public static class TL_photos_uploadContactProfilePhoto extends TLObject { public static final int constructor = 0xe14c4a71; @@ -71078,10 +72520,26 @@ public void serializeToStream(AbstractSerializedData stream) { } } + public static class TL_account_getChannelRestrictedStatusEmojis extends TLObject { + public static final int constructor = 0x35a9e0d5; + + public long hash; + + public TLObject deserializeResponse(AbstractSerializedData stream, int constructor, boolean exception) { + return EmojiList.TLdeserialize(stream, constructor, exception); + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + stream.writeInt64(hash); + } + } + public static class TL_channels_updateColor extends TLObject { - public static final int constructor = 0x621a201f; + public static final int constructor = 0xd8aa3671; public int flags; + public boolean for_profile; public InputChannel channel; public int color; public long background_emoji_id; @@ -71092,9 +72550,12 @@ public TLObject deserializeResponse(AbstractSerializedData stream, int construct public void serializeToStream(AbstractSerializedData stream) { stream.writeInt32(constructor); + flags = for_profile ? (flags | 2) : (flags &~ 2); stream.writeInt32(flags); channel.serializeToStream(stream); - stream.writeInt32(color); + if ((flags & 4) != 0) { + stream.writeInt32(color); + } if ((flags & 1) != 0) { stream.writeInt64(background_emoji_id); } @@ -71143,7 +72604,7 @@ public void serializeToStream(AbstractSerializedData stream) { public static class TL_updateChannelViewForumAsMessages extends Update { public static final int constructor = 0x7b68920; - + public long channel_id; public boolean enabled; @@ -71159,6 +72620,64 @@ public void serializeToStream(AbstractSerializedData stream) { } } + public static class TL_updatePinnedSavedDialogs extends Update { + public static final int constructor = 0x686c85a6; + + public int flags; + public ArrayList order = new ArrayList<>(); + + public void readParams(AbstractSerializedData stream, boolean exception) { + flags = stream.readInt32(exception); + if ((flags & 1) != 0) { + int magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + int count = stream.readInt32(exception); + for (int i = 0; i < count; ++i) { + order.add(DialogPeer.TLdeserialize(stream, stream.readInt32(exception), exception)); + } + } + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + stream.writeInt32(flags); + if ((flags & 1) != 0) { + stream.writeInt32(0x1cb5c415); + int count = order.size(); + stream.writeInt32(count); + for (int i = 0; i < count; ++i) { + order.get(i).serializeToStream(stream); + } + } + } + } + + public static class TL_updateSavedDialogPinned extends Update { + public static final int constructor = 0xaeaf9e74; + + public int flags; + public boolean pinned; + public DialogPeer peer; + + public void readParams(AbstractSerializedData stream, boolean exception) { + flags = stream.readInt32(exception); + pinned = (flags & 1) != 0; + peer = DialogPeer.TLdeserialize(stream, stream.readInt32(exception), exception); + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + flags = pinned ? (flags | 1) : (flags &~ 1); + stream.writeInt32(flags); + peer.serializeToStream(stream); + } + } + public static class TL_updatePeerWallpaper extends Update { public static final int constructor = 0xae3f101d; @@ -71222,12 +72741,12 @@ public void serializeToStream(AbstractSerializedData stream) { public static class TL_contacts_setBlocked extends TLObject { public static final int constructor = 0x94c65c76; - + public int flags; public boolean my_stories_from; public ArrayList id = new ArrayList<>(); public int limit; - + public TLObject deserializeResponse(AbstractSerializedData stream, int constructor, boolean exception) { return Bool.TLdeserialize(stream, constructor, exception); } @@ -71249,7 +72768,7 @@ public static class TL_editCloseFriends extends TLObject { public static final int constructor = 0xba6705f0; public ArrayList id = new ArrayList<>(); - + public TLObject deserializeResponse(AbstractSerializedData stream, int constructor, boolean exception) { return Bool.TLdeserialize(stream, constructor, exception); } @@ -71637,14 +73156,64 @@ public void serializeToStream(AbstractSerializedData stream) { } public static class TL_messageActionGiftCode extends MessageAction { - public static int constructor = 0xd2cfdb0e; + public static final int constructor = 0x678c2e09; public boolean via_giveaway; public boolean unclaimed; public Peer boost_peer; - public int months; public String slug; + public void readParams(AbstractSerializedData stream, boolean exception) { + flags = stream.readInt32(exception); + via_giveaway = (flags & 1) != 0; + unclaimed = (flags & 4) != 0; + if ((flags & 2) != 0) { + boost_peer = Peer.TLdeserialize(stream, stream.readInt32(exception), exception); + } + months = stream.readInt32(exception); + slug = stream.readString(exception); + if ((flags & 4) != 0) { + currency = stream.readString(exception); + } + if ((flags & 4) != 0) { + amount = stream.readInt64(exception); + } + if ((flags & 8) != 0) { + cryptoCurrency = stream.readString(exception); + } + if ((flags & 8) != 0) { + cryptoAmount = stream.readInt64(exception); + } + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + flags = via_giveaway ? (flags | 1) : (flags &~ 1); + flags = unclaimed ? (flags | 4) : (flags &~ 4); + stream.writeInt32(flags); + if ((flags & 2) != 0) { + boost_peer.serializeToStream(stream); + } + stream.writeInt32(months); + stream.writeString(slug); + if ((flags & 4) != 0) { + stream.writeString(currency); + } + if ((flags & 4) != 0) { + stream.writeInt64(amount); + } + if ((flags & 8) != 0) { + stream.writeString(cryptoCurrency); + } + if ((flags & 8) != 0) { + stream.writeInt64(cryptoAmount); + } + } + } + + public static class TL_messageActionGiftCode_layer167 extends TL_messageActionGiftCode { + public static final int constructor = 0xd2cfdb0e; + public void readParams(AbstractSerializedData stream, boolean exception) { flags = stream.readInt32(exception); via_giveaway = (flags & 1) != 0; @@ -71670,7 +73239,7 @@ public void serializeToStream(AbstractSerializedData stream) { } public static class TL_inputStorePaymentPremiumGiftCode extends InputStorePaymentPurpose { - public static int constructor = 0xa3805f3f; + public static final int constructor = 0xa3805f3f; public int flags; public ArrayList users = new ArrayList<>(); @@ -71720,13 +73289,15 @@ public void serializeToStream(AbstractSerializedData stream) { } public static class TL_inputStorePaymentPremiumGiveaway extends InputStorePaymentPurpose { - public static int constructor = 0x7c9375e6; + public static final int constructor = 0x160544ca; public int flags; public boolean only_new_subscribers; + public boolean winners_are_visible; public InputPeer boost_peer; public ArrayList additional_peers = new ArrayList<>(); public ArrayList countries_iso2 = new ArrayList<>(); + public String prize_description; public long random_id; public int until_date; public String currency; @@ -71735,6 +73306,7 @@ public static class TL_inputStorePaymentPremiumGiveaway extends InputStorePaymen public void readParams(AbstractSerializedData stream, boolean exception) { flags = stream.readInt32(exception); only_new_subscribers = (flags & 1) != 0; + winners_are_visible = (flags & 8) != 0; boost_peer = InputPeer.TLdeserialize(stream, stream.readInt32(exception), exception); if ((flags & 2) != 0) { int magic = stream.readInt32(exception); @@ -71766,6 +73338,9 @@ public void readParams(AbstractSerializedData stream, boolean exception) { countries_iso2.add(stream.readString(exception)); } } + if ((flags & 16) != 0) { + prize_description = stream.readString(exception); + } random_id = stream.readInt64(exception); until_date = stream.readInt32(exception); currency = stream.readString(exception); @@ -71775,6 +73350,7 @@ public void readParams(AbstractSerializedData stream, boolean exception) { public void serializeToStream(AbstractSerializedData stream) { stream.writeInt32(constructor); flags = only_new_subscribers ? (flags | 1) : (flags &~ 1); + flags = winners_are_visible ? (flags | 8) : (flags &~ 8); stream.writeInt32(flags); boost_peer.serializeToStream(stream); if ((flags & 2) != 0) { @@ -71793,6 +73369,9 @@ public void serializeToStream(AbstractSerializedData stream) { stream.writeString(countries_iso2.get(a)); } } + if ((flags & 16) != 0) { + stream.writeString(prize_description); + } stream.writeInt64(random_id); stream.writeInt32(until_date); stream.writeString(currency); @@ -71818,16 +73397,155 @@ public void serializeToStream(AbstractSerializedData stream) { } } + public static class TL_messageMediaGiveawayResults extends MessageMedia { + public static final int constructor = 0xc6991068; + + public boolean only_new_subscribers; + public boolean refunded; + public long channel_id; + public int additional_peers_count; + public int launch_msg_id; + public int winners_count; + public int unclaimed_count; + public ArrayList winners = new ArrayList<>(); + public int months; + public String prize_description; + public int until_date; + + public void readParams(AbstractSerializedData stream, boolean exception) { + flags = stream.readInt32(exception); + only_new_subscribers = (flags & 1) != 0; + refunded = (flags & 4) != 0; + channel_id = stream.readInt64(exception); + if ((flags & 8) != 0) { + additional_peers_count = stream.readInt32(exception); + } + launch_msg_id = stream.readInt32(exception); + winners_count = stream.readInt32(exception); + unclaimed_count = stream.readInt32(exception); + int magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + int count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + winners.add(stream.readInt64(exception)); + } + months = stream.readInt32(exception); + if ((flags & 2) != 0) { + prize_description = stream.readString(exception); + } + until_date = stream.readInt32(exception); + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + flags = only_new_subscribers ? (flags | 1) : (flags &~ 1); + flags = refunded ? (flags | 4) : (flags &~ 4); + stream.writeInt32(flags); + stream.writeInt64(channel_id); + if ((flags & 8) != 0) { + stream.writeInt32(additional_peers_count); + } + stream.writeInt32(launch_msg_id); + stream.writeInt32(winners_count); + stream.writeInt32(unclaimed_count); + stream.writeInt32(0x1cb5c415); + int count = winners.size(); + stream.writeInt32(count); + for (int a = 0; a < count; a++) { + stream.writeInt64(winners.get(a)); + } + stream.writeInt32(months); + if ((flags & 2) != 0) { + stream.writeString(prize_description); + } + stream.writeInt32(until_date); + } + } + public static class TL_messageMediaGiveaway extends MessageMedia { - public static int constructor = 0x58260664; + public static final int constructor = 0xdaad85b0; public boolean only_new_subscribers; + public boolean winners_are_visible; public ArrayList channels = new ArrayList<>(); public ArrayList countries_iso2 = new ArrayList<>(); + public String prize_description; public int quantity; public int months; public int until_date; + public void readParams(AbstractSerializedData stream, boolean exception) { + flags = stream.readInt32(exception); + only_new_subscribers = (flags & 1) != 0; + winners_are_visible = (flags & 4) != 0; + int magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + int count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + channels.add(stream.readInt64(exception)); + } + if ((flags & 2) != 0) { + magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + countries_iso2.add(stream.readString(exception)); + } + } + if ((flags & 8) != 0) { + prize_description = stream.readString(exception); + } + quantity = stream.readInt32(exception); + months = stream.readInt32(exception); + until_date = stream.readInt32(exception); + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + flags = only_new_subscribers ? (flags | 1) : (flags &~ 1); + flags = winners_are_visible ? (flags | 4) : (flags &~ 4); + stream.writeInt32(flags); + stream.writeInt32(0x1cb5c415); + int count = channels.size(); + stream.writeInt32(count); + for (int a = 0; a < count; a++) { + stream.writeInt64(channels.get(a)); + } + if ((flags & 2) != 0) { + stream.writeInt32(0x1cb5c415); + count = countries_iso2.size(); + stream.writeInt32(count); + for (int a = 0; a < count; a++) { + stream.writeString(countries_iso2.get(a)); + } + } + if ((flags & 8) != 0) { + stream.writeString(prize_description); + } + stream.writeInt32(quantity); + stream.writeInt32(months); + stream.writeInt32(until_date); + } + } + + public static class TL_messageMediaGiveaway_layer167 extends TL_messageMediaGiveaway { + public static final int constructor = 0x58260664; + public void readParams(AbstractSerializedData stream, boolean exception) { flags = stream.readInt32(exception); only_new_subscribers = (flags & 1) != 0; @@ -71997,7 +73715,7 @@ public void serializeToStream(AbstractSerializedData stream) { } public static class TL_payments_giveawayInfoResults extends payments_GiveawayInfo { - public static int constructor = 0xcd5570; + public static final int constructor = 0xcd5570; public int flags; public boolean winner; @@ -72037,7 +73755,7 @@ public void serializeToStream(AbstractSerializedData stream) { } public static class TL_payments_giveawayInfo extends payments_GiveawayInfo { - public static int constructor = 0x4367daa0; + public static final int constructor = 0x4367daa0; public int flags; public boolean participating; @@ -72086,10 +73804,10 @@ public static abstract class payments_GiveawayInfo extends TLObject { public static payments_GiveawayInfo TLdeserialize(AbstractSerializedData stream, int constructor, boolean exception) { payments_GiveawayInfo result = null; switch (constructor) { - case 0xcd5570: + case TL_payments_giveawayInfoResults.constructor: result = new TL_payments_giveawayInfoResults(); break; - case 0x4367daa0: + case TL_payments_giveawayInfo.constructor: result = new TL_payments_giveawayInfo(); break; } @@ -72121,7 +73839,7 @@ public void serializeToStream(AbstractSerializedData stream) { } public static class TL_payments_checkedGiftCode extends TLObject { - public static int constructor = 0xb722f158; + public static final int constructor = 0x284a1096; public static final long NO_USER_ID = -1L; //custom public int flags; @@ -72152,7 +73870,9 @@ public static TL_payments_checkedGiftCode TLdeserialize(AbstractSerializedData s public void readParams(AbstractSerializedData stream, boolean exception) { flags = stream.readInt32(exception); via_giveaway = (flags & 4) != 0; - from_id = Peer.TLdeserialize(stream, stream.readInt32(exception), exception); + if ((flags & 16) != 0) { + from_id = Peer.TLdeserialize(stream, stream.readInt32(exception), exception); + } if ((flags & 8) != 0) { giveaway_msg_id = stream.readInt32(exception); } @@ -72198,9 +73918,11 @@ public void readParams(AbstractSerializedData stream, boolean exception) { public void serializeToStream(AbstractSerializedData stream) { stream.writeInt32(constructor); - flags = via_giveaway ? (flags | 4) : (flags &~ 4); + flags = via_giveaway ? (flags | 4) : (flags & ~4); stream.writeInt32(flags); - from_id.serializeToStream(stream); + if ((flags & 16) != 0) { + from_id.serializeToStream(stream); + } if ((flags & 8) != 0) { stream.writeInt32(giveaway_msg_id); } @@ -72256,7 +73978,7 @@ public void serializeToStream(AbstractSerializedData stream) { public static class TL_messageActionGiveawayResults extends MessageAction { public static int constructor = 0x2a9fadc5; - + public int winners_count; public int unclaimed_count; @@ -72273,6 +73995,25 @@ public void serializeToStream(AbstractSerializedData stream) { } } + public static class TL_channels_updateEmojiStatus extends TLObject { + public static int constructor = 0xf0d3e6a8; + + public InputChannel channel; + public EmojiStatus emoji_status; + + @Override + public TLObject deserializeResponse(AbstractSerializedData stream, int constructor, boolean exception) { + return Updates.TLdeserialize(stream, constructor, exception); + } + + @Override + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + channel.serializeToStream(stream); + emoji_status.serializeToStream(stream); + } + } + public static class TL_channels_getChannelRecommendations extends TLObject { public static int constructor = 0x83b70d97; @@ -72282,7 +74023,7 @@ public static class TL_channels_getChannelRecommendations extends TLObject { public TLObject deserializeResponse(AbstractSerializedData stream, int constructor, boolean exception) { return messages_Chats.TLdeserialize(stream, constructor, exception); } - + public void serializeToStream(AbstractSerializedData stream) { stream.writeInt32(constructor); channel.serializeToStream(stream); @@ -72348,7 +74089,7 @@ public void serializeToStream(AbstractSerializedData stream) { } } } - + public static class help_PeerColorSet extends TLObject { public static help_PeerColorSet TLdeserialize(AbstractSerializedData stream, int constructor, boolean exception) { help_PeerColorSet result = null; @@ -72473,13 +74214,14 @@ public void serializeToStream(AbstractSerializedData stream) { } public static class TL_help_peerColorOption extends TLObject { - public static final int constructor = 0x135bd42f; - + public static final int constructor = 0xef8430ab; + public int flags; public boolean hidden; public int color_id; public help_PeerColorSet colors; public help_PeerColorSet dark_colors; + public int channel_min_level; public static TL_help_peerColorOption TLdeserialize(AbstractSerializedData stream, int constructor, boolean exception) { if (TL_help_peerColorOption.constructor != constructor) { @@ -72505,6 +74247,9 @@ public void readParams(AbstractSerializedData stream, boolean exception) { if ((flags & 4) != 0) { dark_colors = help_PeerColorSet.TLdeserialize(stream, stream.readInt32(exception), exception); } + if ((flags & 8) != 0) { + channel_min_level = stream.readInt32(exception); + } } @Override @@ -72519,6 +74264,9 @@ public void serializeToStream(AbstractSerializedData stream) { if ((flags & 4) != 0) { dark_colors.serializeToStream(stream); } + if ((flags & 8) != 0) { + stream.writeInt32(channel_min_level); + } } } @@ -72542,7 +74290,7 @@ public static help_PeerColors TLdeserialize(AbstractSerializedData stream, int c return result; } } - + public static class TL_help_peerColorsNotModified extends help_PeerColors { public static final int constructor = 0x2ba1f5ce; @@ -72594,7 +74342,7 @@ public void serializeToStream(AbstractSerializedData stream) { public static class TL_help_getPeerColors extends TLObject { public static final int constructor = 0xda80f42f; - + public int hash; @Override @@ -72626,6 +74374,382 @@ public void serializeToStream(AbstractSerializedData stream) { } } + public static class TL_savedDialog extends TLObject { + public static final int constructor = 0xbd87cb6c; + + public int flags; + public boolean pinned; + public Peer peer; + public int top_message; + + public static TL_savedDialog TLdeserialize(AbstractSerializedData stream, int constructor, boolean exception) { + if (TL_savedDialog.constructor != constructor) { + if (exception) { + throw new RuntimeException(String.format("can't parse magic %x in TL_savedDialog", constructor)); + } else { + return null; + } + } + TL_savedDialog result = new TL_savedDialog(); + result.readParams(stream, exception); + return result; + } + + @Override + public void readParams(AbstractSerializedData stream, boolean exception) { + flags = stream.readInt32(exception); + pinned = (flags & 4) != 0; + peer = Peer.TLdeserialize(stream, stream.readInt32(exception), exception); + top_message = stream.readInt32(exception); + } + + @Override + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + flags = pinned ? (flags | 4) : (flags &~ 4); + stream.writeInt32(flags); + peer.serializeToStream(stream); + stream.writeInt32(top_message); + } + } + + public static class messages_SavedDialogs extends TLObject { + public static messages_SavedDialogs TLdeserialize(AbstractSerializedData stream, int constructor, boolean exception) { + messages_SavedDialogs result = null; + switch (constructor) { + case TL_messages_savedDialogs.constructor: + result = new TL_messages_savedDialogs(); + break; + case TL_messages_savedDialogsSlice.constructor: + result = new TL_messages_savedDialogsSlice(); + break; + case TL_messages_savedDialogsNotModified.constructor: + result = new TL_messages_savedDialogsNotModified(); + break; + } + if (result == null && exception) { + throw new RuntimeException(String.format("can't parse magic %x in messages_SavedDialogs", constructor)); + } + if (result != null) { + result.readParams(stream, exception); + } + return result; + } + } + + public static class TL_messages_savedDialogs extends messages_SavedDialogs { + public static final int constructor = 0xf83ae221; + + public ArrayList dialogs = new ArrayList<>(); + public ArrayList messages = new ArrayList<>(); + public ArrayList chats = new ArrayList<>(); + public ArrayList users = new ArrayList<>(); + + @Override + public void readParams(AbstractSerializedData stream, boolean exception) { + int magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + int count = stream.readInt32(exception); + for (int i = 0; i < count; ++i) { + dialogs.add(TL_savedDialog.TLdeserialize(stream, stream.readInt32(exception), exception)); + } + magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + count = stream.readInt32(exception); + for (int i = 0; i < count; ++i) { + messages.add(Message.TLdeserialize(stream, stream.readInt32(exception), exception)); + } + magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + count = stream.readInt32(exception); + for (int i = 0; i < count; ++i) { + chats.add(Chat.TLdeserialize(stream, stream.readInt32(exception), exception)); + } + magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + count = stream.readInt32(exception); + for (int i = 0; i < count; ++i) { + users.add(User.TLdeserialize(stream, stream.readInt32(exception), exception)); + } + } + + @Override + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + stream.writeInt32(0x1cb5c415); + int count = dialogs.size(); + stream.writeInt32(count); + for (int i = 0; i < count; ++i) { + dialogs.get(i).serializeToStream(stream); + } + stream.writeInt32(0x1cb5c415); + count = messages.size(); + stream.writeInt32(count); + for (int i = 0; i < count; ++i) { + messages.get(i).serializeToStream(stream); + } + stream.writeInt32(0x1cb5c415); + count = users.size(); + stream.writeInt32(count); + for (int i = 0; i < count; ++i) { + users.get(i).serializeToStream(stream); + } + stream.writeInt32(0x1cb5c415); + count = chats.size(); + stream.writeInt32(count); + for (int i = 0; i < count; ++i) { + chats.get(i).serializeToStream(stream); + } + } + } + + public static class TL_messages_savedDialogsSlice extends messages_SavedDialogs { + public static final int constructor = 0x44ba9dd9; + + public int count; + public ArrayList dialogs = new ArrayList<>(); + public ArrayList messages = new ArrayList<>(); + public ArrayList chats = new ArrayList<>(); + public ArrayList users = new ArrayList<>(); + + @Override + public void readParams(AbstractSerializedData stream, boolean exception) { + count = stream.readInt32(exception); + int magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + int count = stream.readInt32(exception); + for (int i = 0; i < count; ++i) { + dialogs.add(TL_savedDialog.TLdeserialize(stream, stream.readInt32(exception), exception)); + } + magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + count = stream.readInt32(exception); + for (int i = 0; i < count; ++i) { + messages.add(Message.TLdeserialize(stream, stream.readInt32(exception), exception)); + } + magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + count = stream.readInt32(exception); + for (int i = 0; i < count; ++i) { + chats.add(Chat.TLdeserialize(stream, stream.readInt32(exception), exception)); + } + magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + count = stream.readInt32(exception); + for (int i = 0; i < count; ++i) { + users.add(User.TLdeserialize(stream, stream.readInt32(exception), exception)); + } + } + + @Override + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + stream.writeInt32(count); + stream.writeInt32(0x1cb5c415); + int count = dialogs.size(); + stream.writeInt32(count); + for (int i = 0; i < count; ++i) { + dialogs.get(i).serializeToStream(stream); + } + stream.writeInt32(0x1cb5c415); + count = messages.size(); + stream.writeInt32(count); + for (int i = 0; i < count; ++i) { + messages.get(i).serializeToStream(stream); + } + stream.writeInt32(0x1cb5c415); + count = users.size(); + stream.writeInt32(count); + for (int i = 0; i < count; ++i) { + users.get(i).serializeToStream(stream); + } + stream.writeInt32(0x1cb5c415); + count = chats.size(); + stream.writeInt32(count); + for (int i = 0; i < count; ++i) { + chats.get(i).serializeToStream(stream); + } + } + } + + public static class TL_messages_savedDialogsNotModified extends messages_SavedDialogs { + public static final int constructor = 0xc01f6fe8; + + public int count; + + @Override + public void readParams(AbstractSerializedData stream, boolean exception) { + count = stream.readInt32(exception); + } + + @Override + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + stream.writeInt32(count); + } + } + + public static class TL_messages_getPinnedSavedDialogs extends TLObject { + public static final int constructor = 0xd63d94e0; + + @Override + public TLObject deserializeResponse(AbstractSerializedData stream, int constructor, boolean exception) { + return messages_SavedDialogs.TLdeserialize(stream, constructor, exception); + } + + @Override + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + } + } + + public static class TL_messages_getSavedDialogs extends TLObject { + public static final int constructor = 0x5381d21a; + + public int flags; + public boolean exclude_pinned; + public int offset_date; + public int offset_id; + public InputPeer offset_peer; + public int limit; + public long hash; + + @Override + public TLObject deserializeResponse(AbstractSerializedData stream, int constructor, boolean exception) { + return messages_SavedDialogs.TLdeserialize(stream, constructor, exception); + } + + @Override + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + flags = exclude_pinned ? (flags | 1) : (flags &~ 1); + stream.writeInt32(flags); + stream.writeInt32(offset_date); + stream.writeInt32(offset_id); + offset_peer.serializeToStream(stream); + stream.writeInt32(limit); + stream.writeInt64(hash); + } + } + + public static class TL_messages_getSavedHistory extends TLObject { + public static final int constructor = 0x3d9a414d; + + public InputPeer peer; + public int offset_id; + public int offset_date; + public int add_offset; + public int limit; + public int max_id; + public int min_id; + public long hash; + + @Override + public TLObject deserializeResponse(AbstractSerializedData stream, int constructor, boolean exception) { + return messages_Messages.TLdeserialize(stream, constructor, exception); + } + + @Override + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + peer.serializeToStream(stream); + stream.writeInt32(offset_id); + stream.writeInt32(offset_date); + stream.writeInt32(add_offset); + stream.writeInt32(limit); + stream.writeInt32(max_id); + stream.writeInt32(min_id); + stream.writeInt64(hash); + } + } + + public static class TL_messages_toggleSavedDialogPin extends TLObject { + public static final int constructor = 0xac81bbde; + + public int flags; + public boolean pinned; + public InputDialogPeer peer; + + @Override + public TLObject deserializeResponse(AbstractSerializedData stream, int constructor, boolean exception) { + return Bool.TLdeserialize(stream, constructor, exception); + } + + @Override + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + flags = pinned ? (flags | 1) : (flags &~ 1); + stream.writeInt32(flags); + peer.serializeToStream(stream); + } + } + + public static class TL_messages_reorderPinnedSavedDialogs extends TLObject { + public static final int constructor = 0x8b716587; + + public int flags; + public boolean force; + public ArrayList order = new ArrayList<>(); + + @Override + public TLObject deserializeResponse(AbstractSerializedData stream, int constructor, boolean exception) { + return Bool.TLdeserialize(stream, constructor, exception); + } + + @Override + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + flags = force ? (flags | 1) : (flags &~ 1); + stream.writeInt32(flags); + stream.writeInt32(0x1cb5c415); + int count = order.size(); + stream.writeInt32(count); + for (int i = 0; i < count; ++i) { + order.get(i).serializeToStream(stream); + } + } + } + public static class Vector extends TLObject { public static final int constructor = 0x1cb5c415; public ArrayList objects = new ArrayList<>(); diff --git a/TMessagesProj/src/main/java/org/telegram/tgnet/tl/TL_stories.java b/TMessagesProj/src/main/java/org/telegram/tgnet/tl/TL_stories.java index f2d7899d2bb..2aa0bf5de5c 100644 --- a/TMessagesProj/src/main/java/org/telegram/tgnet/tl/TL_stories.java +++ b/TMessagesProj/src/main/java/org/telegram/tgnet/tl/TL_stories.java @@ -1,5 +1,6 @@ package org.telegram.tgnet.tl; +import org.telegram.messenger.DialogObject; import org.telegram.tgnet.AbstractSerializedData; import org.telegram.tgnet.TLObject; import org.telegram.tgnet.TLRPC; @@ -78,8 +79,7 @@ public void serializeToStream(AbstractSerializedData stream) { } } - public static class TL_storyView extends TLObject { - public static final int constructor = 0xb0bdeac5; + public static class StoryView extends TLObject { public int flags; public boolean blocked; @@ -87,19 +87,35 @@ public static class TL_storyView extends TLObject { public long user_id; public int date; public TLRPC.Reaction reaction; + public TLRPC.Message message; + public TLRPC.Peer peer_id; + public StoryItem story; - public static TL_storyView TLdeserialize(AbstractSerializedData stream, int constructor, boolean exception) { - if (TL_storyView.constructor != constructor) { - if (exception) { - throw new RuntimeException(String.format("can't parse magic %x in TL_storyView", constructor)); - } else { - return null; - } + public static StoryView TLdeserialize(AbstractSerializedData stream, int constructor, boolean exception) { + StoryView result = null; + switch (constructor) { + case TL_storyView.constructor: + result = new TL_storyView(); + break; + case TL_storyViewPublicForward.constructor: + result = new TL_storyViewPublicForward(); + break; + case TL_storyViewPublicRepost.constructor: + result = new TL_storyViewPublicRepost(); + break; + } + if (result == null && exception) { + throw new RuntimeException(String.format("can't parse magic %x in StoryView", constructor)); + } + if (result != null) { + result.readParams(stream, exception); } - TL_storyView result = new TL_storyView(); - result.readParams(stream, exception); return result; } + } + + public static class TL_storyView extends StoryView { + public static final int constructor = 0xb0bdeac5; public void readParams(AbstractSerializedData stream, boolean exception) { flags = stream.readInt32(exception); @@ -125,6 +141,46 @@ public void serializeToStream(AbstractSerializedData stream) { } } + public static class TL_storyViewPublicForward extends StoryView { + public static final int constructor = 0x9083670b; + + public void readParams(AbstractSerializedData stream, boolean exception) { + flags = stream.readInt32(exception); + blocked = (flags & 1) != 0; + blocked_my_stories_from = (flags & 2) != 0; + message = TLRPC.Message.TLdeserialize(stream, stream.readInt32(exception), exception); + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + flags = blocked ? (flags | 1) : (flags &~ 1); + flags = blocked_my_stories_from ? (flags | 2) : (flags &~ 2); + stream.writeInt32(flags); + message.serializeToStream(stream); + } + } + + public static class TL_storyViewPublicRepost extends StoryView { + public static final int constructor = 0xbd74cf49; + + public void readParams(AbstractSerializedData stream, boolean exception) { + flags = stream.readInt32(exception); + blocked = (flags & 1) != 0; + blocked_my_stories_from = (flags & 2) != 0; + peer_id = TLRPC.Peer.TLdeserialize(stream, stream.readInt32(exception), exception); + story = StoryItem.TLdeserialize(stream, stream.readInt32(exception), exception); + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + flags = blocked ? (flags | 1) : (flags &~ 1); + flags = blocked_my_stories_from ? (flags | 2) : (flags &~ 2); + stream.writeInt32(flags); + peer_id.serializeToStream(stream); + story.serializeToStream(stream); + } + } + public static abstract class PeerStories extends TLObject { public int flags; @@ -862,28 +918,131 @@ public void serializeToStream(AbstractSerializedData stream) { } } - public static class TL_stories_storyViewsList extends TLObject { - public static final int constructor = 0x46e9b9ec; + public static class StoryViewsList extends TLObject { public int flags; public int count; + public int views_count; + public int forwards_count; public int reactions_count; - public ArrayList views = new ArrayList<>(); + public ArrayList views = new ArrayList<>(); + public ArrayList chats = new ArrayList<>(); public ArrayList users = new ArrayList<>(); public String next_offset = ""; - public static TL_stories_storyViewsList TLdeserialize(AbstractSerializedData stream, int constructor, boolean exception) { - if (TL_stories_storyViewsList.constructor != constructor) { + public static StoryViewsList TLdeserialize(AbstractSerializedData stream, int constructor, boolean exception) { + StoryViewsList result = null; + switch (constructor) { + case TL_storyViewsList.constructor: + result = new TL_storyViewsList(); + break; + case TL_storyViewsList_layer167.constructor: + result = new TL_storyViewsList_layer167(); + break; + } + if (result == null && exception) { + throw new RuntimeException(String.format("can't parse magic %x in StoryViewsList", constructor)); + } + if (result != null) { + result.readParams(stream, exception); + } + return result; + } + } + + public static class TL_storyViewsList extends StoryViewsList { + public static final int constructor = 0x59d78fc5; + + public void readParams(AbstractSerializedData stream, boolean exception) { + flags = stream.readInt32(exception); + count = stream.readInt32(exception); + views_count = stream.readInt32(exception); + forwards_count = stream.readInt32(exception); + reactions_count = stream.readInt32(exception); + int magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { if (exception) { - throw new RuntimeException(String.format("can't parse magic %x in TL_stories_storyViewsList", constructor)); - } else { - return null; + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); } + return; + } + int count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + StoryView object = StoryView.TLdeserialize(stream, stream.readInt32(exception), exception); + if (object == null) { + return; + } + views.add(object); + } + magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + TLRPC.Chat object = TLRPC.Chat.TLdeserialize(stream, stream.readInt32(exception), exception); + if (object == null) { + return; + } + chats.add(object); + } + magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + TLRPC.User object = TLRPC.User.TLdeserialize(stream, stream.readInt32(exception), exception); + if (object == null) { + return; + } + users.add(object); + } + if ((flags & 1) != 0) { + next_offset = stream.readString(exception); + } + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + stream.writeInt32(flags); + stream.writeInt32(count); + stream.writeInt32(views_count); + stream.writeInt32(forwards_count); + stream.writeInt32(reactions_count); + stream.writeInt32(0x1cb5c415); + int count = views.size(); + stream.writeInt32(count); + for (int a = 0; a < count; a++) { + views.get(a).serializeToStream(stream); + } + stream.writeInt32(0x1cb5c415); + count = chats.size(); + stream.writeInt32(count); + for (int a = 0; a < count; a++) { + chats.get(a).serializeToStream(stream); + } + stream.writeInt32(0x1cb5c415); + count = users.size(); + stream.writeInt32(count); + for (int a = 0; a < count; a++) { + users.get(a).serializeToStream(stream); + } + if ((flags & 1) != 0) { + stream.writeString(next_offset); } - TL_stories_storyViewsList result = new TL_stories_storyViewsList(); - result.readParams(stream, exception); - return result; } + } + + + public static class TL_storyViewsList_layer167 extends StoryViewsList { + public static final int constructor = 0x46e9b9ec; public void readParams(AbstractSerializedData stream, boolean exception) { flags = stream.readInt32(exception); @@ -898,7 +1057,7 @@ public void readParams(AbstractSerializedData stream, boolean exception) { } int count = stream.readInt32(exception); for (int a = 0; a < count; a++) { - TL_storyView object = TL_storyView.TLdeserialize(stream, stream.readInt32(exception), exception); + StoryView object = StoryView.TLdeserialize(stream, stream.readInt32(exception), exception); if (object == null) { return; } @@ -975,6 +1134,7 @@ public static class TL_stories_getStoryViewsList extends TLObject { public int flags; public boolean just_contacts; public boolean reactions_first; + public boolean forwards_first; public TLRPC.InputPeer peer; public String q; public int id; @@ -982,13 +1142,14 @@ public static class TL_stories_getStoryViewsList extends TLObject { public int limit; public TLObject deserializeResponse(AbstractSerializedData stream, int constructor, boolean exception) { - return TL_stories_storyViewsList.TLdeserialize(stream, constructor, exception); + return StoryViewsList.TLdeserialize(stream, constructor, exception); } public void serializeToStream(AbstractSerializedData stream) { stream.writeInt32(constructor); flags = just_contacts ? (flags | 1) : (flags &~ 1); flags = reactions_first ? (flags | 4) : (flags &~ 4); + flags = forwards_first ? (flags | 8) : (flags &~ 8); stream.writeInt32(flags); peer.serializeToStream(stream); if ((flags & 2) != 0) { @@ -2543,9 +2704,15 @@ public static MediaArea TLdeserialize(AbstractSerializedData stream, int constru case TL_inputMediaAreaVenue.constructor: result = new TL_inputMediaAreaVenue(); break; + case TL_inputMediaAreaChannelPost.constructor: + result = new TL_inputMediaAreaChannelPost(); + break; case TL_mediaAreaSuggestedReaction.constructor: result = new TL_mediaAreaSuggestedReaction(); break; + case TL_mediaAreaChannelPost.constructor: + result = new TL_mediaAreaChannelPost(); + break; } if (result == null && exception) { throw new RuntimeException(String.format("can't parse magic %x in MediaArea", constructor)); @@ -2578,6 +2745,26 @@ public void serializeToStream(AbstractSerializedData stream) { } } + public static class TL_mediaAreaChannelPost extends MediaArea { + public static final int constructor = 0x770416af; + + public long channel_id; + public int msg_id; + + public void readParams(AbstractSerializedData stream, boolean exception) { + coordinates = TL_mediaAreaCoordinates.TLdeserialize(stream, stream.readInt32(exception), exception); + channel_id = stream.readInt64(exception); + msg_id = stream.readInt32(exception); + } + + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + coordinates.serializeToStream(stream); + stream.writeInt64(channel_id); + stream.writeInt32(msg_id); + } + } + public static class TL_mediaAreaVenue extends MediaArea { public static final int constructor = 0xbe82db9c; @@ -2634,6 +2821,28 @@ public void serializeToStream(AbstractSerializedData stream) { } } + public static class TL_inputMediaAreaChannelPost extends MediaArea { + public static final int constructor = 0x2271f2bf; + + public TLRPC.InputChannel channel; + public int msg_id; + + @Override + public void readParams(AbstractSerializedData stream, boolean exception) { + coordinates = TL_mediaAreaCoordinates.TLdeserialize(stream, stream.readInt32(exception), exception); + channel = TLRPC.InputChannel.TLdeserialize(stream, stream.readInt32(exception), exception); + msg_id = stream.readInt32(exception); + } + + @Override + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + coordinates.serializeToStream(stream); + channel.serializeToStream(stream); + stream.writeInt32(msg_id); + } + } + public static class TL_mediaAreaGeoPoint extends MediaArea { public static final int constructor = 0xdf8b3b22; @@ -2741,4 +2950,229 @@ public void serializeToStream(AbstractSerializedData stream) { stream.writeInt32(id); } } + + public static class StoryReaction extends TLObject { + + public TLRPC.Peer peer_id; + public StoryItem story; + public TLRPC.Message message; + + public static StoryReaction TLdeserialize(AbstractSerializedData stream, int constructor, boolean exception) { + StoryReaction result = null; + switch (constructor) { + case TL_storyReaction.constructor: + result = new TL_storyReaction(); + break; + case TL_storyReactionPublicForward.constructor: + result = new TL_storyReactionPublicForward(); + break; + case TL_storyReactionPublicRepost.constructor: + result = new TL_storyReactionPublicRepost(); + break; + } + if (result == null && exception) { + throw new RuntimeException(String.format("can't parse magic %x in StoryReaction", constructor)); + } + if (result != null) { + result.readParams(stream, exception); + } + return result; + } + } + + public static class TL_storyReactionPublicForward extends StoryReaction { + public final static int constructor = 0xbbab2643; + + @Override + public void readParams(AbstractSerializedData stream, boolean exception) { + message = TLRPC.Message.TLdeserialize(stream, stream.readInt32(exception), exception); + } + + @Override + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + message.serializeToStream(stream); + } + } + + public static class TL_storyReactionPublicRepost extends StoryReaction { + public final static int constructor = 0xcfcd0f13; + + @Override + public void readParams(AbstractSerializedData stream, boolean exception) { + peer_id = TLRPC.Peer.TLdeserialize(stream, stream.readInt32(exception), exception); + story = StoryItem.TLdeserialize(stream, stream.readInt32(exception), exception); + if (story != null) { + story.dialogId = DialogObject.getPeerDialogId(peer_id); + } + } + + @Override + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + peer_id.serializeToStream(stream); + story.serializeToStream(stream); + } + } + + public static class TL_storyReaction extends StoryReaction { + public final static int constructor = 0x6090d6d5; + + public int date; + public TLRPC.Reaction reaction; + + @Override + public void readParams(AbstractSerializedData stream, boolean exception) { + peer_id = TLRPC.Peer.TLdeserialize(stream, stream.readInt32(exception), exception); + date = stream.readInt32(exception); + reaction = TLRPC.Reaction.TLdeserialize(stream, stream.readInt32(exception), exception); + } + + @Override + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + peer_id.serializeToStream(stream); + stream.writeInt32(date); + reaction.serializeToStream(stream); + } + } + + public static class TL_storyReactionsList extends TLObject { + public final static int constructor = 0xaa5f789c; + + public int flags; + public int count; + public ArrayList reactions = new ArrayList<>(); + public ArrayList chats = new ArrayList<>(); + public ArrayList users = new ArrayList<>(); + public String next_offset; + + public static TL_storyReactionsList TLdeserialize(AbstractSerializedData stream, int constructor, boolean exception) { + if (TL_storyReactionsList.constructor != constructor) { + if (exception) { + throw new RuntimeException(String.format("can't parse magic %x in TL_storyReactionsList", constructor)); + } else { + return null; + } + } + TL_storyReactionsList result = new TL_storyReactionsList(); + result.readParams(stream, exception); + return result; + } + + @Override + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + stream.writeInt32(flags); + stream.writeInt32(count); + stream.writeInt32(0x1cb5c415); + int count = reactions.size(); + stream.writeInt32(count); + for (int i = 0; i < count; ++i) { + reactions.get(i).serializeToStream(stream); + } + stream.writeInt32(0x1cb5c415); + count = chats.size(); + stream.writeInt32(count); + for (int i = 0; i < count; ++i) { + chats.get(i).serializeToStream(stream); + } + stream.writeInt32(0x1cb5c415); + count = users.size(); + stream.writeInt32(count); + for (int i = 0; i < count; ++i) { + users.get(i).serializeToStream(stream); + } + if ((flags & 1) != 0) { + stream.writeString(next_offset); + } + } + + @Override + public void readParams(AbstractSerializedData stream, boolean exception) { + flags = stream.readInt32(exception); + count = stream.readInt32(exception); + int magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + int count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + StoryReaction object = StoryReaction.TLdeserialize(stream, stream.readInt32(exception), exception); + if (object == null) { + return; + } + reactions.add(object); + } + magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + TLRPC.Chat object = TLRPC.Chat.TLdeserialize(stream, stream.readInt32(exception), exception); + if (object == null) { + return; + } + chats.add(object); + } + magic = stream.readInt32(exception); + if (magic != 0x1cb5c415) { + if (exception) { + throw new RuntimeException(String.format("wrong Vector magic, got %x", magic)); + } + return; + } + count = stream.readInt32(exception); + for (int a = 0; a < count; a++) { + TLRPC.User object = TLRPC.User.TLdeserialize(stream, stream.readInt32(exception), exception); + if (object == null) { + return; + } + users.add(object); + } + if ((flags & 1) != 0) { + next_offset = stream.readString(exception); + } + } + } + + public static class TL_getStoryReactionsList extends TLObject { + public final static int constructor = 0xb9b2881f; + + public int flags; + public boolean forwards_first; + public TLRPC.InputPeer peer; + public int id; + public TLRPC.Reaction reaction; + public String offset; + public int limit; + + @Override + public TLObject deserializeResponse(AbstractSerializedData stream, int constructor, boolean exception) { + return TL_storyReactionsList.TLdeserialize(stream, constructor, exception); + } + + @Override + public void serializeToStream(AbstractSerializedData stream) { + stream.writeInt32(constructor); + flags = forwards_first ? (flags | 4) : (flags &~ 4); + stream.writeInt32(flags); + peer.serializeToStream(stream); + stream.writeInt32(id); + if ((flags & 1) != 0) { + reaction.serializeToStream(stream); + } + if ((flags & 2) != 0) { + stream.writeString(offset); + } + stream.writeInt32(limit); + } + } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/ActionBar.java b/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/ActionBar.java index 8ca2b890521..bc5b09713d8 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/ActionBar.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/ActionBar.java @@ -8,6 +8,8 @@ package org.telegram.ui.ActionBar; +import static org.telegram.messenger.AndroidUtilities.dp; + import android.animation.Animator; import android.animation.AnimatorListenerAdapter; import android.animation.AnimatorSet; @@ -192,7 +194,7 @@ private void createBackButtonImage() { if (itemsColor != 0) { backButtonImageView.setColorFilter(new PorterDuffColorFilter(itemsColor, colorFilterMode)); } - backButtonImageView.setPadding(AndroidUtilities.dp(1), 0, 0, 0); + backButtonImageView.setPadding(dp(1), 0, 0, 0); addView(backButtonImageView, LayoutHelper.createFrame(54, 54, Gravity.LEFT | Gravity.TOP)); backButtonImageView.setOnClickListener(v -> { @@ -302,19 +304,18 @@ protected boolean drawChild(Canvas canvas, View child, long drawingTime) { canvas.clipRect(0, -getTranslationY() + (occupyStatusBar ? AndroidUtilities.statusBarHeight : 0), getMeasuredWidth(), getMeasuredHeight()); } boolean result = super.drawChild(canvas, child, drawingTime); - if (supportsHolidayImage && !titleOverlayShown && !LocaleController.isRTL && (child == titleTextView[0] || child == titleTextView[1])) { + if (supportsHolidayImage && !titleOverlayShown && !LocaleController.isRTL && (child == titleTextView[0] || child == titleTextView[1] || child == titlesContainer && useContainerForTitles)) { Drawable drawable = Theme.getCurrentHolidayDrawable(); if (drawable != null) { - - SimpleTextView titleView = (SimpleTextView) child; - if (titleView.getVisibility() == View.VISIBLE && titleView.getText() instanceof String) { + SimpleTextView titleView = child == titlesContainer ? titleTextView[0] : (SimpleTextView) child; + if (titleView != null && titleView.getVisibility() == View.VISIBLE && titleView.getText() instanceof String) { TextPaint textPaint = titleView.getTextPaint(); textPaint.getFontMetricsInt(fontMetricsInt); textPaint.getTextBounds((String) titleView.getText(), 0, 1, rect); int x = titleView.getTextStartX() + Theme.getCurrentHolidayDrawableXOffset() + (rect.width() - (drawable.getIntrinsicWidth() + Theme.getCurrentHolidayDrawableXOffset())) / 2; - int y = titleView.getTextStartY() + Theme.getCurrentHolidayDrawableYOffset() + (int) Math.ceil((titleView.getTextHeight() - rect.height()) / 2.0f); + int y = titleView.getTextStartY() + Theme.getCurrentHolidayDrawableYOffset() + (int) Math.ceil((titleView.getTextHeight() - rect.height()) / 2.0f) + (int) (dp(8) * (1f - titlesContainer.getScaleY())); drawable.setBounds(x, y - drawable.getIntrinsicHeight(), x + drawable.getIntrinsicWidth(), y); - drawable.setAlpha((int) (255 * titleView.getAlpha())); + drawable.setAlpha((int) (255 * titlesContainer.getAlpha() * titleView.getAlpha())); drawable.draw(canvas); if (overlayTitleAnimationInProgress) { child.invalidate(); @@ -425,9 +426,9 @@ private void createTitleTextView(int i) { titleTextView[i].setTextColor(getThemedColor(Theme.key_actionBarDefaultTitle)); } titleTextView[i].setTypeface(AndroidUtilities.getTypeface("fonts/rmedium.ttf")); - titleTextView[i].setDrawablePadding(AndroidUtilities.dp(4)); - titleTextView[i].setPadding(0, AndroidUtilities.dp(8), 0, AndroidUtilities.dp(8)); - titleTextView[i].setRightDrawableTopPadding(-AndroidUtilities.dp(1)); + titleTextView[i].setDrawablePadding(dp(4)); + titleTextView[i].setPadding(0, dp(8), 0, dp(8)); + titleTextView[i].setRightDrawableTopPadding(-dp(1)); if (useContainerForTitles) { titlesContainer.addView(titleTextView[i], 0, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.LEFT | Gravity.TOP)); } else { @@ -1209,10 +1210,10 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { int textLeft; if (backButtonImageView != null && backButtonImageView.getVisibility() != GONE) { - backButtonImageView.measure(MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(54), MeasureSpec.EXACTLY), actionBarHeightSpec); - textLeft = AndroidUtilities.dp(AndroidUtilities.isTablet() ? 80 : 72); + backButtonImageView.measure(MeasureSpec.makeMeasureSpec(dp(54), MeasureSpec.EXACTLY), actionBarHeightSpec); + textLeft = dp(AndroidUtilities.isTablet() ? 80 : 72); } else { - textLeft = AndroidUtilities.dp(AndroidUtilities.isTablet() ? 26 : 18); + textLeft = dp(AndroidUtilities.isTablet() ? 26 : 18); } if (menu != null && menu.getVisibility() != GONE) { @@ -1222,12 +1223,12 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { menuWidth = MeasureSpec.makeMeasureSpec(width, MeasureSpec.AT_MOST); menu.measure(menuWidth, actionBarHeightSpec); int itemsWidth = menu.getItemsMeasuredWidth(true); - menuWidth = MeasureSpec.makeMeasureSpec(width - AndroidUtilities.dp(AndroidUtilities.isTablet() ? 74 : 66) + menu.getItemsMeasuredWidth(true), MeasureSpec.EXACTLY); + menuWidth = MeasureSpec.makeMeasureSpec(width - dp(AndroidUtilities.isTablet() ? 74 : 66) + menu.getItemsMeasuredWidth(true), MeasureSpec.EXACTLY); if (!isMenuOffsetSuppressed) { menu.translateXItems(-itemsWidth); } } else if (isSearchFieldVisible) { - menuWidth = MeasureSpec.makeMeasureSpec(width - AndroidUtilities.dp(AndroidUtilities.isTablet() ? 74 : 66), MeasureSpec.EXACTLY); + menuWidth = MeasureSpec.makeMeasureSpec(width - dp(AndroidUtilities.isTablet() ? 74 : 66), MeasureSpec.EXACTLY); if (!isMenuOffsetSuppressed) { menu.translateXItems(0); } @@ -1243,7 +1244,7 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { for (int i = 0; i < 2; i++) { if (titleTextView[0] != null && titleTextView[0].getVisibility() != GONE || subtitleTextView != null && subtitleTextView.getVisibility() != GONE) { - int availableWidth = width - (menu != null ? menu.getMeasuredWidth() : 0) - AndroidUtilities.dp(16) - textLeft - titleRightMargin; + int availableWidth = width - (menu != null ? menu.getMeasuredWidth() : 0) - dp(16) - textLeft - titleRightMargin; if (((fromBottom && i == 0) || (!fromBottom && i == 1)) && overlayTitleAnimation && titleAnimationRunning) { titleTextView[i].setTextSize(!AndroidUtilities.isTablet() && getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE ? 18 : 20); @@ -1270,29 +1271,29 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { } if (titleTextView[i] != null && titleTextView[i].getVisibility() != GONE) { - titleTextView[i].measure(MeasureSpec.makeMeasureSpec(availableWidth, MeasureSpec.AT_MOST), MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(24) + titleTextView[i].getPaddingTop() + titleTextView[i].getPaddingBottom(), MeasureSpec.AT_MOST)); + titleTextView[i].measure(MeasureSpec.makeMeasureSpec(availableWidth, MeasureSpec.AT_MOST), MeasureSpec.makeMeasureSpec(dp(24) + titleTextView[i].getPaddingTop() + titleTextView[i].getPaddingBottom(), MeasureSpec.AT_MOST)); if (centerScale) { CharSequence text = titleTextView[i].getText(); titleTextView[i].setPivotX(titleTextView[i].getTextPaint().measureText(text, 0, text.length()) / 2f); - titleTextView[i].setPivotY((AndroidUtilities.dp(24) >> 1)); + titleTextView[i].setPivotY((dp(24) >> 1)); } else { titleTextView[i].setPivotX(0); titleTextView[i].setPivotY(0); } } if (subtitleTextView != null && subtitleTextView.getVisibility() != GONE) { - subtitleTextView.measure(MeasureSpec.makeMeasureSpec(availableWidth, MeasureSpec.AT_MOST), MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(20), MeasureSpec.AT_MOST)); + subtitleTextView.measure(MeasureSpec.makeMeasureSpec(availableWidth, MeasureSpec.AT_MOST), MeasureSpec.makeMeasureSpec(dp(20), MeasureSpec.AT_MOST)); } if (additionalSubtitleTextView != null && additionalSubtitleTextView.getVisibility() != GONE) { - additionalSubtitleTextView.measure(MeasureSpec.makeMeasureSpec(availableWidth, MeasureSpec.AT_MOST), MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(20), MeasureSpec.AT_MOST)); + additionalSubtitleTextView.measure(MeasureSpec.makeMeasureSpec(availableWidth, MeasureSpec.AT_MOST), MeasureSpec.makeMeasureSpec(dp(20), MeasureSpec.AT_MOST)); } } } if (avatarSearchImageView != null) { avatarSearchImageView.measure( - MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(42), MeasureSpec.EXACTLY), - MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(42), MeasureSpec.EXACTLY) + MeasureSpec.makeMeasureSpec(dp(42), MeasureSpec.EXACTLY), + MeasureSpec.makeMeasureSpec(dp(42), MeasureSpec.EXACTLY) ); } @@ -1317,13 +1318,13 @@ protected void onLayout(boolean changed, int left, int top, int right, int botto int textLeft; if (backButtonImageView != null && backButtonImageView.getVisibility() != GONE) { backButtonImageView.layout(0, additionalTop, backButtonImageView.getMeasuredWidth(), additionalTop + backButtonImageView.getMeasuredHeight()); - textLeft = AndroidUtilities.dp(AndroidUtilities.isTablet() ? 80 : 72); + textLeft = dp(AndroidUtilities.isTablet() ? 80 : 72); } else { - textLeft = AndroidUtilities.dp(AndroidUtilities.isTablet() ? 26 : 18); + textLeft = dp(AndroidUtilities.isTablet() ? 26 : 18); } if (menu != null && menu.getVisibility() != GONE) { - int menuLeft = menu.searchFieldVisible() ? AndroidUtilities.dp(AndroidUtilities.isTablet() ? 74 : 66) : (right - left) - menu.getMeasuredWidth(); + int menuLeft = menu.searchFieldVisible() ? dp(AndroidUtilities.isTablet() ? 74 : 66) : (right - left) - menu.getMeasuredWidth(); menu.layout(menuLeft, additionalTop, menuLeft + menu.getMeasuredWidth(), additionalTop + menu.getMeasuredHeight()); } @@ -1334,7 +1335,7 @@ protected void onLayout(boolean changed, int left, int top, int right, int botto textTop = (getCurrentActionBarHeight() - titleTextView[i].getTextHeight()) / 2; } else { if ((subtitleTextView != null && subtitleTextView.getVisibility() != GONE)) { - textTop = (getCurrentActionBarHeight() / 2 - titleTextView[i].getTextHeight()) / 2 + AndroidUtilities.dp(!AndroidUtilities.isTablet() && getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE ? 2 : 3); + textTop = (getCurrentActionBarHeight() / 2 - titleTextView[i].getTextHeight()) / 2 + dp(!AndroidUtilities.isTablet() && getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE ? 2 : 3); } else { textTop = (getCurrentActionBarHeight() - titleTextView[i].getTextHeight()) / 2; } @@ -1343,20 +1344,20 @@ protected void onLayout(boolean changed, int left, int top, int right, int botto } } if (subtitleTextView != null && subtitleTextView.getVisibility() != GONE) { - int textTop = getCurrentActionBarHeight() / 2 + (getCurrentActionBarHeight() / 2 - subtitleTextView.getTextHeight()) / 2 - AndroidUtilities.dp(!AndroidUtilities.isTablet() && getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE ? 1 : 1); + int textTop = getCurrentActionBarHeight() / 2 + (getCurrentActionBarHeight() / 2 - subtitleTextView.getTextHeight()) / 2 - dp(!AndroidUtilities.isTablet() && getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE ? 1 : 1); subtitleTextView.layout(textLeft, additionalTop + textTop, textLeft + subtitleTextView.getMeasuredWidth(), additionalTop + textTop + subtitleTextView.getTextHeight()); } if (additionalSubtitleTextView != null && additionalSubtitleTextView.getVisibility() != GONE) { - int textTop = getCurrentActionBarHeight() / 2 + (getCurrentActionBarHeight() / 2 - additionalSubtitleTextView.getTextHeight()) / 2 - AndroidUtilities.dp(!AndroidUtilities.isTablet() && getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE ? 1 : 1); + int textTop = getCurrentActionBarHeight() / 2 + (getCurrentActionBarHeight() / 2 - additionalSubtitleTextView.getTextHeight()) / 2 - dp(!AndroidUtilities.isTablet() && getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE ? 1 : 1); additionalSubtitleTextView.layout(textLeft, additionalTop + textTop, textLeft + additionalSubtitleTextView.getMeasuredWidth(), additionalTop + textTop + additionalSubtitleTextView.getTextHeight()); } if (avatarSearchImageView != null) { avatarSearchImageView.layout( - AndroidUtilities.dp(56 + 8), + dp(56 + 8), additionalTop + (getCurrentActionBarHeight() - avatarSearchImageView.getMeasuredHeight()) / 2, - AndroidUtilities.dp(56 + 8) + avatarSearchImageView.getMeasuredWidth(), + dp(56 + 8) + avatarSearchImageView.getMeasuredWidth(), additionalTop + (getCurrentActionBarHeight() + avatarSearchImageView.getMeasuredHeight()) / 2 ); } @@ -1476,7 +1477,7 @@ public void setTitleOverlayText(String title, int titleId, Runnable action) { invalidate(); } titleTextView[0].setText(textToSet); - titleTextView[0].setDrawablePadding(AndroidUtilities.dp(4)); + titleTextView[0].setDrawablePadding(dp(4)); titleTextView[0].setRightDrawable(rightDrawableToSet); titleTextView[0].setRightDrawableOnClick(rightDrawableOnClickListener); if (rightDrawableToSet instanceof AnimatedEmojiDrawable.SwapAnimatedEmojiDrawable) { @@ -1496,7 +1497,7 @@ public void setTitleOverlayText(String title, int titleId, Runnable action) { createTitleTextView(1); } titleTextView[1].setText(textToSet); - titleTextView[1].setDrawablePadding(AndroidUtilities.dp(4)); + titleTextView[1].setDrawablePadding(dp(4)); titleTextView[1].setRightDrawable(rightDrawableToSet); titleTextView[1].setRightDrawableOnClick(rightDrawableOnClickListener); if (rightDrawableToSet instanceof AnimatedEmojiDrawable.SwapAnimatedEmojiDrawable) { @@ -1510,7 +1511,7 @@ public void setTitleOverlayText(String title, int titleId, Runnable action) { titleTextView[1] = titleTextView[0]; titleTextView[0] = tmp; titleTextView[0].setAlpha(0); - titleTextView[0].setTranslationY(-AndroidUtilities.dp(20)); + titleTextView[0].setTranslationY(-dp(20)); titleTextView[0].animate() .alpha(1f) .translationY(0) @@ -1518,7 +1519,7 @@ public void setTitleOverlayText(String title, int titleId, Runnable action) { ViewPropertyAnimator animator = titleTextView[1].animate() .alpha(0); if (subtitleTextView == null) { - animator.translationY(AndroidUtilities.dp(20)); + animator.translationY(dp(20)); } else { animator.scaleY(0.7f).scaleX(0.7f); } @@ -1627,11 +1628,11 @@ public boolean onTouchEvent(MotionEvent event) { public static int getCurrentActionBarHeight() { if (AndroidUtilities.isTablet()) { - return AndroidUtilities.dp(64); + return dp(64); } else if (AndroidUtilities.displaySize.x > AndroidUtilities.displaySize.y) { - return AndroidUtilities.dp(48); + return dp(48); } else { - return AndroidUtilities.dp(56); + return dp(56); } } @@ -1665,7 +1666,7 @@ public void setTitleAnimated(CharSequence title, boolean fromBottom, long durati this.fromBottom = fromBottom; titleTextView[0].setAlpha(0); if (!crossfade) { - titleTextView[0].setTranslationY(fromBottom ? AndroidUtilities.dp(20) : -AndroidUtilities.dp(20)); + titleTextView[0].setTranslationY(fromBottom ? dp(20) : -dp(20)); } ViewPropertyAnimator a1 = titleTextView[0].animate().alpha(1f).translationY(0).setDuration(duration); if (interpolator != null) { @@ -1676,7 +1677,7 @@ public void setTitleAnimated(CharSequence title, boolean fromBottom, long durati titleAnimationRunning = true; ViewPropertyAnimator a = titleTextView[1].animate().alpha(0); if (!crossfade) { - a.translationY(fromBottom ? -AndroidUtilities.dp(20) : AndroidUtilities.dp(20)); + a.translationY(fromBottom ? -dp(20) : dp(20)); } if (interpolator != null) { a.setInterpolator(interpolator); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/ActionBarLayout.java b/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/ActionBarLayout.java index 304910e892f..9e9019ad6f9 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/ActionBarLayout.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/ActionBarLayout.java @@ -396,6 +396,7 @@ public void processMenuButtonsTouch(MotionEvent event) { private float themeAnimationValue; private boolean animateThemeAfterAnimation; private Theme.ThemeInfo animateSetThemeAfterAnimation; + private boolean animateSetThemeAfterAnimationApply; private boolean animateSetThemeNightAfterAnimation; private int animateSetThemeAccentIdAfterAnimation; private boolean rebuildAfterAnimation; @@ -1256,10 +1257,10 @@ public boolean presentFragment(NavigationParams params) { } BaseFragment lastFragment = getLastFragment(); Dialog dialog = lastFragment != null ? lastFragment.getVisibleDialog() : null; - if (dialog == null && LaunchActivity.instance != null && LaunchActivity.instance.visibleDialog != null) { - dialog = LaunchActivity.instance.visibleDialog; + if (dialog == null && LaunchActivity.instance != null && LaunchActivity.instance.getVisibleDialog() != null) { + dialog = LaunchActivity.instance.getVisibleDialog(); } - if (shouldOpenFragmentOverlay(dialog)) { + if (lastFragment != null && shouldOpenFragmentOverlay(dialog)) { BaseFragment.BottomSheetParams bottomSheetParams = new BaseFragment.BottomSheetParams(); bottomSheetParams.transitionFromLeft = true; bottomSheetParams.allowNestedScroll = false; @@ -1564,7 +1565,7 @@ public void run() { } private boolean shouldOpenFragmentOverlay(Dialog visibleDialog) { - return visibleDialog instanceof ChatAttachAlert || visibleDialog instanceof BotWebViewSheet; + return (visibleDialog != null && visibleDialog.isShowing()) && (visibleDialog instanceof ChatAttachAlert || visibleDialog instanceof BotWebViewSheet); } @Override @@ -1726,6 +1727,10 @@ public void onAnimationEnd(Animator animation) { fragment.setInPreviewMode(false); fragment.setInMenuMode(false); + + try { + AndroidUtilities.setLightStatusBar(parentActivity.getWindow(), Theme.getColor(Theme.key_actionBarDefault) == Color.WHITE || (fragment.hasForceLightStatusBar() && !Theme.getCurrentTheme().isDark()), fragment.hasForceLightStatusBar()); + } catch (Exception ignore) {} } @Override @@ -2124,6 +2129,7 @@ public void animateThemedValues(ThemeAnimationSettings settings, Runnable onDone animateSetThemeAfterAnimation = settings.theme; animateSetThemeNightAfterAnimation = settings.nightTheme; animateSetThemeAccentIdAfterAnimation = settings.accentId; + animateSetThemeAfterAnimationApply = settings.applyTrulyTheme; if (onDone != null) { onDone.run(); } @@ -2259,7 +2265,7 @@ public void onAnimationCancel(Animator animation) { onDone.run(); } }; - if (fragmentCount >= 1 && settings.applyTheme) { + if (fragmentCount >= 1 && settings.applyTheme && settings.applyTrulyTheme) { if (settings.accentId != -1 && settings.theme != null) { settings.theme.setCurrentAccentId(settings.accentId); Theme.saveThemeAccents(settings.theme, true, false, true, false); @@ -2363,7 +2369,11 @@ private void checkNeedRebuild() { rebuildAllFragmentViews(rebuildLastAfterAnimation, showLastAfterAnimation); rebuildAfterAnimation = false; } else if (animateThemeAfterAnimation) { - animateThemedValues(animateSetThemeAfterAnimation, animateSetThemeAccentIdAfterAnimation, animateSetThemeNightAfterAnimation, false); + ThemeAnimationSettings settings = new ThemeAnimationSettings(animateSetThemeAfterAnimation, animateSetThemeAccentIdAfterAnimation, animateSetThemeNightAfterAnimation, false); + if (!animateSetThemeAfterAnimationApply) { + settings.applyTheme = settings.applyTrulyTheme = animateSetThemeAfterAnimationApply; + } + animateThemedValues(settings, null); animateSetThemeAfterAnimation = null; animateThemeAfterAnimation = false; } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/AlertDialog.java b/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/AlertDialog.java index 8f3d3b14147..5f727c5ef43 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/AlertDialog.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/AlertDialog.java @@ -83,6 +83,7 @@ public class AlertDialog extends Dialog implements Drawable.Callback, Notificati private View customView; private View bottomView; + private View aboveMessageView; private int customViewHeight = LayoutHelper.WRAP_CONTENT; private TextView titleTextView; private TextView secondTitleTextView; @@ -766,9 +767,12 @@ protected boolean drawChild(Canvas canvas, View child, long drawingTime) { progressView.setProgressColor(getThemedColor(Theme.key_dialog_inlineProgress)); progressViewContainer.addView(progressView, LayoutHelper.createFrame(86, 86, Gravity.CENTER)); } else { + if (aboveMessageView != null) { + scrollContainer.addView(aboveMessageView, LayoutHelper.createLinear(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, 22, 4, 22, 12)); + } scrollContainer.addView(messageTextView, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, (topAnimationIsNew ? Gravity.CENTER_HORIZONTAL : LocaleController.isRTL ? Gravity.RIGHT : Gravity.LEFT) | Gravity.TOP, 24, 0, 24, customView != null || items != null ? customViewOffset : 0)); if (bottomView != null) { - scrollContainer.addView(bottomView, LayoutHelper.createLinear(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, 24, -24, 24, 0)); + scrollContainer.addView(bottomView, LayoutHelper.createLinear(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, 22, 12, 22, 0)); } } if (!TextUtils.isEmpty(message)) { @@ -1170,6 +1174,21 @@ public void setTextColor(int color) { } } + public void setTextSize(int titleSizeDp, int messageSizeDp) { + if (titleTextView != null) { + titleTextView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, titleSizeDp); + } + if (messageTextView != null) { + messageTextView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, messageSizeDp); + } + } + + public void setMessageLineSpacing(float spaceDp) { + if (messageTextView != null) { + messageTextView.setLineSpacing(AndroidUtilities.dp(spaceDp), 1.0f); + } + } + private void showCancelAlert() { if (!canCacnel || cancelDialog != null) { return; @@ -1523,6 +1542,11 @@ public Builder setView(View view, int height) { return this; } + public Builder aboveMessageView(View view) { + alertDialog.aboveMessageView = view; + return this; + } + public Builder addBottomView(View view) { alertDialog.bottomView = view; return this; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/BottomSheet.java b/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/BottomSheet.java index 67f034d895e..9e3105c1f5f 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/BottomSheet.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/BottomSheet.java @@ -109,6 +109,8 @@ public class BottomSheet extends Dialog { protected boolean fullWidth; protected boolean isFullscreen; private boolean fullHeight; + private int cellType; + private Integer selectedPos; protected ColorDrawable backDrawable = new ColorDrawable(0xff000000) { @Override public void setAlpha(int alpha) { @@ -897,7 +899,9 @@ public BottomSheetCell(Context context, int type, Theme.ResourcesProvider resour this.resourcesProvider = resourcesProvider; currentType = type; - setBackgroundDrawable(Theme.getSelectorDrawable(false, resourcesProvider)); + if (type != Builder.CELL_TYPE_CALL) { + setBackgroundDrawable(Theme.getSelectorDrawable(false, resourcesProvider)); + } //setPadding(AndroidUtilities.dp(16), 0, AndroidUtilities.dp(16), 0); imageView = new ImageView(context); @@ -910,7 +914,7 @@ public BottomSheetCell(Context context, int type, Theme.ResourcesProvider resour textView.setSingleLine(true); textView.setGravity(Gravity.CENTER_HORIZONTAL); textView.setEllipsize(TextUtils.TruncateAt.END); - if (type == 0) { + if (type == 0 || type == Builder.CELL_TYPE_CALL) { textView.setTextColor(getThemedColor(Theme.key_dialogTextBlack)); textView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 16); addView(textView, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, (LocaleController.isRTL ? Gravity.RIGHT : Gravity.LEFT) | Gravity.CENTER_VERTICAL)); @@ -1234,7 +1238,7 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { if (items[a] == null) { continue; } - BottomSheetCell cell = new BottomSheetCell(getContext(), 0, resourcesProvider); + BottomSheetCell cell = new BottomSheetCell(getContext(), cellType, resourcesProvider); cell.setTextAndIcon(items[a], itemIcons != null ? itemIcons[a] : 0, null, bigTitle); containerView.addView(cell, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 48, Gravity.LEFT | Gravity.TOP, 0, topOffset, 0, 0)); topOffset += 48; @@ -1617,7 +1621,7 @@ public void dismissWithButtonClick(final int item) { ObjectAnimator.ofFloat(containerView, View.TRANSLATION_Y, getContainerViewHeight() + keyboardHeight + AndroidUtilities.dp(10) + (scrollNavBar ? getBottomInset() : 0)), ObjectAnimator.ofInt(backDrawable, AnimationProperties.COLOR_DRAWABLE_ALPHA, 0) ); - currentSheetAnimation.setDuration(180); + currentSheetAnimation.setDuration(cellType == Builder.CELL_TYPE_CALL ? 330 : 180); currentSheetAnimation.setInterpolator(CubicBezierInterpolator.EASE_OUT); currentSheetAnimation.addListener(new AnimatorListenerAdapter() { @Override @@ -1652,6 +1656,27 @@ public void onAnimationCancel(Animator animation) { }); NotificationCenter.getGlobalInstance().postNotificationName(NotificationCenter.stopAllHeavyOperations, 512); currentSheetAnimation.start(); + + if (cellType == Builder.CELL_TYPE_CALL && selectedPos != null && Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { + int color1 = getItemViews().get(selectedPos).getTextView().getCurrentTextColor(); + int color2 = getItemViews().get(item).getTextView().getCurrentTextColor(); + ValueAnimator animator = ValueAnimator.ofArgb(color1, color2); + animator.addUpdateListener(a -> { + int color = (int) a.getAnimatedValue(); + setItemColor(selectedPos, color, color); + }); + animator.setDuration(130); + animator.setInterpolator(CubicBezierInterpolator.DEFAULT); + animator.start(); + ValueAnimator animator2 = ValueAnimator.ofArgb(color2, color1); + animator2.addUpdateListener(a -> { + int color = (int) a.getAnimatedValue(); + setItemColor(item, color, color); + }); + animator2.setDuration(130); + animator2.setInterpolator(CubicBezierInterpolator.DEFAULT); + animator2.start(); + } } @Override @@ -1796,6 +1821,8 @@ protected boolean onCustomOpenAnimation() { public static class Builder { + public static int CELL_TYPE_CALL = 4; + private BottomSheet bottomSheet; public Builder(Context context) { @@ -1863,6 +1890,16 @@ public Builder setTitle(CharSequence title, boolean big) { return this; } + public Builder selectedPos(Integer pos) { + bottomSheet.selectedPos = pos; + return this; + } + + public Builder setCellType(int cellType) { + bottomSheet.cellType = cellType; + return this; + } + public Builder setTitleMultipleLines(boolean allowMultipleLines) { bottomSheet.multipleLinesTitle = allowMultipleLines; return this; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/EmojiThemes.java b/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/EmojiThemes.java index 33922de823e..bc9282cb743 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/EmojiThemes.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/EmojiThemes.java @@ -254,10 +254,12 @@ public SparseIntArray getPreviewColors(int currentAccount, int index) { } else { baseTheme = Theme.getTheme("Blue"); } - themeInfo = new Theme.ThemeInfo(baseTheme); - accent = themeInfo.createNewAccent(tlTheme, currentAccount, true, settingsIndex); - if (accent != null) { - themeInfo.setCurrentAccentId(accent.id); + if (baseTheme != null) { + themeInfo = new Theme.ThemeInfo(baseTheme); + accent = themeInfo.createNewAccent(tlTheme, currentAccount, true, settingsIndex); + if (accent != null) { + themeInfo.setCurrentAccentId(accent.id); + } } } else { if (themeInfo.themeAccentsMap != null) { @@ -265,6 +267,10 @@ public SparseIntArray getPreviewColors(int currentAccount, int index) { } } + if (themeInfo == null) { + return currentColors; + } + SparseIntArray currentColorsNoAccent; String[] wallpaperLink = new String[1]; if (themeInfo.pathToFile != null) { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/FloatingToolbar.java b/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/FloatingToolbar.java index df3f19d93bb..2722ae095c7 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/FloatingToolbar.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/FloatingToolbar.java @@ -1007,7 +1007,7 @@ private void layoutOverflowPanelItems(List menuItems) { }); } final int size = menuItems.size(); - final boolean premiumLocked = MessagesController.getInstance(UserConfig.selectedAccount).premiumLocked; + final boolean premiumLocked = MessagesController.getInstance(UserConfig.selectedAccount).premiumFeaturesBlocked(); for (int i = 0; i < size; i++) { final MenuItem menuItem = menuItems.get(i); final boolean show; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/INavigationLayout.java b/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/INavigationLayout.java index 497bcef107c..8fe20bab782 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/INavigationLayout.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/INavigationLayout.java @@ -368,6 +368,7 @@ class ThemeAnimationSettings { public final boolean instant; public boolean onlyTopFragment; public boolean applyTheme = true; + public boolean applyTrulyTheme = true; public Runnable afterStartDescriptionsAddedRunnable; public Runnable beforeAnimationRunnable; public Runnable afterAnimationRunnable; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/SimpleTextView.java b/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/SimpleTextView.java index 38a52e7fa93..c796ea69e88 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/SimpleTextView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/SimpleTextView.java @@ -61,6 +61,7 @@ public class SimpleTextView extends View implements Drawable.Callback { private SpannableStringBuilder spannableStringBuilder; private Drawable leftDrawable; private Drawable rightDrawable; + private Drawable rightDrawable2; private Drawable replacedDrawable; private String replacedText; private int replacingDrawableTextIndex; @@ -273,6 +274,10 @@ public int getSideDrawablesSize() { int dw = (int) (rightDrawable.getIntrinsicWidth() * rightDrawableScale); size += dw + drawablePadding; } + if (rightDrawable2 != null) { + int dw = (int) (rightDrawable2.getIntrinsicWidth() * rightDrawableScale); + size += dw + drawablePadding; + } return size; } @@ -342,6 +347,11 @@ protected boolean createLayout(int width) { width -= rightDrawableWidth; width -= drawablePadding; } + if (rightDrawable2 != null && !rightDrawableOutside) { + rightDrawableWidth = (int) (rightDrawable2.getIntrinsicWidth() * rightDrawableScale); + width -= rightDrawableWidth; + width -= drawablePadding; + } if (replacedText != null && replacedDrawable != null) { replacingDrawableTextIndex = text.toString().indexOf(replacedText); if (replacingDrawableTextIndex >= 0) { @@ -450,7 +460,7 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { scrollingOffset = 0; currentScrollDelay = SCROLL_DELAY_MS; } - createLayout(width - getPaddingLeft() - getPaddingRight() - minusWidth - (rightDrawableOutside && rightDrawable != null ? rightDrawable.getIntrinsicWidth() + drawablePadding : 0)); + createLayout(width - getPaddingLeft() - getPaddingRight() - minusWidth - (rightDrawableOutside && rightDrawable != null ? rightDrawable.getIntrinsicWidth() + drawablePadding : 0) - (rightDrawableOutside && rightDrawable2 != null ? rightDrawable2.getIntrinsicWidth() + drawablePadding : 0)); int finalHeight; if (MeasureSpec.getMode(heightMeasureSpec) == MeasureSpec.EXACTLY) { @@ -460,7 +470,7 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { } if (widthWrapContent) { // textWidth = (int) Math.ceil(layout.getLineWidth(0)); - width = Math.min(width, getPaddingLeft() + textWidth + getPaddingRight() + minusWidth + (rightDrawableOutside && rightDrawable != null ? rightDrawable.getIntrinsicWidth() + drawablePadding : 0)); + width = Math.min(width, getPaddingLeft() + textWidth + getPaddingRight() + minusWidth + (rightDrawableOutside && rightDrawable != null ? rightDrawable.getIntrinsicWidth() + drawablePadding : 0) + (rightDrawableOutside && rightDrawable2 != null ? rightDrawable2.getIntrinsicWidth() + drawablePadding : 0)); } setMeasuredDimension(width, finalHeight); @@ -549,7 +559,7 @@ public void setLeftDrawable(Drawable drawable) { @Override protected boolean verifyDrawable(@NonNull Drawable who) { - return who == rightDrawable || who == leftDrawable || super.verifyDrawable(who); + return who == rightDrawable || who == rightDrawable2 || who == leftDrawable || super.verifyDrawable(who); } public void replaceTextWithDrawable(Drawable drawable, String replacedText) { @@ -599,6 +609,26 @@ public void setRightDrawable(Drawable drawable) { } } + public void setRightDrawable2(Drawable drawable) { + if (rightDrawable2 == drawable) { + return; + } + if (rightDrawable2 != null) { + rightDrawable2.setCallback(null); + } + rightDrawable2 = drawable; + if (drawable != null) { + drawable.setCallback(this); + } + if (!recreateLayoutMaybe()) { + invalidate(); + } + } + + public Drawable getRightDrawable2() { + return rightDrawable2; + } + public void setRightDrawableScale(float scale) { rightDrawableScale = scale; } @@ -717,6 +747,11 @@ public void setRightPadding(int padding) { width -= rightDrawableWidth; width -= drawablePadding; } + if (rightDrawable2 != null && !rightDrawableOutside) { + rightDrawableWidth = (int) (rightDrawable2.getIntrinsicWidth() * rightDrawableScale); + width -= rightDrawableWidth; + width -= drawablePadding; + } if (replacedText != null && replacedDrawable != null) { if ((replacingDrawableTextIndex = text.toString().indexOf(replacedText)) < 0) { width -= replacedDrawable.getIntrinsicWidth(); @@ -808,6 +843,27 @@ protected void onDraw(Canvas canvas) { rightDrawable.draw(canvas); totalWidth += drawablePadding + dw; } + if (rightDrawable2 != null && !rightDrawableHidden && rightDrawableScale > 0 && !rightDrawableOutside) { + int x = textOffsetX + textWidth + drawablePadding + (int) -scrollingOffset; + if (rightDrawable != null) { + x += (int) (rightDrawable.getIntrinsicWidth() * rightDrawableScale) + drawablePadding; + } + if ((gravity & Gravity.HORIZONTAL_GRAVITY_MASK) == Gravity.CENTER_HORIZONTAL || + (gravity & Gravity.HORIZONTAL_GRAVITY_MASK) == Gravity.RIGHT) { + x += offsetX; + } + int dw = (int) (rightDrawable2.getIntrinsicWidth() * rightDrawableScale); + int dh = (int) (rightDrawable2.getIntrinsicHeight() * rightDrawableScale); + int y; + if ((gravity & Gravity.VERTICAL_GRAVITY_MASK) == Gravity.CENTER_VERTICAL) { + y = (getMeasuredHeight() - dh) / 2 + rightDrawableTopPadding; + } else { + y = getPaddingTop() + (textHeight - dh) / 2 + rightDrawableTopPadding; + } + rightDrawable2.setBounds(x, y, x + dw, y + dh); + rightDrawable2.draw(canvas); + totalWidth += drawablePadding + dw; + } int nextScrollX = totalWidth + AndroidUtilities.dp(DIST_BETWEEN_SCROLLING_TEXT); if (scrollingOffset != 0) { @@ -835,6 +891,22 @@ protected void onDraw(Canvas canvas) { rightDrawable.setBounds(x, y, x + dw, y + dh); rightDrawable.draw(canvas); } + if (rightDrawable2 != null && !rightDrawableOutside) { + int dw = (int) (rightDrawable2.getIntrinsicWidth() * rightDrawableScale); + int dh = (int) (rightDrawable2.getIntrinsicHeight() * rightDrawableScale); + int x = textOffsetX + textWidth + drawablePadding + (int) -scrollingOffset + nextScrollX; + if (rightDrawable != null) { + x += (int) (rightDrawable.getIntrinsicWidth() * rightDrawableScale) + drawablePadding; + } + int y; + if ((gravity & Gravity.VERTICAL_GRAVITY_MASK) == Gravity.CENTER_VERTICAL) { + y = (getMeasuredHeight() - dh) / 2 + rightDrawableTopPadding; + } else { + y = getPaddingTop() + (textHeight - dh) / 2 + rightDrawableTopPadding; + } + rightDrawable2.setBounds(x, y, x + dw, y + dh); + rightDrawable2.draw(canvas); + } } if (layout != null) { @@ -939,6 +1011,25 @@ protected void onDraw(Canvas canvas) { rightDrawableY = y + (dh >> 1); rightDrawable.draw(canvas); } + if (rightDrawable2 != null && rightDrawableOutside) { + int x = Math.min( + textOffsetX + textWidth + drawablePadding + (scrollingOffset == 0 ? -nextScrollX : (int) -scrollingOffset) + nextScrollX, + getMaxTextWidth() - paddingRight + drawablePadding + ); + if (rightDrawable != null) { + x += (int) (rightDrawable.getIntrinsicWidth() * rightDrawableScale) + drawablePadding; + } + int dw = (int) (rightDrawable2.getIntrinsicWidth() * rightDrawableScale); + int dh = (int) (rightDrawable2.getIntrinsicHeight() * rightDrawableScale); + int y; + if ((gravity & Gravity.VERTICAL_GRAVITY_MASK) == Gravity.CENTER_VERTICAL) { + y = (getMeasuredHeight() - dh) / 2 + rightDrawableTopPadding; + } else { + y = getPaddingTop() + (textHeight - dh) / 2 + rightDrawableTopPadding; + } + rightDrawable2.setBounds(x, y, x + dw, y + dh); + rightDrawable2.draw(canvas); + } } public int getRightDrawableX() { @@ -950,7 +1041,7 @@ public int getRightDrawableY() { } private int getMaxTextWidth() { - return getMeasuredWidth() - (rightDrawableOutside && rightDrawable != null ? rightDrawable.getIntrinsicWidth() + drawablePadding : 0); + return getMeasuredWidth() - (rightDrawableOutside && rightDrawable != null ? rightDrawable.getIntrinsicWidth() + drawablePadding : 0) - (rightDrawableOutside && rightDrawable2 != null ? rightDrawable2.getIntrinsicWidth() + drawablePadding : 0); } private void drawLayout(Canvas canvas) { @@ -1035,6 +1126,8 @@ public void invalidateDrawable(Drawable who) { invalidate(leftDrawable.getBounds()); } else if (who == rightDrawable) { invalidate(rightDrawable.getBounds()); + } else if (who == rightDrawable2) { + invalidate(rightDrawable2.getBounds()); } else if (who == replacedDrawable) { invalidate(replacedDrawable.getBounds()); } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/Theme.java b/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/Theme.java index a007d091df5..7c88f7c8bbd 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/Theme.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/Theme.java @@ -958,7 +958,7 @@ private void generatePath(Path path, Rect bounds, int padding, int rad, int smal } if (currentType == TYPE_MEDIA) { if (customPaint || drawFullBottom) { - int radToUse = isBottomNear ? nearRad : rad; + int radToUse = isBottomNear || botButtonsBottom ? nearRad : rad; path.lineTo(bounds.left + padding, bounds.bottom - padding - radToUse); rect.set(bounds.left + padding, bounds.bottom - padding - radToUse * 2, bounds.left + padding + radToUse * 2, bounds.bottom - padding); @@ -1749,7 +1749,6 @@ public boolean fillAccentColors(SparseIntArray currentColorsNoAccent, SparseIntA Math.max(0, Color.blue(submenuBackground) - 10) )); - currentColors.put(key_chat_inCodeBackground, codeBackground(inBubble, isDarkTheme)); if (isDarkTheme && currentColors.get(key_chat_outBubbleGradient1) != 0) { int outBubbleAverage = averageColor(currentColors, key_chat_outBubbleGradient1, key_chat_outBubbleGradient2, key_chat_outBubbleGradient3); Color.colorToHSV(outBubbleAverage, tempHSV); @@ -4126,7 +4125,6 @@ public void run() { public static final int key_stories_circle_closeFriends1 = colorsCount++; public static final int key_stories_circle_closeFriends2 = colorsCount++; - public static final int key_code_background = colorsCount++; public static final int key_chat_inCodeBackground = colorsCount++; public static final int key_chat_outCodeBackground = colorsCount++; public static final int key_code_keyword = colorsCount++; @@ -4411,6 +4409,7 @@ public void run() { themeAccentExclusionKeys.add(key_statisticChartLine_lightgreen); themeAccentExclusionKeys.add(key_statisticChartLine_orange); themeAccentExclusionKeys.add(key_statisticChartLine_indigo); + themeAccentExclusionKeys.add(key_chat_inCodeBackground); themeAccentExclusionKeys.add(key_voipgroup_checkMenu); themeAccentExclusionKeys.add(key_voipgroup_muteButton); @@ -5757,7 +5756,7 @@ private static Drawable createRect(Drawable background, int rippleColor, float . ripple = new ShapeDrawable(new RectShape()); ((ShapeDrawable) ripple).getPaint().setColor(rippleColor); } - Drawable pressed = new LayerDrawable(new Drawable[] { background, ripple }); + Drawable pressed = background == null ? ripple : new LayerDrawable(new Drawable[] { background, ripple }); stateListDrawable.addState(new int[]{android.R.attr.state_pressed}, pressed); stateListDrawable.addState(new int[]{android.R.attr.state_selected}, pressed); stateListDrawable.addState(StateSet.WILD_CARD, background); @@ -5791,7 +5790,7 @@ private static Drawable createCircle(Drawable background, int rippleColor, float } else { StateListDrawable stateListDrawable = new StateListDrawable(); Drawable ripple = new CircleDrawable(radius, rippleColor); - Drawable pressed = new LayerDrawable(new Drawable[] { background, ripple }); + Drawable pressed = background == null ? ripple : new LayerDrawable(new Drawable[] { background, ripple }); stateListDrawable.addState(new int[]{android.R.attr.state_pressed}, pressed); stateListDrawable.addState(new int[]{android.R.attr.state_selected}, pressed); stateListDrawable.addState(StateSet.WILD_CARD, background); @@ -9868,9 +9867,13 @@ public static BackgroundDrawableSettings createBackgroundDrawable( MotionBackgroundDrawable motionBackgroundDrawable = new MotionBackgroundDrawable(backgroundColor, gradientToColor1, gradientToColor2, gradientToColor3, false); Bitmap patternBitmap = null; - if (wallpaperFile != null && wallpaperDocument != null) { - File f = FileLoader.getInstance(UserConfig.selectedAccount).getPathToAttach(wallpaperDocument, true); - patternBitmap = SvgHelper.getBitmap(f, AndroidUtilities.dp(360), AndroidUtilities.dp(640), false); + if (wallpaperFile != null && !isCustomTheme()) { + if (wallpaperDocument != null) { + File f = FileLoader.getInstance(UserConfig.selectedAccount).getPathToAttach(wallpaperDocument, true); + patternBitmap = SvgHelper.getBitmap(f, AndroidUtilities.dp(360), AndroidUtilities.dp(640), false); + } else { + patternBitmap = SvgHelper.getBitmap(R.raw.default_pattern, AndroidUtilities.dp(360), AndroidUtilities.dp(640), Color.WHITE); + } if (patternBitmap != null) { FileOutputStream stream = null; try { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/ThemeColors.java b/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/ThemeColors.java index da994f51417..d3a508cc582 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/ThemeColors.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/ThemeColors.java @@ -771,8 +771,7 @@ public static int[] createDefaultColors() { defaultColors[key_stories_circle_closeFriends1] = 0xFFC9EB38; defaultColors[key_stories_circle_closeFriends2] = 0xFF09C167; - defaultColors[key_code_background] = 0x20000000; - defaultColors[key_chat_inCodeBackground] = 0x08484848; + defaultColors[key_chat_inCodeBackground] = 0xff6F889E; defaultColors[key_chat_outCodeBackground] = 0x123c7503; defaultColors[key_code_keyword] = 0xFFE05356; defaultColors[key_code_operator] = 0xFF4DBBFF; @@ -1513,7 +1512,6 @@ public static SparseArray createColorKeysMap() { colorKeysMap.put(key_stories_circle_dialog2, "stories_circle_dialog2"); colorKeysMap.put(key_stories_circle_closeFriends1, "stories_circle_closeFriends1"); colorKeysMap.put(key_stories_circle_closeFriends2, "stories_circle_closeFriends2"); - colorKeysMap.put(key_code_background, "code_background"); colorKeysMap.put(key_chat_inCodeBackground, "chat_inCodeBackground"); colorKeysMap.put(key_chat_outCodeBackground, "chat_outCodeBackground"); colorKeysMap.put(key_code_keyword, "code_keyword"); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Cells/AppIconsSelectorCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Cells/AppIconsSelectorCell.java index 699f81395d3..24a61a9bcf8 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Cells/AppIconsSelectorCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Cells/AppIconsSelectorCell.java @@ -152,7 +152,7 @@ protected float calculateSpeedPerPixel(DisplayMetrics displayMetrics) { private void updateIconsVisibility() { availableIcons.clear(); availableIcons.addAll(Arrays.asList(LauncherIconController.LauncherIcon.values())); - if (MessagesController.getInstance(currentAccount).premiumLocked) { + if (MessagesController.getInstance(currentAccount).premiumFeaturesBlocked()) { for (int i = 0; i < availableIcons.size(); i++) { if (availableIcons.get(i).premium) { availableIcons.remove(i); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Cells/BaseCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Cells/BaseCell.java index deff68f91b5..b62f91235fe 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Cells/BaseCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Cells/BaseCell.java @@ -9,12 +9,15 @@ package org.telegram.ui.Cells; import android.content.Context; +import android.graphics.Canvas; import android.graphics.drawable.Drawable; import android.view.HapticFeedbackConstants; import android.view.MotionEvent; import android.view.ViewConfiguration; import android.view.ViewGroup; +import org.telegram.ui.ActionBar.Theme; + public abstract class BaseCell extends ViewGroup { private final class CheckForTap implements Runnable { @@ -110,4 +113,12 @@ public boolean hasOverlappingRendering() { protected boolean onLongPress() { return true; } + + public int getBoundsLeft() { + return 0; + } + + public int getBoundsRight() { + return getWidth(); + } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Cells/ChatActionCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Cells/ChatActionCell.java index 6c683cbd368..de8837a389f 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Cells/ChatActionCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Cells/ChatActionCell.java @@ -22,6 +22,7 @@ import android.graphics.PorterDuffColorFilter; import android.graphics.Rect; import android.graphics.RectF; +import android.graphics.drawable.Drawable; import android.os.Build; import android.text.Layout; import android.text.Spannable; @@ -42,11 +43,13 @@ import android.view.ViewGroup; import android.view.accessibility.AccessibilityNodeInfo; +import androidx.annotation.NonNull; import androidx.core.graphics.ColorUtils; import org.telegram.messenger.AndroidUtilities; import org.telegram.messenger.ApplicationLoader; import org.telegram.messenger.BuildVars; +import org.telegram.messenger.ChatThemeController; import org.telegram.messenger.DialogObject; import org.telegram.messenger.DocumentObject; import org.telegram.messenger.DownloadController; @@ -65,6 +68,7 @@ import org.telegram.messenger.SharedConfig; import org.telegram.messenger.SvgHelper; import org.telegram.messenger.UserConfig; +import org.telegram.messenger.UserObject; import org.telegram.messenger.Utilities; import org.telegram.messenger.browser.Browser; import org.telegram.tgnet.TLObject; @@ -94,6 +98,7 @@ import org.telegram.ui.Stories.StoriesUtilities; import org.telegram.ui.Stories.UploadingDotsSpannable; import org.telegram.ui.Stories.recorder.HintView2; +import org.telegram.ui.Stories.recorder.PreviewView; import java.util.ArrayList; import java.util.HashMap; @@ -167,7 +172,7 @@ default void didClickImage(ChatActionCell cell) { default void didClickButton(ChatActionCell cell) { } - default void didOpenPremiumGift(ChatActionCell cell, TLRPC.TL_premiumGiftOption giftOption, boolean animateConfetti) { + default void didOpenPremiumGift(ChatActionCell cell, TLRPC.TL_premiumGiftOption giftOption, String slug, boolean animateConfetti) { } default void didOpenPremiumGiftChannel(ChatActionCell cell, String slug, boolean animateConfetti) { @@ -219,6 +224,8 @@ public interface ThemeDelegate extends Theme.ResourcesProvider { private URLSpan pressedLink; private int currentAccount = UserConfig.selectedAccount; private ImageReceiver imageReceiver; + private Drawable wallpaperPreviewDrawable; + private Path clipPath; private AvatarDrawable avatarDrawable; private StaticLayout textLayout; private int textWidth; @@ -267,6 +274,7 @@ public interface ThemeDelegate extends Theme.ResourcesProvider { private ArrayList lineWidths = new ArrayList<>(); private ArrayList lineHeights = new ArrayList<>(); private Path backgroundPath = new Path(); + private int backgroundLeft, backgroundRight; private RectF rect = new RectF(); private boolean invalidatePath = true; private boolean invalidateColors = false; @@ -449,6 +457,9 @@ public void setMessageObject(MessageObject messageObject, boolean force) { previousWidth = 0; imageReceiver.setAutoRepeatCount(0); imageReceiver.clearDecorators(); + if (messageObject.type != MessageObject.TYPE_ACTION_WALLPAPER) { + wallpaperPreviewDrawable = null; + } if (messageObject.isStoryMention()) { TLRPC.User user = MessagesController.getInstance(currentAccount).getUser(messageObject.messageOwner.media.user_id); avatarDrawable.setInfo(currentAccount, user); @@ -470,11 +481,34 @@ public void setMessageObject(MessageObject messageObject, boolean force) { } } } - TLRPC.MessageAction action = messageObject.messageOwner.action; - if (action.wallpaper.uploadingImage != null) { - imageReceiver.setImage(ImageLocation.getForPath(action.wallpaper.uploadingImage), "150_150_wallpaper" + action.wallpaper.id + ChatBackgroundDrawable.hash(action.wallpaper.settings), null, null, ChatBackgroundDrawable.createThumb(action.wallpaper), 0, null, action.wallpaper, 1); + TLRPC.WallPaper wallPaper = null; + if (messageObject.currentEvent != null && messageObject.currentEvent.action instanceof TLRPC.TL_channelAdminLogEventActionChangeWallpaper) { + wallPaper = ((TLRPC.TL_channelAdminLogEventActionChangeWallpaper) messageObject.currentEvent.action).new_value; + } else if (messageObject.messageOwner != null && messageObject.messageOwner.action != null) { + TLRPC.MessageAction action = messageObject.messageOwner.action; + wallPaper = action.wallpaper; + } + if (!TextUtils.isEmpty(ChatThemeController.getWallpaperEmoticon(wallPaper))) { + final boolean isDark = themeDelegate != null ? themeDelegate.isDark() : Theme.isCurrentThemeDark(); + imageReceiver.clearImage(); + wallpaperPreviewDrawable = PreviewView.getBackgroundDrawableFromTheme(currentAccount, ChatThemeController.getWallpaperEmoticon(wallPaper), isDark, false); + if (wallpaperPreviewDrawable != null) { + wallpaperPreviewDrawable.setCallback(this); + } + } else if (wallPaper != null && wallPaper.uploadingImage != null) { + imageReceiver.setImage(ImageLocation.getForPath(wallPaper.uploadingImage), "150_150_wallpaper" + wallPaper.id + ChatBackgroundDrawable.hash(wallPaper.settings), null, null, ChatBackgroundDrawable.createThumb(wallPaper), 0, null, wallPaper, 1); + wallpaperPreviewDrawable = null; + } else if (wallPaper != null) { + TLRPC.Document document = null; + if (messageObject.photoThumbsObject instanceof TLRPC.Document) { + document = (TLRPC.Document) messageObject.photoThumbsObject; + } else if (wallPaper != null) { + document = wallPaper.document; + } + imageReceiver.setImage(ImageLocation.getForDocument(document), "150_150_wallpaper" + wallPaper.id + ChatBackgroundDrawable.hash(wallPaper.settings), null, null, ChatBackgroundDrawable.createThumb(wallPaper), 0, null, wallPaper, 1); + wallpaperPreviewDrawable = null; } else { - imageReceiver.setImage(ImageLocation.getForDocument((TLRPC.Document) messageObject.photoThumbsObject), "150_150_wallpaper" + action.wallpaper.id + ChatBackgroundDrawable.hash(action.wallpaper.settings), null, null, ChatBackgroundDrawable.createThumb(action.wallpaper), 0, null, action.wallpaper, 1); + wallpaperPreviewDrawable = null; } imageReceiver.setRoundRadius((int) (stickerSize / 2f)); @@ -548,12 +582,7 @@ public void setMessageObject(MessageObject messageObject, boolean force) { set = MediaDataController.getInstance(currentAccount).getStickerSetByEmojiOrName(packName); } if (set != null) { - int months; - if (messageObject.messageOwner.action instanceof TLRPC.TL_messageActionGiftCode) { - months = ((TLRPC.TL_messageActionGiftCode) messageObject.messageOwner.action).months; - } else { - months = messageObject.messageOwner.action.months; - } + int months = messageObject.messageOwner.action.months; String monthsEmoticon; if (USE_PREMIUM_GIFT_MONTHS_AS_EMOJI_NUMBERS) { StringBuilder monthsEmoticonBuilder = new StringBuilder(); @@ -916,15 +945,33 @@ private void openPremiumGiftChannel() { } } + private boolean isSelfGiftCode() { + if (currentMessageObject != null && currentMessageObject.messageOwner.action instanceof TLRPC.TL_messageActionGiftCode) { + if (currentMessageObject.messageOwner.from_id instanceof TLRPC.TL_peerUser) { + return MessagesController.getInstance(currentAccount).getUser(currentMessageObject.messageOwner.from_id.user_id).self; + } + } + return false; + } + + private boolean isGiftCode() { + return currentMessageObject != null && currentMessageObject.messageOwner.action instanceof TLRPC.TL_messageActionGiftCode; + } + private void openPremiumGiftPreview() { TLRPC.TL_premiumGiftOption giftOption = new TLRPC.TL_premiumGiftOption(); TLRPC.MessageAction action = currentMessageObject.messageOwner.action; giftOption.amount = action.amount; giftOption.months = action.months; giftOption.currency = action.currency; - + String slug; + if (isGiftCode()) { + slug = isSelfGiftCode() ? null : ((TLRPC.TL_messageActionGiftCode) currentMessageObject.messageOwner.action).slug; + } else { + slug = null; + } if (delegate != null) { - AndroidUtilities.runOnUIThread(() -> delegate.didOpenPremiumGift(ChatActionCell.this, giftOption, false)); + AndroidUtilities.runOnUIThread(() -> delegate.didOpenPremiumGift(ChatActionCell.this, giftOption, slug, false)); } } @@ -1159,7 +1206,13 @@ private void buildLayout() { if (messageObject.messageOwner.media.photo != null) { text = LocaleController.getString(R.string.AttachPhotoExpired); } else if (messageObject.messageOwner.media.document instanceof TLRPC.TL_documentEmpty || messageObject.messageOwner.media instanceof TLRPC.TL_messageMediaDocument && messageObject.messageOwner.media.document == null) { - text = LocaleController.getString(R.string.AttachVideoExpired); + if (messageObject.messageOwner.media.voice) { + text = LocaleController.getString(R.string.AttachVoiceExpired); + } else if (messageObject.messageOwner.media.round) { + text = LocaleController.getString(R.string.AttachRoundExpired); + } else { + text = LocaleController.getString(R.string.AttachVideoExpired); + } } else { text = AnimatedEmojiSpan.cloneSpans(messageObject.messageText); } @@ -1177,7 +1230,9 @@ private void buildLayout() { } else if (messageObject.type == MessageObject.TYPE_GIFT_PREMIUM_CHANNEL) { createGiftPremiumChannelLayouts(); } else if (messageObject.type == MessageObject.TYPE_GIFT_PREMIUM) { - createGiftPremiumLayouts(LocaleController.getString(R.string.ActionGiftPremiumTitle), LocaleController.formatString(R.string.ActionGiftPremiumSubtitle, LocaleController.formatPluralString("Months", messageObject.messageOwner.action.months)), LocaleController.getString(R.string.ActionGiftPremiumView), giftRectSize, true); + String actionName = isGiftCode() && !isSelfGiftCode() ? LocaleController.getString("GiftPremiumUseGiftBtn", R.string.GiftPremiumUseGiftBtn) : + LocaleController.getString("ActionGiftPremiumView", R.string.ActionGiftPremiumView); + createGiftPremiumLayouts(LocaleController.getString(R.string.ActionGiftPremiumTitle), LocaleController.formatString(R.string.ActionGiftPremiumSubtitle, LocaleController.formatPluralString("Months", messageObject.messageOwner.action.months)), actionName, giftRectSize, true); } else if (messageObject.type == MessageObject.TYPE_SUGGEST_PHOTO) { TLRPC.TL_messageActionSuggestProfilePhoto actionSuggestProfilePhoto = (TLRPC.TL_messageActionSuggestProfilePhoto) messageObject.messageOwner.action; String description; @@ -1212,11 +1267,13 @@ private void buildLayout() { CharSequence description; String action = null; boolean actionClickableAsImage = true; - if (!messageObject.isOutOwner() && messageObject.isWallpaperForBoth() && messageObject.isCurrentWallpaper()) { + if (messageObject.getDialogId() < 0) { + description = messageObject.messageText; + } else if (!messageObject.isOutOwner() && messageObject.isWallpaperForBoth() && messageObject.isCurrentWallpaper()) { description = messageObject.messageText; action = LocaleController.getString(R.string.RemoveWallpaperAction); actionClickableAsImage = false; - } else if (user.id == UserConfig.getInstance(currentAccount).clientUserId) { + } else if (user != null && user.id == UserConfig.getInstance(currentAccount).clientUserId) { description = messageObject.messageText; } else { description = messageObject.messageText; @@ -1310,7 +1367,7 @@ private void createGiftPremiumLayouts(CharSequence title, CharSequence subtitle, giftSubtitlePaint.setTextSize(dp(15)); } int subtitleWidth = giftPremiumSubtitleWidth = width; - if (currentMessageObject != null && currentMessageObject.type == MessageObject.TYPE_ACTION_WALLPAPER) { + if (currentMessageObject != null && (currentMessageObject.type == MessageObject.TYPE_ACTION_WALLPAPER && currentMessageObject.getDialogId() >= 0)) { final int recommendedWidthForTwoLines = HintView2.cutInFancyHalf(subtitle, giftSubtitlePaint); if (recommendedWidthForTwoLines < subtitleWidth && recommendedWidthForTwoLines > subtitleWidth / 5f) { subtitleWidth = recommendedWidthForTwoLines; @@ -1384,6 +1441,7 @@ protected void onDraw(Canvas canvas) { } if (giftSubtitlePaint != null && giftSubtitlePaint.getColor() != textPaint.getColor()) { giftSubtitlePaint.setColor(textPaint.getColor()); + giftSubtitlePaint.linkColor = textPaint.getColor(); } } } @@ -1391,7 +1449,20 @@ protected void onDraw(Canvas canvas) { drawBackground(canvas, false); if (isButtonLayout(messageObject) || (messageObject != null && messageObject.type == MessageObject.TYPE_ACTION_PHOTO)) { - if (messageObject.isStoryMention()) { + if (wallpaperPreviewDrawable != null) { + canvas.save(); + canvas.translate(imageReceiver.getImageX(), imageReceiver.getImageY()); + if (clipPath == null) { + clipPath = new Path(); + } else { + clipPath.rewind(); + } + clipPath.addCircle(imageReceiver.getImageWidth() / 2f, imageReceiver.getImageHeight() / 2f, imageReceiver.getImageWidth() / 2f, Path.Direction.CW); + canvas.clipPath(clipPath); + wallpaperPreviewDrawable.setBounds(0, 0, (int) imageReceiver.getImageWidth(), (int) imageReceiver.getImageHeight()); + wallpaperPreviewDrawable.draw(canvas); + canvas.restore(); + } else if (messageObject.isStoryMention()) { long dialogId = messageObject.messageOwner.media.user_id; avatarStoryParams.storyId = messageObject.messageOwner.media.id; StoriesUtilities.drawAvatarWithStory(dialogId, canvas, imageReceiver, avatarStoryParams); @@ -1519,6 +1590,7 @@ protected void onDraw(Canvas canvas) { int oldColor = giftSubtitlePaint.getColor(); settingWallpaperPaint.setAlpha((int) (Color.alpha(oldColor) * (1f - p))); giftSubtitlePaint.setAlpha((int) (Color.alpha(oldColor) * p)); + giftSubtitlePaint.linkColor = giftSubtitlePaint.getColor(); float s = 0.8f + 0.2f * p; canvas.save(); @@ -1528,6 +1600,7 @@ protected void onDraw(Canvas canvas) { canvas.restore(); giftSubtitlePaint.setAlpha((int) (Color.alpha(oldColor) * (1f - p))); + giftSubtitlePaint.linkColor = giftSubtitlePaint.getColor(); s = 0.8f + 0.2f * (1f - p); canvas.save(); canvas.scale(s, s, settingWallpaperLayout.getWidth() / 2f, settingWallpaperLayout.getHeight() / 2f); @@ -1542,6 +1615,7 @@ protected void onDraw(Canvas canvas) { giftSubtitlePaint.setColor(oldColor); + giftSubtitlePaint.linkColor = oldColor; } else { settingWallpaperLayout.draw(canvas); canvas.save(); @@ -1679,6 +1753,11 @@ protected boolean drawChild(Canvas canvas, View child, long drawingTime) { return super.drawChild(canvas, child, drawingTime); } + private void checkLeftRightBounds() { + backgroundLeft = (int) Math.min(backgroundLeft, rect.left); + backgroundRight = (int) Math.max(backgroundRight, rect.right); + } + public void drawBackground(Canvas canvas, boolean fromParent) { if (canDrawInParent) { if (hasGradientService() && !fromParent) { @@ -1706,6 +1785,8 @@ public void drawBackground(Canvas canvas, boolean fromParent) { } if (invalidatePath) { invalidatePath = false; + backgroundLeft = getWidth(); + backgroundRight = 0; lineWidths.clear(); final int count = textLayout == null ? 0 : textLayout.getLineCount(); final int corner = dp(11); @@ -1771,9 +1852,11 @@ public void drawBackground(Canvas canvas, boolean fromParent) { if (a == 0 || lineWidth > prevLineWidth) { rect.set(startX - cornerOffset - corner, y, startX + cornerRest, y + corner * 2); + checkLeftRightBounds(); backgroundPath.arcTo(rect, -90, 90); } else if (lineWidth < prevLineWidth) { rect.set(startX + cornerRest, y, startX + cornerRest + innerCornerRad * 2, y + innerCornerRad * 2); + checkLeftRightBounds(); backgroundPath.arcTo(rect, -90, -90); } y += height; @@ -1790,9 +1873,11 @@ public void drawBackground(Canvas canvas, boolean fromParent) { if (a == count - 1 || lineWidth > nextLineWidth) { rect.set(startX - cornerOffset - corner, y - corner * 2, startX + cornerRest, y); + checkLeftRightBounds(); backgroundPath.arcTo(rect, 0, 90); } else if (lineWidth < nextLineWidth) { rect.set(startX + cornerRest, y - innerCornerRad * 2, startX + cornerRest + innerCornerRad * 2, y); + checkLeftRightBounds(); backgroundPath.arcTo(rect, 180, -90); } @@ -1814,9 +1899,11 @@ public void drawBackground(Canvas canvas, boolean fromParent) { if (a == count - 1 || lineWidth > nextLineWidth) { rect.set(startX - cornerRest, y - corner * 2, startX + cornerOffset + corner, y); + checkLeftRightBounds(); backgroundPath.arcTo(rect, 90, 90); } else if (lineWidth < nextLineWidth) { rect.set(startX - cornerRest - innerCornerRad * 2, y - innerCornerRad * 2, startX - cornerRest, y); + checkLeftRightBounds(); backgroundPath.arcTo(rect, 90, -90); } @@ -1824,9 +1911,11 @@ public void drawBackground(Canvas canvas, boolean fromParent) { if (a == 0 || lineWidth > prevLineWidth) { rect.set(startX - cornerRest, y, startX + cornerOffset + corner, y + corner * 2); + checkLeftRightBounds(); backgroundPath.arcTo(rect, 180, 90); } else if (lineWidth < prevLineWidth) { rect.set(startX - cornerRest - innerCornerRad * 2, y, startX - cornerRest, y + innerCornerRad * 2); + checkLeftRightBounds(); backgroundPath.arcTo(rect, 0, -90); } } @@ -1896,6 +1985,30 @@ public void drawBackground(Canvas canvas, boolean fromParent) { } } + @Override + public int getBoundsLeft() { + if (isButtonLayout(currentMessageObject)) { + return (getWidth() - giftRectSize) / 2; + } + int left = backgroundLeft; + if (imageReceiver != null && imageReceiver.getVisible()) { + left = Math.min((int) imageReceiver.getImageX(), left); + } + return left; + } + + @Override + public int getBoundsRight() { + if (isButtonLayout(currentMessageObject)) { + return (getWidth() + giftRectSize) / 2; + } + int right = backgroundRight; + if (imageReceiver != null && imageReceiver.getVisible()) { + right = Math.max((int) imageReceiver.getImageX2(), right); + } + return right; + } + public boolean hasGradientService() { return overrideBackgroundPaint == null && (themeDelegate != null ? themeDelegate.hasGradientService() : Theme.hasGradientService()); } @@ -1987,7 +2100,7 @@ private int getThemedColor(int key) { return Theme.getColor(key, themeDelegate); } - private Paint getThemedPaint(String paintKey) { + protected Paint getThemedPaint(String paintKey) { Paint paint = themeDelegate != null ? themeDelegate.getPaint(paintKey) : null; return paint != null ? paint : Theme.getThemePaint(paintKey); } @@ -2031,6 +2144,11 @@ public void invalidate(int l, int t, int r, int b) { } } + @Override + protected boolean verifyDrawable(@NonNull Drawable who) { + return who == wallpaperPreviewDrawable || super.verifyDrawable(who); + } + public boolean isFloating() { return false; } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Cells/ChatMessageCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Cells/ChatMessageCell.java index 0396fb8d438..594f0761e4d 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Cells/ChatMessageCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Cells/ChatMessageCell.java @@ -9,6 +9,7 @@ package org.telegram.ui.Cells; import static org.telegram.messenger.AndroidUtilities.dp; +import static org.telegram.messenger.AndroidUtilities.dpf2; import android.animation.Animator; import android.animation.AnimatorListenerAdapter; @@ -34,6 +35,7 @@ import android.graphics.PixelFormat; import android.graphics.PorterDuff; import android.graphics.PorterDuffColorFilter; +import android.graphics.PorterDuffXfermode; import android.graphics.Rect; import android.graphics.RectF; import android.graphics.Region; @@ -81,10 +83,13 @@ import android.widget.Toast; import androidx.annotation.NonNull; +import androidx.annotation.Nullable; import androidx.core.content.ContextCompat; import androidx.core.graphics.ColorUtils; import androidx.core.math.MathUtils; +import com.google.android.exoplayer2.util.Log; + import org.telegram.PhoneFormat.PhoneFormat; import org.telegram.messenger.AccountInstance; import org.telegram.messenger.AndroidUtilities; @@ -134,6 +139,7 @@ import org.telegram.ui.Components.AudioVisualizerDrawable; import org.telegram.ui.Components.AvatarDrawable; import org.telegram.ui.Components.BackgroundGradientDrawable; +import org.telegram.ui.Components.BottomPagerTabs; import org.telegram.ui.Components.ButtonBounce; import org.telegram.ui.Components.CheckBoxBase; import org.telegram.ui.Components.ClipRoundedDrawable; @@ -151,6 +157,7 @@ import org.telegram.ui.Components.MsgClockDrawable; import org.telegram.ui.Components.Point; import org.telegram.ui.Components.Premium.boosts.cells.msg.GiveawayMessageCell; +import org.telegram.ui.Components.Premium.boosts.cells.msg.GiveawayResultsMessageCell; import org.telegram.ui.Components.QuoteHighlight; import org.telegram.ui.Components.RLottieDrawable; import org.telegram.ui.Components.RadialProgress2; @@ -180,6 +187,7 @@ import org.telegram.ui.SecretMediaViewer; import org.telegram.ui.Stories.StoriesUtilities; import org.telegram.ui.Stories.StoryViewer; +import org.telegram.ui.Stories.recorder.CaptionContainerView; import org.telegram.ui.Stories.recorder.DominantColors; import java.io.File; @@ -201,8 +209,8 @@ public class ChatMessageCell extends BaseCell implements SeekBar.SeekBarDelegate private boolean flipImage; private boolean visibleOnScreen = true; public boolean shouldCheckVisibleOnScreen; - float parentBoundsTop; - int parentBoundsBottom; + public float parentBoundsTop; + public int parentBoundsBottom; public ExpiredStoryView expiredStoryView; private boolean skipFrameUpdate; @@ -254,7 +262,7 @@ public void didReceivedNotification(int id, int account, Object... args) { invalidate(); } else if (id == NotificationCenter.didUpdatePremiumGiftStickers) { MessageObject messageObject = currentMessageObject; - if (messageObject != null && messageObject.messageOwner.media instanceof TLRPC.TL_messageMediaGiveaway) { + if (messageObject != null && (messageObject.messageOwner.media instanceof TLRPC.TL_messageMediaGiveaway || messageObject.messageOwner.media instanceof TLRPC.TL_messageMediaGiveawayResults)) { setMessageObject(messageObject, currentMessagesGroup, pinnedBottom, pinnedTop); } } @@ -570,7 +578,7 @@ default String getAdminRank(long uid) { return null; } - default boolean needPlayMessage(MessageObject messageObject, boolean muted) { + default boolean needPlayMessage(ChatMessageCell cell, MessageObject messageObject, boolean muted) { return false; } @@ -755,6 +763,7 @@ public static class PollButton { private boolean groupPhotoInvisible; public final ReactionsLayoutInBubble reactionsLayoutInBubble = new ReactionsLayoutInBubble(this); public final GiveawayMessageCell giveawayMessageCell = new GiveawayMessageCell(this); + public final GiveawayResultsMessageCell giveawayResultsMessageCell = new GiveawayResultsMessageCell(this); private boolean invalidateSpoilersParent; @@ -802,6 +811,12 @@ public boolean isCellAttachedToWindow() { private Drawable locationLoadingThumb; private Drawable gradientDrawable; private Paint gradientLoadingPaint; + private CaptionContainerView.PeriodDrawable oncePeriod; + private Paint onceClearPaint; + private RLottieDrawable onceFire; + private Paint onceRadialPaint; + private Paint onceRadialStrokePaint; + private int onceRadialPaintColor; private boolean disallowLongPress; private float lastTouchX; @@ -815,7 +830,7 @@ public boolean isCellAttachedToWindow() { private boolean checkBoxAnimationInProgress; private float checkBoxAnimationProgress; private long lastCheckBoxAnimationTime; - private int checkBoxTranslation; + public int checkBoxTranslation; public boolean linkPreviewAbove; private boolean isSmallImage; @@ -1311,6 +1326,7 @@ class LoadingDrawableLocation { private Drawable linkPreviewSelector; public int linkPreviewSelectorColor; + @Nullable private ChatMessageCellDelegate delegate; private MessageBackgroundDrawable backgroundDrawable; @@ -1524,6 +1540,10 @@ public void setResourcesProvider(Theme.ResourcesProvider resourcesProvider) { } } + public Theme.ResourcesProvider getResourcesProvider() { + return resourcesProvider; + } + private void createPollUI() { if (pollAvatarImages != null) { return; @@ -2313,13 +2333,13 @@ private boolean checkLinkPreviewMotionEvent(MotionEvent event) { } else { if (documentAttachType == DOCUMENT_ATTACH_TYPE_ROUND) { if (!MediaController.getInstance().isPlayingMessage(currentMessageObject) || MediaController.getInstance().isMessagePaused()) { - delegate.needPlayMessage(currentMessageObject, false); + delegate.needPlayMessage(this, currentMessageObject, false); } else { MediaController.getInstance().pauseMessage(currentMessageObject); } } else if (documentAttachType == DOCUMENT_ATTACH_TYPE_GIF && drawImageButton) { if (buttonState == -1) { - if (SharedConfig.isAutoplayGifs()) { + if (SharedConfig.isAutoplayGifs() && !currentMessageObject.isRepostPreview) { delegate.didPressImage(this, lastTouchX, lastTouchY); } else { buttonState = 2; @@ -2855,7 +2875,7 @@ private boolean checkPhotoImageMotionEvent(MotionEvent event) { if (currentMessageObject.isSendError()) { imagePressed = false; result = false; - } else if (currentMessageObject.type == MessageObject.TYPE_GIF && buttonState == -1 && SharedConfig.isAutoplayGifs() && photoImage.getAnimation() == null) { + } else if (currentMessageObject.type == MessageObject.TYPE_GIF && buttonState == -1 && SharedConfig.isAutoplayGifs() && !currentMessageObject.isRepostPreview && photoImage.getAnimation() == null) { imagePressed = false; result = false; } @@ -2933,7 +2953,7 @@ private boolean checkAudioMotionEvent(MotionEvent event) { int offset = AndroidUtilities.dp(27); area2 = x >= buttonX + offset && x <= buttonX + offset + side && y >= buttonY + offset && y <= buttonY + offset + side; } - if (!area2 && (currentMessageObject == null || !currentMessageObject.hasMediaSpoilers() || currentMessageObject.isMediaSpoilersRevealed)) { + if (!area2 && (currentMessageObject == null || !currentMessageObject.hasMediaSpoilers() || currentMessageObject.isVoice() || currentMessageObject.isMediaSpoilersRevealed)) { if (buttonState == 0 || buttonState == 1 || buttonState == 2) { area = x >= buttonX - AndroidUtilities.dp(12) && x <= buttonX - AndroidUtilities.dp(12) + backgroundWidth && y >= (drawInstantView ? buttonY : namesOffset + mediaOffsetY) && y <= (drawInstantView ? buttonY + side : namesOffset + mediaOffsetY + AndroidUtilities.dp(82)); } else { @@ -3222,7 +3242,7 @@ private boolean checkBotButtonMotionEvent(MotionEvent event) { @Override public boolean onTouchEvent(MotionEvent event) { - if (currentMessageObject == null || !delegate.canPerformActions() || animationRunning) { + if (currentMessageObject == null || delegate != null && !delegate.canPerformActions() || animationRunning) { if (currentMessageObject != null && currentMessageObject.preview) { return checkTextSelection(event); } else { @@ -3325,6 +3345,9 @@ public boolean onTouchEvent(MotionEvent event) { if (!result) { result = giveawayMessageCell.checkMotionEvent(event); } + if (!result) { + result = giveawayResultsMessageCell.checkMotionEvent(event); + } if (event.getAction() == MotionEvent.ACTION_CANCEL) { spoilerPressed = null; @@ -3855,7 +3878,9 @@ public void updatePlayingMessageProgress() { } } } - if (MediaController.getInstance().isPlayingMessage(currentMessageObject)) { + if (overridenDuration >= 0) { + duration = overridenDuration; + } else if (MediaController.getInstance().isPlayingMessage(currentMessageObject)) { duration = Math.max(0, duration - currentMessageObject.audioProgressSec); } if (lastTime != duration) { @@ -3886,7 +3911,9 @@ public void updatePlayingMessageProgress() { double duration = 0; if (documentAttachType == DOCUMENT_ATTACH_TYPE_AUDIO) { - if (!MediaController.getInstance().isPlayingMessage(currentMessageObject)) { + if (overridenDuration >= 0) { + duration = overridenDuration; + } else if (!MediaController.getInstance().isPlayingMessage(currentMessageObject)) { for (int a = 0; a < documentAttach.attributes.size(); a++) { TLRPC.DocumentAttribute attribute = documentAttach.attributes.get(a); if (attribute instanceof TLRPC.TL_documentAttributeAudio) { @@ -3921,6 +3948,11 @@ public void updatePlayingMessageProgress() { } } + private long overridenDuration = -1; + public void overrideDuration(long duration) { + overridenDuration = duration; + } + public void setFullyDraw(boolean draw) { fullyDraw = draw; } @@ -4071,7 +4103,7 @@ private void didClickedImage() { didPressButton(true, false); } else { if (!MediaController.getInstance().isPlayingMessage(currentMessageObject) || MediaController.getInstance().isMessagePaused()) { - delegate.needPlayMessage(currentMessageObject, false); + delegate.needPlayMessage(this, currentMessageObject, false); } else { MediaController.getInstance().pauseMessage(currentMessageObject); } @@ -4344,6 +4376,11 @@ public int getChecksY() { } } + private AudioVisualizerDrawable overridenAudioVisualizer; + public void overrideAudioVisualizer(AudioVisualizerDrawable audioVisualizerDrawable) { + this.overridenAudioVisualizer = audioVisualizerDrawable; + } + public TLRPC.User getCurrentUser() { return currentUser; } @@ -4527,6 +4564,7 @@ private void checkImageReceiversAttachState() { } } giveawayMessageCell.onAttachedToWindow(); + giveawayResultsMessageCell.onAttachedToWindow(); replyImageReceiver.onAttachedToWindow(); locationImageReceiver.onAttachedToWindow(); blurredPhotoImage.onAttachedToWindow(); @@ -4558,6 +4596,7 @@ private void checkImageReceiversAttachState() { photoImage.onDetachedFromWindow(); blurredPhotoImage.onDetachedFromWindow(); giveawayMessageCell.onDetachedFromWindow(); + giveawayResultsMessageCell.onDetachedFromWindow(); AnimatedEmojiSpan.release(this, animatedEmojiDescriptionStack); AnimatedEmojiSpan.release(this, animatedEmojiReplyStack); @@ -4777,8 +4816,8 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe hasNewLineForTime = false; flipImage = false; isThreadPost = isThreadChat && messageObject.messageOwner.fwd_from != null && messageObject.messageOwner.fwd_from.channel_post != 0 && messageObject.messageOwner.reply_to == null; - isAvatarVisible = !isThreadPost && isChat && !messageObject.isOutOwner() && messageObject.needDrawAvatar() && (currentPosition == null || currentPosition.edge); - boolean drawAvatar = isChat && !isThreadPost && !messageObject.isOutOwner() && messageObject.needDrawAvatar(); + isAvatarVisible = needDrawAvatar() && (currentPosition == null || currentPosition.edge); + boolean drawAvatar = needDrawAvatar(); if (messageObject.customAvatarDrawable != null || messageObject.forceAvatar) { isAvatarVisible = true; drawAvatar = true; @@ -4947,7 +4986,7 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe if (messageIdChanged || messageObject.reactionsChanged || wasPlayingRound != isPlayingRound) { messageObject.reactionsChanged = false; - if (!messageObject.isExpiredStory() && (currentPosition == null || ((currentPosition.flags & MessageObject.POSITION_FLAG_BOTTOM) != 0))) { + if (messageObject.shouldDrawReactions() && !messageObject.isExpiredStory() && (currentPosition == null || ((currentPosition.flags & MessageObject.POSITION_FLAG_BOTTOM) != 0))) { boolean isSmall = !messageObject.shouldDrawReactionsInLayout(); if (currentPosition != null) { reactionsLayoutInBubble.setMessage(groupedMessages.findPrimaryMessageObject(), !messageObject.shouldDrawReactionsInLayout(), resourcesProvider); @@ -5112,7 +5151,7 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe forwardedNameLayout[0] = null; forwardedNameLayout[1] = null; drawName = false; - } else if (messageObject.type == MessageObject.TYPE_TEXT || messageObject.type == MessageObject.TYPE_STORY_MENTION || messageObject.isGiveaway() || messageObject.isSponsored()) { + } else if (messageObject.type == MessageObject.TYPE_TEXT || messageObject.type == MessageObject.TYPE_STORY_MENTION || messageObject.isGiveawayOrGiveawayResults() || messageObject.isSponsored()) { drawForwardedName = !isRepliesChat; int maxWidth; @@ -5160,7 +5199,7 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe TLRPC.Document androidThemeDocument = null; TL_stories.StoryItem storyItem = null; TLRPC.ThemeSettings androidThemeSettings = null; - if (messageObject.isGiveaway()) { + if (messageObject.isGiveawayOrGiveawayResults()) { hasLinkPreview = true; } @@ -5185,7 +5224,7 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe } } } - } else if (messageObject.isGiveaway()) { + } else if (messageObject.isGiveawayOrGiveawayResults()) { drawInstantView = true; drawInstantViewType = 19; } else if ("telegram_channel_boost".equals(webpageType)) { @@ -5357,6 +5396,9 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe photosCountLayout = new StaticLayout(str, Theme.chat_durationPaint, photosCountWidth, Layout.Alignment.ALIGN_NORMAL, 1.0f, 0.0f, false); } } + if (messageObject.isRepostPreview) { + drawInstantView = false; + } backgroundWidth = maxWidth; if (hasLinkPreview && !linkPreviewAbove || hasGamePreview || hasInvoicePreview || maxWidth - messageObject.lastLineWidth < timeMore) { backgroundWidth = Math.max(backgroundWidth, messageObject.lastLineWidth) + AndroidUtilities.dp(31); @@ -5376,6 +5418,7 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe setMessageObjectInternal(messageObject); giveawayMessageCell.setMessageContent(messageObject, getParentWidth(), forwardedNameWidth); + giveawayResultsMessageCell.setMessageContent(messageObject, getParentWidth(), forwardedNameWidth); backgroundWidth = messageObject.textWidth + getExtraTextX() * 2 + (hasGamePreview || hasInvoicePreview ? AndroidUtilities.dp(10) : 0); if (messageObject.isSponsored()) { @@ -5385,7 +5428,13 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe } if (!reactionsLayoutInBubble.isSmall) { - reactionsLayoutInBubble.measure(messageObject.isGiveaway() ? giveawayMessageCell.getMeasuredWidth() : maxWidth, Gravity.LEFT); + int availableWidth = maxWidth; + if (messageObject.isGiveaway()) { + availableWidth = giveawayMessageCell.getMeasuredWidth(); + } else if (messageObject.isGiveawayResults()) { + availableWidth = giveawayResultsMessageCell.getMeasuredWidth(); + } + reactionsLayoutInBubble.measure(availableWidth, Gravity.LEFT); if (!reactionsLayoutInBubble.isEmpty) { reactionsLayoutInBubble.totalHeight = reactionsLayoutInBubble.height + AndroidUtilities.dp(8); if (reactionsLayoutInBubble.width > backgroundWidth) { @@ -5416,7 +5465,9 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe if (hasLinkPreview || hasGamePreview || hasInvoicePreview) { int linkPreviewMaxWidth; - if (AndroidUtilities.isTablet()) { + if (currentMessageObject.isRepostPreview) { + linkPreviewMaxWidth = currentMessageObject.getMaxMessageTextWidth(); + } else if (AndroidUtilities.isTablet()) { linkPreviewMaxWidth = AndroidUtilities.getMinTabletSide() - AndroidUtilities.dp(80 + (drawAvatar ? 52 : 0)); } else { linkPreviewMaxWidth = getParentWidth() - AndroidUtilities.dp(80 + (drawAvatar ? 52 : 0)); @@ -5608,7 +5659,10 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe totalHeight += giveawayMessageCell.getMeasuredHeight(); linkPreviewHeight += giveawayMessageCell.getMeasuredHeight(); + totalHeight += giveawayResultsMessageCell.getMeasuredHeight(); + linkPreviewHeight += giveawayResultsMessageCell.getMeasuredHeight(); maxChildWidth = Math.max(maxChildWidth, giveawayMessageCell.getMeasuredWidth()); + maxChildWidth = Math.max(maxChildWidth, giveawayResultsMessageCell.getMeasuredWidth()); if (site_name != null) { try { @@ -5674,11 +5728,15 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe titleCs = Emoji.replaceEmoji(title, Theme.chat_replyNamePaint.getFontMetricsInt(), AndroidUtilities.dp(14), false); } catch (Exception ignore) { } + int maxLines = 4; + if (currentMessageObject.isRepostPreview) { + maxLines = 1; + } if (!isSmallImage) { - titleLayout = StaticLayoutEx.createStaticLayout(titleCs, Theme.chat_replyNamePaint, linkPreviewMaxWidth, Layout.Alignment.ALIGN_NORMAL, 1.0f, AndroidUtilities.dp(1), false, TextUtils.TruncateAt.END, linkPreviewMaxWidth, 4); + titleLayout = StaticLayoutEx.createStaticLayout(titleCs, Theme.chat_replyNamePaint, linkPreviewMaxWidth, Layout.Alignment.ALIGN_NORMAL, 1.0f, AndroidUtilities.dp(1), false, TextUtils.TruncateAt.END, linkPreviewMaxWidth, maxLines); } else { restLines = restLinesCount; - titleLayout = generateStaticLayout(titleCs, Theme.chat_replyNamePaint, linkPreviewMaxWidth, linkPreviewMaxWidth - smallImageSide - smallSideMargin, restLinesCount, 4); + titleLayout = generateStaticLayout(titleCs, Theme.chat_replyNamePaint, linkPreviewMaxWidth, linkPreviewMaxWidth - smallImageSide - smallSideMargin, restLinesCount, maxLines); restLinesCount -= titleLayout.getLineCount(); } @@ -5755,11 +5813,12 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe boolean allowAllLines = site_name != null && site_name.toString().toLowerCase().equals("twitter"); CharSequence text = overrideDescrption != null ? overrideDescrption : messageObject.linkDescription; boolean isRTL = AndroidUtilities.isRTL(text); + int maxLines = allowAllLines ? 100 : (currentMessageObject.isRepostPreview ? 3 : 6); if (restLinesCount == 3 && !isSmallImage) { - descriptionLayout = StaticLayoutEx.createStaticLayout(text, Theme.chat_replyTextPaint, linkPreviewMaxWidth, Layout.Alignment.ALIGN_NORMAL, 1.0f, AndroidUtilities.dp(1), false, TextUtils.TruncateAt.END, linkPreviewMaxWidth, allowAllLines ? 100 : 6); + descriptionLayout = StaticLayoutEx.createStaticLayout(text, Theme.chat_replyTextPaint, linkPreviewMaxWidth, Layout.Alignment.ALIGN_NORMAL, 1.0f, AndroidUtilities.dp(1), false, TextUtils.TruncateAt.END, linkPreviewMaxWidth, maxLines); } else { restLines = restLinesCount; - descriptionLayout = generateStaticLayout(text, Theme.chat_replyTextPaint, linkPreviewMaxWidth - (isRTL ? smallImageSide : 0), linkPreviewMaxWidth - smallImageSide - (isRTL ? 0 : smallSideMargin), restLinesCount, allowAllLines ? 100 : 6); + descriptionLayout = generateStaticLayout(text, Theme.chat_replyTextPaint, linkPreviewMaxWidth - (isRTL ? smallImageSide + 2 * smallSideMargin : 0), linkPreviewMaxWidth - smallImageSide - 2 * smallSideMargin, restLinesCount, maxLines); } animatedEmojiDescriptionStack = AnimatedEmojiSpan.update(AnimatedEmojiDrawable.CACHE_TYPE_MESSAGES, this, false, animatedEmojiDescriptionStack, descriptionLayout); @@ -5838,7 +5897,7 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe isSmallImage = smallImage = false; documentAttachType = DOCUMENT_ATTACH_TYPE_ROUND; } else if (MessageObject.isGifDocument(document, messageObject.hasValidGroupId())) { - if (!messageObject.isGame() && !SharedConfig.isAutoplayGifs()) { + if (!messageObject.isGame() && !(SharedConfig.isAutoplayGifs() && !messageObject.isRepostPreview)) { messageObject.gifState = 1; } photoImage.setAllowStartAnimation(messageObject.gifState != 1); @@ -6226,7 +6285,7 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe } else if (documentAttachType == DOCUMENT_ATTACH_TYPE_VIDEO) { photoImage.setNeedsQualityThumb(true); photoImage.setShouldGenerateQualityThumb(true); - if (!isSmallImage && SharedConfig.isAutoplayVideo() && (!currentMessageObject.hasMediaSpoilers() || currentMessageObject.isMediaSpoilersRevealed || currentMessageObject.revealingMediaSpoilers) && ( + if (!isSmallImage && SharedConfig.isAutoplayVideo() && !currentMessageObject.isRepostPreview && (!currentMessageObject.hasMediaSpoilers() || currentMessageObject.isMediaSpoilersRevealed || currentMessageObject.revealingMediaSpoilers) && ( (currentMessageObject.mediaExists || currentMessageObject.attachPathExists) || messageObject.canStreamVideo() && DownloadController.getInstance(currentAccount).canDownloadMedia(currentMessageObject) )) { @@ -6253,7 +6312,7 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe autoDownload = DownloadController.getInstance(currentAccount).canDownloadMedia(currentMessageObject); } String filter = currentPhotoObject instanceof TLRPC.TL_photoStrippedSize || "s".equals(currentPhotoObject.type) ? currentPhotoFilterThumb : currentPhotoFilter; - if (messageObject.mediaExists || autoDownload) { + if ((messageObject.mediaExists || autoDownload) && !currentMessageObject.isRepostPreview) { autoPlayingMedia = true; TLRPC.VideoSize videoSize = MessageObject.getDocumentVideoThumb(document); if (!messageObject.mediaExists && videoSize != null && (currentPhotoObject == null || currentPhotoObjectThumb == null)) { @@ -6358,11 +6417,11 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe calcBackgroundWidth(maxWidth, timeMore, maxChildWidth); } - if (!hasInvoicePreview && !currentMessageObject.isGiveaway()) { + if (!hasInvoicePreview && !currentMessageObject.isGiveawayOrGiveawayResults()) { linkPreviewHeight += AndroidUtilities.dp(6); totalHeight += AndroidUtilities.dp(6); } - if (!hasInvoicePreview && !currentMessageObject.isGiveaway() && ( + if (!hasInvoicePreview && !currentMessageObject.isGiveawayOrGiveawayResults() && ( currentPhotoObject != null || documentAttachType == DOCUMENT_ATTACH_TYPE_VIDEO ) && (authorLayout != null || descriptionLayout != null || titleLayout != null || siteNameLayout != null)) { linkPreviewHeight += AndroidUtilities.dp(2.66f); @@ -6928,7 +6987,7 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe } else { drawForwardedName = (messageObject.messageOwner.fwd_from != null && !(messageObject.isAnyKindOfSticker() && messageObject.isDice())) || messageObject.type == MessageObject.TYPE_STORY; if (!messageObject.isAnyKindOfSticker() && messageObject.type != MessageObject.TYPE_ROUND_VIDEO) { - drawName = (messageObject.isFromGroup() && messageObject.isSupergroup() || messageObject.isImportedForward() && messageObject.messageOwner.fwd_from.from_id == null) && (currentPosition == null || (currentPosition.flags & MessageObject.POSITION_FLAG_TOP) != 0); + drawName = (messageObject.isFromGroup() && messageObject.isSupergroup() || messageObject.isRepostPreview || messageObject.isImportedForward() && messageObject.messageOwner.fwd_from.from_id == null) && (currentPosition == null || (currentPosition.flags & MessageObject.POSITION_FLAG_TOP) != 0); } mediaBackground = isMedia = messageObject.type != MessageObject.TYPE_FILE; drawImageButton = true; @@ -6938,7 +6997,7 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe int photoHeight = 0; int additionHeight = 0; - if (messageObject.gifState != 2 && !SharedConfig.isAutoplayGifs() && (messageObject.type == MessageObject.TYPE_GIF || messageObject.type == MessageObject.TYPE_ROUND_VIDEO)) { + if (messageObject.gifState != 2 && !(SharedConfig.isAutoplayGifs() && !messageObject.isRepostPreview) && (messageObject.type == MessageObject.TYPE_GIF || messageObject.type == MessageObject.TYPE_ROUND_VIDEO)) { messageObject.gifState = 1; } @@ -7708,7 +7767,7 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe } else { w -= AndroidUtilities.dp(9); } - if ((isChat && !isThreadPost && !m.isOutOwner() || m.forceAvatar) && m.needDrawAvatar() && (rowPosition == null || rowPosition.edge)) { + if (((isChat || m.isRepostPreview) && !isThreadPost && !m.isOutOwner() || m.forceAvatar) && m.needDrawAvatar() && (rowPosition == null || rowPosition.edge)) { w -= AndroidUtilities.dp(48); } w += getAdditionalWidthForPosition(rowPosition); @@ -7907,7 +7966,7 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe currentPhotoObjectThumb.size = -1; } - if (SharedConfig.isAutoplayVideo() && (!currentMessageObject.hasMediaSpoilers() || currentMessageObject.isMediaSpoilersRevealed || currentMessageObject.revealingMediaSpoilers) && messageObject.type == MessageObject.TYPE_VIDEO && !messageObject.needDrawBluredPreview() && + if (SharedConfig.isAutoplayVideo() && !currentMessageObject.isRepostPreview && (!currentMessageObject.hasMediaSpoilers() || currentMessageObject.isMediaSpoilersRevealed || currentMessageObject.revealingMediaSpoilers) && messageObject.type == MessageObject.TYPE_VIDEO && !messageObject.needDrawBluredPreview() && ((currentMessageObject.mediaExists || currentMessageObject.attachPathExists) || messageObject.canStreamVideo() && DownloadController.getInstance(currentAccount).canDownloadMedia(currentMessageObject)) ) { if (currentPosition != null) { @@ -7917,6 +7976,7 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe } } + final int cacheType = currentMessageObject.shouldEncryptPhotoOrVideo() ? ImageLoader.CACHE_TYPE_ENCRYPTED : ImageLoader.CACHE_TYPE_NONE; if (autoPlayingMedia) { photoImage.setAllowStartAnimation(true); photoImage.startAnimation(); @@ -7943,7 +8003,7 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe photoImage.clearImage(); } } else if (messageObject.type == MessageObject.TYPE_EXTENDED_MEDIA_PREVIEW) { - photoImage.setImage(null, null, ImageLocation.getForObject(currentPhotoObjectThumb, photoParentObject), currentPhotoFilterThumb, currentPhotoObjectThumbStripped, 0, null, currentMessageObject, currentMessageObject.shouldEncryptPhotoOrVideo() ? 2 : 0); + photoImage.setImage(null, null, ImageLocation.getForObject(currentPhotoObjectThumb, photoParentObject), currentPhotoFilterThumb, currentPhotoObjectThumbStripped, 0, null, currentMessageObject, cacheType); } else if (messageObject.type == MessageObject.TYPE_PHOTO) { if (messageObject.useCustomPhoto) { photoImage.setImageBitmap(getResources().getDrawable(R.drawable.theme_preview_image)); @@ -7957,11 +8017,11 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe photoExist = false; } if (photoExist || !currentMessageObject.loadingCancelled && DownloadController.getInstance(currentAccount).canDownloadMedia(currentMessageObject) || FileLoader.getInstance(currentAccount).isLoadingFile(fileName)) { - photoImage.setImage(ImageLocation.getForObject(currentPhotoObject, photoParentObject), currentPhotoFilter, ImageLocation.getForObject(currentPhotoObjectThumb, photoParentObject), currentPhotoFilterThumb, currentPhotoObjectThumbStripped, currentPhotoObject.size, null, currentMessageObject, currentMessageObject.shouldEncryptPhotoOrVideo() ? 2 : 0); + photoImage.setImage(ImageLocation.getForObject(currentPhotoObject, photoParentObject), currentPhotoFilter, ImageLocation.getForObject(currentPhotoObjectThumb, photoParentObject), currentPhotoFilterThumb, currentPhotoObjectThumbStripped, currentPhotoObject.size, null, currentMessageObject, cacheType); } else { photoNotSet = true; if (currentPhotoObjectThumb != null || currentPhotoObjectThumbStripped != null) { - photoImage.setImage(null, null, ImageLocation.getForObject(currentPhotoObjectThumb, photoParentObject), currentPhotoFilterThumb, currentPhotoObjectThumbStripped, 0, null, currentMessageObject, currentMessageObject.shouldEncryptPhotoOrVideo() ? 2 : 0); + photoImage.setImage(null, null, ImageLocation.getForObject(currentPhotoObjectThumb, photoParentObject), currentPhotoFilterThumb, currentPhotoObjectThumbStripped, 0, null, currentMessageObject, cacheType); } else { photoImage.setImageBitmap((Drawable) null); } @@ -7993,19 +8053,19 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe photoImage.setCrossfadeDuration(250); } if (localFile == 0 && videoSize != null && (currentPhotoObject == null || currentPhotoObjectThumb == null)) { - photoImage.setImage(ImageLocation.getForDocument(document), ImageLoader.AUTOPLAY_FILTER, ImageLocation.getForDocument(videoSize, documentAttach), null, ImageLocation.getForDocument(currentPhotoObject != null ? currentPhotoObject : currentPhotoObjectThumb, documentAttach), currentPhotoObject != null ? currentPhotoFilter : currentPhotoFilterThumb, currentPhotoObjectThumbStripped, document.size, null, messageObject, 0); + photoImage.setImage(ImageLocation.getForDocument(document), ImageLoader.AUTOPLAY_FILTER, ImageLocation.getForDocument(videoSize, documentAttach), null, ImageLocation.getForDocument(currentPhotoObject != null ? currentPhotoObject : currentPhotoObjectThumb, documentAttach), currentPhotoObject != null ? currentPhotoFilter : currentPhotoFilterThumb, currentPhotoObjectThumbStripped, document.size, null, messageObject, cacheType); } else { if (isRoundVideo && !messageIdChanged && photoImage.hasStaticThumb()) { photoImage.setImage(ImageLocation.getForDocument(document), ImageLoader.AUTOPLAY_FILTER, ImageLocation.getForObject(currentPhotoObject, photoParentObject), currentPhotoFilter, null, null, photoImage.getStaticThumb(), document.size, null, messageObject, 0); } else { - photoImage.setImage(ImageLocation.getForDocument(document), ImageLoader.AUTOPLAY_FILTER, ImageLocation.getForObject(currentPhotoObject, photoParentObject), currentPhotoFilter, ImageLocation.getForObject(currentPhotoObjectThumb, photoParentObject), currentPhotoFilterThumb, currentPhotoObjectThumbStripped, document.size, null, messageObject, 0); + photoImage.setImage(ImageLocation.getForDocument(document), ImageLoader.AUTOPLAY_FILTER, ImageLocation.getForObject(currentPhotoObject, photoParentObject), currentPhotoFilter, ImageLocation.getForObject(currentPhotoObjectThumb, photoParentObject), currentPhotoFilterThumb, currentPhotoObjectThumbStripped, document.size, null, messageObject, cacheType); } } } else if (localFile == 1) { - photoImage.setImage(ImageLocation.getForPath(messageObject.isSendError() ? null : messageObject.messageOwner.attachPath), null, ImageLocation.getForObject(currentPhotoObject, photoParentObject), currentPhotoFilter, ImageLocation.getForObject(currentPhotoObjectThumb, photoParentObject), currentPhotoFilterThumb, currentPhotoObjectThumbStripped, 0, null, messageObject, 0); + photoImage.setImage(ImageLocation.getForPath(messageObject.isSendError() ? null : messageObject.messageOwner.attachPath), null, ImageLocation.getForObject(currentPhotoObject, photoParentObject), currentPhotoFilter, ImageLocation.getForObject(currentPhotoObjectThumb, photoParentObject), currentPhotoFilterThumb, currentPhotoObjectThumbStripped, 0, null, messageObject, cacheType); } else { if (videoSize != null && (currentPhotoObject == null || currentPhotoObjectThumb == null)) { - photoImage.setImage(ImageLocation.getForDocument(document), null, ImageLocation.getForDocument(videoSize, documentAttach), null, ImageLocation.getForDocument(currentPhotoObject != null ? currentPhotoObject : currentPhotoObjectThumb, documentAttach), currentPhotoObject != null ? currentPhotoFilter : currentPhotoFilterThumb, currentPhotoObjectThumbStripped, document.size, null, messageObject, 0); + photoImage.setImage(ImageLocation.getForDocument(document), null, ImageLocation.getForDocument(videoSize, documentAttach), null, ImageLocation.getForDocument(currentPhotoObject != null ? currentPhotoObject : currentPhotoObjectThumb, documentAttach), currentPhotoObject != null ? currentPhotoFilter : currentPhotoFilterThumb, currentPhotoObjectThumbStripped, document.size, null, messageObject, cacheType); } else { photoImage.setImage(ImageLocation.getForDocument(document), null, ImageLocation.getForObject(currentPhotoObject, photoParentObject), currentPhotoFilter, ImageLocation.getForObject(currentPhotoObjectThumb, photoParentObject), currentPhotoFilterThumb, currentPhotoObjectThumbStripped, document.size, null, messageObject, 0); } @@ -8019,19 +8079,19 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe photoImage.setCrossfadeWithOldImage(true); photoImage.setCrossfadeDuration(250); } - photoImage.setImage(ImageLocation.getForObject(currentPhotoObject, photoParentObject), currentPhotoFilter, ImageLocation.getForObject(currentPhotoObjectThumb, photoParentObject), currentPhotoFilterThumb, currentPhotoObjectThumbStripped, 0, null, messageObject, 0); + photoImage.setImage(ImageLocation.getForObject(currentPhotoObject, photoParentObject), currentPhotoFilter, ImageLocation.getForObject(currentPhotoObjectThumb, photoParentObject), currentPhotoFilterThumb, currentPhotoObjectThumbStripped, 0, null, messageObject, cacheType); } } } else { if (messageObject.videoEditedInfo != null && messageObject.type == MessageObject.TYPE_ROUND_VIDEO && !currentMessageObject.needDrawBluredPreview()) { - photoImage.setImage(ImageLocation.getForPath(messageObject.videoEditedInfo.originalPath), currentPhotoFilter, ImageLocation.getForObject(currentPhotoObjectThumb, photoParentObject), currentPhotoFilterThumb, currentPhotoObjectThumbStripped, 0, null, messageObject, currentMessageObject.shouldEncryptPhotoOrVideo() ? 2 : 0); + photoImage.setImage(ImageLocation.getForPath(messageObject.videoEditedInfo.originalPath), currentPhotoFilter, ImageLocation.getForObject(currentPhotoObjectThumb, photoParentObject), currentPhotoFilterThumb, currentPhotoObjectThumbStripped, 0, null, messageObject, cacheType); photoImage.setMediaStartEndTime(currentMessageObject.videoEditedInfo.startTime / 1000, currentMessageObject.videoEditedInfo.endTime / 1000); } else { if (!messageIdChanged && !currentMessageObject.needDrawBluredPreview()) { photoImage.setCrossfadeWithOldImage(true); photoImage.setCrossfadeDuration(250); } - photoImage.setImage(ImageLocation.getForObject(currentPhotoObject, photoParentObject), currentPhotoFilter, ImageLocation.getForObject(currentPhotoObjectThumb, photoParentObject), currentPhotoFilterThumb, currentPhotoObjectThumbStripped, 0, null, messageObject, currentMessageObject.shouldEncryptPhotoOrVideo() ? 2 : 0); + photoImage.setImage(ImageLocation.getForObject(currentPhotoObject, photoParentObject), currentPhotoFilter, ImageLocation.getForObject(currentPhotoObjectThumb, photoParentObject), currentPhotoFilterThumb, currentPhotoObjectThumbStripped, 0, null, messageObject, cacheType); } } } @@ -8281,7 +8341,7 @@ private void setMessageContent(MessageObject messageObject, MessageObject.Groupe botButtonsByPosition.clear(); botButtonsLayout = null; } - if (!messageObject.isRestrictedMessage && currentPosition == null && (messageObject.messageOwner.reply_markup instanceof TLRPC.TL_replyInlineMarkup) && !messageObject.hasExtendedMedia()) { + if (!messageObject.isRestrictedMessage && !messageObject.isRepostPreview && currentPosition == null && (messageObject.messageOwner.reply_markup instanceof TLRPC.TL_replyInlineMarkup) && !messageObject.hasExtendedMedia()) { int rows; if (messageObject.messageOwner.reply_markup instanceof TLRPC.TL_replyInlineMarkup) { @@ -8761,6 +8821,7 @@ public void invalidate() { timePressed = false; gamePreviewPressed = false; giveawayMessageCell.setButtonPressed(false); + giveawayResultsMessageCell.setButtonPressed(false); if (pressedVoteButton != -1 || pollHintPressed || psaHintPressed || instantPressed || otherPressed || commentButtonPressed) { instantPressed = commentButtonPressed = false; @@ -8883,6 +8944,10 @@ public void invalidate() { if (currentMessageObject == null) { return; } + if (invalidateCallback != null) { + invalidateCallback.run(); + return; + } super.invalidate(); if ((invalidatesParent || currentMessagesGroup != null && invalidateParentForce()) && getParent() != null) { View parent = (View) getParent(); @@ -8897,12 +8962,21 @@ public void invalidate() { } } + private Runnable invalidateCallback; + public void setInvalidateCallback(Runnable callback) { + invalidateCallback = callback; + } + @Override public void invalidate(int l, int t, int r, int b) { if (currentMessageObject == null) { return; } + if (invalidateCallback != null) { + invalidateCallback.run(); + return; + } super.invalidate(l, t, r, b); if (invalidatesParent) { if (getParent() != null) { @@ -8996,6 +9070,16 @@ public void onSeekBarReleased() { requestDisallowInterceptTouchEvent(false); } + @Override + public boolean isSeekBarDragAllowed() { + return currentMessageObject == null || !currentMessageObject.isVoiceOnce(); + } + + @Override + public boolean reverseWaveform() { + return currentMessageObject != null && currentMessageObject.isVoiceOnce(); + } + @Override public void onSeekBarDrag(float progress) { if (currentMessageObject == null) { @@ -9045,7 +9129,7 @@ private void updateWaveform() { ) || MessagesController.getInstance(currentAccount).transcribeAudioTrialWeeklyNumber <= 0 && - !MessagesController.getInstance(currentAccount).premiumLocked && + !MessagesController.getInstance(currentAccount).premiumFeaturesBlocked() && !MessagesController.getInstance(currentAccount).didPressTranscribeButtonEnough() && !currentMessageObject.isOutOwner() && ( currentMessageObject.messageOwner != null && currentMessageObject.messageOwner.voiceTranscriptionForce || currentMessageObject.getDuration() >= 60 @@ -9053,7 +9137,8 @@ private void updateWaveform() { ) && ( currentMessageObject.isVoice() && useSeekBarWaveform || currentMessageObject.isRoundVideo() - ) && currentMessageObject.messageOwner != null && !(MessageObject.getMedia(currentMessageObject.messageOwner) instanceof TLRPC.TL_messageMediaWebPage) + ) && currentMessageObject.messageOwner != null && !(MessageObject.getMedia(currentMessageObject.messageOwner) instanceof TLRPC.TL_messageMediaWebPage) && + (currentMessageObject.messageOwner.media == null || currentMessageObject.messageOwner.media.ttl_seconds == 0) ); updateSeekBarWaveformWidth(null); } @@ -9278,7 +9363,7 @@ private void calcBackgroundWidth(int maxWidth, int timeMore, int maxChildWidth) if ( currentMessageObject.hasCodeAtBottom && (reactionsLayoutInBubble.isEmpty || reactionsLayoutInBubble.isSmall) || currentMessageObject.hasQuoteAtBottom && (reactionsLayoutInBubble.isEmpty || reactionsLayoutInBubble.isSmall) - || currentMessageObject.isGiveaway() + || currentMessageObject.isGiveawayOrGiveawayResults() ) { newLineForTime = true; newLineForTimeDp = 18; @@ -9611,7 +9696,7 @@ public void setIsUpdating(boolean value) { } public void setMessageObject(MessageObject messageObject, MessageObject.GroupedMessages groupedMessages, boolean bottomNear, boolean topNear) { - if (attachedToWindow) { + if (attachedToWindow && !frozen) { setMessageContent(messageObject, groupedMessages, bottomNear, topNear); } else { messageObjectToSet = messageObject; @@ -9621,6 +9706,17 @@ public void setMessageObject(MessageObject messageObject, MessageObject.GroupedM } } + private boolean frozen; + public void freezeCell(boolean freeze) { + this.frozen = freeze; + if (!frozen && messageObjectToSet != null && attachedToWindow) { + messageObjectToSet.animateComments = false; + setMessageContent(messageObjectToSet, groupedMessagesToSet, bottomNearToSet, topNearToSet); + messageObjectToSet = null; + groupedMessagesToSet = null; + } + } + private int getAdditionalWidthForPosition(MessageObject.GroupedMessagePosition position) { int w = 0; if (position != null) { @@ -9831,7 +9927,10 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { inLayout = false; } updateSelectionTextPosition(); - setMeasuredDimension(MeasureSpec.getSize(widthMeasureSpec), totalHeight + keyboardHeight); + setMeasuredDimension( + isWidthAdaptive() ? getBoundsRight() - getBoundsLeft() : MeasureSpec.getSize(widthMeasureSpec), + totalHeight + keyboardHeight + ); } public void forceResetMessageObject() { @@ -9907,7 +10006,7 @@ protected void onLayout(boolean changed, int left, int top, int right, int botto if (currentMessageObject.isAnyKindOfSticker()) { timeX = Math.max(AndroidUtilities.dp(26), timeX); } - if (isAvatarVisible) { + if (isAvatarVisible && !isWidthAdaptive()) { timeX += AndroidUtilities.dp(48); } if (currentPosition != null && currentPosition.leftSpanOffset != 0) { @@ -9928,7 +10027,7 @@ protected void onLayout(boolean changed, int left, int top, int right, int botto if (currentMessageObject.isAnyKindOfSticker()) { timeX = Math.max(0, timeX); } - if (isAvatarVisible) { + if (isAvatarVisible && !isWidthAdaptive()) { timeX += AndroidUtilities.dp(48); } if (shouldDrawTimeOnMedia()) { @@ -9951,7 +10050,7 @@ protected void onLayout(boolean changed, int left, int top, int right, int botto } if (isAvatarVisible) { - avatarImage.setImageCoords(AndroidUtilities.dp(6), avatarImage.getImageY(), AndroidUtilities.dp(42), AndroidUtilities.dp(42)); + avatarImage.setImageCoords(dp(currentMessageObject.isRepostPreview ? 15 : 6), avatarImage.getImageY(), dp(currentMessageObject.isRepostPreview ? 36 : 42), dp(currentMessageObject.isRepostPreview ? 36 : 42)); } if (currentMessageObject.type == MessageObject.TYPE_EXTENDED_MEDIA_PREVIEW && currentUnlockString != null) { @@ -9980,7 +10079,7 @@ protected void onLayout(boolean changed, int left, int top, int right, int botto } else if (linkPreviewAbove) { linkPreviewY = textY + AndroidUtilities.dp(10); textY += linkPreviewHeight + AndroidUtilities.dp(13); - if (drawInstantView && !hasInvoicePreview && !currentMessageObject.isGiveaway()) { + if (drawInstantView && !hasInvoicePreview && !currentMessageObject.isGiveawayOrGiveawayResults()) { textY += AndroidUtilities.dp(44); } } else { @@ -9995,16 +10094,14 @@ protected void onLayout(boolean changed, int left, int top, int right, int botto seekBarX = layoutWidth - backgroundWidth + AndroidUtilities.dp(57); buttonX = layoutWidth - backgroundWidth + AndroidUtilities.dp(14); timeAudioX = layoutWidth - backgroundWidth + AndroidUtilities.dp(67); + } else if (needDrawAvatar()) { + seekBarX = AndroidUtilities.dp(114); + buttonX = AndroidUtilities.dp(71); + timeAudioX = AndroidUtilities.dp(124); } else { - if ((isChat && !isThreadPost || currentMessageObject.forceAvatar) && currentMessageObject.needDrawAvatar()) { - seekBarX = AndroidUtilities.dp(114); - buttonX = AndroidUtilities.dp(71); - timeAudioX = AndroidUtilities.dp(124); - } else { - seekBarX = AndroidUtilities.dp(66); - buttonX = AndroidUtilities.dp(23); - timeAudioX = AndroidUtilities.dp(76); - } + seekBarX = AndroidUtilities.dp(66); + buttonX = AndroidUtilities.dp(23); + timeAudioX = AndroidUtilities.dp(76); } if (hasLinkPreview) { seekBarX += AndroidUtilities.dp(10); @@ -10077,7 +10174,7 @@ protected void onLayout(boolean changed, int left, int top, int right, int botto buttonX = layoutWidth - backgroundWidth + AndroidUtilities.dp(14); timeAudioX = layoutWidth - backgroundWidth + AndroidUtilities.dp(67); } else { - if ((isChat && !isThreadPost || currentMessageObject.forceAvatar) && currentMessageObject.needDrawAvatar()) { + if (needDrawAvatar()) { seekBarX = AndroidUtilities.dp(113); buttonX = AndroidUtilities.dp(71); timeAudioX = AndroidUtilities.dp(124); @@ -10102,7 +10199,7 @@ protected void onLayout(boolean changed, int left, int top, int right, int botto if (currentMessageObject.isOutOwner()) { buttonX = layoutWidth - backgroundWidth + AndroidUtilities.dp(14); } else { - if ((isChat && !isThreadPost || currentMessageObject.forceAvatar) && currentMessageObject.needDrawAvatar()) { + if (needDrawAvatar()) { buttonX = AndroidUtilities.dp(71); } else { buttonX = AndroidUtilities.dp(23); @@ -10118,7 +10215,7 @@ protected void onLayout(boolean changed, int left, int top, int right, int botto int x; if (currentMessageObject.isOutOwner()) { x = layoutWidth - backgroundWidth + AndroidUtilities.dp(14); - } else if ((isChat && !isThreadPost || currentMessageObject.forceAvatar) && currentMessageObject.needDrawAvatar()) { + } else if (needDrawAvatar()) { x = AndroidUtilities.dp(72); } else { x = AndroidUtilities.dp(23); @@ -10148,7 +10245,7 @@ protected void onLayout(boolean changed, int left, int top, int right, int botto x = layoutWidth - backgroundWidth + AndroidUtilities.dp(6); } } else { - if (isChat && isAvatarVisible && !isPlayingRound) { + if ((isChat || currentMessageObject.isRepostPreview) && isAvatarVisible && !isPlayingRound) { x = AndroidUtilities.dp(63); } else { x = AndroidUtilities.dp(15); @@ -10189,7 +10286,7 @@ protected void onLayout(boolean changed, int left, int top, int right, int botto } public boolean needDelayRoundProgressDraw() { - return (documentAttachType == DOCUMENT_ATTACH_TYPE_ROUND || documentAttachType == DOCUMENT_ATTACH_TYPE_VIDEO) && currentMessageObject.type != MessageObject.TYPE_ROUND_VIDEO && MediaController.getInstance().isPlayingMessage(currentMessageObject); + return (documentAttachType == DOCUMENT_ATTACH_TYPE_ROUND || documentAttachType == DOCUMENT_ATTACH_TYPE_VIDEO) && currentMessageObject != null && currentMessageObject.type != MessageObject.TYPE_ROUND_VIDEO && MediaController.getInstance().isPlayingMessage(currentMessageObject); } public void drawRoundProgress(Canvas canvas) { @@ -10419,7 +10516,7 @@ public void drawContent(Canvas canvas, boolean preview) { imageDrawn = false; radialProgress.setCircleCrossfadeColor(-1, 0.0f, 1.0f); - if (currentMessageObject.type == MessageObject.TYPE_TEXT || currentMessageObject.type == MessageObject.TYPE_STORY_MENTION || currentMessageObject.type == MessageObject.TYPE_EMOJIS || currentMessageObject.isGiveaway()) { + if (currentMessageObject.type == MessageObject.TYPE_TEXT || currentMessageObject.type == MessageObject.TYPE_STORY_MENTION || currentMessageObject.type == MessageObject.TYPE_EMOJIS || currentMessageObject.isGiveawayOrGiveawayResults()) { layoutTextXY(false); if (!enterTransitionInProgress && currentMessageObject != null && !currentMessageObject.preview) { if (!drawForBlur && animatedEmojiStack != null && ((currentMessageObject.textLayoutBlocks != null && !currentMessageObject.textLayoutBlocks.isEmpty()) || (transitionParams.animateOutTextBlocks != null && !transitionParams.animateOutTextBlocks.isEmpty()))) { @@ -10512,7 +10609,7 @@ public void drawContent(Canvas canvas, boolean preview) { canvas.save(); canvas.scale(-1f, 1, photoImage.getCenterX(), photoImage.getCenterY()); if (allowDrawPhotoImage()) { - imageDrawn = photoImage.draw(canvas); + imageDrawn = drawPhotoImage(canvas); } else { imageDrawn = true; } @@ -10522,7 +10619,7 @@ public void drawContent(Canvas canvas, boolean preview) { canvas.restore(); } else { if (allowDrawPhotoImage()) { - imageDrawn = photoImage.draw(canvas); + imageDrawn = drawPhotoImage(canvas); } else { imageDrawn = true; } @@ -10587,7 +10684,7 @@ public void drawContent(Canvas canvas, boolean preview) { } } if (documentAttachType == DOCUMENT_ATTACH_TYPE_GIF) { - if (drawPhotoImage && photoImage.getVisible() && !hasGamePreview && !currentMessageObject.needDrawBluredPreview() && !currentMessageObject.preview && !isSmallImage) { + if (drawPhotoImage && !currentMessageObject.isRepostPreview && photoImage.getVisible() && !hasGamePreview && !currentMessageObject.needDrawBluredPreview() && !currentMessageObject.preview && !isSmallImage) { int oldAlpha = ((BitmapDrawable) Theme.chat_msgMediaMenuDrawable).getPaint().getAlpha(); Theme.chat_msgMediaMenuDrawable.setAlpha((int) (oldAlpha * controlsAlpha)); setDrawableBounds(Theme.chat_msgMediaMenuDrawable, otherX = (int) (photoImage.getImageX() + photoImage.getImageWidth() - AndroidUtilities.dp(14)), otherY = (int) (photoImage.getImageY() + AndroidUtilities.dp(8.1f))); @@ -10615,6 +10712,7 @@ public void drawContent(Canvas canvas, boolean preview) { } radialProgress.setBackgroundDrawable(isDrawSelectionBackground() ? currentBackgroundSelectedDrawable : currentBackgroundDrawable); + radialProgress.iconScale = 1f; radialProgress.draw(canvas); canvas.save(); @@ -10698,7 +10796,9 @@ public void drawContent(Canvas canvas, boolean preview) { radialProgress.setProgressColor(getThemedColor(isDrawSelectionBackground() || buttonPressed != 0 ? Theme.key_chat_inAudioSelectedProgress : Theme.key_chat_inAudioProgress)); } AudioVisualizerDrawable audioVisualizerDrawable; - if (MediaController.getInstance().isPlayingMessage(currentMessageObject)) { + if (overridenAudioVisualizer != null) { + audioVisualizerDrawable = overridenAudioVisualizer; + } else if (MediaController.getInstance().isPlayingMessage(currentMessageObject)) { audioVisualizerDrawable = Theme.getCurrentAudiVisualizerDrawable(); } else { audioVisualizerDrawable = Theme.getAnimatedOutAudioVisualizerDrawable(currentMessageObject); @@ -10715,7 +10815,8 @@ public void drawContent(Canvas canvas, boolean preview) { if (!enterTransitionInProgress && documentAttachType == DOCUMENT_ATTACH_TYPE_AUDIO) { radialProgress.setBackgroundDrawable(isDrawSelectionBackground() ? currentBackgroundSelectedDrawable : currentBackgroundDrawable); - radialProgress.draw(canvas); + radialProgress.iconScale = 1f; + drawVoiceOnce(canvas, seekBarWaveform == null ? 1f : 1f - seekBarWaveform.explosionRate, () -> radialProgress.draw(canvas)); } int seekBarX = this.seekBarX; @@ -11128,6 +11229,8 @@ public void onAnimationEnd(Animator animation) { animator.start(); } + public boolean drawingToBitmap; + private void drawBlurredPhoto(Canvas canvas) { if (currentMessageObject.isMediaSpoilersRevealed || mediaSpoilerRevealProgress == 1f) { return; @@ -11162,7 +11265,7 @@ private void drawBlurredPhoto(Canvas canvas) { if (mediaSpoilerEffect2 != null) { canvas.translate(photoImage.getImageX(), photoImage.getImageY()); - mediaSpoilerEffect2.draw(canvas, this, (int) photoImage.getImageWidth(), (int) photoImage.getImageHeight(), photoImage.getAlpha()); + mediaSpoilerEffect2.draw(canvas, this, (int) photoImage.getImageWidth(), (int) photoImage.getImageHeight(), photoImage.getAlpha(), drawingToBitmap); canvas.restore(); } else { int sColor = Color.WHITE; @@ -11187,6 +11290,9 @@ private float getUseTranscribeButtonProgress() { } private void updateReactionLayoutPosition() { + if (!currentMessageObject.shouldDrawReactions()) { + return; + } if (!reactionsLayoutInBubble.isEmpty && (currentPosition == null || ((currentPosition.flags & MessageObject.POSITION_FLAG_BOTTOM) != 0 && (currentPosition.flags & MessageObject.POSITION_FLAG_LEFT) != 0)) && !reactionsLayoutInBubble.isSmall) { if (currentMessageObject.type == MessageObject.TYPE_EMOJIS || currentMessageObject.isAnimatedEmoji() || currentMessageObject.isAnyKindOfSticker()) { if (currentMessageObject.isOutOwner()) { @@ -11239,6 +11345,99 @@ private void updateReactionLayoutPosition() { } } + public void drawVoiceOnce(Canvas canvas, float progress, Runnable drawRadialProgress) { + if (currentMessageObject != null && currentMessageObject.isVoiceOnce()) { + final float scale = progress; + final float rcx = radialProgress.progressRect.centerX() + (float) Math.cos(AndroidUtilities.lerp(190, 45, scale) / 180f * Math.PI) * dp(22.6274f); + final float rcy = radialProgress.progressRect.centerY() + (float) Math.sin(AndroidUtilities.lerp(190, 45, scale) / 180f * Math.PI) * dp(22.6274f); + + AndroidUtilities.rectTmp.set(radialProgress.progressRect); + AndroidUtilities.rectTmp.inset(-dp(1), -dp(1)); + canvas.saveLayerAlpha(AndroidUtilities.rectTmp, 0xFF, Canvas.ALL_SAVE_FLAG); + radialProgress.setBackgroundDrawable(isDrawSelectionBackground() ? currentBackgroundSelectedDrawable : currentBackgroundDrawable); + radialProgress.iconScale = scale; + drawRadialProgress.run(); + if (onceClearPaint == null) { + onceClearPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + onceClearPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.CLEAR)); + } + if (scale < 1f) { + canvas.saveLayerAlpha(radialProgress.progressRect, 0xFF, Canvas.ALL_SAVE_FLAG); + final float s = (1f - scale) * .7f; + canvas.scale(s, s, radialProgress.progressRect.centerX(), AndroidUtilities.lerp(radialProgress.progressRect.top, radialProgress.progressRect.bottom, .5f)); + if (onceFire == null) { + onceFire = new RLottieDrawable(R.raw.fire_once, "fire_once", dp(32), dp(32), true, null); + onceFire.setMasterParent(this); + onceFire.setAllowDecodeSingleFrame(true); + onceFire.setAutoRepeat(1); + onceFire.start(); + } + onceFire.setBounds( + (int) radialProgress.progressRect.left, + (int) radialProgress.progressRect.top, + (int) radialProgress.progressRect.right, + (int) radialProgress.progressRect.bottom + ); + if (onceRadialPaint == null) { + onceRadialPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + } + if (onceRadialStrokePaint == null) { + onceRadialStrokePaint = new Paint(Paint.ANTI_ALIAS_FLAG); + onceRadialStrokePaint.setStyle(Paint.Style.STROKE); + } + int iconColor = radialProgress.iconColorKey >= 0 ? getThemedColor(radialProgress.iconColorKey) : radialProgress.iconColor; + if (onceRadialPaintColor != iconColor) { + onceRadialPaint.setColorFilter(new PorterDuffColorFilter(onceRadialPaintColor = iconColor, PorterDuff.Mode.SRC_IN)); + onceRadialStrokePaint.setColorFilter(new PorterDuffColorFilter(onceRadialPaintColor, PorterDuff.Mode.SRC_IN)); + } + radialProgress.mediaActionDrawable.applyShaderMatrix(false); + onceRadialPaint.setShader(radialProgress.mediaActionDrawable.paint2.getShader()); + onceRadialStrokePaint.setShader(radialProgress.mediaActionDrawable.paint2.getShader()); + onceFire.draw(canvas, onceRadialPaint); + canvas.restore(); + + onceRadialStrokePaint.setAlpha((int) (0xFF * (1f - scale))); + onceRadialStrokePaint.setStrokeWidth(dp(1.66f)); + rect.set(radialProgress.progressRect); + rect.inset(dp(3), dp(3)); + canvas.drawArc(rect, -90, -360 * (1f - seekBarWaveform.explodeProgress), false, onceRadialStrokePaint); + if (timerParticles == null) { + timerParticles = new TimerParticles(); + } + timerParticles.draw(canvas, onceRadialStrokePaint, rect, -360 * (1f - seekBarWaveform.explodeProgress), 1f - scale); + } else if (onceFire != null) { + onceFire.recycle(true); + onceFire = null; + if (timerParticles != null) { + timerParticles = null; + } + } + canvas.drawCircle(rcx, rcy, dp(10 + scale * 1.5f) * scale, onceClearPaint); + canvas.restore(); + + if (oncePeriod == null) { + oncePeriod = new CaptionContainerView.PeriodDrawable(3); + oncePeriod.updateColors(0xffffffff, 0, 0); + oncePeriod.diameterDp = 14; + oncePeriod.setTextSize(10); + oncePeriod.strokePaint.setStrokeWidth(dpf2(1.5f)); + oncePeriod.setValue(1, false, false); + oncePeriod.textOffsetX = -dpf2(.33f); + oncePeriod.textOffsetY = dpf2(.33f); + } + oncePeriod.diameterDp = 14 * scale; + oncePeriod.setTextSize(10 * scale); + canvas.saveLayerAlpha(rcx - dp(10), rcy - dp(10), rcx + dp(10), rcy + dp(10), 0xFF, Canvas.ALL_SAVE_FLAG); + canvas.drawCircle(rcx, rcy, dp(10) * scale, radialProgress.circlePaint); + oncePeriod.setClear(AndroidUtilities.computePerceivedBrightness(radialProgress.circlePaint.getColor()) > .8f); + oncePeriod.setCenterXY(rcx, rcy); + oncePeriod.draw(canvas, scale); + canvas.restore(); + } else { + drawRadialProgress.run(); + } + } + public void drawLinkPreview(Canvas canvas, float alpha) { if (!currentMessageObject.isSponsored() && !hasLinkPreview && !hasGamePreview && !hasInvoicePreview) { return; @@ -11298,10 +11497,13 @@ public void drawLinkPreview(Canvas canvas, float alpha) { } Theme.chat_replyNamePaint.setColor(linkLine.check(currentMessageObject, currentUser, currentChat, resourcesProvider, ReplyMessageLine.TYPE_LINK)); + final boolean drawPhotoImageBefore = drawInstantView && (drawInstantViewType != 9 && drawInstantViewType != 2 && drawInstantViewType != 13 && drawInstantViewType != 11 && drawInstantViewType != 1 && drawInstantViewType != 18) || drawInstantViewType == 6 && imageBackgroundColor != 0; + final boolean drawPhotoImageAfter = !drawInstantView || drawInstantViewType == 9 || drawInstantViewType == 2 || drawInstantViewType == 11 || drawInstantViewType == 13 || drawInstantViewType == 1 || drawInstantViewType == 18 || isSmallImage; + boolean restore = false; boolean drawInstantButtonInside = false; boolean loading = delegate != null && delegate.isProgressLoading(this, ChatActivity.PROGRESS_INSTANT); - if (!hasInvoicePreview && !currentMessageObject.isGiveaway()) { + if (!hasInvoicePreview && !currentMessageObject.isGiveawayOrGiveawayResults()) { drawInstantButtonInside = true; if (linkPreviewBounce == null) { @@ -11315,7 +11517,9 @@ public void drawLinkPreview(Canvas canvas, float alpha) { AndroidUtilities.rectTmp.set(linkX, linkPreviewY - AndroidUtilities.dp(6), linkX + width, linkPreviewY + linkPreviewHeight + (drawInstantButtonInside && drawInstantView ? AndroidUtilities.dp(42) : 0)); linkLine.setLoading(loading); float rad = (float) Math.floor(SharedConfig.bubbleRadius / (currentMessageObject.isSponsored() ? 2f : 3f)); - linkLine.drawBackground(canvas, AndroidUtilities.rectTmp, rad, rad, rad, alpha); + linkLine + .offsetEmoji(0, drawPhotoImageBefore ? (1f - isSmallImage()) * (dp(18) + photoImage.getImageHeight() + (siteNameLayout != null ? siteNameLayout.getLineBottom(siteNameLayout.getLineCount() - 1) : 0)) : 0) + .drawBackground(canvas, AndroidUtilities.rectTmp, rad, rad, rad, alpha); int rippleColor = linkLine.getBackgroundColor(); if (linkPreviewSelector == null) { @@ -11351,7 +11555,7 @@ public void drawLinkPreview(Canvas canvas, float alpha) { linkPreviewY += currentMessageObject.textHeight + AndroidUtilities.dp(4); } - if (drawPhotoImage && drawInstantView && (drawInstantViewType != 9 && drawInstantViewType != 2 && drawInstantViewType != 13 && drawInstantViewType != 11 && drawInstantViewType != 1 && drawInstantViewType != 18) || drawInstantViewType == 6 && imageBackgroundColor != 0) { + if (drawPhotoImage && drawPhotoImageBefore) { if (isSmallImage) { if (transitionParams != null && transitionParams.animateSmallImage) { float diff = (linkPreviewY != startY ? AndroidUtilities.dp(2) : 0) + transitionParams.photoImageFromHeight + AndroidUtilities.dp(6); @@ -11579,7 +11783,7 @@ public void drawLinkPreview(Canvas canvas, float alpha) { linkPreviewY += descriptionLayout.getLineBottom(descriptionLayout.getLineCount() - 1); } - if (drawPhotoImage && (!drawInstantView || drawInstantViewType == 9 || drawInstantViewType == 2 || drawInstantViewType == 11 || drawInstantViewType == 13 || drawInstantViewType == 1 || drawInstantViewType == 18 || isSmallImage)) { + if (drawPhotoImage && drawPhotoImageAfter) { if (linkPreviewY != startY) { linkPreviewY += AndroidUtilities.dp(2); } @@ -11697,6 +11901,7 @@ public void drawLinkPreview(Canvas canvas, float alpha) { canvas.restore(); } giveawayMessageCell.draw(canvas, namesOffset, linkX, resourcesProvider); + giveawayResultsMessageCell.draw(canvas, namesOffset, linkX, resourcesProvider); if (drawInstantView) { Drawable instantDrawable; @@ -11836,12 +12041,12 @@ private float isSmallImage() { } private boolean shouldDrawMenuDrawable() { - return (currentMessagesGroup == null || (currentPosition.flags & MessageObject.POSITION_FLAG_TOP) != 0) && !hasLinkPreview; + return (currentMessagesGroup == null || (currentPosition.flags & MessageObject.POSITION_FLAG_TOP) != 0) && !hasLinkPreview && (currentMessageObject == null || !currentMessageObject.isRepostPreview); } private void drawBotButtons(Canvas canvas, ArrayList botButtons, int alpha) { int addX; - if (currentMessageObject.isOutOwner()) { + if (currentMessageObject != null && currentMessageObject.isOutOwner()) { addX = getMeasuredWidth() - widthForButtons - AndroidUtilities.dp(10); } else { addX = backgroundDrawableLeft + AndroidUtilities.dp(mediaBackground || drawPinnedBottom ? 1 : 7); @@ -12008,7 +12213,7 @@ public void layoutTextXY(boolean parent) { } else if (linkPreviewAbove) { linkPreviewY = textY + AndroidUtilities.dp(10); textY += linkPreviewHeight + AndroidUtilities.dp(13); - if (drawInstantView && !hasInvoicePreview && !currentMessageObject.isGiveaway()) { + if (drawInstantView && !hasInvoicePreview && !currentMessageObject.isGiveawayOrGiveawayResults()) { textY += AndroidUtilities.dp(44); } } else { @@ -12307,7 +12512,7 @@ public void drawMessageText(float textX, float textY, Canvas canvas, ArrayList 0) { @@ -13106,14 +13312,14 @@ private void didPressButton(boolean animated, boolean video) { createLoadingProgressLayout(documentAttach); } } else if (documentAttachType == DOCUMENT_ATTACH_TYPE_VIDEO) { - FileLoader.getInstance(currentAccount).loadFile(documentAttach, currentMessageObject, FileLoader.PRIORITY_NORMAL, currentMessageObject.shouldEncryptPhotoOrVideo() ? 2 : 0); + FileLoader.getInstance(currentAccount).loadFile(documentAttach, currentMessageObject, FileLoader.PRIORITY_NORMAL, cacheType); if (currentMessageObject.loadedFileSize > 0) { createLoadingProgressLayout(currentMessageObject.getDocument()); } } else if (currentMessageObject.type == MessageObject.TYPE_TEXT && documentAttachType != DOCUMENT_ATTACH_TYPE_NONE) { if (documentAttachType == DOCUMENT_ATTACH_TYPE_GIF) { photoImage.setForceLoading(true); - photoImage.setImage(ImageLocation.getForDocument(documentAttach), null, ImageLocation.getForDocument(currentPhotoObject, documentAttach), currentPhotoFilterThumb, documentAttach.size, null, currentMessageObject, 0); + photoImage.setImage(ImageLocation.getForDocument(documentAttach), null, ImageLocation.getForDocument(currentPhotoObject, documentAttach), currentPhotoFilterThumb, documentAttach.size, null, currentMessageObject, cacheType); currentMessageObject.gifState = 2; if (currentMessageObject.loadedFileSize > 0) { createLoadingProgressLayout(currentMessageObject.getDocument()); @@ -13188,7 +13394,7 @@ private void didPressButton(boolean animated, boolean video) { FileLoader.getInstance(currentAccount).loadFile(documentAttach, currentMessageObject, FileLoader.PRIORITY_NORMAL_UP, 0); currentMessageObject.loadingCancelled = false; } - if (delegate.needPlayMessage(currentMessageObject, false)) { + if (delegate.needPlayMessage(this, currentMessageObject, false)) { if (hasMiniProgress == 2 && miniButtonState != 1) { miniButtonState = 1; radialProgress.setProgress(0, false); @@ -13203,7 +13409,7 @@ private void didPressButton(boolean animated, boolean video) { wouldBeInPip = true; ChatMessageCell.this.invalidate(); } - } else if (documentAttachType == DOCUMENT_ATTACH_TYPE_AUDIO || documentAttachType == DOCUMENT_ATTACH_TYPE_MUSIC) { + } else if (documentAttachType == DOCUMENT_ATTACH_TYPE_AUDIO || documentAttachType == DOCUMENT_ATTACH_TYPE_MUSIC) { //asdf1 radialProgress.setProgress(0, false); FileLoader.getInstance(currentAccount).loadFile(documentAttach, currentMessageObject, FileLoader.PRIORITY_NORMAL_UP, 0); currentMessageObject.loadingCancelled = false; @@ -13231,9 +13437,11 @@ private void didPressButton(boolean animated, boolean video) { radialProgress.setProgress(0, false); radialProgress.setMiniIcon(getMiniIconForCurrentState(), false, animated); } - delegate.didPressImage(this, 0, 0); + if (delegate != null) { + delegate.didPressImage(this, 0, 0); + } } else if (buttonState == 4) { - if (documentAttachType == DOCUMENT_ATTACH_TYPE_AUDIO || documentAttachType == DOCUMENT_ATTACH_TYPE_MUSIC || documentAttachType == DOCUMENT_ATTACH_TYPE_ROUND && currentMessageObject != null && currentMessageObject.isVoiceTranscriptionOpen()) { + if (documentAttachType == DOCUMENT_ATTACH_TYPE_AUDIO || documentAttachType == DOCUMENT_ATTACH_TYPE_MUSIC || documentAttachType == DOCUMENT_ATTACH_TYPE_ROUND && currentMessageObject != null && currentMessageObject.isVoiceTranscriptionOpen()) { // asdf2 if (currentMessageObject.isOut() && (currentMessageObject.isSending() || currentMessageObject.isEditing()) || currentMessageObject.isSendError()) { if (delegate != null && radialProgress.getIcon() != MediaActionDrawable.ICON_CHECK) { delegate.didPressCancelSendButton(this); @@ -13274,7 +13482,7 @@ public void onSuccessDownload(String fileName) { photoImage.setAllowStartAnimation(true); photoImage.startAnimation(); autoPlayingMedia = true; - } else if (!isSmallImage && SharedConfig.isAutoplayVideo() && documentAttachType == DOCUMENT_ATTACH_TYPE_VIDEO && (currentPosition == null || (currentPosition.flags & MessageObject.POSITION_FLAG_LEFT) != 0 && (currentPosition.flags & MessageObject.POSITION_FLAG_RIGHT) != 0)) { + } else if (!isSmallImage && SharedConfig.isAutoplayVideo() && !currentMessageObject.isRepostPreview && documentAttachType == DOCUMENT_ATTACH_TYPE_VIDEO && (currentPosition == null || (currentPosition.flags & MessageObject.POSITION_FLAG_LEFT) != 0 && (currentPosition.flags & MessageObject.POSITION_FLAG_RIGHT) != 0)) { animatingNoSound = 2; photoImage.setImage(ImageLocation.getForDocument(documentAttach), ImageLoader.AUTOPLAY_FILTER, ImageLocation.getForObject(currentPhotoObject, photoParentObject), currentPhotoObject instanceof TLRPC.TL_photoStrippedSize || currentPhotoObject != null && "s".equals(currentPhotoObject.type) ? currentPhotoFilterThumb : currentPhotoFilter, ImageLocation.getForObject(currentPhotoObjectThumb, photoParentObject), currentPhotoFilterThumb, currentPhotoObjectThumbStripped, documentAttach.size, null, currentMessageObject, 0); if (!PhotoViewer.isPlayingMessage(currentMessageObject)) { @@ -13286,7 +13494,7 @@ public void onSuccessDownload(String fileName) { autoPlayingMedia = true; } else if (documentAttachType == DOCUMENT_ATTACH_TYPE_GIF) { photoImage.setImage(ImageLocation.getForDocument(documentAttach), ImageLoader.AUTOPLAY_FILTER, ImageLocation.getForObject(currentPhotoObject, photoParentObject), currentPhotoObject instanceof TLRPC.TL_photoStrippedSize || currentPhotoObject != null && "s".equals(currentPhotoObject.type) ? currentPhotoFilterThumb : currentPhotoFilter, ImageLocation.getForObject(currentPhotoObjectThumb, photoParentObject), currentPhotoFilterThumb, currentPhotoObjectThumbStripped, documentAttach.size, null, currentMessageObject, 0); - if (SharedConfig.isAutoplayGifs()) { + if (SharedConfig.isAutoplayGifs() && !currentMessageObject.isRepostPreview) { photoImage.setAllowStartAnimation(true); photoImage.startAnimation(); } else { @@ -13548,9 +13756,9 @@ private void createLoadingProgressLayout(long loadedSize, long totalSize) { public void onProvideStructure(ViewStructure structure) { super.onProvideStructure(structure); if (allowAssistant && Build.VERSION.SDK_INT >= 23) { - if (currentMessageObject.messageText != null && currentMessageObject.messageText.length() > 0) { + if (currentMessageObject != null && currentMessageObject.messageText != null && currentMessageObject.messageText.length() > 0) { structure.setText(currentMessageObject.messageText); - } else if (currentMessageObject.caption != null && currentMessageObject.caption.length() > 0) { + } else if (currentMessageObject != null && currentMessageObject.caption != null && currentMessageObject.caption.length() > 0) { structure.setText(currentMessageObject.caption); } } @@ -13624,6 +13832,8 @@ private void measureTime(MessageObject messageObject) { timeString = ""; } else if (currentMessageObject.scheduled && currentMessageObject.messageOwner.date == 0x7FFFFFFE) { timeString = ""; + } else if (currentMessageObject.isRepostPreview) { + timeString = LocaleController.formatSmallDateChat(messageObject.messageOwner.date) + ", " + LocaleController.getInstance().formatterDay.format((long) (messageObject.messageOwner.date) * 1000); } else if (edited) { timeString = LocaleController.getString("EditedMessage", R.string.EditedMessage) + " " + LocaleController.getInstance().formatterDay.format((long) (messageObject.messageOwner.date) * 1000); } else { @@ -13671,7 +13881,7 @@ private void measureTime(MessageObject messageObject) { timeWidth += AndroidUtilities.dp(18); } } - if (reactionsLayoutInBubble.isSmall) { + if (currentMessageObject.shouldDrawReactions() && reactionsLayoutInBubble.isSmall) { reactionsLayoutInBubble.measure(Integer.MAX_VALUE, Gravity.LEFT); timeWidth += reactionsLayoutInBubble.width; } @@ -13734,8 +13944,8 @@ private boolean isOpenChatByShare(MessageObject messageObject) { return messageObject.messageOwner.fwd_from != null && messageObject.messageOwner.fwd_from.saved_from_peer != null && (delegate == null || delegate.isReplyOrSelf()); } - private boolean checkNeedDrawShareButton(MessageObject messageObject) { - if (currentMessageObject.deleted || currentMessageObject.isSponsored()) { + protected boolean checkNeedDrawShareButton(MessageObject messageObject) { + if (currentMessageObject.deleted && !currentMessageObject.deletedByThanos || currentMessageObject.isSponsored()) { return false; } if (currentPosition != null) { @@ -14106,7 +14316,7 @@ protected void onClick() { topicButton = null; } - if ((!isThreadChat || messageObject.getReplyTopMsgId(isForum) != 0 || isForumGeneral) && messageObject.hasValidReplyMessageObject() || messageObject.messageOwner.fwd_from != null && messageObject.isDice() || (messageObject.messageOwner.reply_to != null && (messageObject.messageOwner.reply_to.story_id != 0 || !TextUtils.isEmpty(messageObject.messageOwner.reply_to.quote_text) || messageObject.messageOwner.reply_to.reply_from != null))) { + if (!messageObject.isGiveawayResults() && (!isThreadChat || messageObject.getReplyTopMsgId(isForum) != 0 || isForumGeneral) && messageObject.hasValidReplyMessageObject() || messageObject.messageOwner.fwd_from != null && messageObject.isDice() || (messageObject.messageOwner.reply_to != null && (messageObject.messageOwner.reply_to.story_id != 0 || !TextUtils.isEmpty(messageObject.messageOwner.reply_to.quote_text) || messageObject.messageOwner.reply_to.reply_from != null))) { if (currentPosition == null || currentPosition.minY == 0) { if (!messageObject.isAnyKindOfSticker() && messageObject.type != MessageObject.TYPE_ROUND_VIDEO || messageObject.type == MessageObject.TYPE_EMOJIS) { namesOffset += AndroidUtilities.dp(20) + (Theme.chat_replyTextPaint.getTextSize() + Theme.chat_replyNamePaint.getTextSize()); @@ -14202,7 +14412,7 @@ protected void onClick() { currentReplyPhoto = photoSize; replyImageReceiver.setImage(ImageLocation.getForObject(photoSize, photoObject), hasReplySpoiler ? "5_5_b" : "50_50", ImageLocation.getForObject(thumbPhotoSize, photoObject), hasReplySpoiler ? "50_50_b4" : "50_50_b", size, null, messageObject, cacheType); needReplyImage = true; - maxWidth -= AndroidUtilities.dp(35); + maxWidth -= AndroidUtilities.dp(isReplyQuote ? 3 : 7) + Theme.chat_replyNamePaint.getTextSize() + Theme.chat_replyTextPaint.getTextSize(); } } else if (photoSize == null || messageObject.replyMessageObject == null || messageObject.replyMessageObject.isAnyKindOfSticker() || messageObject.isAnyKindOfSticker() && !AndroidUtilities.isTablet() || messageObject.replyMessageObject.isSecretMedia() || messageObject.replyMessageObject.isWebpageDocument()) { replyImageReceiver.setImageBitmap((Drawable) null); @@ -14216,7 +14426,7 @@ protected void onClick() { currentReplyPhoto = photoSize; replyImageReceiver.setImage(ImageLocation.getForObject(photoSize, photoObject), hasReplySpoiler ? "5_5_b" : "50_50", ImageLocation.getForObject(thumbPhotoSize, photoObject), hasReplySpoiler ? "50_50_b4" : "50_50_b", size, null, messageObject.replyMessageObject, cacheType); needReplyImage = true; - maxWidth -= AndroidUtilities.dp(35); + maxWidth -= AndroidUtilities.dp(isReplyQuote ? 3 : 7) + Theme.chat_replyNamePaint.getTextSize() + Theme.chat_replyTextPaint.getTextSize(); } if (DialogObject.isEncryptedDialog(messageObject.getDialogId())) { @@ -14385,7 +14595,7 @@ protected void onClick() { if (stringFinalName != null) { replyNameLayout = new StaticLayout(stringFinalName, Theme.chat_replyNamePaint, maxWidth + AndroidUtilities.dp(6), Layout.Alignment.ALIGN_NORMAL, 1.0f, 0.0f, false); if (replyNameLayout.getLineCount() > 0) { - replyNameWidth += (int) Math.ceil(replyNameLayout.getLineWidth(0)) + AndroidUtilities.dp(8); + replyNameWidth += (int) Math.ceil(replyNameLayout.getLineWidth(0)) + AndroidUtilities.dp(4); replyNameOffset = (int) replyNameLayout.getLineLeft(0); } } @@ -14410,15 +14620,16 @@ protected void onClick() { } replyTextRTL = AndroidUtilities.isRTL(sb); if (isReplyQuote) { -// maxWidth += AndroidUtilities.dp(12); + maxWidth += AndroidUtilities.dp(24 + 12); // replyTextWidth += AndroidUtilities.dp(24); } if (isReplyQuote && needReplyImage && !replyTextRTL) { final float sz = AndroidUtilities.dp(isReplyQuote ? 3 : 7) + Theme.chat_replyNamePaint.getTextSize() + Theme.chat_replyTextPaint.getTextSize(); sb.setSpan(new LeadingMarginSpan.Standard((int) sz + AndroidUtilities.dp(4), 0), 0, sb.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); replyTextWidth -= sz; + maxWidth += sz; } - if (!isReplyQuote || currentMessageObject.shouldDrawWithoutBackground() || Build.VERSION.SDK_INT < Build.VERSION_CODES.M) { + if (!isReplyQuote && (currentMessageObject.shouldDrawWithoutBackground() || Build.VERSION.SDK_INT < Build.VERSION_CODES.M)) { stringFinalText = TextUtils.ellipsize(sb, textPaint, maxWidth, TextUtils.TruncateAt.END); } else { stringFinalText = sb; @@ -14471,7 +14682,7 @@ protected void onClick() { FileLog.e(e); } } - } else if (!isThreadChat && messageObject.getReplyMsgId() != 0) { + } else if (!isThreadChat && messageObject.getReplyMsgId() != 0 && !messageObject.isGiveawayResults()) { if (!(messageObject.replyMessageObject != null && (messageObject.replyMessageObject.messageOwner instanceof TLRPC.TL_messageEmpty || messageObject.replyMessageObject.messageOwner != null && messageObject.replyMessageObject.messageOwner.action instanceof TLRPC.TL_messageActionTopicCreate))) { if (!messageObject.isAnyKindOfSticker() && messageObject.type != MessageObject.TYPE_ROUND_VIDEO) { namesOffset += AndroidUtilities.dp(14 + 4) + (Theme.chat_replyTextPaint.getTextSize() + Theme.chat_replyNamePaint.getTextSize()); @@ -14515,19 +14726,19 @@ private String getNameFromDialogId(long fromId) { return name; } - private boolean isNeedAuthorName() { + protected boolean isNeedAuthorName() { if (currentMessageObject.forceAvatar) { return true; } if (currentMessageObject.isSponsored()) { return false; } - if (currentMessageObject.isGiveaway()) { + if (currentMessageObject.isGiveawayOrGiveawayResults()) { return false; } return ( isPinnedChat && currentMessageObject.type == MessageObject.TYPE_TEXT || - !pinnedTop && drawName && isChat && (!currentMessageObject.isOutOwner() || currentMessageObject.isSupergroup() && currentMessageObject.isFromGroup()) || + !pinnedTop && drawName && isChat && (!currentMessageObject.isOutOwner() || currentMessageObject.isSupergroup() && currentMessageObject.isFromGroup() || currentMessageObject.isRepostPreview) || currentMessageObject.isImportedForward() && currentMessageObject.messageOwner.fwd_from.from_id == null ); } @@ -14709,6 +14920,20 @@ private void setupTextColors() { } } + protected boolean isWidthAdaptive() { + return false; + } + + @Override + public int getBoundsLeft() { + return Math.max(0, getBackgroundDrawableLeft() - (needDrawAvatar() ? dp(currentPosition != null ? 73 : (currentMessageObject.isRepostPreview ? 42 : 63)) : 0)); + } + + @Override + public int getBoundsRight() { + return getBackgroundDrawableRight() + (checkNeedDrawShareButton(currentMessageObject) ? dp(48) : 0); + } + @SuppressLint("WrongCall") @Override protected void onDraw(Canvas canvas) { @@ -14780,6 +15005,10 @@ protected void onDraw(Canvas canvas) { } } } + if (isWidthAdaptive()) { + canvas.save(); + canvas.translate(-getBoundsLeft(), 0); + } drawBackgroundInternal(canvas, false); if (isHighlightedAnimated) { @@ -14962,6 +15191,9 @@ protected void onDraw(Canvas canvas) { if (restore != Integer.MIN_VALUE) { canvas.restoreToCount(restore); } + if (isWidthAdaptive()) { + canvas.restore(); + } updateSelectionTextPosition(); } @@ -15077,7 +15309,7 @@ public void drawBackgroundInternal(Canvas canvas, boolean fromParent) { currentBackgroundShadowDrawable = currentBackgroundDrawable.getShadowDrawable(); } - backgroundDrawableLeft = AndroidUtilities.dp(((isChat || currentMessageObject != null && currentMessageObject.forceAvatar) && isAvatarVisible ? 48 : 0) + (!mediaBackground ? 3 : 9)); + backgroundDrawableLeft = AndroidUtilities.dp(((isChat || currentMessageObject != null && (currentMessageObject.isRepostPreview || currentMessageObject.forceAvatar)) && isAvatarVisible ? 48 : 0) + (!mediaBackground ? 3 : 9)); backgroundDrawableRight = backgroundWidth - (mediaBackground ? 0 : AndroidUtilities.dp(3)); if (currentMessagesGroup != null && !currentMessagesGroup.isDocuments) { if (!currentPosition.edge) { @@ -15507,6 +15739,9 @@ public void drawOutboundsContent(Canvas canvas) { channelRecommendationsCell.draw(canvas); return; } + if (currentMessageObject == null) { + return; + } if (!enterTransitionInProgress) { drawAnimatedEmojis(canvas, 1f); } @@ -15562,7 +15797,7 @@ public void drawOutboundsContent(Canvas canvas) { MessagesController.PeerColors peerColors = MessagesController.getInstance(currentAccount).peerColors; MessagesController.PeerColor peerColor = peerColors != null ? peerColors.getColor(colorId) : null; if (peerColor != null) { - color = peerColor.getColor1(isDark()); + color = peerColor.getColor(0, resourcesProvider); } else { color = getThemedColor(Theme.key_chat_inForwardedNameText); } @@ -15765,7 +16000,7 @@ private void drawSideButton(Canvas canvas) { sideStartY -= getTranslationY(); } } - if (!reactionsLayoutInBubble.isSmall) { + if (currentMessageObject.shouldDrawReactions() && !reactionsLayoutInBubble.isSmall) { if (isRoundVideo) { sideStartY -= reactionsLayoutInBubble.getCurrentTotalHeight(transitionParams.animateChangeProgress) * (1f - getVideoTranscriptionProgress()); } else if (reactionsLayoutInBubble.drawServiceShaderBackground > 0) { @@ -15843,7 +16078,8 @@ public float getTimeAlpha() { } public int getBackgroundDrawableLeft() { - if (currentMessageObject.isOutOwner()) { + MessageObject messageObject = getMessageObject(); + if (messageObject != null && messageObject.isOutOwner()) { if (isRoundVideo) { return layoutWidth - backgroundWidth - (int) ((1f - getVideoTranscriptionProgress()) * AndroidUtilities.dp(9)); } @@ -15851,10 +16087,10 @@ public int getBackgroundDrawableLeft() { } else { int r; if (isRoundVideo) { - r = AndroidUtilities.dp(((isChat || currentMessageObject != null && currentMessageObject.forceAvatar) && isAvatarVisible ? 48 : 0) + 3); + r = AndroidUtilities.dp(((isChat || messageObject != null && (messageObject.isRepostPreview || messageObject.forceAvatar)) && isAvatarVisible ? 48 : 0) + 3); r += (int) (AndroidUtilities.dp(6) * (1f - getVideoTranscriptionProgress())); } else { - r = AndroidUtilities.dp(((isChat || currentMessageObject != null && currentMessageObject.forceAvatar) && isAvatarVisible ? 48 : 0) + (!mediaBackground ? 3 : 9)); + r = AndroidUtilities.dp(((isChat || messageObject != null && (messageObject.isRepostPreview || messageObject.forceAvatar)) && isAvatarVisible ? 48 : 0) + (!mediaBackground ? 3 : 9)); } if (currentMessagesGroup != null && !currentMessagesGroup.isDocuments) { if (currentPosition.leftSpanOffset != 0) { @@ -15940,7 +16176,7 @@ public int getBackgroundDrawableBottom() { } public void drawBackground(Canvas canvas, int left, int top, int right, int bottom, boolean pinnedTop, boolean pinnedBottom, boolean selected, int keyboardHeight) { - if (currentMessageObject.isOutOwner()) { + if (currentMessageObject != null && currentMessageObject.isOutOwner()) { if (!mediaBackground && !pinnedBottom) { currentBackgroundDrawable = (Theme.MessageDrawable) getThemedDrawable(selected ? Theme.key_drawable_msgOutSelected : Theme.key_drawable_msgOut); } else { @@ -16015,7 +16251,7 @@ public void drawNamesLayout(Canvas canvas, float alpha) { } lastNamesAnimationTime = newAnimationTime; - if (currentMessageObject.deleted && currentMessagesGroup != null && currentMessagesGroup.messages.size() >= 1) { + if (currentMessageObject.deleted && !drawingToBitmap && currentMessagesGroup != null && currentMessagesGroup.messages.size() >= 1) { return; } @@ -16054,9 +16290,9 @@ public void drawNamesLayout(Canvas canvas, float alpha) { float alphaProgress = currentMessageObject.isOut() && (checkBoxVisible || checkBoxAnimationInProgress) ? (1.0f - checkBoxAnimationProgress) : 1.0f; rect.set((int) nameX - AndroidUtilities.dp(12), (int) nameY - AndroidUtilities.dp(5), (int) nameX + AndroidUtilities.dp(12) + nameWidth, (int) nameY + AndroidUtilities.dp(22)); + applyServiceShaderMatrix(); oldAlpha = getThemedPaint(Theme.key_paint_chatActionBackground).getAlpha(); getThemedPaint(Theme.key_paint_chatActionBackground).setAlpha((int) (alphaProgress * oldAlpha * replyForwardAlpha)); - applyServiceShaderMatrix(); canvas.drawRoundRect(rect, AndroidUtilities.dp(6), AndroidUtilities.dp(6), getThemedPaint(Theme.key_paint_chatActionBackground)); if (hasGradientService()) { int oldAlpha2 = Theme.chat_actionBackgroundGradientDarkenPaint.getAlpha(); @@ -16116,7 +16352,7 @@ public void drawNamesLayout(Canvas canvas, float alpha) { MessagesController.PeerColors peerColors = MessagesController.getInstance(currentAccount).peerColors; MessagesController.PeerColor peerColor = peerColors != null ? peerColors.getColor(colorId) : null; if (peerColor != null) { - Theme.chat_namePaint.setColor(peerColor.getColor1(isDark())); + Theme.chat_namePaint.setColor(peerColor.getColor(0, resourcesProvider)); } else { Theme.chat_namePaint.setColor(getThemedColor(Theme.key_chat_inForwardedNameText)); } @@ -16256,7 +16492,7 @@ public void drawNamesLayout(Canvas canvas, float alpha) { } } - float forwardNameXLocal; + float forwardNameXLocal = 0; int forwardNameRight = -1; boolean needDrawReplyBackground = true; if (drawForwardedNameLocal && forwardedNameLayoutLocal[0] != null && forwardedNameLayoutLocal[1] != null && (currentPosition == null || currentPosition.minY == 0 && currentPosition.minX == 0)) { @@ -16298,7 +16534,7 @@ public void drawNamesLayout(Canvas canvas, float alpha) { int backWidth = forwardedNameWidthLocal + AndroidUtilities.dp(14); if (hasReply) { needDrawReplyBackground = false; - int replyBackWidth = Math.max(replyNameWidth, replyTextWidth) + AndroidUtilities.dp(14); + int replyBackWidth = Math.max(replyNameWidth, replyTextWidth) - AndroidUtilities.dp(4); rect.set((int) forwardNameXLocal - AndroidUtilities.dp(7), forwardNameY - AndroidUtilities.dp(6), forwardNameRight = ((int) forwardNameXLocal - AndroidUtilities.dp(7) + Math.max(backWidth, replyBackWidth)), forwardNameY + forwardHeight + AndroidUtilities.dp(6) + AndroidUtilities.dp(41)); } else { rect.set((int) forwardNameXLocal - AndroidUtilities.dp(7), forwardNameY - AndroidUtilities.dp(6), (int) forwardNameXLocal - AndroidUtilities.dp(7) + backWidth, forwardNameY + forwardHeight + AndroidUtilities.dp(6)); @@ -16354,7 +16590,7 @@ public void drawNamesLayout(Canvas canvas, float alpha) { MessagesController.PeerColors peerColors = MessagesController.getInstance(currentAccount).peerColors; MessagesController.PeerColor peerColor = peerColors != null ? peerColors.getColor(colorId) : null; if (peerColor != null) { - Theme.chat_forwardNamePaint.setColor(peerColor.getColor1(isDark())); + Theme.chat_forwardNamePaint.setColor(peerColor.getColor(0, resourcesProvider)); } } } @@ -16494,6 +16730,9 @@ public void drawNamesLayout(Canvas canvas, float alpha) { } replyStartY = AndroidUtilities.lerp(transitionParams.animateFromReplyY, this.replyStartY, transitionParams.animateChangeProgress); } + if (!needDrawReplyBackground) { + replyStartX = forwardNameXLocal; + } final boolean loading = currentMessageObject != null && delegate != null && delegate.isProgressLoading(this, ChatActivity.PROGRESS_REPLY); if (replyPressedFloat == null) { replyPressedFloat = new AnimatedFloat(this); @@ -16829,6 +17068,9 @@ public float getHighlightAlpha(boolean quote) { } public void setCheckBoxVisible(boolean visible, boolean animated) { + if (animated && currentMessageObject != null && currentMessageObject.deletedByThanos) { + return; + } if (visible) { quoteHighlight = null; if (checkBox == null) { @@ -16873,6 +17115,9 @@ public boolean isCheckBoxVisible() { } public void setChecked(boolean checked, boolean allChecked, boolean animated) { + if (!checked && animated && currentMessageObject != null && currentMessageObject.deletedByThanos) { + return; + } if (checkBox != null) { checkBox.setChecked(allChecked, animated); } @@ -16914,7 +17159,7 @@ public Theme.MessageDrawable getCurrentBackgroundDrawable(boolean update) { } private boolean shouldDrawCaptionLayout() { - return !currentMessageObject.preview && (currentPosition == null || (currentMessagesGroup != null && currentMessagesGroup.isDocuments && (currentPosition.flags & MessageObject.POSITION_FLAG_BOTTOM) == 0)) && !transitionParams.animateBackgroundBoundsInner && !(enterTransitionInProgress && currentMessageObject.isVoice()); + return currentMessageObject != null && !currentMessageObject.preview && (currentPosition == null || (currentMessagesGroup != null && currentMessagesGroup.isDocuments && (currentPosition.flags & MessageObject.POSITION_FLAG_BOTTOM) == 0)) && !transitionParams.animateBackgroundBoundsInner && !(enterTransitionInProgress && currentMessageObject.isVoice()); } public void drawCaptionLayout(Canvas canvas, boolean selectionOnly, float alpha) { @@ -16940,7 +17185,7 @@ public void drawCaptionLayout(Canvas canvas, boolean selectionOnly, float alpha) reactionsLayoutInBubble.drawServiceShaderBackground = 1f - getVideoTranscriptionProgress(); } - if (!selectionOnly && (currentPosition == null || ((currentPosition.flags & MessageObject.POSITION_FLAG_BOTTOM) != 0 && (currentPosition.flags & MessageObject.POSITION_FLAG_LEFT) != 0)) && !reactionsLayoutInBubble.isSmall) { + if (!selectionOnly && currentMessageObject.shouldDrawReactions() && (currentPosition == null || ((currentPosition.flags & MessageObject.POSITION_FLAG_BOTTOM) != 0 && (currentPosition.flags & MessageObject.POSITION_FLAG_LEFT) != 0)) && !reactionsLayoutInBubble.isSmall) { if (reactionsLayoutInBubble.drawServiceShaderBackground > 0) { applyServiceShaderMatrix(); } @@ -17218,7 +17463,7 @@ private void drawCaptionLayout(Canvas canvas, MessageObject.TextLayoutBlocks cap } } - if (captionLayout == null || selectionOnly && links.isEmpty() || (currentMessageObject.deleted && currentPosition != null) || alpha == 0) { + if (captionLayout == null || selectionOnly && links.isEmpty() || (currentMessageObject.deleted && !drawingToBitmap && currentPosition != null) || alpha == 0) { return; } setupTextColors(); @@ -17557,7 +17802,7 @@ private void drawTimeInternal(Canvas canvas, float alpha, boolean fromParent, fl timeX += animationOffsetX; timeTitleTimeX += animationOffsetX; } - if (reactionsLayoutInBubble.isSmall) { + if (currentMessageObject.shouldDrawReactions() && reactionsLayoutInBubble.isSmall) { if (transitionParams.animateBackgroundBoundsInner && transitionParams.deltaRight != 0) { timeTitleTimeX += reactionsLayoutInBubble.getCurrentWidth(1f); } else { @@ -17639,7 +17884,7 @@ private void drawTimeInternal(Canvas canvas, float alpha, boolean fromParent, fl alpha = oldAlpha3; float additionalX = -timeLayout.getLineLeft(0); - if (reactionsLayoutInBubble.isSmall) { + if (currentMessageObject.shouldDrawReactions() && reactionsLayoutInBubble.isSmall) { updateReactionLayoutPosition(); reactionsLayoutInBubble.draw(canvas, transitionParams.animateChangeProgress, null); } @@ -17710,7 +17955,7 @@ private void drawTimeInternal(Canvas canvas, float alpha, boolean fromParent, fl timeYOffset = -(drawCommentButton ? AndroidUtilities.dp(43) : 0); } float additionalX = -timeLayout.getLineLeft(0); - if (reactionsLayoutInBubble.isSmall) { + if (currentMessageObject.shouldDrawReactions() && reactionsLayoutInBubble.isSmall) { updateReactionLayoutPosition(); reactionsLayoutInBubble.draw(canvas, transitionParams.animateChangeProgress, null); } @@ -18437,8 +18682,8 @@ public void drawOverlays(Canvas canvas) { if (hasGamePreview) { } else if (currentMessageObject.type == MessageObject.TYPE_VIDEO || currentMessageObject.type == MessageObject.TYPE_PHOTO || currentMessageObject.type == MessageObject.TYPE_EXTENDED_MEDIA_PREVIEW || documentAttachType == DOCUMENT_ATTACH_TYPE_VIDEO || documentAttachType == DOCUMENT_ATTACH_TYPE_GIF) { - if (photoImage.getVisible()) { - if (!currentMessageObject.needDrawBluredPreview() && !currentMessageObject.preview && !isSmallImage) { + if (photoImage.getVisible() && !currentMessageObject.isRepostPreview) { + if (!currentMessageObject.needDrawBluredPreview() && !currentMessageObject.isRepostPreview && !currentMessageObject.preview && !isSmallImage) { if (documentAttachType == DOCUMENT_ATTACH_TYPE_VIDEO) { int oldAlpha = ((BitmapDrawable) Theme.chat_msgMediaMenuDrawable).getPaint().getAlpha(); if (drawMediaCheckBox) { @@ -18465,7 +18710,7 @@ public void drawOverlays(Canvas canvas) { fullWidth = (currentPosition.flags & mask) == mask; } - if ((documentAttachType == DOCUMENT_ATTACH_TYPE_VIDEO || documentAttachType == DOCUMENT_ATTACH_TYPE_GIF) && (buttonState == 1 || buttonState == 2 || buttonState == 0 || buttonState == 3 || buttonState == -1) || currentMessageObject.needDrawBluredPreview()) { + if (((documentAttachType == DOCUMENT_ATTACH_TYPE_VIDEO || documentAttachType == DOCUMENT_ATTACH_TYPE_GIF) && (buttonState == 1 || buttonState == 2 || buttonState == 0 || buttonState == 3 || buttonState == -1) || currentMessageObject.needDrawBluredPreview()) && !currentMessageObject.isRepostVideoPreview) { if (autoPlayingMedia) { updatePlayingMessageProgress(); } @@ -18777,7 +19022,7 @@ public void drawOverlays(Canvas canvas) { if (currentMessageObject.isOutOwner()) { x = layoutWidth - backgroundWidth + AndroidUtilities.dp(16); } else { - if ((isChat && !isThreadPost || currentMessageObject.forceAvatar) && currentMessageObject.needDrawAvatar()) { + if (needDrawAvatar()) { x = AndroidUtilities.dp(74); } else { x = AndroidUtilities.dp(25); @@ -18854,7 +19099,7 @@ public void drawOverlays(Canvas canvas) { if (currentMessageObject.isOutOwner()) { x = layoutWidth - backgroundWidth + AndroidUtilities.dp(11); } else { - if ((isChat && !isThreadPost || currentMessageObject.forceAvatar) && currentMessageObject.needDrawAvatar()) { + if (needDrawAvatar()) { x = AndroidUtilities.dp(68); } else { x = AndroidUtilities.dp(20); @@ -19145,14 +19390,16 @@ public void drawOverlays(Canvas canvas) { canvas.restore(); } - Drawable menuDrawable; - if (currentMessageObject.isOutOwner()) { - menuDrawable = getThemedDrawable(isDrawSelectionBackground() ? Theme.key_drawable_msgOutMenuSelected : Theme.key_drawable_msgOutMenu); - } else { - menuDrawable = isDrawSelectionBackground() ? Theme.chat_msgInMenuSelectedDrawable : Theme.chat_msgInMenuDrawable; + if (!currentMessageObject.isRepostPreview) { + Drawable menuDrawable; + if (currentMessageObject.isOutOwner()) { + menuDrawable = getThemedDrawable(isDrawSelectionBackground() ? Theme.key_drawable_msgOutMenuSelected : Theme.key_drawable_msgOutMenu); + } else { + menuDrawable = isDrawSelectionBackground() ? Theme.chat_msgInMenuSelectedDrawable : Theme.chat_msgInMenuDrawable; + } + setDrawableBounds(menuDrawable, otherX = (int) (photoImage.getImageX() + backgroundWidth - AndroidUtilities.dp(48)), otherY = (int) (photoImage.getImageY() - AndroidUtilities.dp(2))); + menuDrawable.draw(canvas); } - setDrawableBounds(menuDrawable, otherX = (int) (photoImage.getImageX() + backgroundWidth - AndroidUtilities.dp(48)), otherY = (int) (photoImage.getImageY() - AndroidUtilities.dp(2))); - menuDrawable.draw(canvas); if (drawInstantView) { int textX = (int) (photoImage.getImageX() - AndroidUtilities.dp(2)); @@ -19196,7 +19443,7 @@ public void drawOverlays(Canvas canvas) { } } - if (drawImageButton && photoImage.getVisible() && !isSmallImage) { + if (drawImageButton && photoImage.getVisible() && !isSmallImage && !currentMessageObject.isRepostVideoPreview) { if (controlsAlpha != 1.0f) { radialProgress.setOverrideAlpha(controlsAlpha); } @@ -19226,7 +19473,7 @@ public void drawOverlays(Canvas canvas) { canvas.scale(scale, scale, radialProgress.getProgressRect().centerX(), radialProgress.getProgressRect().centerY()); restore = true; } - if ((!isRoundVideo || !hasLinkPreview) && (!currentMessageObject.needDrawBluredPreview() || !MediaController.getInstance().isPlayingMessage(currentMessageObject)) && !(currentMessageObject.hasMediaSpoilers() && (!currentMessageObject.isMediaSpoilersRevealed || !currentMessageObject.revealingMediaSpoilers) && SharedConfig.isAutoplayVideo() && currentMessagesGroup == null && (radialProgress.getIcon() == MediaActionDrawable.ICON_PLAY || radialProgress.getIcon() == MediaActionDrawable.ICON_NONE))) { + if ((!isRoundVideo || !hasLinkPreview) && (!currentMessageObject.needDrawBluredPreview() || !MediaController.getInstance().isPlayingMessage(currentMessageObject)) && !(currentMessageObject.hasMediaSpoilers() && (!currentMessageObject.isMediaSpoilersRevealed || !currentMessageObject.revealingMediaSpoilers) && SharedConfig.isAutoplayVideo() && !currentMessageObject.isRepostPreview && currentMessagesGroup == null && (radialProgress.getIcon() == MediaActionDrawable.ICON_PLAY || radialProgress.getIcon() == MediaActionDrawable.ICON_NONE))) { if (currentMessageObject.needDrawBluredPreview()) { radialProgress.overrideCircleAlpha = 0f; } else if (isRoundVideo && !on) { @@ -19251,6 +19498,7 @@ public void drawOverlays(Canvas canvas) { canvas.drawRoundRect(radialProgress.getProgressRect(), AndroidUtilities.dp(48), AndroidUtilities.dp(48), dimPaint); dimPaint.setAlpha(oldAlpha2); } + radialProgress.iconScale = 1f; radialProgress.draw(canvas); if ((!SharedConfig.isAutoplayVideo() || currentMessagesGroup != null) && currentMessageObject.hasMediaSpoilers() && !currentMessageObject.isMediaSpoilersRevealed && radialProgress.getIcon() == MediaActionDrawable.ICON_PLAY) { canvas.restore(); @@ -19282,7 +19530,7 @@ public void drawOverlays(Canvas canvas) { invalidate(); updateSecretTimeText(currentMessageObject); } - if ((drawVideoImageButton || animatingDrawVideoImageButton != 0) && photoImage.getVisible() && !isSmallImage) { + if ((drawVideoImageButton || animatingDrawVideoImageButton != 0) && !currentMessageObject.isRepostPreview && photoImage.getVisible() && !isSmallImage) { if (controlsAlpha != 1.0f) { videoRadialProgress.setOverrideAlpha(controlsAlpha); } @@ -19339,17 +19587,19 @@ public void drawOverlays(Canvas canvas) { rect.set(x1, y1, x1 + timeWidthAudio + AndroidUtilities.dp(8 + 12 + 2), y1 + AndroidUtilities.dp(17)); - int oldAlpha = getThemedPaint(Theme.key_paint_chatActionBackground).getAlpha(); - getThemedPaint(Theme.key_paint_chatActionBackground).setAlpha((int) (oldAlpha * timeAlpha * (1f - getVideoTranscriptionProgress()))); applyServiceShaderMatrix(); - canvas.drawRoundRect(rect, AndroidUtilities.dp(6), AndroidUtilities.dp(6), getThemedPaint(Theme.key_paint_chatActionBackground)); + Paint bgPaint = getThemedPaint(Theme.key_paint_chatActionBackground); + int oldAlpha = bgPaint.getAlpha(); + bgPaint.setAlpha((int) (oldAlpha * timeAlpha * 0.5f * (1f - getVideoTranscriptionProgress()))); + canvas.drawRoundRect(rect, AndroidUtilities.dp(6), AndroidUtilities.dp(6), bgPaint); if (hasGradientService()) { - int oldAlpha2 = Theme.chat_actionBackgroundGradientDarkenPaint.getAlpha(); - Theme.chat_actionBackgroundGradientDarkenPaint.setAlpha((int) (oldAlpha2 * timeAlpha * (1f - getVideoTranscriptionProgress()))); - canvas.drawRoundRect(rect, AndroidUtilities.dp(6), AndroidUtilities.dp(6), Theme.chat_actionBackgroundGradientDarkenPaint); - Theme.chat_actionBackgroundGradientDarkenPaint.setAlpha(oldAlpha2); + Paint darkenPaint = getThemedPaint(Theme.key_paint_chatActionBackgroundDarken); + int oldAlpha2 = darkenPaint.getAlpha(); + darkenPaint.setAlpha((int) (oldAlpha2 * timeAlpha * (1f - getVideoTranscriptionProgress()))); + canvas.drawRoundRect(rect, AndroidUtilities.dp(6), AndroidUtilities.dp(6), darkenPaint); + darkenPaint.setAlpha(oldAlpha2); } - getThemedPaint(Theme.key_paint_chatActionBackground).setAlpha(oldAlpha); + bgPaint.setAlpha(oldAlpha); boolean showPlayingDrawable = playing || !currentMessageObject.isContentUnread(); @@ -20521,7 +20771,7 @@ public boolean performAction(int virtualViewId, int action, Bundle arguments) { } } else if (action == AccessibilityNodeInfo.ACTION_LONG_CLICK) { ClickableSpan link = getLinkById(virtualViewId, virtualViewId >= LINK_CAPTION_IDS_START); - if (link != null) { + if (link != null && delegate != null) { delegate.didPressUrl(ChatMessageCell.this, link, true); sendAccessibilityEventForVirtualView(virtualViewId, AccessibilityEvent.TYPE_VIEW_LONG_CLICKED); } @@ -21450,4 +21700,12 @@ private boolean isDark() { } return Theme.isCurrentThemeDark(); } + + public boolean needDrawAvatar() { + return isChat && !isThreadPost && currentMessageObject != null && !currentMessageObject.isOutOwner() && currentMessageObject.needDrawAvatar() || currentMessageObject != null && currentMessageObject.forceAvatar; + } + + protected boolean drawPhotoImage(Canvas canvas) { + return photoImage.draw(canvas); + } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Cells/DialogCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Cells/DialogCell.java index d878df5f976..d6d2bc29d52 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Cells/DialogCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Cells/DialogCell.java @@ -1169,6 +1169,11 @@ public void buildLayout() { } else if (chat.fake) { drawScam = 2; Theme.dialogs_fakeDrawable.checkText(); + } else if (DialogObject.getEmojiStatusDocumentId(chat.emoji_status) != 0) { + drawPremium = true; + nameLayoutEllipsizeByGradient = true; + emojiStatus.center = LocaleController.isRTL; + emojiStatus.set(DialogObject.getEmojiStatusDocumentId(chat.emoji_status), false); } else { drawVerified = !forbidVerified && chat.verified; } @@ -1472,7 +1477,13 @@ public void buildLayout() { } else if (message.messageOwner.media instanceof TLRPC.TL_messageMediaPhoto && message.messageOwner.media.photo instanceof TLRPC.TL_photoEmpty && message.messageOwner.media.ttl_seconds != 0) { messageString = LocaleController.getString("AttachPhotoExpired", R.string.AttachPhotoExpired); } else if (message.messageOwner.media instanceof TLRPC.TL_messageMediaDocument && (message.messageOwner.media.document instanceof TLRPC.TL_documentEmpty || message.messageOwner.media.document == null) && message.messageOwner.media.ttl_seconds != 0) { - messageString = LocaleController.getString("AttachVideoExpired", R.string.AttachVideoExpired); + if (message.messageOwner.media.voice) { + messageString = LocaleController.getString(R.string.AttachVoiceExpired); + } else if (message.messageOwner.media.round) { + messageString = LocaleController.getString(R.string.AttachRoundExpired); + } else { + messageString = LocaleController.getString(R.string.AttachVideoExpired); + } } else if (getCaptionMessage() != null) { MessageObject message = getCaptionMessage(); String emoji; @@ -1522,8 +1533,9 @@ public void buildLayout() { currentMessagePaint = Theme.dialogs_messagePrintingPaint[paintIndex]; } else { if (message.messageOwner.media instanceof TLRPC.TL_messageMediaGiveaway) { - TLRPC.TL_messageMediaGiveaway mediaPoll = (TLRPC.TL_messageMediaGiveaway) message.messageOwner.media; messageString = LocaleController.getString("BoostingGiveawayChannelStarted", R.string.BoostingGiveawayChannelStarted); + } else if (message.messageOwner.media instanceof TLRPC.TL_messageMediaGiveawayResults) { + messageString = LocaleController.getString("BoostingGiveawayResults", R.string.BoostingGiveawayResults); } else if (message.messageOwner.media instanceof TLRPC.TL_messageMediaPoll) { TLRPC.TL_messageMediaPoll mediaPoll = (TLRPC.TL_messageMediaPoll) message.messageOwner.media; messageString = "\uD83D\uDCCA " + mediaPoll.poll.question; @@ -2722,17 +2734,29 @@ public boolean update(int mask, boolean animated) { invalidate = true; } } - if (user != null && (mask & MessagesController.UPDATE_MASK_EMOJI_STATUS) != 0) { - user = MessagesController.getInstance(currentAccount).getUser(user.id); - Long emojiStatusId = UserObject.getEmojiStatusDocumentId(user); - if (emojiStatusId != null) { - nameLayoutEllipsizeByGradient = true; - emojiStatus.set(emojiStatusId, animated); - } else { - nameLayoutEllipsizeByGradient = true; - emojiStatus.set(PremiumGradient.getInstance().premiumStarDrawableMini, animated); + if ((mask & MessagesController.UPDATE_MASK_EMOJI_STATUS) != 0) { + if (user != null) { + user = MessagesController.getInstance(currentAccount).getUser(user.id); + if (user != null && DialogObject.getEmojiStatusDocumentId(user.emoji_status) != 0) { + nameLayoutEllipsizeByGradient = true; + emojiStatus.set(DialogObject.getEmojiStatusDocumentId(user.emoji_status), animated); + } else { + nameLayoutEllipsizeByGradient = true; + emojiStatus.set(PremiumGradient.getInstance().premiumStarDrawableMini, animated); + } + invalidate = true; + } + if (chat != null) { + chat = MessagesController.getInstance(currentAccount).getChat(chat.id); + if (chat != null && DialogObject.getEmojiStatusDocumentId(chat.emoji_status) != 0) { + nameLayoutEllipsizeByGradient = true; + emojiStatus.set(DialogObject.getEmojiStatusDocumentId(chat.emoji_status), animated); + } else { + nameLayoutEllipsizeByGradient = true; + emojiStatus.set(PremiumGradient.getInstance().premiumStarDrawableMini, animated); + } + invalidate = true; } - invalidate = true; } if (isDialogCell || isTopic) { if ((mask & MessagesController.UPDATE_MASK_USER_PRINT) != 0) { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Cells/DialogsHintCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Cells/DialogsHintCell.java index a9c23b56ddb..03d271f78e9 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Cells/DialogsHintCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Cells/DialogsHintCell.java @@ -17,16 +17,18 @@ import androidx.annotation.NonNull; import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.Emoji; import org.telegram.messenger.LocaleController; import org.telegram.messenger.R; import org.telegram.ui.ActionBar.Theme; import org.telegram.ui.Components.LayoutHelper; public class DialogsHintCell extends FrameLayout { - private LinearLayout contentView; - private TextView titleView; - private TextView messageView; - private ImageView chevronView; + private final LinearLayout contentView; + private final TextView titleView; + private final TextView messageView; + private final ImageView chevronView; + private final ImageView closeView; public DialogsHintCell(@NonNull Context context) { super(context); @@ -43,7 +45,7 @@ public DialogsHintCell(@NonNull Context context) { titleView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 15); titleView.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); titleView.setSingleLine(); - contentView.addView(titleView, LayoutHelper.createLinear(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, 0, Gravity.TOP)); + contentView.addView(titleView, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, 0, (LocaleController.isRTL ? Gravity.RIGHT : Gravity.LEFT) | Gravity.TOP)); messageView = new TextView(context); messageView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 14); @@ -55,6 +57,12 @@ public DialogsHintCell(@NonNull Context context) { chevronView.setImageResource(R.drawable.arrow_newchat); addView(chevronView, LayoutHelper.createFrame(16, 16, (LocaleController.isRTL ? Gravity.LEFT : Gravity.RIGHT) | Gravity.CENTER_VERTICAL)); + closeView = new ImageView(context); + closeView.setImageResource(R.drawable.msg_close); + closeView.setPadding(dp(6), dp(6), dp(6), dp(6)); + addView(closeView, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, (LocaleController.isRTL ? Gravity.LEFT : Gravity.RIGHT) | Gravity.CENTER_VERTICAL, LocaleController.isRTL ? -15 : 0, 0, LocaleController.isRTL ? 0 : -15, 0)); + closeView.setVisibility(GONE); + setClipToPadding(false); updateColors(); } @@ -62,12 +70,28 @@ public void updateColors() { titleView.setTextColor(Theme.getColor(Theme.key_windowBackgroundWhiteBlackText)); messageView.setTextColor(Theme.getColor(Theme.key_windowBackgroundWhiteGrayText)); chevronView.setColorFilter(Theme.getColor(Theme.key_windowBackgroundWhiteGrayText), PorterDuff.Mode.SRC_IN); + closeView.setColorFilter(Theme.getColor(Theme.key_windowBackgroundWhiteGrayText), PorterDuff.Mode.SRC_IN); + closeView.setBackground(Theme.AdaptiveRipple.filledCircle()); setBackground(Theme.AdaptiveRipple.filledRect()); } public void setText(CharSequence title, CharSequence subtitle) { titleView.setText(title); + titleView.setCompoundDrawables(null, null, null, null); messageView.setText(subtitle); + chevronView.setVisibility(VISIBLE); + closeView.setVisibility(GONE); + } + + public void setChristmasStyle(OnClickListener closeListener) { + chevronView.setVisibility(INVISIBLE); + closeView.setVisibility(VISIBLE); + closeView.setOnClickListener(closeListener); + Emoji.EmojiDrawable drawable = Emoji.getEmojiDrawable("\uD83C\uDF84"); + if (drawable != null) { + drawable.setBounds(dp(2), -dp(2), Emoji.drawImgSize + dp(2), Emoji.drawImgSize - dp(2)); + titleView.setCompoundDrawables(null, null, drawable, null); + } } @Override @@ -85,8 +109,8 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { width = AndroidUtilities.displaySize.x; } contentView.measure( - MeasureSpec.makeMeasureSpec(width - getPaddingLeft() - getPaddingRight(), MeasureSpec.EXACTLY), - MeasureSpec.makeMeasureSpec(AndroidUtilities.displaySize.y, MeasureSpec.AT_MOST) + MeasureSpec.makeMeasureSpec(width - getPaddingLeft() - getPaddingRight(), MeasureSpec.EXACTLY), + MeasureSpec.makeMeasureSpec(AndroidUtilities.displaySize.y, MeasureSpec.AT_MOST) ); this.height = contentView.getMeasuredHeight() + getPaddingTop() + getPaddingBottom() + 1; super.onMeasure(widthMeasureSpec, MeasureSpec.makeMeasureSpec(height, MeasureSpec.EXACTLY)); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Cells/GroupCallUserCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Cells/GroupCallUserCell.java index 48a23680370..6082a03bafa 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Cells/GroupCallUserCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Cells/GroupCallUserCell.java @@ -29,6 +29,7 @@ import org.telegram.messenger.AccountInstance; import org.telegram.messenger.AndroidUtilities; import org.telegram.messenger.ChatObject; +import org.telegram.messenger.DialogObject; import org.telegram.messenger.ImageLocation; import org.telegram.messenger.LiteMode; import org.telegram.messenger.LocaleController; @@ -474,10 +475,8 @@ public void setData(AccountInstance account, TLRPC.TL_groupCallParticipant group nameTextView.setText(UserObject.getUserName(currentUser)); if (currentUser != null && currentUser.verified) { rightDrawable.set(verifiedDrawable = (verifiedDrawable == null ? new VerifiedDrawable(getContext()) : verifiedDrawable), animated); - } else if (currentUser != null && currentUser.emoji_status instanceof TLRPC.TL_emojiStatus) { - rightDrawable.set(((TLRPC.TL_emojiStatus) currentUser.emoji_status).document_id, animated); - } else if (currentUser != null && currentUser.emoji_status instanceof TLRPC.TL_emojiStatusUntil && ((TLRPC.TL_emojiStatusUntil) currentUser.emoji_status).until > (int) (System.currentTimeMillis() / 1000)) { - rightDrawable.set(((TLRPC.TL_emojiStatusUntil) currentUser.emoji_status).document_id, animated); + } else if (currentUser != null && DialogObject.getEmojiStatusDocumentId(currentUser.emoji_status) != 0) { + rightDrawable.set(DialogObject.getEmojiStatusDocumentId(currentUser.emoji_status), animated); } else if (currentUser != null && currentUser.premium) { if (premiumDrawable == null) { premiumDrawable = getContext().getResources().getDrawable(R.drawable.msg_premium_liststar).mutate(); @@ -515,6 +514,8 @@ public void draw(@NonNull Canvas canvas) { nameTextView.setText(currentChat.title); if (currentChat.verified) { rightDrawable.set(verifiedDrawable = (verifiedDrawable == null ? new VerifiedDrawable(getContext()) : verifiedDrawable), animated); + } else if (currentChat != null && DialogObject.getEmojiStatusDocumentId(currentChat.emoji_status) != 0) { + rightDrawable.set(DialogObject.getEmojiStatusDocumentId(currentChat.emoji_status), animated); } else { rightDrawable.set((Drawable) null, animated); } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Cells/ManageChatUserCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Cells/ManageChatUserCell.java index c1478c2bd1b..f1932853e96 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Cells/ManageChatUserCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Cells/ManageChatUserCell.java @@ -21,6 +21,7 @@ import org.telegram.messenger.Emoji; import org.telegram.messenger.LocaleController; import org.telegram.messenger.MessagesController; +import org.telegram.messenger.NotificationCenter; import org.telegram.messenger.R; import org.telegram.messenger.UserConfig; import org.telegram.messenger.UserObject; @@ -106,6 +107,7 @@ protected void onDraw(Canvas canvas) { nameTextView.setTypeface(AndroidUtilities.getTypeface("fonts/rmedium.ttf")); nameTextView.setGravity((LocaleController.isRTL ? Gravity.RIGHT : Gravity.LEFT) | Gravity.TOP); addView(nameTextView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 20, (LocaleController.isRTL ? Gravity.RIGHT : Gravity.LEFT) | Gravity.TOP, LocaleController.isRTL ? 28 + 18 : (68 + namePadding), 11.5f, LocaleController.isRTL ? (68 + namePadding) : 28 + 18, 0)); + NotificationCenter.listenEmojiLoading(nameTextView); statusTextView = new SimpleTextView(context); statusTextView.setTextSize(14); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Cells/ProfileSearchCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Cells/ProfileSearchCell.java index dbd87813ef7..fe844a1e953 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Cells/ProfileSearchCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Cells/ProfileSearchCell.java @@ -308,7 +308,7 @@ public void buildLayout() { nameLeft = AndroidUtilities.dp(11); } nameLockTop = AndroidUtilities.dp(22.0f); - updateStatus(false, null, false); + updateStatus(false, null, null, false); } else if (chat != null) { dialog_id = -chat.id; drawCheck = chat.verified; @@ -317,7 +317,7 @@ public void buildLayout() { } else { nameLeft = AndroidUtilities.dp(11); } - updateStatus(drawCheck, null, false); + updateStatus(drawCheck, null, chat, false); } else if (user != null) { dialog_id = user.id; if (!LocaleController.isRTL) { @@ -328,7 +328,7 @@ public void buildLayout() { nameLockTop = AndroidUtilities.dp(21); drawCheck = user.verified; drawPremium = !savedMessages && MessagesController.getInstance(currentAccount).isPremiumUser(user); - updateStatus(drawCheck, user, false); + updateStatus(drawCheck, user, null, false); } else if (contact != null) { dialog_id = 0; if (!LocaleController.isRTL) { @@ -587,16 +587,16 @@ public void buildLayout() { nameLockLeft += getPaddingLeft(); } - public void updateStatus(boolean verified, TLRPC.User user, boolean animated) { + public void updateStatus(boolean verified, TLRPC.User user, TLRPC.Chat chat, boolean animated) { statusDrawable.center = LocaleController.isRTL; if (verified) { statusDrawable.set(new CombinedDrawable(Theme.dialogs_verifiedDrawable, Theme.dialogs_verifiedCheckDrawable, 0, 0), animated); statusDrawable.setColor(null); - } else if (user != null && !savedMessages && user.emoji_status instanceof TLRPC.TL_emojiStatusUntil && ((TLRPC.TL_emojiStatusUntil) user.emoji_status).until > (int) (System.currentTimeMillis() / 1000)) { - statusDrawable.set(((TLRPC.TL_emojiStatusUntil) user.emoji_status).document_id, animated); + } else if (user != null && !savedMessages && DialogObject.getEmojiStatusDocumentId(user.emoji_status) != 0) { + statusDrawable.set(DialogObject.getEmojiStatusDocumentId(user.emoji_status), animated); statusDrawable.setColor(Theme.getColor(Theme.key_chats_verifiedBackground, resourcesProvider)); - } else if (user != null && !savedMessages && user.emoji_status instanceof TLRPC.TL_emojiStatus) { - statusDrawable.set(((TLRPC.TL_emojiStatus) user.emoji_status).document_id, animated); + } else if (chat != null && !savedMessages && DialogObject.getEmojiStatusDocumentId(chat.emoji_status) != 0) { + statusDrawable.set(DialogObject.getEmojiStatusDocumentId(chat.emoji_status), animated); statusDrawable.setColor(Theme.getColor(Theme.key_chats_verifiedBackground, resourcesProvider)); } else if (user != null && !savedMessages && MessagesController.getInstance(currentAccount).isPremiumUser(user)) { statusDrawable.set(PremiumGradient.getInstance().premiumStarDrawableMini, animated); @@ -662,8 +662,8 @@ public void update(int mask) { continueUpdate = true; } } - if (!continueUpdate && (mask & MessagesController.UPDATE_MASK_EMOJI_STATUS) != 0 && user != null) { - updateStatus(user.verified, user, true); + if (!continueUpdate && (mask & MessagesController.UPDATE_MASK_EMOJI_STATUS) != 0 && (user != null || chat != null)) { + updateStatus(user != null ? user.verified : chat.verified, user, chat, true); } if (!continueUpdate && ((mask & MessagesController.UPDATE_MASK_NAME) != 0 && user != null) || (mask & MessagesController.UPDATE_MASK_CHAT_NAME) != 0 && chat != null) { String newName; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Cells/ReactedUserHolderView.java b/TMessagesProj/src/main/java/org/telegram/ui/Cells/ReactedUserHolderView.java index 616957c36c8..f8cfed7e354 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Cells/ReactedUserHolderView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Cells/ReactedUserHolderView.java @@ -8,7 +8,10 @@ import android.graphics.PorterDuff; import android.graphics.PorterDuffColorFilter; import android.graphics.drawable.Drawable; +import android.text.SpannableStringBuilder; +import android.text.Spanned; import android.text.TextUtils; +import android.text.style.RelativeSizeSpan; import android.view.Gravity; import android.view.MotionEvent; import android.view.View; @@ -23,6 +26,7 @@ import org.telegram.messenger.AndroidUtilities; import org.telegram.messenger.DocumentObject; import org.telegram.messenger.Emoji; +import org.telegram.messenger.FileLoader; import org.telegram.messenger.ImageLocation; import org.telegram.messenger.MediaDataController; import org.telegram.messenger.MessageObject; @@ -33,12 +37,14 @@ import org.telegram.messenger.UserObject; import org.telegram.tgnet.TLObject; import org.telegram.tgnet.TLRPC; +import org.telegram.tgnet.tl.TL_stories; import org.telegram.ui.ActionBar.SimpleTextView; import org.telegram.ui.ActionBar.Theme; import org.telegram.ui.Components.AnimatedEmojiDrawable; import org.telegram.ui.Components.AvatarDrawable; import org.telegram.ui.Components.BackupImageView; import org.telegram.ui.Components.CubicBezierInterpolator; +import org.telegram.ui.Components.DotDividerSpan; import org.telegram.ui.Components.LayoutHelper; import org.telegram.ui.Components.MessageSeenCheckDrawable; import org.telegram.ui.Components.Reactions.ReactionsLayoutInBubble; @@ -57,6 +63,8 @@ public class ReactedUserHolderView extends FrameLayout { SimpleTextView titleView; SimpleTextView subtitleView; BackupImageView reactView; + public BackupImageView storyPreviewView; + public int storyId; AvatarDrawable avatarDrawable = new AvatarDrawable(); View overlaySelectorView; StatusBadgeComponent statusBadgeComponent; @@ -73,6 +81,8 @@ public void openStory(long dialogId, Runnable onDone) { public static final MessageSeenCheckDrawable seenDrawable = new MessageSeenCheckDrawable(R.drawable.msg_mini_checks, Theme.key_windowBackgroundWhiteGrayText); public static final MessageSeenCheckDrawable reactDrawable = new MessageSeenCheckDrawable(R.drawable.msg_reactions, Theme.key_windowBackgroundWhiteGrayText, 16, 16, 5.66f); + public static final MessageSeenCheckDrawable repostDrawable = new MessageSeenCheckDrawable(R.drawable.mini_repost_story, Theme.key_stories_circle1); + public static final MessageSeenCheckDrawable forwardDrawable = new MessageSeenCheckDrawable(R.drawable.mini_forward_story, Theme.key_stories_circle1); public ReactedUserHolderView(int style, int currentAccount, @NonNull Context context, Theme.ResourcesProvider resourcesProvider) { this(style, currentAccount, context, resourcesProvider, true); @@ -148,6 +158,9 @@ public boolean setText(CharSequence value) { reactView = new BackupImageView(context); addView(reactView, LayoutHelper.createFrameRelatively(24, 24, Gravity.END | Gravity.CENTER_VERTICAL, 0, 0, 12, 0)); + storyPreviewView = new BackupImageView(context); + addView(storyPreviewView, LayoutHelper.createFrameRelatively(22, 35, Gravity.END | Gravity.CENTER_VERTICAL, 0, 0, 12, 0)); + if (useOverlaySelector) { overlaySelectorView = new View(context); overlaySelectorView.setBackground(Theme.getSelectorDrawable(false)); @@ -155,7 +168,7 @@ public boolean setText(CharSequence value) { } } - public void setUserReaction(TLRPC.User user, TLRPC.Chat chat, TLRPC.Reaction reaction, boolean like, long date, boolean dateIsSeen, boolean animated) { + public void setUserReaction(TLRPC.User user, TLRPC.Chat chat, TLRPC.Reaction reaction, boolean like, long date, TL_stories.StoryItem storyItem, boolean isForward, boolean dateIsSeen, boolean animated) { TLObject u = user; if (u == null) { u = chat; @@ -223,6 +236,24 @@ public void setUserReaction(TLRPC.User user, TLRPC.Chat chat, TLRPC.Reaction rea contentDescription = LocaleController.formatString("AccDescrPersonHasSeen", R.string.AccDescrPersonHasSeen, titleView.getText()); } + if (storyItem != null) { + storyId = storyItem.id; + if (storyItem.media != null && storyItem.media.photo != null) { + final TLRPC.PhotoSize photoSize = FileLoader.getClosestPhotoSizeWithSize(storyItem.media.photo.sizes, 35, false, null, true); + storyPreviewView.setImage(ImageLocation.getForPhoto(photoSize, storyItem.media.photo), "22_35", null, null, -1, storyItem); + } else if (storyItem.media != null && storyItem.media.document != null) { + final TLRPC.PhotoSize photoSize = FileLoader.getClosestPhotoSizeWithSize(storyItem.media.document.thumbs, 35, false, null, true); + storyPreviewView.setImage(ImageLocation.getForDocument(photoSize, storyItem.media.document), "22_35", null, null, -1, storyItem); + } + storyPreviewView.setRoundRadius(AndroidUtilities.dp(3.33f)); + if (date <= 0) { + date = storyItem.date; + } + } else { + storyId = -1; + storyPreviewView.setImageDrawable(null); + } + if (date != 0) { contentDescription += " " + LocaleController.formatSeenDate(date); } @@ -230,8 +261,41 @@ public void setUserReaction(TLRPC.User user, TLRPC.Chat chat, TLRPC.Reaction rea if (date != 0) { subtitleView.setVisibility(View.VISIBLE); - CharSequence icon = dateIsSeen ? seenDrawable.getSpanned(getContext(), resourcesProvider) : reactDrawable.getSpanned(getContext(), resourcesProvider); - subtitleView.setText(TextUtils.concat(icon, LocaleController.formatSeenDate(date))); + MessageSeenCheckDrawable drawable; + if (storyItem != null) { + drawable = isForward ? forwardDrawable : repostDrawable; + } else if (dateIsSeen) { + drawable = seenDrawable; + } else { + drawable = reactDrawable; + } + SpannableStringBuilder ssb = new SpannableStringBuilder(); + ssb.append(drawable.getSpanned(getContext(), resourcesProvider)); + ssb.append(LocaleController.formatSeenDate(date)); + if (!isForward && storyItem != null && !TextUtils.isEmpty(storyItem.caption)) { + ssb.append(" "); + ssb.append("."); + DotDividerSpan dotSpan = new DotDividerSpan(); + dotSpan.setSize(2.33333f); + dotSpan.setTopPadding(AndroidUtilities.dp(5)); + ssb.setSpan(dotSpan, ssb.length() - 1, ssb.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + ssb.append(" "); + int index = ssb.length(); + ssb.append(LocaleController.getString(R.string.StoryRepostCommented)); + ssb.setSpan(new RelativeSizeSpan(.95f), index, ssb.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + } else if (!isForward && storyItem != null && storyItem.fwd_from != null && storyItem.fwd_from.modified) { + ssb.append(" "); + ssb.append("."); + DotDividerSpan dotSpan = new DotDividerSpan(); + dotSpan.setSize(2.33333f); + dotSpan.setTopPadding(AndroidUtilities.dp(5)); + ssb.setSpan(dotSpan, ssb.length() - 1, ssb.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + ssb.append(" "); + int index = ssb.length(); + ssb.append("edited"); + ssb.setSpan(new RelativeSizeSpan(.95f), index, ssb.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + } + subtitleView.setText(ssb); subtitleView.setTranslationY(!dateIsSeen ? AndroidUtilities.dp(-1) : 0); titleView.setTranslationY(0); if (animated) { @@ -263,7 +327,7 @@ public void setUserReaction(TLRPC.MessagePeerReaction reaction) { } else { chat = MessagesController.getInstance(currentAccount).getChat(-dialogId); } - setUserReaction(user, chat, reaction.reaction, false, reaction.date, reaction.dateIsSeen, false); + setUserReaction(user, chat, reaction.reaction, false, reaction.date, null, false, reaction.dateIsSeen, false); } @Override diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Cells/ShareDialogCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Cells/ShareDialogCell.java index 694593da119..f7dd2b02244 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Cells/ShareDialogCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Cells/ShareDialogCell.java @@ -12,10 +12,13 @@ import android.content.Context; import android.graphics.Canvas; +import android.graphics.Color; import android.graphics.ColorFilter; import android.graphics.LinearGradient; import android.graphics.Paint; import android.graphics.PixelFormat; +import android.graphics.PorterDuff; +import android.graphics.PorterDuffColorFilter; import android.graphics.Shader; import android.graphics.drawable.Drawable; import android.os.SystemClock; @@ -79,12 +82,16 @@ public void invalidateSelf() { private boolean topicWasVisible; private final int currentAccount = UserConfig.selectedAccount; - private final Theme.ResourcesProvider resourcesProvider; + public final Theme.ResourcesProvider resourcesProvider; public static final int TYPE_SHARE = 0; public static final int TYPE_CALL = 1; public static final int TYPE_CREATE = 2; + public BackupImageView getImageView() { + return imageView; + } + public ShareDialogCell(Context context, int type, Theme.ResourcesProvider resourcesProvider) { super(context); this.resourcesProvider = resourcesProvider; @@ -144,11 +151,15 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { super.onMeasure(widthMeasureSpec, MeasureSpec.makeMeasureSpec(dp(currentType == TYPE_CREATE ? 95 : 103), MeasureSpec.EXACTLY)); } + protected String repostToCustomName() { + return LocaleController.getString(R.string.FwdMyStory); + } + public void setDialog(long uid, boolean checked, CharSequence name) { if (uid == Long.MAX_VALUE) { - nameTextView.setText(LocaleController.getString(R.string.FwdMyStory)); + nameTextView.setText(repostToCustomName()); if (repostStoryDrawable == null) { - repostStoryDrawable = new RepostStoryDrawable(imageView, resourcesProvider); + repostStoryDrawable = new RepostStoryDrawable(getContext(), imageView, true, resourcesProvider); } imageView.setImage(null, null, repostStoryDrawable, null); } else if (DialogObject.isUserDialog(uid)) { @@ -321,41 +332,62 @@ public void onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo info) { } } - private static class RepostStoryDrawable extends Drawable { + public static class RepostStoryDrawable extends Drawable { private final LinearGradient gradient; private final Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG); private final RLottieDrawable lottieDrawable; + private final Drawable drawable; - public RepostStoryDrawable(View view, Theme.ResourcesProvider resourcesProvider) { + public RepostStoryDrawable(Context context, View parentView, boolean animate, Theme.ResourcesProvider resourcesProvider) { gradient = new LinearGradient(0, 0, dp(56), dp(56), new int[] { Theme.getColor(Theme.key_stories_circle1, resourcesProvider), Theme.getColor(Theme.key_stories_circle2, resourcesProvider) }, new float[] { 0, 1 }, Shader.TileMode.CLAMP); paint.setShader(gradient); - lottieDrawable = new RLottieDrawable(R.raw.story_repost, "story_repost", dp(42), dp(42), true, null); - lottieDrawable.setMasterParent(view); - AndroidUtilities.runOnUIThread(lottieDrawable::start, 450); + if (animate) { + lottieDrawable = new RLottieDrawable(R.raw.story_repost, "story_repost", dp(42), dp(42), true, null); + lottieDrawable.setMasterParent(parentView); + AndroidUtilities.runOnUIThread(lottieDrawable::start, 450); + drawable = null; + } else { + lottieDrawable = null; + drawable = context.getResources().getDrawable(R.drawable.large_repost_story).mutate(); + drawable.setColorFilter(new PorterDuffColorFilter(Color.WHITE, PorterDuff.Mode.SRC_IN)); + } } + int alpha = 0xFF; @Override public void draw(@NonNull Canvas canvas) { canvas.save(); canvas.translate(getBounds().left, getBounds().top); - canvas.drawCircle(getBounds().width() / 2f, getBounds().height() / 2f, getBounds().width() / 2f, paint); + AndroidUtilities.rectTmp.set(0, 0, getBounds().width(), getBounds().height()); + paint.setAlpha(alpha); + float r2 = Math.min(getBounds().width(), getBounds().height()) / 2f * ((float) alpha / 0xFF); + canvas.drawRoundRect(AndroidUtilities.rectTmp, r2, r2, paint); canvas.restore(); - AndroidUtilities.rectTmp2.set(getBounds()); - AndroidUtilities.rectTmp2.inset(dp(8), dp(8)); - lottieDrawable.setBounds(AndroidUtilities.rectTmp2); - lottieDrawable.draw(canvas); + final int r = dp(lottieDrawable != null ? 20 : 15); + AndroidUtilities.rectTmp2.set( + getBounds().centerX() - r, + getBounds().centerY() - r, + getBounds().centerX() + r, + getBounds().centerY() + r + ); + Drawable drawable = lottieDrawable == null ? this.drawable : lottieDrawable; + if (drawable != null) { + drawable.setBounds(AndroidUtilities.rectTmp2); + drawable.setAlpha(alpha); + drawable.draw(canvas); + } } @Override public void setAlpha(int alpha) { - + this.alpha = alpha; } @Override diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Cells/TextCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Cells/TextCell.java index 61b7f295833..6dbc2f11bf2 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Cells/TextCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Cells/TextCell.java @@ -8,6 +8,8 @@ package org.telegram.ui.Cells; +import static org.telegram.messenger.AndroidUtilities.dp; + import android.content.Context; import android.graphics.Canvas; import android.graphics.Color; @@ -33,6 +35,7 @@ import org.telegram.ui.Components.RLottieImageView; import org.telegram.ui.Components.Switch; import org.telegram.ui.FilterCreateActivity; +import org.telegram.ui.PeerColorActivity; public class TextCell extends FrameLayout { @@ -94,11 +97,11 @@ public TextCell(Context context, int left, boolean dialog, boolean needCheck, Th valueTextView = new AnimatedTextView(context, false, false, true); valueTextView.setTextColor(Theme.getColor(dialog ? Theme.key_dialogTextBlue2 : Theme.key_windowBackgroundWhiteValueText, resourcesProvider)); - valueTextView.setPadding(0, AndroidUtilities.dp(18), 0, AndroidUtilities.dp(18)); - valueTextView.setTextSize(AndroidUtilities.dp(16)); + valueTextView.setPadding(0, dp(18), 0, dp(18)); + valueTextView.setTextSize(dp(16)); valueTextView.setGravity(LocaleController.isRTL ? Gravity.LEFT : Gravity.RIGHT); valueTextView.setImportantForAccessibility(IMPORTANT_FOR_ACCESSIBILITY_NO); - valueTextView.setTranslationY(AndroidUtilities.dp(-2)); + valueTextView.setTranslationY(dp(-2)); addView(valueTextView); valueSpoilersTextView = new SimpleTextView(context); @@ -166,7 +169,7 @@ public void setPrioritizeTitleOverValue(boolean prioritizeTitleOverValue) { @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { int width = MeasureSpec.getSize(widthMeasureSpec); - int height = AndroidUtilities.dp(heightDp); + int height = dp(heightDp); if (lastWidth != 0 && lastWidth != width && valueText != null) { valueTextView.setText(TextUtils.ellipsize(valueText, valueTextView.getPaint(), AndroidUtilities.displaySize.x / 2.5f, TextUtils.TruncateAt.END), false); @@ -175,16 +178,16 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { int valueWidth; if (prioritizeTitleOverValue) { - textView.measure(MeasureSpec.makeMeasureSpec(width - AndroidUtilities.dp(71 + leftPadding), MeasureSpec.AT_MOST), MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(20), MeasureSpec.EXACTLY)); - subtitleView.measure(MeasureSpec.makeMeasureSpec(width - AndroidUtilities.dp(71 + leftPadding), MeasureSpec.AT_MOST), MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(20), MeasureSpec.EXACTLY)); - valueTextView.measure(MeasureSpec.makeMeasureSpec(width - AndroidUtilities.dp(103 + leftPadding) - textView.getTextWidth(), LocaleController.isRTL ? MeasureSpec.AT_MOST : MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(20), MeasureSpec.EXACTLY)); - valueSpoilersTextView.measure(MeasureSpec.makeMeasureSpec(width - AndroidUtilities.dp(103 + leftPadding) - textView.getTextWidth(), LocaleController.isRTL ? MeasureSpec.AT_MOST : MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(20), MeasureSpec.EXACTLY)); + textView.measure(MeasureSpec.makeMeasureSpec(width - dp(71 + leftPadding), MeasureSpec.AT_MOST), MeasureSpec.makeMeasureSpec(dp(20), MeasureSpec.EXACTLY)); + subtitleView.measure(MeasureSpec.makeMeasureSpec(width - dp(71 + leftPadding), MeasureSpec.AT_MOST), MeasureSpec.makeMeasureSpec(dp(20), MeasureSpec.EXACTLY)); + valueTextView.measure(MeasureSpec.makeMeasureSpec(width - dp(103 + leftPadding) - textView.getTextWidth(), LocaleController.isRTL ? MeasureSpec.AT_MOST : MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec(dp(20), MeasureSpec.EXACTLY)); + valueSpoilersTextView.measure(MeasureSpec.makeMeasureSpec(width - dp(103 + leftPadding) - textView.getTextWidth(), LocaleController.isRTL ? MeasureSpec.AT_MOST : MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec(dp(20), MeasureSpec.EXACTLY)); } else { - valueTextView.measure(MeasureSpec.makeMeasureSpec(width - AndroidUtilities.dp(leftPadding), LocaleController.isRTL ? MeasureSpec.AT_MOST : MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(20), MeasureSpec.EXACTLY)); - valueSpoilersTextView.measure(MeasureSpec.makeMeasureSpec(width - AndroidUtilities.dp(leftPadding), LocaleController.isRTL ? MeasureSpec.AT_MOST : MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(20), MeasureSpec.EXACTLY)); + valueTextView.measure(MeasureSpec.makeMeasureSpec(width - dp(leftPadding), LocaleController.isRTL ? MeasureSpec.AT_MOST : MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec(dp(20), MeasureSpec.EXACTLY)); + valueSpoilersTextView.measure(MeasureSpec.makeMeasureSpec(width - dp(leftPadding), LocaleController.isRTL ? MeasureSpec.AT_MOST : MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec(dp(20), MeasureSpec.EXACTLY)); valueWidth = Math.max(valueTextView.width(), valueSpoilersTextView.getTextWidth()); - textView.measure(MeasureSpec.makeMeasureSpec(Math.max(0, width - AndroidUtilities.dp(71 + leftPadding) - valueWidth), MeasureSpec.AT_MOST), MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(20), MeasureSpec.EXACTLY)); - subtitleView.measure(MeasureSpec.makeMeasureSpec(width - AndroidUtilities.dp(71 + leftPadding) - valueWidth, MeasureSpec.AT_MOST), MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(20), MeasureSpec.EXACTLY)); + textView.measure(MeasureSpec.makeMeasureSpec(Math.max(0, width - dp(71 + leftPadding) - valueWidth), MeasureSpec.AT_MOST), MeasureSpec.makeMeasureSpec(dp(20), MeasureSpec.EXACTLY)); + subtitleView.measure(MeasureSpec.makeMeasureSpec(width - dp(71 + leftPadding) - valueWidth, MeasureSpec.AT_MOST), MeasureSpec.makeMeasureSpec(dp(20), MeasureSpec.EXACTLY)); } if (imageView.getVisibility() == VISIBLE) { imageView.measure(MeasureSpec.makeMeasureSpec(width, MeasureSpec.AT_MOST), MeasureSpec.makeMeasureSpec(height, MeasureSpec.AT_MOST)); @@ -193,7 +196,7 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { valueImageView.measure(MeasureSpec.makeMeasureSpec(width, MeasureSpec.AT_MOST), MeasureSpec.makeMeasureSpec(height, MeasureSpec.AT_MOST)); } if (checkBox != null) { - checkBox.measure(MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(37), MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(20), MeasureSpec.EXACTLY)); + checkBox.measure(MeasureSpec.makeMeasureSpec(dp(37), MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec(dp(20), MeasureSpec.EXACTLY)); } setMeasuredDimension(width, height + (needDivider ? 1 : 0)); } @@ -212,43 +215,43 @@ protected void onLayout(boolean changed, int left, int top, int right, int botto int width = right - left; int viewTop = (height - Math.max(valueSpoilersTextView.getTextHeight(), valueTextView.getTextHeight())) / 2; - int viewLeft = LocaleController.isRTL ? AndroidUtilities.dp(leftPadding) : width - valueTextView.getMeasuredWidth() - AndroidUtilities.dp(leftPadding); + int viewLeft = LocaleController.isRTL ? dp(leftPadding) : width - valueTextView.getMeasuredWidth() - dp(leftPadding); if (prioritizeTitleOverValue && !LocaleController.isRTL) { - viewLeft = width - valueTextView.getMeasuredWidth() - AndroidUtilities.dp(leftPadding); + viewLeft = width - valueTextView.getMeasuredWidth() - dp(leftPadding); } valueTextView.layout(viewLeft, viewTop, viewLeft + valueTextView.getMeasuredWidth(), viewTop + valueTextView.getMeasuredHeight()); - viewLeft = LocaleController.isRTL ? AndroidUtilities.dp(leftPadding) : width - valueSpoilersTextView.getMeasuredWidth() - AndroidUtilities.dp(leftPadding); + viewLeft = LocaleController.isRTL ? dp(leftPadding) : width - valueSpoilersTextView.getMeasuredWidth() - dp(leftPadding); valueSpoilersTextView.layout(viewLeft, viewTop, viewLeft + valueSpoilersTextView.getMeasuredWidth(), viewTop + valueSpoilersTextView.getMeasuredHeight()); if (LocaleController.isRTL) { - viewLeft = getMeasuredWidth() - textView.getMeasuredWidth() - AndroidUtilities.dp(imageView.getVisibility() == VISIBLE ? offsetFromImage : leftPadding); + viewLeft = getMeasuredWidth() - textView.getMeasuredWidth() - dp(imageView.getVisibility() == VISIBLE ? offsetFromImage : leftPadding); } else { - viewLeft = AndroidUtilities.dp(imageView.getVisibility() == VISIBLE ? offsetFromImage : leftPadding); + viewLeft = dp(imageView.getVisibility() == VISIBLE ? offsetFromImage : leftPadding); } if (subtitleView.getVisibility() == View.VISIBLE) { int margin = heightDp > 50 ? 4 : 2; - viewTop = (height - textView.getTextHeight() - subtitleView.getTextHeight() - AndroidUtilities.dp(margin)) / 2; + viewTop = (height - textView.getTextHeight() - subtitleView.getTextHeight() - dp(margin)) / 2; textView.layout(viewLeft, viewTop, viewLeft + textView.getMeasuredWidth(), viewTop + textView.getMeasuredHeight()); - viewTop = viewTop + textView.getTextHeight() + AndroidUtilities.dp(margin); + viewTop = viewTop + textView.getTextHeight() + dp(margin); subtitleView.layout(viewLeft, viewTop, viewLeft + subtitleView.getMeasuredWidth(), viewTop + subtitleView.getMeasuredHeight()); } else { viewTop = (height - textView.getTextHeight()) / 2; textView.layout(viewLeft, viewTop, viewLeft + textView.getMeasuredWidth(), viewTop + textView.getMeasuredHeight()); } if (imageView.getVisibility() == VISIBLE) { - viewTop = AndroidUtilities.dp(heightDp > 50 ? 0 : 2) + (height - imageView.getMeasuredHeight()) / 2 - imageView.getPaddingTop(); - viewLeft = !LocaleController.isRTL ? AndroidUtilities.dp(imageLeft) : width - imageView.getMeasuredWidth() - AndroidUtilities.dp(imageLeft); + viewTop = dp(heightDp > 50 ? 0 : 2) + (height - imageView.getMeasuredHeight()) / 2 - imageView.getPaddingTop(); + viewLeft = !LocaleController.isRTL ? dp(imageLeft) : width - imageView.getMeasuredWidth() - dp(imageLeft); imageView.layout(viewLeft, viewTop, viewLeft + imageView.getMeasuredWidth(), viewTop + imageView.getMeasuredHeight()); } if (valueImageView.getVisibility() == VISIBLE) { viewTop = (height - valueImageView.getMeasuredHeight()) / 2; - viewLeft = LocaleController.isRTL ? AndroidUtilities.dp(23) : width - valueImageView.getMeasuredWidth() - AndroidUtilities.dp(23); + viewLeft = LocaleController.isRTL ? dp(23) : width - valueImageView.getMeasuredWidth() - dp(23); valueImageView.layout(viewLeft, viewTop, viewLeft + valueImageView.getMeasuredWidth(), viewTop + valueImageView.getMeasuredHeight()); } if (checkBox != null && checkBox.getVisibility() == VISIBLE) { viewTop = (height - checkBox.getMeasuredHeight()) / 2; - viewLeft = LocaleController.isRTL ? AndroidUtilities.dp(22) : width - checkBox.getMeasuredWidth() - AndroidUtilities.dp(22); + viewLeft = LocaleController.isRTL ? dp(22) : width - checkBox.getMeasuredWidth() - dp(22); checkBox.layout(viewLeft, viewTop, viewLeft + checkBox.getMeasuredWidth(), viewTop + checkBox.getMeasuredHeight()); } } @@ -286,6 +289,7 @@ public void setColors(int icon, int text) { public void setText(String text, boolean divider) { imageLeft = 21; textView.setText(text); + textView.setRightDrawable(null); valueTextView.setText(valueText = null, false); imageView.setVisibility(GONE); valueTextView.setVisibility(GONE); @@ -295,25 +299,32 @@ public void setText(String text, boolean divider) { setWillNotDraw(!needDivider); } - public void setTextAndIcon(String text, int resId, boolean divider) { + public void setLockLevel(boolean plus, int level) { + textView.setRightDrawable(new PeerColorActivity.LevelLock(getContext(), plus, level, resourcesProvider)); + textView.setDrawablePadding(dp(6)); + } + + public void setTextAndIcon(CharSequence text, int resId, boolean divider) { imageLeft = 21; offsetFromImage = getOffsetFromImage(false); textView.setText(text); + textView.setRightDrawable(null); valueTextView.setText(valueText = null, false); imageView.setImageResource(resId); imageView.setVisibility(VISIBLE); valueTextView.setVisibility(GONE); valueSpoilersTextView.setVisibility(GONE); valueImageView.setVisibility(GONE); - imageView.setPadding(0, AndroidUtilities.dp(7), 0, 0); + imageView.setPadding(0, dp(7), 0, 0); needDivider = divider; setWillNotDraw(!needDivider); } - public void setTextAndColorfulIcon(String text, int resId, int color, boolean divider) { + public void setTextAndColorfulIcon(CharSequence text, int resId, int color, boolean divider) { imageLeft = 21; offsetFromImage = 71; textView.setText(text); + textView.setRightDrawable(null); valueTextView.setText(valueText = null, false); setColorfulIcon(color, resId); valueTextView.setVisibility(GONE); @@ -326,6 +337,7 @@ public void setTextAndIcon(String text, Drawable drawable, boolean divider) { offsetFromImage = 68; imageLeft = 18; textView.setText(text); + textView.setRightDrawable(null); valueTextView.setText(valueText = null, false); imageView.setColorFilter(null); if (drawable instanceof RLottieDrawable) { @@ -336,7 +348,7 @@ public void setTextAndIcon(String text, Drawable drawable, boolean divider) { imageView.setVisibility(VISIBLE); valueTextView.setVisibility(GONE); valueImageView.setVisibility(GONE); - imageView.setPadding(0, AndroidUtilities.dp(6), 0, 0); + imageView.setPadding(0, dp(6), 0, 0); needDivider = divider; setWillNotDraw(!needDivider); } @@ -357,6 +369,7 @@ public void setTextAndValue(String text, String value, boolean animated, boolean imageLeft = 21; offsetFromImage = getOffsetFromImage(false); textView.setText(text); + textView.setRightDrawable(null); valueTextView.setText(TextUtils.ellipsize(valueText = value, valueTextView.getPaint(), AndroidUtilities.displaySize.x / 2.5f, TextUtils.TruncateAt.END), animated); valueTextView.setVisibility(VISIBLE); valueSpoilersTextView.setVisibility(GONE); @@ -373,6 +386,7 @@ public void setTextAndValueAndColorfulIcon(String text, CharSequence value, bool imageLeft = 21; offsetFromImage = getOffsetFromImage(false); textView.setText(text); + textView.setRightDrawable(null); valueTextView.setText(TextUtils.ellipsize(valueText = value, valueTextView.getPaint(), AndroidUtilities.displaySize.x / 2.5f, TextUtils.TruncateAt.END), animated); valueTextView.setVisibility(VISIBLE); valueSpoilersTextView.setVisibility(GONE); @@ -389,6 +403,7 @@ public void setTextAndSpoilersValueAndIcon(String text, CharSequence value, int imageLeft = 21; offsetFromImage = getOffsetFromImage(false); textView.setText(text); + textView.setRightDrawable(null); valueSpoilersTextView.setVisibility(VISIBLE); valueSpoilersTextView.setText(value); valueTextView.setVisibility(GONE); @@ -396,7 +411,7 @@ public void setTextAndSpoilersValueAndIcon(String text, CharSequence value, int imageView.setVisibility(VISIBLE); imageView.setTranslationX(0); imageView.setTranslationY(0); - imageView.setPadding(0, AndroidUtilities.dp(7), 0, 0); + imageView.setPadding(0, dp(7), 0, 0); imageView.setImageResource(resId); needDivider = divider; setWillNotDraw(!needDivider); @@ -409,6 +424,7 @@ public void setTextAndSpoilersValueAndColorfulIcon(String text, CharSequence val imageLeft = 21; offsetFromImage = getOffsetFromImage(false); textView.setText(text); + textView.setRightDrawable(null); valueSpoilersTextView.setVisibility(VISIBLE); valueSpoilersTextView.setText(value); valueTextView.setVisibility(GONE); @@ -429,6 +445,7 @@ public void setTextAndValueAndIcon(CharSequence text, String value, boolean anim imageLeft = 21; offsetFromImage = getOffsetFromImage(false); textView.setText(text); + textView.setRightDrawable(null); valueTextView.setText(TextUtils.ellipsize(valueText = value, valueTextView.getPaint(), AndroidUtilities.displaySize.x / 2.5f, TextUtils.TruncateAt.END), animated); valueTextView.setVisibility(VISIBLE); valueSpoilersTextView.setVisibility(GONE); @@ -436,7 +453,7 @@ public void setTextAndValueAndIcon(CharSequence text, String value, boolean anim imageView.setVisibility(VISIBLE); imageView.setTranslationX(0); imageView.setTranslationY(0); - imageView.setPadding(0, AndroidUtilities.dp(7), 0, 0); + imageView.setPadding(0, dp(7), 0, 0); imageView.setImageResource(resId); needDivider = divider; setWillNotDraw(!needDivider); @@ -445,7 +462,7 @@ public void setTextAndValueAndIcon(CharSequence text, String value, boolean anim } } - public static CharSequence applyNewSpan(String str) { + public static CharSequence applyNewSpan(CharSequence str) { SpannableStringBuilder spannableStringBuilder = new SpannableStringBuilder(str); spannableStringBuilder.append(" d"); FilterCreateActivity.NewSpan span = new FilterCreateActivity.NewSpan(10); @@ -457,17 +474,18 @@ public static CharSequence applyNewSpan(String str) { public void setColorfulIcon(int color, int resId) { offsetFromImage = getOffsetFromImage(true); imageView.setVisibility(VISIBLE); - imageView.setPadding(AndroidUtilities.dp(2), AndroidUtilities.dp(2), AndroidUtilities.dp(2), AndroidUtilities.dp(2)); - imageView.setTranslationX(AndroidUtilities.dp(LocaleController.isRTL ? 0 : -3)); + imageView.setPadding(dp(2), dp(2), dp(2), dp(2)); + imageView.setTranslationX(dp(LocaleController.isRTL ? 0 : -3)); imageView.setImageResource(resId); imageView.setColorFilter(new PorterDuffColorFilter(Color.WHITE, PorterDuff.Mode.SRC_IN)); - imageView.setBackground(Theme.createRoundRectDrawable(AndroidUtilities.dp(9), color)); + imageView.setBackground(Theme.createRoundRectDrawable(dp(9), color)); } public void setTextAndCheck(CharSequence text, boolean checked, boolean divider) { imageLeft = 21; offsetFromImage = getOffsetFromImage(false); textView.setText(text); + textView.setRightDrawable(null); imageView.setVisibility(GONE); valueImageView.setVisibility(GONE); needDivider = divider; @@ -482,6 +500,7 @@ public void setTextAndCheckAndIcon(CharSequence text, boolean checked, int resId imageLeft = 21; offsetFromImage = getOffsetFromImage(false); textView.setText(text); + textView.setRightDrawable(null); valueTextView.setVisibility(GONE); valueSpoilersTextView.setVisibility(GONE); valueImageView.setVisibility(GONE); @@ -490,7 +509,7 @@ public void setTextAndCheckAndIcon(CharSequence text, boolean checked, int resId checkBox.setChecked(checked, false); } imageView.setVisibility(VISIBLE); - imageView.setPadding(0, AndroidUtilities.dp(7), 0, 0); + imageView.setPadding(0, dp(7), 0, 0); imageView.setImageResource(resId); needDivider = divider; setWillNotDraw(!needDivider); @@ -500,6 +519,7 @@ public void setTextAndCheckAndIcon(String text, boolean checked, Drawable resDra imageLeft = 21; offsetFromImage = getOffsetFromImage(false); textView.setText(text); + textView.setRightDrawable(null); valueTextView.setVisibility(GONE); valueSpoilersTextView.setVisibility(GONE); valueImageView.setVisibility(GONE); @@ -508,7 +528,7 @@ public void setTextAndCheckAndIcon(String text, boolean checked, Drawable resDra checkBox.setChecked(checked, false); } imageView.setVisibility(VISIBLE); - imageView.setPadding(0, AndroidUtilities.dp(7), 0, 0); + imageView.setPadding(0, dp(7), 0, 0); imageView.setImageDrawable(resDrawable); needDivider = divider; setWillNotDraw(!needDivider); @@ -518,13 +538,14 @@ public void setTextAndValueDrawable(String text, Drawable drawable, boolean divi imageLeft = 21; offsetFromImage = getOffsetFromImage(false); textView.setText(text); + textView.setRightDrawable(null); valueTextView.setText(valueText = null, false); valueImageView.setVisibility(VISIBLE); valueImageView.setImageDrawable(drawable); valueTextView.setVisibility(GONE); valueSpoilersTextView.setVisibility(GONE); imageView.setVisibility(GONE); - imageView.setPadding(0, AndroidUtilities.dp(7), 0, 0); + imageView.setPadding(0, dp(7), 0, 0); needDivider = divider; setWillNotDraw(!needDivider); if (checkBox != null) { @@ -543,7 +564,7 @@ protected void onDraw(Canvas canvas) { if (paint == null) { paint = Theme.dividerPaint; } - canvas.drawLine(LocaleController.isRTL ? 0 : AndroidUtilities.dp(imageView.getVisibility() == VISIBLE ? (inDialogs ? 72 : 68) : 20), getMeasuredHeight() - 1, getMeasuredWidth() - (LocaleController.isRTL ? AndroidUtilities.dp(imageView.getVisibility() == VISIBLE ? (inDialogs ? 72 : 68) : 20) : 0), getMeasuredHeight() - 1, paint); + canvas.drawLine(LocaleController.isRTL ? 0 : dp(imageView.getVisibility() == VISIBLE ? (inDialogs ? 72 : 68) : 20), getMeasuredHeight() - 1, getMeasuredWidth() - (LocaleController.isRTL ? dp(imageView.getVisibility() == VISIBLE ? (inDialogs ? 72 : 68) : 20) : 0), getMeasuredHeight() - 1, paint); } } @@ -693,16 +714,16 @@ protected void dispatchDraw(Canvas canvas) { paint.setAlpha((int) (255 * alpha)); int cy = getMeasuredHeight() >> 1; AndroidUtilities.rectTmp.set( - getMeasuredWidth() - AndroidUtilities.dp(21) - AndroidUtilities.dp(loadingSize), - cy - AndroidUtilities.dp(3), - getMeasuredWidth() - AndroidUtilities.dp(21), - cy + AndroidUtilities.dp(3) + getMeasuredWidth() - dp(21) - dp(loadingSize), + cy - dp(3), + getMeasuredWidth() - dp(21), + cy + dp(3) ); if (LocaleController.isRTL) { AndroidUtilities.rectTmp.left = getMeasuredWidth() - AndroidUtilities.rectTmp.left; AndroidUtilities.rectTmp.right = getMeasuredWidth() - AndroidUtilities.rectTmp.right; } - canvas.drawRoundRect(AndroidUtilities.rectTmp, AndroidUtilities.dp(3), AndroidUtilities.dp(3), paint); + canvas.drawRoundRect(AndroidUtilities.rectTmp, dp(3), dp(3), paint); invalidate(); } valueTextView.setAlpha(1f - drawLoadingProgress); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Cells/TextCheckCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Cells/TextCheckCell.java index c605dab721d..a248222404e 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Cells/TextCheckCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Cells/TextCheckCell.java @@ -385,10 +385,13 @@ protected void onDraw(Canvas canvas) { canvas.drawCircle(cx, cy, animatedRad, animationPaint); } if (needDivider) { - if (imageView != null) { - canvas.drawLine(LocaleController.isRTL ? 0 : padding, getMeasuredHeight() - 1, getMeasuredWidth() - (LocaleController.isRTL ? padding : 0), getMeasuredHeight() - 1, Theme.dividerPaint); - } else { - canvas.drawLine(LocaleController.isRTL ? 0 : AndroidUtilities.dp(20), getMeasuredHeight() - 1, getMeasuredWidth() - (LocaleController.isRTL ? AndroidUtilities.dp(20) : 0), getMeasuredHeight() - 1, Theme.dividerPaint); + Paint dividerPaint = resourcesProvider != null ? resourcesProvider.getPaint(Theme.key_paint_divider) : Theme.dividerPaint; + if (dividerPaint != null) { + if (imageView != null) { + canvas.drawLine(LocaleController.isRTL ? 0 : padding, getMeasuredHeight() - 1, getMeasuredWidth() - (LocaleController.isRTL ? padding : 0), getMeasuredHeight() - 1, dividerPaint); + } else { + canvas.drawLine(LocaleController.isRTL ? 0 : AndroidUtilities.dp(20), getMeasuredHeight() - 1, getMeasuredWidth() - (LocaleController.isRTL ? AndroidUtilities.dp(20) : 0), getMeasuredHeight() - 1, dividerPaint); + } } } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Cells/ThemePreviewMessagesCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Cells/ThemePreviewMessagesCell.java index 8294b7c2ee1..9e32989723c 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Cells/ThemePreviewMessagesCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Cells/ThemePreviewMessagesCell.java @@ -16,6 +16,7 @@ import android.view.ViewTreeObserver; import android.widget.LinearLayout; +import androidx.annotation.NonNull; import androidx.core.content.ContextCompat; import org.telegram.messenger.AndroidUtilities; @@ -30,7 +31,9 @@ import org.telegram.ui.ActionBar.INavigationLayout; import org.telegram.ui.ActionBar.Theme; import org.telegram.ui.ChatActivity; +import org.telegram.ui.ChatBackgroundDrawable; import org.telegram.ui.Components.AnimatedColor; +import org.telegram.ui.Components.AnimatedFloat; import org.telegram.ui.Components.AvatarDrawable; import org.telegram.ui.Components.BackgroundGradientDrawable; import org.telegram.ui.Components.CubicBezierInterpolator; @@ -38,6 +41,7 @@ import org.telegram.ui.Components.MotionBackgroundDrawable; import org.telegram.ui.Components.Reactions.ReactionsEffectOverlay; import org.telegram.ui.Components.Reactions.ReactionsLayoutInBubble; +import org.telegram.ui.Stories.recorder.StoryEntry; public class ThemePreviewMessagesCell extends LinearLayout { @@ -340,10 +344,7 @@ protected void dispatchDraw(Canvas canvas) { if (getMessageObject() != null && getMessageObject().overrideLinkColor >= 0) { final int colorId = getMessageObject().overrideLinkColor; final int color1, color2; - if (getMessageObject().overrideProfilePeerColor != null) { - color1 = getMessageObject().overrideProfilePeerColor.getAvatarColor1(); - color2 = getMessageObject().overrideProfilePeerColor.getAvatarColor2(); - } else if (colorId >= 14) { + if (colorId >= 14) { MessagesController messagesController = MessagesController.getInstance(UserConfig.selectedAccount); MessagesController.PeerColors peerColors = messagesController != null ? messagesController.peerColors : null; MessagesController.PeerColor peerColor = peerColors != null ? peerColors.getColor(colorId) : null; @@ -443,9 +444,32 @@ public void invalidate() { private Drawable overrideDrawable; public void setOverrideBackground(Drawable drawable) { overrideDrawable = drawable; + if (overrideDrawable != null) { + overrideDrawable.setCallback(this); + } + if (overrideDrawable instanceof ChatBackgroundDrawable) { + if (isAttachedToWindow()) { + ((ChatBackgroundDrawable) overrideDrawable).onAttachedToWindow(this); + } + } invalidate(); } + @Override + protected void onAttachedToWindow() { + super.onAttachedToWindow(); + if (overrideDrawable instanceof ChatBackgroundDrawable) { + ((ChatBackgroundDrawable) overrideDrawable).onAttachedToWindow(this); + } + } + + @Override + protected boolean verifyDrawable(@NonNull Drawable who) { + return who == overrideDrawable || who == oldBackgroundDrawable || super.verifyDrawable(who); + } + + public boolean customAnimation; + private final AnimatedFloat overrideDrawableUpdate = new AnimatedFloat(this, 0, 350, CubicBezierInterpolator.EASE_OUT_QUINT); @Override protected void onDraw(Canvas canvas) { @@ -454,7 +478,7 @@ protected void onDraw(Canvas canvas) { invalidate(); } if (newDrawable != backgroundDrawable && newDrawable != null) { - if (Theme.isAnimatingColor()) { + if (Theme.isAnimatingColor() || customAnimation) { oldBackgroundDrawable = backgroundDrawable; oldBackgroundGradientDisposable = backgroundGradientDisposable; } else if (backgroundGradientDisposable != null) { @@ -462,15 +486,16 @@ protected void onDraw(Canvas canvas) { backgroundGradientDisposable = null; } backgroundDrawable = newDrawable; + overrideDrawableUpdate.set(0, true); } - float themeAnimationValue = parentLayout.getThemeAnimationValue(); + float themeAnimationValue = customAnimation ? overrideDrawableUpdate.set(1) : parentLayout.getThemeAnimationValue(); for (int a = 0; a < 2; a++) { Drawable drawable = a == 0 ? oldBackgroundDrawable : backgroundDrawable; if (drawable == null) { continue; } int alpha; - if (a == 1 && oldBackgroundDrawable != null && parentLayout != null) { + if (a == 1 && oldBackgroundDrawable != null && (parentLayout != null || customAnimation)) { alpha = (int) (255 * themeAnimationValue); } else { alpha = 255; @@ -510,6 +535,8 @@ protected void onDraw(Canvas canvas) { } drawable.draw(canvas); canvas.restore(); + } else { + StoryEntry.drawBackgroundDrawable(canvas, drawable, getWidth(), getHeight()); } if (a == 0 && oldBackgroundDrawable != null && themeAnimationValue >= 1.0f) { if (oldBackgroundGradientDisposable != null) { @@ -539,6 +566,9 @@ protected void onDetachedFromWindow() { oldBackgroundGradientDisposable.dispose(); oldBackgroundGradientDisposable = null; } + if (overrideDrawable instanceof ChatBackgroundDrawable) { + ((ChatBackgroundDrawable) overrideDrawable).onDetachedFromWindow(this); + } } @Override diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Cells/WallpaperCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Cells/WallpaperCell.java index 6b39900c314..a4f9d6ce39b 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Cells/WallpaperCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Cells/WallpaperCell.java @@ -354,7 +354,7 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { int width = MeasureSpec.getSize(widthMeasureSpec); int availableWidth = width - AndroidUtilities.dp(14 * 2 + 6 * (spanCount - 1)); int itemWidth = availableWidth / spanCount; - int height = currentType == WallpapersListActivity.TYPE_ALL ? AndroidUtilities.dp(180) : itemWidth; + int height = currentType == WallpapersListActivity.TYPE_ALL || currentType == WallpapersListActivity.TYPE_CHANNEL_PATTERNS || currentType == WallpapersListActivity.TYPE_CHANNEL_CUSTOM ? AndroidUtilities.dp(180) : itemWidth; setMeasuredDimension(width, height + (isTop ? AndroidUtilities.dp(14) : 0) + (AndroidUtilities.dp(isBottom ? 14 : 6))); for (int a = 0; a < spanCount; a++) { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/ChannelAdminLogActivity.java b/TMessagesProj/src/main/java/org/telegram/ui/ChannelAdminLogActivity.java index a03937c1124..9f935c9e3bb 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/ChannelAdminLogActivity.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/ChannelAdminLogActivity.java @@ -2377,7 +2377,7 @@ public void didPressSideButton(ChatMessageCell cell) { } @Override - public boolean needPlayMessage(MessageObject messageObject, boolean muted) { + public boolean needPlayMessage(ChatMessageCell cell, MessageObject messageObject, boolean muted) { if (messageObject.isVoice() || messageObject.isRoundVideo()) { boolean result = MediaController.getInstance().playMessage(messageObject, muted); MediaController.getInstance().setVoiceMessagesPlaylist(null, false); @@ -2674,6 +2674,10 @@ public void onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo info) { @Override public void didClickImage(ChatActionCell cell) { MessageObject message = cell.getMessageObject(); + if (message.type == MessageObject.TYPE_ACTION_WALLPAPER) { + presentFragment(new ChannelColorActivity(getDialogId()).setOnApplied(ChannelAdminLogActivity.this)); + return; + } PhotoViewer.getInstance().setParentActivity(ChannelAdminLogActivity.this); TLRPC.PhotoSize photoSize = FileLoader.getClosestPhotoSizeWithSize(message.photoThumbs, 640); if (photoSize != null) { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/ChannelColorActivity.java b/TMessagesProj/src/main/java/org/telegram/ui/ChannelColorActivity.java new file mode 100644 index 00000000000..c255fb16918 --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/ChannelColorActivity.java @@ -0,0 +1,2185 @@ +package org.telegram.ui; + +import static org.telegram.messenger.AndroidUtilities.dp; +import static org.telegram.messenger.AndroidUtilities.dpf2; + +import android.animation.Animator; +import android.animation.AnimatorListenerAdapter; +import android.animation.ValueAnimator; +import android.annotation.SuppressLint; +import android.app.Activity; +import android.content.Context; +import android.content.DialogInterface; +import android.content.SharedPreferences; +import android.graphics.Bitmap; +import android.graphics.BitmapShader; +import android.graphics.Canvas; +import android.graphics.Color; +import android.graphics.Paint; +import android.graphics.Path; +import android.graphics.PorterDuff; +import android.graphics.PorterDuffXfermode; +import android.graphics.Shader; +import android.graphics.drawable.Drawable; +import android.net.Uri; +import android.os.Build; +import android.os.Bundle; +import android.text.SpannableStringBuilder; +import android.text.Spanned; +import android.text.TextUtils; +import android.util.SparseIntArray; +import android.view.Gravity; +import android.view.MotionEvent; +import android.view.View; +import android.view.ViewGroup; +import android.widget.FrameLayout; +import android.widget.TextView; +import android.widget.Toast; + +import androidx.annotation.NonNull; +import androidx.recyclerview.widget.DefaultItemAnimator; +import androidx.recyclerview.widget.GridLayoutManager; +import androidx.recyclerview.widget.LinearLayoutManager; +import androidx.recyclerview.widget.RecyclerView; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.ApplicationLoader; +import org.telegram.messenger.ChatObject; +import org.telegram.messenger.ChatThemeController; +import org.telegram.messenger.DialogObject; +import org.telegram.messenger.FileLoader; +import org.telegram.messenger.FileLog; +import org.telegram.messenger.LocaleController; +import org.telegram.messenger.MessageObject; +import org.telegram.messenger.MessagesController; +import org.telegram.messenger.NotificationCenter; +import org.telegram.messenger.R; +import org.telegram.messenger.UserObject; +import org.telegram.messenger.Utilities; +import org.telegram.tgnet.ConnectionsManager; +import org.telegram.tgnet.ResultCallback; +import org.telegram.tgnet.TLRPC; +import org.telegram.tgnet.tl.TL_stories; +import org.telegram.ui.ActionBar.ActionBar; +import org.telegram.ui.ActionBar.ActionBarMenuItem; +import org.telegram.ui.ActionBar.AlertDialog; +import org.telegram.ui.ActionBar.BaseFragment; +import org.telegram.ui.ActionBar.EmojiThemes; +import org.telegram.ui.ActionBar.SimpleTextView; +import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.ActionBar.ThemeColors; +import org.telegram.ui.Cells.ChatMessageCell; +import org.telegram.ui.Cells.TextCell; +import org.telegram.ui.Cells.TextInfoPrivacyCell; +import org.telegram.ui.Cells.ThemePreviewMessagesCell; +import org.telegram.ui.Cells.ThemesHorizontalListCell; +import org.telegram.ui.Components.AnimatedEmojiDrawable; +import org.telegram.ui.Components.AnimatedFloat; +import org.telegram.ui.Components.BulletinFactory; +import org.telegram.ui.Components.ButtonBounce; +import org.telegram.ui.Components.ChatThemeBottomSheet; +import org.telegram.ui.Components.ColoredImageSpan; +import org.telegram.ui.Components.CubicBezierInterpolator; +import org.telegram.ui.Components.Easings; +import org.telegram.ui.Components.FlickerLoadingView; +import org.telegram.ui.Components.LayoutHelper; +import org.telegram.ui.Components.Premium.LimitReachedBottomSheet; +import org.telegram.ui.Components.RLottieDrawable; +import org.telegram.ui.Components.RecyclerListView; +import org.telegram.ui.Components.Text; +import org.telegram.ui.Components.ThemeSmallPreviewView; +import org.telegram.ui.Stories.recorder.ButtonWithCounterView; +import org.telegram.ui.Stories.recorder.PreviewView; + +import java.io.File; +import java.io.FileInputStream; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; + +public class ChannelColorActivity extends BaseFragment { + + public final long dialogId; + public int currentLevel; + public TL_stories.TL_premium_boostsStatus boostsStatus; + + public int currentReplyColor, selectedReplyColor; + public long currentReplyEmoji, selectedReplyEmoji; + public int currentProfileColor, selectedProfileColor; + public long currentProfileEmoji, selectedProfileEmoji; + public TLRPC.EmojiStatus currentStatusEmoji, selectedStatusEmoji; + public TLRPC.WallPaper currentWallpaper, selectedWallpaper; + public TLRPC.WallPaper galleryWallpaper; + + public Drawable backgroundDrawable; + + public int minLevelRequired() { + int lvl = 0; + if (currentReplyColor != selectedReplyColor) { + MessagesController.PeerColors peerColors = getMessagesController().peerColors; + MessagesController.PeerColor peerColor = peerColors == null ? null : peerColors.getColor(selectedReplyColor); + if (peerColor != null) { + lvl = Math.max(lvl, peerColor.lvl); + } + } + if (currentReplyEmoji != selectedReplyEmoji) { + lvl = Math.max(lvl, getMessagesController().channelBgIconLevelMin); + } + if (currentProfileColor != selectedProfileColor) { + MessagesController.PeerColors peerColors = getMessagesController().profilePeerColors; + MessagesController.PeerColor peerColor = peerColors == null ? null : peerColors.getColor(selectedProfileColor); + if (peerColor != null) { + lvl = Math.max(lvl, peerColor.lvl); + } + } + if (currentProfileEmoji != selectedProfileEmoji) { + lvl = Math.max(lvl, getMessagesController().channelProfileIconLevelMin); + } + if (!DialogObject.emojiStatusesEqual(currentStatusEmoji, selectedStatusEmoji)) { + lvl = Math.max(lvl, getMessagesController().channelEmojiStatusLevelMin); + } + if (!ChatThemeController.wallpaperEquals(currentWallpaper, selectedWallpaper)) { + if (!TextUtils.isEmpty(ChatThemeController.getWallpaperEmoticon(selectedWallpaper))) { + lvl = Math.max(lvl, getMessagesController().channelWallpaperLevelMin); + } else { + lvl = Math.max(lvl, getMessagesController().channelCustomWallpaperLevelMin); + } + } + return lvl; + } + + private SpannableStringBuilder lock; + public void updateButton(boolean animated) { + if (boostsStatus == null) { + return; + } + int minLevel = minLevelRequired(); + if (currentLevel >= minLevel) { + button.setSubText(null, animated); + } else { + if (lock == null) { + lock = new SpannableStringBuilder("l"); + ColoredImageSpan coloredImageSpan = new ColoredImageSpan(R.drawable.mini_switch_lock); + coloredImageSpan.setTopOffset(1); + lock.setSpan(coloredImageSpan, 0, 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + } + SpannableStringBuilder buttonLockedText = new SpannableStringBuilder(); + buttonLockedText.append(lock).append(LocaleController.formatPluralString("BoostLevelRequired", minLevel)); + button.setSubText(buttonLockedText, animated); + } + } + + public class ThemeDelegate implements Theme.ResourcesProvider { + @Override + public int getColor(int key) { + int index = currentColors.indexOfKey(key); + if (index >= 0) { + return currentColors.valueAt(index); + } + if (parentResourcesProvider != null) { + return parentResourcesProvider.getColor(key); + } + return Theme.getColor(key); + } + + @Override + public Drawable getDrawable(String drawableKey) { + if (drawableKey.equals(Theme.key_drawable_msgIn)) { + return msgInDrawable; + } + if (drawableKey.equals(Theme.key_drawable_msgInSelected)) { + return msgInDrawableSelected; + } + if (parentResourcesProvider != null) { + return parentResourcesProvider.getDrawable(drawableKey); + } + return Theme.getThemeDrawable(drawableKey); + } + + @Override + public Paint getPaint(String paintKey) { + if (paintKey.equals(Theme.key_paint_divider)) { + return dividerPaint; + } + return Theme.ResourcesProvider.super.getPaint(paintKey); + } + + @Override + public boolean isDark() { + return isDark; + } + + public void toggle() { + isDark = !isDark; + updateThemeColors(); + updateColors(); + } + } + + @Override + public boolean onFragmentCreate() { + getMediaDataController().loadRestrictedStatusEmojis(); + return super.onFragmentCreate(); + } + + public ChannelColorActivity(long dialogId) { + super(); + this.dialogId = dialogId; + TLRPC.Chat chat = getMessagesController().getChat(-dialogId); + if (chat != null) { + currentLevel = chat.level; + } + MessagesController.getInstance(currentAccount).getBoostsController().getBoostsStats(dialogId, boostsStatus -> { + this.boostsStatus = boostsStatus; + if (boostsStatus != null) { + this.currentLevel = boostsStatus.level; + if (chat != null) { + chat.flags |= 1024; + chat.level = currentLevel; + } + } + updateButton(true); + if (button != null) { + button.setLoading(false); + } + }); + + resourceProvider = new ThemeDelegate(); + msgInDrawable = new Theme.MessageDrawable(Theme.MessageDrawable.TYPE_TEXT, false, false, resourceProvider); + msgInDrawableSelected = new Theme.MessageDrawable(Theme.MessageDrawable.TYPE_TEXT, false, true, resourceProvider); + } + + @Override + public void setResourceProvider(Theme.ResourcesProvider resourceProvider) { + parentResourcesProvider = resourceProvider; + } + + private boolean isDark = Theme.isCurrentThemeDark(); + private RLottieDrawable sunDrawable; + private ActionBarMenuItem dayNightItem; + + private RecyclerListView listView; + private Adapter adapter; + private FrameLayout buttonContainer; + private ButtonWithCounterView button; + + @Override + public View createView(Context context) { + TLRPC.Chat chat = getMessagesController().getChat(-dialogId); + if (chat != null) { + currentReplyColor = selectedReplyColor = ChatObject.getColorId(chat); + currentReplyEmoji = selectedReplyEmoji = ChatObject.getEmojiId(chat); + currentProfileColor = selectedProfileColor = ChatObject.getProfileColorId(chat); + currentProfileEmoji = selectedProfileEmoji = ChatObject.getProfileEmojiId(chat); + currentStatusEmoji = selectedStatusEmoji = chat.emoji_status; + } + TLRPC.ChatFull chatFull = getMessagesController().getChatFull(-dialogId); + if (chatFull != null) { + currentWallpaper = selectedWallpaper = chatFull.wallpaper; + if (ChatThemeController.isNotEmoticonWallpaper(currentWallpaper)) { + galleryWallpaper = currentWallpaper; + } + } + + actionBar.setBackButtonImage(R.drawable.ic_ab_back); + actionBar.setTitle(LocaleController.getString(R.string.ChannelColorTitle2)); + actionBar.setActionBarMenuOnItemClick(new ActionBar.ActionBarMenuOnItemClick() { + @Override + public void onItemClick(int id) { + if (id == -1) { + if (currentLevel >= minLevelRequired() && hasUnsavedChanged()) { + showUnsavedAlert(); + return; + } + finishFragment(); + } else if (id == 1) { + toggleTheme(); + } + } + }); + + sunDrawable = new RLottieDrawable(R.raw.sun, "" + R.raw.sun, dp(28), dp(28), true, null); + sunDrawable.setPlayInDirectionOfCustomEndFrame(true); + if (!isDark) { + sunDrawable.setCustomEndFrame(0); + sunDrawable.setCurrentFrame(0); + } else { + sunDrawable.setCurrentFrame(35); + sunDrawable.setCustomEndFrame(36); + } + sunDrawable.beginApplyLayerColors(); + int color = Theme.getColor(Theme.key_chats_menuName, resourceProvider); + sunDrawable.setLayerColor("Sunny.**", color); + sunDrawable.setLayerColor("Path 6.**", color); + sunDrawable.setLayerColor("Path.**", color); + sunDrawable.setLayerColor("Path 5.**", color); + dayNightItem = actionBar.createMenu().addItem(1, sunDrawable); + + FrameLayout contentView = new FrameLayout(context); + + updateRows(); + listView = new RecyclerListView(context, resourceProvider); + listView.setAdapter(adapter = new Adapter()); + listView.setLayoutManager(new LinearLayoutManager(context)); + listView.setBackgroundColor(getThemedColor(Theme.key_windowBackgroundGray)); + contentView.addView(listView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT, Gravity.FILL, 0, 0, 0, 68)); + listView.setOnItemClickListener((view, position) -> { + if (view instanceof EmojiCell) { + long selectedEmojiId = 0; + if (position == replyEmojiRow) { + selectedEmojiId = selectedReplyEmoji; + } else if (position == profileEmojiRow) { + selectedEmojiId = selectedProfileEmoji; + } else if (position == statusEmojiRow) { + selectedEmojiId = DialogObject.getEmojiStatusDocumentId(selectedStatusEmoji); + } + showSelectStatusDialog((EmojiCell) view, selectedEmojiId, position == statusEmojiRow, (documentId, until) -> { + if (position == replyEmojiRow) { + selectedReplyEmoji = documentId; + updateMessagesPreview(true); + } else if (position == profileEmojiRow) { + selectedProfileEmoji = documentId; + updateProfilePreview(true); + } else if (position == statusEmojiRow) { + if (documentId == 0) { + selectedStatusEmoji = null; + } else { + if (until != null) { + selectedStatusEmoji = new TLRPC.TL_emojiStatusUntil(); + ((TLRPC.TL_emojiStatusUntil) selectedStatusEmoji).until = until; + ((TLRPC.TL_emojiStatusUntil) selectedStatusEmoji).document_id = documentId; + } else { + selectedStatusEmoji = new TLRPC.TL_emojiStatus(); + ((TLRPC.TL_emojiStatus) selectedStatusEmoji).document_id = documentId; + } + } + updateProfilePreview(true); + } + updateButton(true); + ((EmojiCell) view).setEmoji(documentId, true); + }); + } else if (position == removeProfileColorRow) { + selectedProfileColor = -1; + selectedProfileEmoji = 0; + updateProfilePreview(true); + updateButton(true); + updateRows(); + } else if (position == wallpaperRow) { + ChannelWallpaperActivity activity = new ChannelWallpaperActivity(dialogId, boostsStatus); + activity.setResourceProvider(resourceProvider); + activity.setSelectedWallpaper(selectedWallpaper, galleryWallpaper); + activity.setOnSelectedWallpaperChange((currentWallpaper, selectedWallpaper, galleryWallpaper) -> { + this.currentWallpaper = currentWallpaper; + this.selectedWallpaper = selectedWallpaper; + this.galleryWallpaper = galleryWallpaper; + + updateButton(false); + updateMessagesPreview(false); + }); + presentFragment(activity); + } + }); + DefaultItemAnimator itemAnimator = new DefaultItemAnimator(); + itemAnimator.setDurations(350); + itemAnimator.setInterpolator(CubicBezierInterpolator.EASE_OUT_QUINT); + itemAnimator.setDelayAnimations(false); + itemAnimator.setSupportsChangeAnimations(false); + listView.setItemAnimator(itemAnimator); + + button = new ButtonWithCounterView(context, resourceProvider); + button.setText(LocaleController.getString(R.string.ApplyChanges), false); + button.setOnClickListener(v -> buttonClick()); + updateButton(false); + + buttonContainer = new FrameLayout(context); + buttonContainer.setBackgroundColor(getThemedColor(Theme.key_windowBackgroundGray)); + buttonContainer.addView(button, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 48, Gravity.BOTTOM, 10, 10, 10, 10)); + contentView.addView(buttonContainer, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 68, Gravity.BOTTOM)); + + return fragmentView = contentView; + } + + @Override + public boolean onBackPressed() { + if (currentLevel >= minLevelRequired() && hasUnsavedChanged()) { + showUnsavedAlert(); + return false; + } + return super.onBackPressed(); + } + + @Override + public boolean isSwipeBackEnabled(MotionEvent event) { + return !hasUnsavedChanged() || currentLevel < minLevelRequired(); + } + + private void buttonClick() { + if (boostsStatus == null || button.isLoading()) { + return; + } + if (currentLevel < minLevelRequired()) { + button.setLoading(true); + showLimit(); + return; + } + + int[] reqCount = new int[] { 0 }; + int[] reqReceivedCount = new int[] { 0 }; + boolean[] receivedError = new boolean[] { false }; + Utilities.Callback whenRequestDone = error -> AndroidUtilities.runOnUIThread(() -> { + if (receivedError[0] || reqReceivedCount[0] >= reqCount[0]) return; + if (error != null) { + receivedError[0] = true; + if ("BOOSTS_REQUIRED".equals(error.text)) { + showLimit(); + } else { + button.setLoading(false); + BulletinFactory.of(this).createSimpleBulletin(R.raw.error, LocaleController.formatString(R.string.UnknownErrorCode, error.text)).show(); + } + return; + } + reqReceivedCount[0]++; + if (reqReceivedCount[0] == reqCount[0]) { + finishFragment(); + showBulletin(); + button.setLoading(false); + } + }); + + TLRPC.Chat channel = getMessagesController().getChat(-dialogId); + if (channel == null) { + FileLog.e("channel is null in ChannelColorAcitivity"); + BulletinFactory.of(this).createSimpleBulletin(R.raw.error, LocaleController.getString(R.string.UnknownError)).show(); + return; + } + + button.setLoading(true); + + if (currentReplyColor != selectedReplyColor || currentReplyEmoji != selectedReplyEmoji) { + TLRPC.TL_channels_updateColor req = new TLRPC.TL_channels_updateColor(); + req.channel = getMessagesController().getInputChannel(-dialogId); + req.for_profile = false; + + if (channel.color == null) { + channel.color = new TLRPC.TL_peerColor(); + channel.flags2 |= 128; + } + req.flags |= 4; + req.color = selectedReplyColor; + channel.color.flags |= 1; + channel.color.color = selectedReplyColor; + + if (selectedReplyEmoji != 0) { + req.flags |= 1; + req.background_emoji_id = selectedReplyEmoji; + channel.color.flags |= 2; + channel.color.background_emoji_id = selectedReplyEmoji; + } else { + channel.color.flags &=~ 2; + channel.color.background_emoji_id = 0; + } + + reqCount[0]++; + getConnectionsManager().sendRequest(req, (res, err) -> { + if (res instanceof TLRPC.Updates) { + getMessagesController().processUpdates((TLRPC.Updates) res, false); + } + if (whenRequestDone != null) { + whenRequestDone.run(err); + } + }); + } + + if (currentProfileColor != selectedProfileColor || currentProfileEmoji != selectedProfileEmoji) { + TLRPC.TL_channels_updateColor req = new TLRPC.TL_channels_updateColor(); + req.channel = getMessagesController().getInputChannel(-dialogId); + req.for_profile = true; + + if (channel.profile_color == null) { + channel.profile_color = new TLRPC.TL_peerColor(); + channel.flags2 |= 256; + } + if (selectedProfileColor >= 0) { + req.flags |= 4; + req.color = selectedProfileColor; + channel.profile_color.flags |= 1; + channel.profile_color.color = selectedProfileColor; + } else { + channel.profile_color.flags &=~ 1; + } + + if (selectedProfileEmoji != 0) { + req.flags |= 1; + req.background_emoji_id = selectedProfileEmoji; + channel.profile_color.flags |= 2; + channel.profile_color.background_emoji_id = selectedProfileEmoji; + } else { + channel.profile_color.flags &=~ 2; + channel.profile_color.background_emoji_id = 0; + } + + reqCount[0]++; + getConnectionsManager().sendRequest(req, (res, err) -> { + if (res instanceof TLRPC.Updates) { + getMessagesController().processUpdates((TLRPC.Updates) res, false); + } + if (whenRequestDone != null) { + whenRequestDone.run(err); + } + }); + } + + if (!ChatThemeController.wallpaperEquals(currentWallpaper, selectedWallpaper)) { + TLRPC.TL_messages_setChatWallPaper req = new TLRPC.TL_messages_setChatWallPaper(); + req.peer = getMessagesController().getInputPeer(dialogId); + if (selectedWallpaper != null) { + if (!TextUtils.isEmpty(ChatThemeController.getWallpaperEmoticon(selectedWallpaper))) { + req.flags |= 1; + req.wallpaper = new TLRPC.TL_inputWallPaperNoFile(); + ((TLRPC.TL_inputWallPaperNoFile) req.wallpaper).id = 0; + + req.flags |= 4; + req.settings = new TLRPC.TL_wallPaperSettings(); + req.settings.flags |= 128; + req.settings.emoticon = ChatThemeController.getWallpaperEmoticon(selectedWallpaper); + } else { + req.flags |= 1; + if (selectedWallpaper instanceof TLRPC.TL_wallPaper) { + TLRPC.TL_inputWallPaper wallPaper = new TLRPC.TL_inputWallPaper(); + wallPaper.id = selectedWallpaper.id; + wallPaper.access_hash = selectedWallpaper.access_hash; + req.wallpaper = wallPaper; + } else if (selectedWallpaper instanceof TLRPC.TL_wallPaperNoFile) { + TLRPC.TL_inputWallPaperNoFile wallPaperNoFile = new TLRPC.TL_inputWallPaperNoFile(); + wallPaperNoFile.id = selectedWallpaper.id; + req.wallpaper = wallPaperNoFile; + } + } + } + + reqCount[0]++; + getConnectionsManager().sendRequest(req, (res, err) -> { + if (res instanceof TLRPC.Updates) { + getMessagesController().processUpdates((TLRPC.Updates) res, false); + } + if (whenRequestDone != null) { + whenRequestDone.run(err); + } + }); + TLRPC.ChatFull chatFull1 = getMessagesController().getChatFull(-dialogId); + ChatThemeController.getInstance(currentAccount).saveChatWallpaper(dialogId, selectedWallpaper); + if (chatFull1 != null) { + if (selectedWallpaper == null) { + chatFull1.flags2 &=~ 128; + chatFull1.wallpaper = null; + } else { + chatFull1.flags2 |= 128; + chatFull1.wallpaper = selectedWallpaper; + } + getMessagesController().putChatFull(chatFull1); + getNotificationCenter().postNotificationName(NotificationCenter.chatInfoDidLoad, chatFull1, 0, false, false); + } + } + + if (!DialogObject.emojiStatusesEqual(currentStatusEmoji, selectedStatusEmoji)) { + TLRPC.TL_channels_updateEmojiStatus req = new TLRPC.TL_channels_updateEmojiStatus(); + req.channel = getMessagesController().getInputChannel(-dialogId); + if (selectedStatusEmoji == null || selectedStatusEmoji instanceof TLRPC.TL_emojiStatusEmpty) { + req.emoji_status = new TLRPC.TL_emojiStatusEmpty(); + channel.emoji_status = new TLRPC.TL_emojiStatusEmpty(); + channel.flags2 &=~ 512; + } else { + req.emoji_status = selectedStatusEmoji; + channel.emoji_status = selectedStatusEmoji; + channel.flags |= 512; + } + + getMessagesController().updateEmojiStatusUntilUpdate(dialogId, selectedStatusEmoji); + + reqCount[0]++; + getConnectionsManager().sendRequest(req, (res, err) -> { + if (res instanceof TLRPC.Updates) { + getMessagesController().processUpdates((TLRPC.Updates) res, false); + } + if (whenRequestDone != null) { + whenRequestDone.run(err); + } + }); + } + + if (reqCount[0] == 0) { + finishFragment(); + button.setLoading(false); + } else { + getMessagesController().putChat(channel, false); + getNotificationCenter().postNotificationName(NotificationCenter.updateInterfaces, MessagesController.UPDATE_MASK_EMOJI_STATUS); + } + } + + private void showLimit() { + getMessagesController().getBoostsController().userCanBoostChannel(dialogId, boostsStatus, canApplyBoost -> { + int type = LimitReachedBottomSheet.TYPE_BOOSTS_FOR_COLOR; + int lvl = 0; + if (currentReplyColor != selectedReplyColor) { + MessagesController.PeerColors peerColors = getMessagesController().peerColors; + MessagesController.PeerColor peerColor = peerColors == null ? null : peerColors.getColor(selectedReplyColor); + if (peerColor != null && peerColor.lvl > currentLevel) { + type = LimitReachedBottomSheet.TYPE_BOOSTS_FOR_COLOR; + lvl = peerColor.lvl; + } + } + if (currentProfileColor != selectedProfileColor) { + MessagesController.PeerColors peerColors = getMessagesController().profilePeerColors; + MessagesController.PeerColor peerColor = peerColors == null ? null : peerColors.getColor(selectedProfileColor); + if (peerColor != null && peerColor.lvl > currentLevel) { + type = LimitReachedBottomSheet.TYPE_BOOSTS_FOR_PROFILE_COLOR; + lvl = peerColor.lvl; + } + } + if (currentReplyEmoji != selectedReplyEmoji && getMessagesController().channelBgIconLevelMin > currentLevel) { + type = LimitReachedBottomSheet.TYPE_BOOSTS_FOR_REPLY_ICON; + } + if (currentProfileEmoji != selectedProfileEmoji && getMessagesController().channelProfileIconLevelMin > currentLevel) { + type = LimitReachedBottomSheet.TYPE_BOOSTS_FOR_PROFILE_ICON; + } + if (!DialogObject.emojiStatusesEqual(currentStatusEmoji, selectedStatusEmoji) && getMessagesController().channelEmojiStatusLevelMin > currentLevel) { + type = LimitReachedBottomSheet.TYPE_BOOSTS_FOR_EMOJI_STATUS; + } + if (!ChatThemeController.wallpaperEquals(currentWallpaper, selectedWallpaper)) { + if (!TextUtils.isEmpty(ChatThemeController.getWallpaperEmoticon(selectedWallpaper))) { + type = LimitReachedBottomSheet.TYPE_BOOSTS_FOR_WALLPAPER; + } else { + type = LimitReachedBottomSheet.TYPE_BOOSTS_FOR_CUSTOM_WALLPAPER; + } + } + final int level = lvl; + LimitReachedBottomSheet limitReachedBottomSheet = new LimitReachedBottomSheet(this, getContext(), type, currentAccount, getResourceProvider()) { + @Override + protected int channelColorLevelMin() { + return level; + } + }; + limitReachedBottomSheet.setCanApplyBoost(canApplyBoost); + limitReachedBottomSheet.setBoostsStats(boostsStatus, true); + limitReachedBottomSheet.setDialogId(dialogId); + TLRPC.Chat channel = getMessagesController().getChat(-dialogId); + if (channel != null) { + limitReachedBottomSheet.showStatisticButtonInLink(() -> { + Bundle args = new Bundle(); + args.putLong("chat_id", -dialogId); + args.putBoolean("is_megagroup", channel.megagroup); + args.putBoolean("start_from_boosts", true); + TLRPC.ChatFull chatInfo = getMessagesController().getChatFull(-dialogId); + if (chatInfo == null || !chatInfo.can_view_stats) { + args.putBoolean("only_boosts", true); + } + StatisticActivity fragment = new StatisticActivity(args); + presentFragment(fragment); + }); + } + showDialog(limitReachedBottomSheet); + button.setLoading(false); + }); + } + + private void showUnsavedAlert() { + if (getVisibleDialog() != null) { + return; + } + AlertDialog alertDialog = new AlertDialog.Builder(getContext(), getResourceProvider()) + .setTitle(LocaleController.getString(R.string.ChannelColorUnsaved)) + .setMessage(LocaleController.getString(R.string.ChannelColorUnsavedMessage)) + .setNegativeButton(LocaleController.getString(R.string.Dismiss), (di, w) -> { + finishFragment(); + }) + .setPositiveButton(LocaleController.getString(R.string.ApplyTheme), (di, w) -> { + buttonClick(); + }) + .create(); + showDialog(alertDialog); + ((TextView) alertDialog.getButton(DialogInterface.BUTTON_NEGATIVE)).setTextColor(getThemedColor(Theme.key_text_RedBold)); + } + + private SelectAnimatedEmojiDialog.SelectAnimatedEmojiDialogWindow selectAnimatedEmojiDialog; + public void showSelectStatusDialog(EmojiCell cell, long documentId, boolean emojiStatus, Utilities.Callback2 onSet) { + if (selectAnimatedEmojiDialog != null || cell == null) { + return; + } + final SelectAnimatedEmojiDialog.SelectAnimatedEmojiDialogWindow[] popup = new SelectAnimatedEmojiDialog.SelectAnimatedEmojiDialogWindow[1]; + int xoff = 0, yoff = 0; + + final boolean down = cell.getTop() + cell.getHeight() > listView.getMeasuredHeight() / 2f; + AnimatedEmojiDrawable.SwapAnimatedEmojiDrawable scrimDrawable = null; + View scrimDrawableParent = null; + final int popupHeight = (int) Math.min(AndroidUtilities.dp(410 - 16 - 64), AndroidUtilities.displaySize.y * .75f); + final int popupWidth = (int) Math.min(dp(340 - 16), AndroidUtilities.displaySize.x * .95f); + if (cell != null) { + scrimDrawable = cell.imageDrawable; + scrimDrawableParent = cell; + if (cell.imageDrawable != null) { + cell.imageDrawable.play(); + cell.updateImageBounds(); + AndroidUtilities.rectTmp2.set(cell.imageDrawable.getBounds()); + if (down) { + yoff = -AndroidUtilities.rectTmp2.centerY() + dp(12) - popupHeight; + } else { + yoff = -(cell.getHeight() - AndroidUtilities.rectTmp2.centerY()) - AndroidUtilities.dp(16); + } + xoff = AndroidUtilities.rectTmp2.centerX() - (AndroidUtilities.displaySize.x - popupWidth); + } + } + int type; + if (emojiStatus) { + type = down ? SelectAnimatedEmojiDialog.TYPE_EMOJI_STATUS_CHANNEL_TOP : SelectAnimatedEmojiDialog.TYPE_EMOJI_STATUS_CHANNEL; + } else { + type = down ? SelectAnimatedEmojiDialog.TYPE_SET_REPLY_ICON : SelectAnimatedEmojiDialog.TYPE_SET_REPLY_ICON_BOTTOM; + } + SelectAnimatedEmojiDialog popupLayout = new SelectAnimatedEmojiDialog(ChannelColorActivity.this, getContext(), true, xoff, type, true, getResourceProvider(), down ? 24 : 16, cell.getColor()) { + @Override + protected void onEmojiSelected(View emojiView, Long documentId, TLRPC.Document document, Integer until) { + if (onSet != null) { + onSet.run(documentId == null ? 0 : documentId, until); + } + if (popup[0] != null) { + selectAnimatedEmojiDialog = null; + popup[0].dismiss(); + } + } + + @Override + protected float getScrimDrawableTranslationY() { + return 0; + } + }; + popupLayout.useAccentForPlus = true; + popupLayout.setSelected(documentId == 0 ? null : documentId); + popupLayout.setSaveState(3); + popupLayout.setScrimDrawable(scrimDrawable, scrimDrawableParent); + popup[0] = selectAnimatedEmojiDialog = new SelectAnimatedEmojiDialog.SelectAnimatedEmojiDialogWindow(popupLayout, LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT) { + @Override + public void dismiss() { + super.dismiss(); + selectAnimatedEmojiDialog = null; + } + }; + popup[0].showAsDropDown(cell, 0, yoff, Gravity.TOP | Gravity.RIGHT); + popup[0].dimBehind(); + } + + private static final int VIEW_TYPE_MESSAGE_PREVIEW = 0; + private static final int VIEW_TYPE_PROFILE_PREVIEW = 1; + private static final int VIEW_TYPE_WALLPAPER_THEMES = 2; + private static final int VIEW_TYPE_COLOR_REPLY_GRID = 3; + private static final int VIEW_TYPE_COLOR_PROFILE_GRID = 4; + private static final int VIEW_TYPE_BUTTON = 5; + private static final int VIEW_TYPE_BUTTON_EMOJI = 6; + private static final int VIEW_TYPE_SHADOW = 7; + + private int rowsCount = 0; + + private int messagesPreviewRow; + private int replyColorListRow; + private int replyEmojiRow; + private int replyHintRow; + + private int wallpaperThemesRow; + private int wallpaperRow; + private int wallpaperHintRow; + + private int profilePreviewRow; + private int profileColorGridRow; + private int profileEmojiRow; + private int profileHintRow; + + private int removeProfileColorRow; + private int removeProfileColorShadowRow; + + private int statusEmojiRow; + private int statusHintRow; + + private void updateRows() { + rowsCount = 0; + messagesPreviewRow = rowsCount++; + replyColorListRow = rowsCount++; + replyEmojiRow = rowsCount++; + replyHintRow = rowsCount++; + wallpaperThemesRow = rowsCount++; + wallpaperRow = rowsCount++; + wallpaperHintRow = rowsCount++; + profilePreviewRow = rowsCount++; + profileColorGridRow = rowsCount++; + profileEmojiRow = rowsCount++; + if (selectedProfileEmoji != 0 || selectedProfileColor >= 0) { + boolean wasButton = removeProfileColorRow >= 0; + removeProfileColorRow = rowsCount++; + if (!wasButton && adapter != null) { + adapter.notifyItemInserted(removeProfileColorRow); + adapter.notifyItemChanged(profileEmojiRow); + } + } else { + int wasIndex = removeProfileColorRow; + removeProfileColorRow = -1; + if (wasIndex >= 0 && adapter != null) { + adapter.notifyItemRemoved(wasIndex); + adapter.notifyItemChanged(profileEmojiRow); + } + } + profileHintRow = rowsCount++; + statusEmojiRow = rowsCount++; + statusHintRow = rowsCount++; + } + + private class Adapter extends RecyclerListView.SelectionAdapter { + + @NonNull + @Override + public RecyclerView.ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) { + View view; + if (viewType == VIEW_TYPE_MESSAGE_PREVIEW) { + ThemePreviewMessagesCell messagesCell = new ThemePreviewMessagesCell(getContext(), parentLayout, ThemePreviewMessagesCell.TYPE_PEER_COLOR, dialogId, resourceProvider); + messagesCell.customAnimation = true; + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { + messagesCell.setImportantForAccessibility(View.IMPORTANT_FOR_ACCESSIBILITY_NO_HIDE_DESCENDANTS); + } + messagesCell.fragment = ChannelColorActivity.this; + messagesCell.setOverrideBackground(backgroundDrawable = PreviewView.getBackgroundDrawable(backgroundDrawable, currentAccount, selectedWallpaper, isDark)); + view = messagesCell; + } else if (viewType == VIEW_TYPE_WALLPAPER_THEMES) { + ThemeChooser themesWallpaper = new ThemeChooser(getContext(), false, currentAccount, resourceProvider); + themesWallpaper.setSelectedEmoticon(ChatThemeController.getWallpaperEmoticon(selectedWallpaper), false); + themesWallpaper.setGalleryWallpaper(galleryWallpaper); + themesWallpaper.setOnEmoticonSelected(emoticon -> { + if (emoticon == null) { + selectedWallpaper = galleryWallpaper; + } else { + selectedWallpaper = new TLRPC.TL_wallPaperNoFile(); + selectedWallpaper.id = 0; + selectedWallpaper.flags |= 4; + selectedWallpaper.settings = new TLRPC.TL_wallPaperSettings(); + selectedWallpaper.settings.emoticon = emoticon; + } + updateButton(true); + updateMessagesPreview(true); + }); + themesWallpaper.setBackgroundColor(getThemedColor(Theme.key_windowBackgroundWhite)); + view = themesWallpaper; + } else if (viewType == VIEW_TYPE_BUTTON) { + TextCell textCell = new TextCell(getContext(), getResourceProvider()); + textCell.setBackgroundColor(getThemedColor(Theme.key_windowBackgroundWhite)); + view = textCell; + } else if (viewType == VIEW_TYPE_BUTTON_EMOJI) { + EmojiCell emojiCell = new EmojiCell(getContext(), resourceProvider); + emojiCell.setBackgroundColor(getThemedColor(Theme.key_windowBackgroundWhite)); + view = emojiCell; + } else if (viewType == VIEW_TYPE_COLOR_REPLY_GRID) { + PeerColorPicker listCell = new PeerColorPicker(getContext(), currentAccount, resourceProvider); + listCell.listView.setOnItemClickListener((view2, position) -> { + selectedReplyColor = listCell.toColorId(position); + updateButton(true); + updateMessagesPreview(true); + updateProfilePreview(true); + + if (view2.getLeft() < listCell.listView.getPaddingLeft() + dp(24)) { + listCell.listView.smoothScrollBy((int) -(listCell.listView.getPaddingLeft() + dp(48) - view2.getLeft()), 0); + } else if (view2.getLeft() + view2.getWidth() > listCell.listView.getMeasuredWidth() - listCell.listView.getPaddingRight() - dp(24)) { + listCell.listView.smoothScrollBy((int) (view2.getLeft() + view2.getWidth() - (listCell.listView.getMeasuredWidth() - listCell.listView.getPaddingRight() - dp(48))), 0); + } + }); + listCell.setBackgroundColor(getThemedColor(Theme.key_windowBackgroundWhite)); + view = listCell; +// PeerColorActivity.PeerColorGrid gridCell = new PeerColorActivity.PeerColorGrid(getContext(), PeerColorActivity.PAGE_NAME, currentAccount, resourceProvider); +// gridCell.setOnColorClick(color -> { +// selectedReplyColor = color; +// updateButton(true); +// updateMessagesPreview(true); +// updateProfilePreview(true); +// }); +// gridCell.setBackgroundColor(getThemedColor(Theme.key_windowBackgroundWhite)); +// view = gridCell; + } else if (viewType == VIEW_TYPE_COLOR_PROFILE_GRID) { + PeerColorActivity.PeerColorGrid gridCell = new PeerColorActivity.PeerColorGrid(getContext(), PeerColorActivity.PAGE_PROFILE, currentAccount, resourceProvider); + gridCell.setDivider(false); + gridCell.setOnColorClick(color -> { + selectedProfileColor = color; + updateButton(true); + updateProfilePreview(true); + }); + gridCell.setBackgroundColor(getThemedColor(Theme.key_windowBackgroundWhite)); + view = gridCell; + } else if (viewType == VIEW_TYPE_PROFILE_PREVIEW) { + view = new ProfilePreview(getContext()); + } else { + view = new TextInfoPrivacyCell(getContext()); + } + return new RecyclerListView.Holder(view); + } + + @Override + public void onBindViewHolder(@NonNull RecyclerView.ViewHolder holder, int position) { + switch (holder.getItemViewType()) { + case VIEW_TYPE_BUTTON: + TextCell textCell = (TextCell) holder.itemView; + if (position == removeProfileColorRow) { + textCell.setText(LocaleController.getString(R.string.ChannelProfileColorReset), false); + } else { + textCell.setText(LocaleController.getString(R.string.ChannelWallpaper), false); + if (currentLevel < getMessagesController().channelWallpaperLevelMin) { + textCell.setLockLevel(false, getMessagesController().channelWallpaperLevelMin); + } + } + break; + case VIEW_TYPE_BUTTON_EMOJI: + EmojiCell emojiCell = (EmojiCell) holder.itemView; + emojiCell.setDivider(false); + if (position == replyEmojiRow) { + emojiCell.setAdaptiveEmojiColor(currentAccount, selectedReplyColor, true); + emojiCell.setText(LocaleController.getString(R.string.ChannelReplyLogo)); + if (currentLevel < getMessagesController().channelBgIconLevelMin) { + emojiCell.setLockLevel(getMessagesController().channelBgIconLevelMin); + } + emojiCell.setEmoji(selectedReplyEmoji, false); + } else if (position == profileEmojiRow) { + emojiCell.setAdaptiveEmojiColor(currentAccount, selectedProfileColor, false); + emojiCell.setText(LocaleController.getString(R.string.ChannelProfileLogo)); + emojiCell.setDivider(removeProfileColorRow >= 0); + if (currentLevel < getMessagesController().channelProfileIconLevelMin) { + emojiCell.setLockLevel(getMessagesController().channelProfileIconLevelMin); + } + emojiCell.setEmoji(selectedProfileEmoji, false); + } else if (position == statusEmojiRow) { + emojiCell.setAdaptiveEmojiColor(currentAccount, selectedProfileColor, false); + emojiCell.setText(LocaleController.getString(R.string.ChannelEmojiStatus)); + if (currentLevel < getMessagesController().channelEmojiStatusLevelMin) { + emojiCell.setLockLevel(getMessagesController().channelEmojiStatusLevelMin); + } + emojiCell.setEmoji(DialogObject.getEmojiStatusDocumentId(selectedStatusEmoji), false); + } + break; + case VIEW_TYPE_SHADOW: + TextInfoPrivacyCell infoCell = (TextInfoPrivacyCell) holder.itemView; + infoCell.setFixedSize(0); + if (position == replyHintRow) { + infoCell.setText(LocaleController.getString(R.string.ChannelReplyInfo)); + } else if (position == wallpaperHintRow) { + infoCell.setText(LocaleController.getString(R.string.ChannelWallpaper2Info)); + } else if (position == profileHintRow) { + infoCell.setText(LocaleController.getString(R.string.ChannelProfileInfo)); + } else if (position == statusHintRow) { + infoCell.setText(LocaleController.getString(R.string.ChannelEmojiStatusInfo)); + } else if (position == removeProfileColorShadowRow) { + infoCell.setText(""); + infoCell.setFixedSize(12); + } + infoCell.setBackground(Theme.getThemedDrawableByKey(getContext(), position == statusHintRow ? R.drawable.greydivider_bottom : R.drawable.greydivider, Theme.key_windowBackgroundGrayShadow, resourceProvider)); + break; + case VIEW_TYPE_PROFILE_PREVIEW: + ProfilePreview profilePreview = (ProfilePreview) holder.itemView; + profilePreview.backgroundView.setColor(currentAccount, selectedProfileColor, false); + profilePreview.profileView.setColor(selectedProfileColor, false); + profilePreview.profileView.setEmoji(selectedProfileEmoji, false); + profilePreview.profileView.setStatusEmoji(DialogObject.getEmojiStatusDocumentId(selectedStatusEmoji), false); + profilePreview.profileView.overrideAvatarColor(selectedReplyColor); + break; + case VIEW_TYPE_COLOR_PROFILE_GRID: + ((PeerColorActivity.PeerColorGrid) holder.itemView).setSelected(selectedProfileColor, false); + break; + case VIEW_TYPE_COLOR_REPLY_GRID: +// ((PeerColorActivity.PeerColorGrid) holder.itemView).setSelected(selectedReplyColor, false); + ((PeerColorPicker) holder.itemView).setSelected(selectedReplyColor, false); + break; + } + } + + @Override + public void onViewAttachedToWindow(@NonNull RecyclerView.ViewHolder holder) { + if (holder.itemView instanceof ProfilePreview) { + ProfilePreview profilePreview = (ProfilePreview) holder.itemView; + profilePreview.profileView.setColor(selectedProfileColor, false); + profilePreview.profileView.setEmoji(selectedProfileEmoji, false); + profilePreview.profileView.setStatusEmoji(DialogObject.getEmojiStatusDocumentId(selectedStatusEmoji), false); + profilePreview.profileView.overrideAvatarColor(selectedReplyColor); + } else if (holder.itemView instanceof ThemePreviewMessagesCell) { + ThemePreviewMessagesCell messagesCell = (ThemePreviewMessagesCell) holder.itemView; + messagesCell.setOverrideBackground(backgroundDrawable); + } else { + updateColors(holder.itemView); + } + super.onViewAttachedToWindow(holder); + } + + @Override + public int getItemViewType(int position) { + if (position == messagesPreviewRow) { + return VIEW_TYPE_MESSAGE_PREVIEW; + } else if (position == wallpaperThemesRow) { + return VIEW_TYPE_WALLPAPER_THEMES; + } else if (position == profilePreviewRow) { + return VIEW_TYPE_PROFILE_PREVIEW; + } else if (position == replyColorListRow) { + return VIEW_TYPE_COLOR_REPLY_GRID; + } else if (position == profileColorGridRow) { + return VIEW_TYPE_COLOR_PROFILE_GRID; + } else if (position == replyEmojiRow || position == profileEmojiRow || position == statusEmojiRow) { + return VIEW_TYPE_BUTTON_EMOJI; + } else if (position == wallpaperRow || position == removeProfileColorRow) { + return VIEW_TYPE_BUTTON; + } else { + return VIEW_TYPE_SHADOW; + } + } + + @Override + public int getItemCount() { + return rowsCount; + } + + @Override + public boolean isEnabled(RecyclerView.ViewHolder holder) { + final int viewType = holder.getItemViewType(); + return viewType == VIEW_TYPE_BUTTON || viewType == VIEW_TYPE_BUTTON_EMOJI; + } + + } + + public void updateMessagesPreview(boolean animated) { + View messagesPreview = findChildAt(messagesPreviewRow); + View colorPicker = findChildAt(replyColorListRow); + View emojiPicker = findChildAt(replyEmojiRow); + View wallpaperPicker = findChildAt(wallpaperThemesRow); + + if (messagesPreview instanceof ThemePreviewMessagesCell) { + ThemePreviewMessagesCell messagesCellPreview = (ThemePreviewMessagesCell) messagesPreview; + ChatMessageCell[] cells = messagesCellPreview.getCells(); + for (int i = 0; i < cells.length; ++i) { + if (cells[i] != null) { + MessageObject msg = cells[i].getMessageObject(); + if (msg != null) { + msg.overrideLinkColor = selectedReplyColor; + msg.overrideLinkEmoji = selectedReplyEmoji; + cells[i].setAvatar(msg); + cells[i].invalidate(); + } + } + } + messagesCellPreview.setOverrideBackground(backgroundDrawable = PreviewView.getBackgroundDrawable(backgroundDrawable, currentAccount, selectedWallpaper, isDark)); + } + if (colorPicker instanceof PeerColorActivity.PeerColorGrid) { + ((PeerColorActivity.PeerColorGrid) colorPicker).setSelected(selectedReplyColor, animated); + } else if (colorPicker instanceof PeerColorPicker) { + ((PeerColorPicker) colorPicker).setSelected(selectedReplyColor, animated); + } + if (emojiPicker instanceof EmojiCell) { + ((EmojiCell) emojiPicker).setAdaptiveEmojiColor(currentAccount, selectedReplyColor, true); + ((EmojiCell) emojiPicker).setEmoji(selectedReplyEmoji, animated); + } + if (wallpaperPicker instanceof ThemeChooser) { + ((ThemeChooser) wallpaperPicker).setSelectedEmoticon(ChatThemeController.getWallpaperEmoticon(selectedWallpaper), animated); + ((ThemeChooser) wallpaperPicker).setGalleryWallpaper(galleryWallpaper); + } + } + + public void updateProfilePreview(boolean animated) { + View profilePreview = findChildAt(profilePreviewRow); + View colorPicker = findChildAt(profileColorGridRow); + View emojiPicker = findChildAt(profileEmojiRow); + View emojiStatusPicker = findChildAt(statusEmojiRow); + + if (profilePreview instanceof ProfilePreview) { + ((ProfilePreview) profilePreview).setColor(selectedProfileColor, animated); + ((ProfilePreview) profilePreview).setEmoji(selectedProfileEmoji, animated); + ((ProfilePreview) profilePreview).setEmojiStatus(selectedStatusEmoji, animated); + ((ProfilePreview) profilePreview).profileView.overrideAvatarColor(selectedReplyColor); + } + if (colorPicker instanceof PeerColorActivity.PeerColorGrid) { + ((PeerColorActivity.PeerColorGrid) colorPicker).setSelected(selectedProfileColor, animated); + } else if (colorPicker instanceof PeerColorPicker) { + ((PeerColorPicker) colorPicker).setSelected(selectedReplyColor, animated); + } + if (emojiPicker instanceof EmojiCell) { + ((EmojiCell) emojiPicker).setAdaptiveEmojiColor(currentAccount, selectedProfileColor, false); + ((EmojiCell) emojiPicker).setEmoji(selectedProfileEmoji, animated); + } + if (emojiStatusPicker instanceof EmojiCell) { + ((EmojiCell) emojiStatusPicker).setAdaptiveEmojiColor(currentAccount, selectedProfileColor, false); + ((EmojiCell) emojiStatusPicker).setEmoji(DialogObject.getEmojiStatusDocumentId(selectedStatusEmoji), animated); + } + + updateRows(); + } + + public View findChildAt(int position) { + for (int i = 0; i < listView.getChildCount(); ++i) { + View child = listView.getChildAt(i); + if (listView.getChildAdapterPosition(child) == position) { + return child; + } + } + return null; + } + + private class ProfilePreview extends FrameLayout { + public final PeerColorActivity.ColoredActionBar backgroundView; + public final PeerColorActivity.ProfilePreview profileView; + public ProfilePreview(Context context) { + super(context); + backgroundView = new PeerColorActivity.ColoredActionBar(getContext(), resourceProvider); + backgroundView.setProgressToGradient(1f); + backgroundView.ignoreMeasure = true; + addView(backgroundView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 134, Gravity.FILL)); + profileView = new PeerColorActivity.ProfilePreview(getContext(), currentAccount, dialogId, resourceProvider); + addView(profileView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 104, Gravity.BOTTOM)); + } + + public void setColor(int colorId, boolean animated) { + profileView.setColor(colorId, animated); + backgroundView.setColor(currentAccount, colorId, animated); + } + public void setEmoji(long emojiId, boolean animated) { + profileView.setEmoji(emojiId, animated); + } + public void setEmojiStatus(TLRPC.EmojiStatus emojiStatus, boolean animated) { + profileView.setStatusEmoji(DialogObject.getEmojiStatusDocumentId(emojiStatus), animated); + } + } + + private static class EmojiCell extends FrameLayout { + + private SimpleTextView textView; + private Text offText; + private AnimatedEmojiDrawable.SwapAnimatedEmojiDrawable imageDrawable; + private Theme.ResourcesProvider resourcesProvider; + + public EmojiCell(Context context, Theme.ResourcesProvider resourcesProvider) { + super(context); + + this.resourcesProvider = resourcesProvider; + + setBackgroundColor(Theme.getColor(Theme.key_windowBackgroundWhite, resourcesProvider)); + + textView = new SimpleTextView(context); + textView.setTextSize(16); + textView.setTextColor(Theme.getColor(Theme.key_windowBackgroundWhiteBlackText, resourcesProvider)); + addView(textView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_VERTICAL | Gravity.FILL_HORIZONTAL, 23, 0, 48, 0)); + + imageDrawable = new AnimatedEmojiDrawable.SwapAnimatedEmojiDrawable(this, false, dp(24), AnimatedEmojiDrawable.CACHE_TYPE_ALERT_PREVIEW_STATIC); + } + + private boolean needDivider = false; + public void setDivider(boolean divider) { + setWillNotDraw(!(this.needDivider = divider)); + } + + public void setLockLevel(int lvl) { + if (lvl <= 0) { + textView.setRightDrawable(null); + } else { + textView.setRightDrawable(new PeerColorActivity.LevelLock(getContext(), lvl, resourcesProvider)); + textView.setDrawablePadding(dp(6)); + } + } + + private int color; + public void setAdaptiveEmojiColor(int currentAccount, int colorId, boolean isReply) { + if (colorId < 0) { + if (AndroidUtilities.computePerceivedBrightness(Theme.getColor(Theme.key_actionBarDefault, resourcesProvider)) > .8f) { + color = Theme.getColor(Theme.key_windowBackgroundWhiteBlueText, resourcesProvider); + } else if (AndroidUtilities.computePerceivedBrightness(Theme.getColor(Theme.key_actionBarDefault, resourcesProvider)) < .2f) { + color = Theme.multAlpha(Theme.getColor(Theme.key_actionBarDefaultTitle, resourcesProvider), .5f); + } else { + color = Theme.blendOver(Theme.getColor(Theme.key_windowBackgroundWhite, resourcesProvider), Theme.multAlpha(PeerColorActivity.adaptProfileEmojiColor(Theme.getColor(Theme.key_actionBarDefault, resourcesProvider)), .7f)); + } + } else if (colorId < 7) { + color = Theme.getColor(Theme.keys_avatar_nameInMessage[colorId], resourcesProvider); + } else { + MessagesController.PeerColors peerColors = isReply ? MessagesController.getInstance(currentAccount).peerColors : MessagesController.getInstance(currentAccount).profilePeerColors; + MessagesController.PeerColor peerColor = peerColors == null ? null : peerColors.getColor(colorId); + if (peerColor != null) { + color = peerColor.getColor(0, resourcesProvider); + } else { + color = Theme.getColor(Theme.keys_avatar_nameInMessage[0], resourcesProvider); + } + } + invalidate(); + } + + public void setText(CharSequence text) { + textView.setText(text); + } + + public void setEmoji(long documentId, boolean animated) { + if (documentId == 0) { + imageDrawable.set((Drawable) null, animated); + if (offText == null) { + offText = new Text(LocaleController.getString(R.string.ChannelReplyIconOff), 16); + } + } else { + imageDrawable.set(documentId, animated); + offText = null; + } + } + + public void updateColors() { + textView.setTextColor(Theme.getColor(Theme.key_windowBackgroundWhiteBlackText, resourcesProvider)); + } + + public void updateImageBounds() { + imageDrawable.setBounds( + getWidth() - imageDrawable.getIntrinsicWidth() - dp(21), + (getHeight() - imageDrawable.getIntrinsicHeight()) / 2, + getWidth() - dp(21), + (getHeight() + imageDrawable.getIntrinsicHeight()) / 2 + ); + } + + @Override + protected void dispatchDraw(Canvas canvas) { + super.dispatchDraw(canvas); + updateImageBounds(); + imageDrawable.setColor(color); + if (offText != null) { + offText.draw(canvas, getMeasuredWidth() - offText.getWidth() - dp(19), getMeasuredHeight() / 2f, Theme.getColor(Theme.key_windowBackgroundWhiteBlueText4, resourcesProvider), 1f); + } else { + imageDrawable.draw(canvas); + } + + if (needDivider) { + Paint dividerPaint = resourcesProvider != null ? resourcesProvider.getPaint(Theme.key_paint_divider) : Theme.dividerPaint; + if (dividerPaint != null) { + canvas.drawLine(LocaleController.isRTL ? 0 : AndroidUtilities.dp(23), getMeasuredHeight() - 1, getMeasuredWidth() - (LocaleController.isRTL ? AndroidUtilities.dp(23) : 0), getMeasuredHeight() - 1, dividerPaint); + } + } + } + + public int getColor() { + return color; + } + + @Override + protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { + super.onMeasure( + MeasureSpec.makeMeasureSpec(MeasureSpec.getSize(widthMeasureSpec), MeasureSpec.EXACTLY), + MeasureSpec.makeMeasureSpec(dp(50), MeasureSpec.EXACTLY) + ); + } + + @Override + protected void onDetachedFromWindow() { + super.onDetachedFromWindow(); + imageDrawable.detach(); + } + + @Override + protected void onAttachedToWindow() { + super.onAttachedToWindow(); + imageDrawable.attach(); + } + } + + public static class ThemeChooser extends FrameLayout { + + private final int currentAccount; + private final Theme.ResourcesProvider resourcesProvider; + public final List items = new ArrayList<>(); + private final RecyclerListView listView; + private FlickerLoadingView progressView; + + private final RecyclerListView.SelectionAdapter adapter; + + private boolean dataLoaded; + + private Utilities.Callback onEmoticonSelected; + private String currentEmoticon; + + public void setOnEmoticonSelected(Utilities.Callback callback) { + onEmoticonSelected = callback; + } + + public void setSelectedEmoticon(String emoticon, boolean animated) { + currentEmoticon = emoticon; + + int selectedPosition = -1; + for (int i = 0; i < items.size(); ++i) { + ChatThemeBottomSheet.ChatThemeItem item = items.get(i); + item.isSelected = TextUtils.equals(currentEmoticon, item.getEmoticon()) || TextUtils.isEmpty(emoticon) && item.chatTheme.showAsDefaultStub; + if (item.isSelected) { + selectedPosition = i; + } + } + if (selectedPosition >= 0 && !animated && listView.getLayoutManager() instanceof LinearLayoutManager) { + ((LinearLayoutManager) listView.getLayoutManager()).scrollToPositionWithOffset(selectedPosition, (AndroidUtilities.displaySize.x - dp(83)) / 2); + } + updateSelected(); + } + + private TLRPC.WallPaper fallbackWallpaper; + public void setGalleryWallpaper(TLRPC.WallPaper wallPaper) { + this.fallbackWallpaper = wallPaper; + AndroidUtilities.forEachViews(listView, child -> { + if (child instanceof ThemeSmallPreviewView) { + ((ThemeSmallPreviewView) child).setFallbackWallpaper(fallbackWallpaper); + } + }); + } + + private void updateSelected() { + for (int i = 0; i < listView.getChildCount(); ++i) { + View child = listView.getChildAt(i); + if (child instanceof ThemeSmallPreviewView) { + int position = listView.getChildAdapterPosition(child); + if (position >= 0 && position < items.size()) { + ChatThemeBottomSheet.ChatThemeItem item = items.get(position); + ((ThemeSmallPreviewView) child).setSelected(item.isSelected, true); + } + } + } + } + + public boolean isDark() { + return resourcesProvider != null ? resourcesProvider.isDark() : Theme.isCurrentThemeDark(); + } + + public ThemeChooser(Context context, boolean grid, int currentAccount, Theme.ResourcesProvider resourcesProvider) { + super(context); + this.currentAccount = currentAccount; + this.resourcesProvider = resourcesProvider; + + if (!grid) { + progressView = new FlickerLoadingView(getContext(), resourcesProvider); + progressView.setViewType(FlickerLoadingView.CHAT_THEMES_TYPE); + progressView.setVisibility(View.VISIBLE); + addView(progressView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 104, Gravity.START, 16, 13, 16, 6)); + } + + listView = new RecyclerListView(context, resourcesProvider) { + @Override + public Integer getSelectorColor(int position) { + return 0; + } + }; + listView.setClipToPadding(false); + listView.setPadding(dp(16), dp(13), dp(16), dp(grid ? 13 : 6)); + if (grid) { + listView.setHasFixedSize(false); + GridLayoutManager gridLayoutManager = new GridLayoutManager(getContext(), 3); + gridLayoutManager.setSpanSizeLookup(new GridLayoutManager.SpanSizeLookup() { + @Override + public int getSpanSize(int position) { + return 1; + } + }); + listView.setLayoutManager(gridLayoutManager); + } else { + LinearLayoutManager layoutManager = new LinearLayoutManager(getContext()); + layoutManager.setOrientation(LinearLayoutManager.HORIZONTAL); + listView.setLayoutManager(layoutManager); + listView.setAlpha(0f); + } + listView.setAdapter(adapter = new RecyclerListView.SelectionAdapter() { + @Override + public boolean isEnabled(RecyclerView.ViewHolder holder) { + return true; + } + + @NonNull + @Override + public RecyclerView.ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) { + return new RecyclerListView.Holder(new ThemeSmallPreviewView(parent.getContext(), currentAccount, resourcesProvider, grid ? ThemeSmallPreviewView.TYPE_GRID_CHANNEL : ThemeSmallPreviewView.TYPE_CHANNEL) { + @Override + protected String noThemeString() { + return LocaleController.getString(R.string.ChannelNoWallpaper); + } + + @Override + protected int noThemeStringTextSize() { + if (!grid) { + return 13; + } + return super.noThemeStringTextSize(); + } + }); + } + + @Override + public void onBindViewHolder(@NonNull RecyclerView.ViewHolder holder, int position) { + ThemeSmallPreviewView view = (ThemeSmallPreviewView) holder.itemView; + Theme.ThemeInfo themeInfo = items.get(position).chatTheme.getThemeInfo(items.get(position).themeIndex); + if (themeInfo != null && themeInfo.pathToFile != null && !themeInfo.previewParsed) { + File file = new File(themeInfo.pathToFile); + boolean fileExists = file.exists(); + if (fileExists) { + parseTheme(themeInfo); + } + } + ChatThemeBottomSheet.ChatThemeItem newItem = items.get(position); + view.setEnabled(true); + view.setBackgroundColor(Theme.getColor(Theme.key_dialogBackgroundGray)); + view.setItem(newItem, false); + view.setSelected(newItem.isSelected, false); + view.setFallbackWallpaper(fallbackWallpaper); + } + + @Override + public void onViewAttachedToWindow(@NonNull RecyclerView.ViewHolder holder) { + final int position = holder.getAdapterPosition(); + if (position < 0 || position >= items.size()) { + return; + } + ChatThemeBottomSheet.ChatThemeItem newItem = items.get(position); + ((ThemeSmallPreviewView) holder.itemView).setSelected(newItem.isSelected, false); + ((ThemeSmallPreviewView) holder.itemView).setFallbackWallpaper(fallbackWallpaper); + } + + @Override + public int getItemCount() { + return items.size(); + } + }); + addView(listView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, grid ? LayoutHelper.MATCH_PARENT : 13 + 111 + 6)); + listView.setOnItemClickListener((view, position) -> { + if (position < 0 || position >= items.size()) { + return; + } + ChatThemeBottomSheet.ChatThemeItem thisItem = items.get(position); + if (!grid) { + setSelectedEmoticon(thisItem.getEmoticon(), true); + if (view.getLeft() < listView.getPaddingLeft() + dp(24)) { + listView.smoothScrollBy((int) -(listView.getPaddingLeft() + dp(48) - view.getLeft()), 0); + } else if (view.getLeft() + view.getWidth() > listView.getMeasuredWidth() - listView.getPaddingRight() - dp(24)) { + listView.smoothScrollBy((int) (view.getLeft() + view.getWidth() - (listView.getMeasuredWidth() - listView.getPaddingRight() - dp(48))), 0); + } + } + if (onEmoticonSelected != null) { + onEmoticonSelected.run(thisItem.getEmoticon()); + } + }); + + ChatThemeController chatThemeController = ChatThemeController.getInstance(currentAccount); + chatThemeController.preloadAllWallpaperThumbs(true); + chatThemeController.preloadAllWallpaperThumbs(false); + chatThemeController.preloadAllWallpaperImages(true); + chatThemeController.preloadAllWallpaperImages(false); + chatThemeController.requestAllChatThemes(new ResultCallback>() { + @Override + public void onComplete(List result) { +// if (result != null && !result.isEmpty()) { +// themeDelegate.setCachedThemes(result); +// } + NotificationCenter.getInstance(currentAccount).doOnIdle(() -> { + onDataLoaded(result); + }); + } + + @Override + public void onError(TLRPC.TL_error error) { + Toast.makeText(getContext(), error.text, Toast.LENGTH_SHORT).show(); + } + }, true); + + updateState(false); + } + + public void updateColors() { + final boolean isDark = isDark(); + for (int i = 0; i < items.size(); ++i) { + ChatThemeBottomSheet.ChatThemeItem item = items.get(i); + item.themeIndex = isDark ? 1 : 0; + } + AndroidUtilities.forEachViews(listView, view -> { + ((ThemeSmallPreviewView) view).setBackgroundColor(Theme.getColor(Theme.key_dialogBackgroundGray, resourcesProvider)); + }); + adapter.notifyDataSetChanged(); + } + + private void onDataLoaded(List result) { + if (result == null || result.isEmpty()) { + return; + } + + dataLoaded = true; + items.clear(); + + ChatThemeBottomSheet.ChatThemeItem noThemeItem = new ChatThemeBottomSheet.ChatThemeItem(result.get(0)); + items.add(0, noThemeItem); + + final boolean isDark = resourcesProvider != null ? resourcesProvider.isDark() : Theme.isCurrentThemeDark(); + for (int i = 1; i < result.size(); ++i) { + EmojiThemes chatTheme = result.get(i); + ChatThemeBottomSheet.ChatThemeItem item = new ChatThemeBottomSheet.ChatThemeItem(chatTheme); + + chatTheme.loadPreviewColors(currentAccount); + + item.themeIndex = isDark ? 1 : 0; + items.add(item); + } + + int selectedPosition = -1; + for (int i = 0; i < items.size(); ++i) { + ChatThemeBottomSheet.ChatThemeItem item = items.get(i); + item.isSelected = TextUtils.equals(currentEmoticon, item.getEmoticon()) || TextUtils.isEmpty(currentEmoticon) && item.chatTheme.showAsDefaultStub; + if (item.isSelected) { + selectedPosition = i; + } + } + if (adapter != null) { + adapter.notifyDataSetChanged(); + } + +// resetToPrimaryState(false); + listView.animate().alpha(1f).setDuration(150).start(); + updateState(true); + + if (selectedPosition >= 0 && listView.getLayoutManager() instanceof LinearLayoutManager) { + ((LinearLayoutManager) listView.getLayoutManager()).scrollToPositionWithOffset(selectedPosition, (AndroidUtilities.displaySize.x - dp(83)) / 2); + } + } + + private final HashMap loadingThemes = new HashMap<>(); + private final HashMap loadingWallpapers = new HashMap<>(); + private boolean parseTheme(Theme.ThemeInfo themeInfo) { + if (themeInfo == null || themeInfo.pathToFile == null) { + return false; + } + boolean finished = false; + File file = new File(themeInfo.pathToFile); + try (FileInputStream stream = new FileInputStream(file)) { + int currentPosition = 0; + int idx; + int read; + int linesRead = 0; + while ((read = stream.read(ThemesHorizontalListCell.bytes)) != -1) { + int previousPosition = currentPosition; + int start = 0; + for (int a = 0; a < read; a++) { + if (ThemesHorizontalListCell.bytes[a] == '\n') { + linesRead++; + int len = a - start + 1; + String line = new String(ThemesHorizontalListCell.bytes, start, len - 1, "UTF-8"); + if (line.startsWith("WLS=")) { + String wallpaperLink = line.substring(4); + Uri uri = Uri.parse(wallpaperLink); + themeInfo.slug = uri.getQueryParameter("slug"); + themeInfo.pathToWallpaper = new File(ApplicationLoader.getFilesDirFixed(), Utilities.MD5(wallpaperLink) + ".wp").getAbsolutePath(); + + String mode = uri.getQueryParameter("mode"); + if (mode != null) { + mode = mode.toLowerCase(); + String[] modes = mode.split(" "); + if (modes != null && modes.length > 0) { + for (int b = 0; b < modes.length; b++) { + if ("blur".equals(modes[b])) { + themeInfo.isBlured = true; + break; + } + } + } + } + String pattern = uri.getQueryParameter("pattern"); + if (!TextUtils.isEmpty(pattern)) { + try { + String bgColor = uri.getQueryParameter("bg_color"); + if (!TextUtils.isEmpty(bgColor)) { + themeInfo.patternBgColor = Integer.parseInt(bgColor.substring(0, 6), 16) | 0xff000000; + if (bgColor.length() >= 13 && AndroidUtilities.isValidWallChar(bgColor.charAt(6))) { + themeInfo.patternBgGradientColor1 = Integer.parseInt(bgColor.substring(7, 13), 16) | 0xff000000; + } + if (bgColor.length() >= 20 && AndroidUtilities.isValidWallChar(bgColor.charAt(13))) { + themeInfo.patternBgGradientColor2 = Integer.parseInt(bgColor.substring(14, 20), 16) | 0xff000000; + } + if (bgColor.length() == 27 && AndroidUtilities.isValidWallChar(bgColor.charAt(20))) { + themeInfo.patternBgGradientColor3 = Integer.parseInt(bgColor.substring(21), 16) | 0xff000000; + } + } + } catch (Exception ignore) { + + } + try { + String rotation = uri.getQueryParameter("rotation"); + if (!TextUtils.isEmpty(rotation)) { + themeInfo.patternBgGradientRotation = Utilities.parseInt(rotation); + } + } catch (Exception ignore) { + + } + String intensity = uri.getQueryParameter("intensity"); + if (!TextUtils.isEmpty(intensity)) { + themeInfo.patternIntensity = Utilities.parseInt(intensity); + } + if (themeInfo.patternIntensity == 0) { + themeInfo.patternIntensity = 50; + } + } + } else if (line.startsWith("WPS")) { + themeInfo.previewWallpaperOffset = currentPosition + len; + finished = true; + break; + } else { + if ((idx = line.indexOf('=')) != -1) { + int key = ThemeColors.stringKeyToInt(line.substring(0, idx)); + if (key == Theme.key_chat_inBubble || key == Theme.key_chat_outBubble || key == Theme.key_chat_wallpaper || key == Theme.key_chat_wallpaper_gradient_to1 || key == Theme.key_chat_wallpaper_gradient_to2 || key == Theme.key_chat_wallpaper_gradient_to3) { + String param = line.substring(idx + 1); + int value; + if (param.length() > 0 && param.charAt(0) == '#') { + try { + value = Color.parseColor(param); + } catch (Exception ignore) { + value = Utilities.parseInt(param); + } + } else { + value = Utilities.parseInt(param); + } + if (key == Theme.key_chat_inBubble) { + themeInfo.setPreviewInColor(value); + } else if (key == Theme.key_chat_outBubble) { + themeInfo.setPreviewOutColor(value); + } else if (key == Theme.key_chat_wallpaper) { + themeInfo.setPreviewBackgroundColor(value); + } else if (key == Theme.key_chat_wallpaper_gradient_to1) { + themeInfo.previewBackgroundGradientColor1 = value; + } else if (key == Theme.key_chat_wallpaper_gradient_to2) { + themeInfo.previewBackgroundGradientColor2 = value; + } else if (key == Theme.key_chat_wallpaper_gradient_to3) { + themeInfo.previewBackgroundGradientColor3 = value; + } + } + } + } + start += len; + currentPosition += len; + } + } + if (finished || previousPosition == currentPosition) { + break; + } + stream.getChannel().position(currentPosition); + } + } catch (Throwable e) { + FileLog.e(e); + } + + if (themeInfo.pathToWallpaper != null && !themeInfo.badWallpaper) { + file = new File(themeInfo.pathToWallpaper); + if (!file.exists()) { + if (!loadingWallpapers.containsKey(themeInfo)) { + loadingWallpapers.put(themeInfo, themeInfo.slug); + TLRPC.TL_account_getWallPaper req = new TLRPC.TL_account_getWallPaper(); + TLRPC.TL_inputWallPaperSlug inputWallPaperSlug = new TLRPC.TL_inputWallPaperSlug(); + inputWallPaperSlug.slug = themeInfo.slug; + req.wallpaper = inputWallPaperSlug; + ConnectionsManager.getInstance(themeInfo.account).sendRequest(req, (response, error) -> AndroidUtilities.runOnUIThread(() -> { + if (response instanceof TLRPC.TL_wallPaper) { + TLRPC.WallPaper wallPaper = (TLRPC.WallPaper) response; + String name = FileLoader.getAttachFileName(wallPaper.document); + if (!loadingThemes.containsKey(name)) { + loadingThemes.put(name, themeInfo); + FileLoader.getInstance(themeInfo.account).loadFile(wallPaper.document, wallPaper, FileLoader.PRIORITY_NORMAL, 1); + } + } else { + themeInfo.badWallpaper = true; + } + })); + } + return false; + } + } + themeInfo.previewParsed = true; + return true; + } + + private void updateState(boolean animated) { + if (!dataLoaded) { + AndroidUtilities.updateViewVisibilityAnimated(progressView, true, 1f, true, animated); + } else { + AndroidUtilities.updateViewVisibilityAnimated(progressView, false, 1f, true, animated); + } + } + + @Override + protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { + super.onMeasure(MeasureSpec.makeMeasureSpec(MeasureSpec.getSize(widthMeasureSpec), MeasureSpec.EXACTLY), heightMeasureSpec); + } + } + + private BaseFragment bulletinFragment; + public ChannelColorActivity setOnApplied(BaseFragment bulletinFragment) { + this.bulletinFragment = bulletinFragment; + return this; + } + + private void showBulletin() { + if (bulletinFragment != null) { + if (bulletinFragment instanceof ChatEditActivity) { + ((ChatEditActivity) bulletinFragment).updateColorCell(); + } + BulletinFactory.of(bulletinFragment).createSimpleBulletin( + R.raw.contact_check, + LocaleController.getString(R.string.ChannelAppearanceUpdated) + ).show(); + bulletinFragment = null; + } + } + + public void updateColors() { + actionBar.setBackgroundColor(getThemedColor(Theme.key_actionBarDefault)); + actionBar.setTitleColor(getThemedColor(Theme.key_actionBarDefaultTitle)); + actionBar.setItemsColor(getThemedColor(Theme.key_actionBarDefaultIcon), false); + actionBar.setItemsBackgroundColor(getThemedColor(Theme.key_actionBarDefaultSelector), false); + listView.setBackgroundColor(getThemedColor(Theme.key_windowBackgroundGray)); + adapter.notifyDataSetChanged(); + AndroidUtilities.forEachViews(listView, this::updateColors); + buttonContainer.setBackgroundColor(getThemedColor(Theme.key_windowBackgroundGray)); + button.updateColors(); + setNavigationBarColor(getNavigationBarColor()); + } + + public boolean hasUnsavedChanged() { + return ( + currentReplyColor != selectedReplyColor || + currentReplyEmoji != selectedReplyEmoji || + currentProfileColor != selectedProfileColor || + currentProfileEmoji != selectedProfileEmoji || + !DialogObject.emojiStatusesEqual(currentStatusEmoji, selectedStatusEmoji) || + !ChatThemeController.wallpaperEquals(currentWallpaper, selectedWallpaper) + ); + } + + private void updateColors(View view) { + if (view instanceof TextInfoPrivacyCell) { + ((TextInfoPrivacyCell) view).setBackground(Theme.getThemedDrawableByKey(getContext(), listView.getChildAdapterPosition(view) == statusHintRow ? R.drawable.greydivider_bottom : R.drawable.greydivider, Theme.key_windowBackgroundGrayShadow, resourceProvider)); + } else { + view.setBackgroundColor(getThemedColor(Theme.key_windowBackgroundWhite)); + if (view instanceof EmojiCell) { + ((EmojiCell) view).updateColors(); + } else if (view instanceof TextCell) { + ((TextCell) view).updateColors(); + } else if (view instanceof PeerColorPicker) { + ((PeerColorPicker) view).updateColors(); + } else if (view instanceof ThemeChooser) { + ((ThemeChooser) view).updateColors(); + } + } + } + + private static class PeerColorPicker extends FrameLayout { + private final Theme.ResourcesProvider resourcesProvider; + public final RecyclerListView listView; + public final LinearLayoutManager layoutManager; + public final RecyclerListView.SelectionAdapter adapter; + private final int currentAccount; + + @Override + public boolean onInterceptTouchEvent(MotionEvent e) { + if (getParent() != null) { + getParent().requestDisallowInterceptTouchEvent(canScrollHorizontally(-1) || canScrollHorizontally(1)); + } + return super.onInterceptTouchEvent(e); + } + + public PeerColorPicker(Context context, final int currentAccount, Theme.ResourcesProvider resourcesProvider) { + super(context); + this.currentAccount = currentAccount; + this.resourcesProvider = resourcesProvider; + + listView = new RecyclerListView(context, resourcesProvider) { + @Override + public Integer getSelectorColor(int position) { + return 0; + } + }; + listView.setPadding(dp(6), dp(5), dp(6), 0); + listView.setClipToPadding(false); + + listView.setAdapter(adapter = new RecyclerListView.SelectionAdapter() { + @Override + public boolean isEnabled(RecyclerView.ViewHolder holder) { + return true; + } + + @NonNull + @Override + public RecyclerView.ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) { + return new RecyclerListView.Holder(new ColorCell(context)); + } + + @Override + public void onBindViewHolder(@NonNull RecyclerView.ViewHolder holder, int position) { + ColorCell cell = (ColorCell) holder.itemView; + cell.setBackgroundColor(Theme.getColor(Theme.key_windowBackgroundWhite, resourcesProvider)); + cell.setSelected(position == selectedPosition, false); + MessagesController.PeerColors peerColors = MessagesController.getInstance(currentAccount).peerColors; + if (peerColors != null && position >= 0 && position < peerColors.colors.size()) { + cell.set(peerColors.colors.get(position)); + } + } + + @Override + public int getItemCount() { + MessagesController.PeerColors peerColors = MessagesController.getInstance(currentAccount).peerColors; + return (peerColors == null ? 0 : peerColors.colors.size()); + } + }); + layoutManager = new LinearLayoutManager(context); + layoutManager.setOrientation(LinearLayoutManager.HORIZONTAL); + listView.setLayoutManager(layoutManager); + addView(listView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT)); + } + + private int selectedPosition; + public void setSelected(int color, boolean animated) { + setSelectedPosition(toPosition(color), animated); + } + + public void setSelectedPosition(int position, boolean animated) { + if (position != selectedPosition) { + selectedPosition = position; + if (!animated) { + layoutManager.scrollToPositionWithOffset(position, (AndroidUtilities.displaySize.x - dp(56)) / 2); + } + AndroidUtilities.forEachViews(listView, child -> ((ColorCell) child).setSelected(listView.getChildAdapterPosition(child) == selectedPosition, animated)); + } + } + + public int getColorId() { + return toColorId(selectedPosition); + } + + public int toPosition(final int colorId) { + MessagesController.PeerColors peerColors = MessagesController.getInstance(currentAccount).peerColors; + if (peerColors == null) { + return 0; + } + for (int i = 0; i < peerColors.colors.size(); ++i) { + if (peerColors.colors.get(i).id == colorId) { + return i; + } + } + return 0; + } + + public void updateColors() { + final MessagesController.PeerColors peerColors = MessagesController.getInstance(currentAccount).peerColors; + AndroidUtilities.forEachViews(listView, view -> { + if (view instanceof ColorCell) { + ((ColorCell) view).setBackgroundColor(Theme.getColor(Theme.key_windowBackgroundWhite, resourcesProvider)); + int position = listView.getChildAdapterPosition(view); + if (peerColors != null && position >= 0 && position < peerColors.colors.size()) { + ((ColorCell) view).set(peerColors.colors.get(position)); + } + } + }); + } + + public int toColorId(int position) { + MessagesController.PeerColors peerColors = MessagesController.getInstance(currentAccount).peerColors; + if (peerColors == null || position < 0 || position >= peerColors.colors.size()) { + return 0; + } + return peerColors.colors.get(position).id; + } + + private class ColorCell extends View { + private final Paint backgroundPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + private final Paint paint1 = new Paint(Paint.ANTI_ALIAS_FLAG); + private final Paint paint2 = new Paint(Paint.ANTI_ALIAS_FLAG); + private final Paint paint3 = new Paint(Paint.ANTI_ALIAS_FLAG); + private final Path circlePath = new Path(); + private final Path color2Path = new Path(); + private boolean hasColor2, hasColor3; + + private final ButtonBounce bounce = new ButtonBounce(this); + + public ColorCell(Context context) { + super(context); + backgroundPaint.setStyle(Paint.Style.STROKE); + } + + public void setBackgroundColor(int backgroundColor) { + backgroundPaint.setColor(backgroundColor); + } + + public void set(int color) { + hasColor2 = hasColor3 = false; + paint1.setColor(color); + } + + public void set(int color1, int color2) { + hasColor2 = true; + hasColor3 = false; + paint1.setColor(color1); + paint2.setColor(color2); + } + + public void set(MessagesController.PeerColor color) { + final boolean isDark = resourcesProvider != null ? resourcesProvider.isDark() : Theme.isCurrentThemeDark(); + if (isDark && color.hasColor2() && !color.hasColor3()) { + paint1.setColor(color.getColor(1, resourcesProvider)); + paint2.setColor(color.getColor(0, resourcesProvider)); + } else { + paint1.setColor(color.getColor(0, resourcesProvider)); + paint2.setColor(color.getColor(1, resourcesProvider)); + } + paint3.setColor(color.getColor(2, resourcesProvider)); + hasColor2 = color.hasColor2(); + hasColor3 = color.hasColor3(); + } + + private boolean selected; + private final AnimatedFloat selectedT = new AnimatedFloat(this, 0, 320, CubicBezierInterpolator.EASE_OUT_QUINT); + public void setSelected(boolean selected, boolean animated) { + this.selected = selected; + if (!animated) { + selectedT.set(selected, true); + } + invalidate(); + } + + private static final int VIEW_SIZE_DP = 56; + private static final int CIRCLE_RADIUS_DP = 20; + + @Override + protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { + setMeasuredDimension(dp(VIEW_SIZE_DP), dp(VIEW_SIZE_DP)); + + circlePath.rewind(); + circlePath.addCircle(getMeasuredWidth() / 2f, getMeasuredHeight() / 2f, dp(CIRCLE_RADIUS_DP), Path.Direction.CW); + + color2Path.rewind(); + color2Path.moveTo(getMeasuredWidth(), 0); + color2Path.lineTo(getMeasuredWidth(), getMeasuredHeight()); + color2Path.lineTo(0, getMeasuredHeight()); + color2Path.close(); + } + + @Override + protected void dispatchDraw(Canvas canvas) { + canvas.save(); + final float s = bounce.getScale(.05f); + canvas.scale(s, s, getMeasuredWidth() / 2f, getMeasuredHeight() / 2f); + + canvas.save(); + canvas.clipPath(circlePath); + canvas.drawPaint(paint1); + if (hasColor2) { + canvas.drawPath(color2Path, paint2); + } + canvas.restore(); + + if (hasColor3) { + canvas.save(); + AndroidUtilities.rectTmp.set( + (getMeasuredWidth() - dp(12.4f)) / 2f, + (getMeasuredHeight() - dp(12.4f)) / 2f, + (getMeasuredWidth() + dp(12.4f)) / 2f, + (getMeasuredHeight() + dp(12.4f)) / 2f + ); + canvas.rotate(45f, getMeasuredWidth() / 2f, getMeasuredHeight() / 2f); + canvas.drawRoundRect(AndroidUtilities.rectTmp, dp(2.33f), dp(2.33f), paint3); + canvas.restore(); + } + + final float selectT = selectedT.set(selected); + + if (selectT > 0) { + backgroundPaint.setStrokeWidth(dpf2(2)); + canvas.drawCircle( + getMeasuredWidth() / 2f, getMeasuredHeight() / 2f, + AndroidUtilities.lerp( + dp(CIRCLE_RADIUS_DP) + backgroundPaint.getStrokeWidth() * .5f, + dp(CIRCLE_RADIUS_DP) - backgroundPaint.getStrokeWidth() * 2f, + selectT + ), + backgroundPaint + ); + } + + canvas.restore(); + } + + @Override + public void setPressed(boolean pressed) { + super.setPressed(pressed); + bounce.setPressed(pressed); + } + } + } + + private View changeDayNightView; + private float changeDayNightViewProgress; + private ValueAnimator changeDayNightViewAnimator; + + @SuppressLint("NotifyDataSetChanged") + public void toggleTheme() { + FrameLayout decorView1 = (FrameLayout) getParentActivity().getWindow().getDecorView(); + Bitmap bitmap = Bitmap.createBitmap(decorView1.getWidth(), decorView1.getHeight(), Bitmap.Config.ARGB_8888); + Canvas bitmapCanvas = new Canvas(bitmap); + dayNightItem.setAlpha(0f); + decorView1.draw(bitmapCanvas); + dayNightItem.setAlpha(1f); + + Paint xRefPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + xRefPaint.setColor(0xff000000); + xRefPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.CLEAR)); + + Paint bitmapPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + bitmapPaint.setFilterBitmap(true); + int[] position = new int[2]; + dayNightItem.getLocationInWindow(position); + float x = position[0]; + float y = position[1]; + float cx = x + dayNightItem.getMeasuredWidth() / 2f; + float cy = y + dayNightItem.getMeasuredHeight() / 2f; + + float r = Math.max(bitmap.getHeight(), bitmap.getWidth()) + AndroidUtilities.navigationBarHeight; + + Shader bitmapShader = new BitmapShader(bitmap, Shader.TileMode.CLAMP, Shader.TileMode.CLAMP); + bitmapPaint.setShader(bitmapShader); + changeDayNightView = new View(getContext()) { + @Override + protected void onDraw(Canvas canvas) { + super.onDraw(canvas); + if (isDark) { + if (changeDayNightViewProgress > 0f) { + bitmapCanvas.drawCircle(cx, cy, r * changeDayNightViewProgress, xRefPaint); + } + canvas.drawBitmap(bitmap, 0, 0, bitmapPaint); + } else { + canvas.drawCircle(cx, cy, r * (1f - changeDayNightViewProgress), bitmapPaint); + } + canvas.save(); + canvas.translate(x, y); + dayNightItem.draw(canvas); + canvas.restore(); + } + }; + changeDayNightView.setOnTouchListener((v, event) -> true); + changeDayNightViewProgress = 0f; + changeDayNightViewAnimator = ValueAnimator.ofFloat(0, 1f); + changeDayNightViewAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() { + boolean changedNavigationBarColor = false; + + @Override + public void onAnimationUpdate(ValueAnimator valueAnimator) { + changeDayNightViewProgress = (float) valueAnimator.getAnimatedValue(); + changeDayNightView.invalidate(); + if (!changedNavigationBarColor && changeDayNightViewProgress > .5f) { + changedNavigationBarColor = true; + } + } + }); + changeDayNightViewAnimator.addListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + if (changeDayNightView != null) { + if (changeDayNightView.getParent() != null) { + ((ViewGroup) changeDayNightView.getParent()).removeView(changeDayNightView); + } + changeDayNightView = null; + } + changeDayNightViewAnimator = null; + super.onAnimationEnd(animation); + } + }); + changeDayNightViewAnimator.setDuration(400); + changeDayNightViewAnimator.setInterpolator(Easings.easeInOutQuad); + changeDayNightViewAnimator.start(); + + decorView1.addView(changeDayNightView, new ViewGroup.LayoutParams(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT)); + + AndroidUtilities.runOnUIThread(() -> { + if (resourceProvider instanceof ThemeDelegate) { + ((ThemeDelegate) resourceProvider).toggle(); + } else { + isDark = !isDark; + updateThemeColors(); + } + setForceDark(isDark, true); + updateColors(); + }); + } + + private boolean forceDark = isDark; + public void setForceDark(boolean isDark, boolean playAnimation) { + if (forceDark == isDark) { + return; + } + forceDark = isDark; + if (playAnimation) { + sunDrawable.setCustomEndFrame(isDark ? sunDrawable.getFramesCount() : 0); + if (sunDrawable != null) { + sunDrawable.start(); + } + } else { + int frame = isDark ? sunDrawable.getFramesCount() - 1 : 0; + sunDrawable.setCurrentFrame(frame, false, true); + sunDrawable.setCustomEndFrame(frame); + if (dayNightItem != null) { + dayNightItem.invalidate(); + } + } + } + + private Theme.ResourcesProvider parentResourcesProvider; + private final SparseIntArray currentColors = new SparseIntArray(); + private final Theme.MessageDrawable msgInDrawable, msgInDrawableSelected; + private final Paint dividerPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + { + dividerPaint.setStrokeWidth(1); + dividerPaint.setColor(Theme.getColor(Theme.key_divider, resourceProvider)); + } + + public void updateThemeColors() { + SharedPreferences preferences = ApplicationLoader.applicationContext.getSharedPreferences("themeconfig", Activity.MODE_PRIVATE); + String dayThemeName = preferences.getString("lastDayTheme", "Blue"); + if (Theme.getTheme(dayThemeName) == null || Theme.getTheme(dayThemeName).isDark()) { + dayThemeName = "Blue"; + } + String nightThemeName = preferences.getString("lastDarkTheme", "Dark Blue"); + if (Theme.getTheme(nightThemeName) == null || !Theme.getTheme(nightThemeName).isDark()) { + nightThemeName = "Dark Blue"; + } + Theme.ThemeInfo themeInfo = Theme.getActiveTheme(); + if (dayThemeName.equals(nightThemeName)) { + if (themeInfo.isDark() || dayThemeName.equals("Dark Blue") || dayThemeName.equals("Night")) { + dayThemeName = "Blue"; + } else { + nightThemeName = "Dark Blue"; + } + } + + if (isDark) { + themeInfo = Theme.getTheme(nightThemeName); + } else { + themeInfo = Theme.getTheme(dayThemeName); + } + + currentColors.clear(); + final String[] wallpaperLink = new String[1]; + final SparseIntArray themeColors; + if (themeInfo.assetName != null) { + themeColors = Theme.getThemeFileValues(null, themeInfo.assetName, wallpaperLink); + } else { + themeColors = Theme.getThemeFileValues(new File(themeInfo.pathToFile), null, wallpaperLink); + } + int[] defaultColors = Theme.getDefaultColors(); + if (defaultColors != null) { + for (int i = 0; i < defaultColors.length; ++i) { + currentColors.put(i, defaultColors[i]); + } + } + if (themeColors != null) { + for (int i = 0; i < themeColors.size(); ++i) { + currentColors.put(themeColors.keyAt(i), themeColors.valueAt(i)); + } + Theme.ThemeAccent accent = themeInfo.getAccent(false); + if (accent != null) { + accent.fillAccentColors(themeColors, currentColors); + } + } + dividerPaint.setColor(Theme.getColor(Theme.key_divider, resourceProvider)); + + backgroundDrawable = PreviewView.getBackgroundDrawable(backgroundDrawable, currentAccount, selectedWallpaper, isDark); + View messagesCellPreview = findChildAt(messagesPreviewRow); + if (messagesCellPreview instanceof ThemePreviewMessagesCell) { + ((ThemePreviewMessagesCell) messagesCellPreview).setOverrideBackground(backgroundDrawable); + } + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/ChannelWallpaperActivity.java b/TMessagesProj/src/main/java/org/telegram/ui/ChannelWallpaperActivity.java new file mode 100644 index 00000000000..4bda72b241d --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/ChannelWallpaperActivity.java @@ -0,0 +1,491 @@ +package org.telegram.ui; + +import static org.telegram.messenger.AndroidUtilities.dp; + +import android.animation.Animator; +import android.animation.AnimatorListenerAdapter; +import android.animation.ValueAnimator; +import android.annotation.SuppressLint; +import android.content.Context; +import android.graphics.Bitmap; +import android.graphics.BitmapShader; +import android.graphics.Canvas; +import android.graphics.Paint; +import android.graphics.PorterDuff; +import android.graphics.PorterDuffXfermode; +import android.graphics.Shader; +import android.view.Gravity; +import android.view.View; +import android.view.ViewGroup; +import android.widget.FrameLayout; + +import androidx.annotation.NonNull; +import androidx.recyclerview.widget.DefaultItemAnimator; +import androidx.recyclerview.widget.LinearLayoutManager; +import androidx.recyclerview.widget.RecyclerView; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.ChatThemeController; +import org.telegram.messenger.DialogObject; +import org.telegram.messenger.LocaleController; +import org.telegram.messenger.MessagesController; +import org.telegram.messenger.R; +import org.telegram.messenger.Utilities; +import org.telegram.tgnet.TLRPC; +import org.telegram.tgnet.tl.TL_stories; +import org.telegram.ui.ActionBar.ActionBar; +import org.telegram.ui.ActionBar.ActionBarMenuItem; +import org.telegram.ui.ActionBar.BaseFragment; +import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.Cells.TextCell; +import org.telegram.ui.Cells.TextInfoPrivacyCell; +import org.telegram.ui.Components.ChatThemeBottomSheet; +import org.telegram.ui.Components.CubicBezierInterpolator; +import org.telegram.ui.Components.Easings; +import org.telegram.ui.Components.LayoutHelper; +import org.telegram.ui.Components.RLottieDrawable; +import org.telegram.ui.Components.RecyclerListView; + +import java.util.Locale; + +public class ChannelWallpaperActivity extends BaseFragment { + + public final long dialogId; + public int currentLevel; + public TL_stories.TL_premium_boostsStatus boostsStatus; + public TLRPC.WallPaper galleryWallpaper; + public TLRPC.WallPaper currentWallpaper, selectedWallpaper; + + public ChannelWallpaperActivity(long dialogId, TL_stories.TL_premium_boostsStatus boostsStatus) { + super(); + this.dialogId = dialogId; + TLRPC.Chat chat = getMessagesController().getChat(-dialogId); + if (chat != null) { + currentLevel = chat.level; + } + this.boostsStatus = boostsStatus; + if (boostsStatus == null) { + MessagesController.getInstance(currentAccount).getBoostsController().getBoostsStats(dialogId, loadedBoostsStatus -> { + this.boostsStatus = loadedBoostsStatus; + if (boostsStatus != null) { + this.currentLevel = boostsStatus.level; + if (chat != null) { + chat.flags |= 1024; + chat.level = currentLevel; + } + } + }); + } else { + currentLevel = boostsStatus.level; + } + + TLRPC.ChatFull chatFull = getMessagesController().getChatFull(-dialogId); + if (chatFull != null) { + currentWallpaper = selectedWallpaper = chatFull.wallpaper; + if (ChatThemeController.isNotEmoticonWallpaper(selectedWallpaper)) { + galleryWallpaper = selectedWallpaper; + } + } + } + + public void setSelectedWallpaper(TLRPC.WallPaper wallpaper, TLRPC.WallPaper galleryWallpaper) { + selectedWallpaper = wallpaper; + this.galleryWallpaper = galleryWallpaper; + } + + private Utilities.Callback3 onSelectedWallpaperChange; + public void setOnSelectedWallpaperChange(Utilities.Callback3 listener) { + onSelectedWallpaperChange = listener; + } + + public FrameLayout contentView; + public RecyclerListView listView; + public Adapter adapter; + public boolean isDark() { + return resourceProvider != null ? resourceProvider.isDark() : Theme.isCurrentThemeDark(); + } + + private RLottieDrawable sunDrawable; + private ActionBarMenuItem dayNightItem; + + @Override + public View createView(Context context) { + actionBar.setBackButtonImage(R.drawable.ic_ab_back); + actionBar.setTitle(LocaleController.getString(R.string.ChannelWallpaper)); + actionBar.setActionBarMenuOnItemClick(new ActionBar.ActionBarMenuOnItemClick() { + @Override + public void onItemClick(int id) { + if (id == -1) { + finishFragment(); + } else if (id == 1) { + toggleTheme(); + } + } + }); + + sunDrawable = new RLottieDrawable(R.raw.sun, "" + R.raw.sun, dp(28), dp(28), true, null); + sunDrawable.setPlayInDirectionOfCustomEndFrame(true); + if (!isDark()) { + sunDrawable.setCustomEndFrame(0); + sunDrawable.setCurrentFrame(0); + } else { + sunDrawable.setCurrentFrame(35); + sunDrawable.setCustomEndFrame(36); + } + sunDrawable.beginApplyLayerColors(); + int color = Theme.getColor(Theme.key_chats_menuName, resourceProvider); + sunDrawable.setLayerColor("Sunny.**", color); + sunDrawable.setLayerColor("Path 6.**", color); + sunDrawable.setLayerColor("Path.**", color); + sunDrawable.setLayerColor("Path 5.**", color); + if (resourceProvider instanceof ChannelColorActivity.ThemeDelegate) { + dayNightItem = actionBar.createMenu().addItem(1, sunDrawable); + } + + contentView = new FrameLayout(context); + + updateRows(); + listView = new RecyclerListView(context, resourceProvider); + listView.setAdapter(adapter = new Adapter()); + listView.setLayoutManager(new LinearLayoutManager(context)); + contentView.addView(listView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT, Gravity.FILL)); + listView.setOnItemClickListener((view, position) -> { + if (position == removeRow) { + galleryWallpaper = null; + selectedWallpaper = null; + if (onSelectedWallpaperChange != null) { + onSelectedWallpaperChange.run(currentWallpaper, selectedWallpaper, galleryWallpaper); + } + + View themesView = findChildAt(themesRow); + if (themesView instanceof ChannelColorActivity.ThemeChooser) { + ((ChannelColorActivity.ThemeChooser) themesView).setGalleryWallpaper(galleryWallpaper); + } + updateRows(); + } else if (position == galleryRow) { + ChatThemeBottomSheet.openGalleryForBackground(getParentActivity(), this, dialogId, resourceProvider, wallpaper -> { + galleryWallpaper = currentWallpaper = selectedWallpaper = wallpaper; + if (onSelectedWallpaperChange != null) { + onSelectedWallpaperChange.run(currentWallpaper, selectedWallpaper, galleryWallpaper); + } + finishFragment(); + }, toggleThemeDelegate, boostsStatus); + } + }); + DefaultItemAnimator itemAnimator = new DefaultItemAnimator(); + itemAnimator.setDurations(350); + itemAnimator.setInterpolator(CubicBezierInterpolator.EASE_OUT_QUINT); + itemAnimator.setDelayAnimations(false); + itemAnimator.setSupportsChangeAnimations(false); + listView.setItemAnimator(itemAnimator); + + updateColors(); + + return fragmentView = contentView; + } + + public View findChildAt(int position) { + for (int i = 0; i < listView.getChildCount(); ++i) { + View child = listView.getChildAt(i); + if (listView.getChildAdapterPosition(child) == position) { + return child; + } + } + return null; + } + + public static final int VIEW_TYPE_BUTTON = 0; + public static final int VIEW_TYPE_INFO = 1; + public static final int VIEW_TYPE_THEMES = 2; + + public int rowsCount = 0; + public int galleryRow = -1; + public int removeRow = -1; + public int infoRow = -1; + public int themesRow = -1; + + public void updateRows() { + rowsCount = 0; + galleryRow = rowsCount++; + final int wasRemoveRow = removeRow; + if (galleryWallpaper != null) { + removeRow = rowsCount++; + } else { + removeRow = -1; + } + if (adapter != null) { + if (removeRow != -1 && wasRemoveRow == -1) { + adapter.notifyItemInserted(removeRow); + } + if (removeRow == -1 && wasRemoveRow != -1) { + adapter.notifyItemRemoved(wasRemoveRow); + } + } + infoRow = rowsCount++; + themesRow = rowsCount++; + } + + public class Adapter extends RecyclerListView.SelectionAdapter { + + @NonNull + @Override + public RecyclerView.ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) { + View view; + if (viewType == VIEW_TYPE_BUTTON) { + TextCell textCell = new TextCell(getContext(), resourceProvider); + textCell.setBackgroundColor(getThemedColor(Theme.key_windowBackgroundWhite)); + view = textCell; + } else if (viewType == VIEW_TYPE_THEMES) { + ChannelColorActivity.ThemeChooser themesWallpaper = new ChannelColorActivity.ThemeChooser(getContext(), true, currentAccount, resourceProvider); + themesWallpaper.setSelectedEmoticon(ChatThemeController.getWallpaperEmoticon(selectedWallpaper), false); + themesWallpaper.setGalleryWallpaper(galleryWallpaper); + themesWallpaper.setOnEmoticonSelected(emoticon -> { + if (emoticon == null) { + selectedWallpaper = galleryWallpaper; + themesWallpaper.setSelectedEmoticon(null, false); + if (onSelectedWallpaperChange != null) { + onSelectedWallpaperChange.run(currentWallpaper, selectedWallpaper, galleryWallpaper); + } + updateRows(); + return; + } + + ThemePreviewActivity themePreviewActivity = new ThemePreviewActivity(new WallpapersListActivity.EmojiWallpaper(emoticon), null) { + @Override + public boolean insideBottomSheet() { + return true; + } + }; + themePreviewActivity.boostsStatus = boostsStatus; + themePreviewActivity.setOnSwitchDayNightDelegate(toggleThemeDelegate); + themePreviewActivity.setResourceProvider(resourceProvider); + themePreviewActivity.setInitialModes(false, false, .20f); + themePreviewActivity.setDialogId(dialogId); + themePreviewActivity.setDelegate(wallPaper -> { + selectedWallpaper = new TLRPC.TL_wallPaperNoFile(); + selectedWallpaper.id = 0; + selectedWallpaper.flags |= 4; + selectedWallpaper.settings = new TLRPC.TL_wallPaperSettings(); + selectedWallpaper.settings.emoticon = emoticon; + themesWallpaper.setSelectedEmoticon(emoticon, false); + if (onSelectedWallpaperChange != null) { + onSelectedWallpaperChange.run(currentWallpaper, selectedWallpaper, galleryWallpaper); + } + updateRows(); + finishFragment(); + }); + BaseFragment.BottomSheetParams params = new BaseFragment.BottomSheetParams(); + params.transitionFromLeft = true; + params.allowNestedScroll = false; + params.occupyNavigationBar = true; + showAsSheet(themePreviewActivity, params); + }); + themesWallpaper.setBackgroundColor(getThemedColor(Theme.key_windowBackgroundWhite)); + view = themesWallpaper; + } else { + view = new TextInfoPrivacyCell(getContext()); + } + return new RecyclerListView.Holder(view); + } + + @Override + public int getItemViewType(int position) { + if (position == galleryRow || position == removeRow) { + return VIEW_TYPE_BUTTON; + } + if (position == themesRow) { + return VIEW_TYPE_THEMES; + } + return VIEW_TYPE_INFO; + } + + @Override + public void onBindViewHolder(@NonNull RecyclerView.ViewHolder holder, int position) { + if (position == galleryRow) { + ((TextCell) holder.itemView).setTextAndIcon(LocaleController.getString(R.string.ChooseFromGallery2), R.drawable.msg_background, removeRow != -1); + ((TextCell) holder.itemView).setColors(Theme.key_windowBackgroundWhiteBlueIcon, Theme.key_windowBackgroundWhiteBlueButton); + } else if (position == removeRow) { + ((TextCell) holder.itemView).setTextAndIcon(LocaleController.getString(R.string.ChannelWallpaperRemove), R.drawable.msg_delete, false); + ((TextCell) holder.itemView).setColors(Theme.key_text_RedRegular, Theme.key_text_RedRegular); + } else if (position == infoRow) { + ((TextInfoPrivacyCell) holder.itemView).setText(LocaleController.getString(R.string.ChannelWallpaperInfo)); + ((TextInfoPrivacyCell) holder.itemView).setBackgroundColor(getThemedColor(Theme.key_windowBackgroundGray)); + ((TextInfoPrivacyCell) holder.itemView).setForeground(Theme.getThemedDrawableByKey(getContext(), R.drawable.greydivider, Theme.key_windowBackgroundGrayShadow, resourceProvider)); + } else if (position == themesRow) { + ((ChannelColorActivity.ThemeChooser) holder.itemView).setGalleryWallpaper(galleryWallpaper); + } + } + + @Override + public void onViewAttachedToWindow(@NonNull RecyclerView.ViewHolder holder) { + if (holder.itemView instanceof ChannelColorActivity.ThemeChooser) { + ((ChannelColorActivity.ThemeChooser) holder.itemView).setGalleryWallpaper(galleryWallpaper); + } + super.onViewAttachedToWindow(holder); + } + + @Override + public int getItemCount() { + return rowsCount; + } + + @Override + public boolean isEnabled(RecyclerView.ViewHolder holder) { + return holder.getItemViewType() == VIEW_TYPE_BUTTON; + } + } + + public void updateColors() { + actionBar.setBackgroundColor(getThemedColor(Theme.key_actionBarDefault)); + actionBar.setTitleColor(getThemedColor(Theme.key_actionBarDefaultTitle)); + actionBar.setItemsColor(getThemedColor(Theme.key_actionBarDefaultIcon), false); + actionBar.setItemsBackgroundColor(getThemedColor(Theme.key_actionBarDefaultSelector), false); + listView.setBackgroundColor(getThemedColor(Theme.key_windowBackgroundGray)); + adapter.notifyDataSetChanged(); + AndroidUtilities.forEachViews(listView, this::updateColors); + setNavigationBarColor(getNavigationBarColor()); + contentView.setBackgroundColor(getThemedColor(Theme.key_windowBackgroundWhite)); + } + + private void updateColors(View view) { + if (view instanceof TextInfoPrivacyCell) { + ((TextInfoPrivacyCell) view).setBackgroundColor(getThemedColor(Theme.key_windowBackgroundGray)); + ((TextInfoPrivacyCell) view).setForeground(Theme.getThemedDrawableByKey(getContext(), R.drawable.greydivider, Theme.key_windowBackgroundGrayShadow, resourceProvider)); + } else { + view.setBackgroundColor(getThemedColor(Theme.key_windowBackgroundWhite)); + if (view instanceof TextCell) { + ((TextCell) view).updateColors(); + } else if (view instanceof ChannelColorActivity.ThemeChooser) { + ((ChannelColorActivity.ThemeChooser) view).updateColors(); + } + } + } + + public ThemePreviewActivity.DayNightSwitchDelegate toggleThemeDelegate = new ThemePreviewActivity.DayNightSwitchDelegate() { + @Override + public boolean isDark() { + return ChannelWallpaperActivity.this.isDark(); + } + + @Override + public void switchDayNight(boolean animated) { + if (resourceProvider instanceof ChannelColorActivity.ThemeDelegate) { + ((ChannelColorActivity.ThemeDelegate) resourceProvider).toggle(); + } + setForceDark(isDark(), false); + updateColors(); + } + + @Override + public boolean supportsAnimation() { + return false; + } + }; + + private View changeDayNightView; + private float changeDayNightViewProgress; + private ValueAnimator changeDayNightViewAnimator; + + @SuppressLint("NotifyDataSetChanged") + public void toggleTheme() { + FrameLayout decorView1 = (FrameLayout) getParentActivity().getWindow().getDecorView(); + Bitmap bitmap = Bitmap.createBitmap(decorView1.getWidth(), decorView1.getHeight(), Bitmap.Config.ARGB_8888); + Canvas bitmapCanvas = new Canvas(bitmap); + dayNightItem.setAlpha(0f); + decorView1.draw(bitmapCanvas); + dayNightItem.setAlpha(1f); + + Paint xRefPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + xRefPaint.setColor(0xff000000); + xRefPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.CLEAR)); + + Paint bitmapPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + bitmapPaint.setFilterBitmap(true); + int[] position = new int[2]; + dayNightItem.getLocationInWindow(position); + float x = position[0]; + float y = position[1]; + float cx = x + dayNightItem.getMeasuredWidth() / 2f; + float cy = y + dayNightItem.getMeasuredHeight() / 2f; + + float r = Math.max(bitmap.getHeight(), bitmap.getWidth()) + AndroidUtilities.navigationBarHeight; + + Shader bitmapShader = new BitmapShader(bitmap, Shader.TileMode.CLAMP, Shader.TileMode.CLAMP); + bitmapPaint.setShader(bitmapShader); + changeDayNightView = new View(getContext()) { + @Override + protected void onDraw(Canvas canvas) { + super.onDraw(canvas); + if (isDark()) { + if (changeDayNightViewProgress > 0f) { + bitmapCanvas.drawCircle(cx, cy, r * changeDayNightViewProgress, xRefPaint); + } + canvas.drawBitmap(bitmap, 0, 0, bitmapPaint); + } else { + canvas.drawCircle(cx, cy, r * (1f - changeDayNightViewProgress), bitmapPaint); + } + canvas.save(); + canvas.translate(x, y); + dayNightItem.draw(canvas); + canvas.restore(); + } + }; + changeDayNightView.setOnTouchListener((v, event) -> true); + changeDayNightViewProgress = 0f; + changeDayNightViewAnimator = ValueAnimator.ofFloat(0, 1f); + changeDayNightViewAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() { + boolean changedNavigationBarColor = false; + + @Override + public void onAnimationUpdate(ValueAnimator valueAnimator) { + changeDayNightViewProgress = (float) valueAnimator.getAnimatedValue(); + changeDayNightView.invalidate(); + if (!changedNavigationBarColor && changeDayNightViewProgress > .5f) { + changedNavigationBarColor = true; + } + } + }); + changeDayNightViewAnimator.addListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + if (changeDayNightView != null) { + if (changeDayNightView.getParent() != null) { + ((ViewGroup) changeDayNightView.getParent()).removeView(changeDayNightView); + } + changeDayNightView = null; + } + changeDayNightViewAnimator = null; + super.onAnimationEnd(animation); + } + }); + changeDayNightViewAnimator.setDuration(400); + changeDayNightViewAnimator.setInterpolator(Easings.easeInOutQuad); + changeDayNightViewAnimator.start(); + + decorView1.addView(changeDayNightView, new ViewGroup.LayoutParams(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT)); + + AndroidUtilities.runOnUIThread(() -> { + if (resourceProvider instanceof ChannelColorActivity.ThemeDelegate) { + ((ChannelColorActivity.ThemeDelegate) resourceProvider).toggle(); + } + setForceDark(isDark(), true); + updateColors(); + }); + } + + public void setForceDark(boolean isDark, boolean playAnimation) { + if (playAnimation) { + sunDrawable.setCustomEndFrame(isDark ? sunDrawable.getFramesCount() : 0); + if (sunDrawable != null) { + sunDrawable.start(); + } + } else { + int frame = isDark ? sunDrawable.getFramesCount() - 1 : 0; + sunDrawable.setCurrentFrame(frame, false, true); + sunDrawable.setCustomEndFrame(frame); + if (dayNightItem != null) { + dayNightItem.invalidate(); + } + } + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Charts/view_data/ChartHorizontalLinesData.java b/TMessagesProj/src/main/java/org/telegram/ui/Charts/view_data/ChartHorizontalLinesData.java index 4d8313fcf35..52135892107 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Charts/view_data/ChartHorizontalLinesData.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Charts/view_data/ChartHorizontalLinesData.java @@ -73,17 +73,17 @@ public ChartHorizontalLinesData(int newMaxHeight, int newMinHeight, boolean useM boolean skipFloatValues = step / k < 1; for (int i = 0; i < n; i++) { values[i] = newMinHeight + (int) (i * step); - valuesStr[i] = AndroidUtilities.formatWholeNumber(values[i], dif); + valuesStr[i] = AndroidUtilities.formatWholeNumber(values[i], 0); if (k > 0) { float v = (values[i] / k); if (skipFloatValues) { if (v - ((int) v) < 0.01f) { - valuesStr2[i] = AndroidUtilities.formatWholeNumber((int) v, (int) (dif / k)); + valuesStr2[i] = AndroidUtilities.formatWholeNumber((int) v, 0); } else { valuesStr2[i] = ""; } } else { - valuesStr2[i] = AndroidUtilities.formatWholeNumber((int) v, (int) (dif / k)); + valuesStr2[i] = AndroidUtilities.formatWholeNumber((int) v, 0); } } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/ChatActivity.java b/TMessagesProj/src/main/java/org/telegram/ui/ChatActivity.java index c6dcf3daaf5..5f02c4186a9 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/ChatActivity.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/ChatActivity.java @@ -50,6 +50,7 @@ import android.graphics.drawable.BitmapDrawable; import android.graphics.drawable.ColorDrawable; import android.graphics.drawable.Drawable; +import android.media.AudioManager; import android.net.Uri; import android.os.Build; import android.os.Bundle; @@ -199,122 +200,33 @@ import org.telegram.ui.Cells.ContextLinkCell; import org.telegram.ui.Cells.DialogCell; import org.telegram.ui.Cells.MentionCell; +import org.telegram.ui.Cells.ShareDialogCell; import org.telegram.ui.Cells.StickerCell; import org.telegram.ui.Cells.TextSelectionHelper; -import org.telegram.ui.Components.AlertsCreator; -import org.telegram.ui.Components.AnimatedEmojiDrawable; -import org.telegram.ui.Components.AnimatedEmojiSpan; -import org.telegram.ui.Components.AnimatedFileDrawable; -import org.telegram.ui.Components.AnimationProperties; -import org.telegram.ui.Components.AttachBotIntroTopView; -import org.telegram.ui.Components.AudioPlayerAlert; -import org.telegram.ui.Components.AutoDeletePopupWrapper; -import org.telegram.ui.Components.BackButtonMenu; -import org.telegram.ui.Components.BackupImageView; -import org.telegram.ui.Components.BlurBehindDrawable; -import org.telegram.ui.Components.BluredView; -import org.telegram.ui.Components.BlurredFrameLayout; -import org.telegram.ui.Components.BotCommandsMenuView; -import org.telegram.ui.Components.BotWebViewSheet; -import org.telegram.ui.Components.Bulletin; -import org.telegram.ui.Components.BulletinFactory; -import org.telegram.ui.Components.ChatActivityEnterTopView; -import org.telegram.ui.Components.ChatActivityEnterView; -import org.telegram.ui.Components.ChatActivityInterface; -import org.telegram.ui.Components.ChatAttachAlert; -import org.telegram.ui.Components.ChatAttachAlertDocumentLayout; -import org.telegram.ui.Components.ChatAvatarContainer; -import org.telegram.ui.Components.ChatBigEmptyView; -import org.telegram.ui.Components.ChatGreetingsView; -import org.telegram.ui.Components.ChatNotificationsPopupWrapper; -import org.telegram.ui.Components.ChatScrimPopupContainerLayout; -import org.telegram.ui.Components.ChatThemeBottomSheet; -import org.telegram.ui.Components.ChecksHintView; -import org.telegram.ui.Components.CircularProgressDrawable; -import org.telegram.ui.Components.ClippingImageView; -import org.telegram.ui.Components.CombinedDrawable; -import org.telegram.ui.Components.CounterView; -import org.telegram.ui.Components.CrossfadeDrawable; -import org.telegram.ui.Components.CubicBezierInterpolator; -import org.telegram.ui.Components.EditTextBoldCursor; -import org.telegram.ui.Components.EditTextCaption; -import org.telegram.ui.Components.EmbedBottomSheet; -import org.telegram.ui.Components.EmojiPacksAlert; -import org.telegram.ui.Components.EmojiView; -import org.telegram.ui.Components.ExtendedGridLayoutManager; -import org.telegram.ui.Components.FireworksOverlay; +import org.telegram.ui.Components.*; import org.telegram.ui.Components.FloatingDebug.FloatingDebugController; import org.telegram.ui.Components.FloatingDebug.FloatingDebugProvider; import org.telegram.ui.Components.Forum.ForumUtilities; -import org.telegram.ui.Components.FragmentContextView; -import org.telegram.ui.Components.GigagroupConvertAlert; -import org.telegram.ui.Components.HideViewAfterAnimation; -import org.telegram.ui.Components.HintView; -import org.telegram.ui.Components.ImageUpdater; -import org.telegram.ui.Components.ImportingAlert; -import org.telegram.ui.Components.InstantCameraView; -import org.telegram.ui.Components.InviteMembersBottomSheet; -import org.telegram.ui.Components.JoinGroupAlert; -import org.telegram.ui.Components.LayoutHelper; -import org.telegram.ui.Components.LinkSpanDrawable; -import org.telegram.ui.Components.MediaActivity; -import org.telegram.ui.Components.MentionsContainerView; -import org.telegram.ui.Components.MessageBackgroundDrawable; -import org.telegram.ui.Components.MessageContainsEmojiButton; -import org.telegram.ui.Components.MessagePreviewView; -import org.telegram.ui.Components.MotionBackgroundDrawable; -import org.telegram.ui.Components.NumberTextView; -import org.telegram.ui.Components.PhonebookShareAlert; -import org.telegram.ui.Components.PinnedLineView; -import org.telegram.ui.Components.PipRoundVideoView; -import org.telegram.ui.Components.PollVotesAlert; -import org.telegram.ui.Components.PopupSwipeBackLayout; import org.telegram.ui.Components.Premium.GiftPremiumBottomSheet; +import org.telegram.ui.Components.Premium.LimitReachedBottomSheet; import org.telegram.ui.Components.Premium.PremiumFeatureBottomSheet; import org.telegram.ui.Components.Premium.PremiumPreviewBottomSheet; import org.telegram.ui.Components.Premium.boosts.BoostDialogs; import org.telegram.ui.Components.Premium.boosts.GiftInfoBottomSheet; -import org.telegram.ui.Components.RLottieDrawable; -import org.telegram.ui.Components.RLottieImageView; -import org.telegram.ui.Components.RadialProgressView; -import org.telegram.ui.Components.ReactedHeaderView; -import org.telegram.ui.Components.ReactedUsersListView; -import org.telegram.ui.Components.ReactionTabHolderView; +import org.telegram.ui.Components.Premium.boosts.PremiumPreviewGiftLinkBottomSheet; import org.telegram.ui.Components.Reactions.ChatSelectionReactionMenuOverlay; import org.telegram.ui.Components.Reactions.ReactionsEffectOverlay; import org.telegram.ui.Components.Reactions.ReactionsLayoutInBubble; -import org.telegram.ui.Components.ReactionsContainerLayout; -import org.telegram.ui.Components.RecyclerAnimationScrollHelper; -import org.telegram.ui.Components.RecyclerListView; -import org.telegram.ui.Components.ReportAlert; -import org.telegram.ui.Components.SearchCounterView; -import org.telegram.ui.Components.ShareAlert; -import org.telegram.ui.Components.SharedMediaLayout; -import org.telegram.ui.Components.SizeNotifierFrameLayout; -import org.telegram.ui.Components.StickersAlert; -import org.telegram.ui.Components.SuggestEmojiView; -import org.telegram.ui.Components.TextSelectionHint; -import org.telegram.ui.Components.TextStyleSpan; -import org.telegram.ui.Components.ThemeEditorView; -import org.telegram.ui.Components.TranscribeButton; -import org.telegram.ui.Components.TranslateAlert2; -import org.telegram.ui.Components.TranslateButton; -import org.telegram.ui.Components.TrendingStickersAlert; -import org.telegram.ui.Components.TypefaceSpan; -import org.telegram.ui.Components.URLSpanBotCommand; -import org.telegram.ui.Components.URLSpanMono; -import org.telegram.ui.Components.URLSpanNoUnderline; -import org.telegram.ui.Components.URLSpanReplacement; -import org.telegram.ui.Components.URLSpanUserMention; -import org.telegram.ui.Components.UndoView; -import org.telegram.ui.Components.UnreadCounterTextView; -import org.telegram.ui.Components.ViewHelper; import org.telegram.ui.Components.spoilers.SpoilerEffect; import org.telegram.ui.Components.voip.CellFlickerDrawable; import org.telegram.ui.Components.voip.VoIPHelper; import org.telegram.ui.Delegates.ChatActivityMemberRequestsDelegate; +import org.telegram.ui.Stories.DialogStoriesCell; import org.telegram.ui.Stories.StoriesListPlaceProvider; import org.telegram.ui.Stories.StoriesUtilities; +import org.telegram.ui.Stories.recorder.PreviewView; +import org.telegram.ui.Stories.recorder.StoryEntry; +import org.telegram.ui.Stories.recorder.StoryRecorder; import java.io.BufferedWriter; import java.io.File; @@ -393,6 +305,7 @@ public class ChatActivity extends BaseFragment implements NotificationCenter.Not private ActionBarMenuItem.Item closeTopicItem; private ActionBarMenuItem.Item openForumItem; private ClippingImageView animatingImageView; + private ThanosEffect chatListThanosEffect; private RecyclerListView chatListView; private ChatListItemAnimator chatListItemAnimator; private GridLayoutManagerFixed chatLayoutManager; @@ -920,8 +833,8 @@ public void run() { private boolean openImport; - private float chatListViewPaddingTop; - private int chatListViewPaddingVisibleOffset; + public float chatListViewPaddingTop; + public int chatListViewPaddingVisibleOffset; private int contentPaddingTop; private float contentPanTranslation; @@ -965,6 +878,7 @@ public void run() { }; private ChatSelectionReactionMenuOverlay selectionReactionsOverlay; + private SecretVoicePlayer secretVoicePlayer; private boolean isPauseOnThemePreview; private ChatThemeBottomSheet chatThemeBottomSheet; @@ -1209,7 +1123,6 @@ public Object get(Object object) { } })); } - return items; } @@ -1696,7 +1609,7 @@ public void onMessageSend(CharSequence message, boolean notify, int scheduleDate chatActivityEnterView.getEmojiView().onMessageSend(); } - if (!getMessagesController().premiumLocked && getMessagesController().transcribeAudioTrialWeeklyNumber <= 0 && !getMessagesController().didPressTranscribeButtonEnough() && !getUserConfig().isPremium() && !TextUtils.isEmpty(message) && messages != null) { + if (!getMessagesController().premiumFeaturesBlocked() && getMessagesController().transcribeAudioTrialWeeklyNumber <= 0 && !getMessagesController().didPressTranscribeButtonEnough() && !getUserConfig().isPremium() && !TextUtils.isEmpty(message) && messages != null) { for (int i = 1; i < Math.min(5, messages.size()); ++i) { MessageObject msg = messages.get(i); if (msg != null && !msg.isOutOwner() && (msg.isVoice() || msg.isRoundVideo()) && msg.isContentUnread()) { @@ -1992,7 +1905,7 @@ public void didPressAttachButton() { } @Override - public void needStartRecordVideo(int state, boolean notify, int scheduleDate) { + public void needStartRecordVideo(int state, boolean notify, int scheduleDate, int ttl) { checkInstantCameraView(); if (instantCameraView != null) { if (state == 0) { @@ -2000,7 +1913,7 @@ public void needStartRecordVideo(int state, boolean notify, int scheduleDate) { chatListView.stopScroll(); chatAdapter.updateRowsSafe(); } else if (state == 1 || state == 3 || state == 4) { - instantCameraView.send(state, notify, scheduleDate); + instantCameraView.send(state, notify, scheduleDate, ttl); } else if (state == 2 || state == 5) { instantCameraView.cancel(state == 2); } @@ -2134,6 +2047,11 @@ public void onKeyboardRequested() { checkAdjustResize(); } + @Override + public boolean onceVoiceAvailable() { + return currentUser != null && !UserObject.isUserSelf(currentUser) && !currentUser.bot && currentEncryptedChat == null && chatMode == 0; + } + @Override public ReplyQuote getReplyQuote() { return replyingQuote; @@ -2561,7 +2479,7 @@ public boolean onFragmentCreate() { } themeDelegate = new ThemeDelegate(); - if (themeDelegate.isThemeChangeAvailable()) { + if (themeDelegate.isThemeChangeAvailable(false)) { NotificationCenter.getGlobalInstance().addObserver(this, NotificationCenter.needSetDayNightTheme); } @@ -3566,7 +3484,7 @@ public void toggleMute() { if (currentChat != null && !isTopic) { viewAsTopics = headerItem.lazilyAddSubItem(view_as_topics, R.drawable.msg_topics, LocaleController.getString("TopicViewAsTopics", R.string.TopicViewAsTopics)); } - if (themeDelegate.isThemeChangeAvailable()) { + if (themeDelegate.isThemeChangeAvailable(true)) { headerItem.lazilyAddSubItem(change_colors, R.drawable.msg_colors, LocaleController.getString("SetWallpapers", R.string.SetWallpapers)); } if (!isTopic) { @@ -3767,6 +3685,12 @@ public void setTranslationY(float translationY) { } } + @Override + protected boolean allowSelectChildAtPosition(View child) { + if (child != null && child.getVisibility() == View.INVISIBLE) return false; + return super.allowSelectChildAtPosition(child); + } + @Override protected void onMeasure(int widthSpec, int heightSpec) { saveScrollPosition(); @@ -4625,6 +4549,9 @@ private void drawChatBackgroundElements(Canvas canvas) { for (int a = 0; a < count; a++) { View child = getChildAt(a); + if (child.getVisibility() == View.INVISIBLE || child.getVisibility() == View.GONE) { + continue; + } if (chatAdapter.isBot && child instanceof BotHelpCell) { BotHelpCell botCell = (BotHelpCell) child; float top = (getMeasuredHeight() - chatListViewPaddingTop - blurredViewBottomOffset) / 2 - child.getMeasuredHeight() / 2 + chatListViewPaddingTop; @@ -4756,7 +4683,7 @@ private void drawChatBackgroundElements(Canvas canvas) { View child = chatListView.getChildAt(i); if (child instanceof ChatMessageCell) { ChatMessageCell cell = (ChatMessageCell) child; - if (child.getY() > chatListView.getHeight() || child.getY() + child.getHeight() < 0) { + if (child.getY() > chatListView.getHeight() || child.getY() + child.getHeight() < 0 || cell.getVisibility() == View.GONE) { continue; } MessageObject.GroupedMessages group = cell.getCurrentMessagesGroup(); @@ -4889,7 +4816,7 @@ public boolean drawChild(Canvas canvas, View child, long drawingTime) { ChatActionCell actionCell = null; float cilpTop = chatListViewPaddingTop - chatListViewPaddingVisibleOffset - AndroidUtilities.dp(4); - if (child.getY() > getMeasuredHeight() || child.getY() + child.getMeasuredHeight() < cilpTop) { + if (child.getY() > getMeasuredHeight() || child.getY() + child.getMeasuredHeight() < cilpTop || child.getVisibility() == View.INVISIBLE || child.getVisibility() == View.GONE) { skipDraw = true; } @@ -5031,7 +4958,7 @@ public boolean drawChild(Canvas canvas, View child, long drawingTime) { ImageReceiver imageReceiver = cell.getAvatarImage(); if (imageReceiver != null) { MessageObject.GroupedMessages groupedMessages = getValidGroupedMessage(message); - if (cell.getMessageObject().deleted) { + if (cell.getMessageObject().deleted && !cell.getMessageObject().deletedByThanos) { if (child.getTranslationY() != 0) { canvas.restore(); } @@ -5361,6 +5288,7 @@ public void endAnimations() { }); } }; + chatListItemAnimator.setOnSnapMessage(this::supportsThanosEffect, this::getChatThanosEffect); } chatLayoutManager = new GridLayoutManagerFixed(context, 1000, LinearLayoutManager.VERTICAL, true) { @@ -5662,6 +5590,9 @@ public void onScrollStateChanged(RecyclerView recyclerView, int newState) { @Override public void onScrolled(RecyclerView recyclerView, int dx, int dy) { chatListView.invalidate(); + if (chatListThanosEffect != null) { + chatListThanosEffect.scroll(dx, dy); + } scrollUp = dy < 0; int firstVisibleItem = chatLayoutManager.findFirstVisibleItemPosition(); if (dy != 0 && (scrollByTouch && recyclerView.getScrollState() == RecyclerView.SCROLL_STATE_SETTLING) || recyclerView.getScrollState() == RecyclerView.SCROLL_STATE_DRAGGING) { @@ -11211,6 +11142,7 @@ public void searchLinks(final CharSequence charSequence, final boolean force) { } textToCheck = charSequence; } + final String firstUrl = urls == null || urls.isEmpty() ? null : urls.get(0).toString(); if (currentEncryptedChat != null && messagesController.secretWebpagePreview == 2) { AndroidUtilities.runOnUIThread(() -> { @@ -11983,7 +11915,9 @@ public void showFieldPanel(boolean show, MessageObject messageObjectToReply, Mes messageObjectToReply = messageObjectsToForward.get(0); } } else if (type == MessageObject.TYPE_GIVEAWAY) { - text = LocaleController.getString("BoostingGiveaway", R.string.BoostingGiveaway);; + text = LocaleController.getString("BoostingGiveaway", R.string.BoostingGiveaway); + } else if (type == MessageObject.TYPE_GIVEAWAY_RESULTS) { + text = LocaleController.getString("BoostingGiveawayResults", R.string.BoostingGiveawayResults); } else if (type == MessageObject.TYPE_GEO) { text = LocaleController.formatPluralString("PreviewForwardLocation", messageObjectsToForward.size()); } else if (type == MessageObject.TYPE_VIDEO) { @@ -13546,6 +13480,10 @@ public void onAnimationEnd(Animator animation) { public class ChatActivityFragmentView extends SizeNotifierFrameLayout { + public ChatActivity getChatActivity() { + return ChatActivity.this; + } + public ChatActivityFragmentView(Context context, INavigationLayout parentLayout) { super(context, parentLayout); adjustPanLayoutHelper = new AdjustPanLayoutHelper(this) { @@ -14491,7 +14429,7 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { int contentWidthSpec = View.MeasureSpec.makeMeasureSpec(widthSize, View.MeasureSpec.EXACTLY); int contentHeightSpec = View.MeasureSpec.makeMeasureSpec(h, View.MeasureSpec.EXACTLY); child.measure(contentWidthSpec, contentHeightSpec); - } else if (child == chatListView) { + } else if (child == chatListView || child == chatListThanosEffect) { int contentWidthSpec = View.MeasureSpec.makeMeasureSpec(widthSize, View.MeasureSpec.EXACTLY); int h = heightSize - listViewTopHeight - (inPreviewMode && Build.VERSION.SDK_INT >= 21 ? AndroidUtilities.statusBarHeight : 0) + blurredViewTopOffset + blurredViewBottomOffset; if (keyboardSize > AndroidUtilities.dp(20) && getLayoutParams().height < 0) { @@ -14710,7 +14648,7 @@ protected void onLayout(boolean changed, int l, int t, int r, int b) { } } else if (child == gifHintTextView || child == voiceHintTextView || child == mediaBanTooltip || child == emojiHintTextView) { childTop -= inputFieldHeight; - } else if (child == chatListView || child == floatingDateView || child == infoTopView) { + } else if (child == chatListView || child == chatListThanosEffect || child == floatingDateView || child == infoTopView) { childTop -= blurredViewTopOffset; if (!inPreviewMode) { childTop -= (inputFieldHeight - AndroidUtilities.dp(51)); @@ -15569,7 +15507,7 @@ public void updateTitle(boolean animated) { } else if (chatMode == MODE_PINNED) { avatarContainer.setTitle(LocaleController.formatPluralString("PinnedMessagesCount", getPinnedMessagesCount())); } else if (currentChat != null) { - avatarContainer.setTitle(currentChat.title, currentChat.scam, currentChat.fake, currentChat.verified, false, null, animated); + avatarContainer.setTitle(currentChat.title, currentChat.scam, currentChat.fake, currentChat.verified, false, currentChat.emoji_status, animated); } else if (currentUser != null) { if (currentUser.self) { avatarContainer.setTitle(LocaleController.getString("SavedMessages", R.string.SavedMessages)); @@ -18825,7 +18763,8 @@ public void didReceivedNotification(int id, int account, final Object... args) { } if (chatActivityEnterView != null) { chatActivityEnterView.setBotsCount(botsCount, hasBotsCommands, true); - hasBotWebView = getMessagesController().getUser(info.user_id).bot_menu_webview; + TLRPC.User bot = getMessagesController().getUser(info.user_id); + hasBotWebView = bot != null && bot.bot_menu_webview; chatActivityEnterView.updateBotWebView(true); } } @@ -20802,6 +20741,7 @@ private int getSponsoredMessagesCount() { private void processDeletedMessages(ArrayList markAsDeletedMessages, long channelId) { ArrayList removedIndexes = new ArrayList<>(); + ArrayList messagesIndexes = new ArrayList<>(); int loadIndex = 0; if (ChatObject.isChannel(currentChat)) { if (channelId == 0 && mergeDialogId != 0) { @@ -20902,6 +20842,10 @@ private void processDeletedMessages(ArrayList markAsDeletedMessages, lo MessageObject removed = messages.remove(index); if (chatAdapter != null) { removedIndexes.add(chatAdapter.messagesStartRow + index); + if (removed != null && removed.messageOwner != null && removed.messageOwner.send_state == MessageObject.MESSAGE_SEND_STATE_SENT) { + messagesIndexes.add(chatAdapter.messagesStartRow + index); + removed.deletedByThanos = LiteMode.isEnabled(LiteMode.FLAG_CHAT_THANOS); + } } if (removed.getGroupId() != 0) { MessageObject.GroupedMessages groupedMessages = groupedMessagesMap.get(removed.getGroupId()); @@ -21017,7 +20961,8 @@ private void processDeletedMessages(ArrayList markAsDeletedMessages, lo int prevLoadingUpRow = chatAdapter.loadingUpRow; int prevLoadingDownRow = chatAdapter.loadingDownRow; for (int a = 0, N = removedIndexes.size(); a < N; a++) { - chatAdapter.notifyItemRemoved(removedIndexes.get(a)); + final int pos = removedIndexes.get(a); + chatAdapter.notifyItemRemoved(pos, messagesIndexes.contains(pos)); } if (!isThreadChat() || messages.size() <= 3) { removeUnreadPlane(false); @@ -21383,7 +21328,7 @@ public void onBecomeFullyVisible() { @Override public void onBecomeFullyHidden() { - if (!getMessagesController().premiumLocked && getMessagesController().transcribeAudioTrialWeeklyNumber <= 0 && !getMessagesController().didPressTranscribeButtonEnough() && !getUserConfig().isPremium() && messages != null) { + if (!getMessagesController().premiumFeaturesBlocked() && getMessagesController().transcribeAudioTrialWeeklyNumber <= 0 && !getMessagesController().didPressTranscribeButtonEnough() && !getUserConfig().isPremium() && messages != null) { for (int i = 0; i < messages.size(); ++i) { MessageObject msg = messages.get(i); if (msg != null && !msg.isOutOwner() && (msg.isVoice() || msg.isRoundVideo()) && !msg.isUnread() && (msg.isContentUnread() || ChatObject.isChannelAndNotMegaGroup(currentChat))) { @@ -23174,7 +23119,7 @@ private void updateTopPanel(boolean animated) { boolean showTranslate = ( getUserConfig().isPremium() ? getMessagesController().getTranslateController().isDialogTranslatable(getDialogId()) && !getMessagesController().getTranslateController().isTranslateDialogHidden(getDialogId()) : - !getMessagesController().premiumLocked && preferences.getInt("dialog_show_translate_count" + did, 5) <= 0 + !getMessagesController().premiumFeaturesBlocked() && preferences.getInt("dialog_show_translate_count" + did, 5) <= 0 ); if (showRestartTopic) { shownRestartTopic = true; @@ -24459,13 +24404,7 @@ private boolean createMenu(View v, boolean single, boolean listView, float x, fl final int type = getMessageType(message); if (single) { boolean isGiveawayResultsMessage = false; - if (message.messageOwner.action instanceof TLRPC.TL_messageActionCustomAction) { - TLRPC.TL_messageActionCustomAction customAction = (TLRPC.TL_messageActionCustomAction) message.messageOwner.action; - if (customAction.message != null && customAction.message.contains("giveaway")) { - //fallback for old versions - isGiveawayResultsMessage = true; - } - } else if (message.messageOwner.action instanceof TLRPC.TL_messageActionGiveawayResults) { + if (message.messageOwner.action instanceof TLRPC.TL_messageActionGiveawayResults) { isGiveawayResultsMessage = true; } if (message.messageOwner.action instanceof TLRPC.TL_messageActionPinMessage || isGiveawayResultsMessage) { @@ -24520,7 +24459,9 @@ private boolean createMenu(View v, boolean single, boolean listView, float x, fl return true; } } else if (message.messageOwner.action instanceof TLRPC.TL_messageActionSetChatTheme) { - showChatThemeBottomSheet(); + if (currentChat == null || ChatObject.canChangeChatInfo(currentChat)) { + showChatThemeBottomSheet(); + } return true; } } @@ -24606,7 +24547,7 @@ private boolean createMenu(View v, boolean single, boolean listView, float x, fl icons.add(R.drawable.msg_user_search); } - if (!getUserConfig().isPremium() && !getMessagesController().premiumLocked && message.getDocument() != null && message.getDocument().size >= 150 * 1024 * 1024 && FileLoader.getInstance(currentAccount).isLoadingFile(FileLoader.getAttachFileName(message.getDocument()))) { + if (!getUserConfig().isPremium() && !getMessagesController().premiumFeaturesBlocked() && message.getDocument() != null && message.getDocument().size >= 150 * 1024 * 1024 && FileLoader.getInstance(currentAccount).isLoadingFile(FileLoader.getAttachFileName(message.getDocument()))) { items.add(LocaleController.getString(R.string.PremiumSpeedPromo)); options.add(OPTION_SPEED_PROMO); icons.add(R.drawable.msg_speed); @@ -24680,7 +24621,7 @@ public void setAutoDeleteHistory(int time, int action) { messageTextToTranslate = null; } - if (message.isSponsored() && !getMessagesController().premiumLocked) { + if (message.isSponsored() && !getUserConfig().isPremium() && !getMessagesController().premiumFeaturesBlocked()) { items.add(LocaleController.getString("HideAd", R.string.HideAd)); options.add(OPTION_HIDE_SPONSORED_MESSAGE); icons.add(R.drawable.msg_block2); @@ -26056,7 +25997,7 @@ public void onSwipeBackProgress(PopupSwipeBackLayout layout, float toProgress, f } } - if (stickerSets.size() > 0 && !getMessagesController().premiumLocked) { + if (stickerSets.size() > 0 && !getMessagesController().premiumFeaturesBlocked()) { View gap = new FrameLayout(contentView.getContext()); gap.setBackgroundColor(getThemedColor(Theme.key_actionBarDefaultSubmenuSeparator)); popupLayout.addView(gap, LayoutHelper.createLinear(LayoutHelper.MATCH_PARENT, 8)); @@ -27617,7 +27558,10 @@ public boolean checkRecordLocked(boolean forceCloseOnDiscard) { @Override public boolean onBackPressed() { - if (closeStoryViewer()) { + if (secretVoicePlayer != null && secretVoicePlayer.isShown()) { + secretVoicePlayer.dismiss(); + return false; + } else if (closeStoryViewer()) { return false; } else if (selectionReactionsOverlay != null && !selectionReactionsOverlay.onBackPressed()) { return false; @@ -27856,7 +27800,7 @@ private ArrayList createVoiceMessagesPlaylist(MessageObject start if (messageObject.getDialogId() == mergeDialogId && startMessageObject.getDialogId() != mergeDialogId) { continue; } - if ((currentEncryptedChat == null && messageObject.getId() > messageId || currentEncryptedChat != null && messageObject.getId() < messageId) && (messageObject.isVoice() || messageObject.isRoundVideo()) && (!playingUnreadMedia || messageObject.isContentUnread() && !messageObject.isOut())) { + if ((currentEncryptedChat == null && messageObject.getId() > messageId || currentEncryptedChat != null && messageObject.getId() < messageId) && (messageObject.isVoice() || messageObject.isRoundVideo()) && !messageObject.isVoiceOnce() && !messageObject.isRoundOnce() && (!playingUnreadMedia || messageObject.isContentUnread() && !messageObject.isOut())) { messageObjects.add(messageObject); } } @@ -29238,14 +29182,24 @@ public void onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo info) { ((ChatActionCell) view).setInvalidateColors(true); ((ChatActionCell) view).setDelegate(new ChatActionCell.ChatActionCellDelegate() { @Override - public void didOpenPremiumGift(ChatActionCell cell, TLRPC.TL_premiumGiftOption giftOption, boolean animateConfetti) { - showDialog(new PremiumPreviewBottomSheet(ChatActivity.this, currentAccount, getCurrentUser(), new GiftPremiumBottomSheet.GiftTier(giftOption), themeDelegate) - .setAnimateConfetti(animateConfetti) - .setOutboundGift(cell.getMessageObject().isOut())); + public void didOpenPremiumGift(ChatActionCell cell, TLRPC.TL_premiumGiftOption giftOption, String slug, boolean animateConfetti) { + if (slug != null) { + initGiftProgressDialog(cell); + PremiumPreviewGiftLinkBottomSheet.show(slug, giftOption, getCurrentUser(), progressDialogCurrent); + } else { + showDialog(new PremiumPreviewBottomSheet(ChatActivity.this, currentAccount, getCurrentUser(), new GiftPremiumBottomSheet.GiftTier(giftOption), themeDelegate) + .setAnimateConfetti(animateConfetti) + .setOutboundGift(cell.getMessageObject().isOut())); + } } @Override public void didOpenPremiumGiftChannel(ChatActionCell cell, String slug, boolean animateConfetti) { + initGiftProgressDialog(cell); + GiftInfoBottomSheet.show(getBaseFragment(), slug, progressDialogCurrent); + } + + private void initGiftProgressDialog(ChatActionCell cell) { if (progressDialogCurrent != null) { progressDialogCurrent.cancel(true); } @@ -29270,7 +29224,6 @@ public void end(boolean replaced) { } } }; - GiftInfoBottomSheet.show(getBaseFragment(), slug, progressDialogCurrent); } @Override @@ -29320,7 +29273,7 @@ public void didClickImage(ChatActionCell cell) { } if (cell.hasButton()) { ThemePreviewActivity.showFor(ChatActivity.this, message); - } else { + } else if (currentChat == null || ChatObject.canChangeChatInfo(currentChat)) { showChatThemeBottomSheet(); } return; @@ -30263,6 +30216,26 @@ public void notifyItemRemoved(int position) { } } + public void notifyItemRemoved(int position, boolean thanos) { + if (BuildVars.LOGS_ENABLED) { + FileLog.d("notify item removed " + position + (thanos ? " with thanos effect" : "")); + } + if (!fragmentBeginToShow) { + chatListView.setItemAnimator(null); + } else if (chatListView.getItemAnimator() != chatListItemAnimator) { + chatListView.setItemAnimator(chatListItemAnimator); + } + if (thanos && chatListItemAnimator != null && chatListView.getItemAnimator() == chatListItemAnimator) { + chatListItemAnimator.prepareThanos(chatListView.findViewHolderForAdapterPosition(position)); + } + updateRowsInternal(); + try { + super.notifyItemRemoved(position); + } catch (Exception e) { + FileLog.e(e); + } + } + @Override public void notifyItemRangeRemoved(int positionStart, int itemCount) { if (BuildVars.LOGS_ENABLED) { @@ -30555,7 +30528,9 @@ public void didPressSideButton(ChatMessageCell cell) { arrayList = new ArrayList<>(); arrayList.add(messageObject); } - showDialog(new ShareAlert(getContext(), ChatActivity.this, arrayList, null, null, ChatObject.isChannel(currentChat), null, null, false, false, false, themeDelegate) { + final boolean includeStory = getMessagesController().storiesEnabled() && StoryEntry.canRepostMessage(messageObject); + showDialog(new ShareAlert(getContext(), ChatActivity.this, arrayList, null, null, ChatObject.isChannel(currentChat), null, null, false, false, includeStory, themeDelegate) { + { includeStoryFromMessage = includeStory; } @Override public void dismissInternal() { super.dismissInternal(); @@ -30565,6 +30540,50 @@ public void dismissInternal() { } } + @Override + protected void onShareStory(View cell) { + StoryRecorder.SourceView sourceView = null; + if (cell instanceof ShareDialogCell) { + sourceView = StoryRecorder.SourceView.fromShareCell((ShareDialogCell) cell); + } + final ArrayList messageObjects = new ArrayList<>(); + MessageObject.GroupedMessages groupedMessages = messageObject.getGroupId() != 0 ? groupedMessagesMap.get(messageObject.getGroupId()) : null; + if (groupedMessages != null) { + messageObjects.addAll(groupedMessages.messages); + } else { + messageObjects.add(messageObject); + } + StoryRecorder editor = StoryRecorder.getInstance(getParentActivity(), currentAccount); + editor.setOnPrepareCloseListener((t, close, sent, did) -> { + if (sent) { + AndroidUtilities.runOnUIThread(() -> { + String chatTitle = ""; + if (did < 0) { + TLRPC.Chat chat = getMessagesController().getChat(-did); + if (chat != null) { + chatTitle = chat.title; + } + } + BulletinFactory.of(ChatActivity.this).createSimpleBulletin(R.raw.contact_check, AndroidUtilities.replaceTags( + TextUtils.isEmpty(chatTitle) ? + LocaleController.getString(R.string.RepostedToProfile) : + LocaleController.formatString(R.string.RepostedToChannelProfile, chatTitle) + )).show(); + }); + dismiss(); + editor.replaceSourceView(null); + } else { + StoryRecorder.SourceView sourceView2 = null; + if (cell instanceof ShareDialogCell && cell.isAttachedToWindow()) { + sourceView2 = StoryRecorder.SourceView.fromShareCell((ShareDialogCell) cell); + } + editor.replaceSourceView(sourceView2); + } + AndroidUtilities.runOnUIThread(close); + }); + editor.openRepost(sourceView, StoryEntry.repostMessage(messageObjects)); + } + @Override protected void onSend(LongSparseArray dids, int count, TLRPC.TL_forumTopic topic) { createUndoView(); @@ -30584,8 +30603,30 @@ protected void onSend(LongSparseArray dids, int count, TLRPC.TL_fo } @Override - public boolean needPlayMessage(MessageObject messageObject, boolean muted) { - if (messageObject.isVoice() || messageObject.isRoundVideo()) { + public boolean needPlayMessage(ChatMessageCell cell, MessageObject messageObject, boolean muted) { + if (messageObject.isVoiceOnce()) { + if (secretVoicePlayer != null && secretVoicePlayer.isShown()) return false; + try { + AudioManager audioManager = (AudioManager) ApplicationLoader.applicationContext.getSystemService(Context.AUDIO_SERVICE); + int stream = AudioManager.STREAM_MUSIC; + int volume = audioManager.getStreamVolume(stream); + if (volume == 0) { + audioManager.adjustStreamVolume(stream, volume, AudioManager.FLAG_SHOW_UI); + if (!messageObject.isOutOwner()) { + BulletinFactory.of(ChatActivity.this).createImageBulletin(R.drawable.tooltip_sound, LocaleController.getString(R.string.VoiceOnceTurnOnSound)).show(true); + return false; + } + } + } catch (Exception ignore) {} + secretVoicePlayer = new SecretVoicePlayer(getContext()); + secretVoicePlayer.setCell( + cell, + !messageObject.isOutOwner() ? sendSecretMessageRead(messageObject, true) : null, + !messageObject.isOutOwner() ? sendSecretMediaDelete(messageObject) : null + ); + showDialog(secretVoicePlayer); + return false; + } else if (messageObject.isVoice() || messageObject.isRoundVideo()) { boolean result = MediaController.getInstance().playMessage(messageObject, muted); MediaController.getInstance().setVoiceMessagesPlaylist(result ? createVoiceMessagesPlaylist(messageObject, false) : null, false); return result; @@ -31286,7 +31327,7 @@ public void didPressChannelRecommendationsClose(ChatMessageCell cell) { @Override public boolean didPressAnimatedEmoji(ChatMessageCell cell, AnimatedEmojiSpan span) { - if (getMessagesController().premiumLocked || span == null || span.standard) { + if (getMessagesController().premiumFeaturesBlocked() || span == null || span.standard) { return false; } long documentId = span.getDocumentId(); @@ -31350,6 +31391,9 @@ public void didPressWebPage(ChatMessageCell cell, TLRPC.WebPage webpage, String if (uri == null) { return; } + if (!safe && Browser.isTelegraphUrl(url, false)) { + safe = true; + } if (progressDialogCurrent != null) { progressDialogCurrent.cancel(true); } @@ -31842,14 +31886,21 @@ public void end(boolean replaced) { @Override public void didPressGiveawayChatButton(ChatMessageCell cell, int pressedPos) { - TLRPC.TL_messageMediaGiveaway giveaway = (TLRPC.TL_messageMediaGiveaway) cell.getMessageObject().messageOwner.media; - long channelId = giveaway.channels.get(pressedPos); - if (dialog_id != -channelId) { - presentFragment(ChatActivity.of(-channelId)); - } else { - ViewGroup v = getChatListView(); - AndroidUtilities.shakeViewSpring(v, 5); - BotWebViewVibrationEffect.APP_ERROR.vibrate(); + if (cell.getMessageObject().messageOwner.media instanceof TLRPC.TL_messageMediaGiveaway) { + TLRPC.TL_messageMediaGiveaway giveaway = (TLRPC.TL_messageMediaGiveaway) cell.getMessageObject().messageOwner.media; + long channelId = giveaway.channels.get(pressedPos); + if (dialog_id != -channelId) { + presentFragment(ChatActivity.of(-channelId)); + } else { + ViewGroup v = getChatListView(); + AndroidUtilities.shakeViewSpring(v, 5); + BotWebViewVibrationEffect.APP_ERROR.vibrate(); + } + } + if (cell.getMessageObject().messageOwner.media instanceof TLRPC.TL_messageMediaGiveawayResults) { + TLRPC.TL_messageMediaGiveawayResults giveaway = (TLRPC.TL_messageMediaGiveawayResults) cell.getMessageObject().messageOwner.media; + long id = giveaway.winners.get(pressedPos); + presentFragment(ProfileActivity.of(id)); } } @@ -32366,7 +32417,7 @@ public ArrayList getThemeDescriptions() { } themeDescriptions.add(new ThemeDescription(chatListView, 0, new Class[]{ChatMessageCell.class}, null, new Drawable[]{msgOutDrawable, msgOutMediaDrawable}, null, Theme.key_chat_outBubble)); - if (!themeDelegate.isThemeChangeAvailable()) { + if (!themeDelegate.isThemeChangeAvailable(false)) { themeDescriptions.add(new ThemeDescription(chatListView, 0, new Class[]{ChatMessageCell.class}, null, new Drawable[]{msgOutDrawable, msgOutMediaDrawable}, null, Theme.key_chat_outBubbleGradient1)); themeDescriptions.add(new ThemeDescription(chatListView, 0, new Class[]{ChatMessageCell.class}, null, new Drawable[]{msgOutDrawable, msgOutMediaDrawable}, null, Theme.key_chat_outBubbleGradient2)); themeDescriptions.add(new ThemeDescription(chatListView, 0, new Class[]{ChatMessageCell.class}, null, new Drawable[]{msgOutDrawable, msgOutMediaDrawable}, null, Theme.key_chat_outBubbleGradient3)); @@ -32540,7 +32591,7 @@ public ArrayList getThemeDescriptions() { themeDescriptions.add(new ThemeDescription(chatListView, 0, new Class[]{ChatMessageCell.class}, null, new Drawable[]{Theme.chat_psaHelpDrawable[0]}, null, Theme.key_chat_inViews)); themeDescriptions.add(new ThemeDescription(chatListView, 0, new Class[]{ChatMessageCell.class}, null, new Drawable[]{Theme.chat_psaHelpDrawable[1]}, null, Theme.key_chat_outViews)); - if (!themeDelegate.isThemeChangeAvailable()) { + if (!themeDelegate.isThemeChangeAvailable(false)) { themeDescriptions.add(new ThemeDescription(messagesSearchListView, 0, new Class[]{DialogCell.class}, null, Theme.avatarDrawables, null, Theme.key_avatar_text)); themeDescriptions.add(new ThemeDescription(messagesSearchListView, 0, new Class[]{DialogCell.class}, Theme.dialogs_countPaint, null, null, Theme.key_chats_unreadCounter)); themeDescriptions.add(new ThemeDescription(messagesSearchListView, 0, new Class[]{DialogCell.class}, null, new Paint[]{Theme.dialogs_namePaint[0], Theme.dialogs_namePaint[1], Theme.dialogs_searchNamePaint}, null, null, Theme.key_chats_name)); @@ -33030,6 +33081,10 @@ private void setTransitionToChatProgress(float p) { } private void showChatThemeBottomSheet() { + if (currentChat != null) { + presentFragment(new ChannelColorActivity(getDialogId()).setOnApplied(ChatActivity.this)); + return; + } chatThemeBottomSheet = new ChatThemeBottomSheet(ChatActivity.this, themeDelegate); chatListView.setOnInterceptTouchListener(event -> true); setChildrenEnabled(contentView, false); @@ -33059,9 +33114,9 @@ private void checkThemeEmoticonOrWallpaper() { if (userInfo != null) { emoticon = userInfo.theme_emoticon; } - if (emoticon == null && chatInfo != null) { - emoticon = chatInfo.theme_emoticon; - } +// if (emoticon == null && chatInfo != null) { +// emoticon = chatInfo.theme_emoticon; +// } setChatThemeEmoticon(emoticon); }); } @@ -33077,13 +33132,7 @@ private void setChatThemeEmoticon(final String emoticon) { themeDelegate.setCurrentTheme(result, themeDelegate.wallpaper,openAnimationStartTime != 0, null); }); } - TLRPC.WallPaper wallPaper = null; - if (dialog_id >= 0) { - TLRPC.UserFull userFull = getMessagesController().getUserFull(dialog_id); - if (userFull != null) { - wallPaper = userFull.wallpaper; - } - } + TLRPC.WallPaper wallPaper = chatThemeController.getDialogWallpaper(dialog_id); themeDelegate.setCurrentTheme(themeDelegate.chatTheme, wallPaper, openAnimationStartTime != 0, null); } @@ -33143,7 +33192,7 @@ public class ThemeDelegate implements Theme.ResourcesProvider, ChatActionCell.Th ThemeDelegate() { isDark = Theme.getActiveTheme().isDark(); boolean setup = false; - if (isThemeChangeAvailable()) { + if (isThemeChangeAvailable(false)) { chatTheme = ChatThemeController.getInstance(currentAccount).getDialogTheme(dialog_id); wallpaper = ChatThemeController.getInstance(currentAccount).getDialogWallpaper(dialog_id); if (chatTheme != null || wallpaper != null) { @@ -33257,8 +33306,11 @@ public Paint getPaint(String paintKey) { return chatTheme != null || backgroundDrawable != null ? currentPaints.get(paintKey) : null; } - public boolean isThemeChangeAvailable() { - return currentChat == null && currentEncryptedChat == null && !currentUser.bot && dialog_id >= 0; + public boolean isThemeChangeAvailable(boolean canEdit) { + return currentEncryptedChat == null && ( + (!canEdit /*|| currentChat != null && ChatObject.isChannelAndNotMegaGroup(currentChat) && ChatObject.canChangeChatInfo(currentChat)*/) || + currentChat == null && currentUser != null && !currentUser.bot + ); } public EmojiThemes getCurrentTheme() { @@ -33287,7 +33339,7 @@ public void setCurrentTheme(final EmojiThemes chatTheme, TLRPC.WallPaper newWall String newEmoticon = chatTheme != null ? chatTheme.getEmoticon() : null; String oldEmoticon = this.chatTheme != null ? this.chatTheme.getEmoticon() : null; TLRPC.WallPaper oldWallpaper = this.wallpaper; - if (!force && (!isThemeChangeAvailable() || (TextUtils.equals(oldEmoticon, newEmoticon) && this.isDark == newIsDark && ChatThemeController.equals(newWallpaper, oldWallpaper)))) { + if (!force && (!isThemeChangeAvailable(false) || (TextUtils.equals(oldEmoticon, newEmoticon) && this.isDark == newIsDark && ChatThemeController.equals(newWallpaper, oldWallpaper)))) { return; } @@ -33323,6 +33375,8 @@ public void setCurrentTheme(final EmojiThemes chatTheme, TLRPC.WallPaper newWall } animationSettings.applyTheme = false; + if (dialog_id < 0) + animationSettings.applyTrulyTheme = false; animationSettings.afterStartDescriptionsAddedRunnable = () -> { setupChatTheme(chatTheme, newWallpaper, animated, true); initServiceMessageColors(backgroundDrawable); @@ -33422,7 +33476,9 @@ private void setupChatTheme(EmojiThemes chatTheme, TLRPC.WallPaper wallPaper, bo } else { currentColors = chatTheme.createColors(currentAccount, isDark ? 1 : 0); } - if (wallPaper != null) { + if (!TextUtils.isEmpty(ChatThemeController.getWallpaperEmoticon(wallpaper))) { + backgroundDrawable = PreviewView.getBackgroundDrawable(backgroundDrawable, currentAccount, wallpaper, isDark); + } else if (wallPaper != null) { backgroundDrawable = ChatBackgroundDrawable.getOrCreate(backgroundDrawable, wallPaper, isDark); } else { backgroundDrawable = getBackgroundDrawableFromTheme(chatTheme, prevPhase); @@ -33464,7 +33520,7 @@ public void onAnimationEnd(Animator animation) { patternAlphaAnimator.start(); } - if (chatTheme == null) { + if (chatTheme == null && dialog_id >= 0) { Theme.ThemeInfo activeTheme; if (Theme.getActiveTheme().isDark() == isDark) { activeTheme = Theme.getActiveTheme(); @@ -34023,4 +34079,28 @@ private void checkLeaveChannelButton() { } } } + + public boolean supportsThanosEffect() { + return ThanosEffect.supports() && LiteMode.isEnabled(LiteMode.FLAG_CHAT_THANOS); + } + + public ThanosEffect getChatThanosEffect() { + if (!LiteMode.isEnabled(LiteMode.FLAG_CHAT_THANOS) || !ThanosEffect.supports()) { + return null; + } + if (chatListThanosEffect == null) { + if (getContext() == null || !ThanosEffect.supports() || chatListView == null || contentView == null) { + return null; + } + chatListThanosEffect = new ThanosEffect(getContext(), () -> { + ThanosEffect thisThanosEffect = chatListThanosEffect; + if (thisThanosEffect != null) { + chatListThanosEffect = null; + contentView.removeView(thisThanosEffect); + } + }); + contentView.addView(chatListThanosEffect, 1 + contentView.indexOfChild(chatListView), LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT)); + } + return chatListThanosEffect; + } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/ChatBackgroundDrawable.java b/TMessagesProj/src/main/java/org/telegram/ui/ChatBackgroundDrawable.java index fff49d0454a..3db6c721e33 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/ChatBackgroundDrawable.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/ChatBackgroundDrawable.java @@ -154,7 +154,7 @@ public static Drawable createThumb(TLRPC.WallPaper wallPaper) { } } } else { - if (wallPaper.settings.intensity < 0) { + if (wallPaper.settings == null || wallPaper.settings.intensity < 0) { thumb = bitmapDrawableOf(new ColorDrawable(Color.BLACK)); } else { if (wallPaper.settings.second_background_color == 0) { //one color diff --git a/TMessagesProj/src/main/java/org/telegram/ui/ChatEditActivity.java b/TMessagesProj/src/main/java/org/telegram/ui/ChatEditActivity.java index 45d4d12d168..9e59365262a 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/ChatEditActivity.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/ChatEditActivity.java @@ -45,6 +45,7 @@ import androidx.annotation.NonNull; import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.ChannelBoostsController; import org.telegram.messenger.ChatObject; import org.telegram.messenger.ContactsController; import org.telegram.messenger.Emoji; @@ -888,11 +889,15 @@ public void afterTextChanged(Editable editable) { } if (ChatObject.isChannelAndNotMegaGroup(currentChat) && ChatObject.canChangeChatInfo(currentChat)) { - colorCell = new PeerColorActivity.ChangeNameColorCell(currentAccount, true, context, getResourceProvider()); + colorCell = new PeerColorActivity.ChangeNameColorCell(currentAccount, -currentChat.id, context, getResourceProvider()); colorCell.setBackgroundDrawable(Theme.getSelectorDrawable(true)); typeEditContainer.addView(colorCell, LayoutHelper.createLinear(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT)); colorCell.setOnClickListener(v -> { - presentFragment(new PeerColorActivity(-currentChat.id).setOnApplied(this)); + presentFragment(new ChannelColorActivity(-currentChat.id).setOnApplied(this)); + + MessagesController.getInstance(currentAccount).getMainSettings().edit().putInt("boostingappearance", + MessagesController.getInstance(currentAccount).getMainSettings().getInt("boostingappearance", 0) + 1 + ).apply(); }); } @@ -1925,7 +1930,7 @@ private void updateFields(boolean updateChat, boolean animated) { } } - private void updateColorCell() { + public void updateColorCell() { if (colorCell != null) { colorCell.set(currentChat, (historyCell != null && historyCell.getVisibility() == View.VISIBLE) || (signCell != null && signCell.getVisibility() == View.VISIBLE) || (forumsCell != null && forumsCell.getVisibility() == View.VISIBLE)); } @@ -2034,11 +2039,7 @@ private void updateReactionsCell(boolean animated) { } else { finalString = LocaleController.getString("ReactionsAll", R.string.ReactionsAll); } - if (isChannelAndNotMegaGroup) { - reactionsCell.setTextAndValueAndIcon(TextCell.applyNewSpan(LocaleController.getString("Reactions", R.string.Reactions)), finalString, animated, R.drawable.msg_reactions2, true); - } else { - reactionsCell.setTextAndValueAndIcon(LocaleController.getString("Reactions", R.string.Reactions), finalString, animated, R.drawable.msg_reactions2, true); - } + reactionsCell.setTextAndValueAndIcon(LocaleController.getString("Reactions", R.string.Reactions), finalString, animated, R.drawable.msg_reactions2, true); } @Override diff --git a/TMessagesProj/src/main/java/org/telegram/ui/ChatEditTypeActivity.java b/TMessagesProj/src/main/java/org/telegram/ui/ChatEditTypeActivity.java index 15b931a2175..478761aa79c 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/ChatEditTypeActivity.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/ChatEditTypeActivity.java @@ -263,6 +263,9 @@ public void onItemClick(int id) { if (id == -1) { finishFragment(); } else if (id == done_button) { + if (doneButtonDrawable != null && doneButtonDrawable.getProgress() > 0) { + return; + } processDone(); } } @@ -683,10 +686,7 @@ public void setInfo(TLRPC.ChatFull chatFull) { private void processDone() { AndroidUtilities.runOnUIThread(enableDoneLoading, 200); - if (currentChat.noforwards != isSaveRestricted) { - getMessagesController().toggleChatNoForwards(chatId, currentChat.noforwards = isSaveRestricted); - } - if (trySetUsername() && tryUpdateJoinSettings()) { + if (trySetUsername() && trySetRestrict() && tryUpdateJoinSettings()) { finishFragment(); } } @@ -1117,6 +1117,26 @@ protected void dispatchDraw(Canvas canvas) { } } + private boolean trySetRestrict() { + if (currentChat.noforwards != isSaveRestricted) { + if (!ChatObject.isChannel(currentChat)) { + updateDoneProgress(true); + getMessagesController().convertToMegaGroup(getParentActivity(), chatId, this, param -> { + if (param != 0) { + chatId = param; + currentChat = getMessagesController().getChat(param); + getMessagesController().toggleChatNoForwards(chatId, currentChat.noforwards = isSaveRestricted); + processDone(); + } + }); + return false; + } else { + getMessagesController().toggleChatNoForwards(chatId, currentChat.noforwards = isSaveRestricted); + } + } + return true; + } + private boolean trySetUsername() { if (getParentActivity() == null) { return false; @@ -1165,7 +1185,7 @@ private boolean trySetUsername() { private boolean deactivatingLinks = false; private boolean tryDeactivateAllLinks() { - if (!isPrivate || currentChat.usernames == null) { + if (!isPrivate || currentChat.usernames == null || currentChat.usernames.isEmpty()) { return true; } if (deactivatingLinks) { @@ -1182,7 +1202,7 @@ private boolean tryDeactivateAllLinks() { if (hasActive) { TLRPC.TL_channels_deactivateAllUsernames req = new TLRPC.TL_channels_deactivateAllUsernames(); req.channel = MessagesController.getInputChannel(currentChat); - getConnectionsManager().sendRequest(req, (res, err) -> { + getConnectionsManager().sendRequest(req, (res, err) -> AndroidUtilities.runOnUIThread(() -> { if (res instanceof TLRPC.TL_boolTrue) { for (int i = 0; i < currentChat.usernames.size(); ++i) { final TLRPC.TL_username username = currentChat.usernames.get(i); @@ -1193,7 +1213,9 @@ private boolean tryDeactivateAllLinks() { } deactivatingLinks = false; AndroidUtilities.runOnUIThread(this::processDone); - }); + })); + } else { + deactivatingLinks = false; } return !hasActive; } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/AlertsCreator.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/AlertsCreator.java index 3a8c648c956..5b7d35667db 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/AlertsCreator.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/AlertsCreator.java @@ -5768,28 +5768,32 @@ public static void createDeleteMessagesAlert(BaseFragment fragment, TLRPC.User u } } - boolean isGiveawayAndOwner = false; + boolean isActiveGiveawayAndOwner = false; String giveawayEndDate = null; if (selectedMessage != null) { - isGiveawayAndOwner = selectedMessage.isGiveaway() && !selectedMessage.isForwarded(); - if (isGiveawayAndOwner) { + isActiveGiveawayAndOwner = selectedMessage.isGiveaway() && !selectedMessage.isForwarded(); + if (isActiveGiveawayAndOwner) { TLRPC.TL_messageMediaGiveaway giveaway = (TLRPC.TL_messageMediaGiveaway) selectedMessage.messageOwner.media; - giveawayEndDate = LocaleController.getInstance().formatterGiveawayMonthDayYear.format(new Date(giveaway.until_date * 1000L)); + long untilDate = giveaway.until_date * 1000L; + giveawayEndDate = LocaleController.getInstance().formatterGiveawayMonthDayYear.format(new Date(untilDate)); + isActiveGiveawayAndOwner = System.currentTimeMillis() < untilDate; } } else if (count == 1) { for (int a = 1; a >= 0; a--) { for (int b = 0; b < selectedMessages[a].size(); b++) { MessageObject msg = selectedMessages[a].valueAt(b); - isGiveawayAndOwner = msg.isGiveaway() && !msg.isForwarded(); - if (isGiveawayAndOwner) { + isActiveGiveawayAndOwner = msg.isGiveaway() && !msg.isForwarded(); + if (isActiveGiveawayAndOwner) { TLRPC.TL_messageMediaGiveaway giveaway = (TLRPC.TL_messageMediaGiveaway) msg.messageOwner.media; - giveawayEndDate = LocaleController.getInstance().formatterGiveawayMonthDayYear.format(new Date(giveaway.until_date * 1000L)); + long untilDate = giveaway.until_date * 1000L; + giveawayEndDate = LocaleController.getInstance().formatterGiveawayMonthDayYear.format(new Date(untilDate)); + isActiveGiveawayAndOwner = System.currentTimeMillis() < untilDate; } } } } - if (isGiveawayAndOwner) { + if (isActiveGiveawayAndOwner) { builder.setTitle(LocaleController.getString("BoostingGiveawayDeleteMsgTitle", R.string.BoostingGiveawayDeleteMsgTitle)); builder.setMessage(AndroidUtilities.replaceTags(LocaleController.formatString("BoostingGiveawayDeleteMsgText", R.string.BoostingGiveawayDeleteMsgText, giveawayEndDate))); builder.setNeutralButton(LocaleController.getString("Delete", R.string.Delete), deleteAction); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/AnimatedEmojiDrawable.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/AnimatedEmojiDrawable.java index 145ea376287..bf959b9cfed 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/AnimatedEmojiDrawable.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/AnimatedEmojiDrawable.java @@ -71,6 +71,7 @@ public class AnimatedEmojiDrawable extends Drawable { public static final int CACHE_TYPE_AVATAR_CONSTRUCTOR_PREVIEW = 14; public static final int CACHE_TYPE_AVATAR_CONSTRUCTOR_PREVIEW2 = 15; public static final int CACHE_TYPE_ALERT_PREVIEW_STATIC_WITH_THUMB = 16; + public static final int CACHE_TYPE_EMOJI_CALL = 17; public int rawDrawIndex; @@ -463,7 +464,7 @@ private void updateSize() { sizedp = (int) ((Math.abs(Theme.chat_msgTextPaintEmoji[2].ascent()) + Math.abs(Theme.chat_msgTextPaintEmoji[2].descent())) * 1.15f / AndroidUtilities.density); } else if (this.cacheType == STANDARD_LOTTIE_FRAME) { sizedp = (int) ((Math.abs(Theme.chat_msgTextPaintEmoji[0].ascent()) + Math.abs(Theme.chat_msgTextPaintEmoji[0].descent())) * 1.15f / AndroidUtilities.density); - } else if (cacheType == CACHE_TYPE_AVATAR_CONSTRUCTOR_PREVIEW || cacheType == CACHE_TYPE_AVATAR_CONSTRUCTOR_PREVIEW2) { + } else if (cacheType == CACHE_TYPE_AVATAR_CONSTRUCTOR_PREVIEW || cacheType == CACHE_TYPE_AVATAR_CONSTRUCTOR_PREVIEW2 || cacheType == CACHE_TYPE_EMOJI_CALL) { sizedp = 100; } else { sizedp = 34; @@ -528,10 +529,10 @@ protected boolean setImageBitmapByKey(Drawable drawable, String key, int type, b if (cacheType == CACHE_TYPE_RENDERING_VIDEO) { filter += "_d_nostream"; } - if (cacheType != CACHE_TYPE_AVATAR_CONSTRUCTOR_PREVIEW2 && cacheType != CACHE_TYPE_AVATAR_CONSTRUCTOR_PREVIEW && cacheType != STANDARD_LOTTIE_FRAME && (cacheType != CACHE_TYPE_MESSAGES_LARGE || SharedConfig.getDevicePerformanceClass() < SharedConfig.PERFORMANCE_CLASS_HIGH) && cacheType != CACHE_TYPE_RENDERING_VIDEO) { + if (cacheType != CACHE_TYPE_EMOJI_CALL && cacheType != CACHE_TYPE_AVATAR_CONSTRUCTOR_PREVIEW2 && cacheType != CACHE_TYPE_AVATAR_CONSTRUCTOR_PREVIEW && cacheType != STANDARD_LOTTIE_FRAME && (cacheType != CACHE_TYPE_MESSAGES_LARGE || SharedConfig.getDevicePerformanceClass() < SharedConfig.PERFORMANCE_CLASS_HIGH) && cacheType != CACHE_TYPE_RENDERING_VIDEO) { filter += "_pcache"; } - if (cacheType != CACHE_TYPE_MESSAGES && cacheType != CACHE_TYPE_MESSAGES_LARGE && cacheType != CACHE_TYPE_AVATAR_CONSTRUCTOR_PREVIEW && cacheType != CACHE_TYPE_AVATAR_CONSTRUCTOR_PREVIEW2) { + if (cacheType != CACHE_TYPE_EMOJI_CALL && cacheType != CACHE_TYPE_MESSAGES && cacheType != CACHE_TYPE_MESSAGES_LARGE && cacheType != CACHE_TYPE_AVATAR_CONSTRUCTOR_PREVIEW && cacheType != CACHE_TYPE_AVATAR_CONSTRUCTOR_PREVIEW2) { filter += "_compress"; } if (cacheType == STANDARD_LOTTIE_FRAME) { @@ -585,7 +586,11 @@ protected boolean setImageBitmapByKey(Drawable drawable, String key, int type, b imageReceiver.setImage(ImageLocation.getForDocument(thumb, document), sizedp + "_" + sizedp, thumbLocation, null, thumbDrawable, document.size, null, document, 1); } } else { - imageReceiver.setImage(mediaLocation, mediaFilter, ImageLocation.getForDocument(thumb, document), sizedp + "_" + sizedp, null, null, thumbDrawable, document.size, null, document, 1); + ImageLocation thumbLocation = null; + if (cacheType == CACHE_TYPE_EMOJI_CALL) { + thumbLocation = ImageLocation.getForDocument(thumb, document); + } + imageReceiver.setImage(mediaLocation, mediaFilter, ImageLocation.getForDocument(thumb, document), sizedp + "_" + sizedp, thumbLocation, null, thumbDrawable, document.size, null, document, 1); } } @@ -622,6 +627,8 @@ private void updateAutoRepeat(ImageReceiver imageReceiver) { imageReceiver.setAutoRepeatCount(2); } else if (cacheType == CACHE_TYPE_FORUM_TOPIC_LARGE || cacheType == CACHE_TYPE_AVATAR_CONSTRUCTOR_PREVIEW || cacheType == CACHE_TYPE_TAB_STRIP || cacheType == CACHE_TYPE_ALERT_PREVIEW_TAB_STRIP) { imageReceiver.setAutoRepeatCount(1); + } else if (cacheType == CACHE_TYPE_EMOJI_CALL){ + imageReceiver.setAutoRepeatCount(0); } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/BitmapShaderTools.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/BitmapShaderTools.java index 4ee2471cafa..b2aec0db8b1 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/BitmapShaderTools.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/BitmapShaderTools.java @@ -30,6 +30,13 @@ public BitmapShaderTools() { updateBounds(); } + public BitmapShaderTools(int width, int height) { + bitmap = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888); + canvas = new Canvas(bitmap); + paint.setShader(shader = new BitmapShader(bitmap, Shader.TileMode.CLAMP, Shader.TileMode.CLAMP)); + updateBounds(); + } + public Bitmap getBitmap() { return bitmap; } @@ -64,4 +71,12 @@ public void setBounds(float left, float top, float right, float bottom) { AndroidUtilities.rectTmp.set(left, top, right, bottom); setBounds(AndroidUtilities.rectTmp); } + + public void setMatrix(float offsetX, float offsetY, float scale, float degrees) { + matrix.reset(); + matrix.postRotate(degrees, bitmap.getWidth() / 2f, bitmap.getHeight() / 2f); + matrix.postScale(scale, scale); + matrix.postTranslate(offsetX, offsetY); + shader.setLocalMatrix(matrix); + } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/BlobDrawable.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/BlobDrawable.java index c05dcaf773f..d2519fab2b7 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/BlobDrawable.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/BlobDrawable.java @@ -40,26 +40,26 @@ public class BlobDrawable { private Path path = new Path(); public Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG); - private float[] radius; - private float[] angle; - private float[] radiusNext; - private float[] angleNext; - private float[] progress; - private float[] speed; + protected float[] radius; + protected float[] angle; + protected float[] radiusNext; + protected float[] angleNext; + protected float[] progress; + protected float[] speed; private float[] pointStart = new float[4]; private float[] pointEnd = new float[4]; - final Random random = new Random(); + protected final Random random = new Random(); - private final float N; + protected final float N; private final float L; public float cubicBezierK = 1f; private final Matrix m = new Matrix(); - private final int liteFlag; + protected final int liteFlag; public BlobDrawable(int n) { this(n, LiteMode.FLAG_CALLS_ANIMATIONS); @@ -85,7 +85,7 @@ public BlobDrawable(int n, int liteFlag) { this.liteFlag = liteFlag; } - private void generateBlob(float[] radius, float[] angle, int i) { + protected void generateBlob(float[] radius, float[] angle, int i) { float angleDif = 360f / N * 0.05f; float radDif = maxRadius - minRadius; radius[i] = minRadius + Math.abs(((random.nextInt() % 100f) / 100f)) * radDif; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/BlurringShader.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/BlurringShader.java index 4dd5fa3b165..a086e0d95dd 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/BlurringShader.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/BlurringShader.java @@ -555,7 +555,10 @@ private void invalidateFallbackBlur() { private int i = 0; public void setFallbackBlur(Bitmap bitmap, int orientation) { - fallbackBitmap = thumbBlurer.getBitmap(bitmap, "" + i++, orientation, 0); + setFallbackBlur(bitmap, orientation, false); + } + public void setFallbackBlur(Bitmap bitmap, int orientation, boolean recycleAfter) { + fallbackBitmap = thumbBlurer.getBitmap(bitmap, "" + i++, orientation, 0, recycleAfter); } public void resetBitmap() { @@ -603,7 +606,7 @@ public void destroy() { thumbBitmap = null; } - public Bitmap getBitmap(Bitmap bitmap, String key, int orientation, int invert) { + public Bitmap getBitmap(Bitmap bitmap, String key, int orientation, int invert, boolean recycleAfter) { if (bitmap == null) { return null; } @@ -666,6 +669,10 @@ public Bitmap getBitmap(Bitmap bitmap, String key, int orientation, int invert) } else { resultBitmap.recycle(); } + + if (recycleAfter) { + bitmap.recycle(); + } }); }); return thumbBitmap; @@ -675,14 +682,14 @@ public Bitmap getBitmap(ImageReceiver imageReceiver) { if (imageReceiver == null) { return null; } - return getBitmap(imageReceiver.getBitmap(), imageReceiver.getImageKey(), imageReceiver.getOrientation(), imageReceiver.getInvert()); + return getBitmap(imageReceiver.getBitmap(), imageReceiver.getImageKey(), imageReceiver.getOrientation(), imageReceiver.getInvert(), false); } public Bitmap getBitmap(ImageReceiver.BitmapHolder bitmapHolder) { if (bitmapHolder == null) { return null; } - return getBitmap(bitmapHolder.bitmap, bitmapHolder.getKey(), bitmapHolder.orientation, 0); + return getBitmap(bitmapHolder.bitmap, bitmapHolder.getKey(), bitmapHolder.orientation, 0, false); } } @@ -698,6 +705,7 @@ public static class StoryBlurDrawer { public static final int BLUR_TYPE_EMOJI_VIEW = 7; public static final int BLUR_TYPE_REPLY_BACKGROUND = 8; public static final int BLUR_TYPE_REPLY_TEXT_XFER = 9; + public static final int BLUR_TYPE_ACTION_BACKGROUND = 10; private final BlurManager manager; private final View view; @@ -758,6 +766,10 @@ public StoryBlurDrawer(@Nullable BlurManager manager, @NonNull View view, int ty AndroidUtilities.adjustBrightnessColorMatrix(colorMatrix, +.4f); AndroidUtilities.adjustSaturationColorMatrix(colorMatrix, +.45f); // AndroidUtilities.multiplyBrightnessColorMatrix(colorMatrix, 1.4f); + } else if (type == BLUR_TYPE_ACTION_BACKGROUND) { + colorMatrix.setSaturation(1.6f); + AndroidUtilities.multiplyBrightnessColorMatrix(colorMatrix, wasDark ? .97f : .92f); + AndroidUtilities.adjustBrightnessColorMatrix(colorMatrix, wasDark ? +.12f : -.06f); } paint.setColorFilter(new ColorMatrixColorFilter(colorMatrix)); oldPaint.setColorFilter(new ColorMatrixColorFilter(colorMatrix)); @@ -829,6 +841,22 @@ public void setBounds(RectF bounds) { updateBounds(); } + private boolean wasDark = false; + public StoryBlurDrawer adapt(boolean isDark) { + if (wasDark != isDark) { + wasDark = isDark; + if (type == BLUR_TYPE_ACTION_BACKGROUND) { + final ColorMatrix colorMatrix = new ColorMatrix(); + colorMatrix.setSaturation(1.6f); + AndroidUtilities.multiplyBrightnessColorMatrix(colorMatrix, wasDark ? .97f : .92f); + AndroidUtilities.adjustBrightnessColorMatrix(colorMatrix, wasDark ? +.12f : -.06f); + paint.setColorFilter(new ColorMatrixColorFilter(colorMatrix)); + oldPaint.setColorFilter(new ColorMatrixColorFilter(colorMatrix)); + } + } + return this; + } + @Nullable public Paint getPaint(float alpha) { return getPaint(alpha, 0, 0); @@ -913,6 +941,7 @@ private boolean setupMatrix(int bitmapWidth, int bitmapHeight) { View view = this.view; do { matrix.preScale(1f / view.getScaleX(), 1f / view.getScaleY(), view.getPivotX(), view.getPivotY()); + matrix.preRotate(-view.getRotation(), view.getPivotX(), view.getPivotY()); matrix.preTranslate(-view.getX(), -view.getY()); if (view.getParent() instanceof View) { view = (View) view.getParent(); @@ -930,6 +959,7 @@ private boolean setupMatrix(int bitmapWidth, int bitmapHeight) { } matrix.postTranslate(child.getX(), child.getY()); matrix.postScale(1f / child.getScaleX(), 1f / child.getScaleY(), child.getPivotX(), child.getPivotY()); + matrix.postRotate(child.getRotation(), child.getPivotX(), child.getPivotY()); index++; } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/BotWebViewSheet.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/BotWebViewSheet.java index 71b51360d70..6c01912e155 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/BotWebViewSheet.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/BotWebViewSheet.java @@ -1200,7 +1200,11 @@ public void dismiss(Runnable callback) { NotificationCenter.getGlobalInstance().removeObserver(this, NotificationCenter.didSetNewTheme); swipeContainer.stickTo(swipeContainer.getHeight() + frameLayout.measureKeyboardHeight(), ()->{ - super.dismiss(); + try { + super.dismiss(); + } catch (Exception e) { + FileLog.e(e); + } if (callback != null) { callback.run(); } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/BottomSheetWithRecyclerListView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/BottomSheetWithRecyclerListView.java index eb1e8d095d9..9737c16992f 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/BottomSheetWithRecyclerListView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/BottomSheetWithRecyclerListView.java @@ -251,6 +251,9 @@ private void postDrawInternal(Canvas canvas, View parentView) { headerShadowDrawable.setBounds(backgroundPaddingLeft, actionBar.getBottom(), parentView.getMeasuredWidth() - backgroundPaddingLeft, actionBar.getBottom() + headerShadowDrawable.getIntrinsicHeight()); headerShadowDrawable.setAlpha((int) (255 * actionBar.getAlpha() * shadowAlpha)); headerShadowDrawable.draw(canvas); + if (headerShadowDrawable.getAlpha() < 255) { + parentView.invalidate(); + } } wasDrawn = true; } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Bulletin.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Bulletin.java index d5a99c86524..ac623d3dd80 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Bulletin.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Bulletin.java @@ -674,7 +674,11 @@ public Layout(@NonNull Context context, Theme.ResourcesProvider resourcesProvide } protected void setBackground(int color) { - background = Theme.createRoundRectDrawable(AndroidUtilities.dp(10), color); + setBackground(color, 10); + } + + public void setBackground(int color, int rounding) { + background = Theme.createRoundRectDrawable(AndroidUtilities.dp(rounding), color); } public final static FloatPropertyCompat IN_OUT_OFFSET_Y = new FloatPropertyCompat("offsetY") { @@ -1085,9 +1089,17 @@ public ButtonLayout(@NonNull Context context, Theme.ResourcesProvider resourcesP this.resourcesProvider = resourcesProvider; } + private boolean wrapWidth; + public void setWrapWidth() { + wrapWidth = true; + } + @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { childrenMeasuredWidth = 0; + if (wrapWidth) { + widthMeasureSpec = MeasureSpec.makeMeasureSpec(MeasureSpec.getSize(widthMeasureSpec), MeasureSpec.AT_MOST); + } super.onMeasure(widthMeasureSpec, heightMeasureSpec); if (button != null && MeasureSpec.getMode(widthMeasureSpec) == MeasureSpec.AT_MOST) { setMeasuredDimension(childrenMeasuredWidth + button.getMeasuredWidth(), getMeasuredHeight()); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/BulletinFactory.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/BulletinFactory.java index 9d003eb0127..59673d5dedc 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/BulletinFactory.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/BulletinFactory.java @@ -41,6 +41,7 @@ import org.telegram.ui.ActionBar.Theme; import org.telegram.ui.LaunchActivity; import org.telegram.ui.PremiumPreviewFragment; +import org.telegram.ui.Stories.recorder.HintView2; import java.util.ArrayList; import java.util.List; @@ -189,6 +190,20 @@ public Bulletin createSimpleBulletinWithIconSize(int iconRawId, CharSequence tex return create(layout, text.length() < 20 ? Bulletin.DURATION_SHORT : Bulletin.DURATION_LONG); } + public Bulletin createImageBulletin(int iconRawId, CharSequence title) { + final Bulletin.LottieLayout layout = new Bulletin.LottieLayout(getContext(), resourcesProvider); + layout.setBackground(Theme.getColor(Theme.key_undo_background, resourcesProvider), 12); + layout.imageView.setImageResource(iconRawId); + layout.textView.setText(title); + layout.textView.setSingleLine(false); + layout.textView.setLines(2); + layout.textView.setMaxLines(4); + layout.textView.setMaxWidth(HintView2.cutInFancyHalf(layout.textView.getText(), layout.textView.getPaint())); + ((ViewGroup.MarginLayoutParams) layout.textView.getLayoutParams()).rightMargin = AndroidUtilities.dp(12); + layout.setWrapWidth(); + return create(layout, Bulletin.DURATION_PROLONG); + } + public Bulletin createSimpleLargeBulletin(int iconRawId, CharSequence title, CharSequence subtitle) { final Bulletin.TwoLineLayout layout = new Bulletin.TwoLineLayout(getContext(), resourcesProvider); layout.imageView.setImageResource(iconRawId); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/CaptionPhotoViewer.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/CaptionPhotoViewer.java index 5c5bf8b02d6..41276e7ed0d 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/CaptionPhotoViewer.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/CaptionPhotoViewer.java @@ -303,7 +303,7 @@ protected int additionalKeyboardHeight() { @Override public void updateColors(Theme.ResourcesProvider resourcesProvider) { super.updateColors(resourcesProvider); - timerDrawable.updateColors(0xffffffff, Theme.getColor(Theme.key_chat_editMediaButton, resourcesProvider)); + timerDrawable.updateColors(0xffffffff, Theme.getColor(Theme.key_chat_editMediaButton, resourcesProvider), 0xffffffff); } @Override diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/ChatActivityEnterView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/ChatActivityEnterView.java index 3b886b8e63c..5c4a8bf6ed1 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/ChatActivityEnterView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/ChatActivityEnterView.java @@ -59,9 +59,7 @@ import android.text.TextPaint; import android.text.TextUtils; import android.text.TextWatcher; -import android.text.style.DynamicDrawableSpan; import android.text.style.ImageSpan; -import android.util.Log; import android.util.Property; import android.util.TypedValue; import android.view.ActionMode; @@ -115,7 +113,6 @@ import org.telegram.messenger.BotWebViewVibrationEffect; import org.telegram.messenger.BuildVars; import org.telegram.messenger.ChatObject; -import org.telegram.messenger.CodeHighlighting; import org.telegram.messenger.ContactsController; import org.telegram.messenger.DialogObject; import org.telegram.messenger.Emoji; @@ -126,7 +123,6 @@ import org.telegram.messenger.MediaDataController; import org.telegram.messenger.MessageObject; import org.telegram.messenger.MessagesController; -import org.telegram.messenger.MessagesStorage; import org.telegram.messenger.NotificationCenter; import org.telegram.messenger.NotificationsController; import org.telegram.messenger.R; @@ -137,7 +133,6 @@ import org.telegram.messenger.UserObject; import org.telegram.messenger.Utilities; import org.telegram.messenger.VideoEditedInfo; -import org.telegram.messenger.XiaomiUtilities; import org.telegram.messenger.browser.Browser; import org.telegram.messenger.camera.CameraController; import org.telegram.tgnet.ConnectionsManager; @@ -161,11 +156,13 @@ import org.telegram.ui.DialogsActivity; import org.telegram.ui.GroupStickersActivity; import org.telegram.ui.LaunchActivity; +import org.telegram.ui.MultiContactsSelectorBottomSheet; import org.telegram.ui.PhotoViewer; import org.telegram.ui.PremiumPreviewFragment; import org.telegram.ui.ProfileActivity; import org.telegram.ui.StickersActivity; -import org.telegram.ui.TopicsFragment; +import org.telegram.ui.Stories.recorder.CaptionContainerView; +import org.telegram.ui.Stories.recorder.HintView2; import java.io.File; import java.io.FileOutputStream; @@ -199,6 +196,9 @@ public class ChatActivityEnterView extends BlurredFrameLayout implements Notific private boolean sendButtonEnabled = true; private TLRPC.UserFull userInfo; + public boolean voiceOnce; + public boolean onceVisible; + public void drawRecordedPannel(Canvas canvas) { if (getAlpha() == 0 || recordedAudioPanel == null || recordedAudioPanel.getParent() == null || recordedAudioPanel.getVisibility() != View.VISIBLE) { return; @@ -242,7 +242,7 @@ public interface ChatActivityEnterViewDelegate { void didPressAttachButton(); - void needStartRecordVideo(int state, boolean notify, int scheduleDate); + void needStartRecordVideo(int state, boolean notify, int scheduleDate, int ttl); void needChangeVideoPreviewState(int state, float seekProgress); @@ -323,6 +323,10 @@ default ChatActivity.ReplyQuote getReplyQuote() { default void onKeyboardRequested() { } + + default boolean onceVoiceAvailable() { + return false; + } } public final static int RECORD_STATE_ENTER = 0; @@ -527,6 +531,7 @@ public void onInitializeAccessibilityNodeInfo(View host, AccessibilityNodeInfo i private AnimatorSet scheduledButtonAnimation; @Nullable private RecordCircle recordCircle; + private OnceButton onceButton; private CloseProgressDrawable2 progressDrawable; private Paint dotPaint; private MediaActionDrawable playPauseDrawable; @@ -728,7 +733,7 @@ public void set(RecordCircle object, Float value) { @Override public void run() { if (delegate != null) { - delegate.needStartRecordVideo(0, true, 0); + delegate.needStartRecordVideo(0, true, 0, 0); } } }; @@ -855,6 +860,9 @@ public void updateColors() { drawable.beginApplyLayerColors(); drawable.setLayerColor("Cup Red.**", dotColor); drawable.setLayerColor("Box.**", dotColor); + drawable.setLayerColor("Line 1.**", background); + drawable.setLayerColor("Line 2.**", background); + drawable.setLayerColor("Line 3.**", background); drawable.commitApplyLayerColors(); if (playPauseDrawable != null) { playPauseDrawable.setColor(getThemedColor(Theme.key_chat_recordedVoicePlayPause)); @@ -1009,9 +1017,147 @@ public void set(View object, Float value) { } }; + public class OnceButton extends FrameLayout { + + private HintView2 hintView; + + private CaptionContainerView.PeriodDrawable periodDrawable; + private Paint lockBackgroundPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + + public OnceButton(Context context) { + super(context); + + periodDrawable = new CaptionContainerView.PeriodDrawable(); + periodDrawable.setCallback(this); + periodDrawable.setValue(1, voiceOnce, false); + + setWillNotDraw(false); + updateColors(); + } + + public void showHintView() { + hideHintView(); + hintView = new HintView2(getContext(), HintView2.DIRECTION_RIGHT); + hintView.setJoint(1, 0); + hintView.setMultilineText(true); + hintView.setText(AndroidUtilities.replaceTags(LocaleController.getString(voiceOnce ? R.string.VoiceSetOnceHintEnabled : R.string.VoiceSetOnceHint))); + hintView.setMaxWidthPx(HintView2.cutInFancyHalf(hintView.getText(), hintView.getTextPaint())); + if (voiceOnce) { + hintView.setIcon(R.raw.fire_on); + } else { + MessagesController.getGlobalMainSettings().edit().putInt("voiceoncehint", MessagesController.getGlobalMainSettings().getInt("voiceoncehint", 0) + 1).apply(); + } + addView(hintView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT, Gravity.FILL, 0, 0, 54, 58)); + final HintView2 thisHintView = hintView; + hintView.setOnHiddenListener(() -> { + removeView(thisHintView); + if (hintView == thisHintView) { + hintView = null; + } + }); + hintView.show(); + } + + public void hideHintView() { + if (hintView != null) { + HintView2 oldHintView = hintView; + oldHintView.setOnHiddenListener(() -> removeView(oldHintView)); + oldHintView.hide(); + hintView = null; + } + } + + @Override + protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { + super.onMeasure( + widthMeasureSpec, + MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(194 + 48 + 12), MeasureSpec.EXACTLY) + ); + } + + private final RectF rectF = new RectF(); + public final RectF onceRect = new RectF(); + + @Override + protected void onDraw(Canvas canvas) { + if (recordCircle == null) return; + + int cx = getMeasuredWidth() - AndroidUtilities.dp2(26); + + final float cy = AndroidUtilities.lerp(recordCircle.lockY + recordCircle.translationDy, getMeasuredHeight() - AndroidUtilities.dp(70 + 36), Math.max(recordCircle.exitTransition, Math.min(recordCircle.progressToSeekbarStep1, recordCircle.slideToCancelLockProgress))); + rectF.set(cx - AndroidUtilities.dpf2(18), cy, cx + AndroidUtilities.dpf2(18), cy + recordCircle.lockSize); + rectF.offset(0, getMeasuredHeight() - recordCircle.getMeasuredHeight()); + onceVisible = delegate != null && delegate.onceVoiceAvailable() && !isInVideoMode; + if (onceVisible) { + final float onceOffset = AndroidUtilities.dpf2(AndroidUtilities.lerp(4, 12, recordCircle.moveProgress)); + rectF.set( + rectF.left, rectF.top - AndroidUtilities.dpf2(36) - onceOffset, rectF.right, rectF.top - onceOffset + ); + if (hintView != null) { + hintView.setJointPx(0, rectF.centerY()); + hintView.invalidate(); + } + onceRect.set(rectF); + canvas.save(); + final float s = recordCircle.scale * (1f - recordCircle.exitTransition) * recordCircle.slideToCancelLockProgress * recordCircle.snapAnimationProgress; + canvas.scale(s, s, rectF.centerX(), rectF.centerY()); + lockShadowDrawable.setBounds( + (int) (rectF.left - AndroidUtilities.dpf2(3)), (int) (rectF.top - AndroidUtilities.dpf2(3)), + (int) (rectF.right + AndroidUtilities.dpf2(3)), (int) (rectF.bottom + AndroidUtilities.dpf2(3)) + ); + lockShadowDrawable.draw(canvas); + canvas.drawRoundRect(rectF, AndroidUtilities.dpf2(18), AndroidUtilities.dpf2(18), lockBackgroundPaint); + periodDrawable.setBounds((int) rectF.left, (int) rectF.top, (int) rectF.right, (int) rectF.bottom); + periodDrawable.draw(canvas); + canvas.restore(); + } + } + + public void updateColors() { + periodDrawable.updateColors( + getThemedColor(Theme.key_chat_messagePanelVoiceLock), + getThemedColor(Theme.key_chat_messagePanelVoiceBackground), + 0xFFFFFFFF + ); + lockBackgroundPaint.setColor(getThemedColor(Theme.key_chat_messagePanelVoiceLockBackground)); + } + + private boolean pressed; + @Override + public boolean onTouchEvent(MotionEvent event) { + if (onceVisible && (recordCircle != null && recordCircle.snapAnimationProgress > .1f)) { + int x = (int) event.getX(); + int y = (int) event.getY(); + if (event.getAction() == MotionEvent.ACTION_DOWN) { + return pressed = onceRect.contains(x, y); + } else if (pressed) { + if (event.getAction() == MotionEvent.ACTION_UP) { + if (onceRect.contains(x, y)) { + voiceOnce = !voiceOnce; + periodDrawable.setValue(1, voiceOnce, true); + if (voiceOnce) { + showHintView(); + } else { + hideHintView(); + } + invalidate(); + } + } + return true; + } + } + return false; + } + + @Override + protected boolean verifyDrawable(@NonNull Drawable who) { + return who == periodDrawable || super.verifyDrawable(who); + } + } + public class RecordCircle extends View { - private float scale; + public float scale; private float amplitude; private float animateToAmplitude; private float animateAmplitudeDiff; @@ -1023,7 +1169,7 @@ public class RecordCircle extends View { private boolean pressed; private float transformToSeekbar; private float exitTransition; - private float progressToSeekbarStep3; + public float progressToSeekbarStep3; private float progressToSendButton; public float iconScale; @@ -1057,8 +1203,9 @@ public class RecordCircle extends View { private int paintAlpha; private float touchSlop; - private float slideToCancelProgress; - private float slideToCancelLockProgress; + public float slideToCancelProgress; + public float slideToCancelLockProgress; + private float progressToSeekbarStep1, progressToSeekbarStep2; private int slideDelta; private boolean canceledByGesture; @@ -1225,9 +1372,9 @@ public boolean onTouchEvent(MotionEvent event) { if (event.getAction() == MotionEvent.ACTION_UP) { if (pauseRect.contains(x, y)) { if (isInVideoMode()) { - delegate.needStartRecordVideo(3, true, 0); + delegate.needStartRecordVideo(3, true, 0, voiceOnce ? 0x7FFFFFFF : 0); } else { - MediaController.getInstance().stopRecording(2, true, 0); + MediaController.getInstance().stopRecording(2, true, 0, voiceOnce); delegate.needStartRecordAudio(0); } if (slideText != null) { @@ -1270,6 +1417,10 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { slideDelta = (int) (-distance * (1f - slideToCancelProgress)); } + public float moveProgress; + public float lockY, lockSize; + public float translationDy; + @Override protected void onDraw(Canvas canvas) { if (skipDraw) { @@ -1364,6 +1515,8 @@ protected void onDraw(Canvas canvas) { circleAlpha = Math.max(0, 1f - (exitTransition - 0.6f) / 0.4f); } } + this.progressToSeekbarStep1 = progressToSeekbarStep1; + this.progressToSeekbarStep2 = progressToSeekbarStep2; if (canceledByGesture && slideToCancelProgress > 0.7f) { circleAlpha *= (1f - (slideToCancelProgress - 0.7f) / 0.3f); @@ -1396,7 +1549,7 @@ protected void onDraw(Canvas canvas) { replaceDrawable.setBounds(cx - replaceDrawable.getIntrinsicWidth() / 2, cy - replaceDrawable.getIntrinsicHeight() / 2, cx + replaceDrawable.getIntrinsicWidth() / 2, cy + replaceDrawable.getIntrinsicHeight() / 2); } - float moveProgress = 1.0f - yAdd / AndroidUtilities.dp(57); + float moveProgress = this.moveProgress = 1.0f - yAdd / AndroidUtilities.dp(57); float lockSize; float lockY; @@ -1511,8 +1664,8 @@ protected void onDraw(Canvas canvas) { } if (isSendButtonVisible()) { - lockSize = AndroidUtilities.dp(36); - lockY = AndroidUtilities.dp(60) + multilinTooltipOffset + AndroidUtilities.dpf2(30) * (1.0f - sc) - yAdd + AndroidUtilities.dpf2(14f) * moveProgress; + lockSize = this.lockSize = AndroidUtilities.dp(36); + lockY = this.lockY = AndroidUtilities.dp(60) + multilinTooltipOffset + AndroidUtilities.dpf2(30) * (1.0f - sc) - yAdd + AndroidUtilities.dpf2(14f) * moveProgress; lockMiddleY = lockY + lockSize / 2f - AndroidUtilities.dpf2(8) + AndroidUtilities.dpf2(2); lockTopY = lockY + lockSize / 2f - AndroidUtilities.dpf2(16) + AndroidUtilities.dpf2(2); @@ -1522,15 +1675,14 @@ protected void onDraw(Canvas canvas) { transformToPauseProgress = moveProgress; } else { - lockSize = AndroidUtilities.dp(36) + (int) (AndroidUtilities.dp(14) * moveProgress); - lockY = AndroidUtilities.dp(60) + multilinTooltipOffset + (int) (AndroidUtilities.dp(30) * (1.0f - sc)) - (int) yAdd + (moveProgress) * idleProgress * -AndroidUtilities.dp(8); + lockSize = this.lockSize = AndroidUtilities.dp(36) + (int) (AndroidUtilities.dp(14) * moveProgress); + lockY = this.lockY = AndroidUtilities.dp(60) + multilinTooltipOffset + (int) (AndroidUtilities.dp(30) * (1.0f - sc)) - (int) yAdd + (moveProgress) * idleProgress * -AndroidUtilities.dp(8); lockMiddleY = lockY + lockSize / 2f - AndroidUtilities.dpf2(8) + AndroidUtilities.dpf2(2) + AndroidUtilities.dpf2(2) * moveProgress; lockTopY = lockY + lockSize / 2f - AndroidUtilities.dpf2(16) + AndroidUtilities.dpf2(2) + AndroidUtilities.dpf2(2) * moveProgress; lockRotation = 9 * (1f - moveProgress); snapAnimationProgress = 0; } - if ((showTooltip && System.currentTimeMillis() - showTooltipStartTime > 200) || tooltipAlpha != 0f) { if (moveProgress < 0.8f || isSendButtonVisible() || exitTransition != 0 || transformToSeekbar != 0) { showTooltip = false; @@ -1623,10 +1775,15 @@ protected void onDraw(Canvas canvas) { } float maxTranslationDy = AndroidUtilities.dpf2(72); - float dy = maxTranslationDy * translation - AndroidUtilities.dpf2(20) * (progressToSeekbarStep1) * (1f - translation) + maxTranslationDy * (1f - slideToCancelLockProgress); + float dy = ( + maxTranslationDy * translation + - AndroidUtilities.dpf2(20) * (progressToSeekbarStep1) * (1f - translation) + + maxTranslationDy * (1f - slideToCancelLockProgress) + ); if (dy > maxTranslationDy) { dy = maxTranslationDy; } + this.translationDy = dy; canvas.translate(0, dy); float s = scale * (1f - progressToSeekbarStep2) * (1f - exitProgress2) * slideToCancelLockProgress; canvas.scale(s, s, cx, lockMiddleY); @@ -1689,6 +1846,14 @@ protected void onDraw(Canvas canvas) { drawingCircleRadius = radius; } + @Override + public void invalidate() { + super.invalidate(); + if (onceButton != null) { + onceButton.invalidate(); + } + } + public void drawIcon(Canvas canvas, int cx, int cy, float alpha) { Drawable drawable; Drawable replaceDrawable = null; @@ -2228,12 +2393,12 @@ public boolean onTouchEvent(MotionEvent motionEvent) { if (!hasRecordVideo || calledRecordRunnable) { startedDraggingX = -1; if (hasRecordVideo && isInVideoMode()) { - delegate.needStartRecordVideo(1, true, 0); + delegate.needStartRecordVideo(1, true, 0, voiceOnce ? 0x7FFFFFFF : 0); } else { if (recordingAudioVideo && isInScheduleMode()) { - AlertsCreator.createScheduleDatePickerDialog(parentActivity, parentFragment.getDialogId(), (notify, scheduleDate) -> MediaController.getInstance().stopRecording(1, notify, scheduleDate), () -> MediaController.getInstance().stopRecording(0, false, 0), resourcesProvider); + AlertsCreator.createScheduleDatePickerDialog(parentActivity, parentFragment.getDialogId(), (notify, scheduleDate) -> MediaController.getInstance().stopRecording(1, notify, scheduleDate, false), () -> MediaController.getInstance().stopRecording(0, false, 0, false), resourcesProvider); } - MediaController.getInstance().stopRecording(isInScheduleMode() ? 3 : 1, true, 0); + MediaController.getInstance().stopRecording(isInScheduleMode() ? 3 : 1, true, 0, voiceOnce); delegate.needStartRecordAudio(0); } recordingAudioVideo = false; @@ -2267,10 +2432,10 @@ public boolean onTouchEvent(MotionEvent motionEvent) { if (recordCircle.slideToCancelProgress < 0.7f) { if (hasRecordVideo && isInVideoMode()) { CameraController.getInstance().cancelOnInitRunnable(onFinishInitCameraRunnable); - delegate.needStartRecordVideo(2, true, 0); + delegate.needStartRecordVideo(2, true, 0, voiceOnce ? 0x7FFFFFFF : 0); } else { delegate.needStartRecordAudio(0); - MediaController.getInstance().stopRecording(0, false, 0); + MediaController.getInstance().stopRecording(0, false, 0, voiceOnce); } recordingAudioVideo = false; updateRecordInterface(RECORD_STATE_CANCEL_BY_GESTURE); @@ -2293,10 +2458,10 @@ public boolean onTouchEvent(MotionEvent motionEvent) { if (alpha < 0.45) { if (hasRecordVideo && isInVideoMode()) { CameraController.getInstance().cancelOnInitRunnable(onFinishInitCameraRunnable); - delegate.needStartRecordVideo(2, true, 0); + delegate.needStartRecordVideo(2, true, 0, voiceOnce ? 0x7FFFFFFF : 0); } else { delegate.needStartRecordAudio(0); - MediaController.getInstance().stopRecording(0, false, 0); + MediaController.getInstance().stopRecording(0, false, 0, voiceOnce); } recordingAudioVideo = false; updateRecordInterface(RECORD_STATE_CANCEL_BY_GESTURE); @@ -2315,15 +2480,15 @@ public boolean onTouchEvent(MotionEvent motionEvent) { startedDraggingX = -1; if (hasRecordVideo && isInVideoMode()) { CameraController.getInstance().cancelOnInitRunnable(onFinishInitCameraRunnable); - delegate.needStartRecordVideo(1, true, 0); + delegate.needStartRecordVideo(1, true, 0, voiceOnce ? 0x7FFFFFFF : 0); } else if (!sendVoiceEnabled) { delegate.needShowMediaBanHint(); } else { if (recordingAudioVideo && isInScheduleMode()) { - AlertsCreator.createScheduleDatePickerDialog(parentActivity, parentFragment.getDialogId(), (notify, scheduleDate) -> MediaController.getInstance().stopRecording(1, notify, scheduleDate), () -> MediaController.getInstance().stopRecording(0, false, 0), resourcesProvider); + AlertsCreator.createScheduleDatePickerDialog(parentActivity, parentFragment.getDialogId(), (notify, scheduleDate) -> MediaController.getInstance().stopRecording(1, notify, scheduleDate, false), () -> MediaController.getInstance().stopRecording(0, false, 0, false), resourcesProvider); } delegate.needStartRecordAudio(0); - MediaController.getInstance().stopRecording(isInScheduleMode() ? 3 : 1, true, 0); + MediaController.getInstance().stopRecording(isInScheduleMode() ? 3 : 1, true, 0, voiceOnce); } recordingAudioVideo = false; messageTransitionIsRunning = false; @@ -2375,10 +2540,10 @@ public boolean onTouchEvent(MotionEvent motionEvent) { if (alpha == 0) { if (hasRecordVideo && isInVideoMode()) { CameraController.getInstance().cancelOnInitRunnable(onFinishInitCameraRunnable); - delegate.needStartRecordVideo(2, true, 0); + delegate.needStartRecordVideo(2, true, 0, voiceOnce ? 0x7FFFFFFF : 0); } else { delegate.needStartRecordAudio(0); - MediaController.getInstance().stopRecording(0, false, 0); + MediaController.getInstance().stopRecording(0, false, 0, voiceOnce); } recordingAudioVideo = false; updateRecordInterface(RECORD_STATE_CANCEL_BY_GESTURE); @@ -3096,7 +3261,7 @@ protected void dispatchDraw(Canvas canvas) { private void resetRecordedState() { if (videoToSendMessageObject != null) { CameraController.getInstance().cancelOnInitRunnable(onFinishInitCameraRunnable); - delegate.needStartRecordVideo(2, true, 0); + delegate.needStartRecordVideo(2, true, 0, voiceOnce ? 0x7FFFFFFF : 0); } else { MessageObject playing = MediaController.getInstance().getPlayingMessageObject(); if (playing != null && playing == audioToSendMessageObject) { @@ -3471,6 +3636,7 @@ private void createBotWebViewButton() { } private void createRecordCircle() { + createOnceButton(); if (recordCircle != null) { return; } @@ -3479,6 +3645,18 @@ private void createRecordCircle() { sizeNotifierLayout.addView(recordCircle, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.BOTTOM)); } + private void createOnceButton() { + if (onceButton != null) { + return; + } + if (delegate == null || !delegate.onceVoiceAvailable()) { + return; + } + onceButton = new OnceButton(getContext()); + onceButton.setVisibility(GONE); + sizeNotifierLayout.addView(onceButton, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.BOTTOM)); + } + private void showRestrictedHint() { if (DialogObject.isChatDialog(dialog_id)) { TLRPC.Chat chat = accountInstance.getMessagesController().getChat(-dialog_id); @@ -4570,7 +4748,7 @@ public void onAnimationEnd(Animator animation) { } checkBotMenu(); - if (editingCaption && !captionLimitBulletinShown && !MessagesController.getInstance(currentAccount).premiumLocked && !UserConfig.getInstance(currentAccount).isPremium() && codePointCount > MessagesController.getInstance(currentAccount).captionLengthLimitDefault && codePointCount < MessagesController.getInstance(currentAccount).captionLengthLimitPremium) { + if (editingCaption && !captionLimitBulletinShown && !MessagesController.getInstance(currentAccount).premiumFeaturesBlocked() && !UserConfig.getInstance(currentAccount).isPremium() && codePointCount > MessagesController.getInstance(currentAccount).captionLengthLimitDefault && codePointCount < MessagesController.getInstance(currentAccount).captionLengthLimitPremium) { captionLimitBulletinShown = true; if (heightShouldBeChanged) { AndroidUtilities.runOnUIThread(()->showCaptionLimitBulletin(), 300); @@ -4641,10 +4819,10 @@ public boolean isRecordLocked() { public void cancelRecordingAudioVideo() { if (hasRecordVideo && isInVideoMode()) { CameraController.getInstance().cancelOnInitRunnable(onFinishInitCameraRunnable); - delegate.needStartRecordVideo(5, true, 0); + delegate.needStartRecordVideo(5, true, 0, voiceOnce ? 0x7FFFFFFF : 0); } else { delegate.needStartRecordAudio(0); - MediaController.getInstance().stopRecording(0, false, 0); + MediaController.getInstance().stopRecording(0, false, 0, voiceOnce); } recordingAudioVideo = false; updateRecordInterface(RECORD_STATE_CANCEL); @@ -5498,20 +5676,24 @@ private void hideRecordedAudioPanel(boolean wasSent) { updateEmojiButtonParams(); recordPannelAnimation = new AnimatorSet(); - recordPannelAnimation.playTogether( - ObjectAnimator.ofFloat(emojiButton, EMOJI_BUTTON_ALPHA, emojiButtonRestricted ? 0.5f : 1.0f), - ObjectAnimator.ofFloat(emojiButton, EMOJI_BUTTON_SCALE, 1.0f), - ObjectAnimator.ofFloat(recordDeleteImageView, View.ALPHA, 0.0f), - ObjectAnimator.ofFloat(recordDeleteImageView, View.SCALE_X, 0.0f), - ObjectAnimator.ofFloat(recordDeleteImageView, View.SCALE_Y, 0.0f), - - ObjectAnimator.ofFloat(recordedAudioPanel, View.ALPHA, 0.0f), - ObjectAnimator.ofFloat(attachButton, View.ALPHA, 1.0f), - ObjectAnimator.ofFloat(attachButton, View.SCALE_X, 1.0f), - ObjectAnimator.ofFloat(attachButton, View.SCALE_Y, 1.0f), - ObjectAnimator.ofFloat(messageEditText, View.ALPHA, 1f), - ObjectAnimator.ofFloat(messageEditText, MESSAGE_TEXT_TRANSLATION_X, 0) - ); + ArrayList animators = new ArrayList<>(); + animators.add(ObjectAnimator.ofFloat(emojiButton, EMOJI_BUTTON_ALPHA, emojiButtonRestricted ? 0.5f : 1.0f)); + animators.add(ObjectAnimator.ofFloat(emojiButton, EMOJI_BUTTON_SCALE, 1.0f)); + animators.add(ObjectAnimator.ofFloat(recordDeleteImageView, View.ALPHA, 0.0f)); + animators.add(ObjectAnimator.ofFloat(recordDeleteImageView, View.SCALE_X, 0.0f)); + animators.add(ObjectAnimator.ofFloat(recordDeleteImageView, View.SCALE_Y, 0.0f)); + + animators.add(ObjectAnimator.ofFloat(recordedAudioPanel, View.ALPHA, 0.0f)); + animators.add(ObjectAnimator.ofFloat(attachButton, View.ALPHA, 1.0f)); + animators.add(ObjectAnimator.ofFloat(attachButton, View.SCALE_X, 1.0f)); + animators.add(ObjectAnimator.ofFloat(attachButton, View.SCALE_Y, 1.0f)); + animators.add(ObjectAnimator.ofFloat(messageEditText, View.ALPHA, 1f)); + animators.add(ObjectAnimator.ofFloat(messageEditText, MESSAGE_TEXT_TRANSLATION_X, 0)); + if (onceButton != null) { + animators.add(ObjectAnimator.ofFloat(onceButton, View.ALPHA, 0)); + onceButton.hideHintView(); + } + recordPannelAnimation.playTogether(animators); if (botCommandsMenuButton != null) { botCommandsMenuButton.setAlpha(0f); @@ -5544,12 +5726,16 @@ public void onAnimationEnd(Animator animation) { } AnimatorSet exitAnimation = new AnimatorSet(); + ArrayList animators = new ArrayList<>(); if (isInVideoMode()) { - exitAnimation.playTogether( - ObjectAnimator.ofFloat(videoTimelineView, View.ALPHA, 0.0f), - ObjectAnimator.ofFloat(videoTimelineView, View.TRANSLATION_X, -AndroidUtilities.dp(20)), - ObjectAnimator.ofFloat(messageEditText, MESSAGE_TEXT_TRANSLATION_X, 0) - ); + animators.add(ObjectAnimator.ofFloat(videoTimelineView, View.ALPHA, 0.0f)); + animators.add(ObjectAnimator.ofFloat(videoTimelineView, View.TRANSLATION_X, -AndroidUtilities.dp(20))); + animators.add(ObjectAnimator.ofFloat(messageEditText, MESSAGE_TEXT_TRANSLATION_X, 0)); + if (onceButton != null) { + animators.add(ObjectAnimator.ofFloat(onceButton, View.ALPHA, 0.0f)); + onceButton.hideHintView(); + } + exitAnimation.playTogether(animators); if (emojiButtonPaddingAlpha == 1f) { exitAnimation.playTogether(ObjectAnimator.ofFloat(messageEditText, View.ALPHA, 1f)); } else { @@ -5571,16 +5757,19 @@ public void onAnimationEnd(Animator animation) { messageEditTextAniamtor.setDuration(200); exitAnimation.playTogether(messageEditTextAniamtor); } - exitAnimation.playTogether( - ObjectAnimator.ofFloat(recordedAudioSeekBar, View.ALPHA, 0.0f), - ObjectAnimator.ofFloat(recordedAudioPlayButton, View.ALPHA, 0.0f), - ObjectAnimator.ofFloat(recordedAudioBackground, View.ALPHA, 0.0f), - ObjectAnimator.ofFloat(recordedAudioTimeTextView, View.ALPHA, 0.0f), - ObjectAnimator.ofFloat(recordedAudioSeekBar, View.TRANSLATION_X, -AndroidUtilities.dp(20)), - ObjectAnimator.ofFloat(recordedAudioPlayButton, View.TRANSLATION_X, -AndroidUtilities.dp(20)), - ObjectAnimator.ofFloat(recordedAudioBackground, View.TRANSLATION_X, -AndroidUtilities.dp(20)), - ObjectAnimator.ofFloat(recordedAudioTimeTextView, View.TRANSLATION_X, -AndroidUtilities.dp(20)) - ); + animators.add(ObjectAnimator.ofFloat(recordedAudioSeekBar, View.ALPHA, 0.0f)); + animators.add(ObjectAnimator.ofFloat(recordedAudioPlayButton, View.ALPHA, 0.0f)); + animators.add(ObjectAnimator.ofFloat(recordedAudioBackground, View.ALPHA, 0.0f)); + animators.add(ObjectAnimator.ofFloat(recordedAudioTimeTextView, View.ALPHA, 0.0f)); + animators.add(ObjectAnimator.ofFloat(recordedAudioSeekBar, View.TRANSLATION_X, -AndroidUtilities.dp(20))); + animators.add(ObjectAnimator.ofFloat(recordedAudioPlayButton, View.TRANSLATION_X, -AndroidUtilities.dp(20))); + animators.add(ObjectAnimator.ofFloat(recordedAudioBackground, View.TRANSLATION_X, -AndroidUtilities.dp(20))); + animators.add(ObjectAnimator.ofFloat(recordedAudioTimeTextView, View.TRANSLATION_X, -AndroidUtilities.dp(20))); + if (onceButton != null) { + animators.add(ObjectAnimator.ofFloat(onceButton, View.ALPHA, 0)); + onceButton.hideHintView(); + } + exitAnimation.playTogether(animators); } exitAnimation.setDuration(200); @@ -5655,6 +5844,9 @@ public void onAnimationEnd(Animator animation) { if (recordPannelAnimation != null) { recordPannelAnimation.start(); } + if (onceButton != null) { + onceButton.invalidate(); + } } private void hideRecordedAudioPanelInternal() { @@ -5738,7 +5930,7 @@ private void sendMessageInternal(boolean notify, int scheduleDate, boolean allow return; } if (videoToSendMessageObject != null) { - delegate.needStartRecordVideo(4, notify, scheduleDate); + delegate.needStartRecordVideo(4, notify, scheduleDate, voiceOnce ? 0x7FFFFFFF : 0); hideRecordedAudioPanel(true); checkSendButton(true); return; @@ -5747,7 +5939,7 @@ private void sendMessageInternal(boolean notify, int scheduleDate, boolean allow if (playing != null && playing == audioToSendMessageObject) { MediaController.getInstance().cleanupPlayer(true, true); } - SendMessagesHelper.SendMessageParams params = SendMessagesHelper.SendMessageParams.of(audioToSend, null, audioToSendPath, dialog_id, replyingMessageObject, getThreadMessage(), null, null, null, null, notify, scheduleDate, 0, null, null, false); + SendMessagesHelper.SendMessageParams params = SendMessagesHelper.SendMessageParams.of(audioToSend, null, audioToSendPath, dialog_id, replyingMessageObject, getThreadMessage(), null, null, null, null, notify, scheduleDate, voiceOnce ? 0x7FFFFFFF : 0, null, null, false); applyStoryToSendMessageParams(params); SendMessagesHelper.getInstance(currentAccount).sendMessage(params); if (delegate != null) { @@ -5921,7 +6113,7 @@ public void doneEditingMessage() { } catch (Exception ignored) {} } - if (!MessagesController.getInstance(currentAccount).premiumLocked && MessagesController.getInstance(currentAccount).captionLengthLimitPremium > codePointCount) { + if (!MessagesController.getInstance(currentAccount).premiumFeaturesBlocked() && MessagesController.getInstance(currentAccount).captionLengthLimitPremium > codePointCount) { showCaptionLimitBulletin(); } return; @@ -6916,6 +7108,10 @@ protected void updateRecordInterface(int recordState) { if (recordInterfaceState == 1) { return; } + voiceOnce = false; + if (onceButton != null) { + onceButton.periodDrawable.setValue(1, false, false); + } createRecordAudioPanel(); recordInterfaceState = 1; if (emojiView != null) { @@ -6954,6 +7150,9 @@ protected void updateRecordInterface(int recordState) { recordCircle.setVisibility(VISIBLE); recordCircle.setAmplitude(0); } + if (onceButton != null) { + onceButton.setVisibility(delegate != null && delegate.onceVoiceAvailable() ? VISIBLE : GONE); + } if (recordDot != null) { recordDot.resetAlpha(); @@ -6987,6 +7186,9 @@ protected void updateRecordInterface(int recordState) { ObjectAnimator.ofFloat(slideText, View.TRANSLATION_X, 0), ObjectAnimator.ofFloat(slideText, View.ALPHA, 1) ); + if (onceButton != null) { + iconChanges.playTogether(ObjectAnimator.ofFloat(onceButton, View.ALPHA, 1)); + } if (audioVideoSendButton != null) { iconChanges.playTogether(ObjectAnimator.ofFloat(audioVideoSendButton, View.ALPHA, 0)); } @@ -7103,6 +7305,10 @@ public void onAnimationEnd(Animator animator) { ObjectAnimator.ofFloat(messageEditText, MESSAGE_TEXT_TRANSLATION_X, 0), ObjectAnimator.ofFloat(recordCircle, "slideToCancelProgress", 1f) ); + if (onceButton != null) { + runningAnimationAudio.playTogether(ObjectAnimator.ofFloat(onceButton, View.ALPHA, 0)); + onceButton.hideHintView(); + } if (botCommandsMenuButton != null) { runningAnimationAudio.playTogether( ObjectAnimator.ofFloat(botCommandsMenuButton, View.SCALE_Y, 1), @@ -7319,6 +7525,10 @@ public void onAnimationEnd(Animator animation) { botCommandsMenuButton.setScaleX(0f); botCommandsMenuButton.setScaleY(0f); } + + if (onceButton != null && onceVisible && MessagesController.getGlobalMainSettings().getInt("voiceoncehint", 0) < 3) { + onceButton.showHintView(); + } } }); @@ -7334,6 +7544,10 @@ public void onAnimationEnd(Animator animation) { ObjectAnimator.ofFloat(recordDot, View.SCALE_Y, 0), ObjectAnimator.ofFloat(recordDot, View.SCALE_X, 0) ); + if (onceButton != null) { + iconsAnimator.playTogether(ObjectAnimator.ofFloat(onceButton, View.ALPHA, 0)); + onceButton.hideHintView(); + } if (botCommandsMenuButton != null) { iconsAnimator.playTogether( @@ -7481,6 +7695,10 @@ public void onAnimationEnd(Animator animation) { ObjectAnimator.ofFloat(recordDot, View.SCALE_X, 0), ObjectAnimator.ofFloat(audioVideoButtonContainer, View.ALPHA, 1.0f) ); + if (onceButton != null) { + iconsAnimator.playTogether(ObjectAnimator.ofFloat(onceButton, View.ALPHA, 0)); + onceButton.hideHintView(); + } if (botCommandsMenuButton != null) { iconsAnimator.playTogether( ObjectAnimator.ofFloat(botCommandsMenuButton, View.SCALE_Y, 1), @@ -8164,7 +8382,7 @@ public CharSequence getFieldText() { } public void updateGiftButton(boolean animated) { - boolean visible = !MessagesController.getInstance(currentAccount).premiumLocked && MessagesController.getInstance(currentAccount).giftAttachMenuIcon && + boolean visible = !MessagesController.getInstance(currentAccount).premiumFeaturesBlocked() && MessagesController.getInstance(currentAccount).giftAttachMenuIcon && MessagesController.getInstance(currentAccount).giftTextFieldIcon && getParentFragment() != null && getParentFragment().getCurrentUser() != null && !BuildVars.IS_BILLING_UNAVAILABLE && !getParentFragment().getCurrentUser().self && !getParentFragment().getCurrentUser().premium && getParentFragment().getCurrentUserInfo() != null && !getParentFragment().getCurrentUserInfo().premium_gifts.isEmpty() && !isInScheduleMode() && @@ -8759,6 +8977,21 @@ public void run() { } else if (button instanceof TLRPC.TL_keyboardButtonRequestPeer) { TLRPC.TL_keyboardButtonRequestPeer btn = (TLRPC.TL_keyboardButtonRequestPeer) button; if (btn.peer_type != null && messageObject != null && messageObject.messageOwner != null) { + if (btn.peer_type instanceof TLRPC.TL_requestPeerTypeUser && btn.max_quantity > 1) { + MultiContactsSelectorBottomSheet.open(btn.max_quantity, ids -> { + if (ids != null && !ids.isEmpty()) { + TLRPC.TL_messages_sendBotRequestedPeer req = new TLRPC.TL_messages_sendBotRequestedPeer(); + req.peer = MessagesController.getInstance(currentAccount).getInputPeer(messageObject.messageOwner.peer_id); + req.msg_id = messageObject.getId(); + req.button_id = btn.button_id; + for (Long id : ids) { + req.requested_peers.add(MessagesController.getInstance(currentAccount).getInputPeer(id)); + } + ConnectionsManager.getInstance(currentAccount).sendRequest(req, null); + } + }); + return false; + } Bundle args = new Bundle(); args.putBoolean("onlySelect", true); args.putInt("dialogsType", DialogsActivity.DIALOGS_TYPE_BOT_REQUEST_PEER); @@ -8774,20 +9007,17 @@ public void run() { FileLog.e(e); } DialogsActivity fragment = new DialogsActivity(args); - fragment.setDelegate(new DialogsActivity.DialogsActivityDelegate() { - @Override - public boolean didSelectDialogs(DialogsActivity fragment, ArrayList dids, CharSequence message, boolean param, TopicsFragment topicsFragment) { - if (dids != null && !dids.isEmpty()) { - TLRPC.TL_messages_sendBotRequestedPeer req = new TLRPC.TL_messages_sendBotRequestedPeer(); - req.peer = MessagesController.getInstance(currentAccount).getInputPeer(messageObject.messageOwner.peer_id); - req.msg_id = messageObject.getId(); - req.button_id = btn.button_id; - req.requested_peer = MessagesController.getInstance(currentAccount).getInputPeer(dids.get(0).dialogId); - ConnectionsManager.getInstance(currentAccount).sendRequest(req, null); - } - fragment.finishFragment(); - return true; - } + fragment.setDelegate((dialogFragment, dids, message, param, topicsFragment) -> { + if (dids != null && !dids.isEmpty()) { + TLRPC.TL_messages_sendBotRequestedPeer req = new TLRPC.TL_messages_sendBotRequestedPeer(); + req.peer = MessagesController.getInstance(currentAccount).getInputPeer(messageObject.messageOwner.peer_id); + req.msg_id = messageObject.getId(); + req.button_id = btn.button_id; + req.requested_peers.add(MessagesController.getInstance(currentAccount).getInputPeer(dids.get(0).dialogId)); + ConnectionsManager.getInstance(currentAccount).sendRequest(req, null); + } + dialogFragment.finishFragment(); + return true; }); parentFragment.presentFragment(fragment); return false; @@ -10553,10 +10783,10 @@ public boolean onTouchEvent(MotionEvent event) { public void onCancelButtonPressed() { if (hasRecordVideo && isInVideoMode()) { CameraController.getInstance().cancelOnInitRunnable(onFinishInitCameraRunnable); - delegate.needStartRecordVideo(5, true, 0); + delegate.needStartRecordVideo(5, true, 0, voiceOnce ? 0x7FFFFFFF : 0); } else { delegate.needStartRecordAudio(0); - MediaController.getInstance().stopRecording(0, false, 0); + MediaController.getInstance().stopRecording(0, false, 0, voiceOnce); } recordingAudioVideo = false; updateRecordInterface(RECORD_STATE_CANCEL); @@ -10814,7 +11044,7 @@ protected void onDraw(Canvas canvas) { if (isInVideoMode()) { if (t >= 59500 && !stoppedInternal) { startedDraggingX = -1; - delegate.needStartRecordVideo(3, true, 0); + delegate.needStartRecordVideo(3, true, 0, voiceOnce ? 0x7FFFFFFF : 0); stoppedInternal = true; } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/ChatAttachAlert.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/ChatAttachAlert.java index 40d6645d514..4513c0c6330 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/ChatAttachAlert.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/ChatAttachAlert.java @@ -2557,7 +2557,7 @@ public void onAnimationEnd(Animator animation) { sendButtonColorAnimator.setDuration(150).start(); } -// if (!captionLimitBulletinShown && !MessagesController.getInstance(currentAccount).premiumLocked && !UserConfig.getInstance(currentAccount).isPremium() && codepointCount > MessagesController.getInstance(currentAccount).captionLengthLimitDefault && codepointCount < MessagesController.getInstance(currentAccount).captionLengthLimitPremium) { +// if (!captionLimitBulletinShown && !MessagesController.getInstance(currentAccount).premiumFeaturesBlocked() && !UserConfig.getInstance(currentAccount).isPremium() && codepointCount > MessagesController.getInstance(currentAccount).captionLengthLimitDefault && codepointCount < MessagesController.getInstance(currentAccount).captionLengthLimitPremium) { // captionLimitBulletinShown = true; // showCaptionLimitBulletin(parentFragment); // } @@ -2623,7 +2623,7 @@ public void getOutline(View view, Outline outline) { } catch (Exception ignored) { } - if (!MessagesController.getInstance(currentAccount).premiumLocked && MessagesController.getInstance(currentAccount).captionLengthLimitPremium > codepointCount) { + if (!MessagesController.getInstance(currentAccount).premiumFeaturesBlocked() && MessagesController.getInstance(currentAccount).captionLengthLimitPremium > codepointCount) { showCaptionLimitBulletin(parentFragment); } return; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/ChatAvatarContainer.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/ChatAvatarContainer.java index ed9bf031562..2232ad2f173 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/ChatAvatarContainer.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/ChatAvatarContainer.java @@ -33,6 +33,7 @@ import org.telegram.messenger.AndroidUtilities; import org.telegram.messenger.ApplicationLoader; import org.telegram.messenger.ChatObject; +import org.telegram.messenger.DialogObject; import org.telegram.messenger.Emoji; import org.telegram.messenger.FileLog; import org.telegram.messenger.ImageLoader; @@ -487,6 +488,7 @@ private void fadeOutToLessWidth(int largerWidth) { titleTextLargerCopyView.setTypeface(AndroidUtilities.getTypeface("fonts/rmedium.ttf")); titleTextLargerCopyView.setLeftDrawableTopPadding(-AndroidUtilities.dp(1.3f)); titleTextLargerCopyView.setRightDrawable(titleTextView.getRightDrawable()); + titleTextLargerCopyView.setRightDrawable2(titleTextView.getRightDrawable2()); titleTextLargerCopyView.setRightDrawableOutside(titleTextView.getRightDrawableOutside()); titleTextLargerCopyView.setLeftDrawable(titleTextView.getLeftDrawable()); titleTextLargerCopyView.setText(titleTextView.getText()); @@ -618,16 +620,17 @@ public void setTime(int value, boolean animated) { private boolean rightDrawableIsScamOrVerified = false; private String rightDrawableContentDescription = null; + private String rightDrawable2ContentDescription = null; public void setTitleIcons(Drawable leftIcon, Drawable mutedIcon) { titleTextView.setLeftDrawable(leftIcon); if (!rightDrawableIsScamOrVerified) { if (mutedIcon != null) { - rightDrawableContentDescription = LocaleController.getString("NotificationsMuted", R.string.NotificationsMuted); + rightDrawable2ContentDescription = LocaleController.getString("NotificationsMuted", R.string.NotificationsMuted); } else { - rightDrawableContentDescription = null; + rightDrawable2ContentDescription = null; } - titleTextView.setRightDrawable(mutedIcon); + titleTextView.setRightDrawable2(mutedIcon); } } @@ -644,9 +647,9 @@ public void setTitle(CharSequence value, boolean scam, boolean fake, boolean ver if (!(titleTextView.getRightDrawable() instanceof ScamDrawable)) { ScamDrawable drawable = new ScamDrawable(11, scam ? 0 : 1); drawable.setColor(getThemedColor(Theme.key_actionBarDefaultSubtitle)); - titleTextView.setRightDrawable(drawable); + titleTextView.setRightDrawable2(drawable); // titleTextView.setRightPadding(0); - rightDrawableContentDescription = LocaleController.getString("ScamMessage", R.string.ScamMessage); + rightDrawable2ContentDescription = LocaleController.getString("ScamMessage", R.string.ScamMessage); rightDrawableIsScamOrVerified = true; } } else if (verified) { @@ -655,37 +658,34 @@ public void setTitle(CharSequence value, boolean scam, boolean fake, boolean ver Drawable verifiedCheck = getResources().getDrawable(R.drawable.verified_check).mutate(); verifiedCheck.setColorFilter(new PorterDuffColorFilter(getThemedColor(Theme.key_profile_verifiedCheck), PorterDuff.Mode.MULTIPLY)); Drawable verifiedDrawable = new CombinedDrawable(verifiedBackground, verifiedCheck); - titleTextView.setRightDrawable(verifiedDrawable); -// titleTextView.setRightPadding(titleTextView.getPaddingRight()); + titleTextView.setRightDrawable2(verifiedDrawable); rightDrawableIsScamOrVerified = true; - rightDrawableContentDescription = LocaleController.getString("AccDescrVerified", R.string.AccDescrVerified); - } else if (premium) { - boolean isStatus = emojiStatus instanceof TLRPC.TL_emojiStatus || emojiStatus instanceof TLRPC.TL_emojiStatusUntil && ((TLRPC.TL_emojiStatusUntil) emojiStatus).until > (int) (System.currentTimeMillis() / 1000); -// if (premiumIconHiddable) { -// titleTextView.setCanHideRightDrawable(!isStatus); -// } + rightDrawable2ContentDescription = LocaleController.getString("AccDescrVerified", R.string.AccDescrVerified); + } else if (titleTextView.getRightDrawable() instanceof ScamDrawable) { + titleTextView.setRightDrawable2(null); + rightDrawableIsScamOrVerified = false; + rightDrawable2ContentDescription = null; + } + if (premium || DialogObject.getEmojiStatusDocumentId(emojiStatus) != 0) { if (titleTextView.getRightDrawable() instanceof AnimatedEmojiDrawable.WrapSizeDrawable && ((AnimatedEmojiDrawable.WrapSizeDrawable) titleTextView.getRightDrawable()).getDrawable() instanceof AnimatedEmojiDrawable) { ((AnimatedEmojiDrawable) ((AnimatedEmojiDrawable.WrapSizeDrawable) titleTextView.getRightDrawable()).getDrawable()).removeView(titleTextView); } - if (emojiStatus instanceof TLRPC.TL_emojiStatus) { - emojiStatusDrawable.set(((TLRPC.TL_emojiStatus) emojiStatus).document_id, animated); - } else if (emojiStatus instanceof TLRPC.TL_emojiStatusUntil && ((TLRPC.TL_emojiStatusUntil) emojiStatus).until > (int) (System.currentTimeMillis() / 1000)) { - emojiStatusDrawable.set(((TLRPC.TL_emojiStatusUntil) emojiStatus).document_id, animated); - } else { + if (DialogObject.getEmojiStatusDocumentId(emojiStatus) != 0) { + emojiStatusDrawable.set(DialogObject.getEmojiStatusDocumentId(emojiStatus), animated); + } else if (premium) { Drawable drawable = ContextCompat.getDrawable(ApplicationLoader.applicationContext, R.drawable.msg_premium_liststar).mutate(); drawable.setColorFilter(new PorterDuffColorFilter(getThemedColor(Theme.key_profile_verifiedBackground), PorterDuff.Mode.MULTIPLY)); emojiStatusDrawable.set(drawable, animated); + } else { + emojiStatusDrawable.set((Drawable) null, animated); } emojiStatusDrawable.setColor(getThemedColor(Theme.key_profile_verifiedBackground)); titleTextView.setRightDrawable(emojiStatusDrawable); -// titleTextView.setRightPadding(titleTextView.getPaddingRight()); - rightDrawableIsScamOrVerified = true; + rightDrawableIsScamOrVerified = false; rightDrawableContentDescription = LocaleController.getString("AccDescrPremium", R.string.AccDescrPremium); - } else if (titleTextView.getRightDrawable() instanceof ScamDrawable) { + } else { titleTextView.setRightDrawable(null); -// titleTextView.setRightPadding(0); - rightDrawableIsScamOrVerified = false; rightDrawableContentDescription = null; } } @@ -1146,6 +1146,10 @@ public void onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo info) { sb.append(", "); sb.append(rightDrawableContentDescription); } + if (rightDrawable2ContentDescription != null) { + sb.append(", "); + sb.append(rightDrawable2ContentDescription); + } sb.append("\n"); sb.append(subtitleTextView.getText()); info.setContentDescription(sb); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/ChatThemeBottomSheet.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/ChatThemeBottomSheet.java index 1923c90b849..4f2ad0a0bb4 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/ChatThemeBottomSheet.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/ChatThemeBottomSheet.java @@ -1,5 +1,7 @@ package org.telegram.ui.Components; +import static org.telegram.messenger.AndroidUtilities.dp; + import android.animation.Animator; import android.animation.AnimatorListenerAdapter; import android.animation.ValueAnimator; @@ -20,6 +22,8 @@ import android.net.Uri; import android.os.Build; import android.os.Bundle; +import android.text.SpannableStringBuilder; +import android.text.Spanned; import android.text.TextUtils; import android.util.Log; import android.util.TypedValue; @@ -51,10 +55,12 @@ import org.telegram.messenger.NotificationCenter; import org.telegram.messenger.R; import org.telegram.messenger.SharedConfig; +import org.telegram.messenger.UserObject; import org.telegram.messenger.Utilities; import org.telegram.tgnet.ConnectionsManager; import org.telegram.tgnet.ResultCallback; import org.telegram.tgnet.TLRPC; +import org.telegram.tgnet.tl.TL_stories; import org.telegram.ui.ActionBar.AlertDialog; import org.telegram.ui.ActionBar.BackDrawable; import org.telegram.ui.ActionBar.BaseFragment; @@ -66,7 +72,9 @@ import org.telegram.ui.Cells.DrawerProfileCell; import org.telegram.ui.Cells.ThemesHorizontalListCell; import org.telegram.ui.ChatActivity; +import org.telegram.ui.Components.Premium.LimitReachedBottomSheet; import org.telegram.ui.PhotoViewer; +import org.telegram.ui.StatisticActivity; import org.telegram.ui.ThemePreviewActivity; import org.telegram.ui.WallpapersListActivity; @@ -101,6 +109,7 @@ public class ChatThemeBottomSheet extends BottomSheet implements NotificationCen private final LinearSmoothScroller scroller; private final View applyButton; private AnimatedTextView applyTextView; + private AnimatedTextView applySubTextView; private TextView chooseBackgroundTextView; private ChatThemeItem selectedItem; private boolean forceDark; @@ -113,7 +122,7 @@ public class ChatThemeBottomSheet extends BottomSheet implements NotificationCen HintView hintView; private boolean dataLoaded; private EmojiThemes currentTheme; - ThemePreviewActivity overlayFragment; + BaseFragment overlayFragment; public ChatAttachAlert chatAttachAlert; private FrameLayout chatAttachButton; private AnimatedTextView chatAttachButtonText; @@ -150,10 +159,10 @@ public ChatThemeBottomSheet(final ChatActivity chatActivity, ChatActivity.ThemeD titleView.setTextColor(getThemedColor(Theme.key_dialogTextBlack)); titleView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 20); titleView.setTypeface(AndroidUtilities.getTypeface("fonts/rmedium.ttf")); - titleView.setPadding(AndroidUtilities.dp(12), AndroidUtilities.dp(6), AndroidUtilities.dp(12), AndroidUtilities.dp(8)); + titleView.setPadding(dp(12), dp(6), dp(12), dp(8)); backButtonView = new ImageView(getContext()); - int padding = AndroidUtilities.dp(10); + int padding = dp(10); backButtonView.setPadding(padding, padding, padding, padding); backButtonDrawable = new BackDrawable(false); backButtonView.setImageDrawable(backButtonDrawable); @@ -169,7 +178,7 @@ public ChatThemeBottomSheet(final ChatActivity chatActivity, ChatActivity.ThemeD rootLayout.addView(titleView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.TOP | Gravity.START, 44, 0, 62, 0)); int drawableColor = getThemedColor(Theme.key_featuredStickers_addButton); - int drawableSize = AndroidUtilities.dp(28); + int drawableSize = dp(28); darkThemeDrawable = new RLottieDrawable(R.raw.sun_outline, "" + R.raw.sun_outline, drawableSize, drawableSize, false, null); forceDark = !Theme.getActiveTheme().isDark(); setForceDark(Theme.getActiveTheme().isDark(), false); @@ -213,7 +222,7 @@ protected int calculateTimeForScrolling(int dx) { recyclerView.setItemAnimator(null); recyclerView.setNestedScrollingEnabled(false); recyclerView.setLayoutManager(layoutManager = new LinearLayoutManager(getContext(), LinearLayoutManager.HORIZONTAL, false)); - recyclerView.setPadding(AndroidUtilities.dp(12), 0, AndroidUtilities.dp(12), 0); + recyclerView.setPadding(dp(12), 0, dp(12), 0); recyclerView.setOnItemClickListener((view, position) -> { if (adapter.items.get(position) == selectedItem || changeDayNightView != null) { return; @@ -255,7 +264,7 @@ public void run() { rootLayout.addView(recyclerView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 104, Gravity.START, 0, 44, 0, 0)); applyButton = new View(getContext()); - applyButton.setBackground(Theme.createSimpleSelectorRoundRectDrawable(AndroidUtilities.dp(6), getThemedColor(Theme.key_featuredStickers_addButton), getThemedColor(Theme.key_featuredStickers_addButtonPressed))); + applyButton.setBackground(Theme.createSimpleSelectorRoundRectDrawable(dp(6), getThemedColor(Theme.key_featuredStickers_addButton), getThemedColor(Theme.key_featuredStickers_addButtonPressed))); applyButton.setOnClickListener((view) -> applySelectedTheme()); rootLayout.addView(applyButton, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 48, Gravity.START, 16, 162, 16, 16)); @@ -284,10 +293,19 @@ public void onClick(View v) { applyTextView.adaptWidth = false; applyTextView.setGravity(Gravity.CENTER); applyTextView.setTextColor(getThemedColor(Theme.key_featuredStickers_buttonText)); - applyTextView.setTextSize(AndroidUtilities.dp(15)); + applyTextView.setTextSize(dp(15)); applyTextView.setTypeface(AndroidUtilities.getTypeface("fonts/rmedium.ttf")); rootLayout.addView(applyTextView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 48, Gravity.START, 16, 162, 16, 16)); + applySubTextView = new AnimatedTextView(getContext(), true, true, true); + applySubTextView.getDrawable().setEllipsizeByGradient(true); + applySubTextView.adaptWidth = false; + applySubTextView.setGravity(Gravity.CENTER); + applySubTextView.setTextColor(getThemedColor(Theme.key_featuredStickers_buttonText)); + applySubTextView.setTextSize(dp(12)); + applySubTextView.setAlpha(0f); + applySubTextView.setTranslationY(dp(11)); + rootLayout.addView(applySubTextView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 48, Gravity.START, 16, 162, 16, 16)); if (currentWallpaper != null) { cancelOrResetTextView = new TextView(getContext()); @@ -315,7 +333,13 @@ public void onClick(View v) { themeHintTextView.setGravity(Gravity.CENTER); themeHintTextView.setLines(1); themeHintTextView.setSingleLine(true); - themeHintTextView.setText(LocaleController.formatString("ChatThemeApplyHint", R.string.ChatThemeApplyHint, chatActivity.getCurrentUser().first_name)); + String name = ""; + if (chatActivity.getCurrentUser() != null) { + name = UserObject.getFirstName(chatActivity.getCurrentUser()); + } else if (chatActivity.getCurrentChat() != null) { + name = chatActivity.getCurrentChat().title; + } + themeHintTextView.setText(LocaleController.formatString("ChatThemeApplyHint", R.string.ChatThemeApplyHint, name)); themeHintTextView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 15); rootLayout.addView(themeHintTextView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 48, Gravity.START, 16, 214, 16, 12)); } @@ -326,11 +350,11 @@ public void onClick(View v) { private void updateButtonColors() { if (themeHintTextView != null) { themeHintTextView.setTextColor(getThemedColor(Theme.key_dialogTextGray)); - themeHintTextView.setBackground(Theme.createSimpleSelectorRoundRectDrawable(AndroidUtilities.dp(6), Color.TRANSPARENT, ColorUtils.setAlphaComponent(getThemedColor(Theme.key_featuredStickers_addButton), (int) (0.3f * 255)))); + themeHintTextView.setBackground(Theme.createSimpleSelectorRoundRectDrawable(dp(6), Color.TRANSPARENT, ColorUtils.setAlphaComponent(getThemedColor(Theme.key_featuredStickers_addButton), (int) (0.3f * 255)))); } if (cancelOrResetTextView != null) { cancelOrResetTextView.setTextColor(getThemedColor(Theme.key_text_RedRegular)); - cancelOrResetTextView.setBackground(Theme.createSimpleSelectorRoundRectDrawable(AndroidUtilities.dp(6), Color.TRANSPARENT, ColorUtils.setAlphaComponent(getThemedColor(Theme.key_text_RedRegular), (int) (0.3f * 255)))); + cancelOrResetTextView.setBackground(Theme.createSimpleSelectorRoundRectDrawable(dp(6), Color.TRANSPARENT, ColorUtils.setAlphaComponent(getThemedColor(Theme.key_text_RedRegular), (int) (0.3f * 255)))); } backButtonView.setBackground(Theme.createSelectorDrawable(ColorUtils.setAlphaComponent(getThemedColor(Theme.key_dialogTextBlack), 30), 1)); backButtonDrawable.setColor(getThemedColor(Theme.key_dialogTextBlack)); @@ -339,7 +363,7 @@ private void updateButtonColors() { darkThemeView.setBackground(Theme.createSelectorDrawable(ColorUtils.setAlphaComponent(getThemedColor(Theme.key_featuredStickers_addButton), 30), 1)); chooseBackgroundTextView.setTextColor(getThemedColor(Theme.key_dialogTextBlue)); - chooseBackgroundTextView.setBackground(Theme.createSimpleSelectorRoundRectDrawable(AndroidUtilities.dp(6), Color.TRANSPARENT, ColorUtils.setAlphaComponent(getThemedColor(Theme.key_featuredStickers_addButton), (int) (0.3f * 255)))); + chooseBackgroundTextView.setBackground(Theme.createSimpleSelectorRoundRectDrawable(dp(6), Color.TRANSPARENT, ColorUtils.setAlphaComponent(getThemedColor(Theme.key_featuredStickers_addButton), (int) (0.3f * 255)))); } @@ -357,7 +381,12 @@ private void previewSelectedTheme() { } } + private ColoredImageSpan lockSpan; private void updateState(boolean animated) { + TLRPC.Chat chat = chatActivity.getCurrentChat(); + if (chat != null) { + checkBoostsLevel(); + } if (!dataLoaded) { backButtonDrawable.setRotation(1f, animated); applyButton.setEnabled(false); @@ -365,6 +394,7 @@ private void updateState(boolean animated) { AndroidUtilities.updateViewVisibilityAnimated(cancelOrResetTextView, false, 0.9f, false, animated); AndroidUtilities.updateViewVisibilityAnimated(applyButton, false, 1f, false, animated); AndroidUtilities.updateViewVisibilityAnimated(applyTextView, false, 0.9f, false, animated); + AndroidUtilities.updateViewVisibilityAnimated(applySubTextView, false, 0.9f, false, animated); AndroidUtilities.updateViewVisibilityAnimated(themeHintTextView, false, 0.9f, false, animated); AndroidUtilities.updateViewVisibilityAnimated(progressView, true, 1f, true, animated); } else { @@ -372,16 +402,30 @@ private void updateState(boolean animated) { if (hasChanges()) { backButtonDrawable.setRotation(0, animated); applyButton.setEnabled(true); - AndroidUtilities.updateViewVisibilityAnimated(chooseBackgroundTextView, false, 0.9f, false, animated); - AndroidUtilities.updateViewVisibilityAnimated(cancelOrResetTextView, false, 0.9f, false, animated); - AndroidUtilities.updateViewVisibilityAnimated(applyButton, true, 1f, false, animated); - AndroidUtilities.updateViewVisibilityAnimated(applyTextView, true, 0.9f, false, animated); - AndroidUtilities.updateViewVisibilityAnimated(themeHintTextView, true, 0.9f, false, animated); + boolean showSubText = false; if (selectedItem != null && selectedItem.chatTheme != null && selectedItem.chatTheme.showAsDefaultStub && selectedItem.chatTheme.wallpaper == null) { applyTextView.setText(LocaleController.getString("ChatResetTheme", R.string.ChatResetTheme)); } else { applyTextView.setText(LocaleController.getString("ChatApplyTheme", R.string.ChatApplyTheme)); + if (chat != null && boostsStatus != null && boostsStatus.level < chatActivity.getMessagesController().channelWallpaperLevelMin) { + showSubText = true; + SpannableStringBuilder text = new SpannableStringBuilder("l"); + if (lockSpan == null) { + lockSpan = new ColoredImageSpan(R.drawable.mini_switch_lock); + lockSpan.setTopOffset(1); + } + text.setSpan(lockSpan, 0, 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + text.append(" ").append(LocaleController.formatPluralString("ReactionLevelRequiredBtn", chatActivity.getMessagesController().channelWallpaperLevelMin)); + applySubTextView.setText(text); + } } + updateApplySubTextTranslation(showSubText, animated && applyTextView.getAlpha() > .8f); + AndroidUtilities.updateViewVisibilityAnimated(chooseBackgroundTextView, false, 0.9f, false, animated); + AndroidUtilities.updateViewVisibilityAnimated(cancelOrResetTextView, false, 0.9f, false, animated); + AndroidUtilities.updateViewVisibilityAnimated(applyButton, true, 1f, false, animated); + AndroidUtilities.updateViewVisibilityAnimated(applyTextView, true, 0.9f, false, animated); + AndroidUtilities.updateViewVisibilityAnimated(applySubTextView, showSubText, 0.9f, false, 0.7f, animated); + AndroidUtilities.updateViewVisibilityAnimated(themeHintTextView, true, 0.9f, false, animated); } else { backButtonDrawable.setRotation(1f, animated); applyButton.setEnabled(false); @@ -389,11 +433,54 @@ private void updateState(boolean animated) { AndroidUtilities.updateViewVisibilityAnimated(cancelOrResetTextView, true, 0.9f, false, animated); AndroidUtilities.updateViewVisibilityAnimated(applyButton, false, 1f, false, animated); AndroidUtilities.updateViewVisibilityAnimated(applyTextView, false, 0.9f, false, animated); + AndroidUtilities.updateViewVisibilityAnimated(applySubTextView, false, 0.9f, false, animated); AndroidUtilities.updateViewVisibilityAnimated(themeHintTextView, false, 0.9f, false, animated); } } } + private boolean checkingBoostsLevel = false, checkedBoostsLevel = false; + private TL_stories.TL_premium_boostsStatus boostsStatus; + private void checkBoostsLevel() { + if (chatActivity == null || checkingBoostsLevel || checkedBoostsLevel || boostsStatus != null) { + return; + } + checkingBoostsLevel = true; + chatActivity.getMessagesController().getBoostsController().getBoostsStats(chatActivity.getDialogId(), boostsStatus -> { + this.boostsStatus = boostsStatus; + checkedBoostsLevel = true; + updateState(true); + checkingBoostsLevel = false; + }); + } + + private float subTextTranslation = 0; + private ValueAnimator subTextTranslationAnimator; + private void updateApplySubTextTranslation(boolean subtextShown, boolean animated) { + if (subTextTranslationAnimator != null) { + subTextTranslationAnimator.cancel(); + subTextTranslationAnimator = null; + } + if (animated) { + subTextTranslationAnimator = ValueAnimator.ofFloat(subTextTranslation, subtextShown ? 1 : 0); + subTextTranslationAnimator.addUpdateListener(anm -> { + subTextTranslation = (float) anm.getAnimatedValue(); + applyTextView.setTranslationY(-dp(7) * subTextTranslation); + }); + subTextTranslationAnimator.addListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + subTextTranslation = subtextShown ? 1 : 0; + applyTextView.setTranslationY(-dp(7) * subTextTranslation); + } + }); + subTextTranslationAnimator.start(); + } else { + subTextTranslation = subtextShown ? 1 : 0; + applyTextView.setTranslationY(-dp(7) * subTextTranslation); + } + } + @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); @@ -433,7 +520,7 @@ public void onError(TLRPC.TL_error error) { hintView = new HintView(getContext(), 9, chatActivity.getResourceProvider()); hintView.setVisibility(View.INVISIBLE); hintView.setShowingDuration(5000); - hintView.setBottomOffset(-AndroidUtilities.dp(8)); + hintView.setBottomOffset(-dp(8)); if (forceDark) { hintView.setText(AndroidUtilities.replaceTags(LocaleController.formatString("ChatThemeDaySwitchTooltip", R.string.ChatThemeDaySwitchTooltip))); } else { @@ -538,7 +625,7 @@ public void onAnimationProgress(float progress) { } updateButtonColors(); if (chatAttachButton != null) { - chatAttachButton.setBackground(Theme.createSimpleSelectorRoundRectDrawable(AndroidUtilities.dp(0), getThemedColor(Theme.key_windowBackgroundWhite), ColorUtils.setAlphaComponent(getThemedColor(Theme.key_featuredStickers_addButton), (int) (0.3f * 255)))); + chatAttachButton.setBackground(Theme.createSimpleSelectorRoundRectDrawable(dp(0), getThemedColor(Theme.key_windowBackgroundWhite), ColorUtils.setAlphaComponent(getThemedColor(Theme.key_featuredStickers_addButton), (int) (0.3f * 255)))); } if (chatAttachButtonText != null) { chatAttachButtonText.setTextColor(getThemedColor(Theme.key_featuredStickers_addButton)); @@ -550,8 +637,8 @@ public void didSetColor() { } }; ArrayList themeDescriptions = new ArrayList<>(); - if (chatActivity.forceDisallowRedrawThemeDescriptions && overlayFragment != null) { - themeDescriptions.addAll(overlayFragment.getThemeDescriptionsInternal()); + if (chatActivity.forceDisallowRedrawThemeDescriptions && overlayFragment instanceof ThemePreviewActivity) { + themeDescriptions.addAll(((ThemePreviewActivity) overlayFragment).getThemeDescriptionsInternal()); return themeDescriptions; } if (chatAttachAlert != null) { @@ -833,6 +920,35 @@ private void setItemsAnimationProgress(float progress) { } private void applySelectedTheme() { + if (checkingBoostsLevel) { + return; + } + if (boostsStatus != null && boostsStatus.level < chatActivity.getMessagesController().channelWallpaperLevelMin) { + chatActivity.getMessagesController().getBoostsController().userCanBoostChannel(chatActivity.getDialogId(), boostsStatus, canApplyBoost -> { + if (getContext() == null) { + return; + } + LimitReachedBottomSheet limitReachedBottomSheet = new LimitReachedBottomSheet(chatActivity, getContext(), LimitReachedBottomSheet.TYPE_BOOSTS_FOR_WALLPAPER, currentAccount, resourcesProvider); + limitReachedBottomSheet.setCanApplyBoost(canApplyBoost); + limitReachedBottomSheet.setBoostsStats(boostsStatus, true); + limitReachedBottomSheet.setDialogId(chatActivity.getDialogId()); + limitReachedBottomSheet.showStatisticButtonInLink(() -> { + TLRPC.Chat chat = chatActivity.getMessagesController().getChat(-chatActivity.getDialogId()); + Bundle args = new Bundle(); + args.putLong("chat_id", -chatActivity.getDialogId()); + args.putBoolean("is_megagroup", chat.megagroup); + args.putBoolean("start_from_boosts", true); + TLRPC.ChatFull chatInfo = chatActivity.getMessagesController().getChatFull(-chatActivity.getDialogId()); + if (chatInfo == null || !chatInfo.can_view_stats) { + args.putBoolean("only_boosts", true); + }; + StatisticActivity fragment = new StatisticActivity(args); + showAsSheet(fragment); + }); + limitReachedBottomSheet.show(); + }); + return; + } Bulletin bulletin = null; EmojiThemes newTheme = selectedItem.chatTheme; if (selectedItem != null && newTheme != currentTheme) { @@ -1124,6 +1240,147 @@ public boolean isEnabled(RecyclerView.ViewHolder holder) { } } + public static void openGalleryForBackground( + Activity activity, + BaseFragment fragment, + long dialogId, + Theme.ResourcesProvider resourcesProvider, + Utilities.Callback onSet, + ThemePreviewActivity.DayNightSwitchDelegate toggleTheme, + TL_stories.TL_premium_boostsStatus cachedBoostsStatus + ) { + ChatAttachAlert chatAttachAlert = new ChatAttachAlert(activity, fragment, false, false, false, resourcesProvider); + chatAttachAlert.drawNavigationBar = true; + chatAttachAlert.setupPhotoPicker(LocaleController.getString("ChooseBackground", R.string.ChooseBackground)); + chatAttachAlert.setDelegate(new ChatAttachAlert.ChatAttachViewDelegate() { + long start; + @Override + public boolean selectItemOnClicking() { + start = System.currentTimeMillis(); + return true; + } + + @Override + public void didPressedButton(int button, boolean arg, boolean notify, int scheduleDate, boolean forceDocument) { + try { + HashMap photos = chatAttachAlert.getPhotoLayout().getSelectedPhotos(); + if (!photos.isEmpty()) { + MediaController.PhotoEntry entry = (MediaController.PhotoEntry) photos.values().iterator().next(); + String path; + if (entry.imagePath != null) { + path = entry.imagePath; + } else { + path = entry.path; + } + if (path != null) { + File currentWallpaperPath = new File(FileLoader.getDirectory(FileLoader.MEDIA_DIR_CACHE), Utilities.random.nextInt() + ".jpg"); + Point screenSize = AndroidUtilities.getRealScreenSize(); + Bitmap bitmap = ImageLoader.loadBitmap(path, null, screenSize.x, screenSize.y, true); + FileOutputStream stream = new FileOutputStream(currentWallpaperPath); + bitmap.compress(Bitmap.CompressFormat.JPEG, 87, stream); + + ThemePreviewActivity themePreviewActivity = new ThemePreviewActivity(new WallpapersListActivity.FileWallpaper("", currentWallpaperPath, currentWallpaperPath), bitmap) { + @Override + public boolean insideBottomSheet() { + return true; + } + }; + themePreviewActivity.boostsStatus = cachedBoostsStatus; + themePreviewActivity.setResourceProvider(resourcesProvider); + themePreviewActivity.setOnSwitchDayNightDelegate(toggleTheme); + themePreviewActivity.setInitialModes(false, false, .20f); + themePreviewActivity.setDialogId(dialogId); + themePreviewActivity.setDelegate(wallPaper -> { + chatAttachAlert.dismissInternal(); + if (onSet != null) { + onSet.run(wallPaper); + } + }); + BaseFragment.BottomSheetParams params = new BaseFragment.BottomSheetParams(); + params.transitionFromLeft = true; + params.allowNestedScroll = false; + params.occupyNavigationBar = true; + fragment.showAsSheet(themePreviewActivity, params); + + chatAttachAlert.dismiss(); + } + } + } catch (Throwable e) { + FileLog.e(e); + } + } + + @Override + public void onWallpaperSelected(Object object) { + ThemePreviewActivity wallpaperActivity = new ThemePreviewActivity(object, null, true, false) { + @Override + public boolean insideBottomSheet() { + return true; + } + }; + wallpaperActivity.boostsStatus = cachedBoostsStatus; + wallpaperActivity.setResourceProvider(resourcesProvider); + wallpaperActivity.setOnSwitchDayNightDelegate(toggleTheme); + wallpaperActivity.setDialogId(dialogId); + wallpaperActivity.setDelegate(wallPaper -> { + chatAttachAlert.dismissInternal(); + if (onSet != null) { + onSet.run(wallPaper); + } + }); + BaseFragment.BottomSheetParams params = new BaseFragment.BottomSheetParams(); + params.transitionFromLeft = true; + params.allowNestedScroll = false; + params.occupyNavigationBar = true; + fragment.showAsSheet(wallpaperActivity, params); + } + }); + chatAttachAlert.setMaxSelectedPhotos(1, false); + chatAttachAlert.init(); + chatAttachAlert.getPhotoLayout().loadGalleryPhotos(); + chatAttachAlert.show(); + + FrameLayout chatAttachButton = new FrameLayout(activity) { + + Paint paint = new Paint(); + + @Override + protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { + super.onMeasure(widthMeasureSpec, MeasureSpec.makeMeasureSpec(dp(48), MeasureSpec.EXACTLY)); + } + + @Override + protected void dispatchDraw(Canvas canvas) { + super.dispatchDraw(canvas); + paint.setColor(Theme.getColor(Theme.key_divider, resourcesProvider)); + canvas.drawRect(0, 0, getMeasuredWidth(), 1, paint); + } + }; + AnimatedTextView chatAttachButtonText = new AnimatedTextView(activity, true, true, true); + chatAttachButtonText.setTextSize(dp(14)); + chatAttachButtonText.setText(LocaleController.getString(R.string.SetColorAsBackground)); + chatAttachButtonText.setGravity(Gravity.CENTER); + chatAttachButtonText.setTextColor(Theme.getColor(Theme.key_featuredStickers_addButton, resourcesProvider)); + chatAttachButton.addView(chatAttachButtonText, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER)); + chatAttachButton.setBackground(Theme.createSimpleSelectorRoundRectDrawable(dp(0), Theme.getColor(Theme.key_windowBackgroundWhite, resourcesProvider), ColorUtils.setAlphaComponent(Theme.getColor(Theme.key_featuredStickers_addButton, resourcesProvider), (int) (0.3f * 255)))); + chatAttachButton.setOnClickListener(v -> { + if (chatAttachAlert.getCurrentAttachLayout() == chatAttachAlert.getPhotoLayout()) { + chatAttachButtonText.setText(LocaleController.getString(R.string.ChooseBackgroundFromGallery)); + chatAttachAlert.openColorsLayout(); +// chatAttachAlert.colorsLayout.updateColors(forceDark); + } else { + chatAttachButtonText.setText(LocaleController.getString(R.string.SetColorAsBackground)); + chatAttachAlert.showLayout(chatAttachAlert.getPhotoLayout()); + } +// WallpapersListActivity wallpapersListActivity = new WallpapersListActivity(WallpapersListActivity.TYPE_ALL, chatActivity.getDialogId()); +// chatActivity.presentFragment(wallpapersListActivity); +// chatAttachAlert.dismiss(); +// dismiss(); + }); + chatAttachAlert.sizeNotifierFrameLayout.addView(chatAttachButton, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.BOTTOM)); + } + + private void openGalleryForBackground() { chatAttachAlert = new ChatAttachAlert(chatActivity.getParentActivity(), chatActivity, false, false, false, chatActivity.getResourceProvider()); chatAttachAlert.drawNavigationBar = true; @@ -1161,9 +1418,10 @@ public boolean insideBottomSheet() { return true; } }; + themePreviewActivity.boostsStatus = boostsStatus; themePreviewActivity.setInitialModes(false, false, .20f); themePreviewActivity.setDialogId(chatActivity.getDialogId()); - themePreviewActivity.setDelegate(() -> { + themePreviewActivity.setDelegate(wallPaper -> { chatAttachAlert.dismissInternal(); dismiss(); }); @@ -1183,8 +1441,9 @@ public boolean insideBottomSheet() { return true; } }; + wallpaperActivity.boostsStatus = boostsStatus; wallpaperActivity.setDialogId(chatActivity.getDialogId()); - wallpaperActivity.setDelegate(() -> { + wallpaperActivity.setDelegate(wallPaper -> { chatAttachAlert.dismissInternal(); dismiss(); }); @@ -1202,7 +1461,7 @@ public boolean insideBottomSheet() { @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { - super.onMeasure(widthMeasureSpec, MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(48), MeasureSpec.EXACTLY)); + super.onMeasure(widthMeasureSpec, MeasureSpec.makeMeasureSpec(dp(48), MeasureSpec.EXACTLY)); } @Override @@ -1213,12 +1472,12 @@ protected void dispatchDraw(Canvas canvas) { } }; chatAttachButtonText = new AnimatedTextView(getContext(), true, true, true); - chatAttachButtonText.setTextSize(AndroidUtilities.dp(14)); + chatAttachButtonText.setTextSize(dp(14)); chatAttachButtonText.setText(LocaleController.getString("SetColorAsBackground", R.string.SetColorAsBackground)); chatAttachButtonText.setGravity(Gravity.CENTER); chatAttachButtonText.setTextColor(getThemedColor(Theme.key_featuredStickers_addButton)); chatAttachButton.addView(chatAttachButtonText, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER)); - chatAttachButton.setBackground(Theme.createSimpleSelectorRoundRectDrawable(AndroidUtilities.dp(0), getThemedColor(Theme.key_windowBackgroundWhite), ColorUtils.setAlphaComponent(getThemedColor(Theme.key_featuredStickers_addButton), (int) (0.3f * 255)))); + chatAttachButton.setBackground(Theme.createSimpleSelectorRoundRectDrawable(dp(0), getThemedColor(Theme.key_windowBackgroundWhite), ColorUtils.setAlphaComponent(getThemedColor(Theme.key_featuredStickers_addButton), (int) (0.3f * 255)))); chatAttachButton.setOnClickListener(v -> { if (chatAttachAlert.getCurrentAttachLayout() == chatAttachAlert.getPhotoLayout()) { chatAttachButtonText.setText(LocaleController.getString("ChooseBackgroundFromGallery", R.string.ChooseBackgroundFromGallery)); @@ -1261,6 +1520,27 @@ private void fixColorsAfterAnotherWindow() { } } + private void showAsSheet(BaseFragment fragment) { + if (fragment == null) { + return; + } + BaseFragment.BottomSheetParams params = new BaseFragment.BottomSheetParams(); + params.transitionFromLeft = true; + params.allowNestedScroll = false; + fragment.setResourceProvider(chatActivity.getResourceProvider()); + params.onOpenAnimationFinished = () -> { + PhotoViewer.getInstance().closePhoto(false, false); + }; + params.onPreFinished = () -> { + fixColorsAfterAnotherWindow(); + }; + params.onDismiss = () -> { + overlayFragment = null; + }; + params.occupyNavigationBar = true; + chatActivity.showAsSheet(overlayFragment = fragment, params); + } + private void showAsSheet(ThemePreviewActivity themePreviewActivity) { BaseFragment.BottomSheetParams params = new BaseFragment.BottomSheetParams(); params.transitionFromLeft = true; @@ -1274,6 +1554,11 @@ public boolean isDark() { return forceDark; } + @Override + public boolean supportsAnimation() { + return true; + } + @Override public void switchDayNight(boolean animated) { forceDark = !forceDark; @@ -1316,5 +1601,12 @@ public static class ChatThemeItem { public ChatThemeItem(EmojiThemes chatTheme) { this.chatTheme = chatTheme; } + + public String getEmoticon() { + if (chatTheme == null || chatTheme.showAsDefaultStub) { + return null; + } + return chatTheme.getEmoticon(); + } } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/DotDividerSpan.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/DotDividerSpan.java index b886854f120..b8e820c777c 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/DotDividerSpan.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/DotDividerSpan.java @@ -14,7 +14,7 @@ public class DotDividerSpan extends ReplacementSpan { Paint p = new Paint(Paint.ANTI_ALIAS_FLAG); int color; int topPadding; - private int size = 3; + private float size = 3; @Override public int getSize(@NonNull Paint paint, CharSequence charSequence, int i, int i1, @Nullable Paint.FontMetricsInt fontMetricsInt) { @@ -35,7 +35,7 @@ public void setTopPadding(int topPadding) { this.topPadding = topPadding; } - public void setSize(int size) { + public void setSize(float size) { this.size = size; } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/EditTextEmoji.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/EditTextEmoji.java index 8af807ab9be..26c4620d783 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/EditTextEmoji.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/EditTextEmoji.java @@ -231,7 +231,7 @@ protected void onSelectionChanged(int selStart, int selEnd) { editText.setHintTextColor(getThemedColor(Theme.key_windowBackgroundWhiteHintText)); editText.setTextColor(getThemedColor(Theme.key_windowBackgroundWhiteBlackText)); editText.setHandlesColor(getThemedColor(Theme.key_chat_TextSelectionCursor)); - editText.setPadding(LocaleController.isRTL ? AndroidUtilities.dp(40) : 0, 0, LocaleController.isRTL ? 0 : AndroidUtilities.dp(40), AndroidUtilities.dp(8)); + editText.setPadding(LocaleController.isRTL ? AndroidUtilities.dp(40) : 0, 0, LocaleController.isRTL ? 0 : AndroidUtilities.dp(40), AndroidUtilities.dp(11)); addView(editText, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.LEFT | Gravity.CENTER_VERTICAL, LocaleController.isRTL ? 11 : 0, 1, LocaleController.isRTL ? 0 : 11, 0)); } else if (style == STYLE_STORY || style == STYLE_PHOTOVIEWER) { editText.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 16); @@ -276,7 +276,7 @@ protected void dispatchDraw(Canvas canvas) { if (style == STYLE_FRAGMENT) { emojiIconDrawable.setColorFilter(new PorterDuffColorFilter(getThemedColor(Theme.key_chat_messagePanelIcons), PorterDuff.Mode.MULTIPLY)); emojiIconDrawable.setIcon(R.drawable.smiles_tab_smiles, false); - addView(emojiButton, LayoutHelper.createFrame(48, 48, Gravity.CENTER_VERTICAL | (LocaleController.isRTL ? Gravity.LEFT : Gravity.RIGHT), 0, 0, 0, 7)); + addView(emojiButton, LayoutHelper.createFrame(48, 48, Gravity.CENTER_VERTICAL | (LocaleController.isRTL ? Gravity.LEFT : Gravity.RIGHT), 0, 0, 0, 5)); } else if (style == STYLE_STORY || style == STYLE_PHOTOVIEWER) { emojiIconDrawable.setColorFilter(new PorterDuffColorFilter(0x8cffffff, PorterDuff.Mode.MULTIPLY)); emojiIconDrawable.setIcon(R.drawable.input_smile, false); @@ -367,10 +367,11 @@ public void didReceivedNotification(int id, int account, Object... args) { public void setEnabled(boolean enabled) { editText.setEnabled(enabled); emojiButton.setVisibility(enabled ? VISIBLE : GONE); + int bottomPadding = AndroidUtilities.dp(currentStyle == STYLE_FRAGMENT ? 11 : 8); if (enabled) { - editText.setPadding(LocaleController.isRTL ? AndroidUtilities.dp(40) : 0, 0, LocaleController.isRTL ? 0 : AndroidUtilities.dp(40), AndroidUtilities.dp(8)); + editText.setPadding(LocaleController.isRTL ? AndroidUtilities.dp(40) : 0, 0, LocaleController.isRTL ? 0 : AndroidUtilities.dp(40), bottomPadding); } else { - editText.setPadding(0, 0, 0, AndroidUtilities.dp(8)); + editText.setPadding(0, 0, 0, bottomPadding); } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/EmojiView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/EmojiView.java index b0f45edd6f5..9b4ed058932 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/EmojiView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/EmojiView.java @@ -5774,7 +5774,7 @@ private void checkDocuments(boolean isGif) { } } } - if (MessagesController.getInstance(currentAccount).premiumLocked) { + if (MessagesController.getInstance(currentAccount).premiumFeaturesBlocked()) { for (int a = 0; a < favouriteStickers.size(); a++) { if (MessageObject.isPremiumSticker(favouriteStickers.get(a))) { favouriteStickers.remove(a); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/FilterTabsView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/FilterTabsView.java index 5c53c20cbfd..abcc5a92a97 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/FilterTabsView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/FilterTabsView.java @@ -1706,7 +1706,7 @@ public boolean isLongPressDragEnabled() { @Override public int getMovementFlags(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) { - if (MessagesController.getInstance(UserConfig.selectedAccount).premiumLocked && (!isEditing || (viewHolder.getAdapterPosition() == 0 && tabs.get(0).isDefault && !UserConfig.getInstance(UserConfig.selectedAccount).isPremium()))) { + if (MessagesController.getInstance(UserConfig.selectedAccount).premiumFeaturesBlocked() && (!isEditing || (viewHolder.getAdapterPosition() == 0 && tabs.get(0).isDefault && !UserConfig.getInstance(UserConfig.selectedAccount).isPremium()))) { return makeMovementFlags(0, 0); } return makeMovementFlags(ItemTouchHelper.LEFT | ItemTouchHelper.RIGHT, 0); @@ -1714,7 +1714,7 @@ public int getMovementFlags(RecyclerView recyclerView, RecyclerView.ViewHolder v @Override public boolean onMove(RecyclerView recyclerView, RecyclerView.ViewHolder source, RecyclerView.ViewHolder target) { - if (MessagesController.getInstance(UserConfig.selectedAccount).premiumLocked && ((source.getAdapterPosition() == 0 || target.getAdapterPosition() == 0) && !UserConfig.getInstance(UserConfig.selectedAccount).isPremium())) { + if (MessagesController.getInstance(UserConfig.selectedAccount).premiumFeaturesBlocked() && ((source.getAdapterPosition() == 0 || target.getAdapterPosition() == 0) && !UserConfig.getInstance(UserConfig.selectedAccount).isPremium())) { return false; } adapter.swapElements(source.getAdapterPosition(), target.getAdapterPosition()); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/FireworksOverlay.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/FireworksOverlay.java index 917795a897f..d285a80d65c 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/FireworksOverlay.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/FireworksOverlay.java @@ -32,11 +32,13 @@ public class FireworksOverlay extends View { private float speedCoef = 1.0f; private int fallingDownCount; private static Drawable[] heartDrawable; + private static Drawable[] starsDrawable; private static final int particlesCount = SharedConfig.getDevicePerformanceClass() == SharedConfig.PERFORMANCE_CLASS_LOW ? 50 : 60; private static final int fallParticlesCount = SharedConfig.getDevicePerformanceClass() == SharedConfig.PERFORMANCE_CLASS_LOW ? 20 : 30; private boolean isFebruary14; + private boolean withStars; - private static int[] colors = new int[] { + private static int[] colors = new int[]{ 0xff2CBCE8, 0xff9E04D0, 0xffFECB02, @@ -45,7 +47,7 @@ public class FireworksOverlay extends View { 0xff59B86C }; - private static int[] heartColors = new int[] { + private static int[] heartColors = new int[]{ 0xffE2557B, 0xff5FCDF2, 0xffFFDA69, @@ -53,6 +55,14 @@ public class FireworksOverlay extends View { 0xffE376B0 }; + private static int[] starsColors = new int[]{ + 0xff1e80ff, + 0xff10c689, + 0xffff5997, + 0xffff9724, + 0xff2fe1f9 + }; + static { paint = new Paint[colors.length]; for (int a = 0; a < paint.length; a++) { @@ -85,15 +95,23 @@ private void draw(Canvas canvas) { canvas.drawRoundRect(rect, AndroidUtilities.dp(2), AndroidUtilities.dp(2), paint[colorType]); canvas.restore(); } else if (type == 2) { - Drawable drawable = heartDrawable[colorType]; - int w = drawable.getIntrinsicWidth() / 2; - int h = drawable.getIntrinsicHeight() / 2; - drawable.setBounds((int) x - w, (int) y - h, (int) x + w, (int) y + h); - canvas.save(); - canvas.rotate(rotation, x, y); - canvas.scale(typeSize / 6.0f, typeSize / 6.0f, x, y); - drawable.draw(canvas); - canvas.restore(); + Drawable drawable = null; + if (starsDrawable != null) { + drawable = starsDrawable[colorType]; + } + if (heartDrawable != null) { + drawable = heartDrawable[colorType]; + } + if (drawable != null) { + int w = drawable.getIntrinsicWidth() / 2; + int h = drawable.getIntrinsicHeight() / 2; + drawable.setBounds((int) x - w, (int) y - h, (int) x + w, (int) y + h); + canvas.save(); + canvas.rotate(rotation, x, y); + canvas.scale(typeSize / 6.0f, typeSize / 6.0f, x, y); + drawable.draw(canvas); + canvas.restore(); + } } } @@ -170,16 +188,27 @@ private void loadHeartDrawables() { } } + private void loadStarsDrawables() { + if (starsDrawable != null) { + return; + } + starsDrawable = new Drawable[starsColors.length]; + for (int a = 0; a < starsDrawable.length; a++) { + starsDrawable[a] = ApplicationLoader.applicationContext.getResources().getDrawable(R.drawable.msg_settings_premium).mutate(); + starsDrawable[a].setColorFilter(new PorterDuffColorFilter(starsColors[a], PorterDuff.Mode.MULTIPLY)); + } + } + private int getHeightForAnimation() { if (getMeasuredHeight() == 0) { - return ((View)getParent()).getHeight(); + return ((View) getParent()).getHeight(); } return getMeasuredHeight(); } private int getWidthForAnimation() { if (getMeasuredWidth() == 0) { - return ((View)getParent()).getWidth(); + return ((View) getParent()).getWidth(); } return getMeasuredWidth(); } @@ -190,6 +219,9 @@ private Particle createParticle(boolean fall) { if (isFebruary14 && particle.type == 0) { particle.type = 2; particle.colorType = (byte) Utilities.random.nextInt(heartColors.length); + } else if (withStars && Utilities.random.nextBoolean()) { + particle.type = 2; + particle.colorType = (byte) Utilities.random.nextInt(starsColors.length); } else { particle.colorType = (byte) Utilities.random.nextInt(colors.length); } @@ -223,11 +255,10 @@ public boolean isStarted() { return started; } - public void start() { + public void start(boolean withStars) { + this.withStars = withStars; particles.clear(); - if (Build.VERSION.SDK_INT >= 18) { - setLayerType(View.LAYER_TYPE_HARDWARE, null); - } + setLayerType(View.LAYER_TYPE_HARDWARE, null); started = true; startedFall = false; fallingDownCount = 0; @@ -239,6 +270,8 @@ public void start() { isFebruary14 = month == 1 && (BuildVars.DEBUG_PRIVATE_VERSION || day == 14); if (isFebruary14) { loadHeartDrawables(); + } else if (withStars) { + loadStarsDrawables(); } for (int a = 0; a < particlesCount; a++) { particles.add(createParticle(false)); @@ -246,6 +279,10 @@ public void start() { invalidate(); } + public void start() { + start(false); + } + private void startFall() { if (startedFall) { return; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/FolderBottomSheet.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/FolderBottomSheet.java index 742895ed1b9..a4a4bce6fa6 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/FolderBottomSheet.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/FolderBottomSheet.java @@ -115,6 +115,9 @@ public static void showForDeletion(final BaseFragment fragment, final int filter req.chatlist = new TL_chatlists.TL_inputChatlistDialogFilter(); req.chatlist.filter_id = filterId; fragment.getConnectionsManager().sendRequest(req, (res, err) -> AndroidUtilities.runOnUIThread(() -> { + if (fragment.getParentActivity() == null) { + return; + } FolderBottomSheet sheet; if (res instanceof TLRPC.Vector) { ArrayList suggestions = new ArrayList<>(); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/InstantCameraView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/InstantCameraView.java index bb99aa76dc1..6d6afa2ba22 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/InstantCameraView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/InstantCameraView.java @@ -650,7 +650,7 @@ public void onSurfaceTextureSizeChanged(SurfaceTexture surface, final int width, @Override public boolean onSurfaceTextureDestroyed(SurfaceTexture surface) { if (cameraThread != null) { - cameraThread.shutdown(0); + cameraThread.shutdown(0, 0); cameraThread = null; } if (cameraSession != null) { @@ -765,7 +765,7 @@ public void changeVideoPreviewState(int state, float progress) { } } - public void send(int state, boolean notify, int scheduleDate) { + public void send(int state, boolean notify, int scheduleDate, int ttl) { if (textureView == null) { return; } @@ -803,7 +803,9 @@ public void send(int state, boolean notify, int scheduleDate) { videoEditedInfo.encryptedFile = encryptedFile; videoEditedInfo.key = key; videoEditedInfo.iv = iv; - delegate.sendMedia(new MediaController.PhotoEntry(0, 0, 0, cameraFile.getAbsolutePath(), 0, true, 0, 0, 0), videoEditedInfo, notify, scheduleDate, false); + MediaController.PhotoEntry entry = new MediaController.PhotoEntry(0, 0, 0, cameraFile.getAbsolutePath(), 0, true, 0, 0, 0); + entry.ttl = ttl; + delegate.sendMedia(entry, videoEditedInfo, notify, scheduleDate, false); if (scheduleDate != 0) { startAnimation(false); } @@ -828,7 +830,7 @@ public void send(int state, boolean notify, int scheduleDate) { send = 1; } saveLastCameraBitmap(); - cameraThread.shutdown(send); + cameraThread.shutdown(send, ttl); cameraThread = null; } if (cancelled) { @@ -871,7 +873,7 @@ public void cancel(boolean byGesture) { NotificationCenter.getInstance(currentAccount).postNotificationName(NotificationCenter.recordStopped, recordingGuid, byGesture ? 0 : 6); if (cameraThread != null) { saveLastCameraBitmap(); - cameraThread.shutdown(0); + cameraThread.shutdown(0, 0); cameraThread = null; } if (cameraFile != null) { @@ -1562,7 +1564,7 @@ public void handleMessage(Message inputMessage) { case DO_SHUTDOWN_MESSAGE: finish(); if (recording) { - videoEncoder.stopRecording(inputMessage.arg1); + videoEncoder.stopRecording(inputMessage.arg1, inputMessage.arg2); } Looper looper = Looper.myLooper(); if (looper != null) { @@ -1636,10 +1638,10 @@ public void handleMessage(Message inputMessage) { } } - public void shutdown(int send) { + public void shutdown(int send, int ttl) { Handler handler = getHandler(); if (handler != null) { - sendMessage(handler.obtainMessage(DO_SHUTDOWN_MESSAGE, send, 0), 0); + sendMessage(handler.obtainMessage(DO_SHUTDOWN_MESSAGE, send, ttl), 0); } } @@ -1682,7 +1684,7 @@ public void handleMessage(Message inputMessage) { encoder.prepareEncoder(); } catch (Exception e) { FileLog.e(e); - encoder.handleStopRecording(0); + encoder.handleStopRecording(0, 0); Looper.myLooper().quit(); } break; @@ -1691,7 +1693,7 @@ public void handleMessage(Message inputMessage) { if (BuildVars.LOGS_ENABLED) { FileLog.e("InstantCamera stop encoder"); } - encoder.handleStopRecording(inputMessage.arg1); + encoder.handleStopRecording(inputMessage.arg1, inputMessage.arg2); break; } case MSG_VIDEOFRAME_AVAILABLE: { @@ -1777,6 +1779,7 @@ private class VideoRecorder implements Runnable { private boolean ready; private volatile boolean running; private volatile int sendWhenDone; + private volatile int sendWhenDoneTTL; private long skippedTime; private boolean skippedFirst; @@ -1901,7 +1904,7 @@ public void run() { } catch (Exception e) { FileLog.e(e); } - handler.sendMessage(handler.obtainMessage(MSG_STOP_RECORDING, sendWhenDone, 0)); + handler.sendMessage(handler.obtainMessage(MSG_STOP_RECORDING, sendWhenDone, sendWhenDoneTTL)); } }; @@ -1950,8 +1953,8 @@ public void startRecording(File outputFile, android.opengl.EGLContext sharedCont handler.sendMessage(handler.obtainMessage(MSG_START_RECORDING)); } - public void stopRecording(int send) { - handler.sendMessage(handler.obtainMessage(MSG_STOP_RECORDING, send, 0)); + public void stopRecording(int send, int ttl) { + handler.sendMessage(handler.obtainMessage(MSG_STOP_RECORDING, send, ttl)); AndroidUtilities.runOnUIThread(() -> { NotificationCenter.getInstance(currentAccount).postNotificationName(NotificationCenter.stopAllHeavyOperations, 512); }); @@ -2279,10 +2282,11 @@ public void run() { } } - private void handleStopRecording(final int send) { + private void handleStopRecording(final int send, final int ttl) { if (running) { FileLog.d("InstantCamera handleStopRecording running=false"); sendWhenDone = send; + sendWhenDoneTTL = ttl; running = false; return; } @@ -2374,13 +2378,17 @@ private void handleStopRecording(final int send) { if (send == 1) { if (delegate.isInScheduleMode()) { AlertsCreator.createScheduleDatePickerDialog(delegate.getParentActivity(), delegate.getDialogId(), (notify, scheduleDate) -> { - delegate.sendMedia(new MediaController.PhotoEntry(0, 0, 0, videoFile.getAbsolutePath(), 0, true, 0, 0, 0), videoEditedInfo, notify, scheduleDate, false); + MediaController.PhotoEntry entry = new MediaController.PhotoEntry(0, 0, 0, videoFile.getAbsolutePath(), 0, true, 0, 0, 0); + entry.ttl = ttl; + delegate.sendMedia(entry, videoEditedInfo, notify, scheduleDate, false); startAnimation(false); }, () -> { startAnimation(false); }, resourcesProvider); } else { - delegate.sendMedia(new MediaController.PhotoEntry(0, 0, 0, videoFile.getAbsolutePath(), 0, true, 0, 0, 0), videoEditedInfo, true, 0, false); + MediaController.PhotoEntry entry = new MediaController.PhotoEntry(0, 0, 0, videoFile.getAbsolutePath(), 0, true, 0, 0, 0); + entry.ttl = ttl; + delegate.sendMedia(entry, videoEditedInfo, true, 0, false); } } else { videoPlayer = new VideoPlayer(); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/MediaActionDrawable.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/MediaActionDrawable.java index efe52e57a42..7e49a8f61a3 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/MediaActionDrawable.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/MediaActionDrawable.java @@ -39,9 +39,9 @@ public class MediaActionDrawable extends Drawable { public static final int ICON_UPDATE = 15; private TextPaint textPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG); - private Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG); + public Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG); private Paint backPaint = new Paint(Paint.ANTI_ALIAS_FLAG); - private Paint paint2 = new Paint(Paint.ANTI_ALIAS_FLAG); + public Paint paint2 = new Paint(Paint.ANTI_ALIAS_FLAG); private Paint paint3 = new Paint(Paint.ANTI_ALIAS_FLAG); private RectF rect = new RectF(); private ColorFilter colorFilter; @@ -274,7 +274,7 @@ public void invalidateSelf() { } } - private void applyShaderMatrix(boolean path) { + public void applyShaderMatrix(boolean path) { if (messageDrawable != null && messageDrawable.hasGradient() && !hasOverlayImage) { android.graphics.Rect bounds = getBounds(); Shader shader = messageDrawable.getGradientShader(); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/MenuToItemOptions.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/MenuToItemOptions.java index fa892ea4ef6..c388ecd0e5d 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/MenuToItemOptions.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/MenuToItemOptions.java @@ -30,7 +30,7 @@ public MenuToItemOptions(@NonNull ItemOptions itemOptions, @NonNull Utilities.Ca @Override public MenuItem add(int groupId, int itemId, int order, CharSequence title) { - if (premiumLock != null && FloatingToolbar.premiumOptions.contains(itemId) && MessagesController.getInstance(UserConfig.selectedAccount).premiumLocked) { + if (premiumLock != null && FloatingToolbar.premiumOptions.contains(itemId) && MessagesController.getInstance(UserConfig.selectedAccount).premiumFeaturesBlocked()) { return null; } itemOptions.add(title, () -> onMenuClicked.run(itemId)); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/MessagePreviewView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/MessagePreviewView.java index 6fad4fcf071..a4c57f54058 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/MessagePreviewView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/MessagePreviewView.java @@ -910,7 +910,7 @@ public void updateBackground() { menu.addView(btn1, LayoutHelper.createLinear(LayoutHelper.MATCH_PARENT, 48)); } - if (!messagePreviewParams.noforwards) { + if (!messagePreviewParams.noforwards && !messagePreviewParams.hasSecretMessages) { FrameLayout btn2 = new FrameLayout(context); replyAnotherChatButton = new ActionBarMenuSubItem(context, true, false, false, resourcesProvider); replyAnotherChatButton.setTextAndIcon(LocaleController.getString(R.string.ReplyToAnotherChat), R.drawable.msg_forward_replace); @@ -923,7 +923,7 @@ public void updateBackground() { menu.addView(btn2, LayoutHelper.createLinear(LayoutHelper.MATCH_PARENT, 48)); } - if (!messagePreviewParams.isSecret) { + if (!messagePreviewParams.noforwards && !messagePreviewParams.hasSecretMessages) { ActionBarPopupWindow.GapView gap2 = new ActionBarPopupWindow.GapView(context, resourcesProvider); gap2.setTag(R.id.fit_width_tag, 1); menu.addView(gap2, LayoutHelper.createLinear(LayoutHelper.MATCH_PARENT, 8)); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/MotionBackgroundDrawable.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/MotionBackgroundDrawable.java index 4cade217721..de8d17479b9 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/MotionBackgroundDrawable.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/MotionBackgroundDrawable.java @@ -58,7 +58,7 @@ public class MotionBackgroundDrawable extends Drawable { private int translationY; - private boolean isPreview; + public boolean isPreview; public float posAnimationProgress = 1.0f; private int phase; @@ -115,6 +115,8 @@ public class MotionBackgroundDrawable extends Drawable { private float indeterminateSpeedScale = 1f; private boolean isIndeterminateAnimation; private Paint overrideBitmapPaint; + private int bitmapWidth = 60; + private int bitmapHeight = 80; public MotionBackgroundDrawable() { super(); @@ -122,11 +124,19 @@ public MotionBackgroundDrawable() { } public MotionBackgroundDrawable(int c1, int c2, int c3, int c4, boolean preview) { - this(c1, c2, c3 ,c4, 0, preview); + this(c1, c2, c3, c4, 0, preview); } public MotionBackgroundDrawable(int c1, int c2, int c3, int c4, int rotation, boolean preview) { + this(c1, c2, c3, c4, rotation, preview, false); + } + + public MotionBackgroundDrawable(int c1, int c2, int c3, int c4, int rotation, boolean preview, boolean square) { super(); + if (square) { + bitmapWidth = 80; + bitmapHeight = 80; + } isPreview = preview; setColors(c1, c2, c3, c4, rotation, false); init(); @@ -134,13 +144,13 @@ public MotionBackgroundDrawable(int c1, int c2, int c3, int c4, int rotation, bo @SuppressLint("NewApi") private void init() { - currentBitmap = Bitmap.createBitmap(60, 80, Bitmap.Config.ARGB_8888); + currentBitmap = Bitmap.createBitmap(bitmapWidth, bitmapHeight, Bitmap.Config.ARGB_8888); for (int i = 0; i < ANIMATION_CACHE_BITMAPS_COUNT; i++) { - gradientToBitmap[i] = Bitmap.createBitmap(60, 80, Bitmap.Config.ARGB_8888); + gradientToBitmap[i] = Bitmap.createBitmap(bitmapWidth, bitmapHeight, Bitmap.Config.ARGB_8888); } gradientCanvas = new Canvas(currentBitmap); - gradientFromBitmap = Bitmap.createBitmap(60, 80, Bitmap.Config.ARGB_8888); + gradientFromBitmap = Bitmap.createBitmap(bitmapWidth, bitmapHeight, Bitmap.Config.ARGB_8888); gradientFromCanvas = new Canvas(gradientFromBitmap); Utilities.generateGradient(currentBitmap, true, phase, interpolator.getInterpolation(posAnimationProgress), currentBitmap.getWidth(), currentBitmap.getHeight(), currentBitmap.getRowBytes(), colors); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/EntityView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/EntityView.java index 0798d2a61d4..0ac98ccf361 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/EntityView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/EntityView.java @@ -64,6 +64,9 @@ default void onEntityDragMultitouchEnd() {} default void onEntityDragEnd(boolean delete) {} default void onEntityDragTrash(boolean enter) {} default void onEntityHandleTouched() {} + default boolean isEntityDeletable() { + return true; + } } private float previousLocationX, previousLocationY; @@ -223,7 +226,11 @@ private boolean onTouchMove(float x1, float y1, boolean multitouch, float x2, fl delegate.onEntityDraggedBottom(position.y + getHeight() / 2f * scale > ((View) getParent()).getHeight() - dp(64 + 50)); } - updateTrash(!multitouch && MathUtils.distance(x, y, ((View) getParent()).getWidth() / 2f, ((View) getParent()).getHeight() - dp(76)) < dp(32)); + updateTrash( + (delegate == null || delegate.isEntityDeletable()) && + !multitouch && + MathUtils.distance(x, y, ((View) getParent()).getWidth() / 2f, ((View) getParent()).getHeight() - dp(76)) < dp(32) + ); bounce.setPressed(false); @@ -709,7 +716,7 @@ private void rotateInternal(float angle) { updateSelectionView(); } - protected Rect getSelectionBounds() { + public Rect getSelectionBounds() { return new Rect(0, 0, 0, 0); } @@ -1006,9 +1013,13 @@ public boolean trashCenter() { return false; } + protected float getBounceScale() { + return .04f; + } + @Override protected void dispatchDraw(Canvas canvas) { - final float scale = bounce.getScale(.05f); + final float scale = bounce.getScale(getBounceScale()); canvas.save(); canvas.scale(scale, scale, getWidth() / 2f, getHeight() / 2f); if (getParent() instanceof View) { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/LPhotoPaintView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/LPhotoPaintView.java index 564526e7dad..38aff838f80 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/LPhotoPaintView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/LPhotoPaintView.java @@ -6,7 +6,9 @@ import android.annotation.SuppressLint; import android.app.Activity; import android.content.Context; +import android.content.DialogInterface; import android.graphics.Bitmap; +import android.graphics.BitmapFactory; import android.graphics.Canvas; import android.graphics.Color; import android.graphics.Matrix; @@ -17,6 +19,8 @@ import android.graphics.Rect; import android.graphics.SweepGradient; import android.graphics.drawable.GradientDrawable; +import android.location.Address; +import android.location.Geocoder; import android.os.Build; import android.os.Looper; import android.text.Layout; @@ -24,6 +28,7 @@ import android.text.Spanned; import android.text.TextUtils; import android.util.Log; +import android.util.Pair; import android.util.SparseArray; import android.util.TypedValue; import android.view.Gravity; @@ -48,12 +53,14 @@ import com.google.android.gms.vision.face.FaceDetector; import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.ApplicationLoader; import org.telegram.messenger.Bitmaps; import org.telegram.messenger.BuildVars; import org.telegram.messenger.DispatchQueue; import org.telegram.messenger.Emoji; import org.telegram.messenger.FileLoader; import org.telegram.messenger.FileLog; +import org.telegram.messenger.ImageLoader; import org.telegram.messenger.LocaleController; import org.telegram.messenger.MediaController; import org.telegram.messenger.MessageObject; @@ -65,15 +72,18 @@ import org.telegram.messenger.Utilities; import org.telegram.messenger.VideoEditedInfo; import org.telegram.tgnet.TLRPC; +import org.telegram.tgnet.tl.TL_stories; import org.telegram.ui.ActionBar.ActionBar; import org.telegram.ui.ActionBar.ActionBarPopupWindow; import org.telegram.ui.ActionBar.AdjustPanLayoutHelper; import org.telegram.ui.ActionBar.AlertDialog; import org.telegram.ui.ActionBar.Theme; import org.telegram.ui.BubbleActivity; +import org.telegram.ui.ChatActivity; import org.telegram.ui.Components.AnimatedEmojiDrawable; import org.telegram.ui.Components.AnimatedEmojiSpan; import org.telegram.ui.Components.ChatActivityEnterViewAnimatedIconView; +import org.telegram.ui.Components.ChatAttachAlert; import org.telegram.ui.Components.CubicBezierInterpolator; import org.telegram.ui.Components.EmojiView; import org.telegram.ui.Components.IPhotoPaintView; @@ -93,12 +103,19 @@ import org.telegram.ui.Components.SizeNotifierFrameLayout; import org.telegram.ui.Components.SizeNotifierFrameLayoutPhoto; import org.telegram.ui.Components.StickerMasksAlert; +import org.telegram.ui.Components.ThanosEffect; import org.telegram.ui.PhotoViewer; import org.telegram.ui.Stories.recorder.EmojiBottomSheet; +import org.telegram.ui.Stories.recorder.PaintView; +import org.telegram.ui.ThemePreviewActivity; +import org.telegram.ui.WallpapersListActivity; +import java.io.File; +import java.io.FileOutputStream; import java.math.BigInteger; import java.util.ArrayList; import java.util.Arrays; +import java.util.HashMap; import java.util.List; public class LPhotoPaintView extends SizeNotifierFrameLayoutPhoto implements IPhotoPaintView, PaintToolsView.Delegate, EntityView.EntityViewDelegate, PaintTextOptionsView.Delegate, SizeNotifierFrameLayout.SizeNotifierFrameLayoutDelegate, NotificationCenter.NotificationCenterDelegate { @@ -129,6 +146,7 @@ public class LPhotoPaintView extends SizeNotifierFrameLayoutPhoto implements IPh private View renderInputView; private FrameLayout selectionContainerView; private EntitiesContainerView entitiesView; + private ThanosEffect thanosEffect; private FrameLayout topLayout; public FrameLayout bottomLayout; public FrameLayout overlayLayout; @@ -440,7 +458,7 @@ protected void onDraw(Canvas canvas) { for (int a = 0, N = entities.size(); a < N; a++) { VideoEditedInfo.MediaEntity entity = entities.get(a); EntityView view; - if (entity.type == 0) { + if (entity.type == VideoEditedInfo.MediaEntity.TYPE_STICKER) { StickerView stickerView = createSticker(entity.parentObject, entity.document, false); if ((entity.subType & 2) != 0) { stickerView.mirror(); @@ -449,7 +467,7 @@ protected void onDraw(Canvas canvas) { ViewGroup.LayoutParams layoutParams = view.getLayoutParams(); layoutParams.width = entity.viewWidth; layoutParams.height = entity.viewHeight; - } else if (entity.type == 1) { + } else if (entity.type == VideoEditedInfo.MediaEntity.TYPE_TEXT) { TextPaintView textPaintView = createText(false); textPaintView.setType(entity.subType); textPaintView.setTypeface(entity.textTypeface); @@ -474,6 +492,19 @@ protected void onDraw(Canvas canvas) { swatch.color = entity.color; textPaintView.setSwatch(swatch); view = textPaintView; + } else if (entity.type == VideoEditedInfo.MediaEntity.TYPE_PHOTO) { + PhotoView photoView = createPhoto(entity.text, false); + photoView.preloadSegmented(entity.segmentedPath); + if ((entity.subType & 2) != 0) { + photoView.mirror(); + } + if ((entity.subType & 16) != 0) { + photoView.toggleSegmented(false); + } + view = photoView; + ViewGroup.LayoutParams layoutParams = view.getLayoutParams(); + layoutParams.width = entity.viewWidth; + layoutParams.height = entity.viewHeight; } else { continue; } @@ -1371,7 +1402,7 @@ private void openStickersView() { EmojiBottomSheet emojiBottomSheet = new EmojiBottomSheet(getContext(), false, resourcesProvider) { @Override public boolean canShowWidget(Integer id) { - return false; + return id == WIDGET_PHOTO; } }; emojiBottomSheet.whenDocumentSelected((parentObject, document, isGif) -> { @@ -1380,6 +1411,11 @@ public boolean canShowWidget(Integer id) { stickerView.setScale(1.5f); } }); + emojiBottomSheet.whenWidgetSelected(widget -> { + if (widget == EmojiBottomSheet.WIDGET_PHOTO) { + showPhotoAlert(); + } + }); emojiBottomSheet.setOnDismissListener(di -> { onOpenCloseStickersAlert(false); switchTab(wasSelectedIndex); @@ -1388,6 +1424,132 @@ public boolean canShowWidget(Integer id) { onOpenCloseStickersAlert(true); } + private void showPhotoAlert() { + ChatAttachAlert chatAttachAlert = new ChatAttachAlert(getContext(), new ChatActivity(null) { + @Override + public long getDialogId() { + return 0; + } + + @Override + public Theme.ResourcesProvider getResourceProvider() { + return resourcesProvider; + } + + @Override + public boolean isKeyboardVisible() { + return false; + } + + @Override + public Activity getParentActivity() { + return AndroidUtilities.findActivity(LPhotoPaintView.this.getContext()); + } + + @Override + public TLRPC.User getCurrentUser() { + return UserConfig.getInstance(currentAccount).getCurrentUser(); + } + + @Override + public boolean isLightStatusBar() { + return false; + } + + }, false, false, false, resourcesProvider); + chatAttachAlert.drawNavigationBar = true; + chatAttachAlert.setupPhotoPicker(LocaleController.getString(R.string.AddImage)); + chatAttachAlert.setDelegate(new ChatAttachAlert.ChatAttachViewDelegate() { + long start; + @Override + public boolean selectItemOnClicking() { + start = System.currentTimeMillis(); + return true; + } + + @Override + public void didPressedButton(int button, boolean arg, boolean notify, int scheduleDate, boolean forceDocument) { + try { + HashMap photos = chatAttachAlert.getPhotoLayout().getSelectedPhotos(); + if (!photos.isEmpty()) { + MediaController.PhotoEntry entry = (MediaController.PhotoEntry) photos.values().iterator().next(); + String path; + if (entry.imagePath != null) { + path = entry.imagePath; + } else { + path = entry.path; + } + appearAnimation(createPhoto(path, true)); + chatAttachAlert.dismiss(); + } + } catch (Throwable e) { + FileLog.e(e); + } + } + }); + chatAttachAlert.setOnDismissListener(dialog -> { + MediaController.forceBroadcastNewPhotos = false; + }); + chatAttachAlert.setMaxSelectedPhotos(1, false); + chatAttachAlert.init(); + MediaController.forceBroadcastNewPhotos = true; + chatAttachAlert.getPhotoLayout().loadGalleryPhotos(); + chatAttachAlert.show(); + } + + public void appearAnimation(View view) { + float scaleX = view.getScaleX(), scaleY = view.getScaleY(); + view.setScaleX(scaleX * .5f); + view.setScaleY(scaleY * .5f); + view.setAlpha(0f); + view.animate().scaleX(scaleX).scaleY(scaleY).alpha(1f).setInterpolator(new OvershootInterpolator(3f)).setDuration(240).withEndAction(() -> { + if (view instanceof EntityView) { + ((EntityView) view).updateSelectionView(); + selectEntity((EntityView) view); + } + }).start(); + } + + public PhotoView createPhoto(String path, boolean select) { + Size size = basePhotoSize(path); + Pair orientation = AndroidUtilities.getImageOrientation(path); + if ((orientation.first / 90 % 2) == 1) { + float w = size.width; + size.width = size.height; + size.height = w; + } + PhotoView view = new PhotoView(getContext(), centerPositionForEntity(), 0, 1f, size, path, orientation.first, orientation.second); + view.centerImage.setLayerNum(4 + 8); +// view.setHasStickyX(true); +// view.setHasStickyY(true); + view.setDelegate(this); + entitiesView.addView(view); + if (select) { + registerRemovalUndo(view); + selectEntity(view); + } + return view; + } + + private Size basePhotoSize(String path) { + float a = 1f; + try { + BitmapFactory.Options opts = new BitmapFactory.Options(); + opts.inJustDecodeBounds = true; + BitmapFactory.decodeFile(path, opts); + a = (float) opts.outWidth / opts.outHeight; + } catch (Exception e) { + FileLog.e(e); + } + if (a > 1) { + float side = (float) Math.floor(entitiesView.getMeasuredWidth() * 0.5); + return new Size(side, side / a); + } else { + float side = (float) Math.floor(entitiesView.getMeasuredHeight() * 0.5); + return new Size(side * a, side); + } + } + protected void onOpenCloseStickersAlert(boolean open) {} protected void onTextAdd() {} @@ -1434,6 +1596,11 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { entitiesView.setScaleX(baseScale); entitiesView.setScaleY(baseScale); entitiesView.measure(MeasureSpec.makeMeasureSpec((int) paintingSize.width, MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec((int) paintingSize.height, MeasureSpec.EXACTLY)); + if (thanosEffect != null) { + thanosEffect.measure(MeasureSpec.makeMeasureSpec((int) paintingSize.width, MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec((int) paintingSize.height, MeasureSpec.EXACTLY)); + thanosEffect.setScaleX(baseScale); + thanosEffect.setScaleY(baseScale); + } updateEntitiesSelections(); selectionContainerView.measure(MeasureSpec.makeMeasureSpec((int) renderWidth, MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec((int) renderHeight, MeasureSpec.EXACTLY)); measureChild(bottomLayout, widthMeasureSpec, heightMeasureSpec); @@ -1497,6 +1664,9 @@ protected void onLayout(boolean changed, int left, int top, int right, int botto int x2 = x + (renderView.getMeasuredWidth() - entitiesView.getMeasuredWidth()) / 2; int y2 = y + (renderView.getMeasuredHeight() - entitiesView.getMeasuredHeight()) / 2; entitiesView.layout(x2, y2, x2 + entitiesView.getMeasuredWidth(), y2 + entitiesView.getMeasuredHeight()); + if (thanosEffect != null) { + thanosEffect.layout(x2, y2, x2 + entitiesView.getMeasuredWidth(), y2 + entitiesView.getMeasuredHeight()); + } selectionContainerView.layout(x, y, x + selectionContainerView.getMeasuredWidth(), y + selectionContainerView.getMeasuredHeight()); } @@ -1753,6 +1923,23 @@ public Bitmap getBitmap(ArrayList entities, Bitmap[ if (stickerView.isMirrored()) { mediaEntity.subType |= 2; } + } else if (entity instanceof PhotoView) { + PhotoView photoView = (PhotoView) entity; + mediaEntity.type = VideoEditedInfo.MediaEntity.TYPE_PHOTO; + Size size = photoView.getBaseSize(); + mediaEntity.width = size.width; + mediaEntity.height = size.height; + mediaEntity.text = photoView.getPath(currentAccount); + if (photoView.isMirrored()) { + mediaEntity.subType |= 2; + } + if (photoView.hasSegmentedImage() && photoView.isSegmented()) { + File segmentedFile = photoView.saveSegmentedImage(currentAccount); + if (segmentedFile != null) { + mediaEntity.subType |= 16; + mediaEntity.segmentedPath = segmentedFile.getPath(); + } + } } else { continue; } @@ -2573,6 +2760,30 @@ private void showMenuForEntity(final EntityView entityView) { }); parent.addView(duplicateView, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, 48)); + if (entityView instanceof PhotoView && ((PhotoView) entityView).hasSegmentedImage()) { + PhotoView photoView = (PhotoView) entityView; + TextView cutView = new TextView(getContext()); + cutView.setTextColor(getThemedColor(Theme.key_actionBarDefaultSubmenuItem)); + cutView.setBackgroundDrawable(Theme.getSelectorDrawable(false)); + cutView.setGravity(Gravity.CENTER_VERTICAL); + cutView.setEllipsize(TextUtils.TruncateAt.END); + cutView.setPadding(AndroidUtilities.dp(14), 0, AndroidUtilities.dp(16), 0); + cutView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 14); + cutView.setTag(5); + cutView.setText(LocaleController.getString(photoView.isSegmented() ? R.string.SegmentationUndoCutOut : R.string.SegmentationCutOut)); + cutView.setOnClickListener(v -> { + photoView.toggleSegmented(true); + if (photoView.isSegmented()) { + onSwitchSegmentedAnimation(photoView); + } + if (popupWindow != null && popupWindow.isShowing()) { + popupWindow.dismiss(true); + } + }); + parent.addView(cutView, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, 44)); + photoView.highlightSegmented(); + } + popupLayout.addView(parent); LinearLayout.LayoutParams params = (LinearLayout.LayoutParams) parent.getLayoutParams(); @@ -2883,6 +3094,9 @@ private void removeEntity(EntityView entityView) { } private void registerRemovalUndo(final EntityView entityView) { + if (entityView == null) { + return; + } undoStore.registerUndo(entityView.getUUID(), () -> removeEntity(entityView)); } @@ -3447,4 +3661,64 @@ public void didReceivedNotification(int id, int account, Object... args) { } } } + + public ThanosEffect getThanosEffect() { + if (!ThanosEffect.supports()) { + return null; + } + if (thanosEffect == null) { + addView(thanosEffect = new ThanosEffect(getContext(), () -> { + ThanosEffect thisThanosEffect = thanosEffect; + if (thisThanosEffect != null) { + thanosEffect = null; + removeView(thisThanosEffect); + } + })); + } + return thanosEffect; + } + + public void onSwitchSegmentedAnimation(PhotoView photoView) { + if (photoView == null) { + return; + } + ThanosEffect thanosEffect = getThanosEffect(); + if (thanosEffect == null) { + photoView.onSwitchSegmentedAnimationStarted(false); + return; + } + Bitmap bitmap = photoView.getSegmentedOutBitmap(); + if (bitmap == null) { + photoView.onSwitchSegmentedAnimationStarted(false); + return; + } + Matrix matrix = new Matrix(); + float w = photoView.getWidth(), h = photoView.getHeight(); + float tx = 0, ty = 0; + if (photoView.getRotation() != 0) { + final float bw = bitmap.getWidth(); + final float bh = bitmap.getHeight(); + final float r = (float) Math.sqrt((bw / 2f) * (bw / 2f) + (bh / 2f) * (bh / 2f)); + final float d = 2 * r; + Bitmap newBitmap = Bitmap.createBitmap((int) d, (int) d, Bitmap.Config.ARGB_8888); + Canvas canvas = new Canvas(newBitmap); + canvas.save(); + canvas.rotate(photoView.getRotation(), r, r); + canvas.drawBitmap(bitmap, (d - bw) / 2, (d - bh) / 2, null); + bitmap.recycle(); + bitmap = newBitmap; + + final float pd = 2 * (float) Math.sqrt((w / 2f) * (w / 2f) + (h / 2f) * (h / 2f)); + tx = -(pd - w) / 2; + ty = -(pd - h) / 2; + w = pd; + h = pd; + } + matrix.postScale(w, h); + matrix.postScale(photoView.getScaleX(), photoView.getScaleY(), w / 2f, h / 2f); + matrix.postTranslate(photoView.getX() + tx, photoView.getY() + ty); + thanosEffect.animate(matrix, bitmap, () -> { + photoView.onSwitchSegmentedAnimationStarted(true); + }, () -> {}); + } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/LocationView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/LocationView.java index e58a66dfd7c..347aa67bdaf 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/LocationView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/LocationView.java @@ -145,7 +145,7 @@ protected float getMaxScale() { } @Override - protected Rect getSelectionBounds() { + public Rect getSelectionBounds() { ViewGroup parentView = (ViewGroup) getParent(); if (parentView == null) { return new Rect(); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/MessageEntityView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/MessageEntityView.java new file mode 100644 index 00000000000..e357f7460ee --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/MessageEntityView.java @@ -0,0 +1,1361 @@ +package org.telegram.ui.Components.Paint.Views; + +import static org.telegram.messenger.AndroidUtilities.dp; + +import android.app.Activity; +import android.content.Context; +import android.content.SharedPreferences; +import android.graphics.Bitmap; +import android.graphics.Canvas; +import android.graphics.Matrix; +import android.graphics.Paint; +import android.graphics.Path; +import android.graphics.PorterDuff; +import android.graphics.PorterDuffXfermode; +import android.graphics.RectF; +import android.graphics.drawable.Drawable; +import android.text.TextPaint; +import android.util.Log; +import android.util.SparseIntArray; +import android.view.MotionEvent; +import android.view.TextureView; +import android.view.View; +import android.view.ViewGroup; +import android.widget.FrameLayout; + +import androidx.annotation.NonNull; +import androidx.recyclerview.widget.GridLayoutManager; +import androidx.recyclerview.widget.GridLayoutManagerFixed; +import androidx.recyclerview.widget.LinearLayoutManager; +import androidx.recyclerview.widget.RecyclerView; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.ApplicationLoader; +import org.telegram.messenger.ImageReceiver; +import org.telegram.messenger.MediaController; +import org.telegram.messenger.MessageObject; +import org.telegram.messenger.MessagesController; +import org.telegram.messenger.SharedConfig; +import org.telegram.tgnet.TLRPC; +import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.Cells.BotHelpCell; +import org.telegram.ui.Cells.ChatActionCell; +import org.telegram.ui.Cells.ChatMessageCell; +import org.telegram.ui.Components.AnimatedFloat; +import org.telegram.ui.Components.BlurringShader; +import org.telegram.ui.Components.CubicBezierInterpolator; +import org.telegram.ui.Components.LayoutHelper; +import org.telegram.ui.Components.MessageBackgroundDrawable; +import org.telegram.ui.Components.Point; +import org.telegram.ui.Components.Rect; +import org.telegram.ui.Components.RecyclerListView; +import org.telegram.ui.Components.Size; +import org.telegram.ui.Stories.recorder.PreviewView; +import org.telegram.ui.Stories.recorder.StoryEntry; + +import java.io.File; +import java.util.ArrayList; + +public class MessageEntityView extends EntityView { + +// private final ChatActionCell dateCell; + public final FrameLayout container; + public final RecyclerListView listView; + public final ArrayList messageObjects = new ArrayList<>(); + private MessageObject.GroupedMessages groupedMessages; + private final BlurringShader.BlurManager blurManager; + private boolean clipVideoMessageForBitmap; + private boolean usesBackgroundPaint; + private PreviewView.TextureViewHolder videoTextureHolder; + private TextureView textureView; + private boolean textureViewActive; + private int videoWidth = 1, videoHeight = 1; + + public boolean drawForBitmap() { + return false; + } + + public MessageEntityView(Context context, Point position, ArrayList messageObjects, BlurringShader.BlurManager blurManager, boolean isRepostVideoPreview, PreviewView.TextureViewHolder videoTextureHolder) { + this(context, position, 0.0f, 1.0f, messageObjects, blurManager, isRepostVideoPreview, videoTextureHolder); + } + + public MessageEntityView(Context context, Point position, float angle, float scale, ArrayList thisMessageObjects, BlurringShader.BlurManager blurManager, boolean isRepostVideoPreview, PreviewView.TextureViewHolder videoTextureHolder) { + super(context, position); + this.blurManager = blurManager; + setRotation(angle); + setScale(scale); + int date = 0; + for (int i = 0; i < thisMessageObjects.size(); ++i) { + MessageObject msg = thisMessageObjects.get(i); + date = msg.messageOwner.date; + TLRPC.Message messageOwner = copyMessage(msg.messageOwner); + Boolean b = StoryEntry.useForwardForRepost(msg); + if (b != null && b && messageOwner.fwd_from != null && messageOwner.fwd_from.from_id != null) { + messageOwner.from_id = messageOwner.fwd_from.from_id; + messageOwner.peer_id = messageOwner.fwd_from.from_id; + messageOwner.flags &=~ 4; + messageOwner.fwd_from = null; + } + messageOwner.voiceTranscriptionOpen = false; + MessageObject newMsg = new MessageObject(msg.currentAccount, messageOwner, msg.replyMessageObject, MessagesController.getInstance(msg.currentAccount).getUsers(), MessagesController.getInstance(msg.currentAccount).getChats(), null, null, true, true, 0, true, isRepostVideoPreview); + messageObjects.add(newMsg); + } +// dateCell = new ChatActionCell(context, false, resourcesProvider) { +// public final BlurringShader.StoryBlurDrawer blurDrawer = new BlurringShader.StoryBlurDrawer(blurManager, this, BlurringShader.StoryBlurDrawer.BLUR_TYPE_ACTION_BACKGROUND); +// private final TextPaint textPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG); { +// textPaint.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); +// textPaint.setTextSize(AndroidUtilities.dp(Math.max(16, SharedConfig.fontSize) - 2)); +// textPaint.setColor(0xffffffff); +// } +// +// @Override +// protected Paint getThemedPaint(String paintKey) { +// if (Theme.key_paint_chatActionText.equals(paintKey) || Theme.key_paint_chatActionText2.equals(paintKey)) { +// return textPaint; +// } +// if (Theme.key_paint_chatActionBackground.equals(paintKey)) { +// usesBackgroundPaint = true; +// Paint paint = blurDrawer.adapt(isDark).getPaint(1f); +// if (paint != null) { +// return paint; +// } +// } +// return super.getThemedPaint(paintKey); +// } +// }; +// dateCell.setTranslationX(dp(26)); +// dateCell.setCustomDate(date, false, false); +// addView(dateCell, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT)); + groupedMessages = null; + if (messageObjects.size() > 1) { + groupedMessages = new MessageObject.GroupedMessages(); + groupedMessages.messages.addAll(messageObjects); + groupedMessages.groupId = messageObjects.get(0).getGroupId(); + groupedMessages.calculate(); + } + container = new FrameLayout(context) { + @Override + protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { + listView.measure( + widthMeasureSpec, + MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED) + ); + if (textureView != null) { + textureView.measure( + MeasureSpec.makeMeasureSpec(listView.getMeasuredWidth(), MeasureSpec.EXACTLY), + MeasureSpec.makeMeasureSpec(listView.getMeasuredHeight(), MeasureSpec.EXACTLY) + ); + } + int left = listView.getMeasuredWidth(); + int right = 0; + for (int i = 0; i < listView.getChildCount(); ++i) { + View child = listView.getChildAt(i); + int childleft = child.getLeft(), childright = child.getRight(); + if (child instanceof ChatMessageCell) { + childleft = child.getLeft() + ((ChatMessageCell) child).getBoundsLeft(); + childright = child.getLeft() + ((ChatMessageCell) child).getBoundsRight(); + } + left = Math.min(childleft, left); + right = Math.max(childright, right); + } + setMeasuredDimension(right - left, listView.getMeasuredHeight()); + } + + @Override + protected void onLayout(boolean changed, int left, int top, int right, int bottom) { + int cleft = listView.getMeasuredWidth(); + int cright = 0; + for (int i = 0; i < listView.getChildCount(); ++i) { + View child = listView.getChildAt(i); + int childleft = child.getLeft(), childright = child.getRight(); + if (child instanceof ChatMessageCell) { + childleft = child.getLeft() + ((ChatMessageCell) child).getBoundsLeft(); + childright = child.getLeft() + ((ChatMessageCell) child).getBoundsRight(); + } + cleft = Math.min(childleft, cleft); + cright = Math.max(childright, cright); + } + listView.layout(-cleft, 0, listView.getMeasuredWidth() - cleft, listView.getMeasuredHeight()); + if (textureView != null) { + textureView.layout(0, 0, getMeasuredWidth(), listView.getMeasuredHeight()); + } + } + + private final Matrix videoMatrix = new Matrix(); + private final float[] radii = new float[8]; + private final Path clipPath = new Path(); + + @Override + protected boolean drawChild(Canvas canvas, View child, long drawingTime) { + if (child == textureView) { + ChatMessageCell cell = getCell(); + if (cell == null) return false; + ImageReceiver photoImage = cell.getPhotoImage(); + if (photoImage == null) return false; + videoMatrix.reset(); + float scale = Math.max(photoImage.getImageWidth() / videoWidth, photoImage.getImageHeight() / videoHeight); + videoMatrix.postScale((float) videoWidth / textureView.getWidth() * scale, (float) videoHeight / textureView.getHeight() * scale); + videoMatrix.postTranslate(listView.getX() + cell.getX() + photoImage.getCenterX() - videoWidth * scale / 2f, listView.getY() + cell.getY() + photoImage.getCenterY() - videoHeight * scale / 2f); + textureView.setTransform(videoMatrix); + canvas.save(); + clipPath.rewind(); + AndroidUtilities.rectTmp.set(listView.getX() + cell.getX() + photoImage.getImageX(), listView.getY() + cell.getY() + photoImage.getImageY(), listView.getX() + cell.getX() + photoImage.getImageX2(), listView.getY() + cell.getY() + photoImage.getImageY2()); + for (int a = 0; a < photoImage.getRoundRadius().length; a++) { + radii[a * 2] = photoImage.getRoundRadius()[a]; + radii[a * 2 + 1] = photoImage.getRoundRadius()[a]; + } + clipPath.addRoundRect(AndroidUtilities.rectTmp, radii, Path.Direction.CW); + canvas.clipPath(clipPath); + boolean r = super.drawChild(canvas, child, drawingTime); + canvas.restore(); + return r; + } + return super.drawChild(canvas, child, drawingTime); + } + }; + addView(container, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT)); + listView = new RecyclerListView(context, resourcesProvider) { + + private final ArrayList drawTimeAfter = new ArrayList<>(); + private final ArrayList drawNamesAfter = new ArrayList<>(); + private final ArrayList drawCaptionAfter = new ArrayList<>(); + private final ArrayList drawingGroups = new ArrayList<>(10); + + @Override + protected void dispatchDraw(Canvas canvas) { + canvas.save(); + selectorRect.setEmpty(); + drawChatBackgroundElements(canvas); + super.dispatchDraw(canvas); + drawChatForegroundElements(canvas); + canvas.restore(); + } + + private void drawChatForegroundElements(Canvas canvas) { + int size = drawTimeAfter.size(); + if (size > 0) { + for (int a = 0; a < size; a++) { + ChatMessageCell cell = drawTimeAfter.get(a); + canvas.save(); + canvas.translate(cell.getLeft() + cell.getNonAnimationTranslationX(false), cell.getY()); + cell.drawTime(canvas, cell.shouldDrawAlphaLayer() ? cell.getAlpha() : 1f, true); + canvas.restore(); + } + drawTimeAfter.clear(); + } + size = drawNamesAfter.size(); + if (size > 0) { + for (int a = 0; a < size; a++) { + ChatMessageCell cell = drawNamesAfter.get(a); + float canvasOffsetX = cell.getLeft() + cell.getNonAnimationTranslationX(false); + float canvasOffsetY = cell.getY(); + float alpha = cell.shouldDrawAlphaLayer() ? cell.getAlpha() : 1f; + + canvas.save(); + canvas.translate(canvasOffsetX, canvasOffsetY); + cell.setInvalidatesParent(true); + cell.drawNamesLayout(canvas, alpha); + cell.setInvalidatesParent(false); + canvas.restore(); + } + drawNamesAfter.clear(); + } + size = drawCaptionAfter.size(); + if (size > 0) { + for (int a = 0; a < size; a++) { + ChatMessageCell cell = drawCaptionAfter.get(a); + boolean selectionOnly = false; + if (cell.getCurrentPosition() != null) { + selectionOnly = (cell.getCurrentPosition().flags & MessageObject.POSITION_FLAG_LEFT) == 0; + } + float alpha = cell.shouldDrawAlphaLayer() ? cell.getAlpha() : 1f; + float canvasOffsetX = cell.getLeft() + cell.getNonAnimationTranslationX(false); + float canvasOffsetY = cell.getY(); + canvas.save(); + MessageObject.GroupedMessages groupedMessages = cell.getCurrentMessagesGroup(); + if (groupedMessages != null && groupedMessages.transitionParams.backgroundChangeBounds) { + float x = cell.getNonAnimationTranslationX(true); + float l = (groupedMessages.transitionParams.left + x + groupedMessages.transitionParams.offsetLeft); + float t = (groupedMessages.transitionParams.top + groupedMessages.transitionParams.offsetTop); + float r = (groupedMessages.transitionParams.right + x + groupedMessages.transitionParams.offsetRight); + float b = (groupedMessages.transitionParams.bottom + groupedMessages.transitionParams.offsetBottom); + + if (!groupedMessages.transitionParams.backgroundChangeBounds) { + t += cell.getTranslationY(); + b += cell.getTranslationY(); + } + canvas.clipRect( + l + AndroidUtilities.dp(8), t + AndroidUtilities.dp(8), + r - AndroidUtilities.dp(8), b - AndroidUtilities.dp(8) + ); + } + if (cell.getTransitionParams().wasDraw) { + canvas.translate(canvasOffsetX, canvasOffsetY); + cell.setInvalidatesParent(true); + cell.drawCaptionLayout(canvas, selectionOnly, alpha); + cell.setInvalidatesParent(false); + canvas.restore(); + } + } + drawCaptionAfter.clear(); + } + } + + private void drawChatBackgroundElements(Canvas canvas) { + int count = getChildCount(); + MessageObject.GroupedMessages lastDrawnGroup = null; + + for (int a = 0; a < count; a++) { + View child = getChildAt(a); + if (child.getVisibility() == View.INVISIBLE) { + continue; + } + if (child instanceof ChatMessageCell) { + ChatMessageCell cell = (ChatMessageCell) child; + MessageObject.GroupedMessages group = cell.getCurrentMessagesGroup(); + if (group == null || group != lastDrawnGroup) { + lastDrawnGroup = group; + MessageObject.GroupedMessagePosition position = cell.getCurrentPosition(); + MessageBackgroundDrawable backgroundDrawable = cell.getBackgroundDrawable(); + if ((backgroundDrawable.isAnimationInProgress() || cell.isDrawingSelectionBackground()) && (position == null || (position.flags & MessageObject.POSITION_FLAG_RIGHT) != 0)) { + int y = (int) cell.getY(); + int height; + canvas.save(); + if (position == null) { + height = cell.getMeasuredHeight(); + } else { + height = y + cell.getMeasuredHeight(); + long time = 0; + float touchX = 0; + float touchY = 0; + for (int i = 0; i < count; i++) { + View inner = getChildAt(i); + if (inner instanceof ChatMessageCell) { + ChatMessageCell innerCell = (ChatMessageCell) inner; + MessageObject.GroupedMessages innerGroup = innerCell.getCurrentMessagesGroup(); + if (innerGroup == group) { + MessageBackgroundDrawable drawable = innerCell.getBackgroundDrawable(); + y = Math.min(y, (int) innerCell.getY()); + height = Math.max(height, (int) innerCell.getY() + innerCell.getMeasuredHeight()); + long touchTime = drawable.getLastTouchTime(); + if (touchTime > time) { + touchX = drawable.getTouchX() + innerCell.getX(); + touchY = drawable.getTouchY() + innerCell.getY(); + time = touchTime; + } + } + } + } + backgroundDrawable.setTouchCoordsOverride(touchX, touchY - y); + height -= y; + } + canvas.clipRect(0, y, getMeasuredWidth(), y + height); + backgroundDrawable.setCustomPaint(null); + backgroundDrawable.setColor(getThemedColor(Theme.key_chat_selectedBackground)); + backgroundDrawable.setBounds(0, y, getMeasuredWidth(), y + height); + backgroundDrawable.draw(canvas); + canvas.restore(); + } + } + } else if (child instanceof ChatActionCell) { + ChatActionCell cell = (ChatActionCell) child; + if (cell.hasGradientService()) { + canvas.save(); + canvas.translate(cell.getX(), cell.getY()); + canvas.scale(cell.getScaleX(), cell.getScaleY(), cell.getMeasuredWidth() / 2f, cell.getMeasuredHeight() / 2f); + cell.drawBackground(canvas, true); + canvas.restore(); + } + } + } + for (int k = 0; k < 3; k++) { + drawingGroups.clear(); + if (k == 2 && !isFastScrollAnimationRunning()) { + continue; + } + for (int i = 0; i < count; i++) { + View child = getChildAt(i); + if (child instanceof ChatMessageCell) { + ChatMessageCell cell = (ChatMessageCell) child; + if (child.getY() > getHeight() || child.getY() + child.getHeight() < 0 || cell.getVisibility() == View.INVISIBLE || cell.getVisibility() == View.GONE) { + continue; + } + MessageObject.GroupedMessages group = cell.getCurrentMessagesGroup(); + if (group == null || (k == 0 && group.messages.size() == 1) || (k == 1 && !group.transitionParams.drawBackgroundForDeletedItems)) { + continue; + } + if ((k == 0 && cell.getMessageObject().deleted) || (k == 1 && !cell.getMessageObject().deleted)) { + continue; + } + if ((k == 2 && !cell.willRemovedAfterAnimation()) || (k != 2 && cell.willRemovedAfterAnimation())) { + continue; + } + + if (!drawingGroups.contains(group)) { + group.transitionParams.left = 0; + group.transitionParams.top = 0; + group.transitionParams.right = 0; + group.transitionParams.bottom = 0; + + group.transitionParams.pinnedBotton = false; + group.transitionParams.pinnedTop = false; + group.transitionParams.cell = cell; + drawingGroups.add(group); + } + + group.transitionParams.pinnedTop = cell.isPinnedTop(); + group.transitionParams.pinnedBotton = cell.isPinnedBottom(); + + int left = (cell.getLeft() + cell.getBackgroundDrawableLeft()); + int right = (cell.getLeft() + cell.getBackgroundDrawableRight()); + int top = (cell.getTop() + cell.getBackgroundDrawableTop()); + int bottom = (cell.getTop() + cell.getBackgroundDrawableBottom()); + + if ((cell.getCurrentPosition().flags & MessageObject.POSITION_FLAG_TOP) == 0) { + top -= AndroidUtilities.dp(10); + } + + if ((cell.getCurrentPosition().flags & MessageObject.POSITION_FLAG_BOTTOM) == 0) { + bottom += AndroidUtilities.dp(10); + } + + if (cell.willRemovedAfterAnimation()) { + group.transitionParams.cell = cell; + } + + if (group.transitionParams.top == 0 || top < group.transitionParams.top) { + group.transitionParams.top = top; + } + if (group.transitionParams.bottom == 0 || bottom > group.transitionParams.bottom) { + group.transitionParams.bottom = bottom; + } + if (group.transitionParams.left == 0 || left < group.transitionParams.left) { + group.transitionParams.left = left; + } + if (group.transitionParams.right == 0 || right > group.transitionParams.right) { + group.transitionParams.right = right; + } + } + } + + for (int i = 0; i < drawingGroups.size(); i++) { + MessageObject.GroupedMessages group = drawingGroups.get(i); + float x = group.transitionParams.cell.getNonAnimationTranslationX(true); + float l = (group.transitionParams.left + x + group.transitionParams.offsetLeft); + float t = (group.transitionParams.top + group.transitionParams.offsetTop); + float r = (group.transitionParams.right + x + group.transitionParams.offsetRight); + float b = (group.transitionParams.bottom + group.transitionParams.offsetBottom); + + if (!group.transitionParams.backgroundChangeBounds) { + t += group.transitionParams.cell.getTranslationY(); + b += group.transitionParams.cell.getTranslationY(); + } + + boolean useScale = group.transitionParams.cell.getScaleX() != 1f || group.transitionParams.cell.getScaleY() != 1f; + if (useScale) { + canvas.save(); + canvas.scale(group.transitionParams.cell.getScaleX(), group.transitionParams.cell.getScaleY(), l + (r - l) / 2, t + (b - t) / 2); + } + boolean selected = false; + group.transitionParams.cell.drawBackground(canvas, (int) l, (int) t, (int) r, (int) b, group.transitionParams.pinnedTop, group.transitionParams.pinnedBotton, selected, 0); + group.transitionParams.cell = null; + group.transitionParams.drawCaptionLayout = group.hasCaption; + if (useScale) { + canvas.restore(); + for (int ii = 0; ii < count; ii++) { + View child = getChildAt(ii); + if (child instanceof ChatMessageCell && ((ChatMessageCell) child).getCurrentMessagesGroup() == group) { + ChatMessageCell cell = ((ChatMessageCell) child); + int left = cell.getLeft(); + int top = cell.getTop(); + child.setPivotX(l - left + (r - l) / 2); + child.setPivotY(t - top + (b - t) / 2); + } + } + } + } + } + } + + @Override + public boolean drawChild(Canvas canvas, View child, long drawingTime) { + ChatMessageCell cell = null; + ChatActionCell actionCell = null; + + if (child instanceof ChatMessageCell) { + cell = (ChatMessageCell) child; + } else if (child instanceof ChatActionCell) { + actionCell = (ChatActionCell) child; + } + + boolean result = super.drawChild(canvas, child, drawingTime); + if (cell != null && cell.hasOutboundsContent()) { + canvas.save(); + canvas.translate(cell.getX(), cell.getY()); + cell.drawOutboundsContent(canvas); + canvas.restore(); + } else if (actionCell != null) { + canvas.save(); + canvas.translate(actionCell.getX(), actionCell.getY()); + actionCell.drawOutboundsContent(canvas); + canvas.restore(); + } + + if (child.getTranslationY() != 0) { + canvas.save(); + canvas.translate(0, child.getTranslationY()); + } + + if (cell != null) { + cell.drawCheckBox(canvas); + } + + if (child.getTranslationY() != 0) { + canvas.restore(); + } + + if (child.getTranslationY() != 0) { + canvas.save(); + canvas.translate(0, child.getTranslationY()); + } + + if (cell != null) { + MessageObject message = cell.getMessageObject(); + MessageObject.GroupedMessagePosition position = cell.getCurrentPosition(); + if (position != null || cell.getTransitionParams().animateBackgroundBoundsInner) { + if (position == null || (position.last || position.minX == 0 && position.minY == 0)) { + if (position == null || position.last) { + drawTimeAfter.add(cell); + } + if ((position == null || (position.minX == 0 && position.minY == 0)) && cell.hasNameLayout()) { + drawNamesAfter.add(cell); + } + } + if (position != null || cell.getTransitionParams().transformGroupToSingleMessage || cell.getTransitionParams().animateBackgroundBoundsInner) { + if (position == null || (position.flags & MessageObject.POSITION_FLAG_BOTTOM) != 0) { + drawCaptionAfter.add(cell); + } + } + } + ImageReceiver imageReceiver = cell.getAvatarImage(); + if (imageReceiver != null) { + boolean replaceAnimation = isFastScrollAnimationRunning() || (groupedMessages != null && groupedMessages.transitionParams.backgroundChangeBounds); + int top = replaceAnimation ? child.getTop() : (int) child.getY(); + if (cell.drawPinnedBottom()) { + int p; + ViewHolder holder = listView.getChildViewHolder(child); + p = holder.getAdapterPosition(); + + if (p >= 0) { + int nextPosition; + if (groupedMessages != null && position != null) { + int idx = groupedMessages.posArray.indexOf(position); + int size = groupedMessages.posArray.size(); + if ((position.flags & MessageObject.POSITION_FLAG_BOTTOM) != 0) { + nextPosition = p - size + idx; + } else { + nextPosition = p - 1; + for (int a = idx + 1; a < size; a++) { + if (groupedMessages.posArray.get(a).minY > position.maxY) { + break; + } else { + nextPosition--; + } + } + } + } else { + nextPosition = p - 1; + } + holder = findViewHolderForAdapterPosition(nextPosition); + if (holder != null) { + if (child.getTranslationY() != 0) { + canvas.restore(); + } + imageReceiver.setVisible(false, false); + return result; + } + } + } + float tx = cell.getSlidingOffsetX() + cell.getCheckBoxTranslation(); + int y = (int) ((replaceAnimation ? child.getTop() : child.getY()) + cell.getLayoutHeight() + cell.getTransitionParams().deltaBottom); + int maxY = getMeasuredHeight() - getPaddingBottom(); + boolean canUpdateTx = cell.isCheckBoxVisible() && tx == 0; + if (cell.isPlayingRound() || cell.getTransitionParams().animatePlayingRound) { + if (cell.getTransitionParams().animatePlayingRound) { + float progressLocal = cell.getTransitionParams().animateChangeProgress; + if (!cell.isPlayingRound()) { + progressLocal = 1f - progressLocal; + } + int fromY = y; + int toY = Math.min(y, maxY); + y = (int) (fromY * progressLocal + toY * (1f - progressLocal)); + } + } else { + if (y > maxY) { + y = maxY; + } + } + + if (!replaceAnimation && child.getTranslationY() != 0) { + canvas.restore(); + } + if (cell.drawPinnedTop()) { + int p; + ViewHolder holder = getChildViewHolder(child); + p = holder.getAdapterPosition(); + if (p >= 0) { + int tries = 0; + while (true) { + if (tries >= 20) { + break; + } + tries++; + + int prevPosition; + if (groupedMessages != null && position != null) { + int idx = groupedMessages.posArray.indexOf(position); + if (idx < 0) { + break; + } + int size = groupedMessages.posArray.size(); + if ((position.flags & MessageObject.POSITION_FLAG_TOP) != 0) { + prevPosition = p + idx + 1; + } else { + prevPosition = p + 1; + for (int a = idx - 1; a >= 0; a--) { + if (groupedMessages.posArray.get(a).maxY < position.minY) { + break; + } else { + prevPosition++; + } + } + } + } else { + prevPosition = p + 1; + } + holder = findViewHolderForAdapterPosition(prevPosition); + if (holder != null) { + top = holder.itemView.getTop(); + if (holder.itemView instanceof ChatMessageCell) { + cell = (ChatMessageCell) holder.itemView; + float newTx = cell.getSlidingOffsetX() + cell.getCheckBoxTranslation(); + if (canUpdateTx && newTx > 0) { + tx = newTx; + } + if (!cell.drawPinnedTop()) { + break; + } else { + p = prevPosition; + } + } else { + break; + } + } else { + break; + } + } + } + } + if (y - AndroidUtilities.dp(42) < top) { + y = top + AndroidUtilities.dp(42); + } + if (!cell.drawPinnedBottom()) { + int cellBottom = replaceAnimation ? cell.getBottom() : (int) (cell.getY() + cell.getMeasuredHeight() + cell.getTransitionParams().deltaBottom); + if (y > cellBottom) { + y = cellBottom; + } + } + canvas.save(); + if (tx != 0) { + canvas.translate(tx, 0); + } + if (cell.getCurrentMessagesGroup() != null) { + if (cell.getCurrentMessagesGroup().transitionParams.backgroundChangeBounds) { + y -= cell.getTranslationY(); + } + } + imageReceiver.setImageY(y - AndroidUtilities.dp(40)); + if (cell.shouldDrawAlphaLayer()) { + imageReceiver.setAlpha(cell.getAlpha()); + canvas.scale( + cell.getScaleX(), cell.getScaleY(), + cell.getX() + cell.getPivotX(), cell.getY() + (cell.getHeight() >> 1) + ); + } else { + imageReceiver.setAlpha(1f); + } + imageReceiver.setVisible(true, false); + imageReceiver.draw(canvas); + canvas.restore(); + + if (!replaceAnimation && child.getTranslationY() != 0) { + canvas.save(); + } + } + } + + if (child.getTranslationY() != 0) { + canvas.restore(); + } + return result; + } + }; + listView.setAdapter(new RecyclerListView.SelectionAdapter() { + @Override + public boolean isEnabled(RecyclerView.ViewHolder holder) { + return true; + } + + @NonNull + @Override + public RecyclerView.ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) { + ChatMessageCell cell = new ChatMessageCell(context, false, null, resourcesProvider) { + public BlurringShader.StoryBlurDrawer blurDrawer = new BlurringShader.StoryBlurDrawer(blurManager, this, BlurringShader.StoryBlurDrawer.BLUR_TYPE_ACTION_BACKGROUND); + + @Override + protected void onDraw(Canvas canvas) { + if (videoTextureHolder != null && videoTextureHolder.active && videoTextureHolder.textureViewActive || clipVideoMessageForBitmap) { + canvas.saveLayerAlpha(0, 0, getWidth(), getHeight(), 0xFF, Canvas.ALL_SAVE_FLAG); + } else { + canvas.save(); + } + super.onDraw(canvas); + canvas.restore(); + } + + @Override + public boolean onTouchEvent(MotionEvent event) { + return false; + } + + @Override + public Paint getThemedPaint(String paintKey) { + if (Theme.key_paint_chatActionBackground.equals(paintKey)) { + usesBackgroundPaint = true; + Paint paint = blurDrawer.getPaint(1f); + if (paint != null) { + return paint; + } + } + return super.getThemedPaint(paintKey); + } + + private final float[] radii = new float[8]; + private final Path clipPath = new Path(); + private final Paint clearPaint = new Paint(); + { clearPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.CLEAR)); } + private final android.graphics.Rect src = new android.graphics.Rect(); + private final android.graphics.RectF dst = new android.graphics.RectF(); + + @Override + protected boolean drawPhotoImage(Canvas canvas) { + ImageReceiver photoImage = getPhotoImage(); + if (isRepostVideoPreview && photoImage != null && (videoTextureHolder != null && videoTextureHolder.active && videoTextureHolder.textureViewActive && textureViewActive || clipVideoMessageForBitmap || textureView != null && drawForBitmap())) { + for (int a = 0; a < photoImage.getRoundRadius().length; a++) { + radii[a * 2] = photoImage.getRoundRadius()[a]; + radii[a * 2 + 1] = photoImage.getRoundRadius()[a]; + } + AndroidUtilities.rectTmp.set(photoImage.getImageX(), photoImage.getImageY(), photoImage.getImageX2(), photoImage.getImageY2()); + clipPath.rewind(); + clipPath.addRoundRect(AndroidUtilities.rectTmp, radii, Path.Direction.CW); + if (textureView != null && drawForBitmap()) { + Bitmap bitmap = textureView.getBitmap(); + if (bitmap != null) { + canvas.save(); + canvas.clipPath(clipPath); + canvas.translate(-getX(), -getY()); + float scale = Math.max(photoImage.getImageWidth() / videoWidth, photoImage.getImageHeight() / videoHeight); + canvas.translate(photoImage.getCenterX() - videoWidth * scale / 2f, photoImage.getCenterY() - videoHeight * scale / 2f); + canvas.scale((float) videoWidth / textureView.getWidth() * scale, (float) videoHeight / textureView.getHeight() * scale); + src.set(0, 0, bitmap.getWidth(), bitmap.getHeight()); + dst.set(0, 0, textureView.getWidth(), textureView.getHeight()); + canvas.drawBitmap(bitmap, src, dst, null); + canvas.restore(); + } else { + return super.drawPhotoImage(canvas); + } + } else { + canvas.drawPath(clipPath, clearPaint); + } + return true; + } + return super.drawPhotoImage(canvas); + } + }; + cell.isChat = true; + return new RecyclerListView.Holder(cell); + } + + @Override + public void onBindViewHolder(@NonNull RecyclerView.ViewHolder holder, int position) { + position = messageObjects.size() - 1 - position; + MessageObject message = messageObjects.get(position); + boolean pinnedTop = false; + if (groupedMessages != null) { + MessageObject.GroupedMessagePosition p = groupedMessages.positions.get(message); + if (p != null) { + pinnedTop = p.minY != 0; + } + } + ((ChatMessageCell) holder.itemView).setMessageObject(message, groupedMessages, groupedMessages != null, pinnedTop); + } + + @Override + public int getItemCount() { + return messageObjects.size(); + } + }); + GridLayoutManagerFixed layoutManager = new GridLayoutManagerFixed(context, 1000, LinearLayoutManager.VERTICAL, true) { + + @Override + public boolean supportsPredictiveItemAnimations() { + return false; + } + + @Override + public boolean shouldLayoutChildFromOpositeSide(View child) { + if (child instanceof ChatMessageCell) { + return !((ChatMessageCell) child).getMessageObject().isOutOwner(); + } + return false; + } + + @Override + protected boolean hasSiblingChild(int position) { + position = messageObjects.size() - 1 - position; + if (groupedMessages != null && position >= 0 && position < messageObjects.size()) { + MessageObject message = messageObjects.get(position); + MessageObject.GroupedMessagePosition pos = groupedMessages.positions.get(message); + if (pos == null || pos.minX == pos.maxX || pos.minY != pos.maxY || pos.minY == 0) { + return false; + } + int count = groupedMessages.posArray.size(); + for (int a = 0; a < count; a++) { + MessageObject.GroupedMessagePosition p = groupedMessages.posArray.get(a); + if (p == pos) { + continue; + } + if (p.minY <= pos.minY && p.maxY >= pos.minY) { + return true; + } + } + } + return false; + } + }; + layoutManager.setSpanSizeLookup(new GridLayoutManager.SpanSizeLookup() { + @Override + public int getSpanSize(int position) { + position = messageObjects.size() - 1 - position; + if (groupedMessages != null && position >= 0 && position < groupedMessages.messages.size()) { + MessageObject message = groupedMessages.messages.get(position); + MessageObject.GroupedMessagePosition groupedPosition = groupedMessages.positions.get(message); + if (groupedPosition != null) { + return groupedPosition.spanSize; + } + } + return 1000; + } + }); + listView.setLayoutManager(layoutManager); + listView.addItemDecoration(new RecyclerView.ItemDecoration() { + @Override + public void getItemOffsets(android.graphics.Rect outRect, View view, RecyclerView parent, RecyclerView.State state) { + outRect.bottom = 0; + if (view instanceof ChatMessageCell) { + ChatMessageCell cell = (ChatMessageCell) view; + MessageObject.GroupedMessages group = cell.getCurrentMessagesGroup(); + if (group != null) { + MessageObject.GroupedMessagePosition position = cell.getCurrentPosition(); + if (position != null && position.siblingHeights != null) { + float maxHeight = Math.max(AndroidUtilities.displaySize.x, AndroidUtilities.displaySize.y) * 0.5f; + int h = cell.getExtraInsetHeight(); + for (int a = 0; a < position.siblingHeights.length; a++) { + h += (int) Math.ceil(maxHeight * position.siblingHeights[a]); + } + h += (position.maxY - position.minY) * Math.round(7 * AndroidUtilities.density); + int count = group.posArray.size(); + for (int a = 0; a < count; a++) { + MessageObject.GroupedMessagePosition pos = group.posArray.get(a); + if (pos.minY != position.minY || pos.minX == position.minX && pos.maxX == position.maxX && pos.minY == position.minY && pos.maxY == position.maxY) { + continue; + } + if (pos.minY == position.minY) { + h -= (int) Math.ceil(maxHeight * pos.ph) - AndroidUtilities.dp(4); + break; + } + } + outRect.bottom = -h; + } + } + } + } + }); + container.addView(listView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT)); + if (videoTextureHolder != null && videoTextureHolder.active) { + videoTextureHolder.takeTextureView(textureView -> { + this.textureView = textureView; + if (textureView != null) { + container.addView(textureView, 0); + } + }, (w, h) -> { + videoWidth = w; + videoHeight = h; + AndroidUtilities.runOnUIThread(() -> { + textureViewActive = true; + invalidateAll(); + }, 60); + }); + } + updatePosition(); + } + + private ChatMessageCell getCell() { + if (listView == null) return null; + for (int i = 0; i < listView.getChildCount(); ++i) { + if (listView.getChildAt(i) instanceof ChatMessageCell) { + return (ChatMessageCell) listView.getChildAt(i); + } + } + return null; + } + + public void getBubbleBounds(RectF rect) { + float left = Integer.MAX_VALUE; + float right = Integer.MIN_VALUE; + float top = Integer.MAX_VALUE; + float bottom = Integer.MIN_VALUE; + for (int i = 0; i < listView.getChildCount(); ++i) { + View child = listView.getChildAt(i); + if (child instanceof ChatMessageCell) { + ChatMessageCell cell = (ChatMessageCell) child; + float cleft, ctop, cright, cbottom; + if (cell.getMessageObject() != null && cell.getMessageObject().isRoundVideo() && cell.getPhotoImage() != null) { + cleft = container.getX() + cell.getX() + cell.getPhotoImage().getImageX(); + cright = container.getX() + cell.getX() + cell.getPhotoImage().getImageX2(); + ctop = container.getY() + cell.getY() + cell.getPhotoImage().getImageY(); + cbottom = container.getY() + cell.getY() + cell.getPhotoImage().getImageY2(); + } else { + cleft = container.getX() + child.getX() + cell.getBackgroundDrawableLeft() + dp(1); + if (groupedMessages == null) { // pinned bottom + cleft += dp(8); + } + cright = container.getX() + child.getX() + cell.getBackgroundDrawableRight() - dp(1); + ctop = container.getY() + child.getY() + cell.getBackgroundDrawableTop() + dp(1.33f); + cbottom = container.getY() + child.getY() + cell.getBackgroundDrawableBottom() - dp(.66f); + } + left = Math.min(left, cleft); + left = Math.min(left, cright); + right = Math.max(right, cleft); + right = Math.max(right, cright); + top = Math.min(top, ctop); + top = Math.min(top, cbottom); + bottom = Math.max(bottom, ctop); + bottom = Math.max(bottom, cbottom); + } + } + rect.set(left, top, right, bottom); + } + + public void invalidateAll() { +// dateCell.invalidate(); + listView.invalidate(); + for (int i = 0; i < listView.getChildCount(); ++i) { + listView.getChildAt(i).invalidate(); + } + } + + public void prepareToDraw(boolean drawingToBitmap) { + clipVideoMessageForBitmap = drawingToBitmap; + for (int i = 0; i < listView.getChildCount(); ++i) { + View child = listView.getChildAt(i); + if (child instanceof ChatMessageCell) { + ((ChatMessageCell) child).drawingToBitmap = drawingToBitmap; + } + } + } + + protected void updatePosition() { + float halfWidth = getMeasuredWidth() / 2.0f; + float halfHeight = getMeasuredHeight() / 2.0f; + setX(getPositionX() - halfWidth); + setY(getPositionY() - halfHeight); + updateSelectionView(); + if (usesBackgroundPaint) { + invalidateAll(); + } + } + + public boolean firstMeasure = true; + + @Override + protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { +// dateCell.measure(container.getMeasuredWidth() > 0 ? MeasureSpec.makeMeasureSpec(container.getMeasuredWidth(), MeasureSpec.EXACTLY) : widthMeasureSpec, heightMeasureSpec); +// container.measure(widthMeasureSpec, MeasureSpec.makeMeasureSpec(MeasureSpec.getSize(heightMeasureSpec) - dateCell.getMeasuredHeight(), MeasureSpec.getMode(heightMeasureSpec))); +// dateCell.measure(container.getMeasuredWidth() > 0 ? MeasureSpec.makeMeasureSpec(container.getMeasuredWidth(), MeasureSpec.EXACTLY) : widthMeasureSpec, heightMeasureSpec); + container.measure(widthMeasureSpec, heightMeasureSpec); + setMeasuredDimension(container.getMeasuredWidth(), container.getMeasuredHeight()); + updatePosition(); + if (firstMeasure) { + int maxWidth = MeasureSpec.getSize(widthMeasureSpec) - dp(22 * 2); + int maxHeight = MeasureSpec.getSize(heightMeasureSpec) - dp(96 * 2); + + int width = getMeasuredWidth(); + int height = getMeasuredHeight(); + + float scale = Math.min((float) maxWidth / width, (float) maxHeight / height); + if (scale < 1) { + setScale(scale); + } + Point p = getPosition(); + p.x -= dp(19) * Math.min(1, scale); + setPosition(p); + + firstMeasure = false; + } + } + + @Override + public Rect getSelectionBounds() { + ViewGroup parentView = (ViewGroup) getParent(); + if (parentView == null) { + return new Rect(); + } + float scale = parentView.getScaleX(); + return new Rect( + getPositionX() * scale - getMeasuredWidth() * getScale() / 2.0f * scale - dp(1.0f + 19.5f + 15), + getPositionY() * scale - getMeasuredHeight() * getScale() / 2.0f * scale - dp(1.0f + 19.5f + 15), + (getMeasuredWidth() * getScale()) * scale + dp((1.0f + 19.5f + 15) * 2), + (getMeasuredHeight() * getScale()) * scale + dp((1.0f + 19.5f + 15) * 2) + ); + } + + @Override + protected float getBounceScale() { + return 0.02f; + } + + @Override + protected SelectionView createSelectionView() { + return new MessageEntityViewSelectionView(getContext()); + } + + public class MessageEntityViewSelectionView extends SelectionView { + + private final Paint clearPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + + public MessageEntityViewSelectionView(Context context) { + super(context); + clearPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.CLEAR)); + } + + @Override + protected int pointInsideHandle(float x, float y) { + float thickness = dp(1.0f); + float radius = dp(19.5f); + + float inset = radius + thickness; + float width = getMeasuredWidth() - inset * 2; + float height = getMeasuredHeight() - inset * 2; + + float middle = inset + height / 2.0f; + + if (x > inset - radius && y > middle - radius && x < inset + radius && y < middle + radius) { + return SELECTION_LEFT_HANDLE; + } else if (x > inset + width - radius && y > middle - radius && x < inset + width + radius && y < middle + radius) { + return SELECTION_RIGHT_HANDLE; + } + + if (x > inset && x < width && y > inset && y < height) { + return SELECTION_WHOLE_HANDLE; + } + + return 0; + } + + private Path path = new Path(); + + @Override + protected void onDraw(Canvas canvas) { + super.onDraw(canvas); + + int count = canvas.getSaveCount(); + + float alpha = getShowAlpha(); + if (alpha <= 0) { + return; + } else if (alpha < 1) { + canvas.saveLayerAlpha(0, 0, getWidth(), getHeight(), (int) (0xFF * alpha), Canvas.ALL_SAVE_FLAG); + } + + float thickness = dp(2.0f); + float radius = AndroidUtilities.dpf2(5.66f); + + float inset = radius + thickness + dp(15); + + float width = getMeasuredWidth() - inset * 2; + float height = getMeasuredHeight() - inset * 2; + + AndroidUtilities.rectTmp.set(inset, inset, inset + width, inset + height); + + float R = dp(12); + float rx = Math.min(R, width / 2f), ry = Math.min(R, height / 2f); + + path.rewind(); + AndroidUtilities.rectTmp.set(inset, inset, inset + rx * 2, inset + ry * 2); + path.arcTo(AndroidUtilities.rectTmp, 180, 90); + AndroidUtilities.rectTmp.set(inset + width - rx * 2, inset, inset + width, inset + ry * 2); + path.arcTo(AndroidUtilities.rectTmp, 270, 90); + canvas.drawPath(path, paint); + + path.rewind(); + AndroidUtilities.rectTmp.set(inset, inset + height - ry * 2, inset + rx * 2, inset + height); + path.arcTo(AndroidUtilities.rectTmp, 180, -90); + AndroidUtilities.rectTmp.set(inset + width - rx * 2, inset + height - ry * 2, inset + width, inset + height); + path.arcTo(AndroidUtilities.rectTmp, 90, -90); + canvas.drawPath(path, paint); + + canvas.drawCircle(inset, inset + height / 2.0f, radius, dotStrokePaint); + canvas.drawCircle(inset, inset + height / 2.0f, radius - dp(1) + 1, dotPaint); + + canvas.drawCircle(inset + width, inset + height / 2.0f, radius, dotStrokePaint); + canvas.drawCircle(inset + width, inset + height / 2.0f, radius - dp(1) + 1, dotPaint); + + canvas.saveLayerAlpha(0, 0, getWidth(), getHeight(), 0xFF, Canvas.ALL_SAVE_FLAG); + + canvas.drawLine(inset, inset + ry, inset, inset + height - ry, paint); + canvas.drawLine(inset + width, inset + ry, inset + width, inset + height - ry, paint); + canvas.drawCircle(inset + width, inset + height / 2.0f, radius + dp(1) - 1, clearPaint); + canvas.drawCircle(inset, inset + height / 2.0f, radius + dp(1) - 1, clearPaint); + + canvas.restoreToCount(count); + } + } + + private boolean isDark = Theme.isCurrentThemeDark(); + private final SparseIntArray currentColors = new SparseIntArray(); + public final Theme.ResourcesProvider resourcesProvider = new Theme.ResourcesProvider() { + public final TextPaint chat_actionTextPaint = new TextPaint(); + public final TextPaint chat_actionTextPaint2 = new TextPaint(); + public final TextPaint chat_botButtonPaint = new TextPaint(); + + public final Paint chat_actionBackgroundPaint = new Paint(Paint.ANTI_ALIAS_FLAG | Paint.FILTER_BITMAP_FLAG); + public final Paint chat_actionBackgroundSelectedPaint = new Paint(Paint.ANTI_ALIAS_FLAG | Paint.FILTER_BITMAP_FLAG); + public final Paint chat_actionBackgroundGradientDarkenPaint = new Paint(Paint.ANTI_ALIAS_FLAG | Paint.FILTER_BITMAP_FLAG); + { + chat_actionTextPaint.setTextSize(AndroidUtilities.dp(Math.max(16, SharedConfig.fontSize) - 2)); + chat_actionTextPaint2.setTextSize(AndroidUtilities.dp(Math.max(16, SharedConfig.fontSize) - 2)); + chat_botButtonPaint.setTextSize(AndroidUtilities.dp(15)); + chat_botButtonPaint.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); + chat_actionBackgroundGradientDarkenPaint.setColor(0x15000000); + } + + @Override + public int getColor(int key) { + return currentColors.get(key, Theme.getColor(key)); + } + + @Override + public Paint getPaint(String paintKey) { + switch (paintKey) { + case Theme.key_paint_chatActionBackgroundSelected: return chat_actionBackgroundSelectedPaint; + case Theme.key_paint_chatActionBackgroundDarken: return chat_actionBackgroundGradientDarkenPaint; + case Theme.key_paint_chatActionText: return chat_actionTextPaint; + case Theme.key_paint_chatActionText2: return chat_actionTextPaint2; + case Theme.key_paint_chatBotButton: return chat_botButtonPaint; + } + return Theme.ResourcesProvider.super.getPaint(paintKey); + } + + @Override + public Drawable getDrawable(String drawableKey) { + if (drawableKey.equals(Theme.key_drawable_msgIn)) { + if (msgInDrawable == null) { + msgInDrawable = new Theme.MessageDrawable(Theme.MessageDrawable.TYPE_TEXT, false, false, resourcesProvider); + } + return msgInDrawable; + } + if (drawableKey.equals(Theme.key_drawable_msgInSelected)) { + if (msgInDrawableSelected == null) { + msgInDrawableSelected = new Theme.MessageDrawable(Theme.MessageDrawable.TYPE_TEXT, false, true, resourcesProvider); + } + return msgInDrawableSelected; + } + if (drawableKey.equals(Theme.key_drawable_msgOut)) { + if (msgOutDrawable == null) { + msgOutDrawable = new Theme.MessageDrawable(Theme.MessageDrawable.TYPE_TEXT, true, false, resourcesProvider); + } + return msgOutDrawable; + } + if (drawableKey.equals(Theme.key_drawable_msgOutSelected)) { + if (msgOutDrawableSelected == null) { + msgOutDrawableSelected = new Theme.MessageDrawable(Theme.MessageDrawable.TYPE_TEXT, true, true, resourcesProvider); + } + return msgOutDrawableSelected; + } + + if (drawableKey.equals(Theme.key_drawable_msgInMedia)) { + if (msgMediaInDrawable == null) { + msgMediaInDrawable = new Theme.MessageDrawable(Theme.MessageDrawable.TYPE_MEDIA, false, false, resourcesProvider); + } + msgMediaInDrawable.invalidateSelf(); + return msgMediaInDrawable; + } + if (drawableKey.equals(Theme.key_drawable_msgInMediaSelected)) { + if (msgMediaInDrawableSelected == null) { + msgMediaInDrawableSelected = new Theme.MessageDrawable(Theme.MessageDrawable.TYPE_MEDIA, false, true, resourcesProvider); + } + return msgMediaInDrawableSelected; + } + if (drawableKey.equals(Theme.key_drawable_msgOutMedia)) { + if (msgMediaOutDrawable == null) { + msgMediaOutDrawable = new Theme.MessageDrawable(Theme.MessageDrawable.TYPE_MEDIA, true, false, resourcesProvider); + } + return msgMediaOutDrawable; + } + if (drawableKey.equals(Theme.key_drawable_msgOutMediaSelected)) { + if (msgMediaOutDrawableSelected == null) { + msgMediaOutDrawableSelected = new Theme.MessageDrawable(Theme.MessageDrawable.TYPE_MEDIA, true, true, resourcesProvider); + } + return msgMediaOutDrawableSelected; + } + + return Theme.getThemeDrawable(drawableKey); + } + + @Override + public boolean isDark() { + return isDark; + } + }; + private Theme.MessageDrawable msgInDrawable, msgInDrawableSelected; + private Theme.MessageDrawable msgOutDrawable, msgOutDrawableSelected; + private Theme.MessageDrawable msgMediaInDrawable, msgMediaInDrawableSelected; + private Theme.MessageDrawable msgMediaOutDrawable, msgMediaOutDrawableSelected; + + public void setupTheme(StoryEntry entry) { + if (entry == null) { + currentColors.clear(); + return; + } + + SharedPreferences preferences = ApplicationLoader.applicationContext.getSharedPreferences("themeconfig", Activity.MODE_PRIVATE); + String dayThemeName = preferences.getString("lastDayTheme", "Blue"); + if (Theme.getTheme(dayThemeName) == null || Theme.getTheme(dayThemeName).isDark()) { + dayThemeName = "Blue"; + } + String nightThemeName = preferences.getString("lastDarkTheme", "Dark Blue"); + if (Theme.getTheme(nightThemeName) == null || !Theme.getTheme(nightThemeName).isDark()) { + nightThemeName = "Dark Blue"; + } + Theme.ThemeInfo themeInfo = Theme.getActiveTheme(); + if (dayThemeName.equals(nightThemeName)) { + if (themeInfo.isDark() || dayThemeName.equals("Dark Blue") || dayThemeName.equals("Night")) { + dayThemeName = "Blue"; + } else { + nightThemeName = "Dark Blue"; + } + } + if (this.isDark = entry.isDark) { + themeInfo = Theme.getTheme(nightThemeName); + } else { + themeInfo = Theme.getTheme(dayThemeName); + } + final String[] wallpaperLink = new String[1]; + final SparseIntArray themeColors; + if (themeInfo.assetName != null) { + themeColors = Theme.getThemeFileValues(null, themeInfo.assetName, wallpaperLink); + } else { + themeColors = Theme.getThemeFileValues(new File(themeInfo.pathToFile), null, wallpaperLink); + } + currentColors.clear(); + int[] defaultColors = Theme.getDefaultColors(); + if (defaultColors != null) { + for (int i = 0; i < defaultColors.length; ++i) { + currentColors.put(i, defaultColors[i]); + } + } + if (themeColors != null) { + for (int i = 0; i < themeColors.size(); ++i) { + currentColors.put(themeColors.keyAt(i), themeColors.valueAt(i)); + } + Theme.ThemeAccent accent = themeInfo.getAccent(false); + if (accent != null) { + accent.fillAccentColors(themeColors, currentColors); + } + } + + invalidateAll(); + } + + public TLRPC.TL_message copyMessage(TLRPC.Message msg) { + TLRPC.TL_message newmsg = new TLRPC.TL_message(); + newmsg.id = msg.id; + newmsg.from_id = msg.from_id; + newmsg.peer_id = msg.peer_id; + newmsg.date = msg.date; + newmsg.expire_date = msg.expire_date; + newmsg.action = msg.action; + newmsg.message = msg.message; + newmsg.media = msg.media; + newmsg.flags = msg.flags; + newmsg.mentioned = msg.mentioned; + newmsg.media_unread = msg.media_unread; + newmsg.out = msg.out; + newmsg.unread = msg.unread; + newmsg.entities = msg.entities; + newmsg.via_bot_name = msg.via_bot_name; + newmsg.reply_markup = msg.reply_markup; + newmsg.views = msg.views; + newmsg.forwards = msg.forwards; + newmsg.replies = msg.replies; + newmsg.edit_date = msg.edit_date; + newmsg.silent = msg.silent; + newmsg.post = msg.post; + newmsg.from_scheduled = msg.from_scheduled; + newmsg.legacy = msg.legacy; + newmsg.edit_hide = msg.edit_hide; + newmsg.pinned = msg.pinned; + newmsg.fwd_from = msg.fwd_from; + newmsg.via_bot_id = msg.via_bot_id; + newmsg.reply_to = msg.reply_to; + newmsg.post_author = msg.post_author; + newmsg.grouped_id = msg.grouped_id; + newmsg.reactions = msg.reactions; + newmsg.restriction_reason = msg.restriction_reason; + newmsg.ttl_period = msg.ttl_period; + newmsg.noforwards = msg.noforwards; + newmsg.invert_media = msg.invert_media; + newmsg.send_state = msg.send_state; + newmsg.fwd_msg_id = msg.fwd_msg_id; + newmsg.attachPath = msg.attachPath; + newmsg.params = msg.params; + newmsg.random_id = msg.random_id; + newmsg.local_id = msg.local_id; + newmsg.dialog_id = msg.dialog_id; + newmsg.ttl = msg.ttl; + newmsg.destroyTime = msg.destroyTime; + newmsg.destroyTimeMillis = msg.destroyTimeMillis; + newmsg.layer = msg.layer; + newmsg.seq_in = msg.seq_in; + newmsg.seq_out = msg.seq_out; + newmsg.with_my_score = msg.with_my_score; + newmsg.replyMessage = msg.replyMessage; + newmsg.reqId = msg.reqId; + newmsg.realId = msg.realId; + newmsg.stickerVerified = msg.stickerVerified; + newmsg.isThreadMessage = msg.isThreadMessage; + newmsg.voiceTranscription = msg.voiceTranscription; + newmsg.voiceTranscriptionOpen = msg.voiceTranscriptionOpen; + newmsg.voiceTranscriptionRated = msg.voiceTranscriptionRated; + newmsg.voiceTranscriptionFinal = msg.voiceTranscriptionFinal; + newmsg.voiceTranscriptionForce = msg.voiceTranscriptionForce; + newmsg.voiceTranscriptionId = msg.voiceTranscriptionId; + newmsg.premiumEffectWasPlayed = msg.premiumEffectWasPlayed; + newmsg.originalLanguage = msg.originalLanguage; + newmsg.translatedToLanguage = msg.translatedToLanguage; + newmsg.translatedText = msg.translatedText; + newmsg.replyStory = msg.replyStory; + return newmsg; + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/PhotoView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/PhotoView.java index f096b18d407..220a6853459 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/PhotoView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/PhotoView.java @@ -1,16 +1,36 @@ package org.telegram.ui.Components.Paint.Views; +import static org.telegram.messenger.AndroidUtilities.dp; + import android.content.Context; +import android.graphics.Bitmap; +import android.graphics.BitmapFactory; +import android.graphics.BitmapShader; import android.graphics.Canvas; +import android.graphics.LinearGradient; +import android.graphics.Matrix; import android.graphics.Paint; import android.graphics.Path; import android.graphics.PorterDuff; import android.graphics.PorterDuffXfermode; import android.graphics.RectF; +import android.graphics.Shader; +import android.graphics.drawable.BitmapDrawable; +import android.graphics.drawable.Drawable; +import android.os.Build; +import android.text.TextUtils; +import android.util.Log; import android.view.ViewGroup; import android.widget.FrameLayout; +import com.google.mlkit.common.MlKitException; +import com.google.mlkit.vision.common.InputImage; +import com.google.mlkit.vision.segmentation.subject.SubjectSegmentation; +import com.google.mlkit.vision.segmentation.subject.SubjectSegmenter; +import com.google.mlkit.vision.segmentation.subject.SubjectSegmenterOptions; + import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.BuildVars; import org.telegram.messenger.FileLoader; import org.telegram.messenger.FileLog; import org.telegram.messenger.ImageLocation; @@ -26,6 +46,11 @@ import org.telegram.ui.Components.RLottieDrawable; import org.telegram.ui.Components.Rect; import org.telegram.ui.Components.Size; +import org.telegram.ui.Stories.recorder.StoryEntry; + +import java.io.File; +import java.io.FileNotFoundException; +import java.io.FileOutputStream; public class PhotoView extends EntityView { @@ -47,9 +72,42 @@ protected void onDraw(Canvas canvas) { private boolean mirrored = false; private final AnimatedFloat mirrorT; private Size baseSize; + private boolean overridenSegmented = false; + + private int orientation, invert; + + private boolean segmented = false; + private AnimatedFloat segmentedT; - private FrameLayoutDrawer containerView; - public final ImageReceiver centerImage = new ImageReceiver(); + private final FrameLayoutDrawer containerView; + public final ImageReceiver centerImage = new ImageReceiver() { + @Override + protected boolean setImageBitmapByKey(Drawable drawable, String key, int type, boolean memCache, int guid) { + if (type == TYPE_IMAGE && drawable instanceof BitmapDrawable) { + segmentImage(((BitmapDrawable) drawable).getBitmap()); + } + return super.setImageBitmapByKey(drawable, key, type, memCache, guid); + } + }; + + private File segmentedFile; + public void preloadSegmented(String path) { + if (TextUtils.isEmpty(path)) return; + segmentingLoading = true; + final int side = Math.round(Math.min(AndroidUtilities.displaySize.x, AndroidUtilities.displaySize.y) * .8f / AndroidUtilities.density); + BitmapFactory.Options opts = new BitmapFactory.Options(); + opts.inJustDecodeBounds = true; + BitmapFactory.decodeFile(path, opts); + opts.inSampleSize = StoryEntry.calculateInSampleSize(opts, side, side); + opts.inJustDecodeBounds = false; + opts.inDither = true; + segmentedImage = BitmapFactory.decodeFile(path, opts); + if (segmentedImage != null) { + segmentedFile = new File(path); + segmentingLoaded = true; + } + segmentingLoading = false; + } public PhotoView(Context context, Point position, float angle, float scale, Size baseSize, String path, int orientation, int invert) { super(context, position); @@ -63,14 +121,16 @@ public PhotoView(Context context, Point position, float angle, float scale, Size addView(containerView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT)); mirrorT = new AnimatedFloat(containerView, 0, 500, CubicBezierInterpolator.EASE_OUT_QUINT); + segmentedT = new AnimatedFloat(containerView, 0, 350, CubicBezierInterpolator.EASE_OUT_QUINT); + this.orientation = orientation; + this.invert = invert; centerImage.setAspectFit(true); centerImage.setInvalidateAll(true); centerImage.setParentView(containerView); - centerImage.setRoundRadius(AndroidUtilities.dp(12)); + centerImage.setRoundRadius(dp(12)); centerImage.setOrientation(orientation, invert, true); - final int side = Math.round(Math.min(AndroidUtilities.displaySize.x, AndroidUtilities.displaySize.y) * .8f / AndroidUtilities.density); - centerImage.setImage(ImageLocation.getForPath(path), side + "_" + side, null, null, null, 1); + centerImage.setImage(ImageLocation.getForPath(path), getImageFilter(), null, null, null, 1); updatePosition(); } @@ -86,21 +146,135 @@ public PhotoView(Context context, Point position, float angle, float scale, Size addView(containerView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT)); mirrorT = new AnimatedFloat(containerView, 0, 500, CubicBezierInterpolator.EASE_OUT_QUINT); + segmentedT = new AnimatedFloat(containerView, 0, 350, CubicBezierInterpolator.EASE_OUT_QUINT); centerImage.setAspectFit(true); centerImage.setInvalidateAll(true); centerImage.setParentView(containerView); - centerImage.setRoundRadius(AndroidUtilities.dp(12)); - final int side = Math.round(Math.min(AndroidUtilities.displaySize.x, AndroidUtilities.displaySize.y) * .8f / AndroidUtilities.density); + centerImage.setRoundRadius(dp(12)); + if (object instanceof TLRPC.Photo) { TLRPC.Photo photo = (TLRPC.Photo) object; TLRPC.PhotoSize photoSize = FileLoader.getClosestPhotoSizeWithSize(photo.sizes, 1000); TLRPC.PhotoSize thumbPhotoSize = FileLoader.getClosestPhotoSizeWithSize(photo.sizes, 90); - centerImage.setImage(ImageLocation.getForPhoto(photoSize, photo), side + "_" + side, ImageLocation.getForPhoto(thumbPhotoSize, photo), side + "_" + side, (String) null, null, 1); + centerImage.setImage(ImageLocation.getForPhoto(photoSize, photo), getImageFilter(), ImageLocation.getForPhoto(thumbPhotoSize, photo), getImageFilter(), (String) null, null, 1); } updatePosition(); } + private String getImageFilter() { + final int side = Math.round(Math.min(AndroidUtilities.displaySize.x, AndroidUtilities.displaySize.y) * .8f / AndroidUtilities.density); + return side + "_" + side; + } + + private boolean segmentingLoading, segmentingLoaded; + public Bitmap segmentedImage; + public void segmentImage(Bitmap source) { + if (segmentingLoaded || segmentingLoading || source == null) return; + if (Build.VERSION.SDK_INT < 24) return; + SubjectSegmenter segmenter = SubjectSegmentation.getClient(new SubjectSegmenterOptions.Builder().enableForegroundBitmap().build()); + segmentingLoading = true; + InputImage inputImage = InputImage.fromBitmap(source, orientation); + segmenter.process(inputImage) + .addOnSuccessListener(result -> { + segmentingLoaded = true; + segmentingLoading = false; + segmentedImage = result.getForegroundBitmap(); + highlightSegmented(); + }) + .addOnFailureListener(error -> { + segmentingLoading = false; + FileLog.e(error); + if (isWaitingMlKitError(error) && isAttachedToWindow()) { + AndroidUtilities.runOnUIThread(() -> segmentImage(source), 2000); + } else { + segmentingLoaded = true; + } + }); + } + + public boolean hasSegmentedImage() { + return segmentedImage != null; + } + + public static boolean isWaitingMlKitError(Exception e) { + if (Build.VERSION.SDK_INT < 24) return false; + return e instanceof MlKitException && e.getMessage() != null && e.getMessage().contains("segmentation optional module to be downloaded"); + } + + public File saveSegmentedImage(int currentAccount) { + if (segmentedImage == null) { + return null; + } + if (segmentedFile == null) { + segmentedFile = StoryEntry.makeCacheFile(currentAccount, "webp"); + try { + segmentedImage.compress(Bitmap.CompressFormat.WEBP, 100, new FileOutputStream(segmentedFile)); + } catch (FileNotFoundException e) { + e.printStackTrace(); + } + } + return segmentedFile; + } + + public void deleteSegmentedFile() { + if (segmentedFile != null) { + try { + segmentedFile.delete(); + } catch (Exception e) {} + segmentedFile = null; + } + } + + public void onSwitchSegmentedAnimationStarted(boolean thanos) { + overridenSegmented = true; + if (containerView != null) { + containerView.invalidate(); + } + } + + public Bitmap getSegmentedOutBitmap() { + if (!(centerImage.getImageDrawable() instanceof BitmapDrawable)) + return null; + + Bitmap source = ((BitmapDrawable) centerImage.getImageDrawable()).getBitmap(); + Bitmap mask = segmentedImage; + + if (source == null || mask == null) + return null; + + int w = source.getWidth(), h = source.getHeight(); + if (orientation == 90 || orientation == 270 || orientation == -90 || orientation == -270) { + w = source.getHeight(); + h = source.getWidth(); + } + Bitmap bitmap = Bitmap.createBitmap(w, h, Bitmap.Config.ARGB_8888); + Canvas canvas = new Canvas(bitmap); + + roundRectPath.rewind(); + AndroidUtilities.rectTmp.set(0, 0, w, h); + float mirrorT = this.mirrorT.get(); + canvas.scale(1 - mirrorT * 2, 1f, w / 2f, 0); + canvas.skew(0, 4 * mirrorT * (1f - mirrorT) * .25f); + roundRectPath.addRoundRect(AndroidUtilities.rectTmp, dp(12) * getScaleX(), dp(12) * getScaleY(), Path.Direction.CW); + canvas.clipPath(roundRectPath); + canvas.translate(w / 2f, h / 2f); + canvas.rotate(orientation); + canvas.translate(-source.getWidth() / 2f, -source.getHeight() / 2f); + + AndroidUtilities.rectTmp.set(0, 0, source.getWidth(), source.getHeight()); + canvas.saveLayerAlpha(AndroidUtilities.rectTmp, 0xFF, Canvas.ALL_SAVE_FLAG); + canvas.drawBitmap(source, 0, 0, null); + Paint clearPaint = new Paint(Paint.ANTI_ALIAS_FLAG | Paint.FILTER_BITMAP_FLAG); + clearPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_OUT)); + canvas.save(); + canvas.drawBitmap(mask, 0, 0, clearPaint); + canvas.restore(); + canvas.restore(); + + return bitmap; + } + @Override protected void onDetachedFromWindow() { super.onDetachedFromWindow(); @@ -126,13 +300,32 @@ public void mirror(boolean animated) { if (!animated) { mirrorT.set(mirrored, true); } - containerView.invalidate(); + if (containerView != null) { + containerView.invalidate(); + } } public boolean isMirrored() { return mirrored; } + public boolean isSegmented() { + return segmented; + } + + public void toggleSegmented(boolean animated) { + segmented = !segmented; + if (animated && segmented) { + overridenSegmented = false; + } + if (!animated) { + segmentedT.set(segmented, true); + } + if (containerView != null) { + containerView.invalidate(); + } + } + protected void updatePosition() { float halfWidth = baseSize.width / 2.0f; float halfHeight = baseSize.height / 2.0f; @@ -141,6 +334,17 @@ protected void updatePosition() { updateSelectionView(); } + private final android.graphics.Rect src = new android.graphics.Rect(); + private final android.graphics.RectF dest = new android.graphics.RectF(); + + private final Paint segmentPaint = new Paint(Paint.ANTI_ALIAS_FLAG | Paint.FILTER_BITMAP_FLAG); + + private long highlightStart = -1; + private LinearGradient highlightGradient; + private Matrix highlightGradientMatrix; + private Paint highlightPaint; + private boolean needHighlight; + protected void stickerDraw(Canvas canvas) { if (containerView == null) { return; @@ -150,11 +354,98 @@ protected void stickerDraw(Canvas canvas) { float mirrorT = this.mirrorT.set(mirrored); canvas.scale(1 - mirrorT * 2, 1f, baseSize.width / 2f, 0); canvas.skew(0, 4 * mirrorT * (1f - mirrorT) * .25f); - centerImage.setImageCoords(0, 0, (int) baseSize.width, (int) baseSize.height); - centerImage.draw(canvas); + + final float segmentedT = this.segmentedT.set(segmented); + if (!segmented) { + centerImage.setAlpha(1f - segmentedT); + centerImage.setImageCoords(0, 0, (int) baseSize.width, (int) baseSize.height); + centerImage.draw(canvas); + if (segmentedT > 0) { + drawSegmented(canvas); + } + + if (segmentedImage != null) { + canvas.saveLayerAlpha(0, 0, baseSize.width, baseSize.height, 0xFF, Canvas.ALL_SAVE_FLAG); + drawSegmented(canvas); + canvas.save(); + final long now = System.currentTimeMillis(); + if (highlightStart <= 0) { + highlightStart = now; + } + final float gradientWidth = .80f * baseSize.width; + final float highlightT = (now - highlightStart) / 1000f; + final float translate = highlightT * (2 * gradientWidth + baseSize.width) - gradientWidth; + if (highlightPaint == null) { + highlightPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + highlightPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.SRC_IN)); + highlightGradient = new LinearGradient(0, 0, gradientWidth, 0, new int[]{0x00feee8c, 0x66feee8c, 0x66feee8c, 0x00feee8c}, new float[]{0, .4f, .6f, 1f}, Shader.TileMode.CLAMP); + highlightGradientMatrix = new Matrix(); + highlightGradient.setLocalMatrix(highlightGradientMatrix); + highlightPaint.setShader(highlightGradient); + } + highlightGradientMatrix.reset(); + highlightGradientMatrix.postTranslate(translate, 0); + highlightGradient.setLocalMatrix(highlightGradientMatrix); + canvas.drawRect(0, 0, (int) baseSize.width, (int) baseSize.height, highlightPaint); + canvas.restore(); + canvas.restore(); + + if ((highlightT > 0 || needHighlight) && highlightT < 1f) { + needHighlight = false; + containerView.invalidate(); + } + } + } else { + highlightStart = -1; + needHighlight = false; + if (!overridenSegmented) { + centerImage.setImageCoords(0, 0, (int) baseSize.width, (int) baseSize.height); + centerImage.setAlpha(1f); + centerImage.draw(canvas); + } + drawSegmented(canvas); + } + canvas.restore(); } + private Path roundRectPath; + private void drawSegmented(Canvas canvas) { + if (segmentedImage == null) return; + src.set(0, 0, segmentedImage.getWidth(), segmentedImage.getHeight()); + int bitmapWidth = segmentedImage.getWidth(), bitmapHeight = segmentedImage.getHeight(); + if (orientation == 90 || orientation == 270 || orientation == -90 || orientation == -270) { + bitmapWidth = segmentedImage.getHeight(); + bitmapHeight = segmentedImage.getWidth(); + } + final float scale = Math.max(bitmapWidth / baseSize.width, bitmapHeight / baseSize.height); + final float bitmapW = segmentedImage.getWidth() / scale; + final float bitmapH = segmentedImage.getHeight() / scale; + dest.set((baseSize.width - bitmapW) / 2, (baseSize.height - bitmapH) / 2, (baseSize.width + bitmapW) / 2, (baseSize.height + bitmapH) / 2); + canvas.save(); + if (orientation != 0) { + canvas.rotate(orientation, dest.centerX(), dest.centerY()); + } + if (roundRectPath == null) { + roundRectPath = new Path(); + } + roundRectPath.rewind(); + roundRectPath.addRoundRect(dest, dp(12), dp(12), Path.Direction.CW); + canvas.clipPath(roundRectPath); + canvas.drawBitmap(segmentedImage, src, dest, segmentPaint); + canvas.restore(); + } + + public void highlightSegmented() { + needHighlight = true; + if (highlightStart <= 0 || System.currentTimeMillis() - highlightStart >= 1000) { + highlightStart = System.currentTimeMillis(); + } + if (containerView != null) { + containerView.invalidate(); + } + } + public long getDuration() { RLottieDrawable rLottieDrawable = centerImage.getLottieAnimation(); if (rLottieDrawable != null) { @@ -174,14 +465,14 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { } @Override - protected Rect getSelectionBounds() { + public Rect getSelectionBounds() { ViewGroup parentView = (ViewGroup) getParent(); if (parentView == null) { return new Rect(); } float scale = parentView.getScaleX(); - float width = getMeasuredWidth() * getScale() + AndroidUtilities.dp(64) / scale; - float height = getMeasuredHeight() * getScale() + AndroidUtilities.dp(64) / scale; + float width = getMeasuredWidth() * getScale() + dp(64) / scale; + float height = getMeasuredHeight() * getScale() + dp(64) / scale; float left = (getPositionX() - width / 2.0f) * scale; float right = left + width * scale; return new Rect(left, (getPositionY() - height / 2.0f) * scale, right - left, height * scale); @@ -217,8 +508,8 @@ public PhotoViewSelectionView(Context context) { @Override protected int pointInsideHandle(float x, float y) { - float thickness = AndroidUtilities.dp(1.0f); - float radius = AndroidUtilities.dp(19.5f); + float thickness = dp(1.0f); + float radius = dp(19.5f); float inset = radius + thickness; float width = getMeasuredWidth() - inset * 2; @@ -254,17 +545,17 @@ protected void onDraw(Canvas canvas) { canvas.saveLayerAlpha(0, 0, getWidth(), getHeight(), (int) (0xFF * alpha), Canvas.ALL_SAVE_FLAG); } - float thickness = AndroidUtilities.dp(2.0f); + float thickness = dp(2.0f); float radius = AndroidUtilities.dpf2(5.66f); - float inset = radius + thickness + AndroidUtilities.dp(15); + float inset = radius + thickness + dp(15); float width = getMeasuredWidth() - inset * 2; float height = getMeasuredHeight() - inset * 2; AndroidUtilities.rectTmp.set(inset, inset, inset + width, inset + height); - float R = AndroidUtilities.dp(12); + float R = dp(12); float rx = Math.min(R, width / 2f), ry = Math.min(R, height / 2f); path.rewind(); @@ -282,17 +573,17 @@ protected void onDraw(Canvas canvas) { canvas.drawPath(path, paint); canvas.drawCircle(inset, inset + height / 2.0f, radius, dotStrokePaint); - canvas.drawCircle(inset, inset + height / 2.0f, radius - AndroidUtilities.dp(1) + 1, dotPaint); + canvas.drawCircle(inset, inset + height / 2.0f, radius - dp(1) + 1, dotPaint); canvas.drawCircle(inset + width, inset + height / 2.0f, radius, dotStrokePaint); - canvas.drawCircle(inset + width, inset + height / 2.0f, radius - AndroidUtilities.dp(1) + 1, dotPaint); + canvas.drawCircle(inset + width, inset + height / 2.0f, radius - dp(1) + 1, dotPaint); canvas.saveLayerAlpha(0, 0, getWidth(), getHeight(), 0xFF, Canvas.ALL_SAVE_FLAG); canvas.drawLine(inset, inset + ry, inset, inset + height - ry, paint); canvas.drawLine(inset + width, inset + ry, inset + width, inset + height - ry, paint); - canvas.drawCircle(inset + width, inset + height / 2.0f, radius + AndroidUtilities.dp(1) - 1, clearPaint); - canvas.drawCircle(inset, inset + height / 2.0f, radius + AndroidUtilities.dp(1) - 1, clearPaint); + canvas.drawCircle(inset + width, inset + height / 2.0f, radius + dp(1) - 1, clearPaint); + canvas.drawCircle(inset, inset + height / 2.0f, radius + dp(1) - 1, clearPaint); canvas.restoreToCount(count); } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/ReactionWidgetEntityView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/ReactionWidgetEntityView.java index 6982f1ddc04..1c50712edfa 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/ReactionWidgetEntityView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/ReactionWidgetEntityView.java @@ -129,7 +129,7 @@ public int getPadding() { } @Override - protected Rect getSelectionBounds() { + public Rect getSelectionBounds() { ViewGroup parentView = (ViewGroup) getParent(); if (parentView == null) { return new Rect(); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/RoundView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/RoundView.java index 1d9488f3cd6..41ec51869f9 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/RoundView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/RoundView.java @@ -192,7 +192,7 @@ protected void updatePosition() { } @Override - protected org.telegram.ui.Components.Rect getSelectionBounds() { + public org.telegram.ui.Components.Rect getSelectionBounds() { ViewGroup parentView = (ViewGroup) getParent(); if (parentView == null) { return new org.telegram.ui.Components.Rect(); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/StickerView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/StickerView.java index 97119d0495d..ce80842776f 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/StickerView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/StickerView.java @@ -174,7 +174,7 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { } @Override - protected Rect getSelectionBounds() { + public Rect getSelectionBounds() { ViewGroup parentView = (ViewGroup) getParent(); if (parentView == null) { return new Rect(); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/TextPaintView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/TextPaintView.java index 25a7decd0b0..41386272b3e 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/TextPaintView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Paint/Views/TextPaintView.java @@ -382,7 +382,7 @@ public void updateColor() { } @Override - protected Rect getSelectionBounds() { + public Rect getSelectionBounds() { ViewGroup parentView = (ViewGroup) getParent(); if (parentView == null) { return new Rect(); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/PhotoViewerCaptionEnterView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/PhotoViewerCaptionEnterView.java index 2a8bf63c77a..0132948ebff 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/PhotoViewerCaptionEnterView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/PhotoViewerCaptionEnterView.java @@ -363,7 +363,7 @@ public void onAnimationEnd(Animator animation) { sendButtonColorAnimator.setDuration(150).start(); } - if (photoViewer.getParentAlert() != null && !photoViewer.getParentAlert().captionLimitBulletinShown && !MessagesController.getInstance(currentAccount).premiumLocked && !UserConfig.getInstance(currentAccount).isPremium() && codePointCount > MessagesController.getInstance(currentAccount).captionLengthLimitDefault && codePointCount < MessagesController.getInstance(currentAccount).captionLengthLimitPremium) { + if (photoViewer.getParentAlert() != null && !photoViewer.getParentAlert().captionLimitBulletinShown && !MessagesController.getInstance(currentAccount).premiumFeaturesBlocked() && !UserConfig.getInstance(currentAccount).isPremium() && codePointCount > MessagesController.getInstance(currentAccount).captionLengthLimitDefault && codePointCount < MessagesController.getInstance(currentAccount).captionLengthLimitPremium) { photoViewer.getParentAlert().captionLimitBulletinShown = true; if (heightShouldBeChanged) { AndroidUtilities.runOnUIThread(()->photoViewer.showCaptionLimitBulletin(parent), 300); @@ -390,7 +390,7 @@ public void onAnimationEnd(Animator animation) { captionLimitView.performHapticFeedback(HapticFeedbackConstants.KEYBOARD_TAP, HapticFeedbackConstants.FLAG_IGNORE_GLOBAL_SETTING); } catch (Exception ignored) {} - if (!MessagesController.getInstance(currentAccount).premiumLocked && MessagesController.getInstance(currentAccount).captionLengthLimitPremium > codePointCount) { + if (!MessagesController.getInstance(currentAccount).premiumFeaturesBlocked() && MessagesController.getInstance(currentAccount).captionLengthLimitPremium > codePointCount) { photoViewer.showCaptionLimitBulletin(parent); } return; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/LimitReachedBottomSheet.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/LimitReachedBottomSheet.java index 56edb641139..573dbce89bb 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/LimitReachedBottomSheet.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/LimitReachedBottomSheet.java @@ -10,6 +10,7 @@ import android.content.Context; import android.content.DialogInterface; import android.graphics.Canvas; +import android.graphics.Color; import android.graphics.Paint; import android.graphics.PorterDuff; import android.graphics.PorterDuffColorFilter; @@ -57,6 +58,7 @@ import org.telegram.tgnet.tl.TL_stories; import org.telegram.ui.ActionBar.AlertDialog; import org.telegram.ui.ActionBar.BaseFragment; +import org.telegram.ui.ActionBar.SimpleTextView; import org.telegram.ui.ActionBar.Theme; import org.telegram.ui.Cells.AdminedChannelCell; import org.telegram.ui.Cells.GroupCreateUserCell; @@ -84,6 +86,7 @@ import org.telegram.ui.Components.RecyclerItemsEnterAnimator; import org.telegram.ui.Components.RecyclerListView; import org.telegram.ui.Components.ScaleStateListAnimator; +import org.telegram.ui.Components.TypefaceSpan; import org.telegram.ui.LaunchActivity; import org.telegram.ui.PremiumPreviewFragment; import org.telegram.ui.ProfileActivity; @@ -120,6 +123,12 @@ public class LimitReachedBottomSheet extends BottomSheetWithRecyclerListView imp public static final int TYPE_BOOSTS_FOR_USERS = 19; public static final int TYPE_BOOSTS_FOR_COLOR = 20; public static final int TYPE_BOOSTS_FOR_REACTIONS = 21; + public static final int TYPE_BOOSTS_FOR_WALLPAPER = 22; + public static final int TYPE_BOOSTS_FOR_CUSTOM_WALLPAPER = 23; + public static final int TYPE_BOOSTS_FOR_PROFILE_COLOR = 24; + public static final int TYPE_BOOSTS_FOR_EMOJI_STATUS = 25; + public static final int TYPE_BOOSTS_FOR_REPLY_ICON = 26; + public static final int TYPE_BOOSTS_FOR_PROFILE_ICON = 27; private boolean canSendLink; private int linkRow = -1; @@ -170,6 +179,81 @@ public static String limitTypeToServerString(int type) { int loadingRow = -1; int emptyViewDividerRow = -1; int bottomRow = -1; + int boostFeaturesStartRow = -1; + ArrayList boostFeatures; + + private static class BoostFeature { + + public final int iconResId; + public final int textKey; + public final String countValue; + public final String textKeyPlural; + public final int countPlural; + + private BoostFeature( + int iconResId, + int textKey, + String countValue, + String textKeyPlural, + int countPlural + ) { + this.iconResId = iconResId; + this.textKey = textKey; + this.countValue = countValue; + this.textKeyPlural = textKeyPlural; + this.countPlural = countPlural; + } + + public static BoostFeature of(int iconResId, int textKey) { + return new BoostFeature(iconResId, textKey, null, null, -1); + } + + public static BoostFeature of(int iconResId, int textKey, String count) { + return new BoostFeature(iconResId, textKey, count, null, -1); + } + + public static BoostFeature of(int iconResId, String textKeyPlural, int count) { + return new BoostFeature(iconResId, -1, null, textKeyPlural, count); + } + + public boolean equals(BoostFeature that) { + if (that == null) { + return false; + } + return ( + this.iconResId == that.iconResId && + this.textKey == that.textKey && + TextUtils.equals(this.countValue, that.countValue) && + TextUtils.equals(this.textKeyPlural, that.textKeyPlural) && + this.countPlural == that.countPlural + ); + } + + public static boolean arraysEqual(ArrayList a, ArrayList b) { + if (a == null && b == null) return true; + if (a != null && b == null || a == null && b != null) return false; + if (a.size() != b.size()) return false; + for (int i = 0; i < a.size(); ++i) { + if (!a.get(i).equals(b.get(i))) { + return false; + } + } + return true; + } + + public static class BoostFeatureLevel extends BoostFeature { + public final int lvl; + public final boolean isFirst; + public BoostFeatureLevel(int lvl) { + this(lvl, false); + } + public BoostFeatureLevel(int lvl, boolean isFirst) { + super(-1, -1, null, null, -1); + this.lvl = lvl; + this.isFirst = isFirst; + } + } + } public boolean parentIsChannel; private int currentValue = -1; @@ -213,8 +297,21 @@ public LimitReachedBottomSheet(BaseFragment fragment, Context context, int type, fireworksOverlay = new FireworksOverlay(getContext()); container.addView(fireworksOverlay, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT)); } - if (type == TYPE_BOOSTS_FOR_POSTING || type == TYPE_BOOSTS_FOR_COLOR || type == TYPE_BOOSTS_FOR_REACTIONS) { + if ( + type == TYPE_BOOSTS_FOR_POSTING || + type == TYPE_BOOSTS_FOR_COLOR || + type == TYPE_BOOSTS_FOR_PROFILE_COLOR || + type == TYPE_BOOSTS_FOR_EMOJI_STATUS || + type == TYPE_BOOSTS_FOR_WALLPAPER || + type == TYPE_BOOSTS_FOR_CUSTOM_WALLPAPER || + type == TYPE_BOOSTS_FOR_REACTIONS || + type == TYPE_BOOSTS_FOR_REPLY_ICON || + type == TYPE_BOOSTS_FOR_PROFILE_ICON + ) { ((ViewGroup) premiumButtonView.getParent()).removeView(premiumButtonView); + if (divider != null) { + ((ViewGroup) divider.getParent()).removeView(divider); + } recyclerListView.setPadding(0, 0, 0, 0); actionBtn = new TextView(context); actionBtn.setGravity(Gravity.CENTER); @@ -247,7 +344,17 @@ public void invalidate() { } }; - if (!hasFixedSize) { + if (!hasFixedSize && !( + type == TYPE_BOOSTS_FOR_POSTING || + type == TYPE_BOOSTS_FOR_COLOR || + type == TYPE_BOOSTS_FOR_PROFILE_COLOR || + type == TYPE_BOOSTS_FOR_EMOJI_STATUS || + type == TYPE_BOOSTS_FOR_WALLPAPER || + type == TYPE_BOOSTS_FOR_CUSTOM_WALLPAPER || + type == TYPE_BOOSTS_FOR_REACTIONS || + type == TYPE_BOOSTS_FOR_REPLY_ICON || + type == TYPE_BOOSTS_FOR_PROFILE_ICON + )) { divider = new View(context) { @Override protected void onDraw(Canvas canvas) { @@ -302,7 +409,7 @@ protected void onDraw(Canvas canvas) { if (type == TYPE_BOOSTS_FOR_USERS) { if (canApplyBoost.empty) { if (UserConfig.getInstance(currentAccount).isPremium() && BoostRepository.isMultiBoostsAvailable()) { - BoostDialogs.showMoreBoostsNeeded(dialogId); + BoostDialogs.showMoreBoostsNeeded(dialogId, this); } else { AlertDialog.Builder builder = new AlertDialog.Builder(context, resourcesProvider); builder.setTitle(LocaleController.getString("PremiumNeeded", R.string.PremiumNeeded)); @@ -401,12 +508,22 @@ protected void onDraw(Canvas canvas) { } return; } - if (type == TYPE_BOOSTS_FOR_POSTING || type == TYPE_BOOSTS_FOR_COLOR || type == TYPE_BOOSTS_FOR_REACTIONS) { + if ( + type == TYPE_BOOSTS_FOR_POSTING || + type == TYPE_BOOSTS_FOR_COLOR || + type == TYPE_BOOSTS_FOR_PROFILE_COLOR || + type == TYPE_BOOSTS_FOR_EMOJI_STATUS || + type == TYPE_BOOSTS_FOR_WALLPAPER || + type == TYPE_BOOSTS_FOR_CUSTOM_WALLPAPER || + type == TYPE_BOOSTS_FOR_REACTIONS || + type == TYPE_BOOSTS_FOR_REPLY_ICON || + type == TYPE_BOOSTS_FOR_PROFILE_ICON + ) { AndroidUtilities.addToClipboard(getBoostLink()); dismiss(); return; } - if (UserConfig.getInstance(currentAccount).isPremium() || MessagesController.getInstance(currentAccount).premiumLocked || isVeryLargeFile) { + if (UserConfig.getInstance(currentAccount).isPremium() || MessagesController.getInstance(currentAccount).premiumFeaturesBlocked() || isVeryLargeFile) { dismiss(); return; } @@ -434,7 +551,7 @@ protected void onDraw(Canvas canvas) { } } else { if (canApplyBoost.alreadyActive && BoostRepository.isMultiBoostsAvailable() && !canApplyBoost.isMaxLvl) { - BoostDialogs.showMoreBoostsNeeded(dialogId); + BoostDialogs.showMoreBoostsNeeded(dialogId, this); } else { dismiss(); } @@ -584,12 +701,22 @@ public void updatePremiumButtonText() { } else { premiumButtonView.buttonTextView.setText(LocaleController.getString("BoostChannel", R.string.BoostChannel)); } - } else if (type == TYPE_BOOSTS_FOR_POSTING || type == TYPE_BOOSTS_FOR_COLOR || type == TYPE_BOOSTS_FOR_REACTIONS) { + } else if ( + type == TYPE_BOOSTS_FOR_POSTING || + type == TYPE_BOOSTS_FOR_COLOR || + type == TYPE_BOOSTS_FOR_PROFILE_COLOR || + type == TYPE_BOOSTS_FOR_EMOJI_STATUS || + type == TYPE_BOOSTS_FOR_WALLPAPER || + type == TYPE_BOOSTS_FOR_CUSTOM_WALLPAPER || + type == TYPE_BOOSTS_FOR_REACTIONS || + type == TYPE_BOOSTS_FOR_REPLY_ICON || + type == TYPE_BOOSTS_FOR_PROFILE_ICON + ) { SpannableStringBuilder spannableStringBuilder = new SpannableStringBuilder("d "); spannableStringBuilder.setSpan(new ColoredImageSpan(R.drawable.msg_copy_filled), 0, 1, 0); spannableStringBuilder.append(LocaleController.getString("CopyLink", R.string.CopyLink)); premiumButtonView.buttonTextView.setText(spannableStringBuilder); - } else if (UserConfig.getInstance(currentAccount).isPremium() || MessagesController.getInstance(currentAccount).premiumLocked || isVeryLargeFile) { + } else if (UserConfig.getInstance(currentAccount).isPremium() || MessagesController.getInstance(currentAccount).premiumFeaturesBlocked() || isVeryLargeFile) { premiumButtonView.buttonTextView.setText(LocaleController.getString("OK", R.string.OK)); premiumButtonView.hideIcon(); } else { @@ -697,21 +824,40 @@ private void updateButton() { } private static boolean hasFixedSize(int type) { - if (type == TYPE_PIN_DIALOGS || type == TYPE_FOLDERS || type == TYPE_CHATS_IN_FOLDER || - type == TYPE_LARGE_FILE || type == TYPE_ACCOUNTS || type == TYPE_FOLDER_INVITES || - type == TYPE_SHARED_FOLDERS || type == TYPE_STORIES_COUNT || type == TYPE_STORIES_WEEK || - type == TYPE_STORIES_MONTH || type == TYPE_BOOSTS_FOR_POSTING || type == TYPE_BOOSTS_FOR_COLOR || type == TYPE_BOOSTS_FOR_USERS || type == TYPE_BOOSTS_FOR_REACTIONS) { - return true; - } - return false; + return ( + type == TYPE_PIN_DIALOGS || + type == TYPE_FOLDERS || + type == TYPE_CHATS_IN_FOLDER || + type == TYPE_LARGE_FILE || + type == TYPE_ACCOUNTS || + type == TYPE_FOLDER_INVITES || + type == TYPE_SHARED_FOLDERS || + type == TYPE_STORIES_COUNT || + type == TYPE_STORIES_WEEK || + type == TYPE_STORIES_MONTH + ); } @Override public CharSequence getTitle() { - if (type == TYPE_ADD_MEMBERS_RESTRICTED) { - return LocaleController.getString("ChannelInviteViaLink", R.string.ChannelInviteViaLink); + switch (type) { + case TYPE_BOOSTS_FOR_USERS: + return LocaleController.getString(R.string.BoostChannel); + case TYPE_BOOSTS_FOR_POSTING: + case TYPE_BOOSTS_FOR_COLOR: + case TYPE_BOOSTS_FOR_WALLPAPER: + case TYPE_BOOSTS_FOR_CUSTOM_WALLPAPER: + case TYPE_BOOSTS_FOR_REACTIONS: + case TYPE_BOOSTS_FOR_EMOJI_STATUS: + case TYPE_BOOSTS_FOR_REPLY_ICON: + case TYPE_BOOSTS_FOR_PROFILE_ICON: + case TYPE_BOOSTS_FOR_PROFILE_COLOR: + return LocaleController.getString(R.string.UnlockBoostChannelFeatures); + case TYPE_ADD_MEMBERS_RESTRICTED: + return LocaleController.getString(R.string.ChannelInviteViaLink); + default: + return LocaleController.getString(R.string.LimitReached); } - return LocaleController.getString("LimitReached", R.string.LimitReached); } @Override @@ -719,6 +865,7 @@ public void onAttachedToWindow() { super.onAttachedToWindow(); NotificationCenter.getInstance(currentAccount).addObserver(this, NotificationCenter.boostByChannelCreated); NotificationCenter.getInstance(currentAccount).addObserver(this, NotificationCenter.boostedChannelByUser); + NotificationCenter.getInstance(currentAccount).addObserver(this, NotificationCenter.didStartedMultiGiftsSelector); } @Override @@ -726,6 +873,7 @@ public void onDetachedFromWindow() { super.onDetachedFromWindow(); NotificationCenter.getInstance(currentAccount).removeObserver(this, NotificationCenter.boostByChannelCreated); NotificationCenter.getInstance(currentAccount).removeObserver(this, NotificationCenter.boostedChannelByUser); + NotificationCenter.getInstance(currentAccount).removeObserver(this, NotificationCenter.didStartedMultiGiftsSelector); } @Override @@ -804,10 +952,23 @@ public void didReceivedNotification(int id, int account, Object... args) { String str = LocaleController.formatPluralString("BoostingReassignedFromPlural", size, LocaleController.formatPluralString("BoostingFromOtherChannel", channels)); BulletinFactory bulletinFactory = BulletinFactory.of(container, resourcesProvider); - bulletinFactory.createSimpleBulletinWithIconSize(R.raw.forward, str, 30).setDuration(4000).show(true); + bulletinFactory.createSimpleBulletinWithIconSize(R.raw.ic_boosts_replace, str, 30).setDuration(4000).show(true); + } else if (id == NotificationCenter.didStartedMultiGiftsSelector) { + dismiss(); } } + private static final int VIEW_TYPE_HEADER = 0; + private static final int VIEW_TYPE_CHANNEL = 1; + private static final int VIEW_TYPE_SHADOW = 2; + private static final int VIEW_TYPE_HEADER_CELL = 3; + private static final int VIEW_TYPE_USER = 4; + private static final int VIEW_TYPE_PROGRESS = 5; + private static final int VIEW_TYPE_SPACE16DP = 6; + private static final int VIEW_TYPE_BOOST_LINK = 7; + private static final int VIEW_TYPE_BOOST_FOOTER = 8; + private static final int VIEW_TYPE_BOOST_FEATURE = 9; + @Override public RecyclerListView.SelectionAdapter createAdapter() { return new RecyclerListView.SelectionAdapter() { @@ -816,7 +977,7 @@ public boolean isEnabled(RecyclerView.ViewHolder holder) { if (type == TYPE_ADD_MEMBERS_RESTRICTED && !canSendLink) { return false; } - return holder.getItemViewType() == 1 || holder.getItemViewType() == 4; + return holder.getItemViewType() == VIEW_TYPE_CHANNEL || holder.getItemViewType() == VIEW_TYPE_USER; } @NonNull @@ -825,8 +986,9 @@ public RecyclerView.ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int View view; Context context = parent.getContext(); switch (viewType) { - case 8: + case VIEW_TYPE_BOOST_FOOTER: LinearLayout wrapperLayout = new LinearLayout(context); + wrapperLayout.setPadding(backgroundPaddingLeft + dp(6), 0, backgroundPaddingLeft + dp(6), 0); wrapperLayout.setOrientation(LinearLayout.VERTICAL); LoginOrView orDividerView = new LoginOrView(context); @@ -871,10 +1033,14 @@ public void onClick(@NonNull View view) { view = wrapperLayout; break; - case 7: + case VIEW_TYPE_BOOST_FEATURE: + view = new BoostFeatureCell(context, resourcesProvider); + break; + case VIEW_TYPE_BOOST_LINK: FrameLayout frameLayout = new FrameLayout(getContext()); + frameLayout.setPadding(backgroundPaddingLeft + dp(6), 0, backgroundPaddingLeft + dp(6), 0); TextView linkView = new TextView(context); - linkView.setPadding(AndroidUtilities.dp(18), AndroidUtilities.dp(13), AndroidUtilities.dp(50), AndroidUtilities.dp(13)); + linkView.setPadding(AndroidUtilities.dp(18), AndroidUtilities.dp(13), AndroidUtilities.dp(statisticClickRunnable == null ? 18 : 50), AndroidUtilities.dp(13)); linkView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 16); linkView.setEllipsize(TextUtils.TruncateAt.MIDDLE); linkView.setSingleLine(true); @@ -901,10 +1067,10 @@ public void onClick(@NonNull View view) { view = frameLayout; break; default: - case 0: + case VIEW_TYPE_HEADER: view = headerView = new HeaderView(context); break; - case 1: + case VIEW_TYPE_CHANNEL: view = new AdminedChannelCell(context, new View.OnClickListener() { @Override public void onClick(View v) { @@ -915,17 +1081,17 @@ public void onClick(View v) { } }, true, 9); break; - case 2: + case VIEW_TYPE_SHADOW: view = new ShadowSectionCell(context, 12, Theme.getColor(Theme.key_windowBackgroundGray, resourcesProvider)); break; - case 3: + case VIEW_TYPE_HEADER_CELL: view = new HeaderCell(context); view.setPadding(0, 0, 0, AndroidUtilities.dp(8)); break; - case 4: + case VIEW_TYPE_USER: view = new GroupCreateUserCell(context, 1, 8, false); break; - case 5: + case VIEW_TYPE_PROGRESS: FlickerLoadingView flickerLoadingView = new FlickerLoadingView(context, null); flickerLoadingView.setViewType(type == TYPE_PUBLIC_LINKS ? FlickerLoadingView.LIMIT_REACHED_LINKS : FlickerLoadingView.LIMIT_REACHED_GROUPS); flickerLoadingView.setIsSingleCell(true); @@ -933,7 +1099,7 @@ public void onClick(View v) { flickerLoadingView.setItemsCount(10); view = flickerLoadingView; break; - case 6: + case VIEW_TYPE_SPACE16DP: view = new View(getContext()) { @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { @@ -948,68 +1114,79 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { @Override public void onBindViewHolder(@NonNull RecyclerView.ViewHolder holder, int position) { - if (holder.getItemViewType() == 4) { - GroupCreateUserCell cell = (GroupCreateUserCell) holder.itemView; - if (type == TYPE_TO0_MANY_COMMUNITIES) { - TLRPC.Chat chat = inactiveChats.get(position - chatStartRow); - String signature = inactiveChatsSignatures.get(position - chatStartRow); - cell.setObject(chat, chat.title, signature, position != chatEndRow - 1f); - cell.setChecked(selectedChats.contains(chat), false); - } else if (type == TYPE_ADD_MEMBERS_RESTRICTED) { - TLRPC.User user = restrictedUsers.get(position - chatStartRow); - String signature = LocaleController.formatUserStatus(currentAccount, user, null, null); - cell.setObject(user, ContactsController.formatName(user.first_name, user.last_name), signature, position != chatEndRow - 1f); - cell.setChecked(selectedChats.contains(user), false); - } - } else if (holder.getItemViewType() == 1) { - TLRPC.Chat chat = chats.get(position - chatStartRow); - AdminedChannelCell adminedChannelCell = (AdminedChannelCell) holder.itemView; - TLRPC.Chat oldChat = adminedChannelCell.getCurrentChannel(); - adminedChannelCell.setChannel(chat, false); - adminedChannelCell.setChecked(selectedChats.contains(chat), oldChat == chat); - } else if (holder.getItemViewType() == 3) { - HeaderCell headerCell = (HeaderCell) holder.itemView; - if (type == TYPE_ADD_MEMBERS_RESTRICTED) { - if (canSendLink) { - headerCell.setText(LocaleController.getString("ChannelInviteViaLink", R.string.ChannelInviteViaLink)); - } else { - if (restrictedUsers.size() == 1) { - headerCell.setText(LocaleController.getString("ChannelInviteViaLinkRestricted2", R.string.ChannelInviteViaLinkRestricted2)); + switch (holder.getItemViewType()) { + case VIEW_TYPE_USER: + GroupCreateUserCell cell = (GroupCreateUserCell) holder.itemView; + if (type == TYPE_TO0_MANY_COMMUNITIES) { + TLRPC.Chat chat = inactiveChats.get(position - chatStartRow); + String signature = inactiveChatsSignatures.get(position - chatStartRow); + cell.setObject(chat, chat.title, signature, position != chatEndRow - 1f); + cell.setChecked(selectedChats.contains(chat), false); + } else if (type == TYPE_ADD_MEMBERS_RESTRICTED) { + TLRPC.User user = restrictedUsers.get(position - chatStartRow); + String signature = LocaleController.formatUserStatus(currentAccount, user, null, null); + cell.setObject(user, ContactsController.formatName(user.first_name, user.last_name), signature, position != chatEndRow - 1f); + cell.setChecked(selectedChats.contains(user), false); + } + break; + case VIEW_TYPE_CHANNEL: + TLRPC.Chat chat = chats.get(position - chatStartRow); + AdminedChannelCell adminedChannelCell = (AdminedChannelCell) holder.itemView; + TLRPC.Chat oldChat = adminedChannelCell.getCurrentChannel(); + adminedChannelCell.setChannel(chat, false); + adminedChannelCell.setChecked(selectedChats.contains(chat), oldChat == chat); + break; + case VIEW_TYPE_HEADER_CELL: + HeaderCell headerCell = (HeaderCell) holder.itemView; + if (type == TYPE_ADD_MEMBERS_RESTRICTED) { + if (canSendLink) { + headerCell.setText(LocaleController.getString("ChannelInviteViaLink", R.string.ChannelInviteViaLink)); } else { - headerCell.setText(LocaleController.getString("ChannelInviteViaLinkRestricted3", R.string.ChannelInviteViaLinkRestricted3)); + if (restrictedUsers.size() == 1) { + headerCell.setText(LocaleController.getString("ChannelInviteViaLinkRestricted2", R.string.ChannelInviteViaLinkRestricted2)); + } else { + headerCell.setText(LocaleController.getString("ChannelInviteViaLinkRestricted3", R.string.ChannelInviteViaLinkRestricted3)); + } } + } else if (type == TYPE_PUBLIC_LINKS) { + headerCell.setText(LocaleController.getString("YourPublicCommunities", R.string.YourPublicCommunities)); + } else { + headerCell.setText(LocaleController.getString("LastActiveCommunities", R.string.LastActiveCommunities)); } - } else if (type == TYPE_PUBLIC_LINKS) { - headerCell.setText(LocaleController.getString("YourPublicCommunities", R.string.YourPublicCommunities)); - } else { - headerCell.setText(LocaleController.getString("LastActiveCommunities", R.string.LastActiveCommunities)); - } - } else if (holder.getItemViewType() == 8) { - + break; + case VIEW_TYPE_BOOST_FEATURE: + final int index = position - boostFeaturesStartRow; + if (boostFeatures != null && index >= 0 && index < boostFeatures.size()) { + ((BoostFeatureCell) holder.itemView).set(boostFeatures.get(index)); + } + break; } } @Override public int getItemViewType(int position) { if (headerRow == position) { - return 0; + return VIEW_TYPE_HEADER; } else if (dividerRow == position) { - return 2; + return VIEW_TYPE_SHADOW; } else if (chatsTitleRow == position) { - return 3; + return VIEW_TYPE_HEADER_CELL; } else if (loadingRow == position) { - return 5; + return VIEW_TYPE_PROGRESS; } else if (emptyViewDividerRow == position) { - return 6; + return VIEW_TYPE_SPACE16DP; } else if (linkRow == position) { - return 7; + return VIEW_TYPE_BOOST_LINK; } else if (bottomRow == position) { - return 8; + return VIEW_TYPE_BOOST_FOOTER; + } + if (boostFeatures != null && position >= boostFeaturesStartRow && position <= boostFeaturesStartRow + boostFeatures.size()) { + return VIEW_TYPE_BOOST_FEATURE; } if (type == TYPE_TO0_MANY_COMMUNITIES || type == TYPE_ADD_MEMBERS_RESTRICTED) { - return 4; + return VIEW_TYPE_USER; } else { - return 1; + return VIEW_TYPE_CHANNEL; } } @@ -1052,6 +1229,7 @@ public void setDialogId(long dialogId) { public void setBoostsStats(TL_stories.TL_premium_boostsStatus boostsStatus, boolean isCurrentChat) { this.boostsStatus = boostsStatus; this.isCurrentChat = isCurrentChat; + updateRows(); } public void setCanApplyBoost(ChannelBoostsController.CanApplyBoost canApplyBoost) { @@ -1075,12 +1253,12 @@ private class HeaderView extends LinearLayout { public HeaderView(Context context) { super(context); setOrientation(LinearLayout.VERTICAL); - setPadding(AndroidUtilities.dp(6), 0, AndroidUtilities.dp(6), 0); + setPadding(backgroundPaddingLeft + dp(6), 0, backgroundPaddingLeft + dp(6), 0); limitParams = getLimitParams(type, currentAccount); int icon = limitParams.icon; String descriptionStr; - boolean premiumLocked = MessagesController.getInstance(currentAccount).premiumLocked; + boolean premiumLocked = MessagesController.getInstance(currentAccount).premiumFeaturesBlocked(); if (type == TYPE_BOOSTS_FOR_USERS) { descriptionStr = getBoostsDescriptionString(); } else if (type == TYPE_BOOSTS_FOR_POSTING) { @@ -1098,8 +1276,38 @@ public HeaderView(Context context) { } } else if (type == TYPE_BOOSTS_FOR_COLOR) { descriptionStr = LocaleController.formatString( - "ChannelNeedBoostsForColorDescription", R.string.ChannelNeedBoostsForColorDescription, - MessagesController.getInstance(currentAccount).channelColorLevelMin + R.string.ChannelNeedBoostsForColorDescription, + channelColorLevelMin() + ); + } else if (type == TYPE_BOOSTS_FOR_PROFILE_COLOR) { + descriptionStr = LocaleController.formatString( + R.string.ChannelNeedBoostsForProfileColorDescription, + channelColorLevelMin() + ); + } else if (type == TYPE_BOOSTS_FOR_EMOJI_STATUS) { + descriptionStr = LocaleController.formatString( + R.string.ChannelNeedBoostsForEmojiStatusDescription, + MessagesController.getInstance(currentAccount).channelEmojiStatusLevelMin + ); + } else if (type == TYPE_BOOSTS_FOR_REPLY_ICON) { + descriptionStr = LocaleController.formatString( + R.string.ChannelNeedBoostsForReplyIconDescription, + MessagesController.getInstance(currentAccount).channelBgIconLevelMin + ); + } else if (type == TYPE_BOOSTS_FOR_PROFILE_ICON) { + descriptionStr = LocaleController.formatString( + R.string.ChannelNeedBoostsForProfileIconDescription, + MessagesController.getInstance(currentAccount).channelProfileIconLevelMin + ); + } else if (type == TYPE_BOOSTS_FOR_WALLPAPER) { + descriptionStr = LocaleController.formatString( + R.string.ChannelNeedBoostsForWallpaperDescription, + MessagesController.getInstance(currentAccount).channelWallpaperLevelMin + ); + } else if (type == TYPE_BOOSTS_FOR_CUSTOM_WALLPAPER) { + descriptionStr = LocaleController.formatString( + R.string.ChannelNeedBoostsForCustomWallpaperDescription, + MessagesController.getInstance(currentAccount).channelCustomWallpaperLevelMin ); } else if (type == TYPE_BOOSTS_FOR_REACTIONS) { descriptionStr = LocaleController.formatPluralString("ReactionReachLvlForReaction", requiredLvl, requiredLvl); @@ -1184,7 +1392,19 @@ public HeaderView(Context context) { percent = defaultLimit / (float) premiumLimit; - if (type == TYPE_BOOSTS_FOR_POSTING || type == TYPE_BOOSTS_FOR_COLOR || type == TYPE_BOOSTS_FOR_USERS || type == TYPE_BOOSTS_FOR_REACTIONS) { + final boolean boostsType = ( + type == TYPE_BOOSTS_FOR_POSTING || + type == TYPE_BOOSTS_FOR_COLOR || + type == TYPE_BOOSTS_FOR_PROFILE_COLOR || + type == TYPE_BOOSTS_FOR_EMOJI_STATUS || + type == TYPE_BOOSTS_FOR_WALLPAPER || + type == TYPE_BOOSTS_FOR_CUSTOM_WALLPAPER || + type == TYPE_BOOSTS_FOR_USERS || + type == TYPE_BOOSTS_FOR_REACTIONS || + type == TYPE_BOOSTS_FOR_REPLY_ICON || + type == TYPE_BOOSTS_FOR_PROFILE_ICON + ); + if (boostsType) { currentValue = 0; } @@ -1197,7 +1417,7 @@ public void invalidate() { super.invalidate(); } }; - if (type == TYPE_BOOSTS_FOR_POSTING || type == TYPE_BOOSTS_FOR_COLOR || type == TYPE_BOOSTS_FOR_USERS || type == TYPE_BOOSTS_FOR_REACTIONS) { + if (boostsType) { if (boostsStatus != null) { limitPreviewView.setBoosts(boostsStatus, canApplyBoost != null && canApplyBoost.boostedNow); } @@ -1240,6 +1460,16 @@ public void invalidate() { title.setText(LocaleController.getString(R.string.ReactionCustomReactions)); } else if (type == TYPE_BOOSTS_FOR_COLOR) { title.setText(LocaleController.getString(R.string.BoostingEnableColor)); + } else if (type == TYPE_BOOSTS_FOR_PROFILE_COLOR) { + title.setText(LocaleController.getString(R.string.BoostingEnableProfileColor)); + } else if (type == TYPE_BOOSTS_FOR_REPLY_ICON) { + title.setText(LocaleController.getString(R.string.BoostingEnableLinkIcon)); + } else if (type == TYPE_BOOSTS_FOR_PROFILE_ICON) { + title.setText(LocaleController.getString(R.string.BoostingEnableProfileIcon)); + } else if (type == TYPE_BOOSTS_FOR_EMOJI_STATUS) { + title.setText(LocaleController.getString(R.string.BoostingEnableEmojiStatus)); + } else if (type == TYPE_BOOSTS_FOR_WALLPAPER || type == TYPE_BOOSTS_FOR_CUSTOM_WALLPAPER) { + title.setText(LocaleController.getString(R.string.BoostingEnableWallpaper)); } else if (type == TYPE_ADD_MEMBERS_RESTRICTED) { if (canSendLink) { title.setText(LocaleController.getString("ChannelInviteViaLink", R.string.ChannelInviteViaLink)); @@ -1264,7 +1494,7 @@ public void invalidate() { titleLinearLayout.setWeightSum(1f); titleLinearLayout.addView(title, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, 1, 0)); titleLinearLayout.addView(boostCounterView, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.TOP, 0, 2, 0, 0)); - addView(titleLinearLayout, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_HORIZONTAL, 12 + 13, premiumLocked ? 8 : 22, 12, 9)); + addView(titleLinearLayout, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_HORIZONTAL, 12, premiumLocked ? 8 : 22, 12, 9)); } else { addView(title, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_HORIZONTAL, 0, premiumLocked ? 8 : 22, 0, 0)); @@ -1364,17 +1594,21 @@ public void recreateTitleAndDescription() { } } + protected int channelColorLevelMin() { + return 0; + } + private String getBoostsTitleString() { if (boostsStatus.next_level_boosts == 0) { return LocaleController.formatString("BoostsMaxLevelReached", R.string.BoostsMaxLevelReached); } else if (boostsStatus.level > 0 && !canApplyBoost.alreadyActive) { - return LocaleController.getString("HelpUpgradeChannel", R.string.HelpUpgradeChannel); + return LocaleController.getString(R.string.BoostChannel); } else if (isCurrentChat) { TLRPC.Chat chat = MessagesController.getInstance(currentAccount).getChat(-dialogId); if (canApplyBoost.alreadyActive) { return LocaleController.formatString("YouBoostedChannel2", R.string.YouBoostedChannel2, chat.title); } else { - return LocaleController.formatString("BoostingEnableStoriesForChannel2", R.string.BoostingEnableStoriesForChannel2, chat.title); + return LocaleController.getString(R.string.BoostChannel); } } else { if (canApplyBoost.alreadyActive) { @@ -1386,6 +1620,8 @@ private String getBoostsTitleString() { } private String getBoostsDescriptionString() { + TLRPC.Chat channel = MessagesController.getInstance(currentAccount).getChat(-dialogId); + String channelTitle = channel == null ? LocaleController.getString(R.string.AccDescrChannel) : channel.title; boolean isZeroBoostsForNextLevel = boostsStatus.boosts == boostsStatus.current_level_boosts; if (isZeroBoostsForNextLevel && canApplyBoost.alreadyActive) { if (boostsStatus.level == 1) { @@ -1399,7 +1635,8 @@ private String getBoostsDescriptionString() { if (canApplyBoost.alreadyActive) { if (boostsStatus.level == 0) { return LocaleController.formatString( - "ChannelNeedBoostsAlreadyBoostedDescriptionLevel1", R.string.ChannelNeedBoostsAlreadyBoostedDescriptionLevel1, + R.string.ChannelNeedBoostsDescriptionForNewFeatures, + channelTitle, LocaleController.formatPluralString("MoreBoosts", boostsStatus.next_level_boosts - boostsStatus.boosts, boostsStatus.next_level_boosts - boostsStatus.boosts) ); } else { @@ -1408,17 +1645,19 @@ private String getBoostsDescriptionString() { boostsStatus.level, LocaleController.formatPluralString("BoostStories", boostsStatus.level + 1)); } else { - return LocaleController.formatString("ChannelNeedBoostsDescriptionLevelNext", R.string.ChannelNeedBoostsDescriptionLevelNext, - LocaleController.formatPluralString("MoreBoosts", boostsStatus.next_level_boosts - boostsStatus.boosts, boostsStatus.next_level_boosts - boostsStatus.boosts), - LocaleController.formatPluralString("BoostStories", boostsStatus.level + 1) + return LocaleController.formatString( + R.string.ChannelNeedBoostsDescriptionForNewFeatures, + channelTitle, + LocaleController.formatPluralString("MoreBoosts", boostsStatus.next_level_boosts - boostsStatus.boosts, boostsStatus.next_level_boosts - boostsStatus.boosts) ); } } } else { if (boostsStatus.level == 0) { return LocaleController.formatString( - "ChannelNeedBoostsDescriptionLevel1", R.string.ChannelNeedBoostsDescriptionLevel1, - LocaleController.formatPluralString("MoreBoosts", boostsStatus.next_level_boosts - boostsStatus.boosts, boostsStatus.next_level_boosts - boostsStatus.boosts) + R.string.ChannelNeedBoostsDescriptionForNewFeatures, + channelTitle, + LocaleController.formatPluralString("MoreBoosts", boostsStatus.next_level_boosts - boostsStatus.boosts, boostsStatus.next_level_boosts - boostsStatus.boosts) ); } else { if (boostsStatus.next_level_boosts == 0) { @@ -1426,9 +1665,10 @@ private String getBoostsDescriptionString() { boostsStatus.level, LocaleController.formatPluralString("BoostStories", boostsStatus.level + 1)); } else { - return LocaleController.formatString("ChannelNeedBoostsDescriptionLevelNext", R.string.ChannelNeedBoostsDescriptionLevelNext, - LocaleController.formatPluralString("MoreBoosts", boostsStatus.next_level_boosts - boostsStatus.boosts, boostsStatus.next_level_boosts - boostsStatus.boosts), - LocaleController.formatPluralString("BoostStories", boostsStatus.level + 1) + return LocaleController.formatString( + R.string.ChannelNeedBoostsDescriptionForNewFeatures, + channelTitle, + LocaleController.formatPluralString("MoreBoosts", boostsStatus.next_level_boosts - boostsStatus.boosts, boostsStatus.next_level_boosts - boostsStatus.boosts) ); } } @@ -1529,7 +1769,7 @@ private static LimitParams getLimitParams(int type, int currentAccount) { limitParams.descriptionStr = LocaleController.formatString("LimitReachedStoriesMonthly", R.string.LimitReachedStoriesMonthly, limitParams.defaultLimit, limitParams.premiumLimit); limitParams.descriptionStrPremium = LocaleController.formatString("LimitReachedStoriesMonthlyPremium", R.string.LimitReachedStoriesMonthlyPremium, limitParams.premiumLimit); limitParams.descriptionStrLocked = LocaleController.formatString("LimitReachedStoriesMonthlyPremium", R.string.LimitReachedStoriesMonthlyPremium, limitParams.defaultLimit); - } else if (type == TYPE_BOOSTS_FOR_POSTING || type == TYPE_BOOSTS_FOR_COLOR || type == TYPE_BOOSTS_FOR_USERS || type == TYPE_BOOSTS_FOR_REACTIONS) { + } else if (type == TYPE_BOOSTS_FOR_POSTING || type == TYPE_BOOSTS_FOR_COLOR || type == TYPE_BOOSTS_FOR_PROFILE_COLOR || type == TYPE_BOOSTS_FOR_REPLY_ICON || type == TYPE_BOOSTS_FOR_PROFILE_ICON || type == TYPE_BOOSTS_FOR_EMOJI_STATUS || type == TYPE_BOOSTS_FOR_WALLPAPER || type == TYPE_BOOSTS_FOR_CUSTOM_WALLPAPER || type == TYPE_BOOSTS_FOR_USERS || type == TYPE_BOOSTS_FOR_REACTIONS) { limitParams.defaultLimit = MessagesController.getInstance(currentAccount).storiesSentMonthlyLimitDefault; limitParams.premiumLimit = MessagesController.getInstance(currentAccount).storiesSentMonthlyLimitPremium; limitParams.icon = R.drawable.filled_limit_boost; @@ -1583,8 +1823,32 @@ private void updateRows() { loadingRow = -1; linkRow = -1; emptyViewDividerRow = -1; + boostFeaturesStartRow = -1; + headerRow = rowCount++; - if (!hasFixedSize(type)) { + if ( + type == TYPE_BOOSTS_FOR_USERS || + type == TYPE_BOOSTS_FOR_POSTING || + type == TYPE_BOOSTS_FOR_COLOR || + type == TYPE_BOOSTS_FOR_PROFILE_COLOR || + type == TYPE_BOOSTS_FOR_REPLY_ICON || + type == TYPE_BOOSTS_FOR_PROFILE_ICON || + type == TYPE_BOOSTS_FOR_WALLPAPER || + type == TYPE_BOOSTS_FOR_CUSTOM_WALLPAPER || + type == TYPE_BOOSTS_FOR_EMOJI_STATUS || + type == TYPE_BOOSTS_FOR_REACTIONS + ) { + if (type != TYPE_BOOSTS_FOR_USERS) { + topPadding = .24f; + linkRow = rowCount++; + if (MessagesController.getInstance(currentAccount).giveawayGiftsPurchaseAvailable) { + bottomRow = rowCount++; + } + } + setupBoostFeatures(); + boostFeaturesStartRow = rowCount++; + rowCount += boostFeatures.size() - 1; + } else if (!hasFixedSize(type)) { dividerRow = rowCount++; chatsTitleRow = rowCount++; if (loading) { @@ -1604,12 +1868,6 @@ private void updateRows() { } } } - if (type == TYPE_BOOSTS_FOR_POSTING || type == TYPE_BOOSTS_FOR_COLOR || type == TYPE_BOOSTS_FOR_REACTIONS) { - linkRow = rowCount++; - if (MessagesController.getInstance(currentAccount).giveawayGiftsPurchaseAvailable) { - bottomRow = rowCount++; - } - } notifyDataSetChanged(); } @@ -1739,4 +1997,201 @@ public static class LimitParams { int premiumLimit = 0; } + private void setupBoostFeatures() { + boostFeatures = new ArrayList<>(); + + ArrayList lastFeatureList = null; + int startLevel = 1; + if (boostsStatus != null) { + startLevel = boostsStatus.level + 1; + } + + int maxlvl = 10; + final MessagesController m = MessagesController.getInstance(currentAccount); + if (m != null) { + maxlvl = Math.max(maxlvl, m.peerColors != null ? m.peerColors.maxLevel() : 0); + maxlvl = Math.max(maxlvl, m.profilePeerColors != null ? m.profilePeerColors.maxLevel() : 0); + maxlvl = Math.max(maxlvl, m.channelBgIconLevelMin); + maxlvl = Math.max(maxlvl, m.channelProfileIconLevelMin); + maxlvl = Math.max(maxlvl, m.channelEmojiStatusLevelMin); + maxlvl = Math.max(maxlvl, m.channelWallpaperLevelMin); + maxlvl = Math.max(maxlvl, m.channelCustomWallpaperLevelMin); + } + + for (int lvl = startLevel; lvl <= maxlvl; ++lvl) { + ArrayList featureList = boostFeaturesForLevel(lvl); + if (lastFeatureList == null || !BoostFeature.arraysEqual(lastFeatureList, featureList)) { + boostFeatures.add(new BoostFeature.BoostFeatureLevel(lvl, boostFeatures.isEmpty())); + boostFeatures.addAll(featureList); + lastFeatureList = featureList; + } + } + } + + private ArrayList boostFeaturesForLevel(int level) { + ArrayList list = new ArrayList<>(); + final MessagesController m = MessagesController.getInstance(currentAccount); + if (m == null) return list; + list.add(BoostFeature.of(R.drawable.menu_feature_stories, "BoostFeatureStoriesPerDay", level)); + list.add(BoostFeature.of(R.drawable.menu_feature_reactions, "BoostFeatureCustomReaction", level)); + final int nameColorsAvailable = m.peerColors != null ? m.peerColors.colorsAvailable(level) : 0; + final int profileColorsAvailable = m.profilePeerColors != null ? m.profilePeerColors.colorsAvailable(level) : 0; + if (nameColorsAvailable > 0) { + list.add(BoostFeature.of(R.drawable.menu_feature_color_name, "BoostFeatureNameColor", 7)); + } + if (nameColorsAvailable > 0) { + list.add(BoostFeature.of(R.drawable.menu_feature_links, "BoostFeatureReplyColor", nameColorsAvailable)); + } + if (level >= m.channelBgIconLevelMin) { + list.add(BoostFeature.of(R.drawable.menu_feature_links2, R.string.BoostFeatureReplyIcon)); + } + if (level >= m.channelEmojiStatusLevelMin) { + list.add(BoostFeature.of(R.drawable.menu_feature_status, R.string.BoostFeatureEmojiStatuses, "1000+")); + } + if (profileColorsAvailable > 0) { + list.add(BoostFeature.of(R.drawable.menu_feature_color_profile, "BoostFeatureProfileColor", profileColorsAvailable)); + } + if (level >= m.channelProfileIconLevelMin) { + list.add(BoostFeature.of(R.drawable.menu_feature_cover, R.string.BoostFeatureProfileIcon)); + } + if (level >= m.channelWallpaperLevelMin) { + list.add(BoostFeature.of(R.drawable.menu_feature_wallpaper, "BoostFeatureBackground", 8)); + } + if (level >= m.channelCustomWallpaperLevelMin) { + list.add(BoostFeature.of(R.drawable.menu_feature_custombg, R.string.BoostFeatureCustomBackground)); + } + return list; + } + + private class BoostFeatureCell extends FrameLayout { + private final Theme.ResourcesProvider resourcesProvider; + + private final ImageView imageView; + private final SimpleTextView textView; + + private final FrameLayout levelLayout; + private final SimpleTextView levelTextView; + + public BoostFeature feature; + public BoostFeature.BoostFeatureLevel level; + + public BoostFeatureCell(Context context, Theme.ResourcesProvider resourcesProvider) { + super(context); + this.resourcesProvider = resourcesProvider; + + setPadding(backgroundPaddingLeft, 0, backgroundPaddingLeft, 0); + + imageView = new ImageView(context); + imageView.setScaleType(ImageView.ScaleType.FIT_CENTER); + imageView.setColorFilter(new PorterDuffColorFilter(Theme.getColor(Theme.key_premiumGradient1, resourcesProvider), PorterDuff.Mode.SRC_IN)); + addView(imageView, LayoutHelper.createFrame(24, 24, Gravity.CENTER_VERTICAL | (LocaleController.isRTL ? Gravity.RIGHT : Gravity.LEFT), 24, 0, 24, 0)); + + textView = new SimpleTextView(context); + textView.setWidthWrapContent(true); + textView.setTextColor(Theme.getColor(Theme.key_dialogTextBlack, resourcesProvider)); + textView.setTextSize(14); + addView(textView, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_VERTICAL | (LocaleController.isRTL ? Gravity.RIGHT : Gravity.LEFT), LocaleController.isRTL ? 30 : 60, 0, LocaleController.isRTL ? 60 : 30, 0)); + + levelTextView = new SimpleTextView(context); + levelTextView.setTextColor(Color.WHITE); + levelTextView.setWidthWrapContent(true); + levelTextView.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); + levelTextView.setTextSize(14); + levelLayout = new FrameLayout(context) { + private final PremiumGradient.PremiumGradientTools gradientTools = new PremiumGradient.PremiumGradientTools(Theme.key_premiumGradient1, Theme.key_premiumGradient2, -1, -1, -1, resourcesProvider); + private final Paint dividerPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + { + dividerPaint.setStyle(Paint.Style.STROKE); + dividerPaint.setStrokeWidth(1); + } + + @Override + protected void dispatchDraw(Canvas canvas) { + dividerPaint.setColor(Theme.getColor(Theme.key_divider, resourcesProvider)); + canvas.drawLine(dp(18), getHeight() / 2f, levelTextView.getLeft() - dp(20), getHeight() / 2f, dividerPaint); + canvas.drawLine(levelTextView.getRight() + dp(20), getHeight() / 2f, getWidth() - dp(18), getHeight() / 2f, dividerPaint); + + AndroidUtilities.rectTmp.set( + levelTextView.getLeft() - dp(15), + (levelTextView.getTop() + levelTextView.getBottom() - dp(30)) / 2f, + levelTextView.getRight() + dp(15), + (levelTextView.getTop() + levelTextView.getBottom() + dp(30)) / 2f + ); + canvas.save(); + canvas.translate(AndroidUtilities.rectTmp.left, AndroidUtilities.rectTmp.top); + AndroidUtilities.rectTmp.set(0, 0, AndroidUtilities.rectTmp.width(), AndroidUtilities.rectTmp.height()); + gradientTools.gradientMatrix(AndroidUtilities.rectTmp); + canvas.drawRoundRect(AndroidUtilities.rectTmp, dp(15), dp(15), gradientTools.paint); + canvas.restore(); + + super.dispatchDraw(canvas); + } + }; + levelLayout.setWillNotDraw(false); + levelLayout.addView(levelTextView, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER)); + addView(levelLayout, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT)); + } + + public void set(BoostFeature boostFeature) { + if (boostFeature instanceof BoostFeature.BoostFeatureLevel) { + level = (BoostFeature.BoostFeatureLevel) boostFeature; + feature = null; + + imageView.setVisibility(View.GONE); + textView.setVisibility(View.GONE); + levelLayout.setVisibility(View.VISIBLE); + levelTextView.setText(LocaleController.formatPluralString(level.isFirst ? "BoostLevelUnlocks" : "BoostLevel", level.lvl)); + } else if (boostFeature != null) { + level = null; + feature = boostFeature; + imageView.setVisibility(View.VISIBLE); + imageView.setImageResource(feature.iconResId); + textView.setVisibility(View.VISIBLE); + if (feature.textKeyPlural != null) { + String key1 = feature.textKeyPlural + "_" + LocaleController.getStringParamForNumber(feature.countPlural); + String text = LocaleController.getString(key1); + if (text == null || text.startsWith("LOC_ERR")) { + text = LocaleController.getString(feature.textKeyPlural + "_other"); + } + if (text == null) { + text = ""; + } + int index; + SpannableStringBuilder ssb = new SpannableStringBuilder(text); + if ((index = text.indexOf("%d")) >= 0) { + ssb = new SpannableStringBuilder(text); + SpannableString boldNumber = new SpannableString(feature.countPlural + ""); + boldNumber.setSpan(new TypefaceSpan(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)), 0, boldNumber.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + ssb.replace(index, index + 2, boldNumber); + } + textView.setText(ssb); + } else { + String text = LocaleController.getString(feature.textKey); + if (text == null) { + text = ""; + } + if (feature.countValue != null) { + int index; + SpannableStringBuilder ssb = new SpannableStringBuilder(text); + if ((index = text.indexOf("%s")) >= 0) { + ssb = new SpannableStringBuilder(text); + SpannableString boldNumber = new SpannableString(feature.countValue); + boldNumber.setSpan(new TypefaceSpan(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)), 0, boldNumber.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + ssb.replace(index, index + 2, boldNumber); + } + textView.setText(ssb); + } else { + textView.setText(text); + } + } + levelLayout.setVisibility(View.GONE); + } + } + + @Override + protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { + super.onMeasure(widthMeasureSpec, MeasureSpec.makeMeasureSpec(dp(level != null ? 49 : 36), MeasureSpec.EXACTLY)); + } + } + } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/PremiumGradient.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/PremiumGradient.java index 280572241dd..be24d248109 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/PremiumGradient.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/PremiumGradient.java @@ -9,6 +9,8 @@ import android.graphics.PorterDuff; import android.graphics.PorterDuffColorFilter; import android.graphics.PorterDuffXfermode; +import android.graphics.Rect; +import android.graphics.RectF; import android.graphics.Shader; import android.graphics.drawable.BitmapDrawable; import android.graphics.drawable.Drawable; @@ -133,7 +135,7 @@ public void setColorFilter(int color, PorterDuff.Mode mode) { } public Paint getMainGradientPaint() { - if (MessagesController.getInstance(UserConfig.selectedAccount).premiumLocked) { + if (MessagesController.getInstance(UserConfig.selectedAccount).premiumFeaturesBlocked()) { if (lockedPremiumPaint == null) { lockedPremiumPaint = new Paint(Paint.ANTI_ALIAS_FLAG); } @@ -182,6 +184,14 @@ public PremiumGradientTools(int colorKey1, int colorKey2, int colorKey3, int col this.colorKey5 = colorKey5; } + public void gradientMatrix(Rect rect) { + gradientMatrix(rect.left, rect.top, rect.right, rect.bottom, 0, 0); + } + + public void gradientMatrix(RectF rect) { + gradientMatrix((int) rect.left, (int) rect.top, (int) rect.right, (int) rect.bottom, 0, 0); + } + public void gradientMatrix(int x, int y, int x1, int y1, float xOffset, float yOffset) { chekColors(); if (exactly) { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/PremiumPreviewBottomSheet.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/PremiumPreviewBottomSheet.java index f1150a566cd..a876bbe5ec5 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/PremiumPreviewBottomSheet.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/PremiumPreviewBottomSheet.java @@ -1,5 +1,7 @@ package org.telegram.ui.Components.Premium; +import static org.telegram.messenger.LocaleController.getString; + import android.animation.Animator; import android.animation.AnimatorListenerAdapter; import android.animation.ValueAnimator; @@ -34,12 +36,14 @@ import org.telegram.messenger.AndroidUtilities; import org.telegram.messenger.ContactsController; import org.telegram.messenger.Emoji; +import org.telegram.messenger.FileLog; import org.telegram.messenger.LocaleController; import org.telegram.messenger.MediaDataController; import org.telegram.messenger.NotificationCenter; import org.telegram.messenger.R; import org.telegram.messenger.UserConfig; import org.telegram.messenger.Utilities; +import org.telegram.messenger.browser.Browser; import org.telegram.tgnet.TLRPC; import org.telegram.ui.ActionBar.BaseFragment; import org.telegram.ui.ActionBar.SimpleTextView; @@ -57,6 +61,7 @@ import org.telegram.ui.Components.LoadingSpan; import org.telegram.ui.Components.Premium.GLIcon.GLIconRenderer; import org.telegram.ui.Components.Premium.GLIcon.GLIconTextureView; +import org.telegram.ui.Components.Premium.boosts.cells.TextInfoCell; import org.telegram.ui.Components.RecyclerListView; import org.telegram.ui.PremiumFeatureCell; import org.telegram.ui.PremiumPreviewFragment; @@ -65,22 +70,25 @@ public class PremiumPreviewBottomSheet extends BottomSheetWithRecyclerListView implements NotificationCenter.NotificationCenterDelegate { - ArrayList premiumFeatures = new ArrayList<>(); + protected ArrayList premiumFeatures = new ArrayList<>(); int currentAccount; - TLRPC.User user; - GiftPremiumBottomSheet.GiftTier giftTier; + protected TLRPC.User user; + protected GiftPremiumBottomSheet.GiftTier giftTier; boolean isOutboundGift; PremiumFeatureCell dummyCell; int totalGradientHeight; - int rowCount; - int paddingRow; - int featuresStartRow; - int featuresEndRow; - int sectionRow; - int helpUsRow; - int buttonRow; + protected int rowCount; + protected int paddingRow; + protected int additionStartRow; + protected int additionEndRow; + protected int featuresStartRow; + protected int featuresEndRow; + protected int sectionRow; + protected int helpUsRow; + protected int buttonRow; + protected int termsRow; FireworksOverlay fireworksOverlay; PremiumGradient.PremiumGradientTools gradientTools; @@ -106,6 +114,7 @@ public class PremiumPreviewBottomSheet extends BottomSheetWithRecyclerListView i ValueAnimator enterAnimator; boolean animateConfetti; + boolean animateConfettiWithStars; FrameLayout buttonContainer; FrameLayout bulletinContainer; @@ -137,26 +146,16 @@ public PremiumPreviewBottomSheet(BaseFragment fragment, int currentAccount, TLRP gradientTools.cx = 0; gradientTools.cy = 0; - paddingRow = rowCount++; - featuresStartRow = rowCount; - rowCount += premiumFeatures.size(); - featuresEndRow = rowCount; - sectionRow = rowCount++; - if (!UserConfig.getInstance(currentAccount).isPremium() && gift == null) { - buttonRow = rowCount++; - } + updateRows(); + recyclerListView.setPadding(AndroidUtilities.dp(6), 0, AndroidUtilities.dp(6), 0); recyclerListView.setOnItemClickListener((view, position) -> { if (view instanceof PremiumFeatureCell) { PremiumFeatureCell cell = (PremiumFeatureCell) view; PremiumPreviewFragment.sentShowFeaturePreview(currentAccount, cell.data.type); -// if (cell.data.type == PremiumPreviewFragment.PREMIUM_FEATURE_LIMITS) { -// DoubledLimitsBottomSheet bottomSheet = new DoubledLimitsBottomSheet(fragment, currentAccount); -// showDialog(bottomSheet); -// } else { - showDialog(new PremiumFeatureBottomSheet(fragment, cell.data.type, false)); - // } + showDialog(new PremiumFeatureBottomSheet(fragment, cell.data.type, false)); } + onAdditionItemClicked(view); }); MediaDataController.getInstance(currentAccount).preloadPremiumPreviewStickers(); @@ -169,6 +168,21 @@ public PremiumPreviewBottomSheet(BaseFragment fragment, int currentAccount, TLRP containerView.addView(bulletinContainer, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 140, Gravity.BOTTOM | Gravity.FILL_HORIZONTAL)); } + protected void onAdditionItemClicked(View view) { + + } + + protected void updateRows() { + paddingRow = rowCount++; + featuresStartRow = rowCount; + rowCount += premiumFeatures.size(); + featuresEndRow = rowCount; + sectionRow = rowCount++; + if (!UserConfig.getInstance(currentAccount).isPremium() && giftTier == null) { + buttonRow = rowCount++; + } + } + public PremiumPreviewBottomSheet setOutboundGift(boolean outboundGift) { isOutboundGift = outboundGift; return this; @@ -179,6 +193,11 @@ public PremiumPreviewBottomSheet setAnimateConfetti(boolean animateConfetti) { return this; } + public PremiumPreviewBottomSheet setAnimateConfettiWithStars(boolean animateConfettiWithStars) { + this.animateConfettiWithStars = animateConfettiWithStars; + return this; + } + private void showDialog(Dialog dialog) { if (iconTextureView != null) { iconTextureView.setDialogVisible(true); @@ -212,13 +231,17 @@ public void onViewCreated(FrameLayout containerView) { buttonDivider.getLayoutParams().height = 1; AndroidUtilities.updateViewVisibilityAnimated(buttonDivider, true, 1f, false); - if (!UserConfig.getInstance(currentAccount).isPremium()) { + if (!UserConfig.getInstance(currentAccount).isPremium() && needDefaultPremiumBtn()) { buttonContainer.addView(premiumButtonView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 48, Gravity.CENTER_VERTICAL, 16, 0, 16, 0)); buttonContainer.setBackgroundColor(getThemedColor(Theme.key_dialogBackground)); containerView.addView(buttonContainer, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 68, Gravity.BOTTOM)); } } + protected boolean needDefaultPremiumBtn() { + return true; + } + @Override protected void onPreMeasure(int widthMeasureSpec, int heightMeasureSpec) { super.onPreMeasure(widthMeasureSpec, heightMeasureSpec); @@ -227,7 +250,8 @@ protected void onPreMeasure(int widthMeasureSpec, int heightMeasureSpec) { } private FrameLayout titleViewContainer; - private LinkSpanDrawable.LinksTextView titleView[]; + protected LinkSpanDrawable.LinksTextView titleView[]; + private void titleLoaded(CharSequence newText, boolean animated) { if (titleView == null) { return; @@ -260,7 +284,8 @@ private void titleLoaded(CharSequence newText, boolean animated) { } } - private TextView subtitleView; + protected TextView subtitleView; + public void setTitle(boolean animated) { if (titleView == null || subtitleView == null) { return; @@ -398,6 +423,14 @@ protected RecyclerListView.SelectionAdapter createAdapter() { return new Adapter(); } + protected void attachIconContainer(LinearLayout container) { + container.addView(overrideTitleIcon, LayoutHelper.createLinear(140, 140, Gravity.CENTER_HORIZONTAL, Gravity.CENTER, 10, 10, 10, 10)); + } + + protected void afterCellCreated(int viewType, View view) { + + } + private class Adapter extends RecyclerListView.SelectionAdapter { @NonNull @@ -405,6 +438,11 @@ private class Adapter extends RecyclerListView.SelectionAdapter { public RecyclerView.ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) { View view; Context context = parent.getContext(); + view = onCreateAdditionCell(viewType, context); + if (view != null) { + view.setLayoutParams(new RecyclerView.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT)); + return new RecyclerListView.Holder(view); + } switch (viewType) { case 0: LinearLayout linearLayout = new LinearLayout(context) { @@ -444,7 +482,7 @@ protected void onDetachedFromWindow() { if (overrideTitleIcon.getParent() != null) { ((ViewGroup) overrideTitleIcon.getParent()).removeView(overrideTitleIcon); } - linearLayout.addView(overrideTitleIcon, LayoutHelper.createLinear(140, 140, Gravity.CENTER_HORIZONTAL, Gravity.CENTER, 10, 10, 10, 10)); + attachIconContainer(linearLayout); } if (titleViewContainer == null) { @@ -499,7 +537,7 @@ public int overrideColor() { linearLayout.addView(titleViewContainer, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, 0, Gravity.CENTER_HORIZONTAL, 40, 0, 40, 0)); if (subtitleView == null) { - subtitleView = new TextView(context); + subtitleView = new LinkSpanDrawable.LinksTextView(getContext(), resourcesProvider); subtitleView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 14); subtitleView.setGravity(Gravity.CENTER_HORIZONTAL); subtitleView.setTextColor(getThemedColor(Theme.key_windowBackgroundWhiteBlackText)); @@ -516,9 +554,18 @@ public int overrideColor() { @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { super.onMeasure(widthMeasureSpec, heightMeasureSpec); - drawable.rect2.set(0, 0, getMeasuredWidth(), getMeasuredHeight() - AndroidUtilities.dp(52)); } + + @Override + protected void configure() { + super.configure(); + drawable.useGradient = true; + drawable.useBlur = false; + drawable.forceMaxAlpha = true; + drawable.checkBounds = true; + drawable.init(); + } }; FrameLayout frameLayout = new FrameLayout(context) { @Override @@ -537,11 +584,6 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { frameLayout.addView(starParticlesView); frameLayout.addView(linearLayout); - starParticlesView.drawable.useGradient = true; - starParticlesView.drawable.useBlur = false; - starParticlesView.drawable.forceMaxAlpha = true; - starParticlesView.drawable.checkBounds = true; - starParticlesView.drawable.init(); if (iconTextureView != null) { iconTextureView.setStarParticlesView(starParticlesView); } @@ -574,8 +616,25 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { case 4: view = new AboutPremiumView(context); break; + case 5: + TextInfoCell cell = new TextInfoCell(context, resourcesProvider); + cell.setBackground(true); + String terms1 = getString("GiftPremiumPrivacyPolicyAndTerms", R.string.GiftPremiumPrivacyPolicyAndTerms); + SpannableStringBuilder stringBuilder1 = AndroidUtilities.replaceSingleTag( + terms1, + Theme.key_chat_messageLinkIn, 0, + () -> Browser.openUrl(fragment.getParentActivity(), LocaleController.getString("TermsOfServiceUrl", R.string.TermsOfServiceUrl))); + String terms2 = getString("GiftPremiumPrivacyPolicy", R.string.GiftPremiumPrivacyPolicy); + SpannableStringBuilder stringBuilder2 = AndroidUtilities.replaceSingleTag( + terms2, + Theme.key_chat_messageLinkIn, 0, + () -> Browser.openUrl(fragment.getParentActivity(), LocaleController.getString("PrivacyPolicyUrl", R.string.PrivacyPolicyUrl))); + cell.setText(AndroidUtilities.replaceCharSequence("%1$s", stringBuilder1, stringBuilder2)); + view = cell; + break; } view.setLayoutParams(new RecyclerView.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT)); + afterCellCreated(viewType, view); return new RecyclerListView.Holder(view); } @@ -583,6 +642,8 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { public void onBindViewHolder(@NonNull RecyclerView.ViewHolder holder, int position) { if (position >= featuresStartRow && position < featuresEndRow) { ((PremiumFeatureCell) holder.itemView).setData(premiumFeatures.get(position - featuresStartRow), position != featuresEndRow - 1); + } else if (position >= additionStartRow && position < additionEndRow) { + onBindAdditionCell(holder.itemView, position); } } @@ -595,6 +656,8 @@ public int getItemCount() { public int getItemViewType(int position) { if (position == paddingRow) { return 0; + } else if (position >= additionStartRow && position < additionEndRow) { + return getAdditionItemViewType(position); } else if (position >= featuresStartRow && position < featuresEndRow) { return 1; } else if (position == sectionRow) { @@ -603,16 +666,34 @@ public int getItemViewType(int position) { return 3; } else if (position == helpUsRow) { return 4; + } else if (position == termsRow) { + return 5; } return super.getItemViewType(position); } @Override public boolean isEnabled(RecyclerView.ViewHolder holder) { - return holder.getItemViewType() == 1; + return holder.getItemViewType() == 1 || isAdditionViewClickable(holder.getItemViewType()); } } + protected boolean isAdditionViewClickable(int viewType) { + return false; + } + + protected int getAdditionItemViewType(int position) { + return 0; + } + + protected View onCreateAdditionCell(int viewType, Context context) { + return null; + } + + protected void onBindAdditionCell(View view, int pos) { + + } + private void measureGradient(int w, int h) { int yOffset = 0; for (int i = 0; i < premiumFeatures.size(); i++) { @@ -634,7 +715,7 @@ public void show() { try { container.performHapticFeedback(HapticFeedbackConstants.KEYBOARD_TAP, HapticFeedbackConstants.FLAG_IGNORE_GLOBAL_SETTING); } catch (Exception ignored) {} - fireworksOverlay.start(); + fireworksOverlay.start(animateConfettiWithStars); }, 200); } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/StarParticlesView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/StarParticlesView.java index 95dfa49ff47..05c536b519f 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/StarParticlesView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/StarParticlesView.java @@ -49,6 +49,10 @@ public StarParticlesView(Context context) { } drawable = new Drawable(particlesCount); + configure(); + } + + protected void configure() { drawable.type = 100; drawable.roundEffect = true; drawable.useRotate = true; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/BoostCounterView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/BoostCounterView.java index 0e348a4ac74..fb9d287f9d1 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/BoostCounterView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/BoostCounterView.java @@ -83,14 +83,19 @@ public void setCount(int count, boolean animated) { animateCount(); } lastCount = count; + int oldLength = countText.getText().length(); countText.setText("x" + count, animated); + int newLength = countText.getText().length(); invalidate(); + if (oldLength != newLength) { + requestLayout(); + } } @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { super.onMeasure( - MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(26), MeasureSpec.EXACTLY), + MeasureSpec.makeMeasureSpec((int) (dp(8 + 3 + 4) + countText.getWidth()), MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(26), MeasureSpec.EXACTLY) ); } @@ -100,7 +105,7 @@ protected void onDraw(Canvas canvas) { super.onDraw(canvas); canvas.save(); canvas.translate(AndroidUtilities.dp(3), AndroidUtilities.dp(3)); - AndroidUtilities.rectTmp2.set(0, 0, AndroidUtilities.dp(20), AndroidUtilities.dp(20)); + AndroidUtilities.rectTmp2.set(0, 0, dp(8) + (int) countText.getCurrentWidth(), AndroidUtilities.dp(20)); AndroidUtilities.rectTmp.set(AndroidUtilities.rectTmp2); if (countScale != 1) { @@ -109,7 +114,7 @@ protected void onDraw(Canvas canvas) { } canvas.drawRoundRect(AndroidUtilities.rectTmp, dp(10), dp(10), bgPaint); - AndroidUtilities.rectTmp2.set(0, 0, AndroidUtilities.dp(20), AndroidUtilities.dp(19)); + AndroidUtilities.rectTmp2.set(0, 0, (int) AndroidUtilities.rectTmp.width(), AndroidUtilities.dp(19)); countText.setBounds(AndroidUtilities.rectTmp2); countText.draw(canvas); if (countScale != 1) { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/BoostDialogs.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/BoostDialogs.java index 8ba91f29eaf..176fbef6569 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/BoostDialogs.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/BoostDialogs.java @@ -19,6 +19,7 @@ import android.util.TypedValue; import android.view.Gravity; import android.view.HapticFeedbackConstants; +import android.view.ViewGroup; import android.widget.Button; import android.widget.FrameLayout; import android.widget.LinearLayout; @@ -29,9 +30,11 @@ import org.telegram.messenger.AndroidUtilities; import org.telegram.messenger.DialogObject; +import org.telegram.messenger.FileLog; import org.telegram.messenger.LocaleController; import org.telegram.messenger.MessageObject; import org.telegram.messenger.MessagesController; +import org.telegram.messenger.NotificationCenter; import org.telegram.messenger.R; import org.telegram.messenger.UserConfig; import org.telegram.messenger.UserObject; @@ -45,6 +48,7 @@ import org.telegram.ui.Components.AlertsCreator; import org.telegram.ui.Components.Bulletin; import org.telegram.ui.Components.BulletinFactory; +import org.telegram.ui.Components.EffectsTextView; import org.telegram.ui.Components.LayoutHelper; import org.telegram.ui.Components.NumberPicker; import org.telegram.ui.LaunchActivity; @@ -69,6 +73,32 @@ public static void showToastError(Context context, TLRPC.TL_error error) { } } + public static void processApplyGiftCodeError(TLRPC.TL_error error, FrameLayout containerLayout, Theme.ResourcesProvider resourcesProvider, Runnable share) { + if (error == null || error.text == null) { + return; + } + if (error.text.contains("PREMIUM_SUB_ACTIVE_UNTIL_")) { + String strDate = error.text.replace("PREMIUM_SUB_ACTIVE_UNTIL_", ""); + int date = Integer.parseInt(strDate); + String formattedDate = LocaleController.getInstance().formatterBoostExpired.format(new Date(date * 1000L)); + String subTitleText = getString("GiftPremiumActivateErrorText", R.string.GiftPremiumActivateErrorText); + SpannableStringBuilder subTitleWithLink = AndroidUtilities.replaceSingleTag( + subTitleText, + Theme.key_undo_cancelColor, 0, + share); + BulletinFactory.of(containerLayout, resourcesProvider).createSimpleBulletin(R.raw.chats_infotip, + LocaleController.getString("GiftPremiumActivateErrorTitle", R.string.GiftPremiumActivateErrorTitle), + AndroidUtilities.replaceCharSequence("%1$s", subTitleWithLink, replaceTags("**" + formattedDate + "**")) + ).show(); + try { + containerLayout.performHapticFeedback(HapticFeedbackConstants.KEYBOARD_TAP, HapticFeedbackConstants.FLAG_IGNORE_GLOBAL_SETTING); + } catch (Exception ignore) { + } + } else { + BoostDialogs.showToastError(containerLayout.getContext(), error); + } + } + private static void showBulletin(final BulletinFactory bulletinFactory, Theme.ResourcesProvider resourcesProvider, final TLRPC.Chat chat, final boolean isGiveaway) { AndroidUtilities.runOnUIThread(() -> bulletinFactory.createSimpleBulletin(R.raw.star_premium_2, isGiveaway ? getString("BoostingGiveawayCreated", R.string.BoostingGiveawayCreated) @@ -491,6 +521,11 @@ public static void showAbout(String from, long msgDate, TLRPC.TL_payments_giveaw stringBuilder.append(replaceTags(formatPluralString("BoostingGiveawayHowItWorksText", quantity, from, quantity, months))); stringBuilder.append("\n\n"); + if (giveaway.prize_description != null && !giveaway.prize_description.isEmpty()) { + stringBuilder.append(replaceTags(formatPluralString("BoostingGiveawayHowItWorksIncludeText", quantity, from, giveaway.prize_description))); + stringBuilder.append("\n\n"); + } + if (giveaway.only_new_subscribers) { if (isSeveralChats) { String andStr = formatPluralString("BoostingGiveawayHowItWorksSubTextDateSeveral2", giveaway.channels.size() - 1, fromTime, fromDate); @@ -536,10 +571,13 @@ public static void showAbout(String from, long msgDate, TLRPC.TL_payments_giveaw builder.setPositiveButton(getString("OK", R.string.OK), (dialogInterface, i) -> { }); - builder.show(); + applyDialogStyle(builder.show(), false); } public static void showAboutEnd(String from, long msgDate, TLRPC.TL_payments_giveawayInfoResults giveawayInfo, TLRPC.TL_messageMediaGiveaway giveaway, Context context, Theme.ResourcesProvider resourcesProvider) { + if (giveaway.until_date == 0) { + giveaway.until_date = giveawayInfo.finish_date; + } int quantity = giveaway.quantity; String months = formatPluralString("BoldMonths", giveaway.months); String endDate = LocaleController.getInstance().formatterGiveawayMonthDay.format(new Date(giveaway.until_date * 1000L)); @@ -554,6 +592,11 @@ public static void showAboutEnd(String from, long msgDate, TLRPC.TL_payments_giv stringBuilder.append(replaceTags(formatPluralString("BoostingGiveawayHowItWorksTextEnd", quantity, from, quantity, months))); stringBuilder.append("\n\n"); + if (giveaway.prize_description != null && !giveaway.prize_description.isEmpty()) { + stringBuilder.append(replaceTags(formatPluralString("BoostingGiveawayHowItWorksIncludeText", quantity, from, giveaway.prize_description))); + stringBuilder.append("\n\n"); + } + if (giveaway.only_new_subscribers) { if (isSeveralChats) { String andStr = formatPluralString("BoostingGiveawayHowItWorksSubTextDateSeveral2", giveaway.channels.size() - 1, fromTime, fromDate); @@ -574,28 +617,28 @@ public static void showAboutEnd(String from, long msgDate, TLRPC.TL_payments_giv if (giveawayInfo.activated_count > 0) { stringBuilder.append(replaceTags(formatPluralString("BoostingGiveawayUsedLinksPlural", giveawayInfo.activated_count))); } - stringBuilder.append("\n\n"); if (giveawayInfo.refunded) { String str = getString("BoostingGiveawayCanceledByPayment", R.string.BoostingGiveawayCanceledByPayment); TextView bottomTextView = new TextView(context); - bottomTextView.setTextColor(Theme.getColor(Theme.key_dialogTextBlack, resourcesProvider)); - bottomTextView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 16); + bottomTextView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 14); bottomTextView.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); bottomTextView.setGravity(Gravity.CENTER); bottomTextView.setText(str); bottomTextView.setTextColor(Theme.getColor(Theme.key_text_RedRegular, resourcesProvider)); bottomTextView.setBackground(Theme.createRoundRectDrawable(dp(10), dp(10), Theme.multAlpha(Theme.getColor(Theme.key_text_RedRegular, resourcesProvider), 0.1f))); - bottomTextView.setPadding(dp(8), dp(12), dp(8), dp(12)); + bottomTextView.setPadding(dp(12), dp(12), dp(12), dp(12)); builder.addBottomView(bottomTextView); builder.setMessage(stringBuilder); builder.setPositiveButton(getString("Close", R.string.Close), (dialogInterface, i) -> { }); + applyDialogStyle(builder.show(), true); } else { + builder.setMessage(stringBuilder); + String str; if (giveawayInfo.winner) { - stringBuilder.append(getString("BoostingGiveawayYouWon", R.string.BoostingGiveawayYouWon)); - builder.setMessage(stringBuilder); + str = getString("BoostingGiveawayYouWon", R.string.BoostingGiveawayYouWon); builder.setPositiveButton(getString("BoostingGiveawayViewPrize", R.string.BoostingGiveawayViewPrize), (dialogInterface, i) -> { BaseFragment fragment = LaunchActivity.getLastFragment(); if (fragment == null) { @@ -607,15 +650,30 @@ public static void showAboutEnd(String from, long msgDate, TLRPC.TL_payments_giv }); } else { - stringBuilder.append(getString("BoostingGiveawayYouNotWon", R.string.BoostingGiveawayYouNotWon)); - builder.setMessage(stringBuilder); + str = getString("BoostingGiveawayYouNotWon", R.string.BoostingGiveawayYouNotWon); builder.setPositiveButton(getString("Close", R.string.Close), (dialogInterface, i) -> { }); } + EffectsTextView topTextView = new EffectsTextView(context); + NotificationCenter.listenEmojiLoading(topTextView); + topTextView.setTextColor(Theme.getColor(Theme.key_dialogTextBlack, resourcesProvider)); + topTextView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 14); + topTextView.setGravity(Gravity.CENTER); + topTextView.setText(str); + topTextView.setBackground(Theme.createRoundRectDrawable(dp(8), dp(8), Theme.getColor(Theme.key_profile_actionPressedBackground, resourcesProvider))); + topTextView.setPadding(dp(8), dp(8), dp(8), dp(9)); + builder.aboveMessageView(topTextView); + applyDialogStyle(builder.show(), false); } + } - builder.show(); + public static void applyDialogStyle(AlertDialog dialog, boolean defaultMarginTop) { + dialog.setTextSize(20, 14); + dialog.setMessageLineSpacing(2.5f); + if (!defaultMarginTop) { + ((ViewGroup.MarginLayoutParams) dialog.getButtonsLayout().getLayoutParams()).topMargin = AndroidUtilities.dp(-14); + } } public static void showPrivateChannelAlert(Context context, Theme.ResourcesProvider resourcesProvider, Runnable onCanceled, Runnable onAccepted) { @@ -642,7 +700,20 @@ public static void openGiveAwayStatusDialog(MessageObject messageObject, Browser final AtomicBoolean isCanceled = new AtomicBoolean(false); progress.init(); progress.onCancel(() -> isCanceled.set(true)); - final TLRPC.TL_messageMediaGiveaway giveaway = (TLRPC.TL_messageMediaGiveaway) messageObject.messageOwner.media; + + final TLRPC.TL_messageMediaGiveaway giveaway; + if (messageObject.messageOwner.media instanceof TLRPC.TL_messageMediaGiveawayResults) { + TLRPC.TL_messageMediaGiveawayResults giveawayResults = (TLRPC.TL_messageMediaGiveawayResults) messageObject.messageOwner.media; + giveaway = new TLRPC.TL_messageMediaGiveaway(); + giveaway.prize_description = giveawayResults.prize_description; + giveaway.months = giveawayResults.months; + giveaway.quantity = giveawayResults.winners_count + giveawayResults.unclaimed_count; + giveaway.only_new_subscribers = giveawayResults.only_new_subscribers; + giveaway.until_date = giveawayResults.until_date; + } else { + giveaway = (TLRPC.TL_messageMediaGiveaway) messageObject.messageOwner.media; + } + final String fromName = getGiveawayCreatorName(messageObject); final long msgDate = messageObject.messageOwner.date * 1000L; BoostRepository.getGiveawayInfo(messageObject, result -> { @@ -682,11 +753,22 @@ private static String getGiveawayCreatorName(MessageObject messageObject) { } public static void showBulletinAbout(MessageObject messageObject) { - if (messageObject == null) { + if (messageObject == null || messageObject.messageOwner == null) { return; } BoostRepository.getGiveawayInfo(messageObject, result -> { - final TLRPC.TL_messageMediaGiveaway giveaway = (TLRPC.TL_messageMediaGiveaway) messageObject.messageOwner.media; + final TLRPC.TL_messageMediaGiveaway giveaway; + if (messageObject.messageOwner.media instanceof TLRPC.TL_messageMediaGiveawayResults) { + TLRPC.TL_messageMediaGiveawayResults giveawayResults = (TLRPC.TL_messageMediaGiveawayResults) messageObject.messageOwner.media; + giveaway = new TLRPC.TL_messageMediaGiveaway(); + giveaway.prize_description = giveawayResults.prize_description; + giveaway.months = giveawayResults.months; + giveaway.quantity = giveawayResults.winners_count + giveawayResults.unclaimed_count; + giveaway.only_new_subscribers = giveawayResults.only_new_subscribers; + giveaway.until_date = giveawayResults.until_date; + } else { + giveaway = (TLRPC.TL_messageMediaGiveaway) messageObject.messageOwner.media; + } final long msgDate = messageObject.messageOwner.date * 1000L; BaseFragment fragment = LaunchActivity.getLastFragment(); if (fragment == null) { @@ -729,7 +811,7 @@ public static void showBulletinAbout(MessageObject messageObject) { }); } - public static void showMoreBoostsNeeded(long dialogId) { + public static void showMoreBoostsNeeded(long dialogId, BottomSheet bottomSheet) { TLRPC.Chat chat = MessagesController.getInstance(UserConfig.selectedAccount).getChat(-dialogId); BaseFragment baseFragment = LaunchActivity.getLastFragment(); if (baseFragment == null) { @@ -737,8 +819,12 @@ public static void showMoreBoostsNeeded(long dialogId) { } AlertDialog.Builder builder = new AlertDialog.Builder(baseFragment.getContext(), baseFragment.getResourceProvider()); builder.setTitle(LocaleController.getString("BoostingMoreBoostsNeeded", R.string.BoostingMoreBoostsNeeded)); - builder.setMessage(AndroidUtilities.replaceTags(LocaleController.formatString("BoostingGetMoreBoostByGifting", R.string.BoostingGetMoreBoostByGifting, chat.title))); - builder.setPositiveButton(getString("OK", R.string.OK), (dialogInterface, i) -> { + builder.setMessage(AndroidUtilities.replaceTags(LocaleController.formatPluralString("BoostingGetMoreBoostByGiftingCount", BoostRepository.boostsPerSentGift(), chat.title))); + builder.setNegativeButton(getString("GiftPremium", R.string.GiftPremium), (dialogInterface, i) -> { + bottomSheet.dismiss(); + UserSelectorBottomSheet.open(); + }); + builder.setPositiveButton(getString("Close", R.string.Close), (dialogInterface, i) -> { }); builder.show(); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/BoostRepository.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/BoostRepository.java index 54e68c61bc0..59a64747f9b 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/BoostRepository.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/BoostRepository.java @@ -126,14 +126,18 @@ public static void payGiftCodeByInvoice(List users, TLRPC.TL_premiumGi TLRPC.TL_inputInvoicePremiumGiftCode invoice = new TLRPC.TL_inputInvoicePremiumGiftCode(); TLRPC.TL_inputStorePaymentPremiumGiftCode payload = new TLRPC.TL_inputStorePaymentPremiumGiftCode(); - payload.flags = 1; payload.users = new ArrayList<>(); for (TLObject user : users) { if (user instanceof TLRPC.User) { payload.users.add(controller.getInputUser((TLRPC.User) user)); } } - payload.boost_peer = controller.getInputPeer(-chat.id); + + if (chat != null) { + payload.flags = 1; + payload.boost_peer = controller.getInputPeer(-chat.id); + } + payload.currency = option.currency; payload.amount = option.amount; @@ -182,14 +186,16 @@ public static void payGiftCodeByGoogle(List users, TLRPC.TL_premiumGif ConnectionsManager connection = ConnectionsManager.getInstance(UserConfig.selectedAccount); TLRPC.TL_inputStorePaymentPremiumGiftCode payload = new TLRPC.TL_inputStorePaymentPremiumGiftCode(); - payload.flags = 1; payload.users = new ArrayList<>(); for (TLObject user : users) { if (user instanceof TLRPC.User) { payload.users.add(controller.getInputUser((TLRPC.User) user)); } } - payload.boost_peer = controller.getInputPeer(-chat.id); + if (chat != null) { + payload.flags = 1; + payload.boost_peer = controller.getInputPeer(-chat.id); + } QueryProductDetailsParams.Product product = QueryProductDetailsParams.Product.newBuilder() .setProductType(BillingClient.ProductType.INAPP) @@ -227,15 +233,22 @@ public static void payGiftCodeByGoogle(List users, TLRPC.TL_premiumGif }); } - public static void launchPreparedGiveaway(TL_stories.TL_prepaidGiveaway prepaidGiveaway, List chats, List selectedCountries, TLRPC.Chat chat, int date, boolean onlyNewSubscribers, Utilities.Callback onSuccess, Utilities.Callback onError) { + public static void launchPreparedGiveaway(TL_stories.TL_prepaidGiveaway prepaidGiveaway, List chats, List selectedCountries, + TLRPC.Chat chat, int date, boolean onlyNewSubscribers, boolean winnersVisible, boolean withAdditionPrize, String prizeDesc, + Utilities.Callback onSuccess, Utilities.Callback onError) { MessagesController controller = MessagesController.getInstance(UserConfig.selectedAccount); ConnectionsManager connection = ConnectionsManager.getInstance(UserConfig.selectedAccount); TLRPC.TL_inputStorePaymentPremiumGiveaway purpose = new TLRPC.TL_inputStorePaymentPremiumGiveaway(); purpose.only_new_subscribers = onlyNewSubscribers; + purpose.winners_are_visible = winnersVisible; + purpose.prize_description = prizeDesc; purpose.until_date = date; purpose.flags |= 2; purpose.flags |= 4; + if (withAdditionPrize) { + purpose.flags |= 16; + } purpose.random_id = System.currentTimeMillis(); purpose.additional_peers = new ArrayList<>(); purpose.boost_peer = controller.getInputPeer(-chat.id); @@ -267,15 +280,21 @@ public static void launchPreparedGiveaway(TL_stories.TL_prepaidGiveaway prepaidG }); } - public static void payGiveAway(List chats, List selectedCountries, TLRPC.TL_premiumGiftCodeOption option, TLRPC.Chat chat, int date, boolean onlyNewSubscribers, BaseFragment baseFragment, Utilities.Callback onSuccess, Utilities.Callback onError) { + public static void payGiveAway(List chats, List selectedCountries, TLRPC.TL_premiumGiftCodeOption option, + TLRPC.Chat chat, int date, boolean onlyNewSubscribers, BaseFragment baseFragment, + boolean winnersVisible, boolean withAdditionPrize, String prizeDesc, + Utilities.Callback onSuccess, Utilities.Callback onError) { if (!isGoogleBillingAvailable()) { - payGiveAwayByInvoice(chats, selectedCountries, option, chat, date, onlyNewSubscribers, baseFragment, onSuccess, onError); + payGiveAwayByInvoice(chats, selectedCountries, option, chat, date, onlyNewSubscribers, baseFragment, winnersVisible, withAdditionPrize, prizeDesc, onSuccess, onError); } else { - payGiveAwayByGoogle(chats, selectedCountries, option, chat, date, onlyNewSubscribers, baseFragment, onSuccess, onError); + payGiveAwayByGoogle(chats, selectedCountries, option, chat, date, onlyNewSubscribers, baseFragment, winnersVisible, withAdditionPrize, prizeDesc, onSuccess, onError); } } - public static void payGiveAwayByInvoice(List chats, List selectedCountries, TLRPC.TL_premiumGiftCodeOption option, TLRPC.Chat chat, int date, boolean onlyNewSubscribers, BaseFragment baseFragment, Utilities.Callback onSuccess, Utilities.Callback onError) { + public static void payGiveAwayByInvoice(List chats, List selectedCountries, TLRPC.TL_premiumGiftCodeOption option, + TLRPC.Chat chat, int date, boolean onlyNewSubscribers, BaseFragment baseFragment, + boolean winnersVisible, boolean withAdditionPrize, String prizeDesc, + Utilities.Callback onSuccess, Utilities.Callback onError) { MessagesController controller = MessagesController.getInstance(UserConfig.selectedAccount); ConnectionsManager connection = ConnectionsManager.getInstance(UserConfig.selectedAccount); @@ -284,9 +303,14 @@ public static void payGiveAwayByInvoice(List chats, List sel TLRPC.TL_inputStorePaymentPremiumGiveaway payload = new TLRPC.TL_inputStorePaymentPremiumGiveaway(); payload.only_new_subscribers = onlyNewSubscribers; + payload.winners_are_visible = winnersVisible; + payload.prize_description = prizeDesc; payload.until_date = date; payload.flags |= 2; payload.flags |= 4; + if (withAdditionPrize) { + payload.flags |= 16; + } payload.random_id = System.currentTimeMillis(); payload.additional_peers = new ArrayList<>(); for (TLObject o : chats) { @@ -344,15 +368,23 @@ public static void payGiveAwayByInvoice(List chats, List sel })); } - public static void payGiveAwayByGoogle(List chats, List selectedCountries, TLRPC.TL_premiumGiftCodeOption option, TLRPC.Chat chat, int date, boolean onlyNewSubscribers, BaseFragment baseFragment, Utilities.Callback onSuccess, Utilities.Callback onError) { + public static void payGiveAwayByGoogle(List chats, List selectedCountries, TLRPC.TL_premiumGiftCodeOption option, + TLRPC.Chat chat, int date, boolean onlyNewSubscribers, BaseFragment baseFragment, + boolean winnersVisible, boolean withAdditionPrize, String prizeDesc, + Utilities.Callback onSuccess, Utilities.Callback onError) { MessagesController controller = MessagesController.getInstance(UserConfig.selectedAccount); ConnectionsManager connection = ConnectionsManager.getInstance(UserConfig.selectedAccount); TLRPC.TL_inputStorePaymentPremiumGiveaway payload = new TLRPC.TL_inputStorePaymentPremiumGiveaway(); payload.only_new_subscribers = onlyNewSubscribers; + payload.winners_are_visible = winnersVisible; + payload.prize_description = prizeDesc; payload.until_date = date; payload.flags |= 2; payload.flags |= 4; + if (withAdditionPrize) { + payload.flags |= 16; + } payload.random_id = System.currentTimeMillis(); payload.additional_peers = new ArrayList<>(); for (TLObject o : chats) { @@ -420,6 +452,21 @@ public static List filterGiftOptions(List filterGiftOptionsByBilling(List list) { + if (BoostRepository.isGoogleBillingAvailable()) { + List result = new ArrayList<>(); + for (TLRPC.TL_premiumGiftCodeOption item : list) { + boolean isAvailableInGoogleStore = item.store_product != null; + if (isAvailableInGoogleStore) { + result.add(item); + } + } + return result; + } else { + return list; + } + } + public static void loadCountries(Utilities.Callback>, List>> onDone) { ConnectionsManager connection = ConnectionsManager.getInstance(UserConfig.selectedAccount); @@ -469,8 +516,10 @@ public static void loadGiftOptions(TLRPC.Chat chat, Utilities.Callback { if (response != null) { @@ -508,6 +557,35 @@ public static void loadGiftOptions(TLRPC.Chat chat, Utilities.Callback> onDone) { + MessagesController controller = MessagesController.getInstance(UserConfig.selectedAccount); + ConnectionsManager connection = ConnectionsManager.getInstance(UserConfig.selectedAccount); + if (reqId != 0) { + connection.cancelRequest(reqId, false); + } + if (query == null || query.isEmpty()) { + AndroidUtilities.runOnUIThread(() -> onDone.run(Collections.emptyList())); + return 0; + } + TLRPC.TL_contacts_search req = new TLRPC.TL_contacts_search(); + req.q = query; + req.limit = 50; + return connection.sendRequest(req, (response, error) -> { + if (response instanceof TLRPC.TL_contacts_found) { + TLRPC.TL_contacts_found res = (TLRPC.TL_contacts_found) response; + controller.putUsers(res.users, false); + List result = new ArrayList<>(); + for (int a = 0; a < res.users.size(); a++) { + TLRPC.User user = res.users.get(a); + if (!user.self && !UserObject.isDeleted(user) && !user.bot && !UserObject.isService(user.id)) { + result.add(user); + } + } + AndroidUtilities.runOnUIThread(() -> onDone.run(result)); + } + }); + } + public static void searchChats(long currentChatId, int guid, String query, int count, Utilities.Callback> onDone) { MessagesController controller = MessagesController.getInstance(UserConfig.selectedAccount); ConnectionsManager connection = ConnectionsManager.getInstance(UserConfig.selectedAccount); @@ -591,7 +669,7 @@ public static void applyGiftCode(String slug, Utilities.Callback onDone, U return; } onDone.run(null); - })); + }), ConnectionsManager.RequestFlagFailOnServerErrors); } public static void getGiveawayInfo(MessageObject messageObject, Utilities.Callback onDone, Utilities.Callback onError) { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/BoostViaGiftsBottomSheet.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/BoostViaGiftsBottomSheet.java index 9f89f3fa231..33d1785886d 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/BoostViaGiftsBottomSheet.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/BoostViaGiftsBottomSheet.java @@ -5,16 +5,16 @@ import android.view.Gravity; import android.view.View; import android.view.ViewGroup; -import android.widget.Toast; +import androidx.annotation.NonNull; import androidx.recyclerview.widget.DefaultItemAnimator; +import androidx.recyclerview.widget.RecyclerView; import org.telegram.messenger.AndroidUtilities; import org.telegram.messenger.LocaleController; import org.telegram.messenger.MessagesController; import org.telegram.messenger.NotificationCenter; import org.telegram.messenger.R; -import org.telegram.messenger.SharedConfig; import org.telegram.messenger.UserConfig; import org.telegram.tgnet.TLObject; import org.telegram.tgnet.TLRPC; @@ -33,6 +33,7 @@ import org.telegram.ui.Components.Premium.boosts.cells.ParticipantsTypeCell; import org.telegram.ui.Components.Premium.boosts.cells.BoostTypeCell; import org.telegram.ui.Components.Premium.boosts.cells.DurationCell; +import org.telegram.ui.Components.Premium.boosts.cells.SwitcherCell; import org.telegram.ui.Components.RecyclerListView; import org.telegram.ui.Components.Premium.boosts.adapters.BoostAdapter.Item; @@ -70,6 +71,10 @@ public interface ActionListener { private int top; private Runnable onCloseClick; private final TL_stories.TL_prepaidGiveaway prepaidGiveaway; + private String additionalPrize = ""; + private boolean isAdditionalPrizeSelected; + private boolean isShowWinnersSelected = true; + private final Runnable hideKeyboardRunnable = () -> AndroidUtilities.hideKeyboard(recyclerListView); public BoostViaGiftsBottomSheet(BaseFragment fragment, boolean needFocus, boolean hasFixedSize, long dialogId, TL_stories.TL_prepaidGiveaway prepaidGiveaway) { super(fragment, needFocus, hasFixedSize); @@ -90,7 +95,36 @@ public BoostViaGiftsBottomSheet(BaseFragment fragment, boolean needFocus, boolea recyclerListView.setPadding(backgroundPaddingLeft, 0, backgroundPaddingLeft, AndroidUtilities.dp(BOTTOM_HEIGHT_DP)); recyclerListView.setItemAnimator(itemAnimator); + recyclerListView.setOnScrollListener(new RecyclerView.OnScrollListener() { + @Override + public void onScrollStateChanged(@NonNull RecyclerView recyclerView, int newState) { + if (newState == RecyclerView.SCROLL_STATE_DRAGGING) { + AndroidUtilities.hideKeyboard(recyclerView); + } + } + }); recyclerListView.setOnItemClickListener((view, position) -> { + if (view instanceof SwitcherCell) { + SwitcherCell cell = ((SwitcherCell) view); + int type = cell.getType(); + boolean isChecked = !cell.isChecked(); + cell.setChecked(isChecked); + if (type == SwitcherCell.TYPE_WINNERS) { + isShowWinnersSelected = isChecked; + updateRows(false, false); + } else if (type == SwitcherCell.TYPE_ADDITION_PRIZE) { + cell.setDivider(isChecked); + isAdditionalPrizeSelected = isChecked; + updateRows(false, false); + adapter.notifyAdditionalPrizeItem(isChecked); + adapter.notifyAllVisibleTextDividers(); + if (!isAdditionalPrizeSelected) { + AndroidUtilities.runOnUIThread(hideKeyboardRunnable, 250); + } else { + AndroidUtilities.cancelRunOnUIThread(hideKeyboardRunnable); + } + } + } if (view instanceof BaseCell) { if (view instanceof BoostTypeCell) { int boostType = ((BoostTypeCell) view).getSelectedType(); @@ -121,6 +155,7 @@ public BoostViaGiftsBottomSheet(BaseFragment fragment, boolean needFocus, boolea } else if (view instanceof DurationCell) { selectedMonths = ((TLRPC.TL_premiumGiftCodeOption) ((DurationCell) view).getGifCode()).months; updateRows(false, false); + adapter.notifyAllVisibleTextDividers(); } else if (view instanceof DateEndCell) { BoostDialogs.showDatePicker(fragment.getContext(), selectedEndDate, (notify, timeSec) -> { selectedEndDate = timeSec * 1000L; @@ -141,6 +176,10 @@ public BoostViaGiftsBottomSheet(BaseFragment fragment, boolean needFocus, boolea }, deletedChat -> { selectedChats.remove(deletedChat); updateRows(true, true); + }, text -> { + additionalPrize = text; + updateRows(false, false); + updateRows(true, true); }); updateRows(false, false); actionBtn = new ActionBtnCell(getContext(), resourcesProvider); @@ -175,7 +214,7 @@ public BoostViaGiftsBottomSheet(BaseFragment fragment, boolean needFocus, boolea int dateInt = BoostRepository.prepareServerDate(selectedEndDate); boolean onlyNewSubscribers = selectedParticipantsType == ParticipantsTypeCell.TYPE_NEW; actionBtn.updateLoading(true); - BoostRepository.launchPreparedGiveaway(prepaidGiveaway, selectedChats, selectedCountries, currentChat, dateInt, onlyNewSubscribers, + BoostRepository.launchPreparedGiveaway(prepaidGiveaway, selectedChats, selectedCountries, currentChat, dateInt, onlyNewSubscribers, isShowWinnersSelected, isAdditionalPrizeSelected, additionalPrize, result -> { dismiss(); AndroidUtilities.runOnUIThread(() -> NotificationCenter.getInstance(UserConfig.selectedAccount).postNotificationName(NotificationCenter.boostByChannelCreated, currentChat, true, prepaidGiveaway), 220); @@ -198,7 +237,7 @@ public BoostViaGiftsBottomSheet(BaseFragment fragment, boolean needFocus, boolea boolean onlyNewSubscribers = selectedParticipantsType == ParticipantsTypeCell.TYPE_NEW; int dateInt = BoostRepository.prepareServerDate(selectedEndDate); actionBtn.updateLoading(true); - BoostRepository.payGiveAway(selectedChats, selectedCountries, option, currentChat, dateInt, onlyNewSubscribers, fragment, result -> { + BoostRepository.payGiveAway(selectedChats, selectedCountries, option, currentChat, dateInt, onlyNewSubscribers, fragment, isShowWinnersSelected, isAdditionalPrizeSelected, additionalPrize, result -> { dismiss(); AndroidUtilities.runOnUIThread(() -> NotificationCenter.getInstance(UserConfig.selectedAccount).postNotificationName(NotificationCenter.boostByChannelCreated, currentChat, true), 220); }, error -> { @@ -321,11 +360,6 @@ private void updateRows(boolean animated, boolean notify) { items.add(Item.asParticipants(ParticipantsTypeCell.TYPE_ALL, selectedParticipantsType, true, selectedCountries)); items.add(Item.asParticipants(ParticipantsTypeCell.TYPE_NEW, selectedParticipantsType, false, selectedCountries)); items.add(Item.asDivider(LocaleController.getString("BoostingChooseLimitGiveaway", R.string.BoostingChooseLimitGiveaway), false)); - items.add(Item.asSubTitle(LocaleController.getString("BoostingDateWhenGiveawayEnds", R.string.BoostingDateWhenGiveawayEnds))); - items.add(Item.asDateEnd(selectedEndDate)); - if (!isPreparedGiveaway()) { - items.add(Item.asDivider(LocaleController.formatPluralString("BoostingChooseRandom", getSelectedSliderValue()), false)); - } } if (!isPreparedGiveaway()) { @@ -336,17 +370,56 @@ private void updateRows(boolean animated, boolean notify) { items.add(Item.asDuration(option, option.months, isGiveaway() ? getSelectedSliderValue() : selectedUsers.size(), option.amount, selectedMonths, option.currency, i != options.size() - 1)); } } - String textDivider = !isPreparedGiveaway() ? LocaleController.getString("BoostingStoriesFeaturesAndTerms", R.string.BoostingStoriesFeaturesAndTerms) - : LocaleController.formatPluralString("BoostingChooseRandom", prepaidGiveaway.quantity) + "\n\n" + LocaleController.getString("BoostingStoriesFeaturesAndTerms", R.string.BoostingStoriesFeaturesAndTerms); - items.add(Item.asDivider(AndroidUtilities.replaceSingleTag( - textDivider, - Theme.key_chat_messageLinkIn, 0, () -> { - PremiumPreviewBottomSheet previewBottomSheet = new PremiumPreviewBottomSheet(getBaseFragment(), currentAccount, null, resourcesProvider); - previewBottomSheet.setOnDismissListener(dialog -> adapter.setPausedStars(false)); - previewBottomSheet.setOnShowListener(dialog -> adapter.setPausedStars(true)); - previewBottomSheet.show(); - }, - resourcesProvider), true)); + + if (!isPreparedGiveaway()) { + items.add(Item.asDivider(AndroidUtilities.replaceSingleTag( + LocaleController.getString("BoostingStoriesFeaturesAndTerms", R.string.BoostingStoriesFeaturesAndTerms), + Theme.key_chat_messageLinkIn, 0, () -> { + PremiumPreviewBottomSheet previewBottomSheet = new PremiumPreviewBottomSheet(getBaseFragment(), currentAccount, null, resourcesProvider); + previewBottomSheet.setOnDismissListener(dialog -> adapter.setPausedStars(false)); + previewBottomSheet.setOnShowListener(dialog -> adapter.setPausedStars(true)); + previewBottomSheet.show(); + }, + resourcesProvider), true)); + } + + if (selectedBoostType == BoostTypeCell.TYPE_GIVEAWAY) { + items.add(Item.asSwitcher(LocaleController.getString("BoostingGiveawayAdditionalPrizes", R.string.BoostingGiveawayAdditionalPrizes), isAdditionalPrizeSelected, isAdditionalPrizeSelected, SwitcherCell.TYPE_ADDITION_PRIZE)); + + if (isAdditionalPrizeSelected) { + int quantity = isPreparedGiveaway() ? prepaidGiveaway.quantity : getSelectedSliderValue(); + items.add(Item.asEnterPrize(quantity)); + String months = LocaleController.formatPluralString("BoldMonths", selectedMonths); + if (additionalPrize.isEmpty()) { + items.add(Item.asDivider(AndroidUtilities.replaceTags(LocaleController.formatPluralString("BoostingGiveawayAdditionPrizeCountHint", quantity, months)), false)); + } else { + items.add(Item.asDivider(AndroidUtilities.replaceTags(LocaleController.formatPluralString("BoostingGiveawayAdditionPrizeCountNameHint", quantity, additionalPrize, months)), false)); + } + } else { + items.add(Item.asDivider(LocaleController.getString("BoostingGiveawayAdditionPrizeHint", R.string.BoostingGiveawayAdditionPrizeHint), false)); + } + + items.add(Item.asSwitcher(LocaleController.getString("BoostingGiveawayShowWinners", R.string.BoostingGiveawayShowWinners), isShowWinnersSelected, false, SwitcherCell.TYPE_WINNERS)); + items.add(Item.asDivider(LocaleController.getString("BoostingGiveawayShowWinnersHint", R.string.BoostingGiveawayShowWinnersHint), false)); + + items.add(Item.asSubTitle(LocaleController.getString("BoostingDateWhenGiveawayEnds", R.string.BoostingDateWhenGiveawayEnds))); + items.add(Item.asDateEnd(selectedEndDate)); + + if (!isPreparedGiveaway()) { + items.add(Item.asDivider(LocaleController.formatPluralString("BoostingChooseRandom", getSelectedSliderValue()), false)); + } else { + items.add(Item.asDivider(AndroidUtilities.replaceSingleTag( + LocaleController.formatPluralString("BoostingChooseRandom", prepaidGiveaway.quantity) + "\n\n" + LocaleController.getString("BoostingStoriesFeaturesAndTerms", R.string.BoostingStoriesFeaturesAndTerms), + Theme.key_chat_messageLinkIn, 0, () -> { + PremiumPreviewBottomSheet previewBottomSheet = new PremiumPreviewBottomSheet(getBaseFragment(), currentAccount, null, resourcesProvider); + previewBottomSheet.setOnDismissListener(dialog -> adapter.setPausedStars(false)); + previewBottomSheet.setOnShowListener(dialog -> adapter.setPausedStars(true)); + previewBottomSheet.show(); + }, + resourcesProvider), true)); + } + } + if (adapter == null) { return; } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/DiscountSpan.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/DiscountSpan.java new file mode 100644 index 00000000000..00a72c9f365 --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/DiscountSpan.java @@ -0,0 +1,84 @@ +package org.telegram.ui.Components.Premium.boosts; + +import static org.telegram.messenger.AndroidUtilities.dp; + +import android.graphics.Canvas; +import android.graphics.Color; +import android.graphics.Paint; +import android.text.Layout; +import android.text.SpannableStringBuilder; +import android.text.StaticLayout; +import android.text.TextPaint; +import android.text.style.ReplacementSpan; + +import androidx.annotation.NonNull; +import androidx.annotation.Nullable; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.LocaleController; +import org.telegram.messenger.R; +import org.telegram.ui.ActionBar.Theme; + +public class DiscountSpan extends ReplacementSpan { + + public static CharSequence applySpan(CharSequence str, int discount) { + SpannableStringBuilder spannableStringBuilder = new SpannableStringBuilder("d "); + spannableStringBuilder.append(str); + DiscountSpan span = new DiscountSpan(11, discount); + span.setColor(Theme.getColor(Theme.key_premiumGradient1)); + spannableStringBuilder.setSpan(span, 0, 1, 0); + return spannableStringBuilder; + } + + TextPaint textPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG); + Paint bgPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + StaticLayout layout; + float width, height; + int discount; + private int color; + + public DiscountSpan(float textSize, int discount) { + textPaint.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); + bgPaint.setStyle(Paint.Style.FILL); + textPaint.setTextSize(dp(textSize)); + this.discount = discount; + } + + public void setColor(int color) { + this.color = color; + } + + public void makeLayout() { + if (layout == null) { + layout = new StaticLayout(LocaleController.formatString(R.string.GiftPremiumOptionDiscount, discount), textPaint, AndroidUtilities.displaySize.x, Layout.Alignment.ALIGN_NORMAL, 1, 0, false); + width = layout.getLineWidth(0); + height = layout.getHeight(); + } + } + + @Override + public int getSize(@NonNull Paint paint, CharSequence text, int start, int end, @Nullable Paint.FontMetricsInt fm) { + makeLayout(); + return (int) (dp(13) + width); + } + + @Override + public void draw(@NonNull Canvas canvas, CharSequence text, int start, int end, float _x, int top, int _y, int bottom, @NonNull Paint paint) { + makeLayout(); + int color = this.color; + if (color == 0) { + color = paint.getColor(); + } + bgPaint.setColor(color); + textPaint.setColor(AndroidUtilities.computePerceivedBrightness(color) > .721f ? Color.BLACK : Color.WHITE); + float x = _x + dp(6), y = _y - height + dp(2f); + AndroidUtilities.rectTmp.set(x, y, x + width, y + height); + float r = dp(4f); + AndroidUtilities.rectTmp.inset(dp(-4.5f), dp(-1.66f)); + canvas.drawRoundRect(AndroidUtilities.rectTmp, r, r, bgPaint); + canvas.save(); + canvas.translate(x, y); + layout.draw(canvas); + canvas.restore(); + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/GiftInfoBottomSheet.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/GiftInfoBottomSheet.java index d9540ae320f..f5ef0d54c08 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/GiftInfoBottomSheet.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/GiftInfoBottomSheet.java @@ -44,8 +44,23 @@ public static void show(BaseFragment fragment, String slug, Browser.Progress pro if (fragment.getParentActivity() == null) { return; } - GiftInfoBottomSheet alert = new GiftInfoBottomSheet(fragment, false, true, giftCode, slug); - fragment.showDialog(alert); + + if (giftCode.from_id == null) { + TLRPC.TL_premiumGiftOption giftOption = new TLRPC.TL_premiumGiftOption(); + giftOption.months = giftCode.months; + TLRPC.User user = null; + if (fragment instanceof ChatActivity) { + user = ((ChatActivity) fragment).getCurrentUser(); + } + if (user == null || user.self) { + user = new TLRPC.TL_user(); + } + boolean isUsed = giftCode.used_date != 0; + PremiumPreviewGiftLinkBottomSheet.show(slug, giftOption, user, isUsed); + } else { + fragment.showDialog(new GiftInfoBottomSheet(fragment, false, true, giftCode, slug)); + } + if (progress != null) { progress.end(); } @@ -110,7 +125,7 @@ public GiftInfoBottomSheet(BaseFragment fragment, boolean needFocus, boolean has setApplyBottomPadding(false); fixNavigationBar(); updateTitle(); - adapter.init(fragment, giftCode, slug); + adapter.init(fragment, giftCode, slug, container); } @Override diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/GradientButtonWithCounterView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/GradientButtonWithCounterView.java new file mode 100644 index 00000000000..cd82c38961c --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/GradientButtonWithCounterView.java @@ -0,0 +1,53 @@ +package org.telegram.ui.Components.Premium.boosts; + +import static org.telegram.messenger.AndroidUtilities.dp; + +import android.annotation.SuppressLint; +import android.content.Context; +import android.graphics.Canvas; +import android.graphics.RectF; + +import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.Components.Premium.PremiumGradient; +import org.telegram.ui.Components.voip.CellFlickerDrawable; +import org.telegram.ui.Stories.recorder.ButtonWithCounterView; + +@SuppressLint("ViewConstructor") +public class GradientButtonWithCounterView extends ButtonWithCounterView { + + private final RectF rect = new RectF(); + private boolean incGradient; + private float progress; + private final CellFlickerDrawable flickerDrawable; + + public GradientButtonWithCounterView(Context context, boolean filled, Theme.ResourcesProvider resourcesProvider) { + super(context, filled, resourcesProvider); + flickerDrawable = new CellFlickerDrawable(); + flickerDrawable.animationSpeedScale = 1.2f; + flickerDrawable.drawFrame = false; + flickerDrawable.repeatProgress = 4f; + } + + @Override + protected void onDraw(Canvas canvas) { + if (incGradient) { + progress += 16f / 1000f; + if (progress > 3) { + incGradient = false; + } + } else { + progress -= 16f / 1000f; + if (progress < 1) { + incGradient = true; + } + } + + rect.set(0, 0, getMeasuredWidth(), getMeasuredHeight()); + PremiumGradient.getInstance().updateMainGradientMatrix(0, 0, getMeasuredWidth(), getMeasuredHeight(), -getMeasuredWidth() * 0.1f * progress, 0); + canvas.drawRoundRect(rect, dp(8), dp(8), PremiumGradient.getInstance().getMainGradientPaint()); + flickerDrawable.setParentWidth(getMeasuredWidth()); + flickerDrawable.draw(canvas, rect, dp(8), null); + super.onDraw(canvas); + invalidate(); + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/PremiumPreviewGiftLinkBottomSheet.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/PremiumPreviewGiftLinkBottomSheet.java new file mode 100644 index 00000000000..262722dc415 --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/PremiumPreviewGiftLinkBottomSheet.java @@ -0,0 +1,170 @@ +package org.telegram.ui.Components.Premium.boosts; + +import static org.telegram.messenger.AndroidUtilities.dp; +import static org.telegram.messenger.AndroidUtilities.replaceTags; +import static org.telegram.messenger.LocaleController.getString; + +import android.content.Context; +import android.os.Bundle; +import android.text.SpannableStringBuilder; +import android.view.Gravity; +import android.view.View; +import android.view.ViewGroup; +import android.widget.FrameLayout; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.FileLog; +import org.telegram.messenger.R; +import org.telegram.messenger.SendMessagesHelper; +import org.telegram.messenger.UserConfig; +import org.telegram.messenger.browser.Browser; +import org.telegram.tgnet.TLRPC; +import org.telegram.ui.ActionBar.BaseFragment; +import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.Components.Bulletin; +import org.telegram.ui.Components.LayoutHelper; +import org.telegram.ui.Components.Premium.GiftPremiumBottomSheet; +import org.telegram.ui.Components.Premium.PremiumPreviewBottomSheet; +import org.telegram.ui.Components.Premium.boosts.cells.ActionBtnCell; +import org.telegram.ui.Components.Premium.boosts.cells.LinkCell; +import org.telegram.ui.DialogsActivity; +import org.telegram.ui.LaunchActivity; + +public class PremiumPreviewGiftLinkBottomSheet extends PremiumPreviewBottomSheet { + private static final int BOTTOM_HEIGHT_DP = 68; + private static final int CELL_TYPE_LINK = 6; + private static PremiumPreviewGiftLinkBottomSheet instance; + + private ActionBtnCell actionBtn; + private final String slug; + private final boolean isUsed; + + public static void show(String slug, TLRPC.TL_premiumGiftOption giftOption, TLRPC.User user, Browser.Progress progress) { + GiftInfoBottomSheet.show(LaunchActivity.getLastFragment(), slug, progress); + } + + public static void show(String slug, TLRPC.TL_premiumGiftOption giftOption, TLRPC.User user, boolean isUsed) { + BaseFragment fragment = LaunchActivity.getLastFragment(); + if (fragment == null || instance != null) { + return; + } + GiftPremiumBottomSheet.GiftTier tier = new GiftPremiumBottomSheet.GiftTier(giftOption); + PremiumPreviewGiftLinkBottomSheet sheet = new PremiumPreviewGiftLinkBottomSheet(fragment, UserConfig.selectedAccount, user, tier, slug, isUsed, fragment.getResourceProvider()); + sheet.show(); + instance = sheet; + } + + public PremiumPreviewGiftLinkBottomSheet(BaseFragment fragment, int currentAccount, TLRPC.User user, GiftPremiumBottomSheet.GiftTier gift, String slug, boolean isUsed, Theme.ResourcesProvider resourcesProvider) { + super(fragment, currentAccount, user, gift, resourcesProvider); + this.slug = slug; + this.isUsed = isUsed; + init(); + } + + @Override + protected void updateRows() { + paddingRow = rowCount++; + additionStartRow = rowCount; + additionEndRow = ++rowCount; + featuresStartRow = rowCount; + rowCount += premiumFeatures.size(); + featuresEndRow = rowCount; + sectionRow = rowCount++; + } + + @Override + public void setTitle(boolean animated) { + super.setTitle(animated); + subtitleView.setLineSpacing(AndroidUtilities.dp(2), 1f); + ((ViewGroup.MarginLayoutParams) subtitleView.getLayoutParams()).bottomMargin = dp(14); + ((ViewGroup.MarginLayoutParams) subtitleView.getLayoutParams()).topMargin = dp(12); + String subTitleText = getString("GiftPremiumAboutThisLink", R.string.GiftPremiumAboutThisLink); + SpannableStringBuilder subTitleWithLink = AndroidUtilities.replaceSingleTag( + subTitleText, + Theme.key_chat_messageLinkIn, 0, + this::share); + subtitleView.setText(AndroidUtilities.replaceCharSequence("%1$s", subTitleWithLink, replaceTags(getString("GiftPremiumAboutThisLinkEnd", R.string.GiftPremiumAboutThisLinkEnd)))); + } + + private void share() { + final String slugLink = "https://t.me/giftcode/" + slug; + Bundle args = new Bundle(); + args.putBoolean("onlySelect", true); + args.putInt("dialogsType", DialogsActivity.DIALOGS_TYPE_FORWARD); + DialogsActivity dialogFragment = new DialogsActivity(args); + dialogFragment.setDelegate((fragment1, dids, message, param, topicsFragment) -> { + long did = 0; + for (int a = 0; a < dids.size(); a++) { + did = dids.get(a).dialogId; + getBaseFragment().getSendMessagesHelper().sendMessage(SendMessagesHelper.SendMessageParams.of(slugLink, did, null, null, null, true, null, null, null, true, 0, null, false)); + } + fragment1.finishFragment(); + BoostDialogs.showGiftLinkForwardedBulletin(did); + return true; + }); + getBaseFragment().presentFragment(dialogFragment); + dismiss(); + } + + @Override + protected View onCreateAdditionCell(int viewType, Context context) { + if (viewType == CELL_TYPE_LINK) { + LinkCell cell = new LinkCell(context, getBaseFragment(), resourcesProvider); + cell.setPadding(0, 0, 0, dp(8)); + return cell; + } + return null; + } + + @Override + protected void onBindAdditionCell(View view, int pos) { + ((LinkCell) view).setSlug(slug); + } + + @Override + protected int getAdditionItemViewType(int position) { + return CELL_TYPE_LINK; + } + + private void init() { + Bulletin.addDelegate((FrameLayout) containerView, new Bulletin.Delegate() { + @Override + public int getBottomOffset(int tag) { + return dp(BOTTOM_HEIGHT_DP); + } + }); + if (!isUsed) { + recyclerListView.setPadding(backgroundPaddingLeft, 0, backgroundPaddingLeft, dp(BOTTOM_HEIGHT_DP)); + actionBtn = new ActionBtnCell(getContext(), resourcesProvider); + actionBtn.setOnClickListener(v -> { + if (actionBtn.isLoading()) { + return; + } + actionBtn.updateLoading(true); + BoostRepository.applyGiftCode(slug, result -> { + actionBtn.updateLoading(false); + dismiss(); + AndroidUtilities.runOnUIThread(() -> { + PremiumPreviewBottomSheet previewBottomSheet = new PremiumPreviewBottomSheet(getBaseFragment(), UserConfig.selectedAccount, null, null, resourcesProvider) + .setAnimateConfetti(true) + .setAnimateConfettiWithStars(true) + .setOutboundGift(true); + getBaseFragment().showDialog(previewBottomSheet); + }, 200); + }, error -> { + actionBtn.updateLoading(false); + BoostDialogs.processApplyGiftCodeError(error, (FrameLayout) containerView, resourcesProvider, this::share); + }); + }); + actionBtn.setActivateForFreeStyle(); + containerView.addView(actionBtn, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, BOTTOM_HEIGHT_DP, Gravity.BOTTOM, 0, 0, 0, 0)); + } + fixNavigationBar(); + } + + @Override + public void dismissInternal() { + super.dismissInternal(); + instance = null; + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/PremiumPreviewGiftSentBottomSheet.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/PremiumPreviewGiftSentBottomSheet.java new file mode 100644 index 00000000000..69f34c6f9f0 --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/PremiumPreviewGiftSentBottomSheet.java @@ -0,0 +1,158 @@ +package org.telegram.ui.Components.Premium.boosts; + +import static org.telegram.messenger.AndroidUtilities.dp; +import static org.telegram.messenger.AndroidUtilities.replaceTags; +import static org.telegram.messenger.LocaleController.formatPluralString; +import static org.telegram.messenger.LocaleController.formatString; +import static org.telegram.messenger.LocaleController.getPluralString; +import static org.telegram.messenger.LocaleController.getString; + +import android.graphics.Outline; +import android.os.Build; +import android.util.TypedValue; +import android.view.Gravity; +import android.view.View; +import android.view.ViewGroup; +import android.view.ViewOutlineProvider; +import android.widget.LinearLayout; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.R; +import org.telegram.messenger.UserConfig; +import org.telegram.messenger.UserObject; +import org.telegram.tgnet.TLRPC; +import org.telegram.ui.ActionBar.BaseFragment; +import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.Components.LayoutHelper; +import org.telegram.ui.Components.Premium.PremiumPreviewBottomSheet; +import org.telegram.ui.Components.Premium.boosts.cells.ActionBtnCell; +import org.telegram.ui.LaunchActivity; + +import java.util.ArrayList; +import java.util.List; + +public class PremiumPreviewGiftSentBottomSheet extends PremiumPreviewBottomSheet { + private static final int BOTTOM_HEIGHT_DP = 64; + + private final List selectedUsers = new ArrayList<>(); + + public static void show(List selectedUsers) { + BaseFragment fragment = LaunchActivity.getLastFragment(); + if (fragment == null) { + return; + } + PremiumPreviewGiftSentBottomSheet sheet = new PremiumPreviewGiftSentBottomSheet(fragment, UserConfig.selectedAccount, selectedUsers, fragment.getResourceProvider()); + sheet.setAnimateConfetti(true); + sheet.setAnimateConfettiWithStars(true); + sheet.show(); + } + + public PremiumPreviewGiftSentBottomSheet(BaseFragment fragment, int currentAccount, List selectedUsers, Theme.ResourcesProvider resourcesProvider) { + super(fragment, currentAccount, null, null, resourcesProvider); + this.selectedUsers.addAll(selectedUsers); + init(); + } + + @Override + protected boolean needDefaultPremiumBtn() { + return false; + } + + @Override + protected void updateRows() { + rowCount = 0; + paddingRow = rowCount++; + featuresStartRow = rowCount; + rowCount += premiumFeatures.size(); + featuresEndRow = rowCount; + termsRow = rowCount++; + } + + @Override + public void setTitle(boolean animated) { + titleView[0].setTextSize(TypedValue.COMPLEX_UNIT_DIP, 20); + subtitleView.setPadding(dp(30), 0, dp(30), 0); + subtitleView.setLineSpacing(AndroidUtilities.dp(2), 1f); + titleView[0].setText(getPluralString("GiftPremiumGiftsSent", selectedUsers.size())); + ((ViewGroup.MarginLayoutParams) subtitleView.getLayoutParams()).bottomMargin = dp(16); + ((ViewGroup.MarginLayoutParams) subtitleView.getLayoutParams()).topMargin = dp(4f); + + String subTitle; + switch (selectedUsers.size()) { + case 1: { + String names = formatString("GiftPremiumUsersOne", R.string.GiftPremiumUsersOne, UserObject.getFirstName(selectedUsers.get(0))); + subTitle = formatString("GiftPremiumUsersPurchasedManyZero", R.string.GiftPremiumUsersPurchasedManyZero, names); + break; + } + case 2: { + String names = formatString("GiftPremiumUsersTwo", R.string.GiftPremiumUsersTwo, UserObject.getFirstName(selectedUsers.get(0)), UserObject.getFirstName(selectedUsers.get(1))); + subTitle = formatString("GiftPremiumUsersPurchasedManyZero", R.string.GiftPremiumUsersPurchasedManyZero, names); + break; + } + case 3: { + String names = formatString("GiftPremiumUsersThree", R.string.GiftPremiumUsersThree, UserObject.getFirstName(selectedUsers.get(0)), UserObject.getFirstName(selectedUsers.get(1)), UserObject.getFirstName(selectedUsers.get(2))); + subTitle = formatString("GiftPremiumUsersPurchasedManyZero", R.string.GiftPremiumUsersPurchasedManyZero, names); + break; + } + default: { + String names = formatString("GiftPremiumUsersThree", R.string.GiftPremiumUsersThree, UserObject.getFirstName(selectedUsers.get(0)), UserObject.getFirstName(selectedUsers.get(1)), UserObject.getFirstName(selectedUsers.get(2))); + subTitle = formatPluralString("GiftPremiumUsersPurchasedMany", selectedUsers.size() - 3, names); + break; + } + } + subtitleView.setText(replaceTags(subTitle)); + + subtitleView.append("\n"); + subtitleView.append("\n"); + + if (selectedUsers.size() == 1) { + subtitleView.append(replaceTags(formatString("GiftPremiumGiftsSentStatusForUser", R.string.GiftPremiumGiftsSentStatusForUser, UserObject.getFirstName(selectedUsers.get(0))))); + } else { + subtitleView.append(replaceTags(getString("GiftPremiumGiftsSentStatus", R.string.GiftPremiumGiftsSentStatus))); + } + } + + private void init() { + updateRows(); + useBackgroundTopPadding = false; + setApplyTopPadding(false); + backgroundPaddingTop = 0; + ActionBtnCell actionBtn = new ActionBtnCell(getContext(), resourcesProvider); + actionBtn.setOnClickListener(v -> dismiss()); + actionBtn.setCloseStyle(true); + containerView.addView(actionBtn, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, BOTTOM_HEIGHT_DP, Gravity.BOTTOM, 0, 0, 0, 0)); + + recyclerListView.setPadding(backgroundPaddingLeft, 0, backgroundPaddingLeft, dp(BOTTOM_HEIGHT_DP)); + overrideTitleIcon = PremiumPreviewGiftToUsersBottomSheet.AvatarHolderView.createAvatarsContainer(getContext(), selectedUsers); + fixNavigationBar(); + } + + protected void afterCellCreated(int viewType, View view) { + if (viewType == 0) { + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { + view.setOutlineProvider(new ViewOutlineProvider() { + @Override + public void getOutline(View view, Outline outline) { + float cornerRadius = AndroidUtilities.dp(12); + outline.setRoundRect(0, 0, view.getWidth(), (int) (view.getHeight() + cornerRadius), cornerRadius); + } + }); + view.setClipToOutline(true); + view.setBackgroundColor(Theme.getColor(Theme.key_windowBackgroundGray, resourcesProvider)); + } + ((ViewGroup.MarginLayoutParams) view.getLayoutParams()).topMargin = -dp(6); + } + } + + @Override + protected void attachIconContainer(LinearLayout container) { + container.addView(overrideTitleIcon, LayoutHelper.createLinear( + LayoutHelper.MATCH_PARENT, + selectedUsers.size() == 1 ? 94 : 83, + 0, + selectedUsers.size() == 1 ? 28 : 34, + 0, + selectedUsers.size() == 1 ? 9 : 14) + ); + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/PremiumPreviewGiftToUsersBottomSheet.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/PremiumPreviewGiftToUsersBottomSheet.java new file mode 100644 index 00000000000..eb584bf3ed1 --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/PremiumPreviewGiftToUsersBottomSheet.java @@ -0,0 +1,402 @@ +package org.telegram.ui.Components.Premium.boosts; + +import static org.telegram.messenger.AndroidUtilities.dp; +import static org.telegram.messenger.AndroidUtilities.replaceTags; +import static org.telegram.messenger.LocaleController.formatPluralString; +import static org.telegram.messenger.LocaleController.formatString; +import static org.telegram.messenger.LocaleController.getString; + +import android.annotation.SuppressLint; +import android.content.Context; +import android.graphics.Canvas; +import android.graphics.Outline; +import android.graphics.Paint; +import android.os.Build; +import android.text.Spannable; +import android.text.SpannableStringBuilder; +import android.text.TextPaint; +import android.text.TextUtils; +import android.util.TypedValue; +import android.view.Gravity; +import android.view.View; +import android.view.ViewGroup; +import android.view.ViewOutlineProvider; +import android.widget.FrameLayout; +import android.widget.LinearLayout; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.BillingController; +import org.telegram.messenger.NotificationCenter; +import org.telegram.messenger.R; +import org.telegram.messenger.UserConfig; +import org.telegram.messenger.UserObject; +import org.telegram.tgnet.TLRPC; +import org.telegram.ui.ActionBar.BaseFragment; +import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.Cells.HeaderCell; +import org.telegram.ui.Cells.ShadowSectionCell; +import org.telegram.ui.Components.AvatarDrawable; +import org.telegram.ui.Components.BackupImageView; +import org.telegram.ui.Components.ColoredImageSpan; +import org.telegram.ui.Components.LayoutHelper; +import org.telegram.ui.Components.Premium.PremiumGradient; +import org.telegram.ui.Components.Premium.PremiumPreviewBottomSheet; +import org.telegram.ui.Components.Premium.StarParticlesView; +import org.telegram.ui.Components.Premium.boosts.cells.DurationWithDiscountCell; +import org.telegram.ui.Components.Premium.boosts.cells.selector.SelectorBtnCell; +import org.telegram.ui.LaunchActivity; + +import java.util.ArrayList; +import java.util.Collections; +import java.util.Comparator; +import java.util.List; + +public class PremiumPreviewGiftToUsersBottomSheet extends PremiumPreviewBottomSheet { + private static final int BOTTOM_HEIGHT_DP = 64; + private static final int CELL_TYPE_HEADER = 6, + CELL_TYPE_SHADOW = 7, + CELL_TYPE_DURATION = 8; + + private GradientButtonWithCounterView actionBtn; + private SelectorBtnCell buttonContainer; + private final List selectedUsers = new ArrayList<>(); + private final List giftCodeOptions = new ArrayList<>(); + private int selectedMonths = 3; + + public static void show(List selectedUsers, List giftCodeOptions) { + BaseFragment fragment = LaunchActivity.getLastFragment(); + if (fragment == null) { + return; + } + new PremiumPreviewGiftToUsersBottomSheet(fragment, UserConfig.selectedAccount, selectedUsers, giftCodeOptions, fragment.getResourceProvider()).show(); + } + + public PremiumPreviewGiftToUsersBottomSheet(BaseFragment fragment, int currentAccount, List selectedUsers, List giftCodeOptions, Theme.ResourcesProvider resourcesProvider) { + super(fragment, currentAccount, null, null, resourcesProvider); + this.selectedUsers.addAll(selectedUsers); + this.giftCodeOptions.addAll(giftCodeOptions); + Collections.sort(giftCodeOptions, Comparator.comparingLong(o -> o.amount)); + init(); + } + + @Override + protected boolean needDefaultPremiumBtn() { + return false; + } + + @Override + protected void updateRows() { + rowCount = 0; + paddingRow = rowCount++; + additionStartRow = rowCount; + rowCount += (giftCodeOptions != null ? giftCodeOptions.size() : 0) + 2; + additionEndRow = rowCount; + featuresStartRow = rowCount; + rowCount += premiumFeatures.size(); + featuresEndRow = rowCount; + termsRow = rowCount++; + } + + @Override + public void setTitle(boolean animated) { + titleView[0].setTextSize(TypedValue.COMPLEX_UNIT_DIP, 20); + subtitleView.setPadding(dp(30), 0, dp(30), 0); + subtitleView.setLineSpacing(AndroidUtilities.dp(2), 1f); + titleView[0].setText(getString("GiftTelegramPremiumTitle", R.string.GiftTelegramPremiumTitle)); + ((ViewGroup.MarginLayoutParams) subtitleView.getLayoutParams()).bottomMargin = dp(16); + ((ViewGroup.MarginLayoutParams) subtitleView.getLayoutParams()).topMargin = dp(4f); + + String subTitle; + switch (selectedUsers.size()) { + case 1: { + String names = formatString("GiftPremiumUsersOne", R.string.GiftPremiumUsersOne, UserObject.getFirstName(selectedUsers.get(0))); + subTitle = formatString("GiftPremiumUsersGiveAccessManyZero", R.string.GiftPremiumUsersGiveAccessManyZero, names); + break; + } + case 2: { + String names = formatString("GiftPremiumUsersTwo", R.string.GiftPremiumUsersTwo, UserObject.getFirstName(selectedUsers.get(0)), UserObject.getFirstName(selectedUsers.get(1))); + subTitle = formatString("GiftPremiumUsersGiveAccessManyZero", R.string.GiftPremiumUsersGiveAccessManyZero, names); + break; + } + case 3: { + String names = formatString("GiftPremiumUsersThree", R.string.GiftPremiumUsersThree, UserObject.getFirstName(selectedUsers.get(0)), UserObject.getFirstName(selectedUsers.get(1)), UserObject.getFirstName(selectedUsers.get(2))); + subTitle = formatString("GiftPremiumUsersGiveAccessManyZero", R.string.GiftPremiumUsersGiveAccessManyZero, names); + break; + } + default: { + String names = formatString("GiftPremiumUsersThree", R.string.GiftPremiumUsersThree, UserObject.getFirstName(selectedUsers.get(0)), UserObject.getFirstName(selectedUsers.get(1)), UserObject.getFirstName(selectedUsers.get(2))); + subTitle = formatPluralString("GiftPremiumUsersGiveAccessMany", selectedUsers.size() - 3, names); + break; + } + } + subtitleView.setText(replaceTags(subTitle)); + + subtitleView.append("\n"); + subtitleView.append("\n"); + CharSequence boostInfo = replaceTags(formatPluralString("GiftPremiumWillReceiveBoostsPlural", selectedUsers.size() * BoostRepository.boostsPerSentGift())); + SpannableStringBuilder boostInfoSpannableBuilder = new SpannableStringBuilder(boostInfo); + ColoredImageSpan span = new ColoredImageSpan(R.drawable.mini_boost_button); + span.setSize(dp(20)); + span.setWidth(dp(11)); + span.setTranslateX(-dp(4)); + span.setTranslateY(-dp(1)); + span.setColorKey(Theme.key_windowBackgroundWhiteBlueText4); + String lightning = "⚡"; + int lightningIndex = TextUtils.indexOf(boostInfo, lightning); + if (lightningIndex >= 0) { + boostInfoSpannableBuilder.setSpan(span, lightningIndex, lightningIndex + lightning.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE); + } + subtitleView.append(boostInfoSpannableBuilder); + } + + @Override + protected View onCreateAdditionCell(int viewType, Context context) { + switch (viewType) { + case CELL_TYPE_HEADER: { + HeaderCell cell = new HeaderCell(context,Theme.key_windowBackgroundWhiteBlueHeader, 21, 12, false, resourcesProvider); + cell.setTextSize(15); + cell.setPadding(0, 0, 0, dp(2)); + cell.setText(getString("GiftPremiumWhatsIncluded", R.string.GiftPremiumWhatsIncluded)); + return cell; + } + case CELL_TYPE_SHADOW: { + return new ShadowSectionCell(context, 12, Theme.getColor(Theme.key_windowBackgroundGray, resourcesProvider)); + } + case CELL_TYPE_DURATION: { + return new DurationWithDiscountCell(context, resourcesProvider); + } + default: + return null; + } + } + + @Override + protected void onAdditionItemClicked(View view) { + if (view instanceof DurationWithDiscountCell) { + DurationWithDiscountCell cell = ((DurationWithDiscountCell) view); + selectedMonths = cell.getOption().months; + cell.markChecked(recyclerListView); + updateActionButton(true); + } + } + + @Override + protected boolean isAdditionViewClickable(int viewType) { + return viewType == 8; + } + + @Override + protected void onBindAdditionCell(View view, int pos) { + if (view instanceof DurationWithDiscountCell) { + pos = pos - 1; + TLRPC.TL_premiumGiftCodeOption option = giftCodeOptions.get(pos); + ((DurationWithDiscountCell) view).setDuration(option, giftCodeOptions.get(giftCodeOptions.size() - 1), selectedUsers.size(), pos != giftCodeOptions.size() - 1, selectedMonths == option.months); + } + } + + @Override + protected int getAdditionItemViewType(int position) { + if (position <= giftCodeOptions.size()) { + return CELL_TYPE_DURATION; + } else if (position == giftCodeOptions.size() + 1) { + return CELL_TYPE_SHADOW; + } else if (position == giftCodeOptions.size() + 2) { + return CELL_TYPE_HEADER; + } + return 0; + } + + private TLRPC.TL_premiumGiftCodeOption getSelectedOption() { + for (TLRPC.TL_premiumGiftCodeOption giftCodeOption : giftCodeOptions) { + if (giftCodeOption.months == selectedMonths) { + return giftCodeOption; + } + } + return giftCodeOptions.get(0); + } + + private void updateActionButton(boolean animated) { + TLRPC.TL_premiumGiftCodeOption giftCodeOption = getSelectedOption(); + String priceStr = BillingController.getInstance().formatCurrency(giftCodeOption.amount, giftCodeOption.currency); + if (selectedUsers.size() == 1) { + actionBtn.setText(formatString("GiftSubscriptionFor", R.string.GiftSubscriptionFor, priceStr), animated); + } else { + actionBtn.setText(formatPluralString("GiftSubscriptionCountFor", selectedUsers.size(), priceStr), animated); + } + } + + private void chooseMaxSelectedMonths() { + for (TLRPC.TL_premiumGiftCodeOption giftCodeOption : giftCodeOptions) { + selectedMonths = Math.max(giftCodeOption.months, selectedMonths); + } + } + + private void init() { + chooseMaxSelectedMonths(); + updateRows(); + useBackgroundTopPadding = false; + setApplyTopPadding(false); + backgroundPaddingTop = 0; + recyclerListView.setPadding(backgroundPaddingLeft, 0, backgroundPaddingLeft, dp(BOTTOM_HEIGHT_DP)); + + buttonContainer = new SelectorBtnCell(getContext(), resourcesProvider, recyclerListView); + buttonContainer.setClickable(true); + buttonContainer.setOrientation(LinearLayout.VERTICAL); + buttonContainer.setPadding(dp(8), dp(8), dp(8), dp(8)); + buttonContainer.setBackgroundColor(Theme.getColor(Theme.key_dialogBackground, resourcesProvider)); + actionBtn = new GradientButtonWithCounterView(getContext(), true, resourcesProvider); + actionBtn.setOnClickListener(v -> { + if (actionBtn.isLoading()) { + return; + } + actionBtn.setLoading(true); + BoostRepository.payGiftCode(new ArrayList<>(selectedUsers), getSelectedOption(), null, getBaseFragment(), result -> { + dismiss(); + NotificationCenter.getInstance(UserConfig.selectedAccount).postNotificationName(NotificationCenter.giftsToUserSent); + AndroidUtilities.runOnUIThread(() -> PremiumPreviewGiftSentBottomSheet.show(selectedUsers), 250); + }, error -> { + actionBtn.setLoading(false); + BoostDialogs.showToastError(getContext(), error); + }); + }); + buttonContainer.addView(actionBtn, LayoutHelper.createLinear(LayoutHelper.MATCH_PARENT, 48, Gravity.BOTTOM | Gravity.FILL_HORIZONTAL)); + containerView.addView(buttonContainer, LayoutHelper.createFrameMarginPx(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.BOTTOM | Gravity.FILL_HORIZONTAL, backgroundPaddingLeft, 0, backgroundPaddingLeft, 0)); + + overrideTitleIcon = AvatarHolderView.createAvatarsContainer(getContext(), selectedUsers); + updateActionButton(false); + fixNavigationBar(); + } + + protected void afterCellCreated(int viewType, View view) { + if (viewType == 0) { + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { + view.setOutlineProvider(new ViewOutlineProvider() { + @Override + public void getOutline(View view, Outline outline) { + float cornerRadius = AndroidUtilities.dp(12); + outline.setRoundRect(0, 0, view.getWidth(), (int) (view.getHeight() + cornerRadius), cornerRadius); + } + }); + view.setClipToOutline(true); + view.setBackgroundColor(Theme.getColor(Theme.key_windowBackgroundGray, resourcesProvider)); + } + ((ViewGroup.MarginLayoutParams) view.getLayoutParams()).topMargin = -dp(6); + } + } + + @Override + protected void attachIconContainer(LinearLayout container) { + container.addView(overrideTitleIcon, LayoutHelper.createLinear( + LayoutHelper.MATCH_PARENT, + selectedUsers.size() == 1 ? 94 : 83, + 0, + selectedUsers.size() == 1 ? 28 : 34, + 0, + selectedUsers.size() == 1 ? 9 : 14) + ); + } + + @SuppressLint("ViewConstructor") + static class AvatarHolderView extends FrameLayout { + + public static View createAvatarsContainer(Context context, List selectedUsers) { + FrameLayout avatarsContainer = new FrameLayout(context); + avatarsContainer.setClipChildren(false); + FrameLayout avatarsWrapper = new FrameLayout(context); + avatarsWrapper.setClipChildren(false); + + if (selectedUsers.size() == 1) { + avatarsContainer.addView(avatarsWrapper, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 94, 0, 0, 0, 0, 0)); + PremiumPreviewGiftToUsersBottomSheet.AvatarHolderView avatarHolderView = new PremiumPreviewGiftToUsersBottomSheet.AvatarHolderView(context, 47); + avatarHolderView.drawCycle = false; + avatarHolderView.setUser(selectedUsers.get(0)); + avatarsWrapper.addView(avatarHolderView, 0, LayoutHelper.createFrame(94, 94, Gravity.CENTER)); + } else { + avatarsContainer.addView(avatarsWrapper, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 83, 0, 0, 0, 0, 0)); + int visibleCount = 0; + for (int i = 0; i < selectedUsers.size(); i++) { + TLRPC.User user = selectedUsers.get(i); + PremiumPreviewGiftToUsersBottomSheet.AvatarHolderView avatarHolderView = new PremiumPreviewGiftToUsersBottomSheet.AvatarHolderView(context, 41.5f); + avatarHolderView.setUser(user); + avatarsWrapper.addView(avatarHolderView, 0, LayoutHelper.createFrame(83, 83, Gravity.CENTER)); + avatarHolderView.setTranslationX(-i * dp(29)); + if (i == 0 && selectedUsers.size() > 3) { + avatarHolderView.iconView.setAlpha(1f); + avatarHolderView.iconView.count = selectedUsers.size() - 3; + } + visibleCount++; + if (i == 2) { + break; + } + } + avatarsContainer.setTranslationX(dp(29 / 2f) * (visibleCount - 1)); + } + return avatarsContainer; + } + + private final BackupImageView imageView; + protected final AdditionalCounterView iconView; + private final Paint bgPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + public TLRPC.User user; + public boolean drawCycle = true; + AvatarDrawable fromAvatarDrawable = new AvatarDrawable(); + + public AvatarHolderView(Context context, float radiusDp) { + super(context); + imageView = new BackupImageView(getContext()); + imageView.setRoundRadius(AndroidUtilities.dp(radiusDp)); + iconView = new AdditionalCounterView(context); + iconView.setAlpha(0f); + addView(imageView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT, 0, 5, 5, 5, 5)); + addView(iconView, LayoutHelper.createFrame(26, 26, Gravity.BOTTOM | Gravity.RIGHT, 0, 0, 1, 3)); + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { + bgPaint.setColor(Theme.getColor(Theme.key_windowBackgroundGray)); + } else { + bgPaint.setColor(Theme.getColor(Theme.key_dialogBackground)); + } + } + + public void setUser(TLRPC.User user) { + this.user = user; + fromAvatarDrawable.setInfo(user); + imageView.setForUserOrChat(user, fromAvatarDrawable); + } + + @Override + protected void dispatchDraw(Canvas canvas) { + if (drawCycle) { + canvas.drawCircle(getMeasuredWidth() / 2f, getMeasuredHeight() / 2f, (getMeasuredHeight() / 2f) - dp(2f), bgPaint); + } + super.dispatchDraw(canvas); + } + } + + static class AdditionalCounterView extends View { + + TextPaint paint = new TextPaint(Paint.ANTI_ALIAS_FLAG); + int count; + + public AdditionalCounterView(Context context) { + super(context); + paint.setTextAlign(Paint.Align.CENTER); + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { + paint.setColor(Theme.getColor(Theme.key_windowBackgroundGray)); + } else { + paint.setColor(Theme.getColor(Theme.key_dialogBackground)); + } + paint.setTextSize(dp(11.5f)); + paint.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); + } + + @Override + protected void onDraw(Canvas canvas) { + float cx = getMeasuredWidth() / 2f; + float cy = getMeasuredHeight() / 2f; + canvas.drawCircle(cx, cy, getMeasuredWidth() / 2f, paint); + PremiumGradient.getInstance().updateMainGradientMatrix(0, 0, getMeasuredWidth(), getMeasuredHeight(), -AndroidUtilities.dp(10), 0); + canvas.drawCircle(cx, cy, getMeasuredWidth() / 2f - AndroidUtilities.dp(1.5f), PremiumGradient.getInstance().getMainGradientPaint()); + cy = (int) (cy - ((paint.descent() + paint.ascent()) / 2f)); + canvas.drawText("+" + count, cx, cy, paint); + } + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/ReassignBoostBottomSheet.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/ReassignBoostBottomSheet.java index 5f86010612d..87268ccb816 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/ReassignBoostBottomSheet.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/ReassignBoostBottomSheet.java @@ -1,16 +1,18 @@ package org.telegram.ui.Components.Premium.boosts; +import static org.telegram.messenger.AndroidUtilities.REPLACING_TAG_TYPE_LINKBOLD; import static org.telegram.messenger.AndroidUtilities.dp; +import static org.telegram.messenger.LocaleController.getString; import android.animation.Animator; import android.animation.AnimatorListenerAdapter; import android.content.Context; import android.graphics.Canvas; import android.graphics.Paint; -import android.graphics.RectF; import android.graphics.drawable.Drawable; import android.os.CountDownTimer; import android.text.SpannableStringBuilder; +import android.text.TextUtils; import android.util.TypedValue; import android.view.Gravity; import android.view.View; @@ -26,6 +28,7 @@ import org.telegram.messenger.AndroidUtilities; import org.telegram.messenger.DialogObject; +import org.telegram.messenger.FileLog; import org.telegram.messenger.LocaleController; import org.telegram.messenger.MessagesController; import org.telegram.messenger.NotificationCenter; @@ -34,6 +37,7 @@ import org.telegram.tgnet.TLRPC; import org.telegram.tgnet.tl.TL_stories; import org.telegram.ui.ActionBar.BaseFragment; +import org.telegram.ui.ActionBar.BottomSheet; import org.telegram.ui.ActionBar.Theme; import org.telegram.ui.Cells.HeaderCell; import org.telegram.ui.Cells.ShadowSectionCell; @@ -44,6 +48,7 @@ import org.telegram.ui.Components.BulletinFactory; import org.telegram.ui.Components.CubicBezierInterpolator; import org.telegram.ui.Components.LayoutHelper; +import org.telegram.ui.Components.LinkSpanDrawable; import org.telegram.ui.Components.Premium.PremiumGradient; import org.telegram.ui.Components.Premium.boosts.cells.selector.SelectorBtnCell; import org.telegram.ui.Components.Premium.boosts.cells.selector.SelectorUserCell; @@ -97,32 +102,7 @@ public ReassignBoostBottomSheet(BaseFragment fragment, TL_stories.TL_premium_myB buttonContainer.setOrientation(LinearLayout.VERTICAL); buttonContainer.setPadding(dp(8), dp(8), dp(8), dp(8)); buttonContainer.setBackgroundColor(Theme.getColor(Theme.key_dialogBackground, resourcesProvider)); - actionButton = new ButtonWithCounterView(getContext(), true, resourcesProvider) { - - private final RectF rect = new RectF(); - private boolean incGradient; - private float progress; - - @Override - protected void onDraw(Canvas canvas) { - if (incGradient) { - progress += 16f / 1000f; - if (progress > 3) { - incGradient = false; - } - } else { - progress -= 16f / 1000f; - if (progress < 1) { - incGradient = true; - } - } - rect.set(0, 0, getMeasuredWidth(), getMeasuredHeight()); - PremiumGradient.getInstance().updateMainGradientMatrix(0, 0, getMeasuredWidth(), getMeasuredHeight(), -getMeasuredWidth() * 0.1f * progress, 0); - canvas.drawRoundRect(rect, dp(8), dp(8), PremiumGradient.getInstance().getMainGradientPaint()); - invalidate(); - super.onDraw(canvas); - } - }; + actionButton = new GradientButtonWithCounterView(getContext(), true, resourcesProvider); actionButton.withCounterIcon(); actionButton.setCounterColor(0xFF9874fc); actionButton.setOnClickListener(view -> { @@ -295,7 +275,7 @@ public void onBindViewHolder(@NonNull RecyclerView.ViewHolder holder, int positi cell.setText(LocaleController.getString("BoostingRemoveBoostFrom", R.string.BoostingRemoveBoostFrom)); } else if (holder.getItemViewType() == HOLDER_TYPE_HEADER) { topCell = (TopCell) holder.itemView; - topCell.setData(currentChat); + topCell.setData(currentChat, ReassignBoostBottomSheet.this); } } @@ -356,7 +336,7 @@ public TopCell(Context context) { title.setTextColor(Theme.getColor(Theme.key_windowBackgroundWhiteBlackText)); addView(title, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_HORIZONTAL, 0, 15, 0, 7)); - description = new TextView(context); + description = new LinkSpanDrawable.LinksTextView(getContext()); description.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 14); description.setGravity(Gravity.CENTER_HORIZONTAL); description.setTextColor(Theme.getColor(Theme.key_dialogTextBlack)); @@ -365,8 +345,34 @@ public TopCell(Context context) { addView(description, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_HORIZONTAL, 28, 0, 28, 18)); } - public void setData(TLRPC.Chat chat) { - description.setText(AndroidUtilities.replaceTags(LocaleController.formatPluralString("BoostingReassignBoostTextPlural", BoostRepository.boostsPerSentGift(), chat == null ? "" : chat.title))); + public void setData(TLRPC.Chat chat, BottomSheet bottomSheet) { + try { + String replacer = "%3$s"; + SpannableStringBuilder text = AndroidUtilities.replaceTags(LocaleController.formatPluralString("BoostingReassignBoostTextPluralWithLink", BoostRepository.boostsPerSentGift(), chat == null ? "" : chat.title, replacer)); + SpannableStringBuilder link = AndroidUtilities.replaceSingleTag( + getString("BoostingReassignBoostTextLink", R.string.BoostingReassignBoostTextLink), + Theme.key_chat_messageLinkIn, REPLACING_TAG_TYPE_LINKBOLD, + () -> { + bottomSheet.dismiss(); + NotificationCenter.getInstance(UserConfig.selectedAccount).postNotificationName(NotificationCenter.didStartedMultiGiftsSelector); + AndroidUtilities.runOnUIThread(UserSelectorBottomSheet::open, 220); + }); + int indexOfReplacer = TextUtils.indexOf(text, replacer); + text.replace(indexOfReplacer, indexOfReplacer + replacer.length(), link); + description.setText(text, TextView.BufferType.EDITABLE); + description.post(() -> { + try { + int linkLine = description.getLayout().getLineForOffset(indexOfReplacer); + if (linkLine == 0) { + description.getEditableText().insert(indexOfReplacer, "\n"); + } + } catch (Exception e) { + FileLog.e(e); + } + }); + } catch (Exception e) { + FileLog.e(e); + } } public void showBoosts(List selectedBoosts, TLRPC.Chat currentChat) { @@ -409,7 +415,7 @@ public void showChats(List selectedChats, TLRPC.Chat currentChat) { avatar.setChat(chat); int childCount = allViews.size(); avatarsWrapper.addView(avatar, 0, LayoutHelper.createFrame(70, 70, Gravity.CENTER)); - avatar.setTranslationX(-childCount * (52 + 18)); + avatar.setTranslationX(-childCount * dp(23)); avatar.setAlpha(0f); avatar.setScaleX(0.1f); avatar.setScaleY(0.1f); @@ -435,7 +441,7 @@ public void showChats(List selectedChats, TLRPC.Chat currentChat) { final AvatarHolderView finalRemovedAvatar = removedAvatar; finalRemovedAvatar.setTag("REMOVED"); finalRemovedAvatar.animate() - .alpha(0f).translationXBy((52 + 18)) + .alpha(0f).translationXBy(dp(23)) .scaleX(0.1f).scaleY(0.1f) .setInterpolator(interpolator) .setDuration(duration).setListener(new AnimatorListenerAdapter() { @@ -451,7 +457,7 @@ public void onAnimationEnd(Animator animation) { if (view != finalRemovedAvatar) { pos++; childCount -= pos; - view.animate().translationX(-childCount * (52 + 18)) + view.animate().translationX(-childCount * dp(23)) .setInterpolator(interpolator) .setDuration(duration).start(); } @@ -474,7 +480,7 @@ public void onAnimationEnd(Animator animation) { avatarsContainer.animate().setInterpolator(interpolator).translationX(0).setDuration(duration).start(); } else { int count = addedChats.size() - 1; - avatarsContainer.animate().setInterpolator(interpolator).translationX(dp(13) * count).setDuration(duration).start(); + avatarsContainer.animate().setInterpolator(interpolator).translationX(dp(23 / 2f) * count).setDuration(duration).start(); } toAvatar.animate().cancel(); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/SelectorBottomSheet.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/SelectorBottomSheet.java index 83d6a4e3da8..103ad2ad549 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/SelectorBottomSheet.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/SelectorBottomSheet.java @@ -615,7 +615,7 @@ public void updateItems(boolean animated, boolean notify) { if (!countryItems.isEmpty()) { h += dp(32); - items.add(Item.asLetter(countriesLetter)); + items.add(Item.asLetter(countriesLetter.toUpperCase())); items.addAll(countryItems); } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/UserSelectorBottomSheet.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/UserSelectorBottomSheet.java new file mode 100644 index 00000000000..aa55dd99a84 --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/UserSelectorBottomSheet.java @@ -0,0 +1,593 @@ +package org.telegram.ui.Components.Premium.boosts; + +import static org.telegram.messenger.AndroidUtilities.dp; + +import android.annotation.SuppressLint; +import android.content.Intent; +import android.content.res.Configuration; +import android.graphics.Canvas; +import android.graphics.Paint; +import android.graphics.Rect; +import android.net.Uri; +import android.text.SpannableStringBuilder; +import android.text.Spanned; +import android.text.TextUtils; +import android.text.style.ReplacementSpan; +import android.view.Gravity; +import android.view.HapticFeedbackConstants; +import android.view.View; +import android.widget.LinearLayout; + +import androidx.annotation.NonNull; +import androidx.annotation.Nullable; +import androidx.recyclerview.widget.DefaultItemAnimator; +import androidx.recyclerview.widget.LinearSmoothScrollerCustom; +import androidx.recyclerview.widget.RecyclerView; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.ContactsController; +import org.telegram.messenger.LocaleController; +import org.telegram.messenger.MediaDataController; +import org.telegram.messenger.MessagesController; +import org.telegram.messenger.NotificationCenter; +import org.telegram.messenger.R; +import org.telegram.messenger.UserConfig; +import org.telegram.messenger.UserObject; +import org.telegram.messenger.browser.Browser; +import org.telegram.tgnet.TLRPC; +import org.telegram.ui.ActionBar.BaseFragment; +import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.Components.BottomSheetWithRecyclerListView; +import org.telegram.ui.Components.BulletinFactory; +import org.telegram.ui.Components.CubicBezierInterpolator; +import org.telegram.ui.Components.LayoutHelper; +import org.telegram.ui.Components.Premium.boosts.adapters.SelectorAdapter; +import org.telegram.ui.Components.Premium.boosts.adapters.SelectorAdapter.Item; +import org.telegram.ui.Components.Premium.boosts.cells.selector.SelectorBtnCell; +import org.telegram.ui.Components.Premium.boosts.cells.selector.SelectorHeaderCell; +import org.telegram.ui.Components.Premium.boosts.cells.selector.SelectorSearchCell; +import org.telegram.ui.Components.Premium.boosts.cells.selector.SelectorUserCell; +import org.telegram.ui.Components.RecyclerListView; +import org.telegram.ui.LaunchActivity; +import org.telegram.ui.Stories.recorder.ButtonWithCounterView; + +import java.util.ArrayList; +import java.util.HashMap; +import java.util.HashSet; +import java.util.LinkedHashMap; +import java.util.List; +import java.util.Map; + +public class UserSelectorBottomSheet extends BottomSheetWithRecyclerListView implements NotificationCenter.NotificationCenterDelegate { + private static UserSelectorBottomSheet instance; + + public static void open() { + BaseFragment fragment = LaunchActivity.getLastFragment(); + if (fragment == null) { + return; + } + if (instance != null) { + return; + } + UserSelectorBottomSheet sheet = new UserSelectorBottomSheet(fragment, true); + sheet.show(); + instance = sheet; + } + + public static boolean handleIntent(Intent intent, Browser.Progress progress) { + Uri data = intent.getData(); + if (data != null) { + String scheme = data.getScheme(); + if (scheme != null) { + if ((scheme.equals("http") || scheme.equals("https"))) { + String host = data.getHost().toLowerCase(); + if (host.equals("telegram.me") || host.equals("t.me") || host.equals("telegram.dog")) { + String path = data.getPath(); + if (path != null) { + if (path.startsWith("/premium_multigift")) { + open(); + return true; + } + } + } + } else if (scheme.equals("tg")) { + String url = data.toString(); + if (url.startsWith("tg:premium_multigift") || url.startsWith("tg://premium_multigift")) { + open(); + return true; + } + } + } + } + return false; + } + + private static final int BOTTOM_HEIGHT_DP = 60; + + private final ButtonWithCounterView actionButton; + private final SelectorSearchCell searchField; + private final View sectionCell; + private final SelectorHeaderCell headerView; + private final SelectorBtnCell buttonContainer; + + private final ArrayList oldItems = new ArrayList<>(); + private final ArrayList items = new ArrayList<>(); + private final HashSet selectedIds = new HashSet<>(); + private final List contacts = new ArrayList<>(); + private final List hints = new ArrayList<>(); + private final List foundedUsers = new ArrayList<>(); + private final Map> contactsMap = new HashMap<>(); + private final List contactsLetters = new ArrayList<>(); + private final HashMap allSelectedObjects = new LinkedHashMap<>(); + private String query; + private SelectorAdapter selectorAdapter; + private int listPaddingTop = AndroidUtilities.dp(56 + 64); + private final List paymentOptions = new ArrayList<>(); + private boolean isHintSearchText = false; + private int lastRequestId; + private float recipientsBtnExtraSpace; + private ReplacementSpan recipientsBtnSpaceSpan; + + private final Runnable remoteSearchRunnable = new Runnable() { + @Override + public void run() { + final String finalQuery = query; + if (finalQuery != null) { + loadData(finalQuery); + } + } + }; + + private void loadData(String query) { + lastRequestId = BoostRepository.searchContacts(lastRequestId, query, arg -> { + foundedUsers.clear(); + foundedUsers.addAll(arg); + updateList(true, true); + }); + } + + private void checkEditTextHint() { + if (selectedIds.size() > 0) { + if (!isHintSearchText) { + isHintSearchText = true; + AndroidUtilities.runOnUIThread(() -> searchField.setHintText(LocaleController.getString("Search", R.string.Search), true), 10); + } + } else { + if (isHintSearchText) { + isHintSearchText = false; + AndroidUtilities.runOnUIThread(() -> searchField.setHintText(LocaleController.getString("GiftPremiumUsersSearchHint", R.string.GiftPremiumUsersSearchHint), true), 10); + } + } + } + + private void createRecipientsBtnSpaceSpan() { + recipientsBtnSpaceSpan = new ReplacementSpan() { + @Override + public int getSize(@NonNull Paint paint, CharSequence text, int start, int end, @Nullable Paint.FontMetricsInt fm) { + return (int) recipientsBtnExtraSpace; + } + + @Override + public void draw(@NonNull Canvas canvas, CharSequence text, int start, int end, float x, int top, int y, int bottom, @NonNull Paint paint) { + + } + }; + } + + public UserSelectorBottomSheet(BaseFragment fragment, boolean needFocus) { + super(fragment, needFocus, false, false, fragment.getResourceProvider()); + + headerView = new SelectorHeaderCell(getContext(), resourcesProvider) { + @Override + protected int getHeaderHeight() { + if (getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE) { + return dp(48); + } else { + return dp(54); + } + } + }; + headerView.setOnCloseClickListener(this::dismiss); + headerView.setText(getTitle()); + headerView.setCloseImageVisible(false); + headerView.backDrawable.setRotation(0f, false); + + createRecipientsBtnSpaceSpan(); + + searchField = new SelectorSearchCell(getContext(), resourcesProvider, null) { + private boolean isKeyboardVisible; + + @Override + protected void onLayout(boolean changed, int l, int t, int r, int b) { + super.onLayout(changed, l, t, r, b); + listPaddingTop = getMeasuredHeight() + dp(64); + selectorAdapter.notifyChangedLast(); + if (isKeyboardVisible != isKeyboardVisible()) { + isKeyboardVisible = isKeyboardVisible(); + if (isKeyboardVisible) { + scrollToTop(true); + } + } + } + }; + searchField.setBackgroundColor(getThemedColor(Theme.key_dialogBackground)); + searchField.setOnSearchTextChange(this::onSearch); + searchField.setHintText(LocaleController.getString("GiftPremiumUsersSearchHint", R.string.GiftPremiumUsersSearchHint), false); + + sectionCell = new View(getContext()) { + @Override + protected void onDraw(Canvas canvas) { + canvas.drawColor(getThemedColor(Theme.key_graySection)); + } + }; + + containerView.addView(headerView, 0, LayoutHelper.createFrameMarginPx(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.TOP | Gravity.FILL_HORIZONTAL, backgroundPaddingLeft, 0, backgroundPaddingLeft, 0)); + containerView.addView(searchField, LayoutHelper.createFrameMarginPx(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.TOP | Gravity.FILL_HORIZONTAL, backgroundPaddingLeft, 0, backgroundPaddingLeft, 0)); + containerView.addView(sectionCell, LayoutHelper.createFrameMarginPx(LayoutHelper.MATCH_PARENT, 1, Gravity.TOP | Gravity.FILL_HORIZONTAL, backgroundPaddingLeft, 0, backgroundPaddingLeft, 0)); + + buttonContainer = new SelectorBtnCell(getContext(), resourcesProvider, null); + buttonContainer.setClickable(true); + buttonContainer.setOrientation(LinearLayout.VERTICAL); + buttonContainer.setPadding(dp(10), dp(10), dp(10), dp(10)); + buttonContainer.setBackgroundColor(Theme.getColor(Theme.key_dialogBackground, resourcesProvider)); + actionButton = new ButtonWithCounterView(getContext(), resourcesProvider) { + @Override + protected float calculateCounterWidth(float width, float percent) { + boolean needUpdateActionBtn = recipientsBtnExtraSpace == 0; + recipientsBtnExtraSpace = width; + if (needUpdateActionBtn) { + createRecipientsBtnSpaceSpan(); + updateActionButton(false); + } + return width; + } + }; + actionButton.setOnClickListener(v -> next()); + buttonContainer.addView(actionButton, LayoutHelper.createLinear(LayoutHelper.MATCH_PARENT, 48, Gravity.BOTTOM | Gravity.FILL_HORIZONTAL)); + containerView.addView(buttonContainer, LayoutHelper.createFrameMarginPx(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.BOTTOM | Gravity.FILL_HORIZONTAL, backgroundPaddingLeft, 0, backgroundPaddingLeft, 0)); + + selectorAdapter.setData(items, recyclerListView); + recyclerListView.setPadding(backgroundPaddingLeft, 0, backgroundPaddingLeft, dp(BOTTOM_HEIGHT_DP)); + recyclerListView.addOnScrollListener(new RecyclerView.OnScrollListener() { + @Override + public void onScrollStateChanged(@NonNull RecyclerView recyclerView, int newState) { + if (newState == RecyclerView.SCROLL_STATE_DRAGGING) { + AndroidUtilities.hideKeyboard(searchField.getEditText()); + } + } + }); + recyclerListView.setOnItemClickListener((view, position, x, y) -> { + if (view instanceof SelectorUserCell) { + TLRPC.User user = ((SelectorUserCell) view).getUser(); + long id = user.id; + if (selectedIds.contains(id)) { + selectedIds.remove(id); + } else { + selectedIds.add(id); + allSelectedObjects.put(id, user); + } + if (selectedIds.size() == 11) { + selectedIds.remove(id); + showMaximumUsersToast(); + return; + } + checkEditTextHint(); + searchField.updateSpans(true, selectedIds, () -> { + checkEditTextHint(); + updateList(true, false); + }, null); + updateList(true, false); + clearSearchAfterSelect(); + } + }); + DefaultItemAnimator itemAnimator = new DefaultItemAnimator(); + itemAnimator.setDurations(350); + itemAnimator.setInterpolator(CubicBezierInterpolator.EASE_OUT_QUINT); + itemAnimator.setDelayAnimations(false); + itemAnimator.setSupportsChangeAnimations(false); + recyclerListView.setItemAnimator(itemAnimator); + recyclerListView.addItemDecoration(new RecyclerView.ItemDecoration() { + @Override + public void getItemOffsets(@NonNull Rect outRect, @NonNull View view, @NonNull RecyclerView parent, @NonNull RecyclerView.State state) { + super.getItemOffsets(outRect, view, parent, state); + int position = parent.getChildAdapterPosition(view); + if (position == items.size()) { + outRect.bottom = listPaddingTop; + } + } + }); + + searchField.setText(""); + searchField.spansContainer.removeAllSpans(false); + searchField.updateSpans(false, selectedIds, () -> { + checkEditTextHint(); + updateList(true, false); + }, null); + headerView.setText(getTitle()); + updateActionButton(false); + initContacts(false); + initHints(false); + updateList(false, true); + fixNavigationBar(); + BoostRepository.loadGiftOptions(null, arg -> { + paymentOptions.clear(); + paymentOptions.addAll(arg); + }); + } + + private void initContacts(boolean needUpdate) { + if (contacts.isEmpty()) { + contacts.addAll(ContactsController.getInstance(currentAccount).contacts); + contactsMap.putAll(ContactsController.getInstance(currentAccount).usersSectionsDict); + contactsLetters.addAll(ContactsController.getInstance(currentAccount).sortedUsersSectionsArray); + if (needUpdate) { + updateItems(true, true); + } + } + } + + private void initHints(boolean needUpdate) { + if (hints.isEmpty()) { + hints.addAll(MediaDataController.getInstance(currentAccount).hints); + if (needUpdate) { + updateItems(true, true); + } + } + } + + @Override + protected void onPreDraw(Canvas canvas, int top, float progressToFullView) { + float minTop = AndroidUtilities.statusBarHeight + (headerView.getMeasuredHeight() - AndroidUtilities.statusBarHeight - AndroidUtilities.dp(40)) / 2f; + float fromY = Math.max(top, minTop) + AndroidUtilities.dp(8); + headerView.setTranslationY(fromY); + searchField.setTranslationY(headerView.getTranslationY() + headerView.getMeasuredHeight()); + sectionCell.setTranslationY(searchField.getTranslationY() + searchField.getMeasuredHeight()); + recyclerListView.setTranslationY(headerView.getMeasuredHeight() + searchField.getMeasuredHeight() + sectionCell.getMeasuredHeight() - AndroidUtilities.dp(8)); + } + + private void next() { + if (selectedIds.size() == 0 || paymentOptions.isEmpty()) { + return; + } + List selectedUsers = new ArrayList<>(); + for (TLRPC.User object : allSelectedObjects.values()) { + if (selectedIds.contains(object.id)) { + selectedUsers.add(object); + } + } + AndroidUtilities.hideKeyboard(searchField.getEditText()); + List options = BoostRepository.filterGiftOptions(paymentOptions, selectedUsers.size()); + options = BoostRepository.filterGiftOptionsByBilling(options); + PremiumPreviewGiftToUsersBottomSheet.show(selectedUsers, options); + } + + public void scrollToTop(boolean animate) { + if (animate) { + LinearSmoothScrollerCustom linearSmoothScroller = new LinearSmoothScrollerCustom(getContext(), LinearSmoothScrollerCustom.POSITION_TOP, .6f); + linearSmoothScroller.setTargetPosition(1); + linearSmoothScroller.setOffset(AndroidUtilities.dp(36)); + recyclerListView.getLayoutManager().startSmoothScroll(linearSmoothScroller); + } else { + recyclerListView.scrollToPosition(0); + } + } + + @Override + public void onAttachedToWindow() { + super.onAttachedToWindow(); + NotificationCenter.getInstance(currentAccount).addObserver(this, NotificationCenter.giftsToUserSent); + NotificationCenter.getInstance(currentAccount).addObserver(this, NotificationCenter.contactsDidLoad); + NotificationCenter.getInstance(currentAccount).addObserver(this, NotificationCenter.reloadHints); + } + + @Override + public void onDetachedFromWindow() { + super.onDetachedFromWindow(); + NotificationCenter.getInstance(currentAccount).removeObserver(this, NotificationCenter.giftsToUserSent); + NotificationCenter.getInstance(currentAccount).removeObserver(this, NotificationCenter.contactsDidLoad); + NotificationCenter.getInstance(currentAccount).removeObserver(this, NotificationCenter.reloadHints); + } + + @Override + public void dismissInternal() { + super.dismissInternal(); + instance = null; + AndroidUtilities.cancelRunOnUIThread(remoteSearchRunnable); + } + + private void showMaximumUsersToast() { + String text = LocaleController.getString("BoostingSelectUpToWarningUsers", R.string.BoostingSelectUpToWarningUsers); + BulletinFactory.of(container, resourcesProvider).createSimpleBulletin(R.raw.chats_infotip, text).show(true); + try { + container.performHapticFeedback(HapticFeedbackConstants.KEYBOARD_TAP, HapticFeedbackConstants.FLAG_IGNORE_GLOBAL_SETTING); + } catch (Exception ignore) { + } + } + + private void updateList(boolean animated, boolean notify) { + updateItems(animated, notify); + updateCheckboxes(animated); + updateActionButton(animated); + } + + private void updateCheckboxes(boolean animated) { + int visibleItemsFrom = -1; + int visibleItemsTo = 0; + for (int i = 0; i < recyclerListView.getChildCount(); ++i) { + View child = recyclerListView.getChildAt(i); + if (child instanceof SelectorUserCell) { + int position = recyclerListView.getChildAdapterPosition(child); + if (position <= 0) { + continue; + } + if (visibleItemsFrom == -1) { + visibleItemsFrom = position; + } + visibleItemsTo = position; + Item item = items.get(position - 1); + SelectorUserCell cell = (SelectorUserCell) child; + cell.setChecked(item.checked, animated); + if (item.chat != null) { + cell.setCheckboxAlpha(selectorAdapter.getParticipantsCount(item.chat) > 200 ? .3f : 1f, animated); + } else { + cell.setCheckboxAlpha(1f, animated); + } + } + } + if (animated) { + selectorAdapter.notifyItemRangeChanged(0, visibleItemsFrom); + selectorAdapter.notifyItemRangeChanged(visibleItemsTo, selectorAdapter.getItemCount() - visibleItemsTo); + } + } + + private void updateActionButton(boolean animated) { + actionButton.setShowZero(false); + SpannableStringBuilder stringBuilder = new SpannableStringBuilder(); + if (selectedIds.size() == 0) { + stringBuilder.append("d").setSpan(recipientsBtnSpaceSpan, 0, 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + stringBuilder.append(LocaleController.getString("GiftPremiumChooseRecipientsBtn", R.string.GiftPremiumChooseRecipientsBtn)); + } else { + stringBuilder.append(LocaleController.getString("GiftPremiumProceedBtn", R.string.GiftPremiumProceedBtn)); + } + actionButton.setCount(selectedIds.size(), true); + actionButton.setText(stringBuilder, animated, false); + actionButton.setEnabled(true); + } + + private void onSearch(String text) { + this.query = text; + AndroidUtilities.cancelRunOnUIThread(remoteSearchRunnable); + AndroidUtilities.runOnUIThread(remoteSearchRunnable, 350); + } + + private void clearSearchAfterSelect() { + if (isSearching()) { + query = null; + searchField.setText(""); + AndroidUtilities.cancelRunOnUIThread(remoteSearchRunnable); + updateItems(true, true); + } + } + + private void updateSectionCell(boolean animated) { + if (selectedIds == null) { + return; + } + if (selectedIds.size() > 0) { + selectorAdapter.setTopSectionClickListener(v -> { + selectedIds.clear(); + searchField.spansContainer.removeAllSpans(true); + checkEditTextHint(); + updateList(true, false); + }); + } else { + selectorAdapter.setTopSectionClickListener(null); + } + } + + private boolean isSearching() { + return !TextUtils.isEmpty(query); + } + + @SuppressLint("NotifyDataSetChanged") + public void updateItems(boolean animated, boolean notify) { + oldItems.clear(); + oldItems.addAll(items); + items.clear(); + + int h = 0; + if (isSearching()) { + for (TLRPC.User foundedUser : foundedUsers) { + h += dp(56); + items.add(Item.asUser(foundedUser, selectedIds.contains(foundedUser.id))); + } + } else { + if (!hints.isEmpty()) { + List userItems = new ArrayList<>(); + for (TLRPC.TL_topPeer hint : hints) { + TLRPC.User user = MessagesController.getInstance(currentAccount).getUser(hint.peer.user_id); + if (user.self || user.bot || UserObject.isService(user.id) || UserObject.isDeleted(user)) { + continue; + } + h += dp(56); + userItems.add(Item.asUser(user, selectedIds.contains(user.id))); + } + if (!userItems.isEmpty()) { + h += dp(32); + items.add(Item.asTopSection(LocaleController.getString("GiftPremiumFrequentContacts", R.string.GiftPremiumFrequentContacts))); + items.addAll(userItems); + } + } + for (String contactLetter : contactsLetters) { + List userItems = new ArrayList<>(); + for (TLRPC.TL_contact contact : contactsMap.get(contactLetter)) { + long myUid = UserConfig.getInstance(currentAccount).getClientUserId(); + if (contact.user_id == myUid) { + continue; + } + h += dp(56); + TLRPC.User user = MessagesController.getInstance(currentAccount).getUser(contact.user_id); + userItems.add(Item.asUser(user, selectedIds.contains(user.id))); + } + + if (!userItems.isEmpty()) { + h += dp(32); + items.add(Item.asLetter(contactLetter.toUpperCase())); + items.addAll(userItems); + } + } + } + + if (items.isEmpty()) { + items.add(Item.asNoUsers()); + h += dp(150); + } + int minHeight = (int) (AndroidUtilities.displaySize.y * 0.6f); + items.add(Item.asPad(Math.max(0, minHeight - h))); + + updateSectionCell(animated); + + if (notify && selectorAdapter != null) { + if (animated) { + selectorAdapter.setItems(oldItems, items); + } else { + selectorAdapter.notifyDataSetChanged(); + } + } + } + + @Override + public void onConfigurationChanged(Configuration newConfig) { + super.onConfigurationChanged(newConfig); + updateItems(false, true); + } + + @Override + protected CharSequence getTitle() { + return LocaleController.getString("GiftTelegramPremiumTitle", R.string.GiftTelegramPremiumTitle); + } + + @Override + protected RecyclerListView.SelectionAdapter createAdapter() { + selectorAdapter = new SelectorAdapter(getContext(), resourcesProvider); + selectorAdapter.setGreenSelector(true); + return selectorAdapter; + } + + @Override + public void dismiss() { + AndroidUtilities.hideKeyboard(searchField.getEditText()); + super.dismiss(); + } + + @Override + public void didReceivedNotification(int id, int account, Object... args) { + if (id == NotificationCenter.giftsToUserSent) { + dismiss(); + } else if (id == NotificationCenter.contactsDidLoad) { + AndroidUtilities.runOnUIThread(() -> initContacts(true)); + } else if (id == NotificationCenter.reloadHints) { + AndroidUtilities.runOnUIThread(() -> initHints(true)); + } + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/adapters/BoostAdapter.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/adapters/BoostAdapter.java index 2e4138e887b..0fddfb8dcd9 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/adapters/BoostAdapter.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/adapters/BoostAdapter.java @@ -23,11 +23,13 @@ import org.telegram.ui.Components.Premium.boosts.cells.BoostTypeSingleCell; import org.telegram.ui.Components.Premium.boosts.cells.ChatCell; import org.telegram.ui.Components.Premium.boosts.cells.DateEndCell; +import org.telegram.ui.Components.Premium.boosts.cells.EnterPrizeCell; import org.telegram.ui.Components.Premium.boosts.cells.HeaderCell; import org.telegram.ui.Components.Premium.boosts.cells.ParticipantsTypeCell; import org.telegram.ui.Components.Premium.boosts.cells.DurationCell; import org.telegram.ui.Components.Premium.boosts.cells.SliderCell; import org.telegram.ui.Components.Premium.boosts.cells.SubtitleWithCounterCell; +import org.telegram.ui.Components.Premium.boosts.cells.SwitcherCell; import org.telegram.ui.Components.Premium.boosts.cells.TextInfoCell; import org.telegram.ui.Components.RecyclerListView; import org.telegram.ui.Components.SlideChooseView; @@ -51,7 +53,9 @@ public class BoostAdapter extends AdapterWithDiffUtils { HOLDER_TYPE_PARTICIPANTS = 11, HOLDER_TYPE_DURATION = 12, HOLDER_TYPE_SUBTITLE_WITH_COUNTER = 13, - HOLDER_TYPE_SINGLE_BOOST_TYPE = 14; + HOLDER_TYPE_SINGLE_BOOST_TYPE = 14, + HOLDER_TYPE_SWITCHER = 15, + HOLDER_TYPE_ENTER_PRIZE = 16; private final Theme.ResourcesProvider resourcesProvider; private List items = new ArrayList<>(); @@ -59,16 +63,18 @@ public class BoostAdapter extends AdapterWithDiffUtils { private SlideChooseView.Callback sliderCallback; private ChatCell.ChatDeleteListener chatDeleteListener; private HeaderCell headerCell; + private EnterPrizeCell.AfterTextChangedListener afterTextChangedListener; public BoostAdapter(Theme.ResourcesProvider resourcesProvider) { this.resourcesProvider = resourcesProvider; } - public void setItems(List items, RecyclerListView recyclerListView, SlideChooseView.Callback sliderCallback, ChatCell.ChatDeleteListener chatDeleteListener) { + public void setItems(List items, RecyclerListView recyclerListView, SlideChooseView.Callback sliderCallback, ChatCell.ChatDeleteListener chatDeleteListener, EnterPrizeCell.AfterTextChangedListener afterTextChangedListener) { this.items = items; this.recyclerListView = recyclerListView; this.sliderCallback = sliderCallback; this.chatDeleteListener = chatDeleteListener; + this.afterTextChangedListener = afterTextChangedListener; } public void updateBoostCounter(int value) { @@ -81,13 +87,30 @@ public void updateBoostCounter(int value) { ((ChatCell) child).setCounter(value); } } - notifyItemChanged(8); - //updates all prices - notifyItemChanged(items.size() - 1); - notifyItemChanged(items.size() - 2); - notifyItemChanged(items.size() - 3); - notifyItemChanged(items.size() - 4); - notifyItemChanged(items.size() - 6); + notifyItemChanged(8); //update main channel + notifyItemRangeChanged(items.size() - 12, 12); //updates all prices + } + + public void notifyAllVisibleTextDividers() { + for (int i = 0; i < items.size(); i++) { + if (items.get(i).viewType == HOLDER_TYPE_TEXT_DIVIDER) { + notifyItemChanged(i); + } + } + } + + public void notifyAdditionalPrizeItem(boolean checked) { + for (int i = 0; i < items.size(); i++) { + Item item = items.get(i); + if (item.viewType == HOLDER_TYPE_SWITCHER && item.subType == SwitcherCell.TYPE_ADDITION_PRIZE) { + if (checked) { + notifyItemInserted(i + 1); + } else { + notifyItemRemoved(i + 1); + } + break; + } + } } public void setPausedStars(boolean paused) { @@ -158,6 +181,7 @@ public boolean isEnabled(RecyclerView.ViewHolder holder) { || itemViewType == HOLDER_TYPE_PARTICIPANTS || itemViewType == HOLDER_TYPE_ADD_CHANNEL || itemViewType == HOLDER_TYPE_DATE_END + || itemViewType == HOLDER_TYPE_SWITCHER || itemViewType == HOLDER_TYPE_DURATION; } @@ -177,6 +201,14 @@ public RecyclerView.ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int case HOLDER_TYPE_SINGLE_BOOST_TYPE: view = new BoostTypeSingleCell(context, resourcesProvider); break; + case HOLDER_TYPE_ENTER_PRIZE: + view = new EnterPrizeCell(context, resourcesProvider); + break; + case HOLDER_TYPE_SWITCHER: + SwitcherCell cell = new SwitcherCell(context, resourcesProvider); + cell.setHeight(50); + view = cell; + break; case HOLDER_TYPE_EMPTY: view = new View(context); break; @@ -293,6 +325,17 @@ public void onBindViewHolder(@NonNull RecyclerView.ViewHolder holder, int positi case HOLDER_TYPE_SIMPLE_DIVIDER: { break; } + case HOLDER_TYPE_ENTER_PRIZE: { + EnterPrizeCell cell = (EnterPrizeCell) holder.itemView; + cell.setCount(item.intValue); + cell.setAfterTextChangedListener(afterTextChangedListener); + break; + } + case HOLDER_TYPE_SWITCHER: { + SwitcherCell cell = (SwitcherCell) holder.itemView; + cell.setData(item.text, item.selectable, item.boolValue, item.subType); + break; + } } } @@ -358,6 +401,20 @@ public static Item asPeer(TLRPC.InputPeer peer, boolean removable, int count) { return item; } + public static Item asEnterPrize(int count) { + Item item = new Item(HOLDER_TYPE_ENTER_PRIZE, false); + item.intValue = count; + return item; + } + + public static Item asSwitcher(CharSequence text, boolean isSelected, boolean needDivider, int subType) { + Item item = new Item(HOLDER_TYPE_SWITCHER, isSelected); + item.text = text; + item.boolValue = needDivider; + item.subType = subType; + return item; + } + public static Item asSingleBoost(Object user) { Item item = new Item(HOLDER_TYPE_SINGLE_BOOST_TYPE, false); item.user = user; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/adapters/GiftInfoAdapter.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/adapters/GiftInfoAdapter.java index cf28bbc47b3..081079743ae 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/adapters/GiftInfoAdapter.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/adapters/GiftInfoAdapter.java @@ -9,6 +9,7 @@ import android.view.Gravity; import android.view.View; import android.view.ViewGroup; +import android.widget.FrameLayout; import androidx.annotation.NonNull; import androidx.recyclerview.widget.RecyclerView; @@ -48,16 +49,18 @@ public abstract class GiftInfoAdapter extends RecyclerListView.SelectionAdapter private BaseFragment baseFragment; private TLRPC.TL_payments_checkedGiftCode giftCode; private String slug; + private FrameLayout container; public GiftInfoAdapter(Theme.ResourcesProvider resourcesProvider) { this.resourcesProvider = resourcesProvider; } - public void init(BaseFragment baseFragment, TLRPC.TL_payments_checkedGiftCode giftCode, String slug) { + public void init(BaseFragment baseFragment, TLRPC.TL_payments_checkedGiftCode giftCode, String slug, FrameLayout container) { this.isUnused = giftCode.used_date == 0; this.baseFragment = baseFragment; this.giftCode = giftCode; this.slug = slug; + this.container = container; } @Override @@ -111,7 +114,7 @@ public RecyclerView.ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int break; case HOLDER_TYPE_BUTTON: view = new ActionBtnCell(context, resourcesProvider); - view.setPadding(0,0,0, AndroidUtilities.dp(14)); + view.setPadding(0, 0, 0, AndroidUtilities.dp(14)); break; case HOLDER_TYPE_EMPTY: view = new View(context); @@ -181,25 +184,7 @@ public void onBindViewHolder(@NonNull RecyclerView.ViewHolder holder, int positi LocaleController.getString("BoostingSendLinkToAnyone", R.string.BoostingSendLinkToAnyone) : LocaleController.getString("BoostingSendLinkToFriends", R.string.BoostingSendLinkToFriends), Theme.key_chat_messageLinkIn, 0, - () -> { - final String slugLink = "https://t.me/giftcode/" + slug; - Bundle args = new Bundle(); - args.putBoolean("onlySelect", true); - args.putInt("dialogsType", DialogsActivity.DIALOGS_TYPE_FORWARD); - DialogsActivity dialogFragment = new DialogsActivity(args); - dialogFragment.setDelegate((fragment1, dids, message, param, topicsFragment) -> { - long did = 0; - for (int a = 0; a < dids.size(); a++) { - did = dids.get(a).dialogId; - baseFragment.getSendMessagesHelper().sendMessage(SendMessagesHelper.SendMessageParams.of(slugLink, did, null, null, null, true, null, null, null, true, 0, null, false)); - } - fragment1.finishFragment(); - BoostDialogs.showGiftLinkForwardedBulletin(did); - return true; - }); - baseFragment.presentFragment(dialogFragment); - dismiss(); - }, + this::share, resourcesProvider ); cell.setText(text); @@ -227,7 +212,7 @@ public void onBindViewHolder(@NonNull RecyclerView.ViewHolder holder, int positi dismiss(); }, error -> { cell.updateLoading(false); - BoostDialogs.showToastError(baseFragment.getContext(), error); + BoostDialogs.processApplyGiftCodeError(error, container, resourcesProvider, this::share); }); } else { dismiss(); @@ -245,6 +230,26 @@ public void onBindViewHolder(@NonNull RecyclerView.ViewHolder holder, int positi } } + private void share() { + final String slugLink = "https://t.me/giftcode/" + slug; + Bundle args = new Bundle(); + args.putBoolean("onlySelect", true); + args.putInt("dialogsType", DialogsActivity.DIALOGS_TYPE_FORWARD); + DialogsActivity dialogFragment = new DialogsActivity(args); + dialogFragment.setDelegate((fragment1, dids, message, param, topicsFragment) -> { + long did = 0; + for (int a = 0; a < dids.size(); a++) { + did = dids.get(a).dialogId; + baseFragment.getSendMessagesHelper().sendMessage(SendMessagesHelper.SendMessageParams.of(slugLink, did, null, null, null, true, null, null, null, true, 0, null, false)); + } + fragment1.finishFragment(); + BoostDialogs.showGiftLinkForwardedBulletin(did); + return true; + }); + baseFragment.presentFragment(dialogFragment); + dismiss(); + } + @Override public int getItemCount() { return 5; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/adapters/SelectorAdapter.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/adapters/SelectorAdapter.java index 786473589e0..952ab38ed7d 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/adapters/SelectorAdapter.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/adapters/SelectorAdapter.java @@ -17,6 +17,7 @@ import org.telegram.messenger.UserConfig; import org.telegram.tgnet.TLRPC; import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.Cells.GraySectionCell; import org.telegram.ui.Components.ListView.AdapterWithDiffUtils; import org.telegram.ui.Components.Premium.boosts.BoostRepository; import org.telegram.ui.Components.Premium.boosts.cells.selector.SelectorCountryCell; @@ -35,12 +36,16 @@ public class SelectorAdapter extends AdapterWithDiffUtils { public static final int VIEW_TYPE_NO_USERS = 5; public static final int VIEW_TYPE_COUNTRY = 6; public static final int VIEW_TYPE_LETTER = 7; + public static final int VIEW_TYPE_TOP_SECTION = 8; private final Theme.ResourcesProvider resourcesProvider; private final Context context; private RecyclerListView listView; private List items; private HashMap chatsParticipantsCount = new HashMap<>(); + private View.OnClickListener topSectionClickListener; + private boolean isGreenSelector; + private GraySectionCell topSectionCell; public SelectorAdapter(Context context, Theme.ResourcesProvider resourcesProvider) { this.context = context; @@ -56,6 +61,21 @@ public void setData(List items, RecyclerListView listView) { this.listView = listView; } + public void setTopSectionClickListener(View.OnClickListener topSectionClickListener) { + this.topSectionClickListener = topSectionClickListener; + if (topSectionCell != null) { + if (topSectionClickListener == null) { + topSectionCell.setRightText(null); + } else { + topSectionCell.setRightText(LocaleController.getString(R.string.UsersDeselectAll), true, topSectionClickListener); + } + } + } + + public void setGreenSelector(boolean isGreenSelector) { + this.isGreenSelector = isGreenSelector; + } + @Override public boolean isEnabled(RecyclerView.ViewHolder holder) { return (holder.getItemViewType() == VIEW_TYPE_USER || holder.getItemViewType() == VIEW_TYPE_COUNTRY); @@ -68,7 +88,7 @@ public RecyclerView.ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int if (viewType == VIEW_TYPE_PAD) { view = new View(context); } else if (viewType == VIEW_TYPE_USER) { - view = new SelectorUserCell(context, resourcesProvider, false); + view = new SelectorUserCell(context, resourcesProvider, isGreenSelector); } else if (viewType == VIEW_TYPE_NO_USERS) { StickerEmptyView searchEmptyView = new StickerEmptyView(context, null, StickerEmptyView.STICKER_TYPE_SEARCH, resourcesProvider); searchEmptyView.title.setText(LocaleController.getString("NoResult", R.string.NoResult)); @@ -79,6 +99,8 @@ public RecyclerView.ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int view = new SelectorLetterCell(context, resourcesProvider); } else if (viewType == VIEW_TYPE_COUNTRY) { view = new SelectorCountryCell(context, resourcesProvider); + } else if (viewType == VIEW_TYPE_TOP_SECTION) { + view = new GraySectionCell(context, resourcesProvider); } else { view = new View(context); } @@ -128,9 +150,12 @@ public void onBindViewHolder(@NonNull RecyclerView.ViewHolder holder, int positi userCell.setChecked(item.checked, false); userCell.setCheckboxAlpha(1f, false); userCell.setDivider(position < items.size() - 2); + if ((position + 1 < items.size()) && items.get(position + 1).viewType == VIEW_TYPE_LETTER) { + userCell.setDivider(false); + } } else if (viewType == VIEW_TYPE_COUNTRY) { SelectorCountryCell cell = (SelectorCountryCell) holder.itemView; - boolean needDivider = (position < items.size() - 2) && (position + 1 < items.size() - 2) && (items.get(position + 1).viewType != VIEW_TYPE_LETTER); + boolean needDivider = (position < items.size() - 1) && (position + 1 < items.size() - 1) && (items.get(position + 1).viewType != VIEW_TYPE_LETTER); cell.setCountry(item.country, needDivider); cell.setChecked(item.checked, false); } else if (viewType == VIEW_TYPE_PAD) { @@ -149,6 +174,15 @@ public void onBindViewHolder(@NonNull RecyclerView.ViewHolder holder, int positi ((StickerEmptyView) holder.itemView).stickerView.getImageReceiver().startAnimation(); } catch (Exception ignore) { } + } else if (viewType == VIEW_TYPE_TOP_SECTION) { + GraySectionCell cell = (GraySectionCell) holder.itemView; + cell.setText(item.text); + if (topSectionClickListener == null) { + cell.setRightText(null, null); + } else { + cell.setRightText(LocaleController.getString(R.string.UsersDeselectAll), topSectionClickListener); + } + topSectionCell = cell; } } @@ -262,6 +296,12 @@ public static Item asLetter(String letter) { return item; } + public static Item asTopSection(String text) { + Item item = new Item(VIEW_TYPE_TOP_SECTION, false); + item.text = text; + return item; + } + public static Item asCountry(TLRPC.TL_help_country tlHelpCountry, boolean checked) { Item item = new Item(VIEW_TYPE_COUNTRY, true); item.country = tlHelpCountry; @@ -307,6 +347,8 @@ public boolean equals(Object o) { return false; } else if (viewType == VIEW_TYPE_LETTER && (!TextUtils.equals(text, i.text))) { return false; + } else if (viewType == VIEW_TYPE_TOP_SECTION && (!TextUtils.equals(text, i.text) || checked != i.checked)) { + return false; } return true; } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/ActionBtnCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/ActionBtnCell.java index d6626d363c3..9d0e9591264 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/ActionBtnCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/ActionBtnCell.java @@ -63,6 +63,13 @@ public void setGiftPremiumStyle(int counter, boolean animated, boolean isEnabled backgroundView.setBackgroundColor(Theme.getColor(Theme.key_dialogBackground, resourcesProvider)); } + public void setActivateForFreeStyle() { + drawDivider = true; + button.setEnabled(true); + button.setText(LocaleController.formatString("GiftPremiumActivateForFree", R.string.GiftPremiumActivateForFree), false); + backgroundView.setBackgroundColor(Theme.getColor(Theme.key_dialogBackground, resourcesProvider)); + } + @Override protected void dispatchDraw(Canvas canvas) { super.dispatchDraw(canvas); @@ -99,4 +106,9 @@ public void setCloseStyle(){ button.setEnabled(true); button.setText(LocaleController.formatString("Close", R.string.Close), false); } + + public void setCloseStyle(boolean needDivider){ + setCloseStyle(); + drawDivider = needDivider; + } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/DurationCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/DurationCell.java index 696e416333d..5e0a8c62ace 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/DurationCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/DurationCell.java @@ -13,7 +13,7 @@ @SuppressLint("ViewConstructor") public class DurationCell extends BaseCell { - private final SimpleTextView totalTextView; + protected final SimpleTextView totalTextView; private Object code; public DurationCell(Context context, Theme.ResourcesProvider resourcesProvider) { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/DurationWithDiscountCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/DurationWithDiscountCell.java new file mode 100644 index 00000000000..6cd9276297a --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/DurationWithDiscountCell.java @@ -0,0 +1,83 @@ +package org.telegram.ui.Components.Premium.boosts.cells; + +import android.annotation.SuppressLint; +import android.content.Context; +import android.view.Gravity; +import android.view.View; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.BillingController; +import org.telegram.messenger.LocaleController; +import org.telegram.messenger.R; +import org.telegram.tgnet.TLRPC; +import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.Components.CheckBox2; +import org.telegram.ui.Components.LayoutHelper; +import org.telegram.ui.Components.Premium.boosts.DiscountSpan; + +@SuppressLint("ViewConstructor") +public class DurationWithDiscountCell extends DurationCell { + + protected final CheckBox2 checkBox; + private TLRPC.TL_premiumGiftCodeOption option; + + public DurationWithDiscountCell(Context context, Theme.ResourcesProvider resourcesProvider) { + super(context, resourcesProvider); + checkBox = new CheckBox2(context, 21, resourcesProvider); + checkBox.setColor(Theme.key_premiumGradient1, Theme.key_checkboxDisabled, Theme.key_dialogRoundCheckBoxCheck); + checkBox.setDrawUnchecked(true); + checkBox.setDrawBackgroundAsArc(10); + addView(checkBox); + titleTextView.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); + radioButton.setVisibility(GONE); + updateLayouts(); + } + + @Override + protected void updateLayouts() { + super.updateLayouts(); + titleTextView.setLayoutParams(LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_VERTICAL | (LocaleController.isRTL ? Gravity.RIGHT : Gravity.LEFT), LocaleController.isRTL ? 20 : 102, 0, LocaleController.isRTL ? 102 : 20, 0)); + subtitleTextView.setLayoutParams(LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_VERTICAL | (LocaleController.isRTL ? Gravity.RIGHT : Gravity.LEFT), LocaleController.isRTL ? 20 : 102, 0, LocaleController.isRTL ? 102 : 20, 0)); + if (checkBox != null) { + checkBox.setLayoutParams(LayoutHelper.createFrame(22, 22, Gravity.CENTER_VERTICAL | (LocaleController.isRTL ? Gravity.RIGHT : Gravity.LEFT), LocaleController.isRTL ? 15 : 20, 0, LocaleController.isRTL ? 20 : 15, 0)); + } + } + + @Override + public void setChecked(boolean checked, boolean animated) { + if (checkBox.getVisibility() == View.VISIBLE) { + checkBox.setChecked(checked, animated); + } + } + + public void setDuration(TLRPC.TL_premiumGiftCodeOption option, TLRPC.TL_premiumGiftCodeOption minOption, int usersCount, boolean needDivider, boolean selected) { + this.option = option; + long price = option.amount; + CharSequence currency = option.currency; + titleTextView.setText(LocaleController.formatPluralString("Months", option.months)); + int discount = (int) ((1.0 - (option.amount / (double) option.months) / (minOption.amount / (double) minOption.months)) * 100); + String subTitle; + if (usersCount > 1) { + subTitle = BillingController.getInstance().formatCurrency(price / usersCount, currency.toString()) + " x " + usersCount; + } else { + subTitle = LocaleController.formatString(R.string.PricePerMonth, BillingController.getInstance().formatCurrency(price / option.months, currency.toString())); + } + if (discount > 0) { + setSubtitle(DiscountSpan.applySpan(subTitle, discount)); + } else { + setSubtitle(subTitle); + } + totalTextView.setText(BillingController.getInstance().formatCurrency(usersCount > 0 ? price : 0, currency.toString())); + setDivider(needDivider); + checkBox.setChecked(selected, false); + } + + public TLRPC.TL_premiumGiftCodeOption getOption() { + return option; + } + + @Override + protected boolean needCheck() { + return true; + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/EnterPrizeCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/EnterPrizeCell.java new file mode 100644 index 00000000000..4c03f708b65 --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/EnterPrizeCell.java @@ -0,0 +1,123 @@ +package org.telegram.ui.Components.Premium.boosts.cells; + +import org.telegram.messenger.ApplicationLoader; +import org.telegram.messenger.BotWebViewVibrationEffect; +import org.telegram.messenger.R; + +import android.annotation.SuppressLint; +import android.content.Context; +import android.text.Editable; +import android.text.InputFilter; +import android.text.InputType; +import android.text.Spanned; +import android.text.TextWatcher; +import android.util.TypedValue; +import android.view.Gravity; +import android.view.inputmethod.EditorInfo; +import android.widget.LinearLayout; +import android.widget.TextView; + +import androidx.annotation.NonNull; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.LocaleController; +import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.Components.EditTextCaption; +import org.telegram.ui.Components.LayoutHelper; + +@SuppressLint("ViewConstructor") +public class EnterPrizeCell extends LinearLayout { + private static final int MAX_INPUT_LENGTH = 128; + + public interface AfterTextChangedListener { + void afterTextChanged(String text); + } + + private final Theme.ResourcesProvider resourcesProvider; + private final EditTextCaption editText; + private final TextView textView; + private AfterTextChangedListener afterTextChangedListener; + + public EnterPrizeCell(@NonNull Context context, Theme.ResourcesProvider resourcesProvider) { + super(context); + this.resourcesProvider = resourcesProvider; + setOrientation(LinearLayout.HORIZONTAL); + editText = new EditTextCaption(context, resourcesProvider); + editText.setLines(1); + editText.setSingleLine(true); + InputFilter[] inputFilters = new InputFilter[1]; + inputFilters[0] = new InputFilter.LengthFilter(MAX_INPUT_LENGTH) { + @Override + public CharSequence filter(CharSequence source, int start, int end, Spanned dest, int dstart, int dend) { + CharSequence result = super.filter(source, start, end, dest, dstart, dend); + if (result != null && result.length() == 0) { + AndroidUtilities.shakeView(editText); + BotWebViewVibrationEffect.APP_ERROR.vibrate(); + } + return result; + } + }; + editText.setInputType(InputType.TYPE_TEXT_FLAG_CAP_SENTENCES); + editText.setFilters(inputFilters); + editText.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 16); + editText.setTextColor(Theme.getColor(Theme.key_chat_messagePanelText, resourcesProvider)); + editText.setLinkTextColor(Theme.getColor(Theme.key_chat_messageLinkOut, resourcesProvider)); + editText.setHighlightColor(Theme.getColor(Theme.key_chat_inTextSelectionHighlight, resourcesProvider)); + editText.setHintColor(Theme.getColor(Theme.key_chat_messagePanelHint, resourcesProvider)); + editText.setHintTextColor(Theme.getColor(Theme.key_chat_messagePanelHint, resourcesProvider)); + editText.setCursorColor(Theme.getColor(Theme.key_chat_messagePanelCursor, resourcesProvider)); + editText.setHandlesColor(Theme.getColor(Theme.key_chat_TextSelectionCursor, resourcesProvider)); + editText.setBackground(null); + editText.setHint(LocaleController.getString("BoostingGiveawayEnterYourPrize", R.string.BoostingGiveawayEnterYourPrize)); + editText.addTextChangedListener(new TextWatcher() { + + @Override + public void beforeTextChanged(CharSequence s, int start, int count, int after) { + + } + + @Override + public void onTextChanged(CharSequence s, int start, int before, int count) { + + } + + @Override + public void afterTextChanged(Editable s) { + if (afterTextChangedListener != null) { + afterTextChangedListener.afterTextChanged(s.toString().trim()); + } + } + }); + editText.setImeOptions(EditorInfo.IME_ACTION_DONE); + + textView = new TextView(context); + textView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 16); + textView.setTextColor(Theme.getColor(Theme.key_dialogTextBlack, resourcesProvider)); + + if (LocaleController.isRTL) { + LinearLayout.LayoutParams lp = LayoutHelper.createLinear(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_VERTICAL, 20, 0, 36, 0); + lp.weight = 1; + addView(editText, lp); + addView(textView, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_VERTICAL, 0, 0, 20, 0)); + } else { + addView(textView, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_VERTICAL, 20, 0, 0, 0)); + addView(editText, LayoutHelper.createLinear(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_VERTICAL, 36, 0, 20, 0)); + } + } + + @Override + protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { + super.onMeasure( + MeasureSpec.makeMeasureSpec(MeasureSpec.getSize(widthMeasureSpec), MeasureSpec.EXACTLY), + MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(50), MeasureSpec.EXACTLY) + ); + } + + public void setAfterTextChangedListener(AfterTextChangedListener afterTextChangedListener) { + this.afterTextChangedListener = afterTextChangedListener; + } + + public void setCount(int count) { + textView.setText(String.valueOf(count)); + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/HeaderCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/HeaderCell.java index 59107741c27..b3a38b3f926 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/HeaderCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/HeaderCell.java @@ -1,19 +1,14 @@ package org.telegram.ui.Components.Premium.boosts.cells; import static org.telegram.messenger.AndroidUtilities.REPLACING_TAG_TYPE_LINKBOLD; -import static org.telegram.messenger.AndroidUtilities.dp; import android.annotation.SuppressLint; import android.content.Context; import android.graphics.Bitmap; import android.graphics.Canvas; -import android.graphics.Color; import android.graphics.Outline; import android.os.Build; -import android.os.Bundle; -import android.text.Html; import android.text.SpannableStringBuilder; -import android.text.Spanned; import android.text.method.LinkMovementMethod; import android.util.TypedValue; import android.view.Gravity; @@ -25,31 +20,22 @@ import androidx.annotation.NonNull; import androidx.core.graphics.ColorUtils; -import androidx.core.text.HtmlCompat; import org.telegram.messenger.AndroidUtilities; -import org.telegram.messenger.Emoji; -import org.telegram.messenger.FileLog; import org.telegram.messenger.LocaleController; import org.telegram.messenger.MessagesController; import org.telegram.messenger.R; -import org.telegram.messenger.SendMessagesHelper; import org.telegram.messenger.UserConfig; import org.telegram.messenger.UserObject; import org.telegram.messenger.Utilities; import org.telegram.tgnet.TLObject; import org.telegram.tgnet.TLRPC; import org.telegram.ui.ActionBar.Theme; -import org.telegram.ui.ChatActivity; -import org.telegram.ui.Components.AvatarDrawable; import org.telegram.ui.Components.LayoutHelper; import org.telegram.ui.Components.LinkSpanDrawable; import org.telegram.ui.Components.Premium.GLIcon.GLIconRenderer; import org.telegram.ui.Components.Premium.GLIcon.GLIconTextureView; import org.telegram.ui.Components.Premium.StarParticlesView; -import org.telegram.ui.Components.Premium.boosts.BoostDialogs; -import org.telegram.ui.DialogsActivity; -import org.telegram.ui.LaunchActivity; @SuppressLint("ViewConstructor") public class HeaderCell extends FrameLayout { @@ -178,31 +164,16 @@ public void setUnclaimedText() { public void setGiftLinkToUserText(long toUserId, Utilities.Callback onObjectClicked) { titleView.setText(LocaleController.formatString("BoostingGiftLink", R.string.BoostingGiftLink)); - SpannableStringBuilder stringBuilder = new SpannableStringBuilder(); - try { - String description = LocaleController.getString("BoostingLinkAllowsToUser", R.string.BoostingLinkAllowsToUser); - CharSequence descriptionStart = description.substring(0, description.indexOf("**%1$s**") + 8); - CharSequence descriptionEnd = description.substring(description.indexOf("**%1$s**") + 8); - - TLRPC.User toUser = MessagesController.getInstance(UserConfig.selectedAccount).getUser(toUserId); - SpannableStringBuilder userName = new SpannableStringBuilder(); - userName.append("**"); - userName.append(Emoji.replaceEmoji(UserObject.getUserName(toUser), subtitleView.getPaint().getFontMetricsInt(), false)); - userName.append("**"); - - descriptionStart = AndroidUtilities.replaceSingleTag( - descriptionStart.toString().replace("**%1$s**", userName), - Theme.key_chat_messageLinkIn, REPLACING_TAG_TYPE_LINKBOLD, - () -> onObjectClicked.run(toUser), - resourcesProvider - ); - descriptionEnd = AndroidUtilities.replaceTags(descriptionEnd.toString()); - stringBuilder.append(descriptionStart); - stringBuilder.append(descriptionEnd); - } catch (Exception e) { - FileLog.e(e); - } - subtitleView.setText(stringBuilder); + CharSequence description = AndroidUtilities.replaceTags(LocaleController.getString("BoostingLinkAllowsToUser", R.string.BoostingLinkAllowsToUser)); + TLRPC.User toUser = MessagesController.getInstance(UserConfig.selectedAccount).getUser(toUserId); + + SpannableStringBuilder link = AndroidUtilities.replaceSingleTag( + "**" + UserObject.getUserName(toUser) + "**", + Theme.key_chat_messageLinkIn, REPLACING_TAG_TYPE_LINKBOLD, + () -> onObjectClicked.run(toUser), + resourcesProvider + ); + subtitleView.setText(AndroidUtilities.replaceCharSequence("%1$s", description, link)); } @Override diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/SwitcherCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/SwitcherCell.java new file mode 100644 index 00000000000..006d6587333 --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/SwitcherCell.java @@ -0,0 +1,41 @@ +package org.telegram.ui.Components.Premium.boosts.cells; + +import android.content.Context; + +import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.Cells.TextCheckCell; + +public class SwitcherCell extends TextCheckCell { + public static int TYPE_WINNERS = 0; + public static int TYPE_ADDITION_PRIZE = 1; + private int type; + + public SwitcherCell(Context context) { + super(context); + } + + public SwitcherCell(Context context, int padding) { + super(context, padding); + } + + public SwitcherCell(Context context, Theme.ResourcesProvider resourcesProvider) { + super(context, resourcesProvider); + } + + public SwitcherCell(Context context, int padding, boolean dialog) { + super(context, padding, dialog); + } + + public SwitcherCell(Context context, int padding, boolean dialog, Theme.ResourcesProvider resourcesProvider) { + super(context, padding, dialog, resourcesProvider); + } + + public int getType() { + return type; + } + + public void setData(CharSequence text, boolean checked, boolean divider, int type) { + this.type = type; + setTextAndCheck(text, checked, divider); + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/TableCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/TableCell.java index 5b208ff3a86..053255b7299 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/TableCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/TableCell.java @@ -282,7 +282,6 @@ private TextView createTextView(String text, boolean blueColor) { if (blueColor) { textView = new LinkSpanDrawable.LinksTextView(getContext(), resourcesProvider); - textView.setMovementMethod(LinkMovementMethod.getInstance()); textView.setLinkTextColor(Theme.getColor(Theme.key_windowBackgroundWhiteLinkText, resourcesProvider)); } else { textView = new TextView(getContext()); @@ -290,7 +289,9 @@ private TextView createTextView(String text, boolean blueColor) { textView.setTextColor(Theme.getColor(blueColor ? Theme.key_dialogTextBlue : Theme.key_dialogTextBlack, resourcesProvider)); textView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 14); - textView.setGravity(LocaleController.isRTL ? Gravity.RIGHT : Gravity.LEFT); + if (!blueColor) { + textView.setGravity(LocaleController.isRTL ? Gravity.RIGHT : Gravity.LEFT); + } if (text != null) { textView.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); textView.setText(text); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/msg/GiveawayMessageCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/msg/GiveawayMessageCell.java index 652b603d319..9893bb26238 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/msg/GiveawayMessageCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/msg/GiveawayMessageCell.java @@ -72,39 +72,49 @@ public class GiveawayMessageCell { private ImageReceiver[] avatarImageReceivers; private AvatarDrawable[] avatarDrawables; private final ChatMessageCell parentView; - private final ImageReceiver giftReceiver; + private ImageReceiver giftReceiver; - private CharSequence[] chatTitles = new CharSequence[10]; - private TLRPC.Chat[] chats = new TLRPC.Chat[10]; - private float[] chatTitleWidths = new float[10]; - private boolean[] needNewRow = new boolean[10]; - private Rect[] clickRect = new Rect[10]; + private CharSequence[] chatTitles; + private TLRPC.Chat[] chats; + private float[] chatTitleWidths; + private boolean[] needNewRow; + private Rect[] clickRect; private boolean[] avatarVisible; private int measuredHeight = 0; private int measuredWidth = 0; + + private int additionPrizeHeight; + private float textDividerWidth; + private String textDivider; + private int titleHeight; private int topHeight; private int bottomHeight; private int countriesHeight; private String counterStr; private int diffTextWidth; + + private StaticLayout titleLayout; + private StaticLayout additionPrizeLayout; private StaticLayout topLayout; private StaticLayout bottomLayout; private StaticLayout countriesLayout; - private final TextPaint counterTextPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG); - private final TextPaint chatTextPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG); - private final TextPaint textPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG); - private final TextPaint countriesTextPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG); - private final Paint counterBgPaint = new Paint(Paint.ANTI_ALIAS_FLAG); - private final Paint chatBgPaint = new Paint(Paint.ANTI_ALIAS_FLAG); - - private final Paint saveLayerPaint = new Paint(); - private final Paint clipRectPaint = new Paint(); - private final RectF countRect = new RectF(); - private final RectF chatRect = new RectF(); - private final Rect counterTextBounds = new Rect(); - private final Rect containerRect = new Rect(); - private final int[] pressedState = new int[]{android.R.attr.state_enabled, android.R.attr.state_pressed}; + private TextPaint counterTextPaint; + private TextPaint chatTextPaint; + private TextPaint textPaint; + private TextPaint textDividerPaint; + private Paint lineDividerPaint; + private TextPaint countriesTextPaint; + private Paint counterBgPaint; + private Paint chatBgPaint; + + private Paint saveLayerPaint; + private Paint clipRectPaint; + private RectF countRect; + private RectF chatRect; + private Rect counterTextBounds; + private Rect containerRect; + private int[] pressedState; private int selectorColor; private Drawable selectorDrawable; @@ -115,6 +125,35 @@ public class GiveawayMessageCell { public GiveawayMessageCell(ChatMessageCell parentView) { this.parentView = parentView; + } + + private void init() { + if (counterTextPaint != null) { + return; + } + counterTextPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG); + chatTextPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG); + textPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG); + textDividerPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG); + lineDividerPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + countriesTextPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG); + counterBgPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + chatBgPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + + saveLayerPaint = new Paint(); + clipRectPaint = new Paint(); + countRect = new RectF(); + chatRect = new RectF(); + counterTextBounds = new Rect(); + containerRect = new Rect(); + pressedState = new int[]{android.R.attr.state_enabled, android.R.attr.state_pressed}; + + chatTitles = new CharSequence[10]; + chats = new TLRPC.Chat[10]; + chatTitleWidths = new float[10]; + needNewRow = new boolean[10]; + clickRect = new Rect[10]; + giftReceiver = new ImageReceiver(parentView); giftReceiver.setAllowLoadingOnAttachedOnly(true); @@ -127,6 +166,8 @@ public GiveawayMessageCell(ChatMessageCell parentView) { chatTextPaint.setTextSize(dp(13)); countriesTextPaint.setTextSize(dp(13)); textPaint.setTextSize(dp(14)); + textDividerPaint.setTextSize(dp(14)); + textDividerPaint.setTextAlign(Paint.Align.CENTER); } public boolean checkMotionEvent(MotionEvent event) { @@ -210,15 +251,20 @@ public void unscheduleDrawable(@NonNull Drawable who, @NonNull Runnable what) { public void setMessageContent(MessageObject messageObject, int parentWidth, int forwardedNameWidth) { this.messageObject = null; + titleLayout = null; + additionPrizeLayout = null; topLayout = null; bottomLayout = null; countriesLayout = null; measuredHeight = 0; measuredWidth = 0; + additionPrizeHeight = 0; + textDividerWidth = 0; if (!messageObject.isGiveaway()) { return; } this.messageObject = messageObject; + init(); createImages(); setGiftImage(messageObject); TLRPC.TL_messageMediaGiveaway giveaway = (TLRPC.TL_messageMediaGiveaway) messageObject.messageOwner.media; @@ -233,23 +279,23 @@ public void setMessageContent(MessageObject messageObject, int parentWidth, int } CharSequence giveawayPrizes = replaceTags(getString("BoostingGiveawayPrizes", R.string.BoostingGiveawayPrizes)); - SpannableStringBuilder topStringBuilder = new SpannableStringBuilder(giveawayPrizes); - topStringBuilder.setSpan(new RelativeSizeSpan(1.05f), 0, giveawayPrizes.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); - topStringBuilder.append("\n"); + SpannableStringBuilder titleStringBuilder = new SpannableStringBuilder(giveawayPrizes); + titleStringBuilder.setSpan(new RelativeSizeSpan(1.05f), 0, giveawayPrizes.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + SpannableStringBuilder subTitleBuilder = new SpannableStringBuilder(); subTitleBuilder.append(replaceTags(formatPluralStringComma("BoostingGiveawayMsgInfoPlural1", giveaway.quantity))); subTitleBuilder.append("\n"); subTitleBuilder.append(replaceTags(formatPluralString("BoostingGiveawayMsgInfoPlural2", giveaway.quantity, LocaleController.formatPluralString("BoldMonths", giveaway.months)))); - CharSequence subTitle = subTitleBuilder; + SpannableStringBuilder topStringBuilder = new SpannableStringBuilder(); topStringBuilder.append(subTitleBuilder); topStringBuilder.append("\n\n"); - topStringBuilder.setSpan(new RelativeSizeSpan(0.5f), topStringBuilder.length() - 1, topStringBuilder.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + topStringBuilder.setSpan(new RelativeSizeSpan(0.4f), topStringBuilder.length() - 1, topStringBuilder.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); CharSequence participants = replaceTags(getString("BoostingGiveawayMsgParticipants", R.string.BoostingGiveawayMsgParticipants)); topStringBuilder.append(participants); - topStringBuilder.setSpan(new RelativeSizeSpan(1.05f), giveawayPrizes.length() + subTitle.length() + 2, giveawayPrizes.length() + subTitle.length() + 3 + participants.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + topStringBuilder.setSpan(new RelativeSizeSpan(1.05f), subTitleBuilder.length() + 2, subTitleBuilder.length() + 2 + participants.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); topStringBuilder.append("\n"); if (giveaway.only_new_subscribers) { @@ -267,10 +313,14 @@ public void setMessageContent(MessageObject messageObject, int parentWidth, int bottomStringBuilder.append("\n"); bottomStringBuilder.append(formatString("formatDateAtTime", R.string.formatDateAtTime, monthTxt, timeTxt)); + titleLayout = StaticLayoutEx.createStaticLayout(titleStringBuilder, textPaint, maxWidth, Layout.Alignment.ALIGN_CENTER, 1.0f, AndroidUtilities.dp(2), false, TextUtils.TruncateAt.END, maxWidth, 10); topLayout = StaticLayoutEx.createStaticLayout(topStringBuilder, textPaint, maxWidth, Layout.Alignment.ALIGN_CENTER, 1.0f, AndroidUtilities.dp(2), false, TextUtils.TruncateAt.END, maxWidth, 10); - bottomLayout = StaticLayoutEx.createStaticLayout(bottomStringBuilder, textPaint, maxWidth, Layout.Alignment.ALIGN_CENTER, 1.0f, AndroidUtilities.dp(2), false, TextUtils.TruncateAt.END, maxWidth, 10); + bottomLayout = StaticLayoutEx.createStaticLayout(bottomStringBuilder, textPaint, maxWidth, Layout.Alignment.ALIGN_CENTER, 1.0f, AndroidUtilities.dp(3), false, TextUtils.TruncateAt.END, maxWidth, 10); int maxRowLength = 0; + for (int a = 0; a < titleLayout.getLineCount(); ++a) { + maxRowLength = (int) Math.max(maxRowLength, Math.ceil(titleLayout.getLineWidth(a))); + } for (int a = 0; a < topLayout.getLineCount(); ++a) { maxRowLength = (int) Math.max(maxRowLength, Math.ceil(topLayout.getLineWidth(a))); } @@ -282,6 +332,14 @@ public void setMessageContent(MessageObject messageObject, int parentWidth, int maxRowLength = dp(180); } + if (giveaway.prize_description != null && !giveaway.prize_description.isEmpty()) { + CharSequence txt = Emoji.replaceEmoji(AndroidUtilities.replaceTags(LocaleController.formatPluralString("BoostingGiveawayMsgPrizes", giveaway.quantity, giveaway.prize_description)), countriesTextPaint.getFontMetricsInt(), false); + additionPrizeLayout = StaticLayoutEx.createStaticLayout(txt, textPaint, maxRowLength, Layout.Alignment.ALIGN_CENTER, 1.0f, AndroidUtilities.dp(2), false, TextUtils.TruncateAt.END, maxRowLength, 20); + additionPrizeHeight = additionPrizeLayout.getLineBottom(additionPrizeLayout.getLineCount() - 1) + dp(22); + textDivider = LocaleController.getString("BoostingGiveawayMsgWithDivider", R.string.BoostingGiveawayMsgWithDivider); + textDividerWidth = textDividerPaint.measureText(textDivider, 0, textDivider.length()); + } + if (giveaway.countries_iso2.size() > 0) { List countriesWithFlags = new ArrayList<>(); for (String iso2 : giveaway.countries_iso2) { @@ -308,7 +366,8 @@ public void setMessageContent(MessageObject messageObject, int parentWidth, int giftReceiver.setImageCoords((maxWidth / 2f) - (giftSize / 2f), AndroidUtilities.dp(42) - giftSize / 2f, giftSize, giftSize); - topHeight = topLayout.getLineBottom(topLayout.getLineCount() - 1); + titleHeight = titleLayout.getLineBottom(titleLayout.getLineCount() - 1) + dp(5); + topHeight = titleHeight + additionPrizeHeight + topLayout.getLineBottom(topLayout.getLineCount() - 1); bottomHeight = bottomLayout.getLineBottom(bottomLayout.getLineCount() - 1); countriesHeight = countriesLayout != null ? (countriesLayout.getLineBottom(countriesLayout.getLineCount() - 1) + dp(4 + 8)) : 0; @@ -341,10 +400,10 @@ public void setMessageContent(MessageObject messageObject, int parentWidth, int if (chat != null) { avatarVisible[i] = true; chats[i] = chat; - CharSequence text = Emoji.replaceEmoji(chat.title, chatTextPaint.getFontMetricsInt(), dp(14), false); + CharSequence text = Emoji.replaceEmoji(chat.title, chatTextPaint.getFontMetricsInt(), false); chatTitles[i] = TextUtils.ellipsize(text, chatTextPaint, maxWidth * 0.8f, TextUtils.TruncateAt.END); chatTitleWidths[i] = chatTextPaint.measureText(chatTitles[i], 0, chatTitles[i].length()); - float oneRowWidth = chatTitleWidths[i] + dp(24 + 6 + 12); + float oneRowWidth = chatTitleWidths[i] + dp(24 + 6 + 10); oneRowTotalWidth += oneRowWidth; if (i > 0) { needNewRow[i] = oneRowTotalWidth > maxWidth * 0.9f; @@ -370,6 +429,9 @@ public void setMessageContent(MessageObject messageObject, int parentWidth, int } private int getChatColor(TLRPC.Chat chat, Theme.ResourcesProvider resourcesProvider) { + if (messageObject.isOutOwner()) { + return Theme.getColor(Theme.key_chat_outPreviewInstantText, resourcesProvider); + } final int color; int colorId = ChatObject.getColorId(chat); if (colorId < 7) { @@ -378,7 +440,7 @@ private int getChatColor(TLRPC.Chat chat, Theme.ResourcesProvider resourcesProvi MessagesController.PeerColors peerColors = MessagesController.getInstance(UserConfig.selectedAccount).peerColors; MessagesController.PeerColor peerColor = peerColors == null ? null : peerColors.getColor(colorId); if (peerColor != null) { - color = peerColor.getColor1(); + color = peerColor.getColor(0, resourcesProvider); } else { color = Theme.getColor(Theme.keys_avatar_nameInMessage[0], resourcesProvider); } @@ -396,6 +458,8 @@ public void draw(Canvas canvas, int marginTop, int marginLeft, Theme.ResourcesPr } textPaint.setColor(Theme.chat_msgTextPaint.getColor()); + textDividerPaint.setColor(Theme.multAlpha(Theme.chat_msgTextPaint.getColor(), 0.45f)); + lineDividerPaint.setColor(Theme.multAlpha(Theme.chat_msgTextPaint.getColor(), 0.15f)); countriesTextPaint.setColor(Theme.chat_msgTextPaint.getColor()); if (messageObject.isOutOwner()) { @@ -444,6 +508,24 @@ public void draw(Canvas canvas, int marginTop, int marginLeft, Theme.ResourcesPr canvas.save(); canvas.translate(diffTextWidth / 2f, 0); + titleLayout.draw(canvas); + canvas.translate(0, titleHeight); + + if (additionPrizeLayout != null) { + canvas.restore(); + canvas.save(); + float textDividerCY = titleHeight + additionPrizeHeight - dp(22 - 16); + float textDividerCX = measuredWidth / 2f; + canvas.drawText(textDivider, textDividerCX, textDividerCY, textDividerPaint); + canvas.drawLine(dp(17), textDividerCY - dp(4), textDividerCX - textDividerWidth / 2f - dp(6), textDividerCY - dp(4), lineDividerPaint); + canvas.drawLine(textDividerCX + textDividerWidth / 2f + dp(6), textDividerCY - dp(4), measuredWidth - dp(16), textDividerCY - dp(4), lineDividerPaint); + canvas.translate((measuredWidth - additionPrizeLayout.getWidth()) / 2f, titleHeight); + additionPrizeLayout.draw(canvas); + canvas.restore(); + canvas.save(); + canvas.translate(diffTextWidth / 2f, additionPrizeHeight + titleHeight); + } + topLayout.draw(canvas); canvas.restore(); canvas.translate(0, topHeight + dp(6)); @@ -458,7 +540,7 @@ public void draw(Canvas canvas, int marginTop, int marginLeft, Theme.ResourcesPr float rowWidth = 0; do { - rowWidth += chatTitleWidths[k] + dp(24 + 6 + 12); + rowWidth += chatTitleWidths[k] + dp(24 + 6 + 10); k++; } while (k < avatarVisible.length && !needNewRow[k] && avatarVisible[k]); @@ -478,7 +560,7 @@ public void draw(Canvas canvas, int marginTop, int marginLeft, Theme.ResourcesPr chatBgPaint.setAlpha(25); avatarImageReceivers[k].draw(canvas); canvas.drawText(chatTitles[k], 0, chatTitles[k].length(), dp(24 + 6), dp(16), chatTextPaint); - chatRect.set(0, 0, chatTitleWidths[k] + dp(24 + 6 + 12), dp(24)); + chatRect.set(0, 0, chatTitleWidths[k] + dp(24 + 6 + 10), dp(24)); canvas.drawRoundRect(chatRect, dp(12), dp(12), chatBgPaint); clickRect[k].set(xRow, y, (int) (xRow + chatRect.width()), y + dp(24)); @@ -524,7 +606,9 @@ public void draw(Canvas canvas, int marginTop, int marginLeft, Theme.ResourcesPr } public void onDetachedFromWindow() { - giftReceiver.onDetachedFromWindow(); + if (giftReceiver != null) { + giftReceiver.onDetachedFromWindow(); + } if (avatarImageReceivers != null) { for (ImageReceiver avatarImageReceiver : avatarImageReceivers) { avatarImageReceiver.onDetachedFromWindow(); @@ -533,7 +617,9 @@ public void onDetachedFromWindow() { } public void onAttachedToWindow() { - giftReceiver.onAttachedToWindow(); + if (giftReceiver != null) { + giftReceiver.onAttachedToWindow(); + } if (avatarImageReceivers != null) { for (ImageReceiver avatarImageReceiver : avatarImageReceivers) { avatarImageReceiver.onAttachedToWindow(); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/msg/GiveawayResultsMessageCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/msg/GiveawayResultsMessageCell.java new file mode 100644 index 00000000000..f3a6a2be90f --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/msg/GiveawayResultsMessageCell.java @@ -0,0 +1,652 @@ +package org.telegram.ui.Components.Premium.boosts.cells.msg; + +import static org.telegram.messenger.AndroidUtilities.dp; +import static org.telegram.messenger.AndroidUtilities.replaceTags; +import static org.telegram.messenger.LocaleController.formatPluralString; +import static org.telegram.messenger.LocaleController.getPluralString; +import static org.telegram.messenger.LocaleController.getString; + +import android.graphics.Canvas; +import android.graphics.Paint; +import android.graphics.PorterDuff; +import android.graphics.PorterDuffXfermode; +import android.graphics.Rect; +import android.graphics.RectF; +import android.graphics.drawable.Drawable; +import android.os.Build; +import android.os.Bundle; +import android.text.Layout; +import android.text.SpannableStringBuilder; +import android.text.Spanned; +import android.text.StaticLayout; +import android.text.TextPaint; +import android.text.TextUtils; +import android.text.style.ClickableSpan; +import android.text.style.RelativeSizeSpan; +import android.util.StateSet; +import android.view.MotionEvent; +import android.view.SoundEffectConstants; + +import androidx.annotation.NonNull; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.Emoji; +import org.telegram.messenger.FileLog; +import org.telegram.messenger.ImageReceiver; +import org.telegram.messenger.LocaleController; +import org.telegram.messenger.MessageObject; +import org.telegram.messenger.MessagesController; +import org.telegram.messenger.R; +import org.telegram.messenger.UserConfig; +import org.telegram.messenger.UserObject; +import org.telegram.tgnet.TLRPC; +import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.Cells.ChatMessageCell; +import org.telegram.ui.ChatActivity; +import org.telegram.ui.Components.AvatarDrawable; +import org.telegram.ui.Components.LinkPath; +import org.telegram.ui.Components.LinkSpanDrawable; +import org.telegram.ui.Components.Premium.boosts.BoostDialogs; +import org.telegram.ui.Components.RLottieDrawable; +import org.telegram.ui.Components.StaticLayoutEx; +import org.telegram.ui.LaunchActivity; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; + +public class GiveawayResultsMessageCell { + + private ImageReceiver[] avatarImageReceivers; + private AvatarDrawable[] avatarDrawables; + private final ChatMessageCell parentView; + private ImageReceiver giftReceiver; + private RLottieDrawable giftDrawable; + + private CharSequence[] userTitles; + private TLRPC.User[] users; + private float[] userTitleWidths; + private boolean[] needNewRow; + private Rect[] clickRect; + private boolean[] avatarVisible; + private int measuredHeight = 0; + private int measuredWidth = 0; + + private int titleHeight; + private int topHeight; + private int bottomHeight; + private int countriesHeight; + private String counterStr; + private int diffTextWidth; + + private StaticLayout titleLayout; + private StaticLayout topLayout; + private StaticLayout bottomLayout; + private StaticLayout countriesLayout; + + private TextPaint counterTextPaint; + private TextPaint chatTextPaint; + private TextPaint textPaint; + private TextPaint textDividerPaint; + private TextPaint countriesTextPaint; + private Paint counterBgPaint; + private Paint chatBgPaint; + + private Paint saveLayerPaint; + private Paint clipRectPaint; + private RectF countRect; + private RectF chatRect; + private Rect counterTextBounds; + private Rect containerRect; + private int[] pressedState; + + private int selectorColor; + private Drawable selectorDrawable; + private MessageObject messageObject; + private int pressedPos = -1; + private boolean isButtonPressed = false; + private boolean isContainerPressed = false; + private SpannableStringBuilder topStringBuilder; + private int subTitleMarginTop; + private int subTitleMarginLeft; + private LinkSpanDrawable.LinkCollector links; + + public GiveawayResultsMessageCell(ChatMessageCell parentView) { + this.parentView = parentView; + } + + private void init() { + if (counterTextPaint != null) { + return; + } + counterTextPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG); + chatTextPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG); + textPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG); + textDividerPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG); + countriesTextPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG); + counterBgPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + chatBgPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + + saveLayerPaint = new Paint(); + clipRectPaint = new Paint(); + countRect = new RectF(); + chatRect = new RectF(); + counterTextBounds = new Rect(); + containerRect = new Rect(); + pressedState = new int[]{android.R.attr.state_enabled, android.R.attr.state_pressed}; + + userTitles = new CharSequence[10]; + users = new TLRPC.User[10]; + userTitleWidths = new float[10]; + needNewRow = new boolean[10]; + clickRect = new Rect[10]; + + giftReceiver = new ImageReceiver(parentView); + giftReceiver.setAllowLoadingOnAttachedOnly(true); + + clipRectPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_OUT)); + counterTextPaint.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); + counterTextPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_OUT)); + counterTextPaint.setTextSize(dp(12)); + counterTextPaint.setTextAlign(Paint.Align.CENTER); + chatTextPaint.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); + chatTextPaint.setTextSize(dp(13)); + countriesTextPaint.setTextSize(dp(13)); + textPaint.setTextSize(dp(14)); + textDividerPaint.setTextSize(dp(14)); + textDividerPaint.setTextAlign(Paint.Align.CENTER); + } + + public boolean checkMotionEvent(MotionEvent event) { + if (messageObject == null || !messageObject.isGiveawayResults()) { + return false; + } + + if (links == null) { + links = new LinkSpanDrawable.LinkCollector(parentView); + } + + int action = event.getAction(); + int x = (int) event.getX(); + int y = (int) event.getY(); + + if (action == MotionEvent.ACTION_UP || action == MotionEvent.ACTION_DOWN) { + if (topStringBuilder != null && topLayout != null && (y - subTitleMarginTop) > 0) { + int line = topLayout.getLineForVertical(y - subTitleMarginTop - dp(10)); + int off = topLayout.getOffsetForHorizontal(line, x - subTitleMarginLeft); + ClickableSpan[] link = topStringBuilder.getSpans(off, off, ClickableSpan.class); + if (link.length != 0) { + if (action == MotionEvent.ACTION_UP) { + links.clear(); + link[0].onClick(parentView); + } else { + LinkSpanDrawable pressedLink = new LinkSpanDrawable<>(link[0], null, x, y); + links.addLink(pressedLink); + try { + int start = topStringBuilder.getSpanStart(link[0]); + LinkPath path = pressedLink.obtainNewPath(); + path.setCurrentLayout(topLayout, start, subTitleMarginLeft, subTitleMarginTop); + topLayout.getSelectionPath(start, topStringBuilder.getSpanEnd(link[0]), path); + } catch (Exception e) { + FileLog.e(e); + } + } + return true; + } else { + links.clear(); + } + parentView.invalidate(); + } + } + + if (action == MotionEvent.ACTION_DOWN) { + for (int i = 0; i < clickRect.length; i++) { + Rect rect = clickRect[i]; + if (rect.contains(x, y)) { + pressedPos = i; + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { + selectorDrawable.setHotspot(x, y); + } + isButtonPressed = true; + setButtonPressed(true); + return true; + } + } + if (containerRect.contains(x, y)) { + isContainerPressed = true; + return true; + } + } else if (action == MotionEvent.ACTION_UP) { + if (isButtonPressed) { + if (parentView.getDelegate() != null) { + parentView.getDelegate().didPressGiveawayChatButton(parentView, pressedPos); + } + parentView.playSoundEffect(SoundEffectConstants.CLICK); + setButtonPressed(false); + isButtonPressed = false; + } + if (isContainerPressed) { + isContainerPressed = false; + BoostDialogs.showBulletinAbout(messageObject); + } + } else if (action == MotionEvent.ACTION_MOVE) { + + } else if (action == MotionEvent.ACTION_CANCEL) { + links.clear(); + if (isButtonPressed) { + setButtonPressed(false); + } + isButtonPressed = false; + isContainerPressed = false; + } + return false; + } + + public void setButtonPressed(boolean pressed) { + if (messageObject == null || !messageObject.isGiveawayResults() || selectorDrawable == null) { + return; + } + if (links != null) { + links.clear(); + } + if (pressed) { + selectorDrawable.setCallback(new Drawable.Callback() { + @Override + public void invalidateDrawable(@NonNull Drawable who) { + parentView.invalidate(); + } + + @Override + public void scheduleDrawable(@NonNull Drawable who, @NonNull Runnable what, long when) { + parentView.invalidate(); + } + + @Override + public void unscheduleDrawable(@NonNull Drawable who, @NonNull Runnable what) { + parentView.invalidate(); + } + }); + selectorDrawable.setState(pressedState); + parentView.invalidate(); + } else { + selectorDrawable.setState(StateSet.NOTHING); + parentView.invalidate(); + } + } + + public void setMessageContent(MessageObject messageObject, int parentWidth, int forwardedNameWidth) { + this.messageObject = null; + titleLayout = null; + topLayout = null; + bottomLayout = null; + countriesLayout = null; + measuredHeight = 0; + measuredWidth = 0; + if (!messageObject.isGiveawayResults()) { + return; + } + this.messageObject = messageObject; + init(); + createImages(); + setGiftImage(); + TLRPC.TL_messageMediaGiveawayResults giveaway = (TLRPC.TL_messageMediaGiveawayResults) messageObject.messageOwner.media; + checkArraysLimits(giveaway.winners.size()); + + int giftSize = AndroidUtilities.dp(90); + int maxWidth = AndroidUtilities.dp(230); + + CharSequence winnersSelected = replaceTags(getString("BoostingGiveawayResultsMsgWinnersSelected", R.string.BoostingGiveawayResultsMsgWinnersSelected)); + SpannableStringBuilder titleStringBuilder = new SpannableStringBuilder(winnersSelected); + titleStringBuilder.setSpan(new RelativeSizeSpan(1.05f), 0, winnersSelected.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + + topStringBuilder = new SpannableStringBuilder(); + String subTitleText = getPluralString("BoostingGiveawayResultsMsgWinnersTitle", giveaway.winners_count); + SpannableStringBuilder subTitleWithLink = AndroidUtilities.replaceSingleTag( + subTitleText, + Theme.key_chat_messageLinkIn, 0, + () -> AndroidUtilities.runOnUIThread(() -> { + if (messageObject.getDialogId() == -giveaway.channel_id) { + parentView.getDelegate().didPressReplyMessage(parentView, giveaway.launch_msg_id); + } else { + Bundle bundle = new Bundle(); + bundle.putLong("chat_id", giveaway.channel_id); + bundle.putInt("message_id", giveaway.launch_msg_id); + LaunchActivity.getLastFragment().presentFragment(new ChatActivity(bundle)); + } + }) + ); + topStringBuilder.append(AndroidUtilities.replaceCharSequence("%1$d", subTitleWithLink, replaceTags("**" + giveaway.winners_count + "**"))); + + topStringBuilder.append("\n\n"); + topStringBuilder.setSpan(new RelativeSizeSpan(0.4f), topStringBuilder.length() - 1, topStringBuilder.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + + CharSequence winners = replaceTags(getPluralString("BoostingGiveawayResultsMsgWinners", giveaway.winners_count)); + topStringBuilder.append(winners); + topStringBuilder.setSpan(new RelativeSizeSpan(1.05f), subTitleWithLink.length() + 2, subTitleWithLink.length() + 2 + winners.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + + SpannableStringBuilder bottomStringBuilder = new SpannableStringBuilder(); + if (giveaway.winners_count != giveaway.winners.size()) { + bottomStringBuilder.append(replaceTags(formatPluralString("BoostingGiveawayResultsMsgAllAndMoreWinners", giveaway.winners_count - giveaway.winners.size()))); + bottomStringBuilder.setSpan(new RelativeSizeSpan(1.05f), 0, bottomStringBuilder.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + bottomStringBuilder.append("\n"); + } + bottomStringBuilder.append(LocaleController.getString("BoostingGiveawayResultsMsgAllWinnersReceivedLinks", R.string.BoostingGiveawayResultsMsgAllWinnersReceivedLinks)); + + titleLayout = StaticLayoutEx.createStaticLayout(titleStringBuilder, textPaint, maxWidth, Layout.Alignment.ALIGN_CENTER, 1.0f, AndroidUtilities.dp(2), false, TextUtils.TruncateAt.END, maxWidth, 10); + topLayout = StaticLayoutEx.createStaticLayout(topStringBuilder, textPaint, maxWidth, Layout.Alignment.ALIGN_CENTER, 1.0f, AndroidUtilities.dp(2), false, TextUtils.TruncateAt.END, maxWidth, 10); + bottomLayout = StaticLayoutEx.createStaticLayout(bottomStringBuilder, textPaint, maxWidth, Layout.Alignment.ALIGN_CENTER, 1.0f, AndroidUtilities.dp(3), false, TextUtils.TruncateAt.END, maxWidth, 10); + + int oldMaxWidth = maxWidth; + maxWidth = Math.max(forwardedNameWidth, maxWidth); + diffTextWidth = maxWidth - oldMaxWidth; + + giftReceiver.setImageCoords((maxWidth / 2f) - (giftSize / 2f), AndroidUtilities.dp(70) - giftSize / 2f, giftSize, giftSize); + + titleHeight = titleLayout.getLineBottom(titleLayout.getLineCount() - 1) + dp(5); + topHeight = titleHeight + topLayout.getLineBottom(topLayout.getLineCount() - 1); + bottomHeight = bottomLayout.getLineBottom(bottomLayout.getLineCount() - 1); + countriesHeight = countriesLayout != null ? (countriesLayout.getLineBottom(countriesLayout.getLineCount() - 1) + dp(4 + 8)) : 0; + + measuredHeight += topHeight; + measuredHeight += countriesHeight; + measuredHeight += bottomHeight; + measuredHeight += dp(32 + 96); //gift + measuredWidth = maxWidth; + + counterStr = "x" + giveaway.winners_count; + counterTextPaint.getTextBounds(counterStr, 0, counterStr.length(), counterTextBounds); + + Arrays.fill(avatarVisible, false); + + float oneRowTotalWidth = 0; + measuredHeight += dp(24 + 6); + + List visibleChannels = new ArrayList<>(giveaway.winners.size()); + for (Long uid : giveaway.winners) { + TLRPC.User user = MessagesController.getInstance(UserConfig.selectedAccount).getUser(uid); + if (user != null) { + visibleChannels.add(uid); + } + } + + for (int i = 0; i < visibleChannels.size(); i++) { + long uid = visibleChannels.get(i); + TLRPC.User user = MessagesController.getInstance(UserConfig.selectedAccount).getUser(uid); + + if (user != null) { + avatarVisible[i] = true; + users[i] = user; + CharSequence text = Emoji.replaceEmoji(UserObject.getUserName(user), chatTextPaint.getFontMetricsInt(), false); + userTitles[i] = TextUtils.ellipsize(text, chatTextPaint, maxWidth * 0.8f, TextUtils.TruncateAt.END); + userTitleWidths[i] = chatTextPaint.measureText(userTitles[i], 0, userTitles[i].length()); + float oneRowWidth = userTitleWidths[i] + dp(24 + 6 + 10); + oneRowTotalWidth += oneRowWidth; + if (i > 0) { + needNewRow[i] = oneRowTotalWidth > maxWidth * 0.9f; + if (needNewRow[i]) { + oneRowTotalWidth = oneRowWidth; + measuredHeight += dp(24 + 6); + } + } else { + needNewRow[i] = false; + } + avatarDrawables[i].setInfo(user); + avatarImageReceivers[i].setForUserOrChat(user, avatarDrawables[i]); + avatarImageReceivers[i].setImageCoords(0, 0, dp(24), dp(24)); + } else { + users[i] = null; + avatarVisible[i] = false; + userTitles[i] = ""; + needNewRow[i] = false; + userTitleWidths[i] = dp(20); + avatarDrawables[i].setInfo(uid, "", ""); + } + } + } + + private int getUserColor(TLRPC.User user, Theme.ResourcesProvider resourcesProvider) { + if (messageObject.isOutOwner()) { + return Theme.getColor(Theme.key_chat_outPreviewInstantText, resourcesProvider); + } + final int color; + int colorId = UserObject.getColorId(user); + if (colorId < 7) { + color = Theme.getColor(Theme.keys_avatar_nameInMessage[colorId], resourcesProvider); + } else { + MessagesController.PeerColors peerColors = MessagesController.getInstance(UserConfig.selectedAccount).peerColors; + MessagesController.PeerColor peerColor = peerColors == null ? null : peerColors.getColor(colorId); + if (peerColor != null) { + color = peerColor.getColor1(); + } else { + color = Theme.getColor(Theme.keys_avatar_nameInMessage[0], resourcesProvider); + } + } + return color; + } + + public void draw(Canvas canvas, int marginTop, int marginLeft, Theme.ResourcesProvider resourcesProvider) { + if (messageObject == null || !messageObject.isGiveawayResults()) { + return; + } + + if (selectorDrawable == null) { + selectorDrawable = Theme.createRadSelectorDrawable(selectorColor = Theme.getColor(Theme.key_listSelector), 12, 12); + } + + textPaint.setColor(Theme.chat_msgTextPaint.getColor()); + textDividerPaint.setColor(Theme.getColor(Theme.key_dialogTextGray2)); + countriesTextPaint.setColor(Theme.chat_msgTextPaint.getColor()); + + if (messageObject.isOutOwner()) { + chatTextPaint.setColor(Theme.getColor(Theme.key_chat_outPreviewInstantText, resourcesProvider)); + counterBgPaint.setColor(Theme.getColor(Theme.key_chat_outPreviewInstantText, resourcesProvider)); + chatBgPaint.setColor(Theme.getColor(Theme.key_chat_outReplyLine, resourcesProvider)); + } else { + chatTextPaint.setColor(Theme.getColor(Theme.key_chat_inPreviewInstantText, resourcesProvider)); + counterBgPaint.setColor(Theme.getColor(Theme.key_chat_inPreviewInstantText, resourcesProvider)); + chatBgPaint.setColor(Theme.getColor(Theme.key_chat_inReplyLine, resourcesProvider)); + } + + int x = 0, y = 0; + canvas.save(); + x = marginLeft - dp(4); + y = marginTop; + canvas.translate(x, y); + containerRect.set(x, y, getMeasuredWidth() + x, getMeasuredHeight() + y); + + canvas.saveLayer(0, 0, getMeasuredWidth(), getMeasuredHeight(), saveLayerPaint, Canvas.ALL_SAVE_FLAG); + giftReceiver.draw(canvas); + + float centerX = getMeasuredWidth() / 2f; + float centerY = dp(106); + int textWidth = counterTextBounds.width() + dp(12); + int textHeight = counterTextBounds.height() + dp(10); + countRect.set( + centerX - ((textWidth + dp(2)) / 2f), + centerY - ((textHeight + dp(2)) / 2f), + centerX + ((textWidth + dp(2)) / 2f), + centerY + ((textHeight + dp(2)) / 2f) + ); + canvas.drawRoundRect(countRect, dp(11), dp(11), clipRectPaint); + countRect.set( + centerX - ((textWidth) / 2f), + centerY - ((textHeight) / 2f), + centerX + ((textWidth) / 2f), + centerY + ((textHeight) / 2f) + ); + canvas.drawRoundRect(countRect, dp(10), dp(10), counterBgPaint); + canvas.drawText(counterStr, countRect.centerX(), countRect.centerY() + dp(4), counterTextPaint); + canvas.restore(); + + canvas.translate(0, dp(32 + 96)); + y += dp(32 + 96); + subTitleMarginTop = y + titleHeight; + subTitleMarginLeft = (int) (x + diffTextWidth / 2f); + + canvas.save(); + canvas.translate(diffTextWidth / 2f, 0); + titleLayout.draw(canvas); + canvas.translate(0, titleHeight); + + topLayout.draw(canvas); + canvas.restore(); + canvas.translate(0, topHeight + dp(6)); + y += topHeight + dp(6); + + int selectedChatColor = 0; + int i = 0; + while (i < avatarVisible.length) { + if (avatarVisible[i]) { + canvas.save(); + int k = i; + float rowWidth = 0; + + do { + rowWidth += userTitleWidths[k] + dp(24 + 6 + 10); + k++; + } while (k < avatarVisible.length && !needNewRow[k] && avatarVisible[k]); + + float marginItemsLeft = centerX - (rowWidth / 2f); + canvas.translate(marginItemsLeft, 0); + int xRow = x + (int) (marginItemsLeft); + + k = i; + + do { + int chatColor = getUserColor(users[k], resourcesProvider); + if (pressedPos >= 0 && pressedPos == k) { + selectedChatColor = chatColor; + } + chatTextPaint.setColor(chatColor); + chatBgPaint.setColor(chatColor); + chatBgPaint.setAlpha(25); + avatarImageReceivers[k].draw(canvas); + canvas.drawText(userTitles[k], 0, userTitles[k].length(), dp(24 + 6), dp(16), chatTextPaint); + chatRect.set(0, 0, userTitleWidths[k] + dp(24 + 6 + 10), dp(24)); + canvas.drawRoundRect(chatRect, dp(12), dp(12), chatBgPaint); + + clickRect[k].set(xRow, y, (int) (xRow + chatRect.width()), y + dp(24)); + + canvas.translate(chatRect.width() + dp(6), 0); + xRow += chatRect.width() + dp(6); + + k++; + } while (k < avatarVisible.length && !needNewRow[k] && avatarVisible[k]); + + i = k; + + canvas.restore(); + canvas.translate(0, dp(24 + 6)); + y += dp(24 + 6); + } else { + i++; + } + } + + if (countriesLayout != null) { + canvas.save(); + canvas.translate((measuredWidth - countriesLayout.getWidth()) / 2f, dp(4)); + countriesLayout.draw(canvas); + canvas.restore(); + canvas.translate(0, countriesHeight); + } + + canvas.translate(0, dp(6)); + canvas.save(); + canvas.translate(diffTextWidth / 2f, 0); + bottomLayout.draw(canvas); + canvas.restore(); + canvas.restore(); + if (pressedPos >= 0) { + int rippleColor = Theme.multAlpha(selectedChatColor, Theme.isCurrentThemeDark() ? 0.12f : 0.10f); + if (selectorColor != rippleColor) { + Theme.setSelectorDrawableColor(selectorDrawable, selectorColor = rippleColor, true); + } + selectorDrawable.setBounds(clickRect[pressedPos]); + selectorDrawable.draw(canvas); + } + + if (links != null && links.draw(canvas)) { + parentView.invalidate(); + } + } + + public void onDetachedFromWindow() { + if (giftReceiver != null) { + giftReceiver.onDetachedFromWindow(); + } + if (avatarImageReceivers != null) { + for (ImageReceiver avatarImageReceiver : avatarImageReceivers) { + avatarImageReceiver.onDetachedFromWindow(); + } + } + } + + public void onAttachedToWindow() { + if (giftReceiver != null) { + giftReceiver.onAttachedToWindow(); + } + if (avatarImageReceivers != null) { + for (ImageReceiver avatarImageReceiver : avatarImageReceivers) { + avatarImageReceiver.onAttachedToWindow(); + } + } + } + + public int getMeasuredHeight() { + return measuredHeight; + } + + public int getMeasuredWidth() { + return measuredWidth; + } + + private void createImages() { + if (avatarImageReceivers != null) { + return; + } + + avatarImageReceivers = new ImageReceiver[10]; + avatarDrawables = new AvatarDrawable[10]; + avatarVisible = new boolean[10]; + for (int a = 0; a < avatarImageReceivers.length; a++) { + avatarImageReceivers[a] = new ImageReceiver(parentView); + avatarImageReceivers[a].setAllowLoadingOnAttachedOnly(true); + avatarImageReceivers[a].setRoundRadius(AndroidUtilities.dp(12)); + avatarDrawables[a] = new AvatarDrawable(); + avatarDrawables[a].setTextSize(AndroidUtilities.dp(18)); + clickRect[a] = new Rect(); + } + } + + private void checkArraysLimits(int channelsCount) { + if (avatarImageReceivers.length < channelsCount) { + int oldLength = avatarImageReceivers.length; + avatarImageReceivers = Arrays.copyOf(avatarImageReceivers, channelsCount); + avatarDrawables = Arrays.copyOf(avatarDrawables, channelsCount); + avatarVisible = Arrays.copyOf(avatarVisible, channelsCount); + userTitles = Arrays.copyOf(userTitles, channelsCount); + userTitleWidths = Arrays.copyOf(userTitleWidths, channelsCount); + needNewRow = Arrays.copyOf(needNewRow, channelsCount); + clickRect = Arrays.copyOf(clickRect, channelsCount); + users = Arrays.copyOf(users, channelsCount); + + for (int i = oldLength - 1; i < channelsCount; i++) { + avatarImageReceivers[i] = new ImageReceiver(parentView); + avatarImageReceivers[i].setAllowLoadingOnAttachedOnly(true); + avatarImageReceivers[i].setRoundRadius(AndroidUtilities.dp(12)); + avatarDrawables[i] = new AvatarDrawable(); + avatarDrawables[i].setTextSize(AndroidUtilities.dp(18)); + clickRect[i] = new Rect(); + } + } + } + + private void setGiftImage() { + giftReceiver.setAllowStartLottieAnimation(false); + if (giftDrawable == null) { + giftDrawable = new RLottieDrawable(R.raw.giveaway_results, "" + R.raw.giveaway_results, dp(120), dp(120)); + } + giftReceiver.setImageBitmap(giftDrawable); + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/selector/SelectorHeaderCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/selector/SelectorHeaderCell.java index 469f7b6093e..01a2598c59f 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/selector/SelectorHeaderCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/selector/SelectorHeaderCell.java @@ -1,5 +1,7 @@ package org.telegram.ui.Components.Premium.boosts.cells.selector; +import static org.telegram.messenger.AndroidUtilities.dp; + import android.annotation.SuppressLint; import android.content.Context; import android.graphics.Canvas; @@ -81,7 +83,11 @@ public void setOnCloseClickListener(Runnable onCloseClickListener) { protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { super.onMeasure( MeasureSpec.makeMeasureSpec(MeasureSpec.getSize(widthMeasureSpec), MeasureSpec.EXACTLY), - MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(56), MeasureSpec.EXACTLY) + MeasureSpec.makeMeasureSpec(getHeaderHeight(), MeasureSpec.EXACTLY) ); } + + protected int getHeaderHeight() { + return dp(56); + } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/selector/SelectorLetterCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/selector/SelectorLetterCell.java index d5448547c34..1785e1abbba 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/selector/SelectorLetterCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/selector/SelectorLetterCell.java @@ -33,7 +33,7 @@ public SelectorLetterCell(@NonNull Context context, Theme.ResourcesProvider reso } public void setLetter(String letter) { - textView.setText(letter.toUpperCase()); + textView.setText(letter); } private int getThemedColor(int key) { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/selector/SelectorSearchCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/selector/SelectorSearchCell.java index dec75940863..197eb665f40 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/selector/SelectorSearchCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Premium/boosts/cells/selector/SelectorSearchCell.java @@ -140,6 +140,10 @@ public void afterTextChanged(Editable s) { }); } + public void setHintText(String text, boolean animated) { + editText.setHintText(text, animated); + } + private final AnimatedFloat topGradientAlpha = new AnimatedFloat(this, 0, 300, CubicBezierInterpolator.EASE_OUT_QUINT); private final LinearGradient topGradient = new LinearGradient(0, 0, 0, dp(8), new int[]{0xff000000, 0x00000000}, new float[]{0, 1}, Shader.TileMode.CLAMP); private final Paint topGradientPaint = new Paint(Paint.ANTI_ALIAS_FLAG); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/RadialProgress2.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/RadialProgress2.java index 0a219c4f380..011d52bf423 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/RadialProgress2.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/RadialProgress2.java @@ -26,7 +26,7 @@ public class RadialProgress2 { - private RectF progressRect = new RectF(); + public RectF progressRect = new RectF(); private View parent; private boolean previousCheckDrawable; @@ -36,21 +36,21 @@ public class RadialProgress2 { private Paint miniProgressBackgroundPaint; private Paint overlayPaint = new Paint(Paint.ANTI_ALIAS_FLAG); - private Paint circlePaint = new Paint(Paint.ANTI_ALIAS_FLAG); + public Paint circlePaint = new Paint(Paint.ANTI_ALIAS_FLAG); private Paint circleMiniPaint = new Paint(Paint.ANTI_ALIAS_FLAG); - private MediaActionDrawable mediaActionDrawable; + public MediaActionDrawable mediaActionDrawable; private MediaActionDrawable miniMediaActionDrawable; private float miniIconScale = 1.0f; private int circleColor; private int circlePressedColor; - private int iconColor; + public int iconColor; private int iconPressedColor; private int circleColorKey = -1; private int circleCrossfadeColorKey = -1; private float circleCrossfadeColorProgress; private float circleCheckProgress = 1.0f; private int circlePressedColorKey = -1; - private int iconColorKey = -1; + public int iconColorKey = -1; private int iconPressedColorKey = -1; private ImageReceiver overlayImageView; private int circleRadius; @@ -70,6 +70,8 @@ public class RadialProgress2 { private int maxIconSize; private float overlayImageAlpha = 1f; + public float iconScale = 1f; + public RadialProgress2(View parentView) { this(parentView, null); } @@ -452,6 +454,10 @@ public void draw(Canvas canvas) { if (maxIconSize > 0 && iconSize > maxIconSize) { iconSize = maxIconSize; } + if (iconScale != 1f) { + canvas.save(); + canvas.scale(iconScale, iconScale, centerX, centerY); + } mediaActionDrawable.setBounds(centerX - iconSize, centerY - iconSize, centerX + iconSize, centerY + iconSize); mediaActionDrawable.setHasOverlayImage(overlayImageView.hasBitmapImage()); if ((drawMiniIcon || circleCrossfadeColorKey >= 0)) { @@ -522,6 +528,9 @@ public void draw(Canvas canvas) { canvas.restoreToCount(restore); } } + if (iconScale != 1f) { + canvas.restore(); + } } public int getCircleColorKey() { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/ReactedUsersListView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/ReactedUsersListView.java index 9e18824f4e9..594a3b2b1c0 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/ReactedUsersListView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/ReactedUsersListView.java @@ -134,7 +134,7 @@ public void onBindViewHolder(@NonNull RecyclerView.ViewHolder holder, int positi @Override public int getItemCount() { - return userReactions.size() + (!customReactionsEmoji.isEmpty() && !MessagesController.getInstance(currentAccount).premiumLocked ? 1 : 0); + return userReactions.size() + (!customReactionsEmoji.isEmpty() && !MessagesController.getInstance(currentAccount).premiumFeaturesBlocked() ? 1 : 0); } @Override @@ -180,7 +180,7 @@ public int getAdditionalHeight() { loadingView.setIsSingleCell(true); loadingView.setItemsCount(predictiveCount); addView(loadingView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT)); - if (!addPadding && filter != null && filter instanceof TLRPC.TL_reactionCustomEmoji && !MessagesController.getInstance(currentAccount).premiumLocked) { + if (!addPadding && filter != null && filter instanceof TLRPC.TL_reactionCustomEmoji && !MessagesController.getInstance(currentAccount).premiumFeaturesBlocked()) { customReactionsEmoji.clear(); customReactionsEmoji.add(ReactionsLayoutInBubble.VisibleReaction.fromTLReaction(filter)); updateCustomReactionsButton(); @@ -346,7 +346,7 @@ private void updateCustomReactionsButton() { setIds.add(stickerSet.id); } } - if (MessagesController.getInstance(currentAccount).premiumLocked) { + if (MessagesController.getInstance(currentAccount).premiumFeaturesBlocked()) { return; } customEmojiStickerSets.addAll(sets); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/Reactions/ChatCustomReactionsEditActivity.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/Reactions/ChatCustomReactionsEditActivity.java index 8d27acfc481..739b4193ead 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/Reactions/ChatCustomReactionsEditActivity.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/Reactions/ChatCustomReactionsEditActivity.java @@ -708,12 +708,14 @@ private boolean closeKeyboard() { emojiKeyboardVisible = false; editText.clearFocus(); updateScrollViewMarginBottom(0); + NotificationCenter.getGlobalInstance().postNotificationName(NotificationCenter.stopAllHeavyOperations, 512); bottomDialogLayout.animate().setListener(null).cancel(); bottomDialogLayout.animate().translationY(bottomDialogLayout.getMeasuredHeight()).setDuration(350).withLayer().setInterpolator(CubicBezierInterpolator.DEFAULT).setUpdateListener(animation -> { actionButton.setTranslationY(-(1f - (float) animation.getAnimatedValue()) * bottomDialogLayout.getMeasuredHeight()); }).setListener(new AnimatorListenerAdapter() { @Override public void onAnimationEnd(Animator animation) { + NotificationCenter.getGlobalInstance().postNotificationName(NotificationCenter.startAllHeavyOperations, 512); bottomDialogLayout.setVisibility(View.INVISIBLE); } }).start(); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/ReactionsContainerLayout.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/ReactionsContainerLayout.java index 95846a56745..0c3c43dc4f9 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/ReactionsContainerLayout.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/ReactionsContainerLayout.java @@ -567,11 +567,11 @@ private void invalidateLoopViews() { } public boolean showCustomEmojiReaction() { - return (!MessagesController.getInstance(currentAccount).premiumLocked && allReactionsAvailable) || showExpandableReactions; + return allReactionsAvailable || showExpandableReactions; } private boolean showUnlockPremiumButton() { - return !premiumLockedReactions.isEmpty() && !MessagesController.getInstance(currentAccount).premiumLocked; + return !premiumLockedReactions.isEmpty() && !MessagesController.getInstance(currentAccount).premiumFeaturesBlocked(); } private void showUnlockPremium(float x, float y) { @@ -1085,7 +1085,7 @@ public void setMessage(MessageObject message, TLRPC.ChatFull chatFull) { fillRecentReactionsList(visibleReactions); } filterReactions(visibleReactions); - showExpandableReactions = !allReactionsAvailable && visibleReactions.size() > 16; + showExpandableReactions = !allReactionsAvailable && visibleReactions.size() > 16 || allReactionsAvailable && !UserConfig.getInstance(currentAccount).isPremium() && MessagesController.getInstance(currentAccount).premiumFeaturesBlocked(); setVisibleReactionsList(visibleReactions); if (message != null && message.messageOwner.reactions != null && message.messageOwner.reactions.results != null) { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/RecyclerListView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/RecyclerListView.java index 0a1d27176d3..f07567be2f9 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/RecyclerListView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/RecyclerListView.java @@ -59,6 +59,8 @@ import org.telegram.messenger.R; import org.telegram.messenger.SharedConfig; import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.Cells.ChatActionCell; +import org.telegram.ui.Cells.ChatMessageCell; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; @@ -1256,6 +1258,7 @@ public View findChildViewUnder(float x, float y) { for (int a = 0; a < 2; a++) { for (int i = count - 1; i >= 0; i--) { final View child = getChildAt(i); + if ((child instanceof ChatMessageCell || child instanceof ChatActionCell) && child.getVisibility() == View.INVISIBLE) continue; final float translationX = a == 0 ? child.getTranslationX() : 0; final float translationY = a == 0 ? child.getTranslationY() : 0; if (x >= child.getLeft() + translationX diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/ReplyMessageLine.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/ReplyMessageLine.java index e5bf8e323c7..bc2e55cc3d9 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/ReplyMessageLine.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/ReplyMessageLine.java @@ -52,7 +52,7 @@ public class ReplyMessageLine { private AnimatedEmojiDrawable.SwapAnimatedEmojiDrawable emoji; private long emojiDocumentId; - private int backgroundColor, nameColor, color1, color2, color3; + public int backgroundColor, nameColor, color1, color2, color3; private final View parentView; public final AnimatedColor backgroundColorAnimated; public final AnimatedColor color1Animated, color2Animated, color3Animated; @@ -129,9 +129,9 @@ private void resolveColor(MessageObject messageObject, int colorId, Theme.Resour hasColor2 = hasColor3 = false; return; } - color1 = peerColor.getColor1(dark); - color2 = peerColor.getColor2(dark); - color3 = peerColor.getColor3(dark); + color1 = peerColor.getColor(0, resourcesProvider); + color2 = peerColor.getColor(1, resourcesProvider); + color3 = peerColor.getColor(2, resourcesProvider); hasColor2 = color2 != color1; hasColor3 = color3 != color1; if (hasColor3) { @@ -170,8 +170,14 @@ public int check(MessageObject messageObject, TLRPC.User currentUser, TLRPC.Chat colorId = messageObject.overrideLinkColor; } else if (messageObject.isSponsored() && messageObject.sponsoredChatInvite instanceof TLRPC.TL_chatInvite) { colorId = messageObject.sponsoredChatInvite.color; + if (type == TYPE_LINK && messageObject.sponsoredChatInvite.chat != null) { + emojiDocumentId = ChatObject.getEmojiId(messageObject.sponsoredChatInvite.chat); + } } else if (messageObject.isSponsored() && messageObject.sponsoredChatInvite != null && messageObject.sponsoredChatInvite.chat != null) { colorId = ChatObject.getColorId(messageObject.sponsoredChatInvite.chat); + if (type == TYPE_LINK) { + emojiDocumentId = ChatObject.getEmojiId(messageObject.sponsoredChatInvite.chat); + } } else if (messageObject.messageOwner != null && messageObject.messageOwner.fwd_from != null && messageObject.messageOwner.fwd_from.from_id != null) { long dialogId = DialogObject.getPeerDialogId(messageObject.messageOwner.fwd_from.from_id); if (dialogId < 0) { @@ -179,20 +185,35 @@ public int check(MessageObject messageObject, TLRPC.User currentUser, TLRPC.Chat if (chat != null) { colorId = ChatObject.getColorId(chat); } + if (type == TYPE_LINK) { + emojiDocumentId = ChatObject.getEmojiId(chat); + } } else { TLRPC.User user = MessagesController.getInstance(messageObject.currentAccount).getUser(dialogId); if (user != null) { colorId = UserObject.getColorId(user); } + if (type == TYPE_LINK) { + emojiDocumentId = UserObject.getEmojiId(user); + } } } else if (DialogObject.isEncryptedDialog(messageObject.getDialogId()) && currentUser != null) { TLRPC.User user = messageObject.isOutOwner() ? UserConfig.getInstance(messageObject.currentAccount).getCurrentUser() : currentUser; if (user == null) user = currentUser; colorId = UserObject.getColorId(user); + if (type == TYPE_LINK) { + emojiDocumentId = UserObject.getEmojiId(user); + } } else if (messageObject.isFromUser() && currentUser != null) { colorId = UserObject.getColorId(currentUser); + if (type == TYPE_LINK) { + emojiDocumentId = UserObject.getEmojiId(currentUser); + } } else if (messageObject.isFromChannel() && currentChat != null) { colorId = ChatObject.getColorId(currentChat); + if (type == TYPE_LINK) { + emojiDocumentId = ChatObject.getEmojiId(currentChat); + } } else { colorId = 0; } @@ -256,8 +277,12 @@ public int check(MessageObject messageObject, TLRPC.User currentUser, TLRPC.Chat color1 = color2 = color3 = Color.WHITE; backgroundColor = Color.TRANSPARENT; nameColor = Theme.getColor(Theme.key_chat_stickerReplyNameText, resourcesProvider); - } else if (messageObject.isOutOwner()) { - color1 = color2 = color3 = Theme.getColor(hasColor2 || hasColor3 ? Theme.key_chat_outReplyLine2 : Theme.key_chat_outReplyLine, resourcesProvider); + } else if (messageObject.isOutOwner() || type == TYPE_CODE) { + if (type == TYPE_CODE && !messageObject.isOutOwner()) { + color1 = color2 = color3 = Theme.getColor(Theme.key_chat_inCodeBackground, resourcesProvider); + } else { + color1 = color2 = color3 = Theme.getColor(hasColor2 || hasColor3 ? Theme.key_chat_outReplyLine2 : Theme.key_chat_outReplyLine, resourcesProvider); + } if (hasColor3) { reversedOut = true; color1 = Theme.multAlpha(color1, .20f); @@ -269,7 +294,7 @@ public int check(MessageObject messageObject, TLRPC.User currentUser, TLRPC.Chat backgroundColor = Theme.multAlpha(color3, dark ? 0.12f : 0.10f); nameColor = Theme.getColor(Theme.key_chat_outReplyNameText, resourcesProvider); } - if (type == TYPE_REPLY && messageObject != null && messageObject.overrideLinkEmoji != -1) { + if ((type == TYPE_REPLY || type == TYPE_LINK) && messageObject != null && messageObject.overrideLinkEmoji != -1) { emojiDocumentId = messageObject.overrideLinkEmoji; } if (emojiDocumentId != 0 && emoji == null) { @@ -445,6 +470,13 @@ public void drawBackground(Canvas canvas, RectF rect, float alpha) { drawBackground(canvas, rect, alpha, false, false); } + private float emojiOffsetX, emojiOffsetY; + public ReplyMessageLine offsetEmoji(float ox, float oy) { + this.emojiOffsetX = ox; + this.emojiOffsetY = oy; + return this; + } + public void drawBackground(Canvas canvas, RectF rect, float alpha, boolean hasQuote, boolean emojiOnly) { if (!emojiOnly) { backgroundPath.rewind(); @@ -465,18 +497,19 @@ public void drawBackground(Canvas canvas, RectF rect, float alpha, boolean hasQu new IconCoords(30, 3, .78f, .9f), new IconCoords(46, -17, .6f, .6f), new IconCoords(69.66f, -0.666f, .87f, .7f), - new IconCoords(107, -12.6f, 1.03f, .3f), + new IconCoords(98, -12.6f, 1.03f, .3f), new IconCoords(51, 24, 1f, .5f), new IconCoords(6.33f, 20, .77f, .7f), new IconCoords(-19, 12, .8f, .6f, true), - new IconCoords(26, 42, .78f, .9f), +// new IconCoords(26, 42, .78f, .9f), new IconCoords(-22, 36, .7f, .5f, true), - new IconCoords(-1, 48, 1f, .4f), +// new IconCoords(-1, 48, 1f, .4f), }; } canvas.save(); canvas.clipRect(rect); + canvas.translate(emojiOffsetX, emojiOffsetY); float x0 = Math.max(rect.right - dp(15), rect.centerX()); if (hasQuote) { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/SearchViewPager.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/SearchViewPager.java index 657d457f20c..7494d5536ed 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/SearchViewPager.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/SearchViewPager.java @@ -465,7 +465,7 @@ private void showActionMode(boolean show) { } private boolean isSpeedItemVisible() { - if (UserConfig.getInstance(currentAccount).isPremium() || MessagesController.getInstance(currentAccount).premiumLocked) { + if (UserConfig.getInstance(currentAccount).isPremium() || MessagesController.getInstance(currentAccount).premiumFeaturesBlocked()) { return false; } for (MessageObject obj : selectedFiles.values()) { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/SeekBar.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/SeekBar.java index 7f097e14269..ddd1b0cad0c 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/SeekBar.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/SeekBar.java @@ -39,9 +39,16 @@ public interface SeekBarDelegate { default void onSeekBarContinuousDrag(float progress) { } - default void onSeekBarPressed() {} default void onSeekBarReleased() {} + + default boolean isSeekBarDragAllowed() { + return true; + } + + default boolean reverseWaveform() { + return false; + } } private static Paint paint; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/SeekBarWaveform.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/SeekBarWaveform.java index ce66c1e6e79..9bd9c1b7020 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/SeekBarWaveform.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/SeekBarWaveform.java @@ -8,11 +8,14 @@ package org.telegram.ui.Components; +import static org.telegram.messenger.AndroidUtilities.dp; + import android.content.Context; import android.graphics.Canvas; import android.graphics.LinearGradient; import android.graphics.Paint; import android.graphics.Path; +import android.graphics.RectF; import android.graphics.Shader; import android.os.SystemClock; import android.util.Log; @@ -25,6 +28,8 @@ import org.telegram.messenger.AndroidUtilities; import org.telegram.messenger.MediaController; import org.telegram.messenger.MessageObject; +import org.telegram.messenger.Utilities; +import org.telegram.ui.ActionBar.Theme; import java.util.ArrayList; @@ -94,6 +99,9 @@ public void setColors(int inner, int outer, int selected) { public void setWaveform(byte[] waveform) { waveformBytes = waveform; heights = calculateHeights((int) (width / AndroidUtilities.dpf2(3))); + if (!delegate.isSeekBarDragAllowed()) { + this.progress = 1f; + } } public void setSelected(boolean value) { @@ -117,11 +125,21 @@ public void setParentView(View view) { appearFloat.setParent(view); } + public void invalidate() { + if (parentView != null) { + parentView.invalidate(); + } + } + public boolean isStartDraging() { return startDraging; } public boolean onTouch(int action, float x, float y) { + if (!delegate.isSeekBarDragAllowed()) { + this.progress = 1f; + return false; + } if (action == MotionEvent.ACTION_DOWN) { if (0 <= x && x <= width && y >= 0 && y <= height) { startX = x; @@ -173,6 +191,10 @@ public void setProgress(float progress) { } public void setProgress(float progress, boolean animated) { + if (!delegate.isSeekBarDragAllowed()) { + this.progress = 1f; + return; + } this.progress = isUnread ? 1f : progress; int currentThumbX = isUnread ? width : thumbX; if (animated && currentThumbX != 0 && progress == 0) { @@ -267,6 +289,28 @@ private float[] calculateHeights(int count) { return heights; } + private boolean exploding = false; + public float explodeProgress; + public void explodeAt(float progress) { + exploding = true; + explodeProgress = progress; + invalidate(); + } + public float explosionRate; + public void setExplosionRate(float explosionRate) { + this.explosionRate = explosionRate; + invalidate(); + } + public void stopExploding() { + exploding = false; + if (particles != null) { + particles.clear(); + } + invalidate(); + } + + private Particles particles; + public void draw(Canvas canvas, View parentView) { if (waveformBytes == null || width == 0 || alpha <= 0) { return; @@ -299,6 +343,7 @@ public void draw(Canvas canvas, View parentView) { alphaPath.reset(); } + final boolean reverse = delegate != null && delegate.reverseWaveform(); if (fromHeights != null && toHeights != null) { float t = (width - fromWidth) / (float) (toWidth - fromWidth); int maxlen = Math.max(fromHeights.length, toHeights.length); @@ -312,12 +357,12 @@ public void draw(Canvas canvas, View parentView) { int l = MathUtils.clamp((int) Math.floor(barNum / (float) maxlen * minlen), 0, minlen - 1); if (k < l) { float x = AndroidUtilities.lerp((float) l, (float) barNum, T) * AndroidUtilities.dpf2(3); - float h = AndroidUtilities.dpf2(AndroidUtilities.lerp(minarr[l], maxarr[barNum], T)); + float h = AndroidUtilities.dpf2(AndroidUtilities.lerp(minarr[reverse ? minarr.length - 1 - l : l], maxarr[reverse ? maxarr.length - 1 - barNum : barNum], T)); addBar(path, x, h); k = l; } else { float x = AndroidUtilities.lerp((float) l, (float) barNum, T) * AndroidUtilities.dpf2(3); - float h = AndroidUtilities.dpf2(AndroidUtilities.lerp(minarr[l], maxarr[barNum], T)); + float h = AndroidUtilities.dpf2(AndroidUtilities.lerp(minarr[reverse ? minarr.length - 1 - l : l], maxarr[reverse ? maxarr.length - 1 - barNum : barNum], T)); addBar(alphaPath, x, h); alpha = T; } @@ -329,12 +374,18 @@ public void draw(Canvas canvas, View parentView) { } float x = barNum * AndroidUtilities.dpf2(3); float bart = MathUtils.clamp(appearProgress * totalBarsCount - barNum, 0, 1); - float h = AndroidUtilities.dpf2(heights[barNum]) * bart; + float h = AndroidUtilities.dpf2(heights[reverse ? heights.length - 1 - barNum : barNum]) * bart; h -= AndroidUtilities.dpf2(1) * (1f - bart); addBar(path, x, h); } } + if (exploding || explosionRate > 0) { + canvas.save(); + final float w = totalBarsCount * AndroidUtilities.dpf2(3); + canvas.clipRect(0, 0, w * (1f - explodeProgress * explosionRate), height); + } + if (alpha > 0) { canvas.save(); canvas.clipPath(alphaPath); @@ -346,6 +397,39 @@ public void draw(Canvas canvas, View parentView) { canvas.clipPath(path); drawFill(canvas, this.alpha); canvas.restore(); + + if (exploding || explosionRate > 0) { + canvas.restore(); + if (particles == null) { + particles = new Particles(250, this::invalidate); + } + RectF emitArea = null; + if (explodeProgress < .99f && heights != null) { + int barNum = (int) (totalBarsCount * (1f - explodeProgress)); + if (reverse) { + barNum = (int) (totalBarsCount - 1 - barNum); + } + if (barNum >= 0 && barNum < heights.length) { + float bart = MathUtils.clamp(appearProgress * totalBarsCount - barNum, 0, 1); + float h = AndroidUtilities.dpf2(heights[barNum]) * bart; + emitArea = AndroidUtilities.rectTmp; + final float x = (totalBarsCount * (1f - explodeProgress)) * AndroidUtilities.dpf2(3); + final float strokeWidth = AndroidUtilities.dpf2(2); + final int y = (height - dp(14)) / 2; + h *= waveScaling; + AndroidUtilities.rectTmp.set( + x + AndroidUtilities.dpf2(1) - strokeWidth / 2f, + y + dp(7) + (-h - strokeWidth / 2f), + x + AndroidUtilities.dpf2(1) + strokeWidth / 2f, + y + dp(7) + (h + strokeWidth / 2f) + ); + } + } + particles + .setColor(outerColor) + .setEmitArea(emitArea) + .draw(canvas, explosionRate); + } } private void drawFill(Canvas canvas, float alpha) { @@ -368,7 +452,7 @@ private void drawFill(Canvas canvas, float alpha) { } if (loadingT > 0f) { - if (loadingPaint == null || Math.abs(loadingPaintWidth - width) > AndroidUtilities.dp(8) || loadingPaintColor1 != innerColor || loadingPaintColor2 != outerColor) { + if (loadingPaint == null || Math.abs(loadingPaintWidth - width) > dp(8) || loadingPaintColor1 != innerColor || loadingPaintColor2 != outerColor) { if (loadingPaint == null) { loadingPaint = new Paint(Paint.ANTI_ALIAS_FLAG); } @@ -393,13 +477,13 @@ private void drawFill(Canvas canvas, float alpha) { private void addBar(Path path, float x, float h) { final float strokeWidth = AndroidUtilities.dpf2(2); - final int y = (height - AndroidUtilities.dp(14)) / 2; + final int y = (height - dp(14)) / 2; h *= waveScaling; AndroidUtilities.rectTmp.set( x + AndroidUtilities.dpf2(1) - strokeWidth / 2f, - y + AndroidUtilities.dp(7) + (-h - strokeWidth / 2f), + y + dp(7) + (-h - strokeWidth / 2f), x + AndroidUtilities.dpf2(1) + strokeWidth / 2f, - y + AndroidUtilities.dp(7) + (h + strokeWidth / 2f) + y + dp(7) + (h + strokeWidth / 2f) ); path.addRoundRect(AndroidUtilities.rectTmp, strokeWidth, strokeWidth, Path.Direction.CW); } @@ -417,4 +501,83 @@ public void setLoading(boolean loading) { parentView.invalidate(); } } + + public static class Particles { + private final int count; + private Runnable invalidate; + private final ArrayList particles = new ArrayList<>(50); + private final ArrayList deadParticles = new ArrayList<>(50); + + public Particles(int count, Runnable invalidate) { + this.count = count; + this.invalidate = invalidate; + paint.setStrokeWidth(dp(1.33f)); + } + + private final Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG); + private class Particle { + float x, y, v, vx, vy, t, d; + } + + public Particles setColor(int color) { + paint.setColor(color); + return this; + } + + private RectF emitArea; + public Particles setEmitArea(RectF emitArea) { + this.emitArea = emitArea; + return this; + } + + public void clear() { + deadParticles.addAll(particles); + particles.clear(); + } + + private long lastTime; + public void draw(Canvas canvas, float alpha) { + final long now = System.currentTimeMillis(); + final long dt = Math.min(20, (now - lastTime)); + lastTime = now; + for (int i = 0; i < particles.size(); ++i) { + Particle p = particles.get(i); + p.t -= dt / p.d; + if (p.t < 0) { + deadParticles.add(p); + particles.remove(i); + i--; + } else { + p.x += p.vx * p.v * dt / 500.0f; + p.y += p.vy * p.v * dt / 500.0f; + p.vy -= dp(0.33f) * dt / 500.0f; + } + } + if (emitArea != null) { + int count = Math.min(4, this.count - particles.size()); +// double vx = Math.sin(Math.PI / 180.0 * (0 - 90)); +// double vy = -Math.cos(Math.PI / 180.0 * (0 - 90)); + for (int i = 0; i < count; ++i) { + Particle p = deadParticles.isEmpty() ? new Particle() :deadParticles.remove(0); + p.x = emitArea.left + emitArea.width() * Utilities.random.nextFloat(); + p.y = emitArea.top + emitArea.height() * Utilities.random.nextFloat(); + double angle = (Math.PI / 180.0) * (Utilities.random.nextInt(200) - 125); + p.vx = (float) (Math.cos(angle) - Math.sin(angle)) * .8f; + p.vy = (float) (Math.sin(angle) + Math.cos(angle)) - .2f; + p.t = 1f; + p.v = AndroidUtilities.dp(10 + Utilities.random.nextFloat() * 7); + p.d = AndroidUtilities.lerp(420, 550, Utilities.random.nextFloat()); + particles.add(p); + } + } + for (int i = 0; i < particles.size(); ++i) { + Particle p = particles.get(i); + paint.setAlpha((int) (0xFF * alpha * p.t)); + canvas.drawPoint(p.x, p.y, paint); + } + if (invalidate != null) { + invalidate.run(); + } + } + } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/ShareAlert.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/ShareAlert.java index b29d2f4c72c..8888144a9fa 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/ShareAlert.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/ShareAlert.java @@ -148,6 +148,7 @@ public class ShareAlert extends BottomSheet implements NotificationCenter.Notifi private int containerViewTop = -1; private boolean fullyShown = false; private boolean includeStory; + public boolean includeStoryFromMessage; private ChatActivity parentFragment; private Activity parentActivity; @@ -1569,11 +1570,13 @@ public void setRecentSearch(ArrayList a AndroidUtilities.updateViewVisibilityAnimated(searchGridView, false, 1f, false); } + protected void onShareStory(View cell) { + + } + private void selectDialog(View cell, TLRPC.Dialog dialog) { if (dialog instanceof ShareDialogsAdapter.MyStoryDialog) { - LongSparseArray dids = new LongSparseArray<>(); - dids.put(Long.MAX_VALUE, dialog); - onSend(dids, 1, null); + onShareStory(cell); return; } if (topicsGridView.getVisibility() != View.GONE || parentActivity == null) { @@ -2480,7 +2483,15 @@ public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType View view; switch (viewType) { case 0: { - view = new ShareDialogCell(context, darkTheme ? ShareDialogCell.TYPE_CALL : ShareDialogCell.TYPE_SHARE, resourcesProvider); + view = new ShareDialogCell(context, darkTheme ? ShareDialogCell.TYPE_CALL : ShareDialogCell.TYPE_SHARE, resourcesProvider) { + @Override + protected String repostToCustomName() { + if (includeStoryFromMessage) { + return LocaleController.getString(R.string.RepostToStory); + } + return super.repostToCustomName(); + } + }; view.setLayoutParams(new RecyclerView.LayoutParams(RecyclerView.LayoutParams.MATCH_PARENT, AndroidUtilities.dp(100))); break; } @@ -2499,6 +2510,7 @@ public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) { if (holder.getItemViewType() == 0) { ShareDialogCell cell = (ShareDialogCell) holder.itemView; TLRPC.Dialog dialog = getItem(position); + if (dialog == null) return; cell.setTopic(selectedDialogTopics.get(dialog), false); cell.setDialog(dialog.id, selectedDialogs.indexOfKey(dialog.id) >= 0, null); } @@ -3163,7 +3175,7 @@ public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) { ((ProfileSearchCell) holder.itemView).setData(object, ec, name, null, false, false); ((ProfileSearchCell) holder.itemView).useSeparator = position < getItemCount() - 2; } else if (holder.itemView instanceof ShareDialogCell) { - ((ShareDialogCell) holder.itemView).setDialog((int) id, selectedDialogs.indexOfKey(id) >= 0, name); + ((ShareDialogCell) holder.itemView).setDialog(id, selectedDialogs.indexOfKey(id) >= 0, name); } return; } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/SharedMediaLayout.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/SharedMediaLayout.java index 2b6692f319b..1063714f739 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/SharedMediaLayout.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/SharedMediaLayout.java @@ -58,6 +58,7 @@ import androidx.annotation.NonNull; import androidx.annotation.Nullable; +import androidx.collection.LongSparseArray; import androidx.core.content.ContextCompat; import androidx.core.view.accessibility.AccessibilityNodeInfoCompat; import androidx.recyclerview.widget.GridLayoutManager; @@ -105,6 +106,7 @@ import org.telegram.ui.CalendarActivity; import org.telegram.ui.Cells.ChatActionCell; import org.telegram.ui.Cells.ContextLinkCell; +import org.telegram.ui.Cells.DialogCell; import org.telegram.ui.Cells.DividerCell; import org.telegram.ui.Cells.GraySectionCell; import org.telegram.ui.Cells.LoadingCell; @@ -2285,6 +2287,13 @@ public boolean drawChild(Canvas canvas, View child, long drawingTime) { return super.drawChild(canvas, child, drawingTime); } + @Override + public Integer getSelectorColor(int position) { + if (getAdapter() == channelRecommendationsAdapter && channelRecommendationsAdapter.more > 0 && position == channelRecommendationsAdapter.getItemCount() - 1) { + return 0; + } + return super.getSelectorColor(position); + } }; mediaPages[a].listView.setFastScrollEnabled(RecyclerListView.FastScroll.DATE_TYPE); @@ -2438,7 +2447,7 @@ public void getItemOffsets(android.graphics.Rect outRect, View view, RecyclerVie onItemClick(position, view, messageObject, 0, mediaPage.selectedType); } } else if (mediaPage.selectedType == TAB_RECOMMENDED_CHANNELS) { - if (view instanceof ProfileSearchCell && position >= 0 && position < channelRecommendationsAdapter.chats.size()) { + if ((view instanceof ProfileSearchCell || y < dp(60)) && position >= 0 && position < channelRecommendationsAdapter.chats.size()) { Bundle args = new Bundle(); args.putLong("chat_id", channelRecommendationsAdapter.chats.get(position).id); profileActivity.presentFragment(new ChatActivity(args)); @@ -2513,13 +2522,7 @@ public boolean onItemClick(View view, int position, float x, float y) { return onItemLongClick(messageObject, view, mediaPage.selectedType); } } else if (mediaPage.selectedType == TAB_RECOMMENDED_CHANNELS) { - Bundle args = new Bundle(); - args.putLong("chat_id", channelRecommendationsAdapter.chats.get(position).id); - final BaseFragment fragment = new ChatActivity(args); - if (profileActivity instanceof ProfileActivity) { - ((ProfileActivity) profileActivity).prepareBlurBitmap(); - } - profileActivity.presentFragmentAsPreview(fragment); + channelRecommendationsAdapter.openPreview(position); return true; } return false; @@ -3404,7 +3407,7 @@ private void checkLoadMoreScroll(MediaPage mediaPage, RecyclerListView recyclerV } } - if (mediaPage.selectedType == 7) { + if (mediaPage.selectedType == TAB_GROUPUSERS) { } else if (mediaPage.selectedType == TAB_STORIES) { if (storiesAdapter.storiesList != null && firstVisibleItem + visibleItemCount > storiesAdapter.storiesList.getLoadedCount() - mediaColumnsCount[1]) { @@ -3522,7 +3525,8 @@ public boolean isSearchItemVisible() { mediaPages[0].selectedType != TAB_ARCHIVED_STORIES && mediaPages[0].selectedType != TAB_VOICE && mediaPages[0].selectedType != TAB_GIF && - mediaPages[0].selectedType != TAB_COMMON_GROUPS + mediaPages[0].selectedType != TAB_COMMON_GROUPS && + mediaPages[0].selectedType != TAB_RECOMMENDED_CHANNELS ); } @@ -4651,8 +4655,9 @@ public void didReceivedNotification(int id, int account, Object... args) { } else if (id == NotificationCenter.channelRecommendationsLoaded) { long chatId = (long) args[0]; if (chatId == -dialog_id) { - channelRecommendationsAdapter.update(); + channelRecommendationsAdapter.update(true); updateTabs(true); + checkCurrentTabValid(); } } } @@ -4982,7 +4987,7 @@ private void updateTabs(boolean animated) { if ((hasMedia[6] <= 0) == scrollSlidingTextTabStrip.hasTab(TAB_COMMON_GROUPS)) { changed++; } - hasRecommendations = DialogObject.isChatDialog(dialog_id) && MessagesController.ChannelRecommendations.hasRecommendations(profileActivity.getCurrentAccount(), -dialog_id); + hasRecommendations = !channelRecommendationsAdapter.chats.isEmpty(); if (hasRecommendations != scrollSlidingTextTabStrip.hasTab(TAB_RECOMMENDED_CHANNELS)) { changed++; } @@ -5332,11 +5337,11 @@ private void switchToCurrentSelectedMode(boolean animated) { } } } - if (mediaPages[a].selectedType == 6) { + if (mediaPages[a].selectedType == TAB_COMMON_GROUPS) { if (!commonGroupsAdapter.loading && !commonGroupsAdapter.endReached && commonGroupsAdapter.chats.isEmpty()) { commonGroupsAdapter.getChats(0, 100); } - } else if (mediaPages[a].selectedType == 7) { + } else if (mediaPages[a].selectedType == TAB_GROUPUSERS) { } else if (mediaPages[a].selectedType == TAB_STORIES) { StoriesController.StoriesList storiesList = storiesAdapter.storiesList; @@ -5348,7 +5353,9 @@ private void switchToCurrentSelectedMode(boolean animated) { archivedStoriesAdapter.load(false); mediaPages[a].emptyView.showProgress(storiesList != null && (storiesList.isLoading() || hasInternet() && storiesList.getCount() > 0), animated); fastScrollVisible = storiesList != null && storiesList.getCount() > 0; - } else if (mediaPages[a].selectedType != TAB_RECOMMENDED_CHANNELS) { + } else if (mediaPages[a].selectedType == TAB_RECOMMENDED_CHANNELS) { + + } else { if (!sharedMediaData[mediaPages[a].selectedType].loading && !sharedMediaData[mediaPages[a].selectedType].endReached[0] && sharedMediaData[mediaPages[a].selectedType].messages.isEmpty()) { sharedMediaData[mediaPages[a].selectedType].loading = true; documentsAdapter.notifyDataSetChanged(); @@ -6819,31 +6826,36 @@ private class ChannelRecommendationsAdapter extends RecyclerListView.SelectionAd public ChannelRecommendationsAdapter(Context context) { mContext = context; - update(); + update(false); } - public void update() { + public void update(boolean notify) { if (profileActivity == null || !DialogObject.isChatDialog(dialog_id)) { return; } - TLRPC.Chat chat = MessagesController.getInstance(profileActivity.getCurrentAccount()).getChat(-dialog_id); - if (chat == null || !ChatObject.isChannelAndNotMegaGroup(chat)) { + TLRPC.Chat thisChat = MessagesController.getInstance(profileActivity.getCurrentAccount()).getChat(-dialog_id); + if (thisChat == null || !ChatObject.isChannelAndNotMegaGroup(thisChat)) { return; } - MessagesController.ChannelRecommendations rec = MessagesController.getInstance(profileActivity.getCurrentAccount()).getChannelRecommendations(chat.id); + MessagesController.ChannelRecommendations rec = MessagesController.getInstance(profileActivity.getCurrentAccount()).getChannelRecommendations(thisChat.id); chats.clear(); if (rec != null) { - chats.addAll(rec.chats); - more = UserConfig.getInstance(profileActivity.getCurrentAccount()).isPremium() ? 0 : rec.more; - } else { - more = 0; + for (int i = 0; i < rec.chats.size(); ++i) { + TLRPC.Chat chat = rec.chats.get(i); + if (chat != null && ChatObject.isNotInChat(chat)) { + chats.add(chat); + } + } + } + more = chats.isEmpty() || UserConfig.getInstance(profileActivity.getCurrentAccount()).isPremium() ? 0 : rec.more; + if (notify) { + notifyDataSetChanged(); } - notifyDataSetChanged(); } @Override public boolean isEnabled(RecyclerView.ViewHolder holder) { - return holder.getItemViewType() == 0; + return true; } @Override @@ -6860,12 +6872,6 @@ public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType profileActivity.presentFragment(new PremiumPreviewFragment("similar_channels")); } }); - cell.setOnClickListener(v -> { - if (chats.size() <= 0) return; - Bundle args = new Bundle(); - args.putLong("chat_id", chats.get(chats.size() - 1).id); - profileActivity.presentFragment(new ChatActivity(args)); - }); view = cell; } else { // 0 view = new ProfileSearchCell(mContext, resourcesProvider); @@ -6874,6 +6880,52 @@ public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType return new RecyclerListView.Holder(view); } + public void openPreview(int position) { + if (position < 0 || position >= chats.size()) return; + TLRPC.Chat chat = chats.get(position); + + Bundle args = new Bundle(); + args.putLong("chat_id", chat.id); + final BaseFragment fragment = new ChatActivity(args); + if (profileActivity instanceof ProfileActivity) { + ((ProfileActivity) profileActivity).prepareBlurBitmap(); + } + + ActionBarPopupWindow.ActionBarPopupWindowLayout previewMenu = new ActionBarPopupWindow.ActionBarPopupWindowLayout(getContext(), R.drawable.popup_fixed_alert, resourcesProvider, ActionBarPopupWindow.ActionBarPopupWindowLayout.FLAG_SHOWN_FROM_BOTTOM); + previewMenu.setBackgroundColor(getThemedColor(Theme.key_actionBarDefaultSubmenuBackground)); + + ActionBarMenuSubItem openChannel = new ActionBarMenuSubItem(getContext(), false, false); + openChannel.setTextAndIcon(LocaleController.getString(R.string.OpenChannel2), R.drawable.msg_channel); + openChannel.setMinimumWidth(160); + openChannel.setOnClickListener(view -> { + if (profileActivity != null && profileActivity.getParentLayout() != null) { + profileActivity.getParentLayout().expandPreviewFragment(); + } + }); + previewMenu.addView(openChannel); + + ActionBarMenuSubItem joinChannel = new ActionBarMenuSubItem(getContext(), false, false); + joinChannel.setTextAndIcon(LocaleController.getString(R.string.ProfileJoinChannel), R.drawable.msg_addbot); + joinChannel.setMinimumWidth(160); + joinChannel.setOnClickListener(view -> { + profileActivity.finishPreviewFragment(); + chat.left = false; + update(false); + notifyItemRemoved(position); + if (chats.isEmpty()) { + updateTabs(true); + checkCurrentTabValid(); + } + profileActivity.getNotificationCenter().postNotificationName(NotificationCenter.channelRecommendationsLoaded, -dialog_id); + profileActivity.getMessagesController().addUserToChat(chat.id, profileActivity.getUserConfig().getCurrentUser(), 0, null, profileActivity, () -> { + BulletinFactory.of(profileActivity).createSimpleBulletin(R.raw.contact_check, LocaleController.formatString(R.string.YouJoinedChannel, chat == null ? "" : chat.title)).show(); + }); + }); + previewMenu.addView(joinChannel); + + profileActivity.presentFragmentAsPreviewWithMenu(fragment, previewMenu); + } + @Override public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) { ProfileSearchCell cell = null; @@ -6909,11 +6961,6 @@ private static class MoreRecommendationsCell extends FrameLayout { private final ButtonWithCounterView button; private final LinkSpanDrawable.LinksTextView textView; - @Override - public void setOnClickListener(@Nullable OnClickListener l) { - channelCell.setOnClickListener(l); - } - public MoreRecommendationsCell(int currentAccount, Context context, Theme.ResourcesProvider resourcesProvider, Runnable onPremiumClick) { super(context); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/StatusBadgeComponent.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/StatusBadgeComponent.java index e207e00a5d6..1db8830bd85 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/StatusBadgeComponent.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/StatusBadgeComponent.java @@ -4,6 +4,7 @@ import android.view.View; import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.DialogObject; import org.telegram.tgnet.TLObject; import org.telegram.tgnet.TLRPC; import org.telegram.ui.ActionBar.Theme; @@ -31,16 +32,14 @@ public Drawable updateDrawable(TLRPC.User user, TLRPC.Chat chat, int colorFilter if (chat != null && chat.verified) { statusDrawable.set(verifiedDrawable = (verifiedDrawable == null ? new CombinedDrawable(Theme.dialogs_verifiedDrawable, Theme.dialogs_verifiedCheckDrawable) : verifiedDrawable), animated); statusDrawable.setColor(null); - return statusDrawable; - } - if (user != null && user.verified) { - statusDrawable.set(verifiedDrawable = (verifiedDrawable == null ? new CombinedDrawable(Theme.dialogs_verifiedDrawable, Theme.dialogs_verifiedCheckDrawable) : verifiedDrawable), animated); + } else if (chat != null && DialogObject.getEmojiStatusDocumentId(chat.emoji_status) != 0) { + statusDrawable.set(DialogObject.getEmojiStatusDocumentId(chat.emoji_status), animated); statusDrawable.setColor(null); - } else if (user != null && user.emoji_status instanceof TLRPC.TL_emojiStatus) { - statusDrawable.set(((TLRPC.TL_emojiStatus) user.emoji_status).document_id, animated); + } else if (user != null && user.verified) { + statusDrawable.set(verifiedDrawable = (verifiedDrawable == null ? new CombinedDrawable(Theme.dialogs_verifiedDrawable, Theme.dialogs_verifiedCheckDrawable) : verifiedDrawable), animated); statusDrawable.setColor(null); - } else if (user != null && user.emoji_status instanceof TLRPC.TL_emojiStatusUntil && ((TLRPC.TL_emojiStatusUntil) user.emoji_status).until > (int) (System.currentTimeMillis() / 1000)) { - statusDrawable.set(((TLRPC.TL_emojiStatusUntil) user.emoji_status).document_id, animated); + } else if (user != null && DialogObject.getEmojiStatusDocumentId(user.emoji_status) != 0) { + statusDrawable.set(DialogObject.getEmojiStatusDocumentId(user.emoji_status), animated); statusDrawable.setColor(null); } else if (user != null && user.premium) { statusDrawable.set(PremiumGradient.getInstance().premiumStarDrawableMini, animated); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/StickerSetBulletinLayout.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/StickerSetBulletinLayout.java index 984f189be38..44fbdfea126 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/StickerSetBulletinLayout.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/StickerSetBulletinLayout.java @@ -189,7 +189,7 @@ public StickerSetBulletinLayout(@NonNull Context context, TLObject setObject, in subtitleTextView.setVisibility(ViewPagerFixed.GONE); break; case TYPE_REPLACED_TO_FAVORITES: - if (!UserConfig.getInstance(UserConfig.selectedAccount).isPremium() && !MessagesController.getInstance(UserConfig.selectedAccount).premiumLocked) { + if (!UserConfig.getInstance(UserConfig.selectedAccount).isPremium() && !MessagesController.getInstance(UserConfig.selectedAccount).premiumFeaturesBlocked()) { titleTextView.setText(LocaleController.formatString("LimitReachedFavoriteStickers", R.string.LimitReachedFavoriteStickers, MessagesController.getInstance(UserConfig.selectedAccount).stickersFavedLimitDefault)); CharSequence str = AndroidUtilities.premiumText(LocaleController.formatString("LimitReachedFavoriteStickersSubtitle", R.string.LimitReachedFavoriteStickersSubtitle, MessagesController.getInstance(UserConfig.selectedAccount).stickersFavedLimitPremium), () -> { Activity activity = AndroidUtilities.findActivity(context); @@ -204,7 +204,7 @@ public StickerSetBulletinLayout(@NonNull Context context, TLObject setObject, in } break; case TYPE_REPLACED_TO_FAVORITES_GIFS: - if (!UserConfig.getInstance(UserConfig.selectedAccount).isPremium() && !MessagesController.getInstance(UserConfig.selectedAccount).premiumLocked) { + if (!MessagesController.getInstance(UserConfig.selectedAccount).premiumFeaturesBlocked()) { titleTextView.setText(LocaleController.formatString("LimitReachedFavoriteGifs", R.string.LimitReachedFavoriteGifs, MessagesController.getInstance(UserConfig.selectedAccount).savedGifsLimitDefault)); CharSequence str = AndroidUtilities.premiumText(LocaleController.formatString("LimitReachedFavoriteGifsSubtitle", R.string.LimitReachedFavoriteGifsSubtitle, MessagesController.getInstance(UserConfig.selectedAccount).savedGifsLimitPremium), () -> { Activity activity = AndroidUtilities.findActivity(context); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/ThanosEffect.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/ThanosEffect.java new file mode 100644 index 00000000000..92977347080 --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/ThanosEffect.java @@ -0,0 +1,1115 @@ +package org.telegram.ui.Components; + +import android.content.Context; +import android.graphics.Bitmap; +import android.graphics.Canvas; +import android.graphics.Matrix; +import android.graphics.SurfaceTexture; +import android.opengl.EGL14; +import android.opengl.EGLExt; +import android.opengl.GLES20; +import android.opengl.GLES31; +import android.opengl.GLUtils; +import android.os.Build; +import android.os.Handler; +import android.os.Looper; +import android.os.Message; +import android.util.Log; +import android.view.Choreographer; +import android.view.Surface; +import android.view.TextureView; +import android.view.View; +import android.widget.FrameLayout; + +import androidx.annotation.NonNull; +import androidx.recyclerview.widget.RecyclerView; + +import com.google.zxing.common.detector.MathUtils; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.DispatchQueue; +import org.telegram.messenger.FileLog; +import org.telegram.messenger.ImageReceiver; +import org.telegram.messenger.MessageObject; +import org.telegram.messenger.MessagesController; +import org.telegram.messenger.R; +import org.telegram.messenger.SharedConfig; +import org.telegram.messenger.Utilities; +import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.Cells.BaseCell; +import org.telegram.ui.Cells.ChatActionCell; +import org.telegram.ui.Cells.ChatMessageCell; +import org.telegram.ui.ChatActivity; + +import java.util.ArrayList; + +import javax.microedition.khronos.egl.EGL10; +import javax.microedition.khronos.egl.EGLConfig; +import javax.microedition.khronos.egl.EGLContext; +import javax.microedition.khronos.egl.EGLDisplay; +import javax.microedition.khronos.egl.EGLSurface; +import javax.microedition.khronos.opengles.GL10; + +public class ThanosEffect extends TextureView { + + private static Boolean nothanos = null; + public static boolean supports() { + if (nothanos == null) { + nothanos = MessagesController.getGlobalMainSettings().getBoolean("nothanos", false); + } + return (nothanos == null || !nothanos) && Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP; + } + + private DrawingThread drawThread; + + private final Choreographer.FrameCallback frameCallback = new Choreographer.FrameCallback() { + @Override + public void doFrame(long frameTimeNanos) { + if (drawThread != null) { + drawThread.requestDraw(); + if (drawThread.running) { + Choreographer.getInstance().postFrameCallback(this); + } + } + } + }; + + private final ArrayList toSet = new ArrayList<>(); + private static class ToSet { + public final View view; + public final ArrayList views; + public final Runnable startCallback, doneCallback; + + public final Bitmap bitmap; + public final Matrix matrix; + public ToSet(View view, Runnable callback) { + this.view = view; + this.views = null; + this.startCallback = null; + this.doneCallback = callback; + this.bitmap = null; + this.matrix = null; + } + public ToSet(ArrayList views, Runnable callback) { + this.view = null; + this.views = views; + this.startCallback = null; + this.doneCallback = callback; + this.bitmap = null; + this.matrix = null; + } + public ToSet(Matrix matrix, Bitmap bitmap, Runnable startCallback, Runnable doneCallback) { + this.view = null; + this.views = null; + this.startCallback = startCallback; + this.doneCallback = doneCallback; + this.matrix = matrix; + this.bitmap = bitmap; + } + } + + private Runnable whenDone; + public ThanosEffect(@NonNull Context context, Runnable whenDoneCallback) { + super(context); + this.whenDone = whenDoneCallback; + setOpaque(false); + setSurfaceTextureListener(new TextureView.SurfaceTextureListener() { + @Override + public void onSurfaceTextureAvailable(@NonNull SurfaceTexture surface, int width, int height) { + if (drawThread != null) { + drawThread.kill(); + drawThread = null; + } + drawThread = new DrawingThread(surface, ThanosEffect.this::invalidate, ThanosEffect.this::destroy, width, height); + if (!toSet.isEmpty()) { + for (int i = 0; i < toSet.size(); ++i) { + ToSet toSetObj = toSet.get(i); + if (toSetObj.bitmap != null) { + drawThread.animate(toSetObj.matrix, toSetObj.bitmap, toSetObj.startCallback, toSetObj.doneCallback); + } else if (toSetObj.views != null) { + drawThread.animateGroup(toSetObj.views, toSetObj.doneCallback); + } else { + drawThread.animate(toSetObj.view, toSetObj.doneCallback); + } + } + toSet.clear(); + Choreographer.getInstance().postFrameCallback(frameCallback); + } + } + + @Override + public void onSurfaceTextureSizeChanged(@NonNull SurfaceTexture surface, int width, int height) { + if (drawThread != null) { + drawThread.resize(width, height); + } + } + + @Override + public boolean onSurfaceTextureDestroyed(@NonNull SurfaceTexture surface) { + if (drawThread != null) { + drawThread.kill(); + drawThread = null; + } + if (whenDone != null) { + Runnable runnable = whenDone; + whenDone = null; + runnable.run(); + } + return false; + } + + @Override + public void onSurfaceTextureUpdated(@NonNull SurfaceTexture surface) { + + } + }); + } + + private void destroy() { + if (whenDone != null) { + Runnable runnable = whenDone; + whenDone = null; + runnable.run(); + } + } + + public void scroll(int dx, int dy) { + if (drawThread != null && drawThread.running) { +// post(() -> drawThread.scroll(dx, dy)); + } + } + + public void animateGroup(ArrayList views, Runnable whenDone) { + if (drawThread != null) { + drawThread.animateGroup(views, whenDone); + Choreographer.getInstance().postFrameCallback(frameCallback); + } else { + toSet.add(new ToSet(views, whenDone)); + } + } + + public void animate(View view, Runnable whenDone) { + if (drawThread != null) { + drawThread.animate(view, whenDone); + Choreographer.getInstance().postFrameCallback(frameCallback); + } else { + toSet.add(new ToSet(view, whenDone)); + } + } + + public void animate(Matrix matrix, Bitmap bitmap, Runnable whenStarted, Runnable whenDone) { + if (drawThread != null) { + drawThread.animate(matrix, bitmap, whenStarted, whenDone); + Choreographer.getInstance().postFrameCallback(frameCallback); + } else { + toSet.add(new ToSet(matrix, bitmap, whenStarted, whenDone)); + } + } + + private static class DrawingThread extends DispatchQueue { + + private boolean alive = true; + private final SurfaceTexture surfaceTexture; + private final Runnable invalidate; + private Runnable destroy; + private int width, height; + + public DrawingThread(SurfaceTexture surfaceTexture, Runnable invalidate, Runnable destroy, int width, int height) { + super("ThanosEffect.DrawingThread", false); + + this.surfaceTexture = surfaceTexture; + this.invalidate = invalidate; + this.destroy = destroy; + this.width = width; + this.height = height; + + start(); + } + + public final static int DO_DRAW = 0; + public final static int DO_RESIZE = 1; + public final static int DO_KILL = 2; + public final static int DO_ADD_ANIMATION = 3; + public final static int DO_SCROLL = 4; + + @Override + public void handleMessage(Message inputMessage) { + switch (inputMessage.what) { + case DO_DRAW: { + draw(); + return; + } + case DO_RESIZE: { + resizeInternal(inputMessage.arg1, inputMessage.arg2); + draw(); + return; + } + case DO_KILL: { + killInternal(); + return; + } + case DO_ADD_ANIMATION: { + addAnimationInternal((Animation) inputMessage.obj); + return; + } + case DO_SCROLL: { + for (int i = 0; i < pendingAnimations.size(); ++i) { + Animation anim = pendingAnimations.get(i); + anim.offsetLeft += inputMessage.arg1; + anim.offsetTop += inputMessage.arg2; + } + return; + } + } + } + + @Override + public void run() { + try { + init(); + } catch (Exception e) { + FileLog.e(e); + for (int i = 0; i < toAddAnimations.size(); ++i) { + Animation animation = toAddAnimations.get(i); + if (animation.startCallback != null) { + AndroidUtilities.runOnUIThread(animation.startCallback); + } + animation.done(false); + } + toAddAnimations.clear(); + AndroidUtilities.runOnUIThread(() -> { + MessagesController.getGlobalMainSettings().edit().putBoolean("nothanos", nothanos = true).apply(); + }); + killInternal(); + return; + } + if (!toAddAnimations.isEmpty()) { + for (int i = 0; i < toAddAnimations.size(); ++i) { + addAnimationInternal(toAddAnimations.get(i)); + } + toAddAnimations.clear(); + } + super.run(); + } + + public void requestDraw() { + Handler handler = getHandler(); + if (handler != null) { + handler.sendMessage(handler.obtainMessage(DO_DRAW)); + } + } + + public void resize(int width, int height) { + Handler handler = getHandler(); + if (handler != null) { + handler.sendMessage(handler.obtainMessage(DO_RESIZE, width, height)); + } + } + + public void scroll(int dx, int dy) { + Handler handler = getHandler(); + if (handler != null) { + handler.sendMessage(handler.obtainMessage(DO_SCROLL, dx, dy)); + } + } + + private void resizeInternal(int width, int height) { + if (!alive) return; + this.width = width; + this.height = height; + GLES31.glViewport(0, 0, width, height); + GLES31.glUniform2f(sizeHandle, width, height); + } + + public void kill() { + Handler handler = getHandler(); + if (handler != null) { + handler.sendMessage(handler.obtainMessage(DO_KILL)); + } + } + + private void killInternal() { + if (!alive) return; + alive = false; + for (int i = 0; i < pendingAnimations.size(); ++i) { + Animation animation = pendingAnimations.get(i); + animation.done(false); + } + if (surfaceTexture != null) { + surfaceTexture.release(); + } + Looper looper = Looper.myLooper(); + if (looper != null) { + looper.quit(); + } + if (destroy != null) { + AndroidUtilities.runOnUIThread(destroy); + destroy = null; + } + } + + private EGL10 egl; + private EGLDisplay eglDisplay; + private EGLConfig eglConfig; + private EGLSurface eglSurface; + private EGLContext eglContext; + + private int drawProgram; + + private int matrixHandle; + private int resetHandle; + private int timeHandle; + private int deltaTimeHandle; + private int particlesCountHandle; + private int sizeHandle; + private int gridSizeHandle; + private int rectSizeHandle; + private int seedHandle; + private int rectPosHandle; + private int textureHandle; + private int densityHandle; + private int longevityHandle; + private int offsetHandle; + + public volatile boolean running; + private final ArrayList pendingAnimations = new ArrayList<>(); + + private void init() { + egl = (EGL10) javax.microedition.khronos.egl.EGLContext.getEGL(); + + eglDisplay = egl.eglGetDisplay(EGL14.EGL_DEFAULT_DISPLAY); + if (eglDisplay == egl.EGL_NO_DISPLAY) { + killInternal(); + return; + } + int[] version = new int[2]; + if (!egl.eglInitialize(eglDisplay, version)) { + killInternal(); + return; + } + + int[] configAttributes = { + EGL14.EGL_RED_SIZE, 8, + EGL14.EGL_GREEN_SIZE, 8, + EGL14.EGL_BLUE_SIZE, 8, + EGL14.EGL_ALPHA_SIZE, 8, + EGL14.EGL_RENDERABLE_TYPE, EGLExt.EGL_OPENGL_ES3_BIT_KHR, + EGL14.EGL_NONE + }; + EGLConfig[] eglConfigs = new EGLConfig[1]; + int[] numConfigs = new int[1]; + if (!egl.eglChooseConfig(eglDisplay, configAttributes, eglConfigs, 1, numConfigs)) { + kill(); + return; + } + eglConfig = eglConfigs[0]; + + int[] contextAttributes = { + EGL14.EGL_CONTEXT_CLIENT_VERSION, 3, + EGL14.EGL_NONE + }; + eglContext = egl.eglCreateContext(eglDisplay, eglConfig, egl.EGL_NO_CONTEXT, contextAttributes); + if (eglContext == null) { + killInternal(); + return; + } + + eglSurface = egl.eglCreateWindowSurface(eglDisplay, eglConfig, surfaceTexture, null); + if (eglSurface == null) { + killInternal(); + return; + } + + if (!egl.eglMakeCurrent(eglDisplay, eglSurface, eglSurface, eglContext)) { + killInternal(); + return; + } + + int vertexShader = GLES31.glCreateShader(GLES31.GL_VERTEX_SHADER); + int fragmentShader = GLES31.glCreateShader(GLES31.GL_FRAGMENT_SHADER); + if (vertexShader == 0 || fragmentShader == 0) { + killInternal(); + return; + } + GLES31.glShaderSource(vertexShader, RLottieDrawable.readRes(null, R.raw.thanos_vertex) + "\n// " + Math.random()); + GLES31.glCompileShader(vertexShader); + int[] status = new int[1]; + GLES31.glGetShaderiv(vertexShader, GLES31.GL_COMPILE_STATUS, status, 0); + if (status[0] != GLES31.GL_TRUE) { + FileLog.e("ThanosEffect, compile vertex shader error: " + GLES31.glGetShaderInfoLog(vertexShader)); + GLES31.glDeleteShader(vertexShader); + killInternal(); + return; + } + GLES31.glShaderSource(fragmentShader, RLottieDrawable.readRes(null, R.raw.thanos_fragment) + "\n// " + Math.random()); + GLES31.glCompileShader(fragmentShader); + GLES31.glGetShaderiv(fragmentShader, GLES31.GL_COMPILE_STATUS, status, 0); + if (status[0] != GLES31.GL_TRUE) { + FileLog.e("ThanosEffect, compile fragment shader error: " + GLES31.glGetShaderInfoLog(fragmentShader)); + GLES31.glDeleteShader(fragmentShader); + killInternal(); + return; + } + drawProgram = GLES31.glCreateProgram(); + if (drawProgram == 0) { + killInternal(); + return; + } + GLES31.glAttachShader(drawProgram, vertexShader); + GLES31.glAttachShader(drawProgram, fragmentShader); + + String[] feedbackVaryings = { "outUV", "outPosition", "outVelocity", "outTime" }; + GLES31.glTransformFeedbackVaryings(drawProgram, feedbackVaryings, GLES31.GL_INTERLEAVED_ATTRIBS); + GLES31.glLinkProgram(drawProgram); + GLES31.glGetProgramiv(drawProgram, GLES31.GL_LINK_STATUS, status, 0); + if (status[0] != GLES31.GL_TRUE) { + FileLog.e("ThanosEffect, link program error: " + GLES31.glGetProgramInfoLog(drawProgram)); + killInternal(); + return; + } + + matrixHandle = GLES31.glGetUniformLocation(drawProgram, "matrix"); + rectSizeHandle = GLES31.glGetUniformLocation(drawProgram, "rectSize"); + rectPosHandle = GLES31.glGetUniformLocation(drawProgram, "rectPos"); + resetHandle = GLES31.glGetUniformLocation(drawProgram, "reset"); + timeHandle = GLES31.glGetUniformLocation(drawProgram, "time"); + deltaTimeHandle = GLES31.glGetUniformLocation(drawProgram, "deltaTime"); + particlesCountHandle = GLES31.glGetUniformLocation(drawProgram, "particlesCount"); + sizeHandle = GLES31.glGetUniformLocation(drawProgram, "size"); + gridSizeHandle = GLES31.glGetUniformLocation(drawProgram, "gridSize"); + textureHandle = GLES31.glGetUniformLocation(drawProgram, "tex"); + seedHandle = GLES31.glGetUniformLocation(drawProgram, "seed"); + densityHandle = GLES31.glGetUniformLocation(drawProgram, "dp"); + longevityHandle = GLES31.glGetUniformLocation(drawProgram, "longevity"); + offsetHandle = GLES31.glGetUniformLocation(drawProgram, "offset"); + + GLES31.glViewport(0, 0, width, height); + GLES31.glEnable(GLES31.GL_BLEND); + GLES31.glBlendFunc(GLES20.GL_SRC_ALPHA, GLES20.GL_ONE_MINUS_SRC_ALPHA); + GLES31.glClearColor(0.0f, 0.0f, 0.0f, 0.0f); + + GLES31.glUseProgram(drawProgram); + + GLES31.glUniform2f(sizeHandle, width, height); + } + + private final ArrayList toRunStartCallback = new ArrayList<>(); + + private float animationHeightPart(Animation animation) { + int totalHeight = 0; + for (int i = 0; i < pendingAnimations.size(); ++i) { + totalHeight += pendingAnimations.get(i).viewHeight; + } + return (float) animation.viewHeight / totalHeight; + } + + private boolean drawnAnimations = false; + private void draw() { + if (!alive) return; + + GLES31.glClear(GLES31.GL_COLOR_BUFFER_BIT); + + for (int i = 0; i < pendingAnimations.size(); ++i) { + Animation animation = pendingAnimations.get(i); + if (animation.firstDraw) { + animation.calcParticlesGrid(animationHeightPart(animation)); + if (animation.startCallback != null) { + toRunStartCallback.add(animation); + } + } + drawnAnimations = true; + animation.draw(); + if (animation.isDead()) { + animation.done(true); + pendingAnimations.remove(i); + running = !pendingAnimations.isEmpty(); + i--; + } + } + + checkGlErrors(); + + try { + egl.eglSwapBuffers(eglDisplay, eglSurface); + } catch (Exception e) { + for (int i = 0; i < toRunStartCallback.size(); ++i) { + AndroidUtilities.runOnUIThread(toRunStartCallback.get(i).startCallback); + } + toRunStartCallback.clear(); + for (int i = 0; i < pendingAnimations.size(); ++i) { + pendingAnimations.get(i).done(false); + } + pendingAnimations.clear(); + AndroidUtilities.runOnUIThread(() -> { + MessagesController.getGlobalMainSettings().edit().putBoolean("nothanos", nothanos = true).apply(); + }); + killInternal(); + return; + } + + for (int i = 0; i < toRunStartCallback.size(); ++i) { + AndroidUtilities.runOnUIThread(toRunStartCallback.get(i).startCallback); + } + toRunStartCallback.clear(); + + if (pendingAnimations.isEmpty() && drawnAnimations) { + killInternal(); + } + }; + + public void layoutAnimations() { + + } + + private final ArrayList toAddAnimations = new ArrayList<>(); + public void animateGroup(ArrayList views, Runnable whenDone) { + if (!alive) return; + Animation animation = new Animation(views, whenDone); + Handler handler = getHandler(); + running = true; + if (handler == null) { + toAddAnimations.add(animation); + } else { + handler.sendMessage(handler.obtainMessage(DO_ADD_ANIMATION, animation)); + } + } + public void animate(View view, Runnable whenDone) { + if (!alive) return; + Animation animation = new Animation(view, whenDone); + Handler handler = getHandler(); + running = true; + if (handler == null) { + toAddAnimations.add(animation); + } else { + handler.sendMessage(handler.obtainMessage(DO_ADD_ANIMATION, animation)); + } + } + public void animate(Matrix matrix, Bitmap bitmap, Runnable whenStart, Runnable whenDone) { + if (!alive) return; + Animation animation = new Animation(matrix, bitmap, whenStart, whenDone); + Handler handler = getHandler(); + running = true; + if (handler == null) { + toAddAnimations.add(animation); + } else { + handler.sendMessage(handler.obtainMessage(DO_ADD_ANIMATION, animation)); + } + } + + private void addAnimationInternal(Animation animation) { + GLES31.glGenTextures(1, animation.texture, 0); + GLES20.glBindTexture(GL10.GL_TEXTURE_2D, animation.texture[0]); + GLES20.glTexParameteri(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_MIN_FILTER, GL10.GL_LINEAR); + GLES20.glTexParameteri(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_MAG_FILTER, GL10.GL_LINEAR); + GLES20.glTexParameteri(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_WRAP_S, GL10.GL_CLAMP_TO_EDGE); + GLES20.glTexParameteri(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_WRAP_T, GL10.GL_CLAMP_TO_EDGE); + GLUtils.texImage2D(GL10.GL_TEXTURE_2D, 0, animation.bitmap, 0); + GLES20.glBindTexture(GL10.GL_TEXTURE_2D, 0); + + animation.bitmap.recycle(); + animation.bitmap = null; + + pendingAnimations.add(animation); + running = true; + + animation.ready = true; + } + + private class Animation { + + public ArrayList views = new ArrayList<>(); + private long lastDrawTime = -1; + public float time = 0; + public boolean firstDraw = true; + public Runnable startCallback, doneCallback; + public volatile boolean ready; + + public float offsetLeft = 0, offsetTop = 0; + public float left = 0; + public float top = 0; + public final float density = AndroidUtilities.density; + public float longevity = 1.5f; + public float timeScale = 1.12f; + public boolean invalidateMatrix = true; + public boolean customMatrix = false; + public final float[] glMatrixValues = new float[9]; + public final float[] matrixValues = new float[9]; + public final Matrix matrix = new Matrix(); + + public int particlesCount; + public int viewWidth, viewHeight; + public int gridWidth, gridHeight; + public float gridSize; + + public final float seed = (float) (Math.random() * 2.); + + public int currentBuffer; + public final int[] texture = new int[1]; + public final int[] buffer = new int[2]; + + private Bitmap bitmap; + + public Animation(Matrix matrix, Bitmap bitmap, Runnable whenStarted, Runnable whenDone) { + float[] v = new float[] { 0, 0, 0, 1, 1, 0, 1, 1 }; + matrix.mapPoints(v); + left = v[0]; + top = v[1]; + viewWidth = (int) MathUtils.distance(v[2], v[3], v[6], v[7]); + viewHeight = (int) MathUtils.distance(v[4], v[5], v[6], v[7]); + customMatrix = true; + this.matrix.set(matrix); + retrieveMatrixValues(); + startCallback = whenStarted; + doneCallback = whenDone; +// longevity = 1.5f * Utilities.clamp(viewWidth / (float) AndroidUtilities.displaySize.x, .6f, 0.2f); + this.bitmap = bitmap; + } + + public Animation(ArrayList views, Runnable whenDone) { + this.views.addAll(views); + int mleft = Integer.MAX_VALUE, mright = Integer.MIN_VALUE; + int mtop = Integer.MAX_VALUE, mbottom = Integer.MIN_VALUE; + for (int i = 0; i < views.size(); ++i) { + View view = views.get(i); + mleft = Math.min(mleft, (int) view.getX()); + mright = Math.max(mright, (int) view.getX() + view.getWidth()); + mtop = Math.min(mtop, (int) view.getY()); + mbottom = Math.max(mbottom, (int) view.getY() + view.getHeight()); + } + top = mtop; + left = mleft; + viewWidth = mright - mleft; + viewHeight = mbottom - mtop; + doneCallback = whenDone; + startCallback = () -> { + for (int j = 0; j < views.size(); ++j) { + views.get(j).setVisibility(View.GONE); + if (views.get(j) instanceof ChatMessageCell) { + ((ChatMessageCell) views.get(j)).setCheckBoxVisible(false, false); + ((ChatMessageCell) views.get(j)).setChecked(false, false, false); + } + } + }; +// longevity = 1.6f * .6f; + + for (int i = 0; i < views.size(); ++i) { + if (views.get(i) instanceof ChatMessageCell) { + ((ChatMessageCell) views.get(i)).drawingToBitmap = true; + } + } + + bitmap = Bitmap.createBitmap(viewWidth, viewHeight, Bitmap.Config.ARGB_8888); + final Canvas canvas = new Canvas(bitmap); + if (views.size() <= 0) return; + if (!(views.get(0).getParent() instanceof RecyclerListView)) return; + RecyclerListView chatListView = (RecyclerListView) views.get(0).getParent(); + if (!(chatListView.getParent() instanceof ChatActivity.ChatActivityFragmentView)) return; + ChatActivity.ChatActivityFragmentView contentView = (ChatActivity.ChatActivityFragmentView) chatListView.getParent(); + ChatActivity chatActivity = contentView.getChatActivity(); + final ArrayList drawingGroups = new ArrayList<>(10); + ArrayList drawTimeAfter = new ArrayList<>(); + ArrayList drawNamesAfter = new ArrayList<>(); + ArrayList drawCaptionAfter = new ArrayList<>(); + canvas.save(); + for (int k = 0; k < 3; k++) { + drawingGroups.clear(); + if (k == 2 && !chatListView.isFastScrollAnimationRunning()) { + continue; + } + for (int i = 0; i < views.size(); i++) { + View child = views.get(i); + if (child instanceof ChatMessageCell) { + ChatMessageCell cell = (ChatMessageCell) child; + if (child.getY() > chatListView.getHeight() || child.getY() + child.getHeight() < 0 || cell.getVisibility() == View.INVISIBLE || cell.getVisibility() == View.GONE) { + continue; + } + + MessageObject.GroupedMessages group = cell.getCurrentMessagesGroup(); + MessageObject.GroupedMessagePosition position = group == null || group.positions == null ? null : group.positions.get(cell.getMessageObject()); + if (k == 0) { + if (position != null || cell.getTransitionParams().animateBackgroundBoundsInner) { + if (position == null || (position.last || position.minX == 0 && position.minY == 0)) { + if (position == null || position.last) { + drawTimeAfter.add(cell); + } + if ((position == null || (position.minX == 0 && position.minY == 0)) && cell.hasNameLayout()) { + drawNamesAfter.add(cell); + } + } + if (position != null || cell.getTransitionParams().transformGroupToSingleMessage || cell.getTransitionParams().animateBackgroundBoundsInner) { + if (position == null || (position.flags & MessageObject.POSITION_FLAG_BOTTOM) != 0) { + drawCaptionAfter.add(cell); + } + } + } + } + + if (group == null || (k == 0 && group.messages.size() == 1) || (k == 1 && !group.transitionParams.drawBackgroundForDeletedItems)) { + continue; + } + if ((k == 0 && cell.getMessageObject().deleted) || (k == 1 && !cell.getMessageObject().deleted)) { + continue; + } + if ((k == 2 && !cell.willRemovedAfterAnimation()) || (k != 2 && cell.willRemovedAfterAnimation())) { + continue; + } + + if (!drawingGroups.contains(group)) { + group.transitionParams.left = 0; + group.transitionParams.top = 0; + group.transitionParams.right = 0; + group.transitionParams.bottom = 0; + + group.transitionParams.pinnedBotton = false; + group.transitionParams.pinnedTop = false; + group.transitionParams.cell = cell; + drawingGroups.add(group); + } + + group.transitionParams.pinnedTop = cell.isPinnedTop(); + group.transitionParams.pinnedBotton = cell.isPinnedBottom(); + + int left = (cell.getLeft() + cell.getBackgroundDrawableLeft()); + int right = (cell.getLeft() + cell.getBackgroundDrawableRight()); + int top = (cell.getTop() + cell.getBackgroundDrawableTop()); + int bottom = (cell.getTop() + cell.getBackgroundDrawableBottom()); + + if ((cell.getCurrentPosition().flags & MessageObject.POSITION_FLAG_TOP) == 0) { + top -= AndroidUtilities.dp(10); + } + + if ((cell.getCurrentPosition().flags & MessageObject.POSITION_FLAG_BOTTOM) == 0) { + bottom += AndroidUtilities.dp(10); + } + + if (cell.willRemovedAfterAnimation()) { + group.transitionParams.cell = cell; + } + + if (group.transitionParams.top == 0 || top < group.transitionParams.top) { + group.transitionParams.top = top; + } + if (group.transitionParams.bottom == 0 || bottom > group.transitionParams.bottom) { + group.transitionParams.bottom = bottom; + } + if (group.transitionParams.left == 0 || left < group.transitionParams.left) { + group.transitionParams.left = left; + } + if (group.transitionParams.right == 0 || right > group.transitionParams.right) { + group.transitionParams.right = right; + } + } + } + + for (int i = 0; i < drawingGroups.size(); i++) { + MessageObject.GroupedMessages group = drawingGroups.get(i); + float x = group.transitionParams.cell.getNonAnimationTranslationX(true); + float l = (group.transitionParams.left + x + group.transitionParams.offsetLeft); + float t = (group.transitionParams.top + group.transitionParams.offsetTop); + float r = (group.transitionParams.right + x + group.transitionParams.offsetRight); + float b = (group.transitionParams.bottom + group.transitionParams.offsetBottom); + + if (!group.transitionParams.backgroundChangeBounds) { + t += group.transitionParams.cell.getTranslationY(); + b += group.transitionParams.cell.getTranslationY(); + } + + if (t < chatActivity.chatListViewPaddingTop - chatActivity.chatListViewPaddingVisibleOffset - AndroidUtilities.dp(20)) { + t = chatActivity.chatListViewPaddingTop - chatActivity.chatListViewPaddingVisibleOffset - AndroidUtilities.dp(20); + } + + if (b > chatListView.getMeasuredHeight() + AndroidUtilities.dp(20)) { + b = chatListView.getMeasuredHeight() + AndroidUtilities.dp(20); + } + + t -= top; + b -= top; + + boolean useScale = group.transitionParams.cell.getScaleX() != 1f || group.transitionParams.cell.getScaleY() != 1f; + if (useScale) { + canvas.save(); + canvas.scale(group.transitionParams.cell.getScaleX(), group.transitionParams.cell.getScaleY(), l + (r - l) / 2, t + (b - t) / 2); + } + boolean selected = false; + group.transitionParams.cell.drawBackground(canvas, (int) l, (int) t, (int) r, (int) b, group.transitionParams.pinnedTop, group.transitionParams.pinnedBotton, selected, contentView.getKeyboardHeight()); + group.transitionParams.cell = null; + group.transitionParams.drawCaptionLayout = group.hasCaption; + if (useScale) { + canvas.restore(); + for (int ii = 0; ii < views.size(); ii++) { + View child = views.get(ii); + if (child instanceof ChatMessageCell && ((ChatMessageCell) child).getCurrentMessagesGroup() == group) { + ChatMessageCell cell = ((ChatMessageCell) child); + int left = cell.getLeft(); + int top = cell.getTop(); + child.setPivotX(l - left + (r - l) / 2); + child.setPivotY(t - top + (b - t) / 2); + } + } + } + } + } + for (int i = 0; i < views.size(); ++i) { + View view = views.get(i); + canvas.save(); + canvas.translate(view.getX() - mleft, view.getY() - mtop); + view.draw(canvas); + if (view instanceof ChatMessageCell) { + ((ChatMessageCell) view).drawOutboundsContent(canvas); + } else if (view instanceof ChatActionCell) { + ((ChatActionCell) view).drawOutboundsContent(canvas); + } + canvas.restore(); + } + float listTop = chatListView.getY() + chatActivity.chatListViewPaddingTop - chatActivity.chatListViewPaddingVisibleOffset - AndroidUtilities.dp(4); + int size = drawTimeAfter.size(); + if (size > 0) { + for (int a = 0; a < size; a++) { + ChatMessageCell view = drawTimeAfter.get(a); + drawChildElement(chatListView, chatActivity, canvas, listTop, view, 0, view.getX() - mleft, view.getY() - mtop); + } + drawTimeAfter.clear(); + } + size = drawNamesAfter.size(); + if (size > 0) { + for (int a = 0; a < size; a++) { + ChatMessageCell view = drawNamesAfter.get(a); + drawChildElement(chatListView, chatActivity, canvas, listTop, view, 1, view.getX() - mleft, view.getY() - mtop); + } + drawNamesAfter.clear(); + } + size = drawCaptionAfter.size(); + if (size > 0) { + for (int a = 0; a < size; a++) { + ChatMessageCell cell = drawCaptionAfter.get(a); + if (cell.getCurrentPosition() == null && !cell.getTransitionParams().animateBackgroundBoundsInner) { + continue; + } + drawChildElement(chatListView, chatActivity, canvas, listTop, cell, 2, cell.getX() - mleft, cell.getY() - mtop); + } + drawCaptionAfter.clear(); + } + canvas.restore(); + + for (int i = 0; i < views.size(); ++i) { + if (views.get(i) instanceof ChatMessageCell) { + ((ChatMessageCell) views.get(i)).drawingToBitmap = false; + } + } + } + + private void drawChildElement(View chatListView, ChatActivity chatActivity, Canvas canvas, float listTop, ChatMessageCell cell, int type, float x, float y) { + canvas.save(); + float alpha = cell.shouldDrawAlphaLayer() ? cell.getAlpha() : 1f; +// canvas.clipRect(chatListView.getLeft() - x, listTop - y, chatListView.getRight() - x, chatListView.getY() + chatListView.getMeasuredHeight() - (chatActivity == null ? 0 : chatActivity.blurredViewBottomOffset) - y); + canvas.translate(x, y); + cell.setInvalidatesParent(true); + if (type == 0) { + cell.drawTime(canvas, alpha, true); + } else if (type == 1) { + cell.drawNamesLayout(canvas, alpha); + } else { + cell.drawCaptionLayout(canvas, cell.getCurrentPosition() != null && (cell.getCurrentPosition().flags & MessageObject.POSITION_FLAG_LEFT) == 0, alpha); + } + cell.setInvalidatesParent(false); + canvas.restore(); + } + + public void calcParticlesGrid(float part) { + final int maxParticlesCount; + switch (SharedConfig.getDevicePerformanceClass()) { + case SharedConfig.PERFORMANCE_CLASS_HIGH: + maxParticlesCount = 120_000; + break; + case SharedConfig.PERFORMANCE_CLASS_AVERAGE: + maxParticlesCount = 60_000; + break; + case SharedConfig.PERFORMANCE_CLASS_LOW: + default: + maxParticlesCount = 30_000; + break; + } + float p = Math.max(AndroidUtilities.dpf2(.4f), 1); + particlesCount = Utilities.clamp((int) (viewWidth * viewHeight / (p * p)), (int) (maxParticlesCount * part), 10); + + final float aspectRatio = (float) viewWidth / viewHeight; + gridHeight = (int) Math.round(Math.sqrt(particlesCount / aspectRatio)); + gridWidth = (int) Math.round((float) particlesCount / gridHeight); + while (gridWidth * gridHeight < particlesCount) { + if ((float) gridWidth / gridHeight < aspectRatio) { + gridWidth++; + } else { + gridHeight++; + } + } + particlesCount = gridWidth * gridHeight; + gridSize = Math.max((float) viewWidth / gridWidth, (float) viewHeight / gridHeight); + + GLES31.glGenBuffers(2, buffer, 0); + for (int i = 0; i < 2; ++i) { + GLES31.glBindBuffer(GLES31.GL_ARRAY_BUFFER, buffer[i]); + GLES31.glBufferData(GLES31.GL_ARRAY_BUFFER, particlesCount * 28, null, GLES31.GL_DYNAMIC_DRAW); + } + } + + public Animation(View view, Runnable whenDone) { + this.views.add(view); + viewWidth = view.getWidth(); + viewHeight = view.getHeight(); + top = view.getY(); + left = 0; + if (view instanceof BaseCell) { + viewWidth = Math.max(1, ((BaseCell) view).getBoundsRight() - ((BaseCell) view).getBoundsLeft()); + left += ((BaseCell) view).getBoundsLeft(); + } + doneCallback = whenDone; + startCallback = () -> { + for (int j = 0; j < views.size(); ++j) { + views.get(j).setVisibility(View.GONE); + if (views.get(j) instanceof ChatMessageCell) { + ((ChatMessageCell) views.get(j)).setCheckBoxVisible(false, false); + ((ChatMessageCell) views.get(j)).setChecked(false, false, false); + } + } + }; +// longevity = 1.5f * Utilities.clamp(viewWidth / (float) AndroidUtilities.displaySize.x, .6f, 0.2f); + + bitmap = Bitmap.createBitmap(viewWidth, viewHeight, Bitmap.Config.ARGB_8888); + final Canvas canvas = new Canvas(bitmap); + canvas.save(); + canvas.translate(-left, 0); + if (view instanceof ChatMessageCell) { + ((ChatMessageCell) view).drawingToBitmap = true; + } + if (view instanceof ChatActionCell && ((ChatActionCell) view).hasGradientService()) { + ((ChatActionCell) view).drawBackground(canvas, true); + } else if (view instanceof ChatMessageCell && ((ChatMessageCell) view).drawBackgroundInParent()) { + ((ChatMessageCell) view).drawBackgroundInternal(canvas, true); + } + view.draw(canvas); + if (view instanceof ChatMessageCell) { + ImageReceiver avatarImage = ((ChatMessageCell) view).getAvatarImage(); + if (avatarImage != null && avatarImage.getVisible()) { + canvas.save(); + canvas.translate(0, -view.getY()); + avatarImage.draw(canvas); + canvas.restore(); + } + ((ChatMessageCell) view).drawingToBitmap = false; + } + if (view instanceof ChatMessageCell) { + ((ChatMessageCell) view).drawOutboundsContent(canvas); + } else if (view instanceof ChatActionCell) { + ((ChatActionCell) view).drawOutboundsContent(canvas); + } + canvas.restore(); + + left += view.getX(); + } + + private void retrieveMatrixValues() { + matrix.getValues(matrixValues); + glMatrixValues[0] = matrixValues[0]; + glMatrixValues[1] = matrixValues[3]; + glMatrixValues[2] = matrixValues[6]; + glMatrixValues[3] = matrixValues[1]; + glMatrixValues[4] = matrixValues[4]; + glMatrixValues[5] = matrixValues[7]; + glMatrixValues[6] = matrixValues[2]; + glMatrixValues[7] = matrixValues[5]; + glMatrixValues[8] = matrixValues[8]; + invalidateMatrix = false; + } + + public void draw() { + final long now = System.nanoTime(); + final double Δt = lastDrawTime < 0 ? 0 : (now - lastDrawTime) / 1_000_000_000.; + lastDrawTime = now; + + if (invalidateMatrix && !customMatrix) { + matrix.reset(); + matrix.postScale(viewWidth, viewHeight); + matrix.postTranslate(left, top); + retrieveMatrixValues(); + } + + time += Δt * timeScale; + + GLES31.glUniformMatrix3fv(matrixHandle, 1, false, glMatrixValues, 0); + GLES31.glUniform1f(resetHandle, firstDraw ? 1f : 0f); + GLES31.glUniform1f(timeHandle, time); + GLES31.glUniform1f(deltaTimeHandle, (float) Δt * timeScale); + GLES31.glUniform1f(particlesCountHandle, particlesCount); + GLES31.glUniform3f(gridSizeHandle, gridWidth, gridHeight, gridSize); + GLES31.glUniform2f(offsetHandle, offsetLeft, offsetTop); + + GLES31.glUniform2f(rectSizeHandle, viewWidth, viewHeight); + GLES31.glUniform1f(seedHandle, seed); + GLES31.glUniform2f(rectPosHandle, 0, 0); + GLES31.glUniform1f(densityHandle, density); + GLES31.glUniform1f(longevityHandle, longevity); + + GLES20.glActiveTexture(GLES20.GL_TEXTURE0); + GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, texture[0]); + GLES31.glUniform1i(textureHandle, 0); + + GLES31.glBindBuffer(GLES31.GL_ARRAY_BUFFER, buffer[currentBuffer]); + GLES31.glVertexAttribPointer(0, 2, GLES31.GL_FLOAT, false, 28, 0); // Initial UV (vec2) + GLES31.glEnableVertexAttribArray(0); + GLES31.glVertexAttribPointer(1, 2, GLES31.GL_FLOAT, false, 28, 8); // Position (vec2) + GLES31.glEnableVertexAttribArray(1); + GLES31.glVertexAttribPointer(2, 2, GLES31.GL_FLOAT, false, 28, 16); // Velocity (vec2) + GLES31.glEnableVertexAttribArray(2); + GLES31.glVertexAttribPointer(3, 1, GLES31.GL_FLOAT, false, 28, 24); // Time (float) + GLES31.glEnableVertexAttribArray(3); + GLES31.glBindBufferBase(GLES31.GL_TRANSFORM_FEEDBACK_BUFFER, 0, buffer[1 - currentBuffer]); + GLES31.glVertexAttribPointer(0, 2, GLES31.GL_FLOAT, false, 28, 0); // Initial UV (vec2) + GLES31.glEnableVertexAttribArray(0); + GLES31.glVertexAttribPointer(1, 2, GLES31.GL_FLOAT, false, 28, 8); // Position (vec2) + GLES31.glEnableVertexAttribArray(1); + GLES31.glVertexAttribPointer(2, 2, GLES31.GL_FLOAT, false, 28, 16); // Velocity (vec2) + GLES31.glEnableVertexAttribArray(2); + GLES31.glVertexAttribPointer(3, 1, GLES31.GL_FLOAT, false, 28, 24); // Time (float) + GLES31.glEnableVertexAttribArray(3); + + GLES31.glBeginTransformFeedback(GLES31.GL_POINTS); + GLES31.glDrawArrays(GLES31.GL_POINTS, 0, particlesCount); + GLES31.glEndTransformFeedback(); + + GLES31.glBindBuffer(GLES31.GL_ARRAY_BUFFER, 0); + GLES31.glBindBuffer(GLES31.GL_TRANSFORM_FEEDBACK_BUFFER, 0); + + firstDraw = false; + currentBuffer = 1 - currentBuffer; + } + + public boolean isDead() { + return time > longevity + .9f; + } + + public void done(boolean success) { + try { GLES31.glDeleteBuffers(2, buffer, 0); } catch (Exception e) { FileLog.e(e); }; + if (drawProgram != 0) { + try { GLES31.glDeleteProgram(drawProgram); } catch (Exception e) { FileLog.e(e); }; + drawProgram = 0; + } + try { GLES31.glDeleteTextures(1, texture, 0); } catch (Exception e) { FileLog.e(e); }; + + if (doneCallback != null) { + AndroidUtilities.runOnUIThread(() -> { + if (doneCallback != null) { + doneCallback.run(); + } + }); + } + } + } + + private void checkGlErrors() { + int err; + while ((err = GLES31.glGetError()) != GLES31.GL_NO_ERROR) { + FileLog.e("thanos gles error " + err); + } + } + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/ThemeSmallPreviewView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/ThemeSmallPreviewView.java index f94327b3d25..c47e1176b89 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/ThemeSmallPreviewView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/ThemeSmallPreviewView.java @@ -52,6 +52,8 @@ public class ThemeSmallPreviewView extends FrameLayout implements NotificationCe public final static int TYPE_DEFAULT = 0; public final static int TYPE_GRID = 1; public final static int TYPE_QR = 2; + public final static int TYPE_CHANNEL = 3; + public final static int TYPE_GRID_CHANNEL = 4; private final float STROKE_RADIUS = AndroidUtilities.dp(8); private final float INNER_RADIUS = AndroidUtilities.dp(6); @@ -94,7 +96,7 @@ public ThemeSmallPreviewView(Context context, int currentAccount, Theme.Resource backupImageView.getImageReceiver().setCrossfadeWithOldImage(true); backupImageView.getImageReceiver().setAllowStartLottieAnimation(false); backupImageView.getImageReceiver().setAutoRepeat(0); - if (currentType == TYPE_DEFAULT || currentType == TYPE_QR) { + if (currentType == TYPE_DEFAULT || currentType == TYPE_CHANNEL || currentType == TYPE_QR) { addView(backupImageView, LayoutHelper.createFrame(28, 28, Gravity.CENTER_HORIZONTAL | Gravity.BOTTOM, 0, 0, 0, 12)); } else { addView(backupImageView, LayoutHelper.createFrame(36, 36, Gravity.CENTER_HORIZONTAL | Gravity.BOTTOM, 0, 0, 0, 12)); @@ -107,12 +109,12 @@ public ThemeSmallPreviewView(Context context, int currentAccount, Theme.Resource @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { - if (currentType == TYPE_GRID) { + if (currentType == TYPE_GRID || currentType == TYPE_GRID_CHANNEL) { int width = MeasureSpec.getSize(widthMeasureSpec); int height = (int) (width * 1.2f); super.onMeasure(MeasureSpec.makeMeasureSpec(width, MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec(height, MeasureSpec.EXACTLY)); } else { - int width = AndroidUtilities.dp(77); + int width = AndroidUtilities.dp(currentType == TYPE_DEFAULT ? 77 : 83); int height = MeasureSpec.getSize(heightMeasureSpec); if (height == 0) { height = (int) (width * 1.35f); @@ -173,6 +175,18 @@ protected void dispatchDraw(Canvas canvas) { super.dispatchDraw(canvas); } + public TLRPC.WallPaper fallbackWallpaper; + public void setFallbackWallpaper(TLRPC.WallPaper wallPaper) { + if (fallbackWallpaper != wallPaper) { + this.fallbackWallpaper = wallPaper; + if (chatThemeItem != null && (chatThemeItem.chatTheme == null || chatThemeItem.chatTheme.wallpaper == null)) { + ChatThemeBottomSheet.ChatThemeItem item = chatThemeItem; + chatThemeItem = null; + setItem(item, false); + } + } + } + public int lastThemeIndex; public void setItem(ChatThemeBottomSheet.ChatThemeItem item, boolean animated) { boolean itemChanged = chatThemeItem != item; @@ -203,11 +217,15 @@ public void setItem(ChatThemeBottomSheet.ChatThemeItem item, boolean animated) { thumb = Emoji.getEmojiDrawable(item.chatTheme.getEmoticon()); } backupImageView.setImage(ImageLocation.getForDocument(document), "50_50", thumb, null); - if (item.chatTheme.wallpaper != null) { + TLRPC.WallPaper wallPaper = item.chatTheme.wallpaper; + if (wallPaper == null) { + wallPaper = fallbackWallpaper; + } + if (wallPaper != null) { if (attached && chatBackgroundDrawable != null) { chatBackgroundDrawable.onDetachedFromWindow(ThemeSmallPreviewView.this); } - chatBackgroundDrawable = new ChatBackgroundDrawable(item.chatTheme.wallpaper, false, true); + chatBackgroundDrawable = new ChatBackgroundDrawable(wallPaper, false, true); chatBackgroundDrawable.setParent(this); if (attached) { chatBackgroundDrawable.onAttachedToWindow(ThemeSmallPreviewView.this); @@ -219,6 +237,7 @@ public void setItem(ChatThemeBottomSheet.ChatThemeItem item, boolean animated) { chatBackgroundDrawable = null; } } + backupImageView.setVisibility(item.chatTheme.showAsDefaultStub && fallbackWallpaper != null ? View.GONE : View.VISIBLE); if (itemChanged || darkModeChanged) { if (animated) { @@ -314,7 +333,7 @@ public void setItem(ChatThemeBottomSheet.ChatThemeItem item, boolean animated) { } if (chatThemeItem.chatTheme == null || chatThemeItem.chatTheme.showAsDefaultStub) { - setContentDescription(LocaleController.getString("ChatNoTheme", R.string.ChatNoTheme)); + setContentDescription(LocaleController.getString(R.string.ChatNoTheme)); } else { setContentDescription(chatThemeItem.chatTheme.getEmoticon()); } @@ -494,7 +513,7 @@ private Drawable getPreviewDrawable(EmojiThemes.ThemeItem item) { bitmapDrawable.setFilterBitmap(true); drawable = bitmapDrawable; } - } else { + } else if (!(chatThemeItem.chatTheme != null && chatThemeItem.chatTheme.showAsDefaultStub)) { drawable = new MotionBackgroundDrawable(0xffdbddbb, 0xff6ba587, 0xffd5d88d, 0xff88b884, true); } } @@ -510,19 +529,33 @@ private StaticLayout getNoThemeStaticLayout() { } noThemeTextPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG + TextPaint.SUBPIXEL_TEXT_FLAG); noThemeTextPaint.setColor(getThemedColor(Theme.key_chat_emojiPanelTrendingDescription)); - noThemeTextPaint.setTextSize(AndroidUtilities.dp(14)); - noThemeTextPaint.setTypeface(AndroidUtilities.getTypeface("fonts/rmedium.ttf")); + noThemeTextPaint.setTextSize(AndroidUtilities.dp(noThemeStringTextSize())); + noThemeTextPaint.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); + int width = AndroidUtilities.dp(52); + if (currentType == TYPE_CHANNEL || currentType == TYPE_GRID_CHANNEL) { + width = AndroidUtilities.dp(77); + } textLayout = StaticLayoutEx.createStaticLayout2( - LocaleController.getString("ChatNoTheme", R.string.ChatNoTheme), + noThemeString(), noThemeTextPaint, - AndroidUtilities.dp(52), + width, Layout.Alignment.ALIGN_CENTER, 1f, 0f, true, - TextUtils.TruncateAt.END, AndroidUtilities.dp(52), 3 + TextUtils.TruncateAt.END, + width, + 3 ); return textLayout; } + protected int noThemeStringTextSize() { + return 14; + } + + protected String noThemeString() { + return LocaleController.getString(R.string.ChatNoTheme); + } + private int getThemedColor(int key) { return Theme.getColor(key, resourcesProvider); } @@ -594,7 +627,7 @@ public void drawBackground(Canvas canvas, float alpha) { outlineBackgroundPaint.setAlpha(wasAlpha); } canvas.restore(); - } else { + } else if (!(chatThemeItem != null && chatThemeItem.chatTheme != null && chatThemeItem.chatTheme.showAsDefaultStub && chatBackgroundDrawable != null)) { canvas.drawRoundRect(rectF, INNER_RADIUS, INNER_RADIUS, backgroundFillPaint); } } @@ -616,13 +649,15 @@ public void draw(Canvas canvas, float alpha) { rectF.set(INNER_RECT_SPACE, INNER_RECT_SPACE, getWidth() - INNER_RECT_SPACE, getHeight() - INNER_RECT_SPACE); if (chatThemeItem.chatTheme == null || (chatThemeItem.chatTheme.showAsDefaultStub && chatThemeItem.chatTheme.wallpaper == null)) { - canvas.drawRoundRect(rectF, INNER_RADIUS, INNER_RADIUS, backgroundFillPaint); - canvas.save(); - StaticLayout textLayout = getNoThemeStaticLayout(); - canvas.translate((getWidth() - textLayout.getWidth()) * 0.5f, AndroidUtilities.dp(18)); - textLayout.draw(canvas); - canvas.restore(); - } else { + if (fallbackWallpaper == null) { + canvas.drawRoundRect(rectF, INNER_RADIUS, INNER_RADIUS, backgroundFillPaint); + canvas.save(); + StaticLayout textLayout = getNoThemeStaticLayout(); + canvas.translate((getWidth() - textLayout.getWidth()) * 0.5f, AndroidUtilities.dp(18)); + textLayout.draw(canvas); + canvas.restore(); + } + } else if (currentType != TYPE_GRID_CHANNEL) { if (currentType == TYPE_QR) { if (chatThemeItem.icon != null) { float left = (getWidth() - chatThemeItem.icon.getWidth()) * 0.5f; @@ -630,9 +665,9 @@ public void draw(Canvas canvas, float alpha) { } } else { float bubbleTop = INNER_RECT_SPACE + AndroidUtilities.dp(8); - float bubbleLeft = INNER_RECT_SPACE + AndroidUtilities.dp(22); - if (currentType == TYPE_DEFAULT) { - rectF.set(bubbleLeft, bubbleTop, bubbleLeft + BUBBLE_WIDTH, bubbleTop + BUBBLE_HEIGHT); + float bubbleLeft = INNER_RECT_SPACE + AndroidUtilities.dp(currentType == TYPE_CHANNEL ? 5 : 22); + if (currentType == TYPE_DEFAULT || currentType == TYPE_CHANNEL) { + rectF.set(bubbleLeft, bubbleTop, bubbleLeft + BUBBLE_WIDTH * (currentType == TYPE_CHANNEL ? 1.2f : 1f), bubbleTop + BUBBLE_HEIGHT); } else { bubbleTop = getMeasuredHeight() * 0.12f; bubbleLeft = getMeasuredWidth() - getMeasuredWidth() * 0.65f; @@ -641,8 +676,8 @@ public void draw(Canvas canvas, float alpha) { rectF.set(bubbleLeft, bubbleTop, bubbleRight, bubbleBottom); } - Paint paint = outBubblePaintSecond; - if (currentType == TYPE_DEFAULT) { + Paint paint = currentType == TYPE_CHANNEL ? inBubblePaint : outBubblePaintSecond; + if (currentType == TYPE_DEFAULT || currentType == TYPE_CHANNEL) { canvas.drawRoundRect(rectF, rectF.height() * 0.5f, rectF.height() * 0.5f, paint); } else { messageDrawableOut.setBounds((int) rectF.left, (int) rectF.top - AndroidUtilities.dp(2), (int) rectF.right + AndroidUtilities.dp(4), (int) rectF.bottom + AndroidUtilities.dp(2)); @@ -650,10 +685,10 @@ public void draw(Canvas canvas, float alpha) { messageDrawableOut.draw(canvas, paint); } - if (currentType == TYPE_DEFAULT) { + if (currentType == TYPE_DEFAULT || currentType == TYPE_CHANNEL) { bubbleLeft = INNER_RECT_SPACE + AndroidUtilities.dp(5); bubbleTop += BUBBLE_HEIGHT + AndroidUtilities.dp(4); - rectF.set(bubbleLeft, bubbleTop, bubbleLeft + BUBBLE_WIDTH, bubbleTop + BUBBLE_HEIGHT); + rectF.set(bubbleLeft, bubbleTop, bubbleLeft + BUBBLE_WIDTH * (currentType == TYPE_CHANNEL ? 0.8f : 1f), bubbleTop + BUBBLE_HEIGHT); } else { bubbleTop = getMeasuredHeight() * 0.35f; bubbleLeft = getMeasuredWidth() * 0.1f; @@ -662,7 +697,7 @@ public void draw(Canvas canvas, float alpha) { rectF.set(bubbleLeft, bubbleTop, bubbleRight, bubbleBottom); } - if (currentType == TYPE_DEFAULT) { + if (currentType == TYPE_DEFAULT || currentType == TYPE_CHANNEL) { canvas.drawRoundRect(rectF, rectF.height() * 0.5f, rectF.height() * 0.5f, inBubblePaint); } else { messageDrawableIn.setBounds((int) rectF.left - AndroidUtilities.dp(4), (int) rectF.top - AndroidUtilities.dp(2), (int) rectF.right, (int) rectF.bottom + AndroidUtilities.dp(2)); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/spoilers/SpoilerEffect2.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/spoilers/SpoilerEffect2.java index 098537e4462..261be2988ae 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/spoilers/SpoilerEffect2.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/spoilers/SpoilerEffect2.java @@ -6,6 +6,8 @@ import android.content.Context; import android.graphics.Bitmap; import android.graphics.Canvas; +import android.graphics.Color; +import android.graphics.Paint; import android.graphics.SurfaceTexture; import android.hardware.HardwareBuffer; import android.opengl.EGL14; @@ -87,12 +89,12 @@ public static void pause(boolean pause) { private static int getSize() { switch (SharedConfig.getDevicePerformanceClass()) { case SharedConfig.PERFORMANCE_CLASS_HIGH: - return Math.min(900, (int) (Math.min(AndroidUtilities.displaySize.x, AndroidUtilities.displaySize.y) * 1.0f)); + return Math.min(1280, (int) ((AndroidUtilities.displaySize.x + AndroidUtilities.displaySize.y) / 2f * 1.0f)); case SharedConfig.PERFORMANCE_CLASS_AVERAGE: - return Math.min(900, (int) (Math.min(AndroidUtilities.displaySize.x, AndroidUtilities.displaySize.y) * .8f)); + return Math.min(900, (int) ((AndroidUtilities.displaySize.x + AndroidUtilities.displaySize.y) / 2f * .8f)); default: case SharedConfig.PERFORMANCE_CLASS_LOW: - return Math.min(720, (int) (Math.min(AndroidUtilities.displaySize.x, AndroidUtilities.displaySize.y) * .7f)); + return Math.min(720, (int) ((AndroidUtilities.displaySize.x + AndroidUtilities.displaySize.y) / 2f * .7f)); } } @@ -170,6 +172,10 @@ public void draw(Canvas canvas, View view, int w, int h) { } public void draw(Canvas canvas, View view, int w, int h, float alpha) { + draw(canvas, view, w, h, alpha, false); + } + + public void draw(Canvas canvas, View view, int w, int h, float alpha, boolean toBitmap) { if (canvas == null || view == null) { return; } @@ -192,8 +198,18 @@ public void draw(Canvas canvas, View view, int w, int h, float alpha) { if ((index % 4) == 3) { canvas.scale(1, -1, ow / 2f, oh / 2f); } - textureView.setAlpha(alpha); - textureView.draw(canvas); + if (toBitmap) { + Bitmap bitmap = textureView.getBitmap(); + if (bitmap != null) { + Paint paint = new Paint(Paint.DITHER_FLAG | Paint.FILTER_BITMAP_FLAG | Paint.ANTI_ALIAS_FLAG); + paint.setColor(Color.WHITE); + canvas.drawBitmap(bitmap, 0, 0, paint); + bitmap.recycle(); + } + } else { + textureView.setAlpha(alpha); + textureView.draw(canvas); + } canvas.restore(); } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/AcceptDeclineView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/AcceptDeclineView.java index f5a1655ce46..3ededd5cd47 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/AcceptDeclineView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/AcceptDeclineView.java @@ -8,11 +8,13 @@ import android.graphics.Paint; import android.graphics.PorterDuff; import android.graphics.PorterDuffColorFilter; +import android.graphics.PorterDuffXfermode; import android.graphics.Rect; +import android.graphics.RectF; +import android.graphics.drawable.BitmapDrawable; import android.graphics.drawable.Drawable; import android.os.Build; import android.os.Bundle; -import android.os.SystemClock; import android.text.Layout; import android.text.StaticLayout; import android.text.TextPaint; @@ -34,6 +36,7 @@ import org.telegram.messenger.LocaleController; import org.telegram.messenger.R; import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.Components.RLottieDrawable; public class AcceptDeclineView extends View { @@ -76,15 +79,23 @@ public class AcceptDeclineView extends View { boolean retryMod; Drawable rippleDrawable; - private boolean screenWasWakeup; Paint linePaint = new Paint(Paint.ANTI_ALIAS_FLAG); - Drawable arrowDrawable; - - float arrowProgress; + private RLottieDrawable callAcceptDrawable; + private final ImageWithWavesView.AvatarWavesDrawable avatarWavesDrawable; + private final Paint maskPaint = new Paint(Paint.ANTI_ALIAS_FLAG); public AcceptDeclineView(@NonNull Context context) { super(context); + avatarWavesDrawable = new ImageWithWavesView.AvatarWavesDrawable(AndroidUtilities.dp(45), AndroidUtilities.dp(50), AndroidUtilities.dp(8), 4); + avatarWavesDrawable.muteToStatic = true; + avatarWavesDrawable.muteToStaticProgress = 0f; + avatarWavesDrawable.wavesEnter = 0; + avatarWavesDrawable.setAmplitude(0); + + maskPaint.setColor(Color.BLACK); + maskPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_OUT)); + touchSlop = ViewConfiguration.get(context).getScaledTouchSlop(); buttonWidth = AndroidUtilities.dp(60); acceptDrawable = new FabBackgroundDrawable(); @@ -112,15 +123,16 @@ public AcceptDeclineView(@NonNull Context context) { callDrawable = ContextCompat.getDrawable(context, R.drawable.calls_decline).mutate(); cancelDrawable = ContextCompat.getDrawable(context, R.drawable.ic_close_white).mutate(); cancelDrawable.setColorFilter(new PorterDuffColorFilter(Color.BLACK, PorterDuff.Mode.MULTIPLY)); - - acceptCirclePaint.setColor(0x3f45bc4d); + callAcceptDrawable = new RLottieDrawable(R.raw.call_accept, "" + R.raw.call_accept, AndroidUtilities.dp(48), AndroidUtilities.dp(48), true, null); + callAcceptDrawable.setAutoRepeat(1); + callAcceptDrawable.setCustomEndFrame(90); + callAcceptDrawable.setMasterParent(this); + acceptCirclePaint.setColor(Color.WHITE); + acceptCirclePaint.setAlpha(20); rippleDrawable = Theme.createSimpleSelectorCircleDrawable(AndroidUtilities.dp(52), 0, ColorUtils.setAlphaComponent(Color.WHITE, (int) (255 * 0.3f))); rippleDrawable.setCallback(this); - - arrowDrawable = ContextCompat.getDrawable(context, R.drawable.call_arrow_right); } - @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { super.onMeasure(widthMeasureSpec, heightMeasureSpec); @@ -144,10 +156,11 @@ public boolean onTouchEvent(MotionEvent event) { startX = event.getX(); startY = event.getY(); if (leftAnimator == null && declineRect.contains((int) event.getX(), (int) event.getY())) { - rippleDrawable = Theme.createSimpleSelectorCircleDrawable(AndroidUtilities.dp(52), 0, 0xFFFF3846); + rippleDrawable = Theme.createSimpleSelectorCircleDrawable(AndroidUtilities.dp(52), 0, retryMod ? Color.WHITE : 0xFFFF3846); captured = true; leftDrag = true; setPressed(true); + invalidate(); return true; } if (rightAnimator == null && acceptRect.contains((int) event.getX(), (int) event.getY())) { @@ -158,43 +171,12 @@ public boolean onTouchEvent(MotionEvent event) { if (rightAnimator != null) { rightAnimator.cancel(); } + invalidate(); return true; } break; case MotionEvent.ACTION_MOVE: if (captured) { - float dx = event.getX() - startX; - if (!startDrag && Math.abs(dx) > touchSlop) { - if (!retryMod) { - startX = event.getX(); - dx = 0; - startDrag = true; - setPressed(false); - getParent().requestDisallowInterceptTouchEvent(true); - } else { - setPressed(false); - captured = false; - } - } - if (startDrag) { - if (leftDrag) { - leftOffsetX = dx; - if (leftOffsetX < 0) { - leftOffsetX = 0; - } else if (leftOffsetX > maxOffset) { - leftOffsetX = maxOffset; - dispatchTouchEvent(MotionEvent.obtain(SystemClock.uptimeMillis(), SystemClock.uptimeMillis(), MotionEvent.ACTION_UP, 0, 0, 0)); - } - } else { - rigthOffsetX = dx; - if (rigthOffsetX > 0) { - rigthOffsetX = 0; - } else if (rigthOffsetX < -maxOffset) { - rigthOffsetX = -maxOffset; - dispatchTouchEvent(MotionEvent.obtain(SystemClock.uptimeMillis(), SystemClock.uptimeMillis(), MotionEvent.ACTION_UP, 0, 0, 0)); - } - } - } return true; } break; @@ -212,7 +194,7 @@ public boolean onTouchEvent(MotionEvent event) { animator.start(); leftAnimator = animator; if (listener != null) { - if ((!startDrag && Math.abs(dy) < touchSlop && !screenWasWakeup) || leftOffsetX > maxOffset * 0.8f) { + if ((!startDrag && Math.abs(dy) < touchSlop) || leftOffsetX > maxOffset * 0.8f) { listener.onDecline(); } } @@ -226,7 +208,7 @@ public boolean onTouchEvent(MotionEvent event) { animator.start(); rightAnimator = animator; if (listener != null) { - if ((!startDrag && Math.abs(dy) < touchSlop && !screenWasWakeup) || -rigthOffsetX > maxOffset * 0.8f) { + if ((!startDrag && Math.abs(dy) < touchSlop) || -rigthOffsetX > maxOffset * 0.8f) { listener.onAccept(); } } @@ -238,7 +220,6 @@ public boolean onTouchEvent(MotionEvent event) { setPressed(false); break; } - return false; } @@ -275,68 +256,33 @@ protected void onDraw(Canvas canvas) { invalidate(); } - - float k = 0.6f; - if (screenWasWakeup && !retryMod) { - - arrowProgress += 16 / 1500f; - if (arrowProgress > 1) { - arrowProgress = 0; - } - - int cY = (int) (AndroidUtilities.dp(40) + buttonWidth / 2f); - float startX = AndroidUtilities.dp(46) + buttonWidth + AndroidUtilities.dp(8); - float endX = getMeasuredWidth() / 2f - AndroidUtilities.dp(8); - - float lineLength = AndroidUtilities.dp(10); - - float stepProgress = (1f - k) / 3f; - for (int i = 0; i < 3; i++) { - int x = (int) (startX + (endX - startX - lineLength) / 3 * i); - - float alpha = 0.5f; - float startAlphaFrom = i * stepProgress; - if (arrowProgress > startAlphaFrom && arrowProgress < startAlphaFrom + k) { - float p = (arrowProgress - startAlphaFrom) / k; - if (p > 0.5) p = 1f - p; - alpha = 0.5f + p; - } - canvas.save(); - canvas.clipRect(leftOffsetX + AndroidUtilities.dp(46) + buttonWidth / 2,0,getMeasuredHeight(),getMeasuredWidth() >> 1); - arrowDrawable.setAlpha((int) (255 * alpha)); - arrowDrawable.setBounds(x, cY - arrowDrawable.getIntrinsicHeight() / 2, x + arrowDrawable.getIntrinsicWidth(), cY + arrowDrawable.getIntrinsicHeight() / 2); - arrowDrawable.draw(canvas); - canvas.restore(); - - x = (int) (getMeasuredWidth() - (startX + (endX - startX - lineLength) / 3 * i)); - canvas.save(); - canvas.clipRect(getMeasuredWidth() >> 1, 0, rigthOffsetX + getMeasuredWidth() - AndroidUtilities.dp(46) - buttonWidth / 2, getMeasuredHeight()); - canvas.rotate(180, x - arrowDrawable.getIntrinsicWidth() / 2f, cY); - arrowDrawable.setBounds(x - arrowDrawable.getIntrinsicWidth(), cY - arrowDrawable.getIntrinsicHeight() / 2, x, cY + arrowDrawable.getIntrinsicHeight() / 2); - arrowDrawable.draw(canvas); - canvas.restore(); - } - invalidate(); - } bigRadius += AndroidUtilities.dp(8) * 0.005f; canvas.save(); canvas.translate(0, AndroidUtilities.dp(40)); canvas.save(); - canvas.translate(leftOffsetX + AndroidUtilities.dp(46), 0); - declineDrawable.draw(canvas); - - canvas.save(); - canvas.translate(buttonWidth / 2f - declineLayout.getWidth() / 2f, buttonWidth + AndroidUtilities.dp(8)); - declineLayout.draw(canvas); - declineRect.set(AndroidUtilities.dp(46), AndroidUtilities.dp(40), AndroidUtilities.dp(46) + buttonWidth, AndroidUtilities.dp(40) + buttonWidth); - canvas.restore(); + canvas.translate(rigthOffsetX + getMeasuredWidth() - AndroidUtilities.dp(46) - buttonWidth, 0); if (retryMod) { - cancelDrawable.draw(canvas); + canvas.saveLayer(0, 0, getMeasuredWidth(), getMeasuredHeight(), linePaint, Canvas.ALL_SAVE_FLAG); + declineDrawable.draw(canvas); + if (cancelDrawable instanceof BitmapDrawable) { + BitmapDrawable bitmapDrawable = (BitmapDrawable) cancelDrawable; + if (bitmapDrawable.getBitmap() != null) { + canvas.drawBitmap(bitmapDrawable.getBitmap(), null, bitmapDrawable.getBounds(), maskPaint); + } + } + canvas.restore(); } else { + declineDrawable.draw(canvas); callDrawable.draw(canvas); } + canvas.save(); + canvas.translate(buttonWidth / 2f - declineLayout.getWidth() / 2f, buttonWidth + AndroidUtilities.dp(4)); + declineLayout.draw(canvas); + declineRect.set(getMeasuredWidth() - AndroidUtilities.dp(46) - buttonWidth, AndroidUtilities.dp(40), getMeasuredWidth() - AndroidUtilities.dp(46), AndroidUtilities.dp(40) + buttonWidth); + canvas.restore(); + if (leftDrag) { rippleDrawable.setBounds(AndroidUtilities.dp(4), AndroidUtilities.dp(4), buttonWidth - AndroidUtilities.dp(4), buttonWidth - AndroidUtilities.dp(4)); rippleDrawable.draw(canvas); @@ -345,30 +291,30 @@ protected void onDraw(Canvas canvas) { canvas.restore(); canvas.save(); - canvas.translate(rigthOffsetX + getMeasuredWidth() - AndroidUtilities.dp(46) - buttonWidth, 0); + canvas.translate(leftOffsetX + AndroidUtilities.dp(46), 0); if (!retryMod) { - canvas.drawCircle(buttonWidth / 2f, buttonWidth / 2f, buttonWidth / 2f - AndroidUtilities.dp(4) + bigRadius, acceptCirclePaint); - canvas.drawCircle(buttonWidth / 2f, buttonWidth / 2f, buttonWidth / 2f - AndroidUtilities.dp(4) + smallRadius, acceptCirclePaint); + avatarWavesDrawable.update(); + int cx = (int) (buttonWidth / 2f); + avatarWavesDrawable.draw(canvas, cx, cx, this); } acceptDrawable.draw(canvas); - acceptRect.set(getMeasuredWidth() - AndroidUtilities.dp(46) - buttonWidth, AndroidUtilities.dp(40), getMeasuredWidth() - AndroidUtilities.dp(46), AndroidUtilities.dp(40) + buttonWidth); + acceptRect.set(AndroidUtilities.dp(46), AndroidUtilities.dp(40), AndroidUtilities.dp(46) + buttonWidth, AndroidUtilities.dp(40) + buttonWidth); if (retryMod) { canvas.save(); - canvas.translate(buttonWidth / 2f - retryLayout.getWidth() / 2f, buttonWidth + AndroidUtilities.dp(8)); + canvas.translate(buttonWidth / 2f - retryLayout.getWidth() / 2f, buttonWidth + AndroidUtilities.dp(4)); retryLayout.draw(canvas); canvas.restore(); } else { canvas.save(); - canvas.translate(buttonWidth / 2f - acceptLayout.getWidth() / 2f, buttonWidth + AndroidUtilities.dp(8)); + canvas.translate(buttonWidth / 2f - acceptLayout.getWidth() / 2f, buttonWidth + AndroidUtilities.dp(4)); acceptLayout.draw(canvas); canvas.restore(); } canvas.save(); - canvas.translate(-AndroidUtilities.dp(1), AndroidUtilities.dp(1)); - canvas.rotate(-135, callDrawable.getBounds().centerX(), callDrawable.getBounds().centerY()); - callDrawable.draw(canvas); + canvas.translate(AndroidUtilities.dp(6), AndroidUtilities.dp(6)); + callAcceptDrawable.draw(canvas); canvas.restore(); if (!leftDrag) { @@ -378,6 +324,10 @@ protected void onDraw(Canvas canvas) { canvas.restore(); canvas.restore(); + + if (captured) { + invalidate(); + } } public void setListener(Listener listener) { @@ -390,16 +340,43 @@ public interface Listener { void onDecline(); } + private ValueAnimator callAnimator; + public void setRetryMod(boolean retryMod) { this.retryMod = retryMod; if (retryMod) { declineDrawable.setColor(Color.WHITE); - screenWasWakeup = false; } else { - declineDrawable.setColor(0xFFe61e44); + callAcceptDrawable.start(); + avatarWavesDrawable.setShowWaves(true, this); + declineDrawable.setColor(0xFFF01D2C); + + callAnimator = ValueAnimator.ofInt(0, 60, 0, 0, 60, 0, 0, 0, 0); + callAnimator.addUpdateListener(a -> { + avatarWavesDrawable.setAmplitude((int) a.getAnimatedValue()); + }); + + callAnimator.setDuration(1500); + callAnimator.setRepeatMode(ValueAnimator.RESTART); + callAnimator.setRepeatCount(ValueAnimator.INFINITE); + callAnimator.start(); } } + public void stopAnimations() { + if (callAnimator != null) { + callAnimator.cancel(); + callAnimator = null; + callAcceptDrawable.stop(); + } + } + + @Override + protected void onDetachedFromWindow() { + super.onDetachedFromWindow(); + stopAnimations(); + } + @Override protected void drawableStateChanged() { super.drawableStateChanged(); @@ -407,7 +384,7 @@ protected void drawableStateChanged() { } @Override - public boolean verifyDrawable(Drawable drawable) { + public boolean verifyDrawable(@NonNull Drawable drawable) { return rippleDrawable == drawable || super.verifyDrawable(drawable); } @@ -490,10 +467,6 @@ protected void onVirtualViewClick(int virtualViewId) { return accessibilityNodeProvider; } - public void setScreenWasWakeup(boolean screenWasWakeup) { - this.screenWasWakeup = screenWasWakeup; - } - public static abstract class AcceptDeclineAccessibilityNodeProvider extends AccessibilityNodeProvider { private final View hostView; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/EmojiRationalLayout.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/EmojiRationalLayout.java new file mode 100644 index 00000000000..a97fa69e2c8 --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/EmojiRationalLayout.java @@ -0,0 +1,30 @@ +package org.telegram.ui.Components.voip; + +import static org.telegram.messenger.AndroidUtilities.dp; + +import android.annotation.SuppressLint; +import android.content.Context; +import android.graphics.Canvas; +import android.graphics.RectF; +import android.widget.LinearLayout; + +@SuppressLint("ViewConstructor") +public class EmojiRationalLayout extends LinearLayout { + + private final RectF bgRect = new RectF(); + private final VoIPBackgroundProvider backgroundProvider; + + public EmojiRationalLayout(Context context, VoIPBackgroundProvider backgroundProvider) { + super(context); + this.backgroundProvider = backgroundProvider; + backgroundProvider.attach(this); + } + + @Override + protected void dispatchDraw(Canvas canvas) { + bgRect.set(0, 0, getWidth(), getHeight()); + backgroundProvider.setDarkTranslation(getX(), getY()); + canvas.drawRoundRect(bgRect, dp(20), dp(20), backgroundProvider.getDarkPaint()); + super.dispatchDraw(canvas); + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/EndCloseLayout.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/EndCloseLayout.java new file mode 100644 index 00000000000..1f7fc064d17 --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/EndCloseLayout.java @@ -0,0 +1,252 @@ +package org.telegram.ui.Components.voip; + +import android.animation.Animator; +import android.animation.AnimatorListenerAdapter; +import android.animation.AnimatorSet; +import android.animation.ArgbEvaluator; +import android.animation.ValueAnimator; +import android.content.Context; +import android.graphics.Canvas; +import android.graphics.Color; +import android.graphics.Paint; +import android.graphics.PorterDuff; +import android.graphics.PorterDuffColorFilter; +import android.graphics.PorterDuffXfermode; +import android.graphics.RectF; +import android.graphics.drawable.Drawable; +import android.transition.ChangeBounds; +import android.transition.TransitionManager; +import android.transition.TransitionSet; +import android.transition.TransitionValues; +import android.view.Gravity; +import android.view.MotionEvent; +import android.view.View; +import android.view.ViewGroup; +import android.widget.FrameLayout; + +import androidx.annotation.NonNull; +import androidx.core.content.ContextCompat; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.LocaleController; +import org.telegram.messenger.R; +import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.Components.CubicBezierInterpolator; +import org.telegram.ui.Components.LayoutHelper; + +public class EndCloseLayout extends FrameLayout { + private final EndCloseView endCloseView; + private final TransitionSet transitionSet; + private boolean isClosedState = false; + + public EndCloseLayout(@NonNull Context context) { + super(context); + setWillNotDraw(false); + endCloseView = new EndCloseView(context); + this.addView(endCloseView, LayoutHelper.createFrame(52, 52, Gravity.RIGHT)); + + transitionSet = new TransitionSet(); + transitionSet.setOrdering(TransitionSet.ORDERING_TOGETHER); + transitionSet.addTransition(new ChangeBounds() { + + public void captureStartValues(TransitionValues transitionValues) { + super.captureStartValues(transitionValues); + if (transitionValues.view instanceof EndCloseView) { + int color = ((EndCloseView) transitionValues.view).backColor; + int round = ((EndCloseView) transitionValues.view).round; + int declineCallAlpha = ((EndCloseView) transitionValues.view).callDeclineAlpha; + int closeTextAlpha = ((EndCloseView) transitionValues.view).closeTextAlpha; + transitionValues.values.put("back_color_end_close", color); + transitionValues.values.put("round_end_close", round); + transitionValues.values.put("decline_call_alpha_end_close", declineCallAlpha); + transitionValues.values.put("close_text_alpha_end_close", closeTextAlpha); + } + } + + public void captureEndValues(TransitionValues transitionValues) { + super.captureEndValues(transitionValues); + if (transitionValues.view instanceof EndCloseView) { + int color = ((EndCloseView) transitionValues.view).backColor; + int round = ((EndCloseView) transitionValues.view).round; + int declineCallAlpha = ((EndCloseView) transitionValues.view).callDeclineAlpha; + int closeTextAlpha = ((EndCloseView) transitionValues.view).closeTextAlpha; + transitionValues.values.put("back_color_end_close", color); + transitionValues.values.put("round_end_close", round); + transitionValues.values.put("decline_call_alpha_end_close", declineCallAlpha); + transitionValues.values.put("close_text_alpha_end_close", closeTextAlpha); + } + } + + @Override + public Animator createAnimator(ViewGroup sceneRoot, TransitionValues startValues, TransitionValues endValues) { + if (startValues != null && endValues != null && startValues.view instanceof EndCloseView) { + AnimatorSet animatorSet = new AnimatorSet(); + Animator animator = super.createAnimator(sceneRoot, startValues, endValues); + if (animator != null) { + animatorSet.playTogether(animator); + } + final Integer startTextColor = (Integer) startValues.values.get("back_color_end_close"); + final Integer endTextColor = (Integer) endValues.values.get("back_color_end_close"); + final Integer startRound = (Integer) startValues.values.get("round_end_close"); + final Integer endRound = (Integer) endValues.values.get("round_end_close"); + final Integer startDeclineCallAlpha = (Integer) startValues.values.get("decline_call_alpha_end_close"); + final Integer endDeclineCallAlpha = (Integer) endValues.values.get("decline_call_alpha_end_close"); + final Integer startCloseTextAlpha = (Integer) startValues.values.get("close_text_alpha_end_close"); + final Integer endCloseTextAlpha = (Integer) endValues.values.get("close_text_alpha_end_close"); + + ValueAnimator colorAnimator = new ValueAnimator(); + colorAnimator.setIntValues(startTextColor, endTextColor); + colorAnimator.setEvaluator(new ArgbEvaluator()); + colorAnimator.addUpdateListener(a -> ((EndCloseView) startValues.view).backColor = (int) a.getAnimatedValue()); + animatorSet.playTogether(colorAnimator); + + ValueAnimator roundAnimator = ValueAnimator.ofInt(startRound, endRound); + roundAnimator.addUpdateListener(animation -> ((EndCloseView) startValues.view).round = (int) animation.getAnimatedValue()); + animatorSet.playTogether(roundAnimator); + + ValueAnimator declineCallAlphaAnimator = ValueAnimator.ofInt(startDeclineCallAlpha, endDeclineCallAlpha, endDeclineCallAlpha, endDeclineCallAlpha, endDeclineCallAlpha, endDeclineCallAlpha, endDeclineCallAlpha); + declineCallAlphaAnimator.addUpdateListener(animation -> ((EndCloseView) startValues.view).callDeclineAlpha = (int) animation.getAnimatedValue()); + animatorSet.playTogether(declineCallAlphaAnimator); + + ValueAnimator closeTextAlphaAnimator = ValueAnimator.ofInt(startCloseTextAlpha, startCloseTextAlpha, (int) (endCloseTextAlpha * 0.25f), (int) (endCloseTextAlpha * 0.5f), (int) (endCloseTextAlpha * 0.75f), endCloseTextAlpha); + closeTextAlphaAnimator.addUpdateListener(animation -> ((EndCloseView) startValues.view).closeTextAlpha = (int) animation.getAnimatedValue()); + animatorSet.playTogether(closeTextAlphaAnimator); + + animatorSet.addListener(new AnimatorListenerAdapter() { + + @Override + public void onAnimationStart(Animator animation) { + super.onAnimationStart(animation); + startValues.view.setEnabled(false); + } + + @Override + public void onAnimationEnd(Animator animation) { + super.onAnimationEnd(animation); + startValues.view.setEnabled(true); + } + }); + return animatorSet; + } else { + return super.createAnimator(sceneRoot, startValues, endValues); + } + } + }); + transitionSet.setDuration(500); + transitionSet.setInterpolator(CubicBezierInterpolator.DEFAULT); + } + + public EndCloseView getEndCloseView() { + return endCloseView; + } + + public void switchToClose(OnClickListener onClickListener, boolean animate) { + if (isClosedState) return; + isClosedState = true; + + if (animate) { + TransitionManager.beginDelayedTransition(this, transitionSet); + } + + endCloseView.closeTextAlpha = 255; + endCloseView.backColor = 0xFFffffff; + endCloseView.callDeclineAlpha = 0; + endCloseView.round = AndroidUtilities.dp(8); + ViewGroup.LayoutParams lp = endCloseView.getLayoutParams(); + lp.width = ViewGroup.LayoutParams.MATCH_PARENT; + endCloseView.setLayoutParams(lp); + AndroidUtilities.runOnUIThread(() -> endCloseView.setOnClickListener(onClickListener), 500); + } + + static class EndCloseView extends View { + private Drawable rippleDrawable; + private final Paint backgroundPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + private final Paint textPaintMask = new Paint(Paint.ANTI_ALIAS_FLAG); + private final Paint textPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + private final RectF backgroundRect = new RectF(); + private final Drawable callDeclineDrawable; + private final String closeText; + public int backColor = 0xFFf4606c; + public int round = AndroidUtilities.dp(26); + public int callDeclineAlpha = 255; + public int closeTextAlpha = 0; + + public EndCloseView(@NonNull Context context) { + super(context); + callDeclineDrawable = ContextCompat.getDrawable(getContext(), R.drawable.calls_decline).mutate(); + callDeclineDrawable.setColorFilter(new PorterDuffColorFilter(Color.WHITE, PorterDuff.Mode.MULTIPLY)); + textPaintMask.setTextSize(AndroidUtilities.dp(18)); + textPaintMask.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); + textPaintMask.setTextAlign(Paint.Align.CENTER); + textPaintMask.setColor(0xff000000); + textPaintMask.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_OUT)); + textPaint.setTextSize(AndroidUtilities.dp(18)); + textPaint.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); + textPaint.setTextAlign(Paint.Align.CENTER); + textPaint.setColor(Color.BLACK); + setLayerType(View.LAYER_TYPE_HARDWARE, null); + setClickable(true); + closeText = LocaleController.getString("Close", R.string.Close); + } + + @Override + protected void drawableStateChanged() { + super.drawableStateChanged(); + if (rippleDrawable != null) { + rippleDrawable.setState(getDrawableState()); + } + } + + @Override + public boolean verifyDrawable(@NonNull Drawable drawable) { + return rippleDrawable == drawable || super.verifyDrawable(drawable); + } + + @Override + public void jumpDrawablesToCurrentState() { + super.jumpDrawablesToCurrentState(); + if (rippleDrawable != null) { + rippleDrawable.jumpToCurrentState(); + } + } + + @Override + public boolean dispatchTouchEvent(MotionEvent ev) { + if (!isEnabled()) { + return false; + } + return super.dispatchTouchEvent(ev); + } + + @Override + protected void onDraw(Canvas canvas) { + float cx = getWidth() / 2f; + float cy = getHeight() / 2f; + backgroundPaint.setColor(backColor); + backgroundRect.set(0, 0, getWidth(), getHeight()); + canvas.drawRoundRect(backgroundRect, round, round, backgroundPaint); + + callDeclineDrawable.setBounds( + (int) (cx - callDeclineDrawable.getIntrinsicWidth() / 2f), (int) (cy - callDeclineDrawable.getIntrinsicHeight() / 2), + (int) (cx + callDeclineDrawable.getIntrinsicWidth() / 2), (int) (cy + callDeclineDrawable.getIntrinsicHeight() / 2) + ); + callDeclineDrawable.setAlpha(callDeclineAlpha); + callDeclineDrawable.draw(canvas); + + textPaintMask.setAlpha(closeTextAlpha); + int maxDarkAlpha = (int) (255 * 0.15f); + textPaint.setAlpha(maxDarkAlpha * (closeTextAlpha / 255)); + canvas.drawText(closeText, cx, cy + AndroidUtilities.dp(6), textPaintMask); + canvas.drawText(closeText, cx, cy + AndroidUtilities.dp(6), textPaint); + + if (rippleDrawable == null) { + rippleDrawable = Theme.createRadSelectorDrawable(Theme.getColor(Theme.key_listSelector), 8, 8); + rippleDrawable.setCallback(this); + } + rippleDrawable.setBounds(0, 0, getWidth(), getHeight()); + rippleDrawable.draw(canvas); + } + } +} + + diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/FabBackgroundDrawable.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/FabBackgroundDrawable.java index 00970215cdd..f5b08b01b46 100755 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/FabBackgroundDrawable.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/FabBackgroundDrawable.java @@ -62,7 +62,6 @@ protected void onBoundsChange(Rect bounds) { shadowBitmap = Bitmap.createBitmap(size, size, Bitmap.Config.ALPHA_8); Canvas c = new Canvas(shadowBitmap); Paint p = new Paint(Paint.ANTI_ALIAS_FLAG); - p.setShadowLayer(AndroidUtilities.dp(3.33333f), 0, AndroidUtilities.dp(0.666f), 0xFFFFFFFF); c.drawCircle(size / 2, size / 2, size / 2 - AndroidUtilities.dp(4), p); } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/HideEmojiTextView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/HideEmojiTextView.java new file mode 100644 index 00000000000..4b177f69476 --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/HideEmojiTextView.java @@ -0,0 +1,40 @@ +package org.telegram.ui.Components.voip; + +import static org.telegram.messenger.AndroidUtilities.dp; + +import android.annotation.SuppressLint; +import android.content.Context; +import android.graphics.Canvas; +import android.graphics.Color; +import android.graphics.RectF; +import android.view.View; +import android.widget.TextView; +import org.telegram.messenger.R; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.LocaleController; + +@SuppressLint("ViewConstructor") +public class HideEmojiTextView extends TextView { + + private final RectF bgRect = new RectF(); + private final VoIPBackgroundProvider backgroundProvider; + + public HideEmojiTextView(Context context, VoIPBackgroundProvider backgroundProvider) { + super(context); + this.backgroundProvider = backgroundProvider; + backgroundProvider.attach(this); + setText(LocaleController.getString("VoipHideEmoji", R.string.VoipHideEmoji)); + setTextColor(Color.WHITE); + setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); + setPadding(AndroidUtilities.dp(14), AndroidUtilities.dp(4), AndroidUtilities.dp(14), AndroidUtilities.dp(4)); + } + + @Override + protected void onDraw(Canvas canvas) { + bgRect.set(0, 0, getWidth(), getHeight()); + backgroundProvider.setDarkTranslation(getX() + ((View) getParent()).getX(), getY() + ((View) getParent()).getY()); + canvas.drawRoundRect(bgRect, dp(16), dp(16), backgroundProvider.getDarkPaint()); + super.onDraw(canvas); + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/ImageWithWavesView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/ImageWithWavesView.java new file mode 100644 index 00000000000..d7847f72111 --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/ImageWithWavesView.java @@ -0,0 +1,267 @@ +package org.telegram.ui.Components.voip; + +import android.animation.AnimatorSet; +import android.animation.ObjectAnimator; +import android.animation.ValueAnimator; +import android.content.Context; +import android.graphics.Canvas; +import android.graphics.Color; +import android.graphics.drawable.Drawable; +import android.view.Gravity; +import android.view.View; +import android.widget.FrameLayout; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.ImageLocation; +import org.telegram.messenger.LiteMode; +import org.telegram.ui.Components.BackupImageView; +import org.telegram.ui.Components.CubicBezierInterpolator; +import org.telegram.ui.Components.LayoutHelper; + +public class ImageWithWavesView extends FrameLayout { + private final AvatarWavesDrawable avatarWavesDrawable; + private final BackupImageView backupImageView; + private AnimatorSet animatorSet; + private boolean isConnectedCalled; + private boolean isMuted; + private final boolean allowAnimations; + + public ImageWithWavesView(Context context) { + super(context); + avatarWavesDrawable = new AvatarWavesDrawable(AndroidUtilities.dp(104), AndroidUtilities.dp(111), AndroidUtilities.dp(12), 8); + avatarWavesDrawable.setAmplitude(3f); + avatarWavesDrawable.setShowWaves(true, this); + backupImageView = new BackupImageView(context); + addView(backupImageView, LayoutHelper.createFrame(135, 135, Gravity.CENTER)); + setWillNotDraw(false); + animatorSet = new AnimatorSet(); + animatorSet.playTogether( + ObjectAnimator.ofFloat(this, View.SCALE_X, 1.f, 1.05f, 1f, 1.05f, 1f), + ObjectAnimator.ofFloat(this, View.SCALE_Y, 1.f, 1.05f, 1f, 1.05f, 1f) + ); + animatorSet.setInterpolator(CubicBezierInterpolator.EASE_OUT); + animatorSet.setDuration(3000); + allowAnimations = LiteMode.isEnabled(LiteMode.FLAG_CALLS_ANIMATIONS); + if (allowAnimations) { + animatorSet.start(); + } + setClipChildren(false); + } + + public void setImage(ImageLocation imageLocation, String imageFilter, String ext, Drawable thumb, Object parentObject) { + backupImageView.setImage(imageLocation, imageFilter, ext, thumb, parentObject); + } + + public void setImage(ImageLocation imageLocation, String imageFilter, Drawable thumb, Object parentObject) { + backupImageView.setImage(imageLocation, imageFilter, thumb, parentObject); + } + + public void setRoundRadius(int value) { + backupImageView.setRoundRadius(value); + } + + public void setShowWaves(boolean showWaves) { + avatarWavesDrawable.setShowWaves(showWaves, this); + } + + public void setMute(boolean isMuted, boolean isFast) { + if (this.isMuted != isMuted) { + this.isMuted = isMuted; + if (isMuted) { + avatarWavesDrawable.setAmplitude(3f); + } + avatarWavesDrawable.setMuteToStatic(isMuted, isFast, this); + } + } + + public void setAmplitude(double value) { + if (isMuted) return; + if (value > 1.5f) { + avatarWavesDrawable.setAmplitude(value); + } else { + avatarWavesDrawable.setAmplitude(0); + } + } + + public void onConnected() { + if (isConnectedCalled) { + return; + } + if (animatorSet != null) { + animatorSet.cancel(); + } + isConnectedCalled = true; + animatorSet = new AnimatorSet(); + animatorSet.playTogether( + ObjectAnimator.ofFloat(this, View.SCALE_X, this.getScaleX(), 1.05f, 1f), + ObjectAnimator.ofFloat(this, View.SCALE_Y, this.getScaleY(), 1.05f, 1f) + ); + animatorSet.setInterpolator(CubicBezierInterpolator.EASE_OUT); + animatorSet.setDuration(400); + animatorSet.start(); + } + + public void onNeedRating() { + setShowWaves(false); + if (animatorSet != null) { + animatorSet.cancel(); + } + animatorSet = new AnimatorSet(); + animatorSet.playTogether( + ObjectAnimator.ofFloat(this, View.ALPHA, this.getAlpha(), 1f), + ObjectAnimator.ofFloat(this, View.TRANSLATION_Y, this.getTranslationY(), -(float) AndroidUtilities.dp(24)), + ObjectAnimator.ofFloat(this, View.SCALE_X, this.getScaleX(), 0.9f, 1f), + ObjectAnimator.ofFloat(this, View.SCALE_Y, this.getScaleY(), 0.9f, 1f) + ); + animatorSet.setInterpolator(CubicBezierInterpolator.DEFAULT); + animatorSet.setDuration(300); + animatorSet.setStartDelay(250); + animatorSet.start(); + } + + @Override + protected void onDraw(Canvas canvas) { + if(allowAnimations) { + avatarWavesDrawable.update(); + int cx = getWidth() / 2; + int cy = getHeight() / 2; + avatarWavesDrawable.draw(canvas, cx, cy, this); + } + super.onDraw(canvas); + } + + public static class AvatarWavesDrawable { + + float amplitude; + float animateToAmplitude; + float animateAmplitudeDiff; + float wavesEnter = 0f; + boolean showWaves; + + private final VoipBlobDrawable blobDrawable; + private final VoipBlobDrawable blobDrawable2; + + public boolean muteToStatic = false; + public float muteToStaticProgress = 1f; + + private ValueAnimator animator; + private int muteToStaticInvalidationCount; + + public AvatarWavesDrawable(int minRadius, int maxRadius, int diff, int n) { + blobDrawable = new VoipBlobDrawable(n - 1); + blobDrawable2 = new VoipBlobDrawable(n); + blobDrawable.minRadius = minRadius; + blobDrawable.maxRadius = maxRadius; + blobDrawable2.minRadius = minRadius - diff; + blobDrawable2.maxRadius = maxRadius - diff; + blobDrawable.generateBlob(); + blobDrawable2.generateBlob(); + blobDrawable.paint.setColor(Color.WHITE); + blobDrawable.paint.setAlpha(20); + blobDrawable2.paint.setColor(Color.WHITE); + blobDrawable2.paint.setAlpha(36); + } + + public void update() { + if (animateToAmplitude != amplitude) { + amplitude += animateAmplitudeDiff * 16; + if (animateAmplitudeDiff > 0) { + if (amplitude > animateToAmplitude) { + amplitude = animateToAmplitude; + } + } else { + if (amplitude < animateToAmplitude) { + amplitude = animateToAmplitude; + } + } + } + + if (showWaves && wavesEnter != 1f) { + wavesEnter += 16 / 350f; + if (wavesEnter > 1f) { + wavesEnter = 1f; + } + } else if (!showWaves && wavesEnter != 0) { + wavesEnter -= 16 / 350f; + if (wavesEnter < 0f) { + wavesEnter = 0f; + } + } + } + + public void draw(Canvas canvas, float cx, float cy, View parentView) { + float scaleBlob = 0.8f + 0.4f * amplitude; + if (showWaves || wavesEnter != 0) { + canvas.save(); + float wavesEnter = CubicBezierInterpolator.DEFAULT.getInterpolation(this.wavesEnter); + + canvas.scale(scaleBlob * wavesEnter, scaleBlob * wavesEnter, cx, cy); + + blobDrawable.update(amplitude, 1f, muteToStaticProgress); + blobDrawable.draw(cx, cy, canvas, blobDrawable.paint); + + blobDrawable2.update(amplitude, 1f, muteToStaticProgress); + blobDrawable2.draw(cx, cy, canvas, blobDrawable.paint); + canvas.restore(); + } + + if (muteToStatic && muteToStaticInvalidationCount == 0) { + return; + } + + if (muteToStaticInvalidationCount != 0) { + muteToStaticInvalidationCount--; + } + + if (wavesEnter != 0) { + parentView.invalidate(); + } + } + + public void setShowWaves(boolean show, View parentView) { + if (showWaves != show) { + parentView.invalidate(); + } + showWaves = show; + } + + public void setAmplitude(double value) { + float amplitude = (float) value / 80f; + if (!showWaves) { + amplitude = 0; + } + if (amplitude > 1f) { + amplitude = 1f; + } else if (amplitude < 0) { + amplitude = 0; + } + animateToAmplitude = amplitude; + animateAmplitudeDiff = (animateToAmplitude - this.amplitude) / 200; + } + + public void setMuteToStatic(boolean mute, boolean isFast, View parentView) { + if (muteToStatic != mute) { + muteToStatic = mute; + if (animator != null) { + animator.removeAllUpdateListeners(); + animator.cancel(); + } + if (mute) { + animator = ValueAnimator.ofFloat(muteToStaticProgress, 0f); + muteToStaticInvalidationCount = (int) (2000 / AndroidUtilities.screenRefreshTime); + } else { + muteToStaticInvalidationCount = 0; + animator = ValueAnimator.ofFloat(muteToStaticProgress, 1f); + } + animator.addUpdateListener(a -> muteToStaticProgress = (float) a.getAnimatedValue()); + if (isFast) { + animator.setDuration(150); + } else { + animator.setDuration(1000); + } + animator.start(); + parentView.invalidate(); + } + } + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/PrivateVideoPreviewDialogNew.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/PrivateVideoPreviewDialogNew.java new file mode 100644 index 00000000000..02f6189fd69 --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/PrivateVideoPreviewDialogNew.java @@ -0,0 +1,797 @@ +package org.telegram.ui.Components.voip; + +import android.animation.Animator; +import android.animation.AnimatorListenerAdapter; +import android.animation.ValueAnimator; +import android.annotation.SuppressLint; +import android.annotation.TargetApi; +import android.app.Activity; +import android.app.Dialog; +import android.content.Context; +import android.graphics.Bitmap; +import android.graphics.BitmapFactory; +import android.graphics.Camera; +import android.graphics.Canvas; +import android.graphics.Color; +import android.graphics.Matrix; +import android.graphics.Paint; +import android.graphics.Path; +import android.graphics.PorterDuff; +import android.media.projection.MediaProjectionManager; +import android.os.Build; +import android.util.TypedValue; +import android.view.GestureDetector; +import android.view.Gravity; +import android.view.MotionEvent; +import android.view.View; +import android.view.ViewGroup; +import android.widget.FrameLayout; +import android.widget.ImageView; +import android.widget.LinearLayout; +import android.widget.TextView; + +import androidx.annotation.NonNull; +import androidx.core.graphics.ColorUtils; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.ApplicationLoader; +import org.telegram.messenger.LocaleController; +import org.telegram.messenger.R; +import org.telegram.messenger.Utilities; +import org.telegram.messenger.voip.VideoCapturerDevice; +import org.telegram.messenger.voip.VoIPService; +import org.telegram.ui.ActionBar.ActionBar; +import org.telegram.ui.ActionBar.BackDrawable; +import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.Components.BitmapShaderTools; +import org.telegram.ui.Components.CubicBezierInterpolator; +import org.telegram.ui.Components.LayoutHelper; +import org.telegram.ui.Components.MotionBackgroundDrawable; +import org.telegram.ui.LaunchActivity; +import org.webrtc.RendererCommon; + +import java.io.File; +import java.io.FileOutputStream; + +@TargetApi(21) +public abstract class PrivateVideoPreviewDialogNew extends FrameLayout implements VoIPService.StateListener { + + private boolean isDismissed; + + private FrameLayout viewPager; + private TextView positiveButton; + private LinearLayout titlesLayout; + private VoIpBitmapTextView[] titles; + private VoIPTextureView textureView; + private int visibleCameraPage = 1; + private boolean cameraReady; + private ActionBar actionBar; + + private float pageOffset; + private int strangeCurrentPage; + private int realCurrentPage; + private int previousPage = -1; + + private float openProgress1 = 0f; + private float openProgress2 = 0f; + private float closeProgress = 0f; + private float openTranslationX; + private float openTranslationY; + private final float startLocationX; + private final float startLocationY; + private final Path clipPath = new Path(); + private final Camera camera = new Camera(); + private final Matrix matrixRight = new Matrix(); + private final Matrix matrixLeft = new Matrix(); + private boolean positiveButtonDrawText; + + private final BitmapShaderTools bgGreenShaderTools = new BitmapShaderTools(80, 80); + private final BitmapShaderTools bgBlueVioletShaderTools = new BitmapShaderTools(80, 80); + private final MotionBackgroundDrawable bgGreen = new MotionBackgroundDrawable(0xFF5FD051, 0xFF00B48E, 0xFFA9CC66, 0xFF5AB147, 0, false, true); + private final MotionBackgroundDrawable bgBlueViolet = new MotionBackgroundDrawable(0xFF00A3E6, 0xFF296EF7, 0xFF18CEE2, 0xFF3FB2FF, 0, false, true); + private final GestureDetector scrollGestureDetector; + + public PrivateVideoPreviewDialogNew(Context context, float startLocationX, float startLocationY) { + super(context); + + this.startLocationX = startLocationX; + this.startLocationY = startLocationY; + titles = new VoIpBitmapTextView[3]; + scrollGestureDetector = new GestureDetector(context, new GestureDetector.SimpleOnGestureListener() { + private boolean startDragging; + private boolean lockDragging; + + @Override + public boolean onDown(@NonNull MotionEvent e) { + startDragging = true; + return super.onDown(e); + } + + @Override + public boolean onScroll(@NonNull MotionEvent e1, @NonNull MotionEvent e2, float distanceX, float distanceY) { + float dx = e1.getX() - e2.getX(); + float dy = e1.getY() - e2.getY(); + if (Math.abs(dx) > AndroidUtilities.getPixelsInCM(0.4f, true) && Math.abs(dx) / 3 > dy && startDragging && !lockDragging) { + startDragging = false; + Runnable action = () -> { + if (dx > 0) { + if (realCurrentPage < 2) { + setCurrentPage(realCurrentPage + 1, true); + } + } else { + if (realCurrentPage > 0) { + setCurrentPage(realCurrentPage - 1, true); + } + } + lockDragging = false; + }; + if (scrollAnimator != null) { + lockDragging = true; + AndroidUtilities.runOnUIThread(action, scrollAnimator.getDuration() - scrollAnimator.getCurrentPlayTime() + 50); + } else { + action.run(); + } + } + return super.onScroll(e1, e2, distanceX, distanceY); + } + }); + viewPager = new FrameLayout(context) { + @SuppressLint("ClickableViewAccessibility") + @Override + public boolean onTouchEvent(MotionEvent event) { + scrollGestureDetector.onTouchEvent(event); + return super.onTouchEvent(event); + } + }; + viewPager.setClickable(true); + + addView(viewPager, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT)); + + textureView = new VoIPTextureView(context, false, false); + textureView.renderer.setScalingType(RendererCommon.ScalingType.SCALE_ASPECT_FILL); + textureView.scaleType = VoIPTextureView.SCALE_TYPE_FIT; + textureView.clipToTexture = true; + textureView.renderer.setAlpha(0); + textureView.renderer.setRotateTextureWithScreen(true); + textureView.renderer.setUseCameraRotation(true); + addView(textureView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT)); + + actionBar = new ActionBar(context); + actionBar.setBackButtonDrawable(new BackDrawable(false)); + actionBar.setBackgroundColor(Color.TRANSPARENT); + actionBar.setItemsColor(Theme.getColor(Theme.key_voipgroup_actionBarItems), false); + actionBar.setOccupyStatusBar(true); + actionBar.setActionBarMenuOnItemClick(new ActionBar.ActionBarMenuOnItemClick() { + @Override + public void onItemClick(int id) { + if (id == -1) { + dismiss(false, false); + } + } + }); + addView(actionBar); + + positiveButton = new TextView(getContext()) { + private final Paint whitePaint = new Paint(); + private final Paint[] gradientPaint = new Paint[titles.length]; + + { + bgGreen.setBounds(0, 0, 80, 80); + bgBlueViolet.setBounds(0, 0, 80, 80); + bgGreenShaderTools.setBounds(0, 0, 80, 80); + bgBlueVioletShaderTools.setBounds(0, 0, 80, 80); + bgGreen.setAlpha(255); + bgBlueViolet.setAlpha(255); + bgGreenShaderTools.getCanvas().drawColor(Color.TRANSPARENT, PorterDuff.Mode.CLEAR); + bgBlueVioletShaderTools.getCanvas().drawColor(Color.TRANSPARENT, PorterDuff.Mode.CLEAR); + bgGreen.draw(bgGreenShaderTools.getCanvas()); + bgBlueViolet.draw(bgBlueVioletShaderTools.getCanvas()); + whitePaint.setColor(Color.WHITE); + } + + @Override + protected void onSizeChanged(int w, int h, int oldw, int oldh) { + super.onSizeChanged(w, h, oldw, oldh); + for (int a = 0; a < gradientPaint.length; a++) { + if (a == 0) { + gradientPaint[a] = bgGreenShaderTools.paint; + } else if (a == 1) { + gradientPaint[a] = bgBlueVioletShaderTools.paint; + } else { + gradientPaint[a] = bgGreenShaderTools.paint; + } + } + } + + @Override + protected void onDraw(Canvas canvas) { + bgGreenShaderTools.setBounds(-getX(), -getY(), PrivateVideoPreviewDialogNew.this.getWidth() - getX(), PrivateVideoPreviewDialogNew.this.getHeight() - getY()); + bgBlueVioletShaderTools.setBounds(-getX(), -getY(), PrivateVideoPreviewDialogNew.this.getWidth() - getX(), PrivateVideoPreviewDialogNew.this.getHeight() - getY()); + + AndroidUtilities.rectTmp.set(0, 0, getMeasuredWidth(), getMeasuredHeight()); + gradientPaint[strangeCurrentPage].setAlpha(255); + int round = AndroidUtilities.dp(8) + (int) ((AndroidUtilities.dp(26) - AndroidUtilities.dp(8)) * (1f - openProgress1)); + canvas.drawRoundRect(AndroidUtilities.rectTmp, round, round, gradientPaint[strangeCurrentPage]); + if (pageOffset > 0 && strangeCurrentPage + 1 < gradientPaint.length) { + gradientPaint[strangeCurrentPage + 1].setAlpha((int) (255 * pageOffset)); + canvas.drawRoundRect(AndroidUtilities.rectTmp, round, round, gradientPaint[strangeCurrentPage + 1]); + } + if (openProgress1 < 1f) { + whitePaint.setAlpha((int) (255 * (1f - openProgress1))); + canvas.drawRoundRect(AndroidUtilities.rectTmp, round, round, whitePaint); + } + super.onDraw(canvas); + + if (positiveButtonDrawText) { + int xPos = (getWidth() / 2); + int yPos = (int) ((getHeight() / 2) - ((positiveButton.getPaint().descent() + positiveButton.getPaint().ascent()) / 2)); + canvas.drawText(LocaleController.getString("VoipShareVideo", R.string.VoipShareVideo), xPos, yPos, positiveButton.getPaint()); + } + } + }; + positiveButton.setMaxLines(1); + positiveButton.setEllipsize(null); + positiveButton.setMinWidth(AndroidUtilities.dp(64)); + positiveButton.setTag(Dialog.BUTTON_POSITIVE); + positiveButton.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 14); + positiveButton.setTextColor(Theme.getColor(Theme.key_voipgroup_nameText)); + positiveButton.setGravity(Gravity.CENTER); + positiveButton.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); + positiveButton.getPaint().setTextAlign(Paint.Align.CENTER); + + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { + positiveButton.setForeground(Theme.createSimpleSelectorRoundRectDrawable(AndroidUtilities.dp(8), Color.TRANSPARENT, ColorUtils.setAlphaComponent(Theme.getColor(Theme.key_voipgroup_nameText), (int) (255 * 0.3f)))); + } + positiveButton.setPadding(0, AndroidUtilities.dp(12), 0, AndroidUtilities.dp(12)); + positiveButton.setOnClickListener(view -> { + if (isDismissed) { + return; + } + if (realCurrentPage == 0) { + MediaProjectionManager mediaProjectionManager = (MediaProjectionManager) getContext().getSystemService(Context.MEDIA_PROJECTION_SERVICE); + ((Activity) getContext()).startActivityForResult(mediaProjectionManager.createScreenCaptureIntent(), LaunchActivity.SCREEN_CAPTURE_REQUEST_CODE); + } else { + dismiss(false, true); + } + }); + + addView(positiveButton, LayoutHelper.createFrame(52, 52, Gravity.BOTTOM | Gravity.CENTER_HORIZONTAL, 0, 0, 0, 80)); + + titlesLayout = new LinearLayout(context) { + + @Override + protected void dispatchDraw(Canvas canvas) { + int halfWidth = getWidth() / 2; + int halfHeight = getHeight() / 2; + + camera.save(); + camera.rotateY(7); + camera.getMatrix(matrixRight); + camera.restore(); + matrixRight.preTranslate(-halfWidth, -halfHeight); + matrixRight.postTranslate(halfWidth, halfHeight); + canvas.save(); + canvas.clipRect(halfWidth, 0, getWidth(), getHeight()); + canvas.concat(matrixRight); + super.dispatchDraw(canvas); + canvas.restore(); + + camera.save(); + camera.rotateY(-7); + camera.getMatrix(matrixLeft); + camera.restore(); + matrixLeft.preTranslate(-halfWidth, -halfHeight); + matrixLeft.postTranslate(halfWidth, halfHeight); + canvas.save(); + canvas.clipRect(0, 0, halfWidth, getHeight()); + canvas.concat(matrixLeft); + super.dispatchDraw(canvas); + canvas.restore(); + } + }; + titlesLayout.setClipChildren(false); + addView(titlesLayout, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 64, Gravity.BOTTOM)); + + for (int i = 0; i < titles.length; i++) { + String text; + if (i == 0) { + text = LocaleController.getString("VoipPhoneScreen", R.string.VoipPhoneScreen); + } else if (i == 1) { + text = LocaleController.getString("VoipFrontCamera", R.string.VoipFrontCamera); + } else { + text = LocaleController.getString("VoipBackCamera", R.string.VoipBackCamera); + } + titles[i] = new VoIpBitmapTextView(context, text); + titles[i].setPadding(AndroidUtilities.dp(16), 0, AndroidUtilities.dp(10), 0); + titlesLayout.addView(titles[i], LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, LayoutHelper.MATCH_PARENT)); + final int num = i; + titles[i].setOnClickListener(view -> { + if (scrollAnimator != null || view.getAlpha() == 0f) return; + setCurrentPage(num, true); + }); + } + + setWillNotDraw(false); + + VoIPService service = VoIPService.getSharedInstance(); + if (service != null) { + textureView.renderer.setMirror(service.isFrontFaceCamera()); + textureView.renderer.init(VideoCapturerDevice.getEglBase().getEglBaseContext(), new RendererCommon.RendererEvents() { + @Override + public void onFirstFrameRendered() { + + } + + @Override + public void onFrameResolutionChanged(int videoWidth, int videoHeight, int rotation) { + + } + }); + service.setLocalSink(textureView.renderer, false); + } + createPages(viewPager); + + ValueAnimator openAnimator1 = ValueAnimator.ofFloat(0f, 1f); + openAnimator1.addUpdateListener(animation -> { + openProgress1 = (float) animation.getAnimatedValue(); + float startLocationXWithOffset = startLocationX + AndroidUtilities.dp(28); + float startLocationYWithOffset = startLocationY + AndroidUtilities.dp(52); + openTranslationX = startLocationXWithOffset - (startLocationXWithOffset * openProgress1); + openTranslationY = startLocationYWithOffset - (startLocationYWithOffset * openProgress1); + invalidate(); + }); + openAnimator1.addListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + if (!isDismissed) { + afterOpened(); + } + } + }); + ValueAnimator openAnimator2 = ValueAnimator.ofFloat(0f, 1f); + openAnimator2.addUpdateListener(animation -> { + openProgress2 = (float) animation.getAnimatedValue(); + int w = AndroidUtilities.displaySize.x - AndroidUtilities.dp(36) - AndroidUtilities.dp(52); + positiveButton.getLayoutParams().width = AndroidUtilities.dp(52) + (int) (w * openProgress2); + positiveButton.requestLayout(); + }); + int openAnimationTime = 320; + openAnimator1.setInterpolator(CubicBezierInterpolator.DEFAULT); + openAnimator1.setDuration(openAnimationTime); + openAnimator1.start(); + openAnimator2.setInterpolator(CubicBezierInterpolator.DEFAULT); + openAnimator2.setDuration(openAnimationTime); + openAnimator2.setStartDelay(openAnimationTime / 10); + openAnimator2.start(); + titlesLayout.setAlpha(0f); + titlesLayout.setScaleY(0.8f); + titlesLayout.setScaleX(0.8f); + titlesLayout.animate().alpha(1f).scaleX(1f).scaleY(1f).setStartDelay(120).setDuration(250).start(); + positiveButton.setTranslationY(AndroidUtilities.dp(53)); + positiveButton.setTranslationX(startLocationX - (AndroidUtilities.displaySize.x / 2f) + AndroidUtilities.dp(8) + AndroidUtilities.dp(26)); + positiveButton.animate().translationY(0).translationX(0).setDuration(openAnimationTime).start(); + positiveButtonDrawText = true; + setCurrentPage(1, false); + } + + private void showStub(boolean show, boolean animate) { + ImageView imageView = viewPager.findViewWithTag("image_stab"); + if (!show) { + imageView.setVisibility(GONE); + return; + } + Bitmap bitmap = null; + try { + File file = new File(ApplicationLoader.getFilesDirFixed(), "cthumb" + visibleCameraPage + ".jpg"); + bitmap = BitmapFactory.decodeFile(file.getAbsolutePath()); + } catch (Throwable ignore) { + + } + + if (bitmap != null && bitmap.getPixel(0, 0) != 0) { + imageView.setImageBitmap(bitmap); + } else { + imageView.setImageResource(R.drawable.icplaceholder); + } + if (animate) { + imageView.setVisibility(VISIBLE); + imageView.setAlpha(0f); + imageView.animate().alpha(1f).setDuration(250).start(); + } else { + imageView.setAlpha(1f); + imageView.setVisibility(VISIBLE); + } + } + + private ValueAnimator scrollAnimator; + + private void setCurrentPage(int position, boolean animate) { + if (strangeCurrentPage == position || realCurrentPage == position) return; + + if (animate) { + if (realCurrentPage == 0) { + //switch from screencast to any camera + if (visibleCameraPage != position) { + visibleCameraPage = position; + cameraReady = false; + showStub(true, true); + if (VoIPService.getSharedInstance() != null) { + VoIPService.getSharedInstance().switchCamera(); + } + } else { + showStub(false, false); + textureView.animate().alpha(1f).setDuration(250).start(); + } + } else { + if (position == 0) { + //switch to screencast from any camera + viewPager.findViewWithTag("screencast_stub").setVisibility(VISIBLE); + saveLastCameraBitmap(); + showStub(false, false); + textureView.animate().alpha(0f).setDuration(250).start(); + } else { + //switch between cameras + saveLastCameraBitmap(); + visibleCameraPage = position; + cameraReady = false; + showStub(true, false); + textureView.animate().alpha(0f).setDuration(250).start(); + if (VoIPService.getSharedInstance() != null) { + VoIPService.getSharedInstance().switchCamera(); + } + } + } + + if (position > realCurrentPage) { + //to the right + previousPage = realCurrentPage; + realCurrentPage = realCurrentPage + 1; + scrollAnimator = ValueAnimator.ofFloat(0.1f, 1f); + } else { + //to the left + previousPage = realCurrentPage; + realCurrentPage = realCurrentPage - 1; + strangeCurrentPage = position; + scrollAnimator = ValueAnimator.ofFloat(1f, 0f); + } + + scrollAnimator.addUpdateListener(animation -> { + pageOffset = (float) animation.getAnimatedValue(); + updateTitlesLayout(); + }); + scrollAnimator.addListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + previousPage = -1; + strangeCurrentPage = position; + pageOffset = 0; + scrollAnimator = null; + updateTitlesLayout(); + } + }); + scrollAnimator.setInterpolator(CubicBezierInterpolator.DEFAULT); + scrollAnimator.setDuration(350); + scrollAnimator.start(); + } else { + realCurrentPage = position; + strangeCurrentPage = position; + pageOffset = 0; + updateTitlesLayout(); + textureView.setVisibility(VISIBLE); + cameraReady = false; + visibleCameraPage = 1; + showStub(true, false); + } + } + + private void createPages(FrameLayout container) { + { + FrameLayout frameLayout = new FrameLayout(getContext()); + frameLayout.setBackground(new MotionBackgroundDrawable(0xff212E3A, 0xff2B5B4D, 0xff245863, 0xff274558, true)); + + ImageView imageView = new ImageView(getContext()); + imageView.setScaleType(ImageView.ScaleType.CENTER); + imageView.setImageResource(R.drawable.screencast_big); + frameLayout.addView(imageView, LayoutHelper.createFrame(82, 82, Gravity.CENTER, 0, 0, 0, 60)); + + TextView textView = new TextView(getContext()); + textView.setText(LocaleController.getString("VoipVideoPrivateScreenSharing", R.string.VoipVideoPrivateScreenSharing)); + textView.setGravity(Gravity.CENTER); + textView.setLineSpacing(AndroidUtilities.dp(2), 1.0f); + textView.setTextColor(0xffffffff); + textView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 15); + textView.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); + frameLayout.addView(textView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER, 21, 28, 21, 0)); + frameLayout.setTag("screencast_stub"); + frameLayout.setVisibility(GONE); + container.addView(frameLayout); + } + { + ImageView imageView = new ImageView(getContext()); + imageView.setTag("image_stab"); + imageView.setImageResource(R.drawable.icplaceholder); + imageView.setScaleType(ImageView.ScaleType.FIT_XY); + container.addView(imageView); + } + } + + @Override + protected void dispatchDraw(Canvas canvas) { + if (openProgress1 < 1f) { + int maxWidth = AndroidUtilities.displaySize.x; + int maxHeight = AndroidUtilities.displaySize.y + AndroidUtilities.statusBarHeight + AndroidUtilities.navigationBarHeight; + float rounded = AndroidUtilities.dp(28) - (AndroidUtilities.dp(28) * openProgress1); + clipPath.reset(); + clipPath.addCircle(startLocationX + AndroidUtilities.dp(33.5f), startLocationY + AndroidUtilities.dp(26.6f), AndroidUtilities.dp(26), Path.Direction.CW); + int minWidth = AndroidUtilities.dp(52); + int minHeight = AndroidUtilities.dp(52); + int width = AndroidUtilities.lerp(minWidth, maxWidth, openProgress1); + int height = AndroidUtilities.lerp(minHeight, maxHeight, openProgress1); + float x = openTranslationX - ((1f - openProgress1) * AndroidUtilities.dp(20f)); + float y = openTranslationY - ((1f - openProgress1) * AndroidUtilities.dp(51)); + clipPath.addRoundRect(x, y, x + width, y + height, rounded, rounded, Path.Direction.CW); + canvas.clipPath(clipPath); + } + + if (closeProgress > 0f) { + int[] loc = getFloatingViewLocation(); + int x = (int) (closeProgress * loc[0]); + int y = (int) (closeProgress * loc[1]); + int destWidth = loc[2]; + int w = AndroidUtilities.displaySize.x; + float currentWidth = destWidth + ((w - destWidth) * (1f - closeProgress)); + float scale = currentWidth / w; + clipPath.reset(); + clipPath.addRoundRect(0f, 0f, getWidth() * scale, getHeight() * scale, AndroidUtilities.dp(6), AndroidUtilities.dp(6), Path.Direction.CW); + canvas.translate(x, y); + canvas.clipPath(clipPath); + canvas.scale(scale, scale); + } + + super.dispatchDraw(canvas); + } + + public void dismiss(boolean screencast, boolean apply) { + if (isDismissed || openProgress1 != 1f) { + return; + } + beforeClosed(); + isDismissed = true; + saveLastCameraBitmap(); + onDismiss(screencast, apply); + if (isHasVideoOnMainScreen() && apply) { + ValueAnimator closeAnimator = ValueAnimator.ofFloat(0f, 1f); + closeAnimator.addUpdateListener(animation -> { + closeProgress = (float) animation.getAnimatedValue(); + invalidate(); + }); + closeAnimator.setInterpolator(CubicBezierInterpolator.DEFAULT); + closeAnimator.setStartDelay(60); + closeAnimator.setDuration(350); + closeAnimator.addListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + super.onAnimationEnd(animation); + if (getParent() != null) { + ((ViewGroup) getParent()).removeView(PrivateVideoPreviewDialogNew.this); + } + } + }); + closeAnimator.start(); + positiveButton.animate().setStartDelay(60).alpha(0f).setDuration(100).start(); + actionBar.animate().setStartDelay(60).alpha(0f).setDuration(100).start(); + titlesLayout.animate().setStartDelay(60).alpha(0f).setDuration(100).start(); + } else { + if (apply) { + animate().setStartDelay(60).alpha(0f).setDuration(350).setInterpolator(CubicBezierInterpolator.DEFAULT).setListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + super.onAnimationEnd(animation); + if (getParent() != null) { + ((ViewGroup) getParent()).removeView(PrivateVideoPreviewDialogNew.this); + } + } + }); + } else { + ValueAnimator openAnimator1 = ValueAnimator.ofFloat(1f, 0f); + openAnimator1.addUpdateListener(animation -> { + openProgress1 = (float) animation.getAnimatedValue(); + float startLocationXWithOffset = startLocationX + AndroidUtilities.dp(28); + float startLocationYWithOffset = startLocationY + AndroidUtilities.dp(52); + openTranslationX = startLocationXWithOffset - (startLocationXWithOffset * openProgress1); + openTranslationY = startLocationYWithOffset - (startLocationYWithOffset * openProgress1); + invalidate(); + }); + openAnimator1.addListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + if (getParent() != null) { + ((ViewGroup) getParent()).removeView(PrivateVideoPreviewDialogNew.this); + } + } + }); + ValueAnimator openAnimator2 = ValueAnimator.ofFloat(1f, 0f); + openAnimator2.addUpdateListener(animation -> { + openProgress2 = (float) animation.getAnimatedValue(); + int w = AndroidUtilities.displaySize.x - AndroidUtilities.dp(36) - AndroidUtilities.dp(52); + positiveButton.getLayoutParams().width = AndroidUtilities.dp(52) + (int) (w * openProgress2); + positiveButton.requestLayout(); + }); + int closeAnimationTime = 320; + openAnimator1.setInterpolator(CubicBezierInterpolator.DEFAULT); + openAnimator1.setDuration(closeAnimationTime); + openAnimator1.start(); + openAnimator2.setInterpolator(CubicBezierInterpolator.DEFAULT); + openAnimator2.setDuration(closeAnimationTime); + openAnimator2.start(); + titlesLayout.setAlpha(1f); + titlesLayout.setScaleY(1f); + titlesLayout.setScaleX(1f); + titlesLayout.animate().alpha(0f).scaleX(0.8f).scaleY(0.8f).setDuration(250).start(); + positiveButton.animate().translationY(AndroidUtilities.dp(53)).translationX(startLocationX - (AndroidUtilities.displaySize.x / 2f) + AndroidUtilities.dp(8) + AndroidUtilities.dp(26)).setDuration((long) (closeAnimationTime * 0.6f)).start(); + animate().alpha(0f).setDuration((long) (closeAnimationTime * (2f / 8f))).setStartDelay((long) (closeAnimationTime * (6f / 8f))).start(); + } + } + + invalidate(); + } + + public void setBottomPadding(int padding) { + LayoutParams layoutParams = (LayoutParams) positiveButton.getLayoutParams(); + layoutParams.bottomMargin = AndroidUtilities.dp(80) + padding; + + layoutParams = (LayoutParams) titlesLayout.getLayoutParams(); + layoutParams.bottomMargin = padding; + } + + private void updateTitlesLayout() { + View current = titles[strangeCurrentPage]; + View next = strangeCurrentPage < titles.length - 1 ? titles[strangeCurrentPage + 1] : null; + + float currentCx = current.getLeft() + current.getMeasuredWidth() / 2; + float tx = getMeasuredWidth() / 2 - currentCx; + if (next != null) { + float nextCx = next.getLeft() + next.getMeasuredWidth() / 2; + tx -= (nextCx - currentCx) * pageOffset; + } + for (int i = 0; i < titles.length; i++) { + float alpha; + float scale; + if (i < strangeCurrentPage || i > strangeCurrentPage + 1) { + alpha = 0.7f; + scale = 0.9f; + } else if (i == strangeCurrentPage) { + //movement to the right or selected + alpha = 1.0f - 0.3f * pageOffset; + scale = 1.0f - 0.1f * pageOffset; + } else { + alpha = 0.7f + 0.3f * pageOffset; + scale = 0.9f + 0.1f * pageOffset; + } + titles[i].setAlpha(alpha); + titles[i].setScaleX(scale); + titles[i].setScaleY(scale); + titles[i].setTranslationX(tx); + } + positiveButton.invalidate(); + if (realCurrentPage == 0) { + titles[2].setAlpha(0.7f * pageOffset); + } + if (realCurrentPage == 2) { + if (pageOffset > 0f) { + titles[0].setAlpha(0.7f * (1f - pageOffset)); + } else { + titles[0].setAlpha(0f); + } + } + if (realCurrentPage == 1) { + if (previousPage == 0) { + titles[2].setAlpha(0.7f * pageOffset); + } + if (previousPage == 2) { + titles[0].setAlpha(0.7f * (1f - pageOffset)); + } + } + } + + @Override + protected void onAttachedToWindow() { + super.onAttachedToWindow(); + VoIPService service = VoIPService.getSharedInstance(); + if (service != null) service.registerStateListener(this); + } + + @Override + protected void onDetachedFromWindow() { + super.onDetachedFromWindow(); + VoIPService service = VoIPService.getSharedInstance(); + if (service != null) service.unregisterStateListener(this); + } + + private void saveLastCameraBitmap() { + if (!cameraReady) { + return; + } + try { + Bitmap bitmap = textureView.renderer.getBitmap(); + if (bitmap != null) { + Bitmap newBitmap = Bitmap.createBitmap(bitmap, 0, 0, bitmap.getWidth(), bitmap.getHeight(), textureView.renderer.getMatrix(), true); + bitmap.recycle(); + bitmap = newBitmap; + Bitmap lastBitmap = Bitmap.createScaledBitmap(bitmap, 80, (int) (bitmap.getHeight() / (bitmap.getWidth() / 80.0f)), true); + if (lastBitmap != null) { + if (lastBitmap != bitmap) { + bitmap.recycle(); + } + Utilities.blurBitmap(lastBitmap, 7, 1, lastBitmap.getWidth(), lastBitmap.getHeight(), lastBitmap.getRowBytes()); + File file = new File(ApplicationLoader.getFilesDirFixed(), "cthumb" + visibleCameraPage + ".jpg"); + FileOutputStream stream = new FileOutputStream(file); + lastBitmap.compress(Bitmap.CompressFormat.JPEG, 87, stream); + stream.close(); + View view = viewPager.findViewWithTag("image_stab"); + if (view instanceof ImageView) { + ((ImageView) view).setImageBitmap(lastBitmap); + } + } + } + } catch (Throwable ignore) { + + } + } + + @Override + public void onCameraFirstFrameAvailable() { + if (!cameraReady) { + cameraReady = true; + if (realCurrentPage != 0) textureView.animate().alpha(1f).setDuration(250).start(); + } + } + + @Override + protected void onLayout(boolean changed, int left, int top, int right, int bottom) { + super.onLayout(changed, left, top, right, bottom); + updateTitlesLayout(); + } + + @Override + public boolean onTouchEvent(MotionEvent event) { + return true; + } + + protected void onDismiss(boolean screencast, boolean apply) { + + } + + protected int[] getFloatingViewLocation() { + return null; + } + + protected boolean isHasVideoOnMainScreen() { + return false; + } + + protected void afterOpened() { + + } + + protected void beforeClosed() { + + } + + @Override + protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { + super.onMeasure(widthMeasureSpec, heightMeasureSpec); + measureChildWithMargins(titlesLayout, MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED), 0, MeasureSpec.makeMeasureSpec(AndroidUtilities.dp(64), MeasureSpec.EXACTLY), 0); + } + + @Override + public void onCameraSwitch(boolean isFrontFace) { + update(); + } + + public void update() { + if (VoIPService.getSharedInstance() != null) { + textureView.renderer.setMirror(VoIPService.getSharedInstance().isFrontFaceCamera()); + } + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/RateCallLayout.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/RateCallLayout.java new file mode 100644 index 00000000000..9c93ddc7d8c --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/RateCallLayout.java @@ -0,0 +1,274 @@ +package org.telegram.ui.Components.voip; + +import static org.telegram.messenger.AndroidUtilities.dp; + +import android.animation.AnimatorSet; +import android.animation.ObjectAnimator; +import android.annotation.SuppressLint; +import android.content.Context; +import android.graphics.Canvas; +import android.graphics.Color; +import android.graphics.RectF; +import android.graphics.drawable.Drawable; +import android.util.TypedValue; +import android.view.Gravity; +import android.view.MotionEvent; +import android.view.View; +import android.widget.FrameLayout; +import android.widget.TextView; + +import androidx.annotation.NonNull; +import androidx.core.graphics.ColorUtils; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.LocaleController; +import org.telegram.messenger.R; +import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.Components.CubicBezierInterpolator; +import org.telegram.ui.Components.LayoutHelper; +import org.telegram.ui.Components.RLottieImageView; + +@SuppressLint("ViewConstructor") +public class RateCallLayout extends FrameLayout { + + private final RateCallContainer rateCallContainer; + private final FrameLayout starsContainer; + private final StarContainer[] startsViews = new StarContainer[5]; + private final VoIPBackgroundProvider backgroundProvider; + private OnRateSelected onRateSelected; + + public interface OnRateSelected { + void onRateSelected(int count); + } + + public RateCallLayout(@NonNull Context context, VoIPBackgroundProvider backgroundProvider) { + super(context); + this.backgroundProvider = backgroundProvider; + setWillNotDraw(false); + rateCallContainer = new RateCallContainer(context, backgroundProvider); + starsContainer = new FrameLayout(context); + rateCallContainer.setVisibility(GONE); + starsContainer.setVisibility(GONE); + + int starMargin = 4; + for (int i = 0; i < 5; i++) { + startsViews[i] = new StarContainer(context); + startsViews[i].setAllStarsProvider(() -> startsViews); + startsViews[i].setOnSelectedStar((x, y, starsCount) -> { + if (starsCount >= 4) { + final RLottieImageView img = new RLottieImageView(context); + final int rateAnimationSize = 133; + final int rateAnimationSizeDp = AndroidUtilities.dp(rateAnimationSize); + img.setAnimation(R.raw.rate, rateAnimationSize, rateAnimationSize); + int[] location = new int[2]; + getLocationOnScreen(location); + int viewX = location[0]; + int viewY = location[1]; + addView(img, LayoutHelper.createFrame(rateAnimationSize, rateAnimationSize)); + img.setTranslationX((x - viewX) - (rateAnimationSizeDp / 2f)); + img.setTranslationY((y - viewY) - (rateAnimationSizeDp / 2f)); + + img.setOnAnimationEndListener(() -> AndroidUtilities.runOnUIThread(() -> removeView(img))); + img.playAnimation(); + } + if (onRateSelected != null) onRateSelected.onRateSelected(starsCount); + }, i); + starsContainer.addView(startsViews[i], LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.TOP | Gravity.LEFT, i * (StarContainer.starSize + starMargin), 0, 0, 0)); + } + + addView(rateCallContainer, LayoutHelper.createFrame(300, 152, Gravity.TOP | Gravity.CENTER_HORIZONTAL, 0, 0, 0, 0)); + addView(starsContainer, LayoutHelper.createFrame((StarContainer.starSize * 5) + (starMargin * 4), 100, Gravity.TOP | Gravity.CENTER_HORIZONTAL, 0, 90, 0, 0)); + } + + public void show(OnRateSelected onRateSelected) { + this.onRateSelected = onRateSelected; + + rateCallContainer.setVisibility(VISIBLE); + starsContainer.setVisibility(VISIBLE); + + AnimatorSet backSet = new AnimatorSet(); + backSet.playTogether( + ObjectAnimator.ofFloat(rateCallContainer, View.ALPHA, 0, 1f), + ObjectAnimator.ofFloat(rateCallContainer, View.SCALE_X, 0.7f, 1f), + ObjectAnimator.ofFloat(rateCallContainer, View.SCALE_Y, 0.7f, 1f), + ObjectAnimator.ofFloat(rateCallContainer, View.TRANSLATION_Y, AndroidUtilities.dp(24), 0) + ); + backSet.setInterpolator(CubicBezierInterpolator.DEFAULT); + backSet.setDuration(250); + + for (int i = 0; i < startsViews.length; i++) { + AnimatorSet starSet = new AnimatorSet(); + startsViews[i].setAlpha(0f); + starSet.playTogether( + ObjectAnimator.ofFloat(startsViews[i], View.ALPHA, 0, 1f), + ObjectAnimator.ofFloat(startsViews[i], View.SCALE_X, 0.3f, 1f), + ObjectAnimator.ofFloat(startsViews[i], View.SCALE_Y, 0.3f, 1f), + ObjectAnimator.ofFloat(startsViews[i], View.TRANSLATION_Y, AndroidUtilities.dp(30), 0) + ); + starSet.setDuration(250); + starSet.setStartDelay(i * 16L); + starSet.start(); + } + backSet.start(); + } + + public static class StarContainer extends FrameLayout { + + private static final int starSize = 37; + + interface OnSelectedStar { + void onSelected(float x, float y, int starsCount); + } + + interface AllStarsProvider { + StarContainer[] getAllStartsViews(); + } + + public RLottieImageView defaultStar; + public RLottieImageView selectedStar; + private final Drawable rippleDrawable; + private OnSelectedStar onSelectedStar; + private AllStarsProvider allStarsProvider; + private int pos = 0; + + public void setOnSelectedStar(OnSelectedStar onSelectedStar, int pos) { + this.onSelectedStar = onSelectedStar; + this.pos = pos; + } + + public void setAllStarsProvider(AllStarsProvider allStarsProvider) { + this.allStarsProvider = allStarsProvider; + } + + public StarContainer(@NonNull Context context) { + super(context); + setWillNotDraw(false); + defaultStar = new RLottieImageView(context); + selectedStar = new RLottieImageView(context); + + defaultStar.setAnimation(R.raw.star_stroke, starSize, starSize); + selectedStar.setAnimation(R.raw.star_fill, starSize, starSize); + selectedStar.setAlpha(0f); + addView(defaultStar, LayoutHelper.createFrame(starSize, starSize)); + addView(selectedStar, LayoutHelper.createFrame(starSize, starSize)); + rippleDrawable = Theme.createSimpleSelectorCircleDrawable(AndroidUtilities.dp(starSize), 0, ColorUtils.setAlphaComponent(Color.WHITE, (int) (255 * 0.3f))); + rippleDrawable.setCallback(this); + setClickable(true); + } + + @Override + protected void onDraw(Canvas canvas) { + super.onDraw(canvas); + rippleDrawable.setBounds(0, 0, getWidth(), getHeight()); + rippleDrawable.draw(canvas); + } + + @Override + protected void drawableStateChanged() { + super.drawableStateChanged(); + if (rippleDrawable != null) rippleDrawable.setState(getDrawableState()); + } + + @Override + public boolean verifyDrawable(@NonNull Drawable drawable) { + return rippleDrawable == drawable || super.verifyDrawable(drawable); + } + + @Override + public void jumpDrawablesToCurrentState() { + super.jumpDrawablesToCurrentState(); + if (rippleDrawable != null) rippleDrawable.jumpToCurrentState(); + } + + @Override + public boolean dispatchTouchEvent(MotionEvent event) { + switch (event.getAction()) { + case MotionEvent.ACTION_DOWN: + if (allStarsProvider != null) { + RateCallLayout.StarContainer[] starsViews = allStarsProvider.getAllStartsViews(); + for (int i = 0; i <= pos; i++) { + RLottieImageView defaultStar = starsViews[i].defaultStar; + RLottieImageView selectedStar = starsViews[i].selectedStar; + defaultStar.animate().alpha(0f).scaleX(0.8f).scaleY(0.8f).setDuration(250).start(); + selectedStar.animate().alpha(1f).scaleX(0.8f).scaleY(0.8f).setDuration(250).start(); + } + + for (int i = pos + 1; i < starsViews.length; i++) { + RLottieImageView defaultStar = starsViews[i].defaultStar; + RLottieImageView selectedStar = starsViews[i].selectedStar; + defaultStar.animate().alpha(1f).scaleX(1.0f).scaleY(1.0f).setDuration(250).start(); + selectedStar.animate().alpha(0f).scaleX(1.0f).scaleY(1.0f).setDuration(250).start(); + } + } + break; + case MotionEvent.ACTION_UP: + if (allStarsProvider != null) { + RateCallLayout.StarContainer[] starsViews = allStarsProvider.getAllStartsViews(); + for (int i = 0; i <= pos; i++) { + RLottieImageView defaultStar = starsViews[i].defaultStar; + RLottieImageView selectedStar = starsViews[i].selectedStar; + defaultStar.animate().scaleX(1.0f).scaleY(1.0f).setDuration(250).start(); + selectedStar.animate().scaleX(1.0f).scaleY(1.0f).setDuration(250).start(); + } + } + if (onSelectedStar != null) { + int[] location = new int[2]; + getLocationOnScreen(location); + int viewX = location[0]; + int viewY = location[1]; + onSelectedStar.onSelected(viewX + (getWidth() / 2f), viewY + (getHeight() / 2f), this.pos + 1); + } + break; + case MotionEvent.ACTION_CANCEL: + if (allStarsProvider != null) { + RateCallLayout.StarContainer[] starsViews = allStarsProvider.getAllStartsViews(); + for (StarContainer starsView : starsViews) { + RLottieImageView defaultStar = starsView.defaultStar; + RLottieImageView selectedStar = starsView.selectedStar; + defaultStar.animate().alpha(1f).scaleX(1.0f).scaleY(1.0f).setDuration(250).start(); + selectedStar.animate().alpha(0f).scaleX(1.0f).scaleY(1.0f).setDuration(250).start(); + } + } + break; + } + return super.dispatchTouchEvent(event); + } + } + + public static class RateCallContainer extends FrameLayout { + + private final TextView titleTextView; + private final TextView messageTextView; + private final VoIPBackgroundProvider backgroundProvider; + private final RectF bgRect = new RectF(); + + public RateCallContainer(@NonNull Context context, VoIPBackgroundProvider backgroundProvider) { + super(context); + this.backgroundProvider = backgroundProvider; + backgroundProvider.attach(this); + setWillNotDraw(false); + titleTextView = new TextView(context); + titleTextView.setTextColor(Color.WHITE); + titleTextView.setText(LocaleController.getString("VoipRateCallTitle", R.string.VoipRateCallTitle)); + titleTextView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 15); + titleTextView.setGravity(Gravity.CENTER_HORIZONTAL); + titleTextView.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); + messageTextView = new TextView(context); + messageTextView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 15); + messageTextView.setTextColor(Color.WHITE); + messageTextView.setGravity(Gravity.CENTER_HORIZONTAL); + messageTextView.setText(LocaleController.getString("VoipRateCallDescription", R.string.VoipRateCallDescription)); + + addView(titleTextView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.LEFT, 0, 24, 0, 0)); + addView(messageTextView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.LEFT, 0, 50, 0, 0)); + } + + @Override + protected void dispatchDraw(Canvas canvas) { + bgRect.set(0, 0, getWidth(), getHeight()); + backgroundProvider.setDarkTranslation(getX() + ((View) getParent()).getX(), getY() + ((View) getParent()).getY()); + canvas.drawRoundRect(bgRect, dp(28), dp(28), backgroundProvider.getDarkPaint()); + super.dispatchDraw(canvas); + } + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPBackgroundProvider.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPBackgroundProvider.java new file mode 100644 index 00000000000..5cf24953378 --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPBackgroundProvider.java @@ -0,0 +1,205 @@ +package org.telegram.ui.Components.voip; + +import android.animation.Animator; +import android.animation.AnimatorListenerAdapter; +import android.animation.ValueAnimator; +import android.graphics.Canvas; +import android.graphics.Color; +import android.graphics.Paint; +import android.view.View; +import android.view.animation.LinearInterpolator; + +import androidx.annotation.NonNull; + +import org.telegram.ui.Components.BitmapShaderTools; + +import java.util.ArrayList; +import java.util.List; + +public class VoIPBackgroundProvider { + + public static final float DARK_LIGHT_PERCENT = 0.14f; + public static final int DARK_LIGHT_DEFAULT_ALPHA = (int) (255 * DARK_LIGHT_PERCENT); + public static final int REVEAL_SCALE_FACTOR = 4; + + private final BitmapShaderTools lightShaderTools = new BitmapShaderTools(80, 80); + private final BitmapShaderTools darkShaderTools = new BitmapShaderTools(80, 80); + private BitmapShaderTools revealShaderTools; + private BitmapShaderTools revealDarkShaderTools; + private boolean isReveal; + private int totalWidth = 0; + private int totalHeight = 0; + private int degree; + private boolean hasVideo; + private final Paint whiteVideoPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + private final Paint darkVideoPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + private final Paint darkPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + private final List views = new ArrayList<>(); + public final float scale = 1.12f; + + public VoIPBackgroundProvider() { + darkShaderTools.setBounds(0, 0, 80, 80); + lightShaderTools.setBounds(0, 0, 80, 80); + whiteVideoPaint.setColor(Color.WHITE); + whiteVideoPaint.setAlpha(DARK_LIGHT_DEFAULT_ALPHA); + darkVideoPaint.setColor(Color.BLACK); + darkVideoPaint.setAlpha((int) (255 * 0.4f)); + darkPaint.setColor(Color.BLACK); + darkPaint.setAlpha(DARK_LIGHT_DEFAULT_ALPHA); + darkShaderTools.paint.setAlpha(180); + } + + public void invalidateViews() { + for (View view : views) { + view.invalidate(); + } + } + + public void attach(View view) { + views.add(view); + } + + public void detach(View view) { + views.remove(view); + } + + public void setHasVideo(boolean hasVideo) { + if (this.hasVideo && !hasVideo) { + ValueAnimator valueAnimator = ValueAnimator.ofFloat(1f, 0f); + valueAnimator.addUpdateListener(animation -> { + float val = (float) animation.getAnimatedValue(); + darkPaint.setAlpha((int) (DARK_LIGHT_DEFAULT_ALPHA * val)); + darkVideoPaint.setAlpha((int) ((int) (255 * 0.4f) * val)); + whiteVideoPaint.setAlpha((int) (DARK_LIGHT_DEFAULT_ALPHA * val)); + invalidateViews(); + }); + valueAnimator.setInterpolator(new LinearInterpolator()); + valueAnimator.setDuration(80); + valueAnimator.addListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + VoIPBackgroundProvider.this.hasVideo = false; + darkPaint.setAlpha(DARK_LIGHT_DEFAULT_ALPHA); + darkVideoPaint.setAlpha((int) (255 * 0.4f)); + whiteVideoPaint.setAlpha(DARK_LIGHT_DEFAULT_ALPHA); + invalidateViews(); + } + }); + valueAnimator.start(); + ValueAnimator valueAnimator2 = ValueAnimator.ofFloat(0f, 1f); + valueAnimator2.addUpdateListener(animation -> { + float val = (float) animation.getAnimatedValue(); + darkShaderTools.paint.setAlpha((int) (180 * val)); + lightShaderTools.paint.setAlpha((int) (255 * val)); + invalidateViews(); + }); + valueAnimator2.setInterpolator(new LinearInterpolator()); + valueAnimator2.setStartDelay(80); + valueAnimator2.setDuration(80); + valueAnimator2.start(); + } else { + this.hasVideo = hasVideo; + } + invalidateViews(); + } + + public Canvas getLightCanvas() { + return lightShaderTools.getCanvas(); + } + + public Canvas getRevealCanvas() { + return revealShaderTools.getCanvas(); + } + + public Canvas getRevealDrakCanvas() { + return revealDarkShaderTools.getCanvas(); + } + + public Canvas getDarkCanvas() { + return darkShaderTools.getCanvas(); + } + + public int getTotalWidth() { + return totalWidth; + } + + public void setTotalSize(int totalWidth, int totalHeight) { + this.totalWidth = totalWidth; + this.totalHeight = totalHeight; + this.revealShaderTools = new BitmapShaderTools(totalWidth / REVEAL_SCALE_FACTOR, totalHeight / REVEAL_SCALE_FACTOR); + this.revealDarkShaderTools = new BitmapShaderTools(totalWidth / REVEAL_SCALE_FACTOR, totalHeight / REVEAL_SCALE_FACTOR); + this.revealDarkShaderTools.paint.setAlpha(180); + } + + public int getTotalHeight() { + return totalHeight; + } + + public void setTotalHeight(int totalHeight) { + this.totalHeight = totalHeight; + } + + public int getDegree() { + return degree; + } + + public void setDegree(int degree) { + this.degree = degree; + invalidateViews(); + } + + public void setLightTranslation(float x, float y) { + float finalSize = (float) totalHeight * scale; + float s = finalSize / (float) lightShaderTools.getBitmap().getHeight(); + float dx = (totalHeight * scale - totalWidth) / 2f; + float dy = (totalHeight * scale - totalHeight) / 2f; + lightShaderTools.setMatrix(-x - dx, -y - dy, s, degree); + revealShaderTools.setBounds(-x, -y, totalWidth - x, totalHeight - y); + } + + public void setDarkTranslation(float x, float y) { + float finalSize = (float) totalHeight * scale; + float s = finalSize / (float) darkShaderTools.getBitmap().getHeight(); + float dx = (finalSize - totalWidth) / 2f; + float dy = (finalSize - totalHeight) / 2f; + darkShaderTools.setMatrix(-x - dx, -y - dy, s, degree); + revealDarkShaderTools.setBounds(-x, -y, totalWidth - x, totalHeight - y); + } + + public boolean isReveal() { + return isReveal; + } + + public void setReveal(boolean clipping) { + isReveal = clipping; + } + + public Paint getRevealPaint() { + return revealShaderTools.paint; + } + + public Paint getRevealDarkPaint() { + return revealDarkShaderTools.paint; + } + + public Paint getLightPaint() { + if (hasVideo) { + return whiteVideoPaint; + } + return lightShaderTools.paint; + } + + public Paint getDarkPaint() { + if (hasVideo) { + return darkVideoPaint; + } + return darkShaderTools.paint; + } + + public Paint getDarkPaint(boolean ignoreShader) { + if (ignoreShader) { + return darkPaint; + } + return getDarkPaint(); + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPEllipsizeSpan.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPEllipsizeSpan.java new file mode 100644 index 00000000000..a3ed2e70286 --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPEllipsizeSpan.java @@ -0,0 +1,54 @@ +package org.telegram.ui.Components.voip; + +import static org.telegram.messenger.AndroidUtilities.dp; +import static org.telegram.messenger.AndroidUtilities.dpf2; + +import android.graphics.Canvas; +import android.graphics.Paint; +import android.os.SystemClock; +import android.text.style.ReplacementSpan; +import android.view.View; + +import org.telegram.ui.Components.CubicBezierInterpolator; + +import androidx.annotation.NonNull; +import androidx.annotation.Nullable; + +public class VoIPEllipsizeSpan extends ReplacementSpan { + + private final CubicBezierInterpolator interpolator = new CubicBezierInterpolator(0.33, 0.00, 0.67, 1.00); + + private final View[] parents; + + public VoIPEllipsizeSpan(View... parents) { + this.parents = parents; + } + + @Override + public int getSize(@NonNull Paint paint, CharSequence text, int start, int end, @Nullable Paint.FontMetricsInt fm) { + return dp(20); + } + + @Override + public void draw(@NonNull Canvas canvas, CharSequence text, int start, int end, float x, int top, int y, int bottom, @NonNull Paint paint) { + canvas.save(); + canvas.translate(x + dp(4), y / 2f); + long time = SystemClock.uptimeMillis() % 250 + 500; + for (int i = 0; i < 3; i++) { + long pointTime = (time + i * 250L) % 750; + float moveFraction = Math.min(1, pointTime / 667f); + float scale; + if (moveFraction <= 0.425f) { + scale = interpolator.getInterpolation(moveFraction / 0.425f); + } else { + scale = 1f - interpolator.getInterpolation((moveFraction - 0.425f) / 0.575f); + } + moveFraction = interpolator.getInterpolation(moveFraction); + canvas.drawCircle(dpf2(1.667f + moveFraction * 16f), dp(3), dpf2(2 * scale), paint); + } + canvas.restore(); + for (View parent : parents) { + parent.invalidate(); + } + } +} \ No newline at end of file diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPFloatingLayout.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPFloatingLayout.java index 8825892d1b1..f264bd64081 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPFloatingLayout.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPFloatingLayout.java @@ -78,6 +78,7 @@ public class VoIPFloatingLayout extends FrameLayout { public float updatePositionFromX; public float updatePositionFromY; public boolean switchingToPip; + public boolean isAppearing; Drawable outerShadow; ValueAnimator switchToFloatingModeAnimator; @@ -266,12 +267,16 @@ public boolean onTouchEvent(MotionEvent event) { protected void dispatchDraw(Canvas canvas) { boolean animated = false; if (updatePositionFromX >= 0) { - animate().setListener(null).cancel(); + if(!isAppearing) { + animate().setListener(null).cancel(); + } setTranslationX(updatePositionFromX); setTranslationY(updatePositionFromY); - setScaleX(1f); - setScaleY(1f); - setAlpha(1f); + if(!isAppearing) { + setScaleX(1f); + setScaleY(1f); + setAlpha(1f); + } updatePositionFromX = -1f; updatePositionFromY = -1f; } @@ -359,7 +364,7 @@ private void setRelativePositionInternal(float xRelative, float yRelative, int w .translationX(xPoint) .translationY(yPoint) .alpha(1f) - .setStartDelay(0) + .setStartDelay(uiVisible ? 0 : 150) .setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); } else { if (!alwaysFloating) { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPHelper.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPHelper.java index 9f5e1ee90a9..bdb2fad73b2 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPHelper.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPHelper.java @@ -394,6 +394,23 @@ public static boolean canRateCall(TLRPC.TL_messageActionPhoneCall call) { return false; } + public static void sendCallRating(final long callID, final long accessHash, final int account, int rating) { + final int currentAccount = UserConfig.selectedAccount; + final TLRPC.TL_phone_setCallRating req = new TLRPC.TL_phone_setCallRating(); + req.rating = rating; + req.comment = ""; + req.peer = new TLRPC.TL_inputPhoneCall(); + req.peer.access_hash = accessHash; + req.peer.id = callID; + req.user_initiative = false; + ConnectionsManager.getInstance(account).sendRequest(req, (response, error) -> { + if (response instanceof TLRPC.TL_updates) { + TLRPC.TL_updates updates = (TLRPC.TL_updates) response; + MessagesController.getInstance(currentAccount).processUpdates(updates, false); + } + }); + } + public static void showRateAlert(Context context, TLRPC.TL_messageActionPhoneCall call) { SharedPreferences prefs = MessagesController.getNotificationsSettings(UserConfig.selectedAccount); // always called from chat UI Set hashes = prefs.getStringSet("calls_access_hashes", (Set) Collections.EMPTY_SET); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPNotificationsLayout.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPNotificationsLayout.java index d34b13d073a..caee016900f 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPNotificationsLayout.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPNotificationsLayout.java @@ -1,11 +1,19 @@ package org.telegram.ui.Components.voip; +import static org.telegram.messenger.AndroidUtilities.dp; + import android.animation.Animator; import android.animation.AnimatorSet; import android.animation.ObjectAnimator; +import android.annotation.SuppressLint; import android.content.Context; +import android.graphics.Canvas; import android.graphics.Color; -import android.os.Build; +import android.graphics.RectF; +import android.text.Layout; +import android.text.StaticLayout; +import android.text.TextPaint; +import android.text.TextUtils; import android.transition.ChangeBounds; import android.transition.Fade; import android.transition.TransitionManager; @@ -23,16 +31,16 @@ import android.widget.TextView; import androidx.annotation.NonNull; -import androidx.core.graphics.ColorUtils; import org.telegram.messenger.AndroidUtilities; -import org.telegram.ui.ActionBar.Theme; import org.telegram.ui.Components.CubicBezierInterpolator; import org.telegram.ui.Components.LayoutHelper; +import org.telegram.ui.Components.StaticLayoutEx; import java.util.ArrayList; import java.util.HashMap; +@SuppressLint("ViewConstructor") public class VoIPNotificationsLayout extends LinearLayout { HashMap viewsByTag = new HashMap<>(); @@ -42,32 +50,49 @@ public class VoIPNotificationsLayout extends LinearLayout { boolean lockAnimation; boolean wasChanged; Runnable onViewsUpdated; + VoIPBackgroundProvider backgroundProvider; + TextPaint textPaint = new TextPaint(); - public VoIPNotificationsLayout(Context context) { + public VoIPNotificationsLayout(Context context, VoIPBackgroundProvider backgroundProvider) { super(context); setOrientation(VERTICAL); + this.backgroundProvider = backgroundProvider; + transitionSet = new TransitionSet(); + transitionSet.addTransition(new Fade(Fade.OUT).setDuration(150)) + .addTransition(new ChangeBounds().setDuration(200)) + .addTransition(new Visibility() { + @Override + public Animator onAppear(ViewGroup sceneRoot, View view, TransitionValues startValues, TransitionValues endValues) { + AnimatorSet set = new AnimatorSet(); + view.setAlpha(0); + view.setScaleY(0.6f); + view.setScaleX(0.6f); + set.playTogether( + ObjectAnimator.ofFloat(view, View.ALPHA, 0, 1f), + ObjectAnimator.ofFloat(view, View.SCALE_X, 0.6f, 1f), + ObjectAnimator.ofFloat(view, View.SCALE_Y, 0.6f, 1f) + ); + set.setInterpolator(CubicBezierInterpolator.EASE_OUT_BACK); + return set; + } - if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { - transitionSet = new TransitionSet(); - transitionSet.addTransition(new Fade(Fade.OUT).setDuration(150)) - .addTransition(new ChangeBounds().setDuration(200)) - .addTransition(new Visibility() { - @Override - public Animator onAppear(ViewGroup sceneRoot, View view, TransitionValues startValues, TransitionValues endValues) { - AnimatorSet set = new AnimatorSet(); - view.setAlpha(0); - set.playTogether( - ObjectAnimator.ofFloat(view, View.ALPHA, 0, 1f), - ObjectAnimator.ofFloat(view, View.TRANSLATION_Y, view.getMeasuredHeight(), 0) - ); - - set.setInterpolator(CubicBezierInterpolator.DEFAULT); - - return set; + @Override + public Animator onDisappear(ViewGroup sceneRoot, View view, TransitionValues startValues, TransitionValues endValues) { + AnimatorSet set = new AnimatorSet(); + if (view instanceof NotificationView) { + ((NotificationView) view).ignoreShader = true; } - }.setDuration(200)); - transitionSet.setOrdering(TransitionSet.ORDERING_TOGETHER); - } + set.playTogether( + ObjectAnimator.ofFloat(view, View.ALPHA, 0.7f, 0f), + ObjectAnimator.ofFloat(view, View.SCALE_X, 1f, 0.6f), + ObjectAnimator.ofFloat(view, View.SCALE_Y, 1f, 0.6f) + ); + set.setInterpolator(CubicBezierInterpolator.DEFAULT); + return set; + } + }.setDuration(200)); + transitionSet.setOrdering(TransitionSet.ORDERING_TOGETHER); + textPaint.setTextSize(dp(14)); } public void addNotification(int iconRes, String text, String tag, boolean animated) { @@ -75,15 +100,13 @@ public void addNotification(int iconRes, String text, String tag, boolean animat return; } - NotificationView view = new NotificationView(getContext()); + NotificationView view = new NotificationView(getContext(), backgroundProvider, iconRes); view.tag = tag; + view.setText(text); view.iconView.setImageResource(iconRes); - view.textView.setText(text); + viewsByTag.put(tag, view); - if (animated) { - view.startAnimation(); - } if (lockAnimation) { viewToAdd.add(view); } else { @@ -92,8 +115,16 @@ public void addNotification(int iconRes, String text, String tag, boolean animat } } + public CharSequence ellipsize(CharSequence text) { + if (text == null) { + return ""; + } + return TextUtils.ellipsize(text, textPaint, dp(300), TextUtils.TruncateAt.END); + } + public void removeNotification(String tag) { NotificationView view = viewsByTag.remove(tag); + backgroundProvider.detach(view); if (view != null) { if (lockAnimation) { if (viewToAdd.remove(view)) { @@ -119,11 +150,9 @@ private void runDelayed() { if (viewToAdd.isEmpty() && viewToRemove.isEmpty()) { return; } - if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { - ViewParent parent = getParent(); - if (parent != null) { - TransitionManager.beginDelayedTransition(this, transitionSet); - } + ViewParent parent = getParent(); + if (parent != null) { + TransitionManager.beginDelayedTransition(this, transitionSet); } for (int i = 0; i < viewToAdd.size(); i++) { @@ -160,11 +189,9 @@ private void runDelayed() { public void beforeLayoutChanges() { wasChanged = false; if (!lockAnimation) { - if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { - ViewParent parent = getParent(); - if (parent != null) { - TransitionManager.beginDelayedTransition(this, transitionSet); - } + ViewParent parent = getParent(); + if (parent != null) { + TransitionManager.beginDelayedTransition(this, transitionSet); } } } @@ -186,37 +213,58 @@ private static class NotificationView extends FrameLayout { public String tag; ImageView iconView; TextView textView; + boolean ignoreShader; + private final VoIPBackgroundProvider backgroundProvider; + private final RectF bgRect = new RectF(); - public NotificationView(@NonNull Context context) { + public NotificationView(@NonNull Context context, VoIPBackgroundProvider backgroundProvider, int iconRes) { super(context); setFocusable(true); setFocusableInTouchMode(true); - + this.backgroundProvider = backgroundProvider; + backgroundProvider.attach(this); iconView = new ImageView(context); - setBackground(Theme.createRoundRectDrawable(AndroidUtilities.dp(16), ColorUtils.setAlphaComponent(Color.BLACK, (int) (255 * 0.4f)))); - addView(iconView, LayoutHelper.createFrame(24, 24, 0, 10, 4, 10, 4)); + addView(iconView, LayoutHelper.createFrame(24, 24, Gravity.CENTER_VERTICAL, 10, 2, 10, 2)); textView = new TextView(context); textView.setTextColor(Color.WHITE); textView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 14); - addView(textView, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_VERTICAL, 44, 4, 16, 4)); + addView(textView, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_VERTICAL, iconRes == 0 ? 14 : 42, 2, 14, 2)); + } + + public void setText(CharSequence text) { + int maxWidth = AndroidUtilities.displaySize.x - dp(120); + StaticLayout staticLayout = StaticLayoutEx.createStaticLayout(text, textView.getPaint(), + maxWidth, Layout.Alignment.ALIGN_NORMAL, 1.0f, 0, + false, TextUtils.TruncateAt.END, maxWidth, 10); + int maxRowLength = 0; + for (int a = 0; a < staticLayout.getLineCount(); ++a) { + maxRowLength = (int) Math.max(maxRowLength, Math.ceil(staticLayout.getLineWidth(a))); + } + textView.setMaxWidth(maxRowLength); + textView.setText(text); + } + + @Override + protected void dispatchDraw(Canvas canvas) { + bgRect.set(0, 0, getWidth(), getHeight()); + backgroundProvider.setDarkTranslation(getX() + ((View) getParent()).getX(), getY() + ((View) getParent()).getY()); + canvas.drawRoundRect(bgRect, dp(16), dp(16), backgroundProvider.getDarkPaint(ignoreShader)); + super.dispatchDraw(canvas); } public void startAnimation() { - textView.setVisibility(View.GONE); + textView.setVisibility(View.INVISIBLE); postDelayed(() -> { - if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { - TransitionSet transitionSet = new TransitionSet(); - transitionSet. - addTransition(new Fade(Fade.IN).setDuration(150)) - .addTransition(new ChangeBounds().setDuration(200)); - transitionSet.setOrdering(TransitionSet.ORDERING_TOGETHER); - ViewParent parent = getParent(); - if (parent != null) { - TransitionManager.beginDelayedTransition((ViewGroup) parent, transitionSet); - } + TransitionSet transitionSet = new TransitionSet(); + transitionSet. + addTransition(new Fade(Fade.IN).setDuration(150)) + .addTransition(new ChangeBounds().setDuration(200)); + transitionSet.setOrdering(TransitionSet.ORDERING_TOGETHER); + ViewParent parent = getParent(); + if (parent != null) { + TransitionManager.beginDelayedTransition((ViewGroup) parent, transitionSet); } - textView.setVisibility(View.VISIBLE); }, 400); } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPStatusTextView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPStatusTextView.java index 2f7dde5ab57..e7a2d062ec2 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPStatusTextView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPStatusTextView.java @@ -1,16 +1,19 @@ package org.telegram.ui.Components.voip; +import static org.telegram.messenger.AndroidUtilities.dp; + import android.animation.Animator; import android.animation.AnimatorListenerAdapter; -import android.animation.AnimatorSet; import android.animation.ValueAnimator; +import android.annotation.SuppressLint; import android.content.Context; +import android.graphics.Canvas; import android.graphics.Color; +import android.graphics.RectF; import android.text.SpannableString; import android.text.SpannableStringBuilder; -import android.text.TextPaint; +import android.text.Spanned; import android.text.TextUtils; -import android.text.style.CharacterStyle; import android.util.TypedValue; import android.view.Gravity; import android.view.View; @@ -23,29 +26,29 @@ import org.telegram.messenger.LocaleController; import org.telegram.messenger.R; import org.telegram.ui.Components.CubicBezierInterpolator; -import org.telegram.ui.Components.EllipsizeSpanAnimator; import org.telegram.ui.Components.LayoutHelper; -import java.util.ArrayList; - +@SuppressLint("ViewConstructor") public class VoIPStatusTextView extends FrameLayout { TextView[] textView = new TextView[2]; TextView reconnectTextView; + TextView badConnectionTextView; + FrameLayout badConnectionLayer; VoIPTimerView timerView; CharSequence nextTextToSet; boolean animationInProgress; - private boolean attachedToWindow; - ValueAnimator animator; boolean timerShowing; - EllipsizeSpanAnimator ellipsizeAnimator; + VoIPBackgroundProvider backgroundProvider; - public VoIPStatusTextView(@NonNull Context context) { + public VoIPStatusTextView(@NonNull Context context, VoIPBackgroundProvider backgroundProvider) { super(context); + this.backgroundProvider = backgroundProvider; + for (int i = 0; i < 2; i++) { textView[i] = new TextView(context); textView[i].setTextSize(TypedValue.COMPLEX_UNIT_DIP, 15); @@ -55,6 +58,34 @@ public VoIPStatusTextView(@NonNull Context context) { addView(textView[i]); } + badConnectionLayer = new FrameLayout(context); + badConnectionTextView = new TextView(context) { + + private final RectF bgRect = new RectF(); + + { + backgroundProvider.attach(this); + } + + @Override + protected void onDraw(Canvas canvas) { + bgRect.set(0, 0, getWidth(), getHeight()); + float x = getX() + ((View) getParent()).getX() + VoIPStatusTextView.this.getX() + ((View) VoIPStatusTextView.this.getParent()).getX(); + float y = getY() + ((View) getParent()).getY() + VoIPStatusTextView.this.getY() + ((View) VoIPStatusTextView.this.getParent()).getY(); + backgroundProvider.setDarkTranslation(x, y); + canvas.drawRoundRect(bgRect, dp(16), dp(16), backgroundProvider.getDarkPaint()); + super.onDraw(canvas); + } + }; + badConnectionTextView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 15); + badConnectionTextView.setTextColor(Color.WHITE); + badConnectionTextView.setGravity(Gravity.CENTER_HORIZONTAL); + badConnectionTextView.setPadding(AndroidUtilities.dp(12), AndroidUtilities.dp(2), AndroidUtilities.dp(12), AndroidUtilities.dp(2)); + badConnectionTextView.setText(LocaleController.getString("VoipWeakNetwork", R.string.VoipWeakNetwork)); + badConnectionLayer.addView(badConnectionTextView, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_HORIZONTAL, 0, 0, 0, 0)); + badConnectionLayer.setVisibility(View.GONE); + addView(badConnectionLayer, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, 0, 0, 44, 0, 0)); + reconnectTextView = new TextView(context); reconnectTextView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 15); reconnectTextView.setShadowLayer(AndroidUtilities.dp(3), 0, AndroidUtilities.dp(.666666667f), 0x4C000000); @@ -62,34 +93,25 @@ public VoIPStatusTextView(@NonNull Context context) { reconnectTextView.setGravity(Gravity.CENTER_HORIZONTAL); addView(reconnectTextView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, 0, 0, 22, 0, 0)); - ellipsizeAnimator = new EllipsizeSpanAnimator(this); SpannableStringBuilder ssb = new SpannableStringBuilder(LocaleController.getString("VoipReconnecting", R.string.VoipReconnecting)); - SpannableString ell = new SpannableString("..."); - ellipsizeAnimator.wrap(ell, 0); + SpannableString ell = new SpannableString("."); + ell.setSpan(new VoIPEllipsizeSpan(reconnectTextView), 0, 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); ssb.append(ell); reconnectTextView.setText(ssb); reconnectTextView.setVisibility(View.GONE); timerView = new VoIPTimerView(context); addView(timerView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT)); - } public void setText(String text, boolean ellipsis, boolean animated) { CharSequence nextString = text; if (ellipsis) { SpannableStringBuilder ssb = new SpannableStringBuilder(text); - ellipsizeAnimator.reset(); - SpannableString ell = new SpannableString("..."); - ellipsizeAnimator.wrap(ell, 0); + SpannableString ell = new SpannableString("."); + ell.setSpan(new VoIPEllipsizeSpan(textView), 0, 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); ssb.append(ell); nextString = ssb; - - ellipsizeAnimator.addView(textView[0]); - ellipsizeAnimator.addView(textView[1]); - } else { - ellipsizeAnimator.removeView(textView[0]); - ellipsizeAnimator.removeView(textView[1]); } if (TextUtils.isEmpty(textView[0].getText())) { @@ -153,9 +175,6 @@ public void showTimer(boolean animated) { timerShowing = true; replaceViews(textView[0], timerView, null); } - - ellipsizeAnimator.removeView(textView[0]); - ellipsizeAnimator.removeView(textView[1]); } @@ -169,17 +188,10 @@ private void replaceViews(View out, View in, Runnable onEnd) { animator = ValueAnimator.ofFloat(0, 1f); animator.addUpdateListener(valueAnimator -> { float v = (float) valueAnimator.getAnimatedValue(); - float inScale = 0.4f + 0.6f * v; - float outScale = 0.4f + 0.6f * (1f - v); - in.setTranslationY(AndroidUtilities.dp(10) * (1f - v)); + in.setTranslationY(AndroidUtilities.dp(8) * (1f - v)); in.setAlpha(v); - in.setScaleX(inScale); - in.setScaleY(inScale); - - out.setTranslationY(-AndroidUtilities.dp(10) * v); + out.setTranslationY(-AndroidUtilities.dp(6) * v); out.setAlpha(1f - v); - out.setScaleX(outScale); - out.setScaleY(outScale); }); animator.addListener(new AnimatorListenerAdapter() { @Override @@ -244,26 +256,40 @@ public void onAnimationEnd(Animator animation) { }).setDuration(150).start(); } } + } - if (showReconnecting) { - ellipsizeAnimator.addView(reconnectTextView); + public void showBadConnection(boolean showBadConnection, boolean animated) { + if (!animated) { + badConnectionLayer.animate().setListener(null).cancel(); + badConnectionLayer.setVisibility(showBadConnection ? View.VISIBLE : View.GONE); } else { - ellipsizeAnimator.removeView(reconnectTextView); + if (showBadConnection) { + if (badConnectionLayer.getVisibility() == View.VISIBLE) { + return; + } + badConnectionLayer.setVisibility(View.VISIBLE); + badConnectionLayer.setAlpha(0f); + badConnectionLayer.setScaleY(0.6f); + badConnectionLayer.setScaleX(0.6f); + badConnectionLayer.animate().setListener(null).cancel(); + badConnectionLayer.animate().alpha(1f).scaleX(1f).scaleY(1f) + .setInterpolator(CubicBezierInterpolator.EASE_OUT_BACK) + .setDuration(300).start(); + } else { + if (badConnectionLayer.getVisibility() == View.GONE) { + return; + } + badConnectionLayer.animate().alpha(0f).scaleX(0.6f).scaleY(0.6f).setInterpolator(CubicBezierInterpolator.DEFAULT).setListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + badConnectionLayer.setVisibility(View.GONE); + } + }).setDuration(300).start(); + } } } - @Override - protected void onAttachedToWindow() { - super.onAttachedToWindow(); - attachedToWindow = true; - ellipsizeAnimator.onAttachedToWindow(); + public void setDrawCallIcon() { + timerView.setDrawCallIcon(); } - - @Override - protected void onDetachedFromWindow() { - super.onDetachedFromWindow(); - attachedToWindow = false; - ellipsizeAnimator.onDetachedFromWindow(); - } - } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPTimerView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPTimerView.java index 7b63a6dcfe6..7b55ea5763c 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPTimerView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPTimerView.java @@ -5,14 +5,17 @@ import android.graphics.Color; import android.graphics.Paint; import android.graphics.RectF; +import android.graphics.drawable.Drawable; import android.text.Layout; import android.text.StaticLayout; import android.text.TextPaint; import android.view.View; +import androidx.core.content.ContextCompat; import androidx.core.graphics.ColorUtils; import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.R; import org.telegram.messenger.voip.VoIPService; public class VoIPTimerView extends View { @@ -24,6 +27,8 @@ public class VoIPTimerView extends View { String currentTimeStr; TextPaint textPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG); private int signalBarCount = 4; + private boolean isDrawCallIcon = false; + private final Drawable callsDeclineDrawable; Runnable updater = () -> { if (getVisibility() == View.VISIBLE) { @@ -38,6 +43,8 @@ public VoIPTimerView(Context context) { textPaint.setShadowLayer(AndroidUtilities.dp(3), 0, AndroidUtilities.dp(.666666667f), 0x4C000000); activePaint.setColor(ColorUtils.setAlphaComponent(Color.WHITE, (int) (255 * 0.9f))); inactivePaint.setColor(ColorUtils.setAlphaComponent(Color.WHITE, (int) (255 * 0.4f))); + callsDeclineDrawable = ContextCompat.getDrawable(context, R.drawable.calls_decline); + callsDeclineDrawable.setBounds(0, 0, AndroidUtilities.dp(24), AndroidUtilities.dp(24)); } @Override @@ -92,11 +99,16 @@ protected void onDraw(Canvas canvas) { canvas.save(); canvas.translate((getMeasuredWidth() - totalWidth) / 2f, 0); canvas.save(); - canvas.translate(0, (getMeasuredHeight() - AndroidUtilities.dp(11)) / 2f); - for (int i = 0; i < 4; i++) { - Paint p = i + 1 > signalBarCount ? inactivePaint : activePaint; - rectF.set(AndroidUtilities.dpf2(4.16f) * i, AndroidUtilities.dpf2(2.75f) * (3 - i), AndroidUtilities.dpf2(4.16f) * i + AndroidUtilities.dpf2(2.75f), AndroidUtilities.dp(11)); - canvas.drawRoundRect(rectF, AndroidUtilities.dpf2(0.7f), AndroidUtilities.dpf2(0.7f), p); + if (isDrawCallIcon) { + canvas.translate(-AndroidUtilities.dp(7), -AndroidUtilities.dp(3)); + callsDeclineDrawable.draw(canvas); + } else { + canvas.translate(0, (getMeasuredHeight() - AndroidUtilities.dp(11)) / 2f); + for (int i = 0; i < 4; i++) { + Paint p = i + 1 > signalBarCount ? inactivePaint : activePaint; + rectF.set(AndroidUtilities.dpf2(4.16f) * i, AndroidUtilities.dpf2(2.75f) * (3 - i), AndroidUtilities.dpf2(4.16f) * i + AndroidUtilities.dpf2(2.75f), AndroidUtilities.dp(11)); + canvas.drawRoundRect(rectF, AndroidUtilities.dpf2(0.7f), AndroidUtilities.dpf2(0.7f), p); + } } canvas.restore(); @@ -111,4 +123,9 @@ public void setSignalBarCount(int count) { signalBarCount = count; invalidate(); } + + public void setDrawCallIcon() { + isDrawCallIcon = true; + invalidate(); + } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPToggleButton.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPToggleButton.java index 7cf8598bfb5..1228014d349 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPToggleButton.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPToggleButton.java @@ -76,6 +76,8 @@ public class VoIPToggleButton extends FrameLayout { private float radius; private ValueAnimator checkAnimator; + private ValueAnimator pressedScaleAnimator; + private float pressedScale = 1.0f; private RLottieImageView lottieImageView; @@ -96,7 +98,7 @@ public VoIPToggleButton(@NonNull Context context, float radius) { textView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 11); textView.setTextColor(Color.WHITE); textView.setImportantForAccessibility(IMPORTANT_FOR_ACCESSIBILITY_NO); - textLayoutContainer.addView(textView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, 0, 0, radius + 4, 0, 0)); + textLayoutContainer.addView(textView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, 0, 0, radius + 6, 0, 0)); this.textView[i] = textView; } textView[1].setVisibility(View.GONE); @@ -122,9 +124,25 @@ public void setDrawBackground(boolean value) { drawBackground = value; } + public void setPressedBtn(boolean pressed) { + if (pressedScaleAnimator != null) { + pressedScaleAnimator.cancel(); + } + pressedScaleAnimator = ValueAnimator.ofFloat(pressedScale, pressed ? 0.8f : 1f); + pressedScaleAnimator.addUpdateListener(animation -> { + pressedScale = (float) animation.getAnimatedValue(); + invalidate(); + }); + pressedScaleAnimator.setDuration(150); + pressedScaleAnimator.start(); + } + @SuppressLint("DrawAllocation") @Override protected void onDraw(Canvas canvas) { + canvas.save(); + canvas.scale(pressedScale, pressedScale, getMeasuredWidth() / 2f, getMeasuredHeight() / 2f); + if (animateBackground && replaceProgress != 0) { circlePaint.setColor(ColorUtils.blendARGB(backgroundColor, animateToBackgroundColor, replaceProgress)); } else { @@ -229,6 +247,7 @@ protected void onDraw(Canvas canvas) { } } } + canvas.restore(); } public void setBackgroundColor(int backgroundColor, int backgroundColorChecked) { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPWindowView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPWindowView.java index e5779637bb5..1aeccf27fe7 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPWindowView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIPWindowView.java @@ -84,24 +84,24 @@ public boolean onTouchEvent(MotionEvent event) { } else if (event.getAction() == MotionEvent.ACTION_MOVE) { float dx = event.getX() - startX; float dy = event.getY() - startY; - if (!startDragging && Math.abs(dx) > AndroidUtilities.getPixelsInCM(0.4f, true) && Math.abs(dx) / 3 > dy) { - startX = event.getX(); - dx = 0; + if (!startDragging && Math.abs(dy) > AndroidUtilities.getPixelsInCM(0.4f, true) && Math.abs(dy) / 3 > dx) { + startY = event.getY(); + dy = 0; startDragging = true; } if (startDragging) { - if (dx < 0) { - dx = 0; + if (dy < 0) { + dy = 0; } if (velocityTracker == null) { velocityTracker = VelocityTracker.obtain(); } velocityTracker.addMovement(event); - setTranslationX(dx); + setTranslationY(dy); } return startDragging; } else if (event.getAction() == MotionEvent.ACTION_UP || event.getAction() == MotionEvent.ACTION_CANCEL) { - float x = getTranslationX(); + float y = getTranslationY(); if (velocityTracker == null) { velocityTracker = VelocityTracker.obtain(); } @@ -110,12 +110,12 @@ public boolean onTouchEvent(MotionEvent event) { float velX = velocityTracker.getXVelocity(); float velY = velocityTracker.getYVelocity(); - final boolean backAnimation = x < getMeasuredWidth() / 3.0f && (velX < 3500 || velX < velY); + final boolean backAnimation = y < getMeasuredHeight() / 3.0f && (velX < 3500 || velX < velY); if (!backAnimation) { - float distToMove = getMeasuredWidth() - getTranslationX(); - finish(Math.max((int) (200.0f / getMeasuredWidth() * distToMove), 50)); + float distToMove = getMeasuredHeight() - getTranslationY(); + finish(Math.max((int) (200.0f / getMeasuredHeight() * distToMove), 50)); } else { - animate().translationX(0).start(); + animate().translationY(0).start(); } startDragging = false; } @@ -123,7 +123,7 @@ public boolean onTouchEvent(MotionEvent event) { } public void finish() { - finish(150); + finish(330); } public void finish(long animDuration) { @@ -141,7 +141,7 @@ public void finish(long animDuration) { } else { int account = UserConfig.selectedAccount; notificationsLocker.lock(); - animate().translationX(getMeasuredWidth()).setListener(new AnimatorListenerAdapter() { + animate().translationY(getMeasuredHeight()).alpha(0f).setListener(new AnimatorListenerAdapter() { @Override public void onAnimationEnd(Animator animation) { notificationsLocker.unlock(); @@ -166,8 +166,9 @@ public void onAnimationEnd(Animator animation) { public void startEnterTransition() { if (!lockOnScreen) { - setTranslationX(getMeasuredWidth()); - animate().translationX(0).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); + setTranslationY(getMeasuredHeight()); + setAlpha(0f); + animate().translationY(0).alpha(1f).setDuration(330).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIpBitmapTextView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIpBitmapTextView.java new file mode 100644 index 00000000000..e569d15fa57 --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIpBitmapTextView.java @@ -0,0 +1,70 @@ +package org.telegram.ui.Components.voip; + +import static org.telegram.messenger.AndroidUtilities.dp; + +import android.annotation.SuppressLint; +import android.content.Context; +import android.graphics.Bitmap; +import android.graphics.Canvas; +import android.graphics.Color; +import android.graphics.Paint; +import android.text.TextPaint; +import android.view.View; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.Utilities; + +/** + * Fixed ANR on Samsung after one and a half minutes. + * Anr occurs due to drawing long text. + */ +@SuppressLint("ViewConstructor") +public class VoIpBitmapTextView extends View { + + private final TextPaint textPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG); + private final Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG); + private final float textWidth; + private final String text; + private volatile Bitmap bitmap; + + public VoIpBitmapTextView(Context context, String text) { + super(context); + textPaint.setTextAlign(Paint.Align.CENTER); + textPaint.setTextSize(dp(13)); + textPaint.setColor(Color.WHITE); + textPaint.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); + textWidth = textPaint.measureText(text); + this.text = text; + } + + @Override + protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { + super.onMeasure( + MeasureSpec.makeMeasureSpec((int) textWidth + getPaddingLeft() + getPaddingRight(), MeasureSpec.EXACTLY), + MeasureSpec.makeMeasureSpec(MeasureSpec.getSize(heightMeasureSpec), MeasureSpec.EXACTLY) + ); + } + + @Override + protected void onLayout(boolean changed, int left, int top, int right, int bottom) { + super.onLayout(changed, left, top, right, bottom); + if (changed) { + Utilities.globalQueue.postRunnable(() -> { + bitmap = Bitmap.createBitmap(getMeasuredWidth(), getMeasuredHeight(), Bitmap.Config.ARGB_8888); + Canvas canvas = new Canvas(bitmap); + int xPos = (getMeasuredWidth() / 2); + int yPos = (int) ((getMeasuredHeight() / 2) - ((textPaint.descent() + textPaint.ascent()) / 2)); + canvas.drawText(text, xPos, yPos, textPaint); + postInvalidate(); + }); + } + } + + @Override + protected void onDraw(Canvas canvas) { + super.onDraw(canvas); + if (bitmap != null) { + canvas.drawBitmap(bitmap, 0, 0, paint); + } + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIpGradientLayout.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIpGradientLayout.java new file mode 100644 index 00000000000..0dd78bb0597 --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIpGradientLayout.java @@ -0,0 +1,402 @@ +package org.telegram.ui.Components.voip; + +import static org.telegram.ui.Components.voip.VoIPBackgroundProvider.REVEAL_SCALE_FACTOR; + +import android.animation.Animator; +import android.animation.AnimatorListenerAdapter; +import android.animation.AnimatorSet; +import android.animation.ValueAnimator; +import android.annotation.SuppressLint; +import android.content.Context; +import android.graphics.Canvas; +import android.graphics.Color; +import android.graphics.Path; +import android.graphics.PorterDuff; +import android.view.View; +import android.view.animation.LinearInterpolator; +import android.widget.FrameLayout; + +import androidx.annotation.NonNull; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.LiteMode; +import org.telegram.ui.Components.MotionBackgroundDrawable; + +@SuppressLint("ViewConstructor") +public class VoIpGradientLayout extends FrameLayout { + + public enum GradientState { + CALLING, + CONNECTED, + BAD_CONNECTION + } + + private final MotionBackgroundDrawable bgBlueViolet; + private final MotionBackgroundDrawable bgBlueGreen; + private final MotionBackgroundDrawable bgGreen; + private final MotionBackgroundDrawable bgOrangeRed; + + private final MotionBackgroundDrawable bgBlueVioletDark; + private final MotionBackgroundDrawable bgBlueGreenDark; + private final MotionBackgroundDrawable bgGreenDark; + private final MotionBackgroundDrawable bgOrangeRedDark; + + private final MotionBackgroundDrawable bgBlueVioletLight; + private final MotionBackgroundDrawable bgBlueGreenLight; + private final MotionBackgroundDrawable bgGreenLight; + private final MotionBackgroundDrawable bgOrangeRedLight; + private final MotionBackgroundDrawable bgGreenLightReveal; + private final MotionBackgroundDrawable bgGreenDarkReveal; + + private int alphaBlueViolet = 0; + private int alphaBlueGreen = 0; + private int alphaGreen = 0; + private int alphaOrangeRed = 0; + private float clipRadius = 0f; + private boolean showClip = false; + private final Path clipPath = new Path(); + private int clipCx = 0; + private int clipCy = 0; + private ValueAnimator callingAnimator; + private ValueAnimator badConnectionAnimator; + private AnimatorSet connectedAnimatorSet; + private final AnimatorSet defaultAnimatorSet; + private GradientState state; + private boolean isPaused = false; + public volatile boolean lockDrawing = false; + private final VoIPBackgroundProvider backgroundProvider; + private boolean allowAnimations; + + public VoIpGradientLayout(@NonNull Context context, VoIPBackgroundProvider backgroundProvider) { + super(context); + this.backgroundProvider = backgroundProvider; + allowAnimations = LiteMode.isEnabled(LiteMode.FLAG_CALLS_ANIMATIONS); + bgBlueViolet = new MotionBackgroundDrawable(0xFFB456D8, 0xFF8148EC, 0xFF20A4D7, 0xFF3F8BEA, 0, false, true); + bgBlueGreen = new MotionBackgroundDrawable(0xFF4576E9, 0xFF3B7AF1, 0xFF08B0A3, 0xFF17AAE4, 0, false, true); + bgGreen = new MotionBackgroundDrawable(0xFF07A9AC, 0xFF07BA63, 0xFFA9CC66, 0xFF5AB147, 0, false, true); + bgOrangeRed = new MotionBackgroundDrawable(0xFFE86958, 0xFFE7618F, 0xFFDB904C, 0xFFDE7238, 0, false, true); + + bgBlueVioletDark = new MotionBackgroundDrawable(0xFFA736D0, 0xFF6A2BDD, 0xFF0F95C9, 0xFF287AE1, 0, false, true); + bgBlueGreenDark = new MotionBackgroundDrawable(0xFF2D60D6, 0xFF2C6ADF, 0xFF009595, 0xFF0291C9, 0, false, true); + bgGreenDark = new MotionBackgroundDrawable(0xFF008B8E, 0xFF01934C, 0xFF8FBD37, 0xFF319D27, 0, false, true); + bgOrangeRedDark = new MotionBackgroundDrawable(0xFFE23F29, 0xFFE6306F, 0xFFC77616, 0xFFD75A16, 0, false, true); + + bgBlueVioletLight = new MotionBackgroundDrawable(0xFFD664FF, 0xFF9258FD, 0xFF2DC0F9, 0xFF57A1FF, 0, false, true); + bgBlueGreenLight = new MotionBackgroundDrawable(0xFF558BFF, 0xFF5FABFF, 0xFF04DCCC, 0xFF28C2FF, 0, false, true); + bgGreenLight = new MotionBackgroundDrawable(0xFF00D2D5, 0xFF09E279, 0xFFC7EF60, 0xFF6DD957, 0, false, true); + bgOrangeRedLight = new MotionBackgroundDrawable(0xFFFF7866, 0xFFFF82A5, 0xFFFEB055, 0xFFFF8E51, 0, false, true); + bgGreenLightReveal = new MotionBackgroundDrawable(0xFF00D2D5, 0xFF09E279, 0xFFC7EF60, 0xFF6DD957, 0, false, true); + bgGreenDarkReveal = new MotionBackgroundDrawable(0xFF008B8E, 0xFF01934C, 0xFF8FBD37, 0xFF319D27, 0, false, true); + + bgBlueVioletDark.setBounds(0, 0, 80, 80); + bgBlueGreenDark.setBounds(0, 0, 80, 80); + bgGreenDark.setBounds(0, 0, 80, 80); + bgOrangeRedDark.setBounds(0, 0, 80, 80); + + bgBlueVioletLight.setBounds(0, 0, 80, 80); + bgBlueGreenLight.setBounds(0, 0, 80, 80); + bgGreenLight.setBounds(0, 0, 80, 80); + bgOrangeRedLight.setBounds(0, 0, 80, 80); + + setWillNotDraw(false); + setLayerType(View.LAYER_TYPE_HARDWARE, null); + + defaultAnimatorSet = new AnimatorSet(); + ValueAnimator rotationAnimator = ValueAnimator.ofInt(0, 360); + rotationAnimator.addUpdateListener(animation -> { + backgroundProvider.setDegree((int) animation.getAnimatedValue()); + int degree = backgroundProvider.getDegree(); + if ((degree >= 0 && degree <= 2) || (degree >= 180 && degree <= 182)) { + if (isPaused) { + defaultAnimatorSet.pause(); + if (connectedAnimatorSet != null) { + connectedAnimatorSet.pause(); + } + } + } + }); + rotationAnimator.setRepeatCount(ValueAnimator.INFINITE); + rotationAnimator.setRepeatMode(ValueAnimator.RESTART); + defaultAnimatorSet.setInterpolator(new LinearInterpolator()); + defaultAnimatorSet.playTogether(rotationAnimator); + defaultAnimatorSet.setDuration(12000); + if (allowAnimations) { + defaultAnimatorSet.start(); + } + switchToCalling(); + } + + @Override + protected void onDetachedFromWindow() { + super.onDetachedFromWindow(); + if (defaultAnimatorSet != null) { + defaultAnimatorSet.cancel(); + } + if (connectedAnimatorSet != null) { + connectedAnimatorSet.cancel(); + } + if (callingAnimator != null) { + callingAnimator.cancel(); + } + } + + public void switchToCalling() { + if (state == GradientState.CALLING) { + return; + } + state = GradientState.CALLING; + alphaBlueGreen = 255; + callingAnimator = ValueAnimator.ofInt(255, 0, 255); + callingAnimator.addUpdateListener(animation -> { + alphaBlueViolet = (int) animation.getAnimatedValue(); + invalidate(); + }); + callingAnimator.setRepeatCount(ValueAnimator.INFINITE); + callingAnimator.setRepeatMode(ValueAnimator.RESTART); + callingAnimator.setInterpolator(new LinearInterpolator()); + callingAnimator.setDuration(12000); + if (allowAnimations) { + callingAnimator.start(); + } + } + + public boolean isConnectedCalled() { + return state == GradientState.CONNECTED || state == GradientState.BAD_CONNECTION; + } + + public void switchToCallConnected(int x, int y) { + switchToCallConnected(x, y, true); + } + + public void switchToCallConnected(int x, int y, boolean animated) { + if (state == GradientState.CONNECTED || state == GradientState.BAD_CONNECTION) { + return; + } + state = GradientState.CONNECTED; + if (callingAnimator != null) { + callingAnimator.removeAllUpdateListeners(); + callingAnimator.cancel(); + callingAnimator = null; + } + clipCx = x; + clipCy = y; + int w = AndroidUtilities.displaySize.x; + int h = AndroidUtilities.displaySize.y + AndroidUtilities.statusBarHeight + AndroidUtilities.navigationBarHeight; + double d1 = Math.sqrt((w - x) * (w - x) + (h - y) * (h - y)); + double d2 = Math.sqrt(x * x + (h - y) * (h - y)); + double d3 = Math.sqrt(x * x + y * y); + double d4 = Math.sqrt((w - x) * (w - x) + y * y); + double revealMaxRadius = Math.max(Math.max(Math.max(d1, d2), d3), d4); + + showClip = true; + backgroundProvider.setReveal(true); + ValueAnimator revealAnimator = ValueAnimator.ofFloat(0f, (float) revealMaxRadius); + revealAnimator.addUpdateListener(animation -> { + clipRadius = (float) animation.getAnimatedValue(); + invalidate(); + backgroundProvider.invalidateViews(); + }); + revealAnimator.addListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + showClip = false; + backgroundProvider.setReveal(false); + if (allowAnimations) { + if (defaultAnimatorSet != null) { + defaultAnimatorSet.cancel(); + defaultAnimatorSet.start(); + } + } + switchToConnectedAnimator(); + } + }); + revealAnimator.setDuration(animated ? 400 : 0); + revealAnimator.start(); + } + + private void switchToConnectedAnimator() { + if (connectedAnimatorSet != null) { + return; + } + if (callingAnimator != null) { + callingAnimator.removeAllUpdateListeners(); + callingAnimator.cancel(); + callingAnimator = null; + } + alphaGreen = 255; + connectedAnimatorSet = new AnimatorSet(); + ValueAnimator blueGreenAnimator = ValueAnimator.ofInt(0, 255, 255, 255, 0); + blueGreenAnimator.addUpdateListener(animation2 -> { + alphaBlueGreen = (int) animation2.getAnimatedValue(); + invalidate(); + }); + blueGreenAnimator.setRepeatCount(ValueAnimator.INFINITE); + blueGreenAnimator.setRepeatMode(ValueAnimator.RESTART); + + ValueAnimator blueVioletAnimator = ValueAnimator.ofInt(0, 0, 255, 0, 0); + blueVioletAnimator.addUpdateListener(animation2 -> { + alphaBlueViolet = (int) animation2.getAnimatedValue(); + invalidate(); + }); + blueVioletAnimator.setRepeatCount(ValueAnimator.INFINITE); + blueVioletAnimator.setRepeatMode(ValueAnimator.RESTART); + + connectedAnimatorSet.playTogether(blueVioletAnimator, blueGreenAnimator); + connectedAnimatorSet.setInterpolator(new LinearInterpolator()); + connectedAnimatorSet.setDuration(24000); + if (allowAnimations) { + connectedAnimatorSet.start(); + } else { + alphaBlueGreen = 0; + alphaBlueViolet = 0; + } + invalidate(); + } + + public void showToBadConnection() { + if (state == GradientState.BAD_CONNECTION) { + return; + } + state = GradientState.BAD_CONNECTION; + badConnectionAnimator = ValueAnimator.ofInt(alphaOrangeRed, 255); + badConnectionAnimator.addUpdateListener(animation -> { + alphaOrangeRed = (int) animation.getAnimatedValue(); + invalidate(); + backgroundProvider.invalidateViews(); + }); + badConnectionAnimator.setDuration(500); + badConnectionAnimator.start(); + } + + public void hideBadConnection() { + if (state == GradientState.CONNECTED) { + return; + } + state = GradientState.CONNECTED; + switchToConnectedAnimator(); + if (badConnectionAnimator != null) { + badConnectionAnimator.removeAllUpdateListeners(); + badConnectionAnimator.cancel(); + } + badConnectionAnimator = ValueAnimator.ofInt(alphaOrangeRed, 0); + badConnectionAnimator.addUpdateListener(animation -> { + alphaOrangeRed = (int) animation.getAnimatedValue(); + invalidate(); + backgroundProvider.invalidateViews(); + }); + badConnectionAnimator.setDuration(500); + badConnectionAnimator.start(); + } + + public void pause() { + if (isPaused) { + return; + } + isPaused = true; + } + + public void resume() { + if (!isPaused) { + return; + } + isPaused = false; + if (defaultAnimatorSet.isPaused()) { + defaultAnimatorSet.resume(); + } + if (connectedAnimatorSet != null && connectedAnimatorSet.isPaused()) { + connectedAnimatorSet.resume(); + } + } + + @Override + protected void onLayout(boolean changed, int left, int top, int right, int bottom) { + super.onLayout(changed, left, top, right, bottom); + bgGreen.setBounds(0, 0, getWidth(), getHeight()); + bgOrangeRed.setBounds(0, 0, getWidth(), getHeight()); + bgBlueGreen.setBounds(0, 0, getWidth(), getHeight()); + bgBlueViolet.setBounds(0, 0, getWidth(), getHeight()); + bgGreenLightReveal.setBounds(0, 0, getWidth() / REVEAL_SCALE_FACTOR, getHeight() / REVEAL_SCALE_FACTOR); + bgGreenDarkReveal.setBounds(0, 0, getWidth() / REVEAL_SCALE_FACTOR, getHeight() / REVEAL_SCALE_FACTOR); + backgroundProvider.setTotalSize(getWidth(), getHeight()); + } + + @Override + protected void onDraw(Canvas canvas) { + if (lockDrawing) { + return; + } + float halfWidth = getWidth() / 2f; + float halfHeight = getHeight() / 2f; + canvas.save(); + canvas.scale(backgroundProvider.scale, backgroundProvider.scale, halfWidth, halfHeight); + canvas.rotate(backgroundProvider.getDegree(), halfWidth, halfHeight); + + backgroundProvider.getLightCanvas().drawColor(Color.TRANSPARENT, PorterDuff.Mode.CLEAR); + backgroundProvider.getDarkCanvas().drawColor(Color.TRANSPARENT, PorterDuff.Mode.CLEAR); + + if (alphaGreen != 0 && alphaOrangeRed != 255) { + bgGreen.setAlpha(alphaGreen); + bgGreenLight.setAlpha(alphaGreen); + bgGreenDark.setAlpha(alphaGreen); + + bgGreen.draw(canvas); + bgGreenLight.draw(backgroundProvider.getLightCanvas()); + bgGreenDark.draw(backgroundProvider.getDarkCanvas()); + } + if (alphaBlueGreen != 0 && alphaOrangeRed != 255) { + bgBlueGreen.setAlpha(alphaBlueGreen); + bgBlueGreenDark.setAlpha(alphaBlueGreen); + bgBlueGreenLight.setAlpha(alphaBlueGreen); + + bgBlueGreen.draw(canvas); + bgBlueGreenDark.draw(backgroundProvider.getDarkCanvas()); + bgBlueGreenLight.draw(backgroundProvider.getLightCanvas()); + } + if (alphaBlueViolet != 0 && alphaOrangeRed != 255) { + bgBlueViolet.setAlpha(alphaBlueViolet); + bgBlueVioletDark.setAlpha(alphaBlueViolet); + bgBlueVioletLight.setAlpha(alphaBlueViolet); + + bgBlueViolet.draw(canvas); + bgBlueVioletDark.draw(backgroundProvider.getDarkCanvas()); + bgBlueVioletLight.draw(backgroundProvider.getLightCanvas()); + } + + if (alphaOrangeRed != 0) { + bgOrangeRed.setAlpha(alphaOrangeRed); + bgOrangeRedDark.setAlpha(alphaOrangeRed); + bgOrangeRedLight.setAlpha(alphaOrangeRed); + + bgOrangeRed.draw(canvas); + bgOrangeRedDark.draw(backgroundProvider.getDarkCanvas()); + bgOrangeRedLight.draw(backgroundProvider.getLightCanvas()); + } + canvas.restore(); + + if (showClip) { + clipPath.rewind(); + clipPath.addCircle(clipCx, clipCy, clipRadius, Path.Direction.CW); + canvas.clipPath(clipPath); + canvas.scale(backgroundProvider.scale, backgroundProvider.scale, halfWidth, halfHeight); + bgGreen.setAlpha(255); + bgGreen.draw(canvas); + + clipPath.rewind(); + clipPath.addCircle(clipCx / (float) REVEAL_SCALE_FACTOR, clipCy / (float) REVEAL_SCALE_FACTOR, clipRadius / REVEAL_SCALE_FACTOR, Path.Direction.CW); + backgroundProvider.getRevealCanvas().drawColor(Color.TRANSPARENT, PorterDuff.Mode.CLEAR); + backgroundProvider.getRevealCanvas().save(); + backgroundProvider.getRevealCanvas().clipPath(clipPath); + bgGreenLightReveal.setAlpha(255); + bgGreenLightReveal.draw(backgroundProvider.getRevealCanvas()); + backgroundProvider.getRevealCanvas().restore(); + + backgroundProvider.getRevealDrakCanvas().drawColor(Color.TRANSPARENT, PorterDuff.Mode.CLEAR); + backgroundProvider.getRevealDrakCanvas().save(); + backgroundProvider.getRevealDrakCanvas().clipPath(clipPath); + bgGreenDarkReveal.setAlpha(255); + bgGreenDarkReveal.draw(backgroundProvider.getRevealDrakCanvas()); + backgroundProvider.getRevealDrakCanvas().restore(); + } + super.onDraw(canvas); + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIpHintView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIpHintView.java new file mode 100644 index 00000000000..cec7d97b00f --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIpHintView.java @@ -0,0 +1,44 @@ +package org.telegram.ui.Components.voip; + +import android.annotation.SuppressLint; +import android.content.Context; +import android.graphics.Canvas; +import android.graphics.CornerPathEffect; +import android.graphics.Paint; + +import org.telegram.ui.Stories.recorder.HintView2; + +@SuppressLint("ViewConstructor") +public class VoIpHintView extends HintView2 { + + private final Paint mainPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + private final VoIPBackgroundProvider backgroundProvider; + + public VoIpHintView(Context context, int direction, VoIPBackgroundProvider backgroundProvider, boolean withCloseBtn) { + super(context, direction); + this.backgroundProvider = backgroundProvider; + backgroundProvider.attach(this); + mainPaint.setPathEffect(new CornerPathEffect(rounding)); + if (withCloseBtn) { + setCloseButton(true); + } + } + + @Override + protected void dispatchDraw(Canvas canvas) { + backgroundProvider.setDarkTranslation(getX(), getY()); + super.dispatchDraw(canvas); + } + + protected void drawBgPath(Canvas canvas) { + mainPaint.setShader(backgroundProvider.getDarkPaint().getShader()); + int alpha = Math.min(backgroundPaint.getAlpha(), backgroundProvider.getDarkPaint().getAlpha()); + canvas.saveLayerAlpha(0, 0, getMeasuredWidth(), getMeasuredHeight(), alpha, Canvas.ALL_SAVE_FLAG); + canvas.drawPath(path, mainPaint); + if (backgroundProvider.isReveal()) { + mainPaint.setShader(backgroundProvider.getRevealDarkPaint().getShader()); + canvas.drawPath(path, mainPaint); + } + canvas.restore(); + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIpSnowView.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIpSnowView.java new file mode 100644 index 00000000000..f5e705fadb6 --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIpSnowView.java @@ -0,0 +1,39 @@ +package org.telegram.ui.Components.voip; + +import android.content.Context; +import android.graphics.Canvas; +import android.view.View; + +import org.telegram.messenger.LiteMode; +import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.Components.SnowflakesEffect; + +public class VoIpSnowView extends View { + private SnowflakesEffect snowflakesEffect; + private boolean isPaused; + + public VoIpSnowView(Context context) { + super(context); + } + + { + if (LiteMode.isEnabled(LiteMode.FLAG_CALLS_ANIMATIONS) && Theme.getEventType() == 0) { + snowflakesEffect = new SnowflakesEffect(0); + } + } + + @Override + protected void onDraw(Canvas canvas) { + if (isPaused) { + return; + } + if (snowflakesEffect != null) { + snowflakesEffect.onDraw(this, canvas); + } + } + + public void setState(boolean isPaused) { + this.isPaused = isPaused; + invalidate(); + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIpSwitchLayout.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIpSwitchLayout.java new file mode 100644 index 00000000000..8f48d428d67 --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoIpSwitchLayout.java @@ -0,0 +1,482 @@ +package org.telegram.ui.Components.voip; + +import android.animation.Animator; +import android.animation.AnimatorListenerAdapter; +import android.animation.ValueAnimator; +import android.annotation.SuppressLint; +import android.content.Context; +import android.graphics.Canvas; +import android.graphics.Color; +import android.graphics.Paint; +import android.graphics.Path; +import android.graphics.PorterDuff; +import android.graphics.PorterDuffColorFilter; +import android.graphics.PorterDuffXfermode; +import android.util.TypedValue; +import android.view.Gravity; +import android.view.MotionEvent; +import android.view.View; +import android.widget.FrameLayout; +import android.widget.TextView; + +import androidx.annotation.NonNull; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.LocaleController; +import org.telegram.messenger.R; +import org.telegram.ui.Components.LayoutHelper; +import org.telegram.ui.Components.RLottieDrawable; + +@SuppressLint("ViewConstructor") +public class VoIpSwitchLayout extends FrameLayout { + + public enum Type { + MICRO, + CAMERA, + VIDEO, + BLUETOOTH, + SPEAKER, + } + + private final VoIPBackgroundProvider backgroundProvider; + private VoIpButtonView voIpButtonView; + private Type type; + private final TextView currentTextView; + private final TextView newTextView; + public int animationDelay; + + public void setOnBtnClickedListener(VoIpButtonView.OnBtnClickedListener onBtnClickedListener) { + voIpButtonView.setOnBtnClickedListener(onBtnClickedListener); + } + + public VoIpSwitchLayout(@NonNull Context context, VoIPBackgroundProvider backgroundProvider) { + super(context); + this.backgroundProvider = backgroundProvider; + setWillNotDraw(true); + voIpButtonView = new VoIpButtonView(context, backgroundProvider); + addView(voIpButtonView, LayoutHelper.createFrame(VoIpButtonView.ITEM_SIZE + 1.5f, VoIpButtonView.ITEM_SIZE + 1.5f, Gravity.CENTER_HORIZONTAL)); + + currentTextView = new TextView(context); + currentTextView.setGravity(Gravity.CENTER_HORIZONTAL); + currentTextView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 11); + currentTextView.setTextColor(Color.WHITE); + currentTextView.setImportantForAccessibility(IMPORTANT_FOR_ACCESSIBILITY_NO); + addView(currentTextView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, 0, 0, VoIpButtonView.ITEM_SIZE + 6, 0, 2)); + + newTextView = new TextView(context); + newTextView.setGravity(Gravity.CENTER_HORIZONTAL); + newTextView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 11); + newTextView.setTextColor(Color.WHITE); + newTextView.setImportantForAccessibility(IMPORTANT_FOR_ACCESSIBILITY_NO); + addView(newTextView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, 0, 0, VoIpButtonView.ITEM_SIZE + 6, 0, 2)); + currentTextView.setVisibility(GONE); + newTextView.setVisibility(GONE); + } + + private void setText(Type type, boolean isSelectedState) { + final String newText; + switch (type) { + case MICRO: + if (isSelectedState) { + newText = LocaleController.getString("VoipUnmute", R.string.VoipUnmute); + } else { + newText = LocaleController.getString("VoipMute", R.string.VoipMute); + } + break; + case CAMERA: + newText = LocaleController.getString("VoipFlip", R.string.VoipFlip); + break; + case VIDEO: + if (isSelectedState) { + newText = LocaleController.getString("VoipStartVideo", R.string.VoipStartVideo); + } else { + newText = LocaleController.getString("VoipStopVideo", R.string.VoipStopVideo); + } + break; + case BLUETOOTH: + newText = LocaleController.getString("VoipAudioRoutingBluetooth", R.string.VoipAudioRoutingBluetooth); + break; + case SPEAKER: + newText = LocaleController.getString("VoipSpeaker", R.string.VoipSpeaker); + break; + default: + newText = ""; + } + + if (currentTextView.getVisibility() == GONE && newTextView.getVisibility() == GONE) { + currentTextView.setVisibility(VISIBLE); + currentTextView.setText(newText); + newTextView.setText(newText); + return; + } + + if (newTextView.getText().equals(newText) && currentTextView.getText().equals(newText)) { + return; + } + + currentTextView.animate().alpha(0f).translationY(-AndroidUtilities.dp(4)).setDuration(140).setListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + currentTextView.setText(newText); + currentTextView.setTranslationY(0); + currentTextView.setAlpha(1.0f); + } + }).start(); + newTextView.setText(newText); + newTextView.setVisibility(VISIBLE); + newTextView.setAlpha(0); + newTextView.setTranslationY(AndroidUtilities.dp(5)); + newTextView.animate().alpha(1.0f).translationY(0).setDuration(150).setListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + newTextView.setVisibility(GONE); + } + }).start(); + } + + private void attachNewButton(int rawRes, int size, boolean isSelected, Type type) { + final VoIpButtonView newVoIpButtonView = new VoIpButtonView(getContext(), backgroundProvider); + if (rawRes == R.raw.camera_flip2) { + newVoIpButtonView.singleIcon = new RLottieDrawable(rawRes, "" + rawRes, size, size, true, null); + newVoIpButtonView.singleIcon.setMasterParent(newVoIpButtonView); + } else { + newVoIpButtonView.unSelectedIcon = new RLottieDrawable(rawRes, "" + rawRes, size, size, true, null); + newVoIpButtonView.selectedIcon = new RLottieDrawable(rawRes, "" + rawRes, size, size, true, null); + newVoIpButtonView.selectedIcon.setColorFilter(new PorterDuffColorFilter(Color.BLACK, PorterDuff.Mode.MULTIPLY)); + } + newVoIpButtonView.setSelectedState(isSelected, false, type); + newVoIpButtonView.setAlpha(0f); + newVoIpButtonView.setOnBtnClickedListener(voIpButtonView.onBtnClickedListener); + addView(newVoIpButtonView, LayoutHelper.createFrame(VoIpButtonView.ITEM_SIZE, VoIpButtonView.ITEM_SIZE, Gravity.CENTER_HORIZONTAL)); + final VoIpButtonView oldVoIpButton = voIpButtonView; + voIpButtonView = newVoIpButtonView; + newVoIpButtonView.animate().alpha(1f).setDuration(250).start(); + oldVoIpButton.animate().alpha(0f).setDuration(250).setListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + removeView(oldVoIpButton); + } + }).start(); + } + + public void setType(Type newType, boolean isSelected) { + setType(newType, isSelected, false); + } + + public void setType(Type newType, boolean isSelected, boolean fast) { + if (this.type == newType && isSelected == voIpButtonView.isSelectedState) { + if (getVisibility() != View.VISIBLE) { + setVisibility(View.VISIBLE); + } + return; + } + if (getVisibility() != View.VISIBLE) { + setVisibility(View.VISIBLE); + } + int size = AndroidUtilities.dp(VoIpButtonView.ITEM_SIZE + 1.5f); + boolean ignoreSetState = false; + switch (newType) { + case MICRO: + if (this.type != Type.MICRO) { + voIpButtonView.unSelectedIcon = new RLottieDrawable(R.raw.call_mute, "" + R.raw.call_mute, size, size, true, null); + voIpButtonView.selectedIcon = new RLottieDrawable(R.raw.call_mute, "" + R.raw.call_mute, size, size, true, null); + voIpButtonView.selectedIcon.setColorFilter(new PorterDuffColorFilter(Color.BLACK, PorterDuff.Mode.MULTIPLY)); + voIpButtonView.selectedIcon.setMasterParent(voIpButtonView); + } + break; + case VIDEO: + //R.drawable.calls_sharescreen screencast is not used in the design + if (this.type != Type.VIDEO) { + voIpButtonView.unSelectedIcon = new RLottieDrawable(R.raw.video_stop, "" + R.raw.video_stop, size, size, true, null); + voIpButtonView.selectedIcon = new RLottieDrawable(R.raw.video_stop, "" + R.raw.video_stop, size, size, true, null); + voIpButtonView.selectedIcon.setColorFilter(new PorterDuffColorFilter(Color.BLACK, PorterDuff.Mode.MULTIPLY)); + voIpButtonView.selectedIcon.setMasterParent(voIpButtonView); + } + break; + case CAMERA: + if (this.type == Type.SPEAKER || this.type == Type.BLUETOOTH) { + ignoreSetState = true; + attachNewButton(R.raw.camera_flip2, size, isSelected, newType); + } else if (this.type != Type.CAMERA) { + voIpButtonView.singleIcon = new RLottieDrawable(R.raw.camera_flip2, "" + R.raw.camera_flip2, size, size, true, null); + voIpButtonView.singleIcon.setMasterParent(voIpButtonView); + } + break; + case SPEAKER: + if (this.type == Type.BLUETOOTH) { + ignoreSetState = isSelected == voIpButtonView.isSelectedState; + RLottieDrawable icon = isSelected ? voIpButtonView.selectedIcon : voIpButtonView.unSelectedIcon; + icon.setMasterParent(voIpButtonView); + icon.setOnAnimationEndListener(() -> AndroidUtilities.runOnUIThread(() -> attachSpeakerToBt(size))); + icon.start(); + } else if (this.type == Type.CAMERA) { + ignoreSetState = true; + attachNewButton(R.raw.speaker_to_bt, size, isSelected, newType); + } else if (this.type != Type.SPEAKER) { + attachSpeakerToBt(size); + } + break; + case BLUETOOTH: + if (this.type == Type.SPEAKER) { + ignoreSetState = isSelected == voIpButtonView.isSelectedState; + RLottieDrawable icon = isSelected ? voIpButtonView.selectedIcon : voIpButtonView.unSelectedIcon; + icon.setMasterParent(voIpButtonView); + icon.setOnAnimationEndListener(() -> AndroidUtilities.runOnUIThread(() -> attachBtToSpeaker(size))); + icon.start(); + } else if (this.type == Type.CAMERA) { + ignoreSetState = true; + attachNewButton(R.raw.bt_to_speaker, size, isSelected, newType); + } else if (this.type != Type.BLUETOOTH) { + attachBtToSpeaker(size); + } + break; + } + + if (!ignoreSetState) { + voIpButtonView.setSelectedState(isSelected, this.type != null && !fast, newType); + } + setText(newType, isSelected); + this.type = newType; + } + + private void attachSpeakerToBt(int size) { + voIpButtonView.unSelectedIcon = new RLottieDrawable(R.raw.speaker_to_bt, "" + R.raw.speaker_to_bt, size, size, true, null); + voIpButtonView.selectedIcon = new RLottieDrawable(R.raw.speaker_to_bt, "" + R.raw.speaker_to_bt, size, size, true, null); + voIpButtonView.selectedIcon.setColorFilter(new PorterDuffColorFilter(Color.BLACK, PorterDuff.Mode.MULTIPLY)); + } + + private void attachBtToSpeaker(int size) { + voIpButtonView.unSelectedIcon = new RLottieDrawable(R.raw.bt_to_speaker, "" + R.raw.bt_to_speaker, size, size, true, null); + voIpButtonView.selectedIcon = new RLottieDrawable(R.raw.bt_to_speaker, "" + R.raw.bt_to_speaker, size, size, true, null); + voIpButtonView.selectedIcon.setColorFilter(new PorterDuffColorFilter(Color.BLACK, PorterDuff.Mode.MULTIPLY)); + } + + public static class VoIpButtonView extends View { + private static final int ITEM_SIZE = 52; + + private RLottieDrawable unSelectedIcon; + private RLottieDrawable selectedIcon; + private RLottieDrawable singleIcon; + private final Paint maskPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + private final Paint whiteCirclePaint = new Paint(Paint.ANTI_ALIAS_FLAG); + private final Paint darkPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + private final Path clipPath = new Path(); + private final int maxRadius = AndroidUtilities.dp(ITEM_SIZE / 2f); + private int unselectedRadius = maxRadius; + private int selectedRadius = 0; + private boolean isSelectedState = false; + private int singleIconBackgroundAlphaPercent = 0; + private OnBtnClickedListener onBtnClickedListener; + private ValueAnimator animator; + private final VoIPBackgroundProvider backgroundProvider; + + public void setSelectedState(boolean selectedState, boolean animate, Type type) { + if (animate) { + if (singleIcon != null) { + if (animator != null) { + animator.removeAllUpdateListeners(); + animator.cancel(); + } + animator = selectedState ? ValueAnimator.ofInt(20, 100) : ValueAnimator.ofInt(100, 20); + animator.addUpdateListener(animation -> { + singleIconBackgroundAlphaPercent = (int) animation.getAnimatedValue(); + invalidate(); + }); + animator.setDuration(200); + animator.start(); + if (type == Type.CAMERA) { + singleIcon.setCurrentFrame(0, false); + singleIcon.start(); + } + } else { + if (animator != null) { + animator.removeAllUpdateListeners(); + animator.cancel(); + } + animator = ValueAnimator.ofInt(0, maxRadius); + if (selectedState) { + unselectedRadius = maxRadius; + animator.addUpdateListener(animation -> { + selectedRadius = (int) animation.getAnimatedValue(); + invalidate(); + }); + animator.addListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + unselectedRadius = 0; //switched to selected state + invalidate(); + } + }); + animator.setDuration(200); + animator.start(); + selectedIcon.setCurrentFrame(0, false); + selectedIcon.start(); + } else { + selectedRadius = maxRadius; + animator.addUpdateListener(animation -> { + unselectedRadius = (int) animation.getAnimatedValue(); + invalidate(); + }); + animator.setDuration(200); + animator.addListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + selectedRadius = 0; //switched to NOT selected state + invalidate(); + } + }); + animator.start(); + } + } + } else { + if (selectedState) { + selectedRadius = maxRadius; + unselectedRadius = 0; + singleIconBackgroundAlphaPercent = 100; + if (type == Type.VIDEO || type == Type.MICRO) { + selectedIcon.setCurrentFrame(selectedIcon.getFramesCount() - 1, false); + } + } else { + selectedRadius = 0; + unselectedRadius = maxRadius; + singleIconBackgroundAlphaPercent = 20; + } + } + isSelectedState = selectedState; + invalidate(); + } + + public interface OnBtnClickedListener { + void onClicked(View view); + } + + public void setOnBtnClickedListener(OnBtnClickedListener onBtnClickedListener) { + this.onBtnClickedListener = onBtnClickedListener; + } + + public VoIpButtonView(@NonNull Context context, VoIPBackgroundProvider backgroundProvider) { + super(context); + this.backgroundProvider = backgroundProvider; + backgroundProvider.attach(this); + setLayerType(View.LAYER_TYPE_SOFTWARE, null); + whiteCirclePaint.setColor(Color.WHITE); + + maskPaint.setColor(Color.BLACK); + maskPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_OUT)); + + darkPaint.setColor(Color.BLACK); + darkPaint.setColorFilter(new PorterDuffColorFilter(Color.BLACK, PorterDuff.Mode.SRC_ATOP)); + darkPaint.setAlpha(VoIPBackgroundProvider.DARK_LIGHT_DEFAULT_ALPHA); + } + + @Override + protected void onDraw(Canvas canvas) { + float cx = getWidth() / 2f; + float cy = getHeight() / 2f; + + float left = getX() + ((View) getParent()).getX(); + float top = getY() + ((View) ((View) getParent()).getParent()).getY(); + backgroundProvider.setLightTranslation(left, top); + + if (singleIcon != null) { + if (singleIconBackgroundAlphaPercent > 20) { + darkPaint.setAlpha((int) (VoIPBackgroundProvider.DARK_LIGHT_DEFAULT_ALPHA * singleIconBackgroundAlphaPercent / 100f)); + whiteCirclePaint.setAlpha((int) (255 * singleIconBackgroundAlphaPercent / 100f)); + canvas.drawCircle(cx, cy, maxRadius, whiteCirclePaint); + singleIcon.draw(canvas, maskPaint); + singleIcon.draw(canvas, darkPaint); //dimming icons + } else { + canvas.drawCircle(cx, cy, maxRadius, backgroundProvider.getLightPaint()); //add a light background + if(backgroundProvider.isReveal()) { + canvas.drawCircle(cx, cy, maxRadius, backgroundProvider.getRevealPaint()); + } + singleIcon.draw(canvas); + } + return; + } + if (selectedIcon == null || unSelectedIcon == null) return; + + boolean isUnSelected = unselectedRadius == maxRadius && selectedRadius == 0; + boolean isSelected = selectedRadius == maxRadius && unselectedRadius == 0; + + if (selectedRadius == maxRadius && unselectedRadius > 0 && unselectedRadius != maxRadius) { + //in the process of changing from selected to NOT selected. + canvas.drawCircle(cx, cy, selectedRadius, whiteCirclePaint); + canvas.drawCircle(cx, cy, unselectedRadius, maskPaint); + + selectedIcon.setAlpha(255); + selectedIcon.draw(canvas, maskPaint); + selectedIcon.setAlpha((int) (255 * VoIPBackgroundProvider.DARK_LIGHT_PERCENT)); + selectedIcon.draw(canvas); //dimming icons + + clipPath.reset(); + clipPath.addCircle(cx, cy, unselectedRadius, Path.Direction.CW); + canvas.clipPath(clipPath); + canvas.drawCircle(cx, cy, unselectedRadius, maskPaint); //remove all background + } + + if (isUnSelected || unselectedRadius > 0) { + //not selected or in the process of changing from selected to NOT selected + canvas.drawCircle(cx, cy, unselectedRadius, backgroundProvider.getLightPaint()); //add a light background + if (backgroundProvider.isReveal()) { + canvas.drawCircle(cx, cy, unselectedRadius, backgroundProvider.getRevealPaint()); + } + unSelectedIcon.draw(canvas); + } + + if (isSelected || (selectedRadius > 0 && unselectedRadius == maxRadius)) { + //selected and not in the process of changing or in the process of changing from NOT selected to selected. + clipPath.reset(); + clipPath.addCircle(cx, cy, selectedRadius, Path.Direction.CW); + canvas.clipPath(clipPath); + canvas.drawCircle(cx, cy, selectedRadius, whiteCirclePaint); //circular background + selectedIcon.setAlpha(255); + selectedIcon.draw(canvas, maskPaint); + selectedIcon.setAlpha((int) (255 * VoIPBackgroundProvider.DARK_LIGHT_PERCENT)); + selectedIcon.draw(canvas); //dimming icons + } + } + + private boolean isAnimating() { + boolean isUnSelected = unselectedRadius == maxRadius && selectedRadius == 0; + boolean isSelected = selectedRadius == maxRadius && unselectedRadius == 0; + return !isUnSelected && !isSelected; + } + + private float startX; + private float startY; + + @SuppressLint("ClickableViewAccessibility") + @Override + public boolean onTouchEvent(MotionEvent event) { + switch (event.getAction()) { + case MotionEvent.ACTION_DOWN: + animate().scaleX(0.8f).scaleY(0.8f).setDuration(150).start(); + animate().scaleX(0.8f).scaleY(0.8f).setDuration(150).start(); + startX = event.getX(); + startY = event.getY(); + break; + case MotionEvent.ACTION_UP: + animate().scaleX(1.0f).scaleY(1.0f).setDuration(150).start(); + animate().scaleX(1.0f).scaleY(1.0f).setDuration(150).start(); + float endX = event.getX(); + float endY = event.getY(); + if (isClick(startX, endX, startY, endY) && !isAnimating()) { + if (onBtnClickedListener != null) onBtnClickedListener.onClicked(this); + } + break; + case MotionEvent.ACTION_CANCEL: + animate().scaleX(1.0f).scaleY(1.0f).setDuration(150).start(); + animate().scaleX(1.0f).scaleY(1.0f).setDuration(150).start(); + break; + } + return true; + } + + private boolean isClick(float startX, float endX, float startY, float endY) { + float differenceX = Math.abs(startX - endX); + float differenceY = Math.abs(startY - endY); + return !(differenceX > AndroidUtilities.dp(48) || differenceY > AndroidUtilities.dp(48)); + } + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoipBlobDrawable.java b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoipBlobDrawable.java new file mode 100644 index 00000000000..ccfbc368c2f --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Components/voip/VoipBlobDrawable.java @@ -0,0 +1,42 @@ +package org.telegram.ui.Components.voip; + +import org.telegram.messenger.LiteMode; +import org.telegram.ui.Components.BlobDrawable; + +public class VoipBlobDrawable extends BlobDrawable { + + public VoipBlobDrawable(int n) { + super(n); + } + + public VoipBlobDrawable(int n, int liteFlag) { + super(n, liteFlag); + } + + protected void generateBlob(float[] radius, float[] angle, int i, float muteToStaticProgress) { + float angleDif = 360f / N * 0.05f; + float radDif = maxRadius - minRadius; + radius[i] = minRadius + ((Math.abs(((random.nextInt() % 100f) / 100f)) * radDif) * muteToStaticProgress); + angle[i] = 360f / N * i + (((random.nextInt() * muteToStaticProgress) % 100f) / 100f) * angleDif; + speed[i] = (float) (0.017 + 0.003 * (Math.abs(random.nextInt() % 100f) / 100f)); + } + + public void update(float amplitude, float speedScale, float muteToStaticProgress) { + if (!LiteMode.isEnabled(liteFlag)) { + return; + } + for (int i = 0; i < N; i++) { + progress[i] += (speed[i] * MIN_SPEED) + amplitude * speed[i] * MAX_SPEED * speedScale; + if (progress[i] >= 1f) { + progress[i] = 0; + radius[i] = radiusNext[i]; + angle[i] = angleNext[i]; + if (muteToStaticProgress < 1f) { + generateBlob(radiusNext, angleNext, i, muteToStaticProgress); + } else { + generateBlob(radiusNext, angleNext, i); + } + } + } + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/ContactsActivity.java b/TMessagesProj/src/main/java/org/telegram/ui/ContactsActivity.java index 92ccd78cabc..faafc50823b 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/ContactsActivity.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/ContactsActivity.java @@ -1069,7 +1069,9 @@ public void onPause() { @Override public void didReceivedNotification(int id, int account, Object... args) { if (id == NotificationCenter.storiesUpdated) { - listViewAdapter.setStories(getMessagesController().getStoriesController().getHiddenList(), true); + if (listViewAdapter != null) { + listViewAdapter.setStories(getMessagesController().getStoriesController().getHiddenList(), true); + } MessagesController.getInstance(currentAccount).getStoriesController().loadHiddenStories(); } else if (id == NotificationCenter.contactsDidLoad) { if (listViewAdapter != null) { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/DefaultThemesPreviewCell.java b/TMessagesProj/src/main/java/org/telegram/ui/DefaultThemesPreviewCell.java index 980426c61c8..9b4af25497e 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/DefaultThemesPreviewCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/DefaultThemesPreviewCell.java @@ -44,6 +44,9 @@ @SuppressLint("ViewConstructor") public class DefaultThemesPreviewCell extends LinearLayout { + public final static int TYPE_CUSTOM_LIST = -1; + public final static int TYPE_CUSTOM_GRID = -2; // not implemented + private final RecyclerListView recyclerView; private LinearLayoutManager layoutManager = null; private final FlickerLoadingView progressView; @@ -70,8 +73,13 @@ public DefaultThemesPreviewCell(Context context, BaseFragment parentFragment, in addView(frameLayout, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT)); - adapter = new ChatThemeBottomSheet.Adapter(parentFragment.getCurrentAccount(), null, currentType == ThemeActivity.THEME_TYPE_BASIC ? ThemeSmallPreviewView.TYPE_DEFAULT : ThemeSmallPreviewView.TYPE_GRID); - recyclerView = new RecyclerListView(getContext()); + adapter = new ChatThemeBottomSheet.Adapter(parentFragment.getCurrentAccount(), null, currentType == ThemeActivity.THEME_TYPE_BASIC || currentType == TYPE_CUSTOM_LIST ? ThemeSmallPreviewView.TYPE_DEFAULT : ThemeSmallPreviewView.TYPE_GRID); + recyclerView = new RecyclerListView(getContext()) { + @Override + public Integer getSelectorColor(int position) { + return 0; + } + }; recyclerView.setAdapter(adapter); recyclerView.setSelectorDrawableColor(0); recyclerView.setClipChildren(false); @@ -136,7 +144,7 @@ public DefaultThemesPreviewCell(Context context, BaseFragment parentFragment, in progressView.setViewType(FlickerLoadingView.CHAT_THEMES_TYPE); progressView.setVisibility(View.VISIBLE); - if (currentType == ThemeActivity.THEME_TYPE_BASIC) { + if (currentType == ThemeActivity.THEME_TYPE_BASIC || currentType == TYPE_CUSTOM_LIST) { frameLayout.addView(progressView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 104, Gravity.START, 0, 8, 0, 8)); frameLayout.addView(recyclerView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 104, Gravity.START, 0, 8, 0, 8)); } else { @@ -312,7 +320,7 @@ public void updateLayoutManager() { if (wasPortrait != null && wasPortrait == isPortrait) { return; } - if (currentType == ThemeActivity.THEME_TYPE_BASIC) { + if (currentType == ThemeActivity.THEME_TYPE_BASIC || currentType == TYPE_CUSTOM_LIST) { if (layoutManager == null) { recyclerView.setLayoutManager(layoutManager = new LinearLayoutManager(getContext(), LinearLayoutManager.HORIZONTAL, false)); } @@ -342,7 +350,7 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { } public void updateDayNightMode() { - if (currentType == ThemeActivity.THEME_TYPE_BASIC) { + if (currentType == ThemeActivity.THEME_TYPE_BASIC || currentType == TYPE_CUSTOM_LIST) { themeIndex = !Theme.isCurrentThemeDay() ? 2 : 0; } else { if (Theme.getActiveTheme().getKey().equals("Blue")) { @@ -440,13 +448,18 @@ public void selectTheme(Theme.ThemeInfo themeInfo) { } public void updateColors() { - if (currentType == ThemeActivity.THEME_TYPE_BASIC) { - darkThemeDrawable.setColorFilter(new PorterDuffColorFilter(Theme.getColor(Theme.key_windowBackgroundWhiteBlueText4), PorterDuff.Mode.SRC_IN)); - - Theme.setSelectorDrawableColor(dayNightCell.getBackground(), Theme.getColor(Theme.key_listSelector), true); - browseThemesCell.setBackground(Theme.createSelectorWithBackgroundDrawable(Theme.getColor(Theme.key_windowBackgroundWhite), Theme.getColor(Theme.key_listSelector))); - dayNightCell.setColors(-1, Theme.key_windowBackgroundWhiteBlueText4); - browseThemesCell.setColors(Theme.key_windowBackgroundWhiteBlueText4, Theme.key_windowBackgroundWhiteBlueText4); + if (currentType == ThemeActivity.THEME_TYPE_BASIC || currentType == TYPE_CUSTOM_LIST) { + if (darkThemeDrawable != null) { + darkThemeDrawable.setColorFilter(new PorterDuffColorFilter(Theme.getColor(Theme.key_windowBackgroundWhiteBlueText4), PorterDuff.Mode.SRC_IN)); + } + if (dayNightCell != null) { + Theme.setSelectorDrawableColor(dayNightCell.getBackground(), Theme.getColor(Theme.key_listSelector), true); + dayNightCell.setColors(-1, Theme.key_windowBackgroundWhiteBlueText4); + } + if (browseThemesCell != null) { + browseThemesCell.setBackground(Theme.createSelectorWithBackgroundDrawable(Theme.getColor(Theme.key_windowBackgroundWhite), Theme.getColor(Theme.key_listSelector))); + browseThemesCell.setColors(Theme.key_windowBackgroundWhiteBlueText4, Theme.key_windowBackgroundWhiteBlueText4); + } } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/DialogsActivity.java b/TMessagesProj/src/main/java/org/telegram/ui/DialogsActivity.java index 4a2efd73c0b..dc946e519df 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/DialogsActivity.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/DialogsActivity.java @@ -45,6 +45,8 @@ import android.os.Bundle; import android.text.TextPaint; import android.text.TextUtils; +import android.transition.ChangeBounds; +import android.transition.TransitionManager; import android.util.LongSparseArray; import android.util.Property; import android.util.StateSet; @@ -189,6 +191,7 @@ import org.telegram.ui.Components.PacmanAnimation; import org.telegram.ui.Components.Premium.LimitReachedBottomSheet; import org.telegram.ui.Components.Premium.PremiumFeatureBottomSheet; +import org.telegram.ui.Components.Premium.boosts.UserSelectorBottomSheet; import org.telegram.ui.Components.ProxyDrawable; import org.telegram.ui.Components.PullForegroundDrawable; import org.telegram.ui.Components.RLottieDrawable; @@ -1292,7 +1295,7 @@ public boolean onTouchEvent(MotionEvent ev) { filterTabsView != null && !filterTabsView.isEditing() && !searching && !rightSlidingDialogContainer.hasFragment() && - !parentLayout.checkTransitionAnimation() && !parentLayout.isInPreviewMode() && !parentLayout.isPreviewOpenAnimationInProgress() && !parentLayout.getDrawerLayoutContainer().isDrawerOpened() && + !parentLayout.checkTransitionAnimation() && !parentLayout.isInPreviewMode() && !parentLayout.isPreviewOpenAnimationInProgress() && !(parentLayout.getDrawerLayoutContainer() != null && parentLayout.getDrawerLayoutContainer().isDrawerOpened()) && ( ev == null || startedTracking || @@ -1313,7 +1316,9 @@ public boolean onTouchEvent(MotionEvent ev) { startedTracking = true; startedTrackingPointerId = ev.getPointerId(0); startedTrackingX = (int) ev.getX(); - parentLayout.getDrawerLayoutContainer().setAllowOpenDrawerBySwipe(false); + if (parentLayout.getDrawerLayoutContainer() != null) { + parentLayout.getDrawerLayoutContainer().setAllowOpenDrawerBySwipe(false); + } if (animatingForward) { if (startedTrackingX < viewPages[0].getMeasuredWidth() + viewPages[0].getTranslationX()) { additionalOffset = viewPages[0].getTranslationX(); @@ -3236,7 +3241,7 @@ public void setTranslationY(float translationY) { @Override protected void onDefaultTabMoved() { - if (!getMessagesController().premiumLocked) { + if (!getMessagesController().premiumFeaturesBlocked()) { try { performHapticFeedback(HapticFeedbackConstants.KEYBOARD_PRESS, HapticFeedbackConstants.FLAG_IGNORE_VIEW_SETTING); } catch (Exception ignore) { @@ -4712,7 +4717,7 @@ public void didPressAttachButton() { } @Override - public void needStartRecordVideo(int state, boolean notify, int scheduleDate) { + public void needStartRecordVideo(int state, boolean notify, int scheduleDate, int ttl) { } @@ -5402,14 +5407,21 @@ private int getMaxScrollYOffset() { } public boolean isPremiumRestoreHintVisible() { - if (!MessagesController.getInstance(currentAccount).premiumLocked && folderId == 0) { + if (!MessagesController.getInstance(currentAccount).premiumFeaturesBlocked() && folderId == 0) { return MessagesController.getInstance(currentAccount).pendingSuggestions.contains("PREMIUM_RESTORE") && !getUserConfig().isPremium(); } return false; } + public boolean isPremiumChristmasHintVisible() { + if (!MessagesController.getInstance(currentAccount).premiumFeaturesBlocked() && folderId == 0) { + return MessagesController.getInstance(currentAccount).pendingSuggestions.contains("PREMIUM_CHRISTMAS"); + } + return false; + } + public boolean isPremiumHintVisible() { - if (!MessagesController.getInstance(currentAccount).premiumLocked && folderId == 0) { + if (!MessagesController.getInstance(currentAccount).premiumFeaturesBlocked() && folderId == 0) { if (MessagesController.getInstance(currentAccount).pendingSuggestions.contains("PREMIUM_UPGRADE") && getUserConfig().isPremium() || MessagesController.getInstance(currentAccount).pendingSuggestions.contains("PREMIUM_ANNUAL") && !getUserConfig().isPremium()) { if (UserConfig.getInstance(currentAccount).isPremium() ? !BuildVars.useInvoiceBilling() && MediaDataController.getInstance(currentAccount).getPremiumHintAnnualDiscount(true) != null : MediaDataController.getInstance(currentAccount).getPremiumHintAnnualDiscount(false) != null) { isPremiumHintUpgrade = MessagesController.getInstance(currentAccount).pendingSuggestions.contains("PREMIUM_UPGRADE"); @@ -5497,8 +5509,8 @@ protected void onEmojiSelected(View emojiView, Long documentId, TLRPC.Document d } } }; - if (user != null && user.emoji_status instanceof TLRPC.TL_emojiStatusUntil && ((TLRPC.TL_emojiStatusUntil) user.emoji_status).until > (int) (System.currentTimeMillis() / 1000)) { - popupLayout.setExpireDateHint(((TLRPC.TL_emojiStatusUntil) user.emoji_status).until); + if (user != null && DialogObject.getEmojiStatusUntil(user.emoji_status) > 0) { + popupLayout.setExpireDateHint(DialogObject.getEmojiStatusUntil(user.emoji_status)); } popupLayout.setSelected(statusDrawable.getDrawable() instanceof AnimatedEmojiDrawable ? ((AnimatedEmojiDrawable) statusDrawable.getDrawable()).getDocumentId() : null); popupLayout.setSaveState(1); @@ -5629,6 +5641,31 @@ private void updateDialogsHint() { } authHintCell.set(DialogsActivity.this, currentAccount); updateAuthHintCellVisibility(true); + } else if (isPremiumChristmasHintVisible()) { + dialogsHintCellVisible = true; + dialogsHintCell.setVisibility(View.VISIBLE); + dialogsHintCell.setOnClickListener(v -> UserSelectorBottomSheet.open()); + dialogsHintCell.setText( + AndroidUtilities.replaceSingleTag( + LocaleController.getString("BoostingPremiumChristmasTitle", R.string.BoostingPremiumChristmasTitle), + Theme.key_windowBackgroundWhiteValueText, + AndroidUtilities.REPLACING_TAG_TYPE_LINKBOLD, + null + ), + LocaleController.formatString("BoostingPremiumChristmasSubTitle", R.string.BoostingPremiumChristmasSubTitle) + ); + dialogsHintCell.setChristmasStyle(v -> { + MessagesController.getInstance(currentAccount).removeSuggestion(0, "PREMIUM_CHRISTMAS"); + ChangeBounds transition = new ChangeBounds(); + transition.setDuration(200); + TransitionManager.beginDelayedTransition((ViewGroup) dialogsHintCell.getParent(), transition); + updateDialogsHint(); + BulletinFactory.of(this) + .createSimpleBulletin(R.raw.chats_infotip, LocaleController.getString("BoostingPremiumChristmasToast", R.string.BoostingPremiumChristmasToast), 4) + .setDuration(Bulletin.DURATION_PROLONG) + .show(); + }); + updateAuthHintCellVisibility(false); } else if (isPremiumRestoreHintVisible()) { dialogsHintCellVisible = true; dialogsHintCell.setVisibility(View.VISIBLE); @@ -6082,7 +6119,7 @@ public void updateSpeedItem(boolean visibleByPosition) { break; } } - boolean visible = !getUserConfig().isPremium() && !getMessagesController().premiumLocked && visibleByDownload && visibleByPosition && DISPLAY_SPEEDOMETER_IN_DOWNLOADS_SEARCH; + boolean visible = !getUserConfig().isPremium() && !getMessagesController().premiumFeaturesBlocked() && visibleByDownload && visibleByPosition && DISPLAY_SPEEDOMETER_IN_DOWNLOADS_SEARCH; boolean wasVisible = speedItem.getTag() != null; if (visible != wasVisible) { @@ -6211,7 +6248,7 @@ public void onItemClick(int id) { } } else if (onlySelect || folderId != 0) { finishFragment(); - } else if (parentLayout != null) { + } else if (parentLayout != null && parentLayout.getDrawerLayoutContainer() != null) { parentLayout.getDrawerLayoutContainer().openDrawer(false); } } else if (id == 1) { @@ -7385,6 +7422,10 @@ public void setSearchAnimationProgress(float progress, boolean full) { dialogsHintCell.setVisibility(View.INVISIBLE); } else { dialogsHintCell.setVisibility(View.VISIBLE); + ViewParent dialogsHintCellParent = dialogsHintCell.getParent(); + if (dialogsHintCellParent != null) { + dialogsHintCellParent.requestLayout(); + } } } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/EmojiAnimationsOverlay.java b/TMessagesProj/src/main/java/org/telegram/ui/EmojiAnimationsOverlay.java index 5043cd7bd48..b49e629093a 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/EmojiAnimationsOverlay.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/EmojiAnimationsOverlay.java @@ -787,7 +787,7 @@ private void showStickerSetBulletin(TLRPC.TL_messages_stickerSet stickerSet, Mes if (chatActivity == null) { return; } - if (MessagesController.getInstance(currentAccount).premiumLocked || chatActivity.getParentActivity() == null) { + if (MessagesController.getInstance(currentAccount).premiumFeaturesBlocked() || chatActivity.getParentActivity() == null) { return; } StickerSetBulletinLayout layout = new StickerSetBulletinLayout(contentLayout.getContext(), null, StickerSetBulletinLayout.TYPE_EMPTY, messageObject.getDocument(), chatActivity.getResourceProvider()); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/LanguageSelectActivity.java b/TMessagesProj/src/main/java/org/telegram/ui/LanguageSelectActivity.java index c8ac956a0c2..0d4206c3fd8 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/LanguageSelectActivity.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/LanguageSelectActivity.java @@ -212,7 +212,7 @@ protected void onMoveAnimationUpdate(RecyclerView.ViewHolder holder) { } final boolean currentFullValue = getContextValue() || getChatValue(); if (currentFullValue != prevFullValue) { - int start = 1 + (!getMessagesController().premiumLocked ? 1 : 0); + int start = 1 + (!getMessagesController().premiumFeaturesBlocked() ? 1 : 0); TextCheckCell last = null; for (int i = 0; i < listView.getChildCount(); ++i) { View child = listView.getChildAt(i); @@ -239,7 +239,7 @@ protected void onMoveAnimationUpdate(RecyclerView.ViewHolder holder) { } boolean search = listView.getAdapter() == searchListViewAdapter; if (!search) { - position -= (7 - (!(getChatValue() || getContextValue()) ? 1 : 0) - (getMessagesController().premiumLocked ? 1 : 0)); + position -= (7 - (!(getChatValue() || getContextValue()) ? 1 : 0) - (getMessagesController().premiumFeaturesBlocked() ? 1 : 0)); } LocaleController.LocaleInfo localeInfo; if (search) { @@ -304,7 +304,7 @@ protected void onMoveAnimationUpdate(RecyclerView.ViewHolder holder) { } boolean search = listView.getAdapter() == searchListViewAdapter; if (!search) { - position -= (7 - (!(getChatValue() || getContextValue()) ? 1 : 0) - (getMessagesController().premiumLocked ? 1 : 0)); + position -= (7 - (!(getChatValue() || getContextValue()) ? 1 : 0) - (getMessagesController().premiumFeaturesBlocked() ? 1 : 0)); } LocaleController.LocaleInfo localeInfo; if (search) { @@ -541,7 +541,7 @@ public int getItemCount() { if (!unofficialLanguages.isEmpty()) { count += unofficialLanguages.size() + 1; } - return 4 + (getMessagesController().premiumLocked ? 0 : 1) + (getChatValue() || getContextValue() ? 1 : 0) + 1 + count; + return 4 + (getMessagesController().premiumFeaturesBlocked() ? 0 : 1) + (getChatValue() || getContextValue() ? 1 : 0) + 1 + count; } } @@ -594,7 +594,7 @@ public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) { switch (holder.getItemViewType()) { case VIEW_TYPE_LANGUAGE: { if (!search) { - position -= (7 - (!(getChatValue() || getContextValue()) ? 1 : 0) - (getMessagesController().premiumLocked ? 1 : 0)); + position -= (7 - (!(getChatValue() || getContextValue()) ? 1 : 0) - (getMessagesController().premiumFeaturesBlocked() ? 1 : 0)); } TextRadioCell textSettingsCell = (TextRadioCell) holder.itemView; textSettingsCell.updateRTL(); @@ -692,7 +692,7 @@ public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) { case VIEW_TYPE_INFO: { TextInfoPrivacyCell infoCell = (TextInfoPrivacyCell) holder.itemView; infoCell.updateRTL(); - if (position == (!getMessagesController().premiumLocked && (getContextValue() || getChatValue()) ? 4 : 3)) { + if (position == (!getMessagesController().premiumFeaturesBlocked() && (getContextValue() || getChatValue()) ? 4 : 3)) { infoCell.setText(LocaleController.getString("TranslateMessagesInfo1", R.string.TranslateMessagesInfo1)); infoCell.setBackground(Theme.getThemedDrawableByKey(mContext, R.drawable.greydivider_bottom, Theme.key_windowBackgroundGrayShadow)); infoCell.setTopPadding(11); @@ -720,7 +720,7 @@ public int getItemViewType(int i) { } else { if (i-- == 0) return VIEW_TYPE_HEADER; if (i-- == 0) return VIEW_TYPE_SWITCH; - if (!getMessagesController().premiumLocked) { + if (!getMessagesController().premiumFeaturesBlocked()) { if (i-- == 0) return VIEW_TYPE_SWITCH; } if (getChatValue() || getContextValue()) { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/LaunchActivity.java b/TMessagesProj/src/main/java/org/telegram/ui/LaunchActivity.java index 6cf844c3e5a..d6c220e6630 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/LaunchActivity.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/LaunchActivity.java @@ -167,6 +167,7 @@ import org.telegram.ui.Components.Premium.LimitReachedBottomSheet; import org.telegram.ui.Components.Premium.boosts.BoostPagerBottomSheet; import org.telegram.ui.Components.Premium.boosts.GiftInfoBottomSheet; +import org.telegram.ui.Components.Premium.boosts.UserSelectorBottomSheet; import org.telegram.ui.Components.RLottieDrawable; import org.telegram.ui.Components.RLottieImageView; import org.telegram.ui.Components.RecyclerListView; @@ -257,7 +258,7 @@ public class LaunchActivity extends BasePermissionsActivity implements INavigati private List overlayPasscodeViews = new ArrayList<>(); private TermsOfServiceView termsOfServiceView; private BlockingUpdateView blockingUpdateView; - public Dialog visibleDialog; + public final ArrayList visibleDialogs = new ArrayList<>(); private Dialog proxyErrorDialog; private RecyclerListView sideMenu; private SelectAnimatedEmojiDialog.SelectAnimatedEmojiDialogWindow selectAnimatedEmojiDialog; @@ -265,6 +266,15 @@ public class LaunchActivity extends BasePermissionsActivity implements INavigati private FrameLayout sideMenuContainer; private View rippleAbove; private IUpdateLayout updateLayout; + public Dialog getVisibleDialog() { + for (int i = visibleDialogs.size() - 1; i >= 0; --i) { + Dialog dialog = visibleDialogs.get(i); + if (dialog.isShowing()) { + return dialog; + } + } + return null; + } private Dialog localeDialog; private boolean loadingLocaleDialog; @@ -950,12 +960,8 @@ private void showAttachMenuBot(TLRPC.TL_attachMenuBot attachMenuBot, String star BotWebViewSheet webViewSheet = new BotWebViewSheet(this, getLastFragment().getResourceProvider()); webViewSheet.setParentActivity(this); webViewSheet.requestWebView(currentAccount, attachMenuBot.bot_id, attachMenuBot.bot_id, attachMenuBot.short_name, null, BotWebViewSheet.TYPE_SIMPLE_WEB_VIEW_BUTTON, 0, false, null, null, false, startApp, null, BotWebViewSheet.FLAG_FROM_SIDE_MENU); - if (visibleDialog != null) { - visibleDialog.dismiss(); - visibleDialog = null; - } webViewSheet.show(); - visibleDialog = webViewSheet; + visibleDialogs.add(webViewSheet); } @Override @@ -1213,9 +1219,7 @@ protected void onEmojiSelected(View emojiView, Long documentId, TLRPC.Document d } ((DrawerProfileCell) child).setUser(user, drawerLayoutAdapter.isAccountsShown()); } else if (child instanceof DrawerActionCell && drawerLayoutAdapter.getId(sideMenu.getChildAdapterPosition(child)) == 15) { - boolean hasStatus = - user.emoji_status instanceof TLRPC.TL_emojiStatus || - user.emoji_status instanceof TLRPC.TL_emojiStatusUntil && ((TLRPC.TL_emojiStatusUntil) user.emoji_status).until > (int) (System.currentTimeMillis() / 1000); + boolean hasStatus = user != null && DialogObject.getEmojiStatusDocumentId(user.emoji_status) != 0; ((DrawerActionCell) child).updateTextAndIcon( hasStatus ? LocaleController.getString("ChangeEmojiStatus", R.string.ChangeEmojiStatus) : @@ -1233,8 +1237,8 @@ protected void onEmojiSelected(View emojiView, Long documentId, TLRPC.Document d } } }; - if (user != null && user.emoji_status instanceof TLRPC.TL_emojiStatusUntil && ((TLRPC.TL_emojiStatusUntil) user.emoji_status).until > (int) (System.currentTimeMillis() / 1000)) { - popupLayout.setExpireDateHint(((TLRPC.TL_emojiStatusUntil) user.emoji_status).until); + if (user != null) { + popupLayout.setExpireDateHint(DialogObject.getEmojiStatusUntil(user.emoji_status)); } popupLayout.setSelected(scrimDrawable != null && scrimDrawable.getDrawable() instanceof AnimatedEmojiDrawable ? ((AnimatedEmojiDrawable) scrimDrawable.getDrawable()).getDocumentId() : null); popupLayout.setSaveState(2); @@ -1660,6 +1664,9 @@ private boolean handleIntent(Intent intent, boolean isNew, boolean restore, bool if (GiftInfoBottomSheet.handleIntent(intent, progress)) { return true; } + if (UserSelectorBottomSheet.handleIntent(intent, progress)) { + return true; + } if (AndroidUtilities.handleProxyIntent(this, intent)) { return true; } @@ -3851,7 +3858,7 @@ private void runLinkRequest(final int intentAccount, } }), ConnectionsManager.RequestFlagInvokeAfter | ConnectionsManager.RequestFlagFailOnServerErrors); - processWebAppBot(intentAccount, username, group, sticker, emoji, botUser, botChat, botChannel, botChatAdminParams, message, contactToken, folderSlug, hasUrl, messageId, channelId, threadId, commentId, game, auth, lang, unsupportedUrl, code, loginToken, wallPaper, inputInvoiceSlug, theme, voicechat, livestream, state, videoTimestamp, setAsAttachBot, attachMenuBotToOpen, attachMenuBotChoose, botAppMaybe, botAppStartParam, progress, forceNotInternalForApps, storyId, isBoost, user, dismissLoading, botAttachable); + processWebAppBot(intentAccount, username, group, sticker, emoji, botUser, botChat, botChannel, botChatAdminParams, message, contactToken, folderSlug, hasUrl, messageId, channelId, threadId, commentId, game, auth, lang, unsupportedUrl, code, loginToken, wallPaper, inputInvoiceSlug, theme, voicechat, livestream, state, videoTimestamp, setAsAttachBot, attachMenuBotToOpen, attachMenuBotChoose, botAppMaybe, botAppStartParam, progress, forceNotInternalForApps, storyId, isBoost, user, dismissLoading, botAttachable, true); }, null); } else if (attachBot.request_write_access || forceNotInternalForApps) { AtomicBoolean allowWrite = new AtomicBoolean(true); @@ -3869,15 +3876,15 @@ private void runLinkRequest(final int intentAccount, } }), ConnectionsManager.RequestFlagInvokeAfter | ConnectionsManager.RequestFlagFailOnServerErrors); - processWebAppBot(intentAccount, username, group, sticker, emoji, botUser, botChat, botChannel, botChatAdminParams, message, contactToken, folderSlug, hasUrl, messageId, channelId, threadId, commentId, game, auth, lang, unsupportedUrl, code, loginToken, wallPaper, inputInvoiceSlug, theme, voicechat, livestream, state, videoTimestamp, setAsAttachBot, attachMenuBotToOpen, attachMenuBotChoose, botAppMaybe, botAppStartParam, progress, forceNotInternalForApps, storyId, isBoost, user, dismissLoading, false); + processWebAppBot(intentAccount, username, group, sticker, emoji, botUser, botChat, botChannel, botChatAdminParams, message, contactToken, folderSlug, hasUrl, messageId, channelId, threadId, commentId, game, auth, lang, unsupportedUrl, code, loginToken, wallPaper, inputInvoiceSlug, theme, voicechat, livestream, state, videoTimestamp, setAsAttachBot, attachMenuBotToOpen, attachMenuBotChoose, botAppMaybe, botAppStartParam, progress, forceNotInternalForApps, storyId, isBoost, user, dismissLoading, false, false); }); } else { - processWebAppBot(intentAccount, username, group, sticker, emoji, botUser, botChat, botChannel, botChatAdminParams, message, contactToken, folderSlug, hasUrl, messageId, channelId, threadId, commentId, game, auth, lang, unsupportedUrl, code, loginToken, wallPaper, inputInvoiceSlug, theme, voicechat, livestream, state, videoTimestamp, setAsAttachBot, attachMenuBotToOpen, attachMenuBotChoose, botAppMaybe, botAppStartParam, progress, forceNotInternalForApps, storyId, isBoost, user, dismissLoading, false); + processWebAppBot(intentAccount, username, group, sticker, emoji, botUser, botChat, botChannel, botChatAdminParams, message, contactToken, folderSlug, hasUrl, messageId, channelId, threadId, commentId, game, auth, lang, unsupportedUrl, code, loginToken, wallPaper, inputInvoiceSlug, theme, voicechat, livestream, state, videoTimestamp, setAsAttachBot, attachMenuBotToOpen, attachMenuBotChoose, botAppMaybe, botAppStartParam, progress, forceNotInternalForApps, storyId, isBoost, user, dismissLoading, false, false); } } })); } else { - processWebAppBot(intentAccount, username, group, sticker, emoji, botUser, botChat, botChannel, botChatAdminParams, message, contactToken, folderSlug, hasUrl, messageId, channelId, threadId, commentId, game, auth, lang, unsupportedUrl, code, loginToken, wallPaper, inputInvoiceSlug, theme, voicechat, livestream, state, videoTimestamp, setAsAttachBot, attachMenuBotToOpen, attachMenuBotChoose, botAppMaybe, botAppStartParam, progress, forceNotInternalForApps, storyId, isBoost, user, dismissLoading, false); + processWebAppBot(intentAccount, username, group, sticker, emoji, botUser, botChat, botChannel, botChatAdminParams, message, contactToken, folderSlug, hasUrl, messageId, channelId, threadId, commentId, game, auth, lang, unsupportedUrl, code, loginToken, wallPaper, inputInvoiceSlug, theme, voicechat, livestream, state, videoTimestamp, setAsAttachBot, attachMenuBotToOpen, attachMenuBotChoose, botAppMaybe, botAppStartParam, progress, forceNotInternalForApps, storyId, isBoost, user, dismissLoading, false, false); } return; } @@ -4855,7 +4862,7 @@ private void processWebAppBot(final int intentAccount, final int storyId, final boolean isBoost, TLRPC.User user, - Runnable dismissLoading, boolean botAttachable) { + Runnable dismissLoading, boolean botAttachable, boolean ignoreInactive) { TLRPC.TL_messages_getBotApp getBotApp = new TLRPC.TL_messages_getBotApp(); TLRPC.TL_inputBotAppShortName app = new TLRPC.TL_inputBotAppShortName(); @@ -4879,18 +4886,16 @@ private void processWebAppBot(final int intentAccount, BotWebViewSheet sheet = new BotWebViewSheet(LaunchActivity.this, lastFragment.getResourceProvider()); sheet.setParentActivity(LaunchActivity.this); sheet.requestWebView(intentAccount, user.id, user.id, null, null, BotWebViewSheet.TYPE_WEB_VIEW_BOT_APP, 0, false, lastFragment, botApp.app, allowWrite.get(), botAppStartParam, user); - if (visibleDialog != null) { - visibleDialog.dismiss(); - visibleDialog = null; - } sheet.show(); - visibleDialog = sheet; + visibleDialogs.add(sheet); if (botApp.inactive || forceNotInternalForApps) { sheet.showJustAddedBulletin(); } }; - if (botApp.inactive && botAttachable) { + if (ignoreInactive) { + loadBotSheet.run(); + } else if (botApp.inactive && botAttachable) { WebAppDisclaimerAlert.show(this, (allowSendMessage) -> { loadBotSheet.run(); }, null); @@ -5024,10 +5029,13 @@ private void processAttachMenuBot(int intentAccount, long peerId, String attachM if (lastFragment != null) { lastFragment.dismissCurrentDialog(); } - if (visibleDialog != null) { - visibleDialog.dismiss(); - visibleDialog = null; + for (int i = 0; i < visibleDialogs.size(); ++i) { + Dialog dialog = visibleDialogs.get(i); + if (dialog.isShowing()) { + visibleDialogs.get(i).dismiss(); + } } + visibleDialogs.clear(); presentFragment(dialogsActivity); } else if (lastFragment instanceof ChatActivity) { ChatActivity chatActivity = (ChatActivity) lastFragment; @@ -5058,10 +5066,13 @@ private void processAttachMenuBot(int intentAccount, long peerId, String attachM if (lastFragment != null) { lastFragment.dismissCurrentDialog(); } - if (visibleDialog != null) { - visibleDialog.dismiss(); - visibleDialog = null; + for (int i = 0; i < visibleDialogs.size(); ++i) { + Dialog dialog = visibleDialogs.get(i); + if (dialog.isShowing()) { + visibleDialogs.get(i).dismiss(); + } } + visibleDialogs.clear(); presentFragment(dialogsActivity); } else if (lastFragment instanceof ChatActivity) { ((ChatActivity) lastFragment).openAttachBotLayout(user.id, setAsAttachBot, true); @@ -5249,19 +5260,11 @@ public void checkAppUpdate(boolean force) { public Dialog showAlertDialog(AlertDialog.Builder builder) { try { - if (visibleDialog != null) { - visibleDialog.dismiss(); - visibleDialog = null; - } - } catch (Exception e) { - FileLog.e(e); - } - try { - visibleDialog = builder.show(); - visibleDialog.setCanceledOnTouchOutside(true); - visibleDialog.setOnDismissListener(dialog -> { - if (visibleDialog != null) { - if (visibleDialog == localeDialog) { + AlertDialog dialog = builder.show(); + dialog.setCanceledOnTouchOutside(true); + dialog.setOnDismissListener(d -> { + if (dialog != null) { + if (dialog == localeDialog) { BaseFragment fragment = actionBarLayout == null ? null : actionBarLayout.getLastFragment(); try { String shorname = LocaleController.getInstance().getCurrentLocaleInfo().shortName; @@ -5280,7 +5283,7 @@ public Dialog showAlertDialog(AlertDialog.Builder builder) { FileLog.e(e); } localeDialog = null; - } else if (visibleDialog == proxyErrorDialog) { + } else if (dialog == proxyErrorDialog) { SharedPreferences preferences = MessagesController.getGlobalMainSettings(); SharedPreferences.Editor editor = MessagesController.getGlobalMainSettings().edit(); editor.putBoolean("proxy_enabled", false); @@ -5291,9 +5294,10 @@ public Dialog showAlertDialog(AlertDialog.Builder builder) { proxyErrorDialog = null; } } - visibleDialog = null; + visibleDialogs.remove(dialog); }); - return visibleDialog; + visibleDialogs.add(dialog); + return dialog; } catch (Exception e) { FileLog.e(e); } @@ -5865,10 +5869,13 @@ protected void onDestroy() { editorView.destroy(); } try { - if (visibleDialog != null) { - visibleDialog.dismiss(); - visibleDialog = null; + for (int i = 0; i < visibleDialogs.size(); ++i) { + Dialog dialog = visibleDialogs.get(i); + if (dialog.isShowing()) { + visibleDialogs.get(i).dismiss(); + } } + visibleDialogs.clear(); } catch (Exception e) { FileLog.e(e); } @@ -6835,10 +6842,13 @@ private void showLanguageAlertInternal(LocaleController.LocaleInfo systemInfo, L localeDialog = null; drawerLayoutContainer.closeDrawer(true); presentFragment(new LanguageSelectActivity()); - if (visibleDialog != null) { - visibleDialog.dismiss(); - visibleDialog = null; + for (int i = 0; i < visibleDialogs.size(); ++i) { + Dialog dialog = visibleDialogs.get(i); + if (dialog.isShowing()) { + visibleDialogs.get(i).dismiss(); + } } + visibleDialogs.clear(); }); linearLayout.addView(cell, LayoutHelper.createLinear(LayoutHelper.MATCH_PARENT, 50)); builder.setView(linearLayout); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/LauncherIconController.java b/TMessagesProj/src/main/java/org/telegram/ui/LauncherIconController.java index 6451e336e31..49bf821535d 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/LauncherIconController.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/LauncherIconController.java @@ -39,7 +39,7 @@ public enum LauncherIcon { AQUA("AquaIcon", R.drawable.icon_4_background_sa, R.mipmap.icon_foreground_sa, R.string.AppIconAqua), PREMIUM("PremiumIcon", R.drawable.icon_3_background_sa, R.mipmap.icon_3_foreground_sa, R.string.AppIconPremium, true), TURBO("TurboIcon", R.drawable.icon_5_background_sa, R.mipmap.icon_5_foreground_sa, R.string.AppIconTurbo, true), - NOX("NoxIcon", R.drawable.icon_2_background_sa, R.mipmap.icon_foreground_sa, R.string.AppIconNox, true); + NOX("NoxIcon", R.mipmap.icon_2_background_sa, R.mipmap.icon_foreground_sa, R.string.AppIconNox, true); public final String key; public final int background; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/LiteModeSettingsActivity.java b/TMessagesProj/src/main/java/org/telegram/ui/LiteModeSettingsActivity.java index 1764383f934..2dca64a3ad1 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/LiteModeSettingsActivity.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/LiteModeSettingsActivity.java @@ -65,6 +65,7 @@ import org.telegram.ui.Components.SeekBarAccessibilityDelegate; import org.telegram.ui.Components.SeekBarView; import org.telegram.ui.Components.Switch; +import org.telegram.ui.Components.ThanosEffect; import java.util.ArrayList; @@ -257,6 +258,9 @@ private void updateItems() { items.add(Item.asCheckbox(LocaleController.getString("LiteOptionsBlur"), LiteMode.FLAG_CHAT_BLUR)); } items.add(Item.asCheckbox(LocaleController.getString("LiteOptionsScale"), LiteMode.FLAG_CHAT_SCALE)); + if (ThanosEffect.supports()) { + items.add(Item.asCheckbox(LocaleController.getString("LiteOptionsThanos"), LiteMode.FLAG_CHAT_THANOS)); + } } items.add(Item.asSwitch(R.drawable.msg2_call_earpiece, LocaleController.getString("LiteOptionsCalls"), LiteMode.FLAG_CALLS_ANIMATIONS)); items.add(Item.asSwitch(R.drawable.msg2_videocall, LocaleController.getString("LiteOptionsAutoplayVideo"), LiteMode.FLAG_AUTOPLAY_VIDEOS)); @@ -626,6 +630,9 @@ private int preprocessFlagsCount(int flags) { if (SharedConfig.getDevicePerformanceClass() < SharedConfig.PERFORMANCE_CLASS_AVERAGE && (flags & LiteMode.FLAG_CHAT_BLUR) > 0) { count--; } + if (!ThanosEffect.supports() && (flags & LiteMode.FLAG_CHAT_THANOS) > 0) { + count--; + } return count; } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/MessageStatisticActivity.java b/TMessagesProj/src/main/java/org/telegram/ui/MessageStatisticActivity.java index ffd77d90794..f098e8653e9 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/MessageStatisticActivity.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/MessageStatisticActivity.java @@ -123,7 +123,6 @@ public class MessageStatisticActivity extends BaseFragment implements Notificati private RLottieImageView imageView; private LinearLayout progressLayout; - private int nextRate; private int publicChats; private boolean endReached; @@ -659,31 +658,38 @@ private void loadChats(int count) { req.msg_id = messageObject.getId(); req.channel = getMessagesController().getInputChannel(-messageObject.getDialogId()); } - if (!messages.isEmpty()) { - TLRPC.Message message = messages.get(messages.size() - 1).messageOwner; - req.offset_id = message.id; - req.offset_peer = getMessagesController().getInputPeer(MessageObject.getDialogId(message)); - req.offset_rate = nextRate; - } else { - req.offset_peer = new TLRPC.TL_inputPeerEmpty(); - } + req.offset = nextOffset == null ? "" : nextOffset; int reqId = getConnectionsManager().sendRequest(req, (response, error) -> AndroidUtilities.runOnUIThread(() -> { if (error == null) { - TLRPC.messages_Messages res = (TLRPC.messages_Messages) response; + TLRPC.TL_stats_publicForwards res = (TLRPC.TL_stats_publicForwards) response; if ((res.flags & 1) != 0) { - nextRate = res.next_rate; + nextOffset = res.next_offset; + } else { + nextOffset = null; } if (res.count != 0) { publicChats = res.count; } else if (publicChats == 0) { - publicChats = res.messages.size(); + publicChats = res.forwards.size(); } - endReached = !(res instanceof TLRPC.TL_messages_messagesSlice); + endReached = nextOffset == null; getMessagesController().putChats(res.chats, false); getMessagesController().putUsers(res.users, false); - for (int i = 0; i < res.messages.size(); i++) { - messages.add(new MessageObject(currentAccount, res.messages.get(i), false, true)); + + for (TLRPC.PublicForward forward : res.forwards) { + if (forward instanceof TL_stories.TL_publicForwardStory) { + TL_stories.TL_publicForwardStory forwardStory = (TL_stories.TL_publicForwardStory) forward; + forwardStory.story.dialogId = DialogObject.getPeerDialogId(forwardStory.peer); + forwardStory.story.messageId = forwardStory.story.id; + MessageObject msg = new MessageObject(currentAccount, forwardStory.story); + msg.generateThumbs(false); + messages.add(msg); + } else if (forward instanceof TLRPC.TL_publicForwardMessage) { + TLRPC.TL_publicForwardMessage forwardMessage = (TLRPC.TL_publicForwardMessage) forward; + messages.add(new MessageObject(currentAccount, forwardMessage.message, false, true)); + } } + if (emptyView != null) { emptyView.showTextView(); } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/MultiContactsSelectorBottomSheet.java b/TMessagesProj/src/main/java/org/telegram/ui/MultiContactsSelectorBottomSheet.java new file mode 100644 index 00000000000..9bcf578729b --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/MultiContactsSelectorBottomSheet.java @@ -0,0 +1,497 @@ +package org.telegram.ui; + +import static org.telegram.messenger.AndroidUtilities.dp; + +import android.annotation.SuppressLint; +import android.content.res.Configuration; +import android.graphics.Canvas; +import android.graphics.Paint; +import android.graphics.Rect; +import android.text.SpannableStringBuilder; +import android.text.Spanned; +import android.text.TextUtils; +import android.text.style.ReplacementSpan; +import android.view.Gravity; +import android.view.HapticFeedbackConstants; +import android.view.View; +import android.widget.LinearLayout; + +import androidx.annotation.NonNull; +import androidx.annotation.Nullable; +import androidx.recyclerview.widget.DefaultItemAnimator; +import androidx.recyclerview.widget.LinearSmoothScrollerCustom; +import androidx.recyclerview.widget.RecyclerView; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.ContactsController; +import org.telegram.messenger.LocaleController; +import org.telegram.messenger.MediaDataController; +import org.telegram.messenger.MessagesController; +import org.telegram.messenger.R; +import org.telegram.messenger.UserConfig; +import org.telegram.messenger.UserObject; +import org.telegram.tgnet.TLRPC; +import org.telegram.ui.ActionBar.BaseFragment; +import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.Components.BottomSheetWithRecyclerListView; +import org.telegram.ui.Components.BulletinFactory; +import org.telegram.ui.Components.CubicBezierInterpolator; +import org.telegram.ui.Components.LayoutHelper; +import org.telegram.ui.Components.Premium.boosts.BoostRepository; +import org.telegram.ui.Components.Premium.boosts.adapters.SelectorAdapter; +import org.telegram.ui.Components.Premium.boosts.cells.selector.SelectorBtnCell; +import org.telegram.ui.Components.Premium.boosts.cells.selector.SelectorHeaderCell; +import org.telegram.ui.Components.Premium.boosts.cells.selector.SelectorSearchCell; +import org.telegram.ui.Components.Premium.boosts.cells.selector.SelectorUserCell; +import org.telegram.ui.Components.RecyclerListView; +import org.telegram.ui.Stories.recorder.ButtonWithCounterView; + +import java.util.ArrayList; +import java.util.HashMap; +import java.util.HashSet; +import java.util.LinkedHashMap; +import java.util.List; +import java.util.Map; + +public class MultiContactsSelectorBottomSheet extends BottomSheetWithRecyclerListView { + private static MultiContactsSelectorBottomSheet instance; + + public interface SelectorListener { + void onUserSelected(List ids); + } + + public static void open(int maxCount, SelectorListener selectorListener) { + BaseFragment fragment = LaunchActivity.getLastFragment(); + if (fragment == null) { + return; + } + if (instance != null) { + return; + } + MultiContactsSelectorBottomSheet sheet = new MultiContactsSelectorBottomSheet(fragment, true, maxCount, selectorListener); + sheet.show(); + instance = sheet; + } + + private static final int BOTTOM_HEIGHT_DP = 60; + + private final ButtonWithCounterView actionButton; + private final SelectorSearchCell searchField; + private final View sectionCell; + private final SelectorHeaderCell headerView; + private final SelectorBtnCell buttonContainer; + + private final ArrayList oldItems = new ArrayList<>(); + private final ArrayList items = new ArrayList<>(); + private final HashSet selectedIds = new HashSet<>(); + private final List contacts = new ArrayList<>(); + private final List hints = new ArrayList<>(); + private final List foundedUsers = new ArrayList<>(); + private final Map> contactsMap = new HashMap<>(); + private final List contactsLetters = new ArrayList<>(); + private final HashMap allSelectedObjects = new LinkedHashMap<>(); + private String query; + private SelectorAdapter selectorAdapter; + private int listPaddingTop = AndroidUtilities.dp(56 + 64); + private int lastRequestId; + private float recipientsBtnExtraSpace; + private ReplacementSpan recipientsBtnSpaceSpan; + private int maxCount; + private SelectorListener selectorListener; + + private final Runnable remoteSearchRunnable = new Runnable() { + @Override + public void run() { + final String finalQuery = query; + if (finalQuery != null) { + loadData(finalQuery); + } + } + }; + + private void loadData(String query) { + lastRequestId = BoostRepository.searchContacts(lastRequestId, query, arg -> { + foundedUsers.clear(); + foundedUsers.addAll(arg); + updateList(true, true); + }); + } + + private void createRecipientsBtnSpaceSpan() { + recipientsBtnSpaceSpan = new ReplacementSpan() { + @Override + public int getSize(@NonNull Paint paint, CharSequence text, int start, int end, @Nullable Paint.FontMetricsInt fm) { + return (int) recipientsBtnExtraSpace; + } + + @Override + public void draw(@NonNull Canvas canvas, CharSequence text, int start, int end, float x, int top, int y, int bottom, @NonNull Paint paint) { + + } + }; + } + + public MultiContactsSelectorBottomSheet(BaseFragment fragment, boolean needFocus, int maxCount, SelectorListener selectorListener) { + super(fragment, needFocus, false, false, fragment.getResourceProvider()); + this.maxCount = maxCount; + this.selectorListener = selectorListener; + headerView = new SelectorHeaderCell(getContext(), resourcesProvider) { + @Override + protected int getHeaderHeight() { + if (getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE) { + return dp(48); + } else { + return dp(54); + } + } + }; + headerView.setOnCloseClickListener(this::dismiss); + headerView.setText(getTitle()); + headerView.setCloseImageVisible(false); + headerView.backDrawable.setRotation(0f, false); + + createRecipientsBtnSpaceSpan(); + + searchField = new SelectorSearchCell(getContext(), resourcesProvider, null) { + private boolean isKeyboardVisible; + + @Override + protected void onLayout(boolean changed, int l, int t, int r, int b) { + super.onLayout(changed, l, t, r, b); + listPaddingTop = getMeasuredHeight() + dp(64); + selectorAdapter.notifyChangedLast(); + if (isKeyboardVisible != isKeyboardVisible()) { + isKeyboardVisible = isKeyboardVisible(); + if (isKeyboardVisible) { + scrollToTop(true); + } + } + } + }; + searchField.setBackgroundColor(getThemedColor(Theme.key_dialogBackground)); + searchField.setOnSearchTextChange(this::onSearch); + searchField.setHintText(LocaleController.getString("Search", R.string.Search), false); + + sectionCell = new View(getContext()) { + @Override + protected void onDraw(Canvas canvas) { + canvas.drawColor(getThemedColor(Theme.key_graySection)); + } + }; + + containerView.addView(headerView, 0, LayoutHelper.createFrameMarginPx(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.TOP | Gravity.FILL_HORIZONTAL, backgroundPaddingLeft, 0, backgroundPaddingLeft, 0)); + containerView.addView(searchField, LayoutHelper.createFrameMarginPx(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.TOP | Gravity.FILL_HORIZONTAL, backgroundPaddingLeft, 0, backgroundPaddingLeft, 0)); + containerView.addView(sectionCell, LayoutHelper.createFrameMarginPx(LayoutHelper.MATCH_PARENT, 1, Gravity.TOP | Gravity.FILL_HORIZONTAL, backgroundPaddingLeft, 0, backgroundPaddingLeft, 0)); + + buttonContainer = new SelectorBtnCell(getContext(), resourcesProvider, null); + buttonContainer.setClickable(true); + buttonContainer.setOrientation(LinearLayout.VERTICAL); + buttonContainer.setPadding(dp(10), dp(10), dp(10), dp(10)); + buttonContainer.setBackgroundColor(Theme.getColor(Theme.key_dialogBackground, resourcesProvider)); + actionButton = new ButtonWithCounterView(getContext(), resourcesProvider) { + @Override + protected float calculateCounterWidth(float width, float percent) { + boolean needUpdateActionBtn = recipientsBtnExtraSpace == 0; + recipientsBtnExtraSpace = width; + if (needUpdateActionBtn) { + createRecipientsBtnSpaceSpan(); + updateActionButton(false); + } + return width; + } + }; + actionButton.setOnClickListener(v -> next()); + buttonContainer.addView(actionButton, LayoutHelper.createLinear(LayoutHelper.MATCH_PARENT, 48, Gravity.BOTTOM | Gravity.FILL_HORIZONTAL)); + containerView.addView(buttonContainer, LayoutHelper.createFrameMarginPx(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.BOTTOM | Gravity.FILL_HORIZONTAL, backgroundPaddingLeft, 0, backgroundPaddingLeft, 0)); + + selectorAdapter.setData(items, recyclerListView); + recyclerListView.setPadding(backgroundPaddingLeft, 0, backgroundPaddingLeft, dp(BOTTOM_HEIGHT_DP)); + recyclerListView.addOnScrollListener(new RecyclerView.OnScrollListener() { + @Override + public void onScrollStateChanged(@NonNull RecyclerView recyclerView, int newState) { + if (newState == RecyclerView.SCROLL_STATE_DRAGGING) { + AndroidUtilities.hideKeyboard(searchField.getEditText()); + } + } + }); + recyclerListView.setOnItemClickListener((view, position, x, y) -> { + if (view instanceof SelectorUserCell) { + TLRPC.User user = ((SelectorUserCell) view).getUser(); + long id = user.id; + if (selectedIds.contains(id)) { + selectedIds.remove(id); + } else { + selectedIds.add(id); + allSelectedObjects.put(id, user); + } + if (selectedIds.size() == maxCount + 1) { + selectedIds.remove(id); + showMaximumUsersToast(); + return; + } + searchField.updateSpans(true, selectedIds, () -> { + updateList(true, false); + }, null); + updateList(true, false); + clearSearchAfterSelect(); + } + }); + DefaultItemAnimator itemAnimator = new DefaultItemAnimator(); + itemAnimator.setDurations(350); + itemAnimator.setInterpolator(CubicBezierInterpolator.EASE_OUT_QUINT); + itemAnimator.setDelayAnimations(false); + itemAnimator.setSupportsChangeAnimations(false); + recyclerListView.setItemAnimator(itemAnimator); + recyclerListView.addItemDecoration(new RecyclerView.ItemDecoration() { + @Override + public void getItemOffsets(@NonNull Rect outRect, @NonNull View view, @NonNull RecyclerView parent, @NonNull RecyclerView.State state) { + super.getItemOffsets(outRect, view, parent, state); + int position = parent.getChildAdapterPosition(view); + if (position == items.size()) { + outRect.bottom = listPaddingTop; + } + } + }); + + searchField.setText(""); + searchField.spansContainer.removeAllSpans(false); + searchField.updateSpans(false, selectedIds, () -> { + updateList(true, false); + }, null); + headerView.setText(getTitle()); + updateActionButton(false); + + contacts.addAll(ContactsController.getInstance(currentAccount).contacts); + contactsMap.putAll(ContactsController.getInstance(currentAccount).usersSectionsDict); + contactsLetters.addAll(ContactsController.getInstance(currentAccount).sortedUsersSectionsArray); + hints.addAll(MediaDataController.getInstance(currentAccount).hints); + updateList(false, true); + fixNavigationBar(); + } + + @Override + protected void onPreDraw(Canvas canvas, int top, float progressToFullView) { + float minTop = AndroidUtilities.statusBarHeight + (headerView.getMeasuredHeight() - AndroidUtilities.statusBarHeight - AndroidUtilities.dp(40)) / 2f; + float fromY = Math.max(top, minTop) + AndroidUtilities.dp(8); + headerView.setTranslationY(fromY); + searchField.setTranslationY(headerView.getTranslationY() + headerView.getMeasuredHeight()); + sectionCell.setTranslationY(searchField.getTranslationY() + searchField.getMeasuredHeight()); + recyclerListView.setTranslationY(headerView.getMeasuredHeight() + searchField.getMeasuredHeight() + sectionCell.getMeasuredHeight() - AndroidUtilities.dp(8)); + } + + private void next() { + if (selectedIds.size() == 0 || selectorListener == null) { + return; + } + List selectedUsers = new ArrayList<>(); + for (TLRPC.User object : allSelectedObjects.values()) { + if (selectedIds.contains(object.id)) { + selectedUsers.add(object.id); + } + } + selectorListener.onUserSelected(selectedUsers); + dismiss(); + } + + public void scrollToTop(boolean animate) { + if (animate) { + LinearSmoothScrollerCustom linearSmoothScroller = new LinearSmoothScrollerCustom(getContext(), LinearSmoothScrollerCustom.POSITION_TOP, .6f); + linearSmoothScroller.setTargetPosition(1); + linearSmoothScroller.setOffset(AndroidUtilities.dp(36)); + recyclerListView.getLayoutManager().startSmoothScroll(linearSmoothScroller); + } else { + recyclerListView.scrollToPosition(0); + } + } + + @Override + public void dismissInternal() { + super.dismissInternal(); + instance = null; + AndroidUtilities.cancelRunOnUIThread(remoteSearchRunnable); + } + + private void showMaximumUsersToast() { + String text = LocaleController.formatPluralString("BotMultiContactsSelectorLimit", maxCount); + BulletinFactory.of(container, resourcesProvider).createSimpleBulletin(R.raw.chats_infotip, text).show(true); + try { + container.performHapticFeedback(HapticFeedbackConstants.KEYBOARD_TAP, HapticFeedbackConstants.FLAG_IGNORE_GLOBAL_SETTING); + } catch (Exception ignore) { + } + } + + private void updateList(boolean animated, boolean notify) { + updateItems(animated, notify); + updateCheckboxes(animated); + updateActionButton(animated); + } + + private void updateCheckboxes(boolean animated) { + int visibleItemsFrom = -1; + int visibleItemsTo = 0; + for (int i = 0; i < recyclerListView.getChildCount(); ++i) { + View child = recyclerListView.getChildAt(i); + if (child instanceof SelectorUserCell) { + int position = recyclerListView.getChildAdapterPosition(child); + if (position <= 0) { + continue; + } + if (visibleItemsFrom == -1) { + visibleItemsFrom = position; + } + visibleItemsTo = position; + SelectorAdapter.Item item = items.get(position - 1); + SelectorUserCell cell = (SelectorUserCell) child; + cell.setChecked(item.checked, animated); + if (item.chat != null) { + cell.setCheckboxAlpha(selectorAdapter.getParticipantsCount(item.chat) > 200 ? .3f : 1f, animated); + } else { + cell.setCheckboxAlpha(1f, animated); + } + } + } + if (animated) { + selectorAdapter.notifyItemRangeChanged(0, visibleItemsFrom); + selectorAdapter.notifyItemRangeChanged(visibleItemsTo, selectorAdapter.getItemCount() - visibleItemsTo); + } + } + + private void updateActionButton(boolean animated) { + actionButton.setShowZero(false); + SpannableStringBuilder stringBuilder = new SpannableStringBuilder(); + if (selectedIds.size() == 0) { + stringBuilder.append("d").setSpan(recipientsBtnSpaceSpan, 0, 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + stringBuilder.append(LocaleController.getString("ChooseUsers", R.string.ChooseUsers)); + } else { + stringBuilder.append(LocaleController.getString("GiftPremiumProceedBtn", R.string.GiftPremiumProceedBtn)); + } + actionButton.setCount(selectedIds.size(), true); + actionButton.setText(stringBuilder, animated, false); + actionButton.setEnabled(true); + } + + private void onSearch(String text) { + this.query = text; + AndroidUtilities.cancelRunOnUIThread(remoteSearchRunnable); + AndroidUtilities.runOnUIThread(remoteSearchRunnable, 350); + } + + private void clearSearchAfterSelect() { + if (isSearching()) { + query = null; + searchField.setText(""); + AndroidUtilities.cancelRunOnUIThread(remoteSearchRunnable); + updateItems(true, true); + } + } + + private void updateSectionCell(boolean animated) { + if (selectedIds == null) { + return; + } + if (selectedIds.size() > 0) { + selectorAdapter.setTopSectionClickListener(v -> { + selectedIds.clear(); + searchField.spansContainer.removeAllSpans(true); + updateList(true, false); + }); + } else { + selectorAdapter.setTopSectionClickListener(null); + } + } + + private boolean isSearching() { + return !TextUtils.isEmpty(query); + } + + @SuppressLint("NotifyDataSetChanged") + public void updateItems(boolean animated, boolean notify) { + oldItems.clear(); + oldItems.addAll(items); + items.clear(); + + int h = 0; + if (isSearching()) { + for (TLRPC.User foundedUser : foundedUsers) { + h += dp(56); + items.add(SelectorAdapter.Item.asUser(foundedUser, selectedIds.contains(foundedUser.id))); + } + } else { + if (!hints.isEmpty()) { + List userItems = new ArrayList<>(); + for (TLRPC.TL_topPeer hint : hints) { + TLRPC.User user = MessagesController.getInstance(currentAccount).getUser(hint.peer.user_id); + if (user.self || user.bot || UserObject.isService(user.id) || UserObject.isDeleted(user)) { + continue; + } + h += dp(56); + userItems.add(SelectorAdapter.Item.asUser(user, selectedIds.contains(user.id))); + } + if (!userItems.isEmpty()) { + h += dp(32); + items.add(SelectorAdapter.Item.asTopSection(LocaleController.getString("GiftPremiumFrequentContacts", R.string.GiftPremiumFrequentContacts))); + items.addAll(userItems); + } + } + for (String contactLetter : contactsLetters) { + List userItems = new ArrayList<>(); + for (TLRPC.TL_contact contact : contactsMap.get(contactLetter)) { + long myUid = UserConfig.getInstance(currentAccount).getClientUserId(); + if (contact.user_id == myUid) { + continue; + } + h += dp(56); + TLRPC.User user = MessagesController.getInstance(currentAccount).getUser(contact.user_id); + userItems.add(SelectorAdapter.Item.asUser(user, selectedIds.contains(user.id))); + } + + if (!userItems.isEmpty()) { + h += dp(32); + items.add(SelectorAdapter.Item.asLetter(contactLetter.toUpperCase())); + items.addAll(userItems); + } + } + } + + if (items.isEmpty()) { + items.add(SelectorAdapter.Item.asNoUsers()); + h += dp(150); + } + int minHeight = (int) (AndroidUtilities.displaySize.y * 0.6f); + items.add(SelectorAdapter.Item.asPad(Math.max(0, minHeight - h))); + + updateSectionCell(animated); + + if (notify && selectorAdapter != null) { + if (animated) { + selectorAdapter.setItems(oldItems, items); + } else { + selectorAdapter.notifyDataSetChanged(); + } + } + } + + @Override + public void onConfigurationChanged(Configuration newConfig) { + super.onConfigurationChanged(newConfig); + updateItems(false, true); + } + + @Override + protected CharSequence getTitle() { + return LocaleController.getString("ChooseUsers", R.string.ChooseUsers); + } + + @Override + protected RecyclerListView.SelectionAdapter createAdapter() { + selectorAdapter = new SelectorAdapter(getContext(), resourcesProvider); + selectorAdapter.setGreenSelector(true); + return selectorAdapter; + } + + @Override + public void dismiss() { + AndroidUtilities.hideKeyboard(searchField.getEditText()); + super.dismiss(); + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/PaymentFormActivity.java b/TMessagesProj/src/main/java/org/telegram/ui/PaymentFormActivity.java index 930f861f86e..ccc48d02bf1 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/PaymentFormActivity.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/PaymentFormActivity.java @@ -253,6 +253,7 @@ public class PaymentFormActivity extends BaseFragment implements NotificationCen private boolean loadingPasswordInfo; private PaymentFormActivity passwordFragment; + private String overrideSmartGlocalConnectionUrl; private boolean need_card_country; private boolean need_card_postcode; private boolean need_card_name; @@ -1764,7 +1765,7 @@ protected void onDraw(Canvas canvas) { }); if (a == FIELD_SAVEDPASSWORD) { bottomCell[0] = new TextInfoPrivacyCell(context, resourcesProvider); - bottomCell[0].setText(LocaleController.formatString("PaymentConfirmationMessage", R.string.PaymentConfirmationMessage, savedCredentialsCard.title)); + bottomCell[0].setText(LocaleController.formatString("PaymentConfirmationMessage", R.string.PaymentConfirmationMessage, savedCredentialsCard == null ? "" : savedCredentialsCard.title)); bottomCell[0].setBackgroundDrawable(Theme.getThemedDrawableByKey(context, R.drawable.greydivider, Theme.key_windowBackgroundGrayShadow)); linearLayout2.addView(bottomCell[0], LayoutHelper.createLinear(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT)); @@ -3775,8 +3776,23 @@ protected String doInBackground(Object... objects) { cardObject.put("security_code", "" + card.getCVC()); jsonObject.put("card", cardObject); + String overrideSmartGlocalConnectionUrl = null; + if (paymentForm.native_params != null) { + try { + JSONObject jsonObject2 = new JSONObject(paymentForm.native_params.data); + overrideSmartGlocalConnectionUrl = jsonObject2.getString("tokenize_url"); + if (overrideSmartGlocalConnectionUrl != null && ( + !overrideSmartGlocalConnectionUrl.startsWith("https://") || + overrideSmartGlocalConnectionUrl.endsWith(".smart-glocal.com/cds/v1/tokenize/card") + )) { + overrideSmartGlocalConnectionUrl = null; + } + } catch (Exception e) {} + } URL connectionUrl; - if (paymentForm.invoice.test) { + if (overrideSmartGlocalConnectionUrl != null) { + connectionUrl = new URL(overrideSmartGlocalConnectionUrl); + } else if (paymentForm.invoice.test) { connectionUrl = new URL("https://tgb-playground.smart-glocal.com/cds/v1/tokenize/card"); } else { connectionUrl = new URL("https://tgb.smart-glocal.com/cds/v1/tokenize/card"); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/PeerColorActivity.java b/TMessagesProj/src/main/java/org/telegram/ui/PeerColorActivity.java index 01868fc121f..bdbb25a602d 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/PeerColorActivity.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/PeerColorActivity.java @@ -2,7 +2,6 @@ import static org.telegram.messenger.AndroidUtilities.dp; import static org.telegram.messenger.AndroidUtilities.dpf2; -import static org.telegram.ui.Components.Premium.LimitReachedBottomSheet.TYPE_BOOSTS_FOR_COLOR; import android.animation.Animator; import android.animation.AnimatorListenerAdapter; @@ -29,8 +28,11 @@ import android.graphics.drawable.Drawable; import android.os.Build; import android.os.Bundle; +import android.text.SpannableString; import android.text.SpannableStringBuilder; import android.text.Spanned; +import android.text.TextUtils; +import android.text.style.ReplacementSpan; import android.util.SparseIntArray; import android.util.TypedValue; import android.view.Gravity; @@ -48,11 +50,15 @@ import androidx.recyclerview.widget.LinearLayoutManager; import androidx.recyclerview.widget.RecyclerView; +import com.google.common.io.CharSource; + import org.telegram.messenger.AndroidUtilities; import org.telegram.messenger.ApplicationLoader; import org.telegram.messenger.BotWebViewVibrationEffect; import org.telegram.messenger.BuildVars; import org.telegram.messenger.ChatObject; +import org.telegram.messenger.ChatThemeController; +import org.telegram.messenger.DialogObject; import org.telegram.messenger.Emoji; import org.telegram.messenger.ImageReceiver; import org.telegram.messenger.LocaleController; @@ -88,9 +94,11 @@ import org.telegram.ui.Components.LayoutHelper; import org.telegram.ui.Components.Premium.LimitReachedBottomSheet; import org.telegram.ui.Components.Premium.PremiumFeatureBottomSheet; +import org.telegram.ui.Components.Premium.PremiumGradient; import org.telegram.ui.Components.RLottieDrawable; import org.telegram.ui.Components.RecyclerListView; import org.telegram.ui.Components.SimpleThemeDescription; +import org.telegram.ui.Components.SpannableStringLight; import org.telegram.ui.Components.Text; import org.telegram.ui.Components.ViewPagerFixed; import org.telegram.ui.Stories.StoriesUtilities; @@ -221,22 +229,25 @@ public RecyclerView.ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int view = cell; break; case VIEW_TYPE_COLOR_PICKER: - PeerColorGrid colorPicker = peerColorPicker = new PeerColorGrid(getContext(), type, currentAccount); + PeerColorGrid colorPicker = peerColorPicker = new PeerColorGrid(getContext(), type, currentAccount, resourceProvider); colorPicker.setBackgroundColor(getThemedColor(Theme.key_windowBackgroundWhite)); - colorPicker.setSelected(selectedColor); + colorPicker.setSelected(selectedColor, false); colorPicker.setOnColorClick(colorId -> { selectedColor = colorId; - colorPicker.setSelected(colorId); + colorPicker.setSelected(colorId, true); updateMessages(); if (setReplyIconCell != null) { setReplyIconCell.invalidate(); } if (type == PAGE_PROFILE && colorBar != null) { - colorBar.setColor(selectedColor, true); + colorBar.setColor(currentAccount, selectedColor, true); } if (profilePreview != null) { profilePreview.setColor(selectedColor, true); } + if (profilePage != null && profilePage.profilePreview != null && namePage != null) { + profilePage.profilePreview.overrideAvatarColor(namePage.selectedColor); + } checkResetColorButton(); }); view = colorPicker; @@ -341,7 +352,7 @@ public int getItemViewType(int position) { selectedColor = -1; selectedEmoji = 0; if (peerColorPicker != null) { - peerColorPicker.setSelected(selectedColor); + peerColorPicker.setSelected(selectedColor, true); } updateMessages(); if (type == PAGE_PROFILE) { @@ -351,12 +362,15 @@ public int getItemViewType(int position) { setReplyIconCell.update(true); } if (type == PAGE_PROFILE && colorBar != null) { - colorBar.setColor(selectedColor, true); + colorBar.setColor(currentAccount, selectedColor, true); } if (profilePreview != null) { profilePreview.setColor(selectedColor, true); profilePreview.setEmoji(selectedEmoji, true); } + if (profilePage != null && profilePage.profilePreview != null && namePage != null) { + profilePage.profilePreview.overrideAvatarColor(namePage.selectedColor); + } checkResetColorButton(); } }); @@ -392,7 +406,7 @@ public void onScrolled(@NonNull RecyclerView recyclerView, int dx, int dy) { listView.setItemAnimator(itemAnimator); if (type == PAGE_PROFILE) { - profilePreview = new ProfilePreview(getContext()); + profilePreview = new ProfilePreview(getContext(), currentAccount, dialogId, resourceProvider); profilePreview.setColor(selectedColor, false); profilePreview.setEmoji(selectedEmoji, false); addView(profilePreview, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.TOP | Gravity.FILL_HORIZONTAL)); @@ -490,7 +504,7 @@ public SetReplyIconCell(Context context) { } else { textView.setText(LocaleController.getString(isChannel ? R.string.ChannelProfileIcon : R.string.UserProfileIcon)); } - addView(textView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_VERTICAL | Gravity.FILL_HORIZONTAL, 20, 0, 48, 0)); + addView(textView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_VERTICAL | Gravity.FILL_HORIZONTAL, 20, 0, 20, 0)); imageDrawable = new AnimatedEmojiDrawable.SwapAnimatedEmojiDrawable(this, false, dp(24), AnimatedEmojiDrawable.CACHE_TYPE_ALERT_PREVIEW_STATIC); } @@ -513,10 +527,10 @@ public void update(boolean animated) { public void updateImageBounds() { imageDrawable.setBounds( - getWidth() - imageDrawable.getIntrinsicWidth() - dp(21), - (getHeight() - imageDrawable.getIntrinsicHeight()) / 2, - getWidth() - dp(21), - (getHeight() + imageDrawable.getIntrinsicHeight()) / 2 + LocaleController.isRTL ? dp(21) : getWidth() - imageDrawable.getIntrinsicWidth() - dp(21), + (getHeight() - imageDrawable.getIntrinsicHeight()) / 2, + LocaleController.isRTL ? dp(21) + imageDrawable.getIntrinsicWidth() : getWidth() - dp(21), + (getHeight() + imageDrawable.getIntrinsicHeight()) / 2 ); } @@ -636,7 +650,7 @@ public void dismiss() { selectAnimatedEmojiDialog = null; } }; - popup[0].showAsDropDown(cell, 0, yoff, Gravity.TOP | Gravity.RIGHT); + popup[0].showAsDropDown(cell, 0, yoff, Gravity.TOP | (LocaleController.isRTL ? Gravity.LEFT : Gravity.RIGHT)); popup[0].dimBehind(); } @@ -681,11 +695,6 @@ private void updateMessages() { if (peerColorPicker != null) { msg.overrideLinkColor = peerColorPicker.getColorId(); } - if (profilePage != null && profilePage.selectedColor >= 0 && getMessagesController().profilePeerColors != null) { - msg.overrideProfilePeerColor = getMessagesController().profilePeerColors.getColor(profilePage.selectedColor); - } else { - msg.overrideProfilePeerColor = null; - } msg.overrideLinkEmoji = selectedEmoji; cells[i].setAvatar(msg); cells[i].invalidate(); @@ -698,7 +707,7 @@ private void updateMessages() { public void updateColors() { listView.setBackgroundColor(getThemedColor(Theme.key_windowBackgroundGray)); if (type == PAGE_PROFILE && colorBar != null) { - colorBar.setColor(selectedColor, true); + colorBar.setColor(currentAccount, selectedColor, true); } if (button != null) { button.updateColors(); @@ -774,19 +783,17 @@ public void updateThemeColors() { currentColors.put(i, defaultColors[i]); } } - if (themeColors != null) { - for (int i = 0; i < themeColors.size(); ++i) { - currentColors.put(themeColors.keyAt(i), themeColors.valueAt(i)); - } + for (int i = 0; i < themeColors.size(); ++i) { + currentColors.put(themeColors.keyAt(i), themeColors.valueAt(i)); + } + Theme.ThemeAccent accent = themeInfo.getAccent(false); + if (accent != null) { + accent.fillAccentColors(themeColors, currentColors); } if (namePage != null && namePage.messagesCellPreview != null) { - if (Theme.isCurrentThemeDark() == isDark) { - namePage.messagesCellPreview.setOverrideBackground(null); - } else { - Theme.BackgroundDrawableSettings bg = Theme.createBackgroundDrawable(themeInfo, currentColors, wallpaperLink[0], 0, true); - namePage.messagesCellPreview.setOverrideBackground(bg.themedWallpaper != null ? bg.themedWallpaper : bg.wallpaper); - } + Theme.BackgroundDrawableSettings bg = Theme.createBackgroundDrawable(themeInfo, currentColors, wallpaperLink[0], 0, true); + namePage.messagesCellPreview.setOverrideBackground(bg.themedWallpaper != null ? bg.themedWallpaper : bg.wallpaper); } } @@ -906,9 +913,33 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { }; frameLayout.setFitsSystemWindows(true); - colorBar = new ColoredActionBar(context); + colorBar = new ColoredActionBar(context, resourceProvider) { + @Override + protected void onUpdateColor() { + updateLightStatusBar(); + updateActionBarButtonsColor(); + if (tabsView != null) { + tabsView.setBackgroundColor(getTabsViewBackgroundColor()); + } + } + + private int lastBtnColor = 0; + public void updateActionBarButtonsColor() { + final int btnColor = getActionBarButtonColor(); + if (lastBtnColor != btnColor) { + if (backButton != null) { + lastBtnColor = btnColor; + backButton.setColorFilter(new PorterDuffColorFilter(btnColor, PorterDuff.Mode.SRC_IN)); + } + if (dayNightItem != null) { + lastBtnColor = btnColor; + dayNightItem.setColorFilter(new PorterDuffColorFilter(btnColor, PorterDuff.Mode.SRC_IN)); + } + } + } + }; if (profilePage != null) { - colorBar.setColor(profilePage.selectedColor, false); + colorBar.setColor(currentAccount, profilePage.selectedColor, false); } frameLayout.addView(colorBar, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.TOP | Gravity.FILL_HORIZONTAL)); @@ -922,7 +953,7 @@ protected void onTabAnimationUpdate(boolean manual) { viewPager.setAdapter(new ViewPagerFixed.Adapter() { @Override public int getItemCount() { - return isChannel ? 1 : 2; + return 2; } @Override @@ -961,20 +992,22 @@ public void bindView(View view, int position, int viewType) { actionBarContainer.addView(tabsView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 40, Gravity.CENTER)); } else { titleView = new SimpleTextView(context); - titleView.setText(LocaleController.getString(R.string.ChannelColorTitle)); + titleView.setText(LocaleController.getString(R.string.ChannelColorTitle2)); titleView.setEllipsizeByGradient(true); titleView.setTextSize(20); titleView.setTextColor(getThemedColor(Theme.key_actionBarDefaultTitle)); titleView.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); actionBarContainer.addView(titleView, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_VERTICAL | Gravity.LEFT, 72, 0, 72, 0)); } - if (startAtProfile && !isChannel) { + + if (startAtProfile) { viewPager.setPosition(1); if (tabsView != null) { tabsView.setSelected(1); } if (colorBar != null) { colorBar.setProgressToGradient(1f); + updateLightStatusBar(); } } @@ -1031,39 +1064,13 @@ public boolean hasUnsavedChanged() { return namePage.hasUnsavedChanged() || profilePage.hasUnsavedChanged(); } - private void showBoostLimit(boolean error) { - getMessagesController().getBoostsController().getBoostsStats(dialogId, boostsStatus -> { - if (error || boostsStatus.level < getMessagesController().channelColorLevelMin) { - getMessagesController().getBoostsController().userCanBoostChannel(dialogId, boostsStatus, canApplyBoost -> { - if (getContext() == null) { - return; - } - LimitReachedBottomSheet limitReachedBottomSheet = new LimitReachedBottomSheet(this, getContext(), TYPE_BOOSTS_FOR_COLOR, currentAccount, getResourceProvider()); - limitReachedBottomSheet.setCanApplyBoost(canApplyBoost); - - limitReachedBottomSheet.setBoostsStats(boostsStatus, true); - limitReachedBottomSheet.setDialogId(dialogId); - limitReachedBottomSheet.showStatisticButtonInLink(() -> { - TLRPC.Chat chat = getMessagesController().getChat(-dialogId); - Bundle args = new Bundle(); - args.putLong("chat_id", -dialogId); - args.putBoolean("is_megagroup", chat.megagroup); - args.putBoolean("start_from_boosts", true); - TLRPC.ChatFull chatInfo = getMessagesController().getChatFull(-dialogId); - if (chatInfo == null || !chatInfo.can_view_stats) { - args.putBoolean("only_boosts", true); - }; - StatisticActivity fragment = new StatisticActivity(args); - presentFragment(fragment); - }); - showDialog(limitReachedBottomSheet); - loading = false; - AndroidUtilities.runOnUIThread(() -> getCurrentPage().button.setLoading(false), 300); - }); - } else { - apply(); - } - }); + private void setLoading(boolean loading) { + if (namePage != null && namePage.button != null) { + namePage.button.setLoading(loading); + } + if (profilePage != null && profilePage.button != null) { + profilePage.button.setLoading(loading); + } } @Override @@ -1106,15 +1113,7 @@ private void buttonClick() { return; } if (isChannel) { - final TLRPC.Chat chat = getMessagesController().getChat(-dialogId); - if (namePage != null && chat != null && namePage.selectedColor == ChatObject.getColorId(chat) && namePage.selectedEmoji == ChatObject.getEmojiId(chat)) { - finishFragment(); - return; - } - loading = true; - getCurrentPage().button.setLoading(true); - showBoostLimit(false); - return; + finishFragment(); } else { if (!getUserConfig().isPremium()) { showDialog(new PremiumFeatureBottomSheet(PeerColorActivity.this, PremiumPreviewFragment.PREMIUM_FEATURE_NAME_COLOR, true)); @@ -1135,46 +1134,7 @@ private void apply() { } if (isChannel) { - final TLRPC.Chat chat = getMessagesController().getChat(-dialogId); - if (chat == null) { - return; - } - if (namePage.selectedColor == ChatObject.getColorId(chat) && namePage.selectedEmoji == ChatObject.getEmojiId(chat)) { - getCurrentPage().button.setLoading(loading = false); - finishFragment(); - return; - } - TLRPC.TL_channels_updateColor req = new TLRPC.TL_channels_updateColor(); - req.channel = getMessagesController().getInputChannel(-dialogId); - if (req.channel == null) { - return; - } - chat.flags2 |= 64; - if (chat.color == null) { - chat.color = new TLRPC.TL_peerColor(); - } - chat.flags2 |= 128; - req.color = chat.color.color = namePage.selectedColor; - if (namePage.selectedEmoji != 0) { - chat.color.background_emoji_id = namePage.selectedEmoji; - - req.flags |= 1; - req.background_emoji_id = namePage.selectedEmoji; - } else { - chat.color.background_emoji_id = 0; - } - getCurrentPage().button.setLoading(loading = true); - getMessagesController().putChat(chat, false); - getUserConfig().saveConfig(true); - getConnectionsManager().sendRequest(req, (res, err) -> AndroidUtilities.runOnUIThread(() -> { - applying = false; - if (err != null && "BOOSTS_REQUIRED".equals(err.text)) { - showBoostLimit(true); - } else { - finishFragment(); - showBulletin(); - } - })); + finishFragment(); } else { final TLRPC.User me = getUserConfig().getCurrentUser(); if (me.color == null) { @@ -1225,6 +1185,8 @@ private void apply() { } getMessagesController().putUser(me, false); getUserConfig().saveConfig(true); + finishFragment(); + showBulletin(); } applying = true; getNotificationCenter().postNotificationName(NotificationCenter.updateInterfaces, MessagesController.UPDATE_MASK_EMOJI_STATUS); @@ -1320,6 +1282,67 @@ public void didReceivedNotification(int id, int account, Object... args) { } } + public static class LevelLock extends Drawable { + + private final Theme.ResourcesProvider resourcesProvider; + private final Text text; + private final float lockScale = .875f; + private final Drawable lock; + private final PremiumGradient.PremiumGradientTools gradientTools; + + public LevelLock(Context context, int lvl, Theme.ResourcesProvider resourcesProvider) { + this(context, false, lvl, resourcesProvider); + } + + public LevelLock(Context context, boolean plus, int lvl, Theme.ResourcesProvider resourcesProvider) { + this.resourcesProvider = resourcesProvider; + text = new Text(LocaleController.formatPluralString(plus ? "BoostLevelPlus" : "BoostLevel", lvl), 12, AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); + lock = context.getResources().getDrawable(R.drawable.mini_switch_lock).mutate(); + lock.setColorFilter(new PorterDuffColorFilter(Color.WHITE, PorterDuff.Mode.SRC_IN)); + gradientTools = new PremiumGradient.PremiumGradientTools(Theme.key_premiumGradient1, Theme.key_premiumGradient2, -1, -1, -1, resourcesProvider); + } + + @Override + public void draw(@NonNull Canvas canvas) { + int left = getBounds().left; + int cy = getBounds().centerY(); + + AndroidUtilities.rectTmp.set(left, cy - getIntrinsicHeight() / 2f, left + getIntrinsicWidth(), cy + getIntrinsicHeight() / 2f); + gradientTools.gradientMatrix(AndroidUtilities.rectTmp); + canvas.drawRoundRect(AndroidUtilities.rectTmp, dp(10), dp(10), gradientTools.paint); + + lock.setBounds( + left + dp(3.33f), + (int) (cy - lock.getIntrinsicHeight() * lockScale / 2f), + (int) (left + dp(3.33f) + lock.getIntrinsicWidth() * lockScale), + (int) (cy + lock.getIntrinsicHeight() * lockScale / 2f) + ); + lock.draw(canvas); + + text.draw(canvas, left + dp(3.66f) + lock.getIntrinsicWidth() * lockScale, cy, Color.WHITE, 1f); + } + + @Override + public void setAlpha(int alpha) {} + @Override + public void setColorFilter(@Nullable ColorFilter colorFilter) {} + + @Override + public int getOpacity() { + return PixelFormat.TRANSPARENT; + } + + @Override + public int getIntrinsicWidth() { + return (int) (dp(3.66f + 6) + lock.getIntrinsicWidth() * lockScale + text.getWidth()); + } + + @Override + public int getIntrinsicHeight() { + return dp(18.33f); + } + } + public static class ChangeNameColorCell extends View { private final int currentAccount; private final boolean isChannel; @@ -1327,6 +1350,7 @@ public static class ChangeNameColorCell extends View { private final Drawable drawable; private final Text buttonText; + private LevelLock lock; private final Paint userTextBackgroundPaint = new Paint(Paint.ANTI_ALIAS_FLAG); private Text userText; @@ -1336,16 +1360,51 @@ public static class ChangeNameColorCell extends View { private PeerColorDrawable color1Drawable; private PeerColorDrawable color2Drawable; - public ChangeNameColorCell(int currentAccount, boolean isChannel, Context context, Theme.ResourcesProvider resourcesProvider) { + public ChangeNameColorCell(int currentAccount, long dialogId, Context context, Theme.ResourcesProvider resourcesProvider) { super(context); this.currentAccount = currentAccount; - this.isChannel = isChannel; + this.isChannel = dialogId < 0; this.resourcesProvider = resourcesProvider; drawable = context.getResources().getDrawable(R.drawable.msg_palette).mutate(); drawable.setColorFilter(new PorterDuffColorFilter(Theme.getColor(Theme.key_windowBackgroundWhiteBlueText4, resourcesProvider), PorterDuff.Mode.SRC_IN)); - buttonText = new Text(LocaleController.getString(isChannel ? R.string.ChangeChannelNameColor : R.string.ChangeUserNameColor), 16); + CharSequence button = LocaleController.getString(isChannel ? R.string.ChangeChannelNameColor2 : R.string.ChangeUserNameColor); + if (isChannel && MessagesController.getInstance(currentAccount).getMainSettings().getInt("boostingappearance", 0) < 3) { + MessagesController mc = MessagesController.getInstance(currentAccount); + int minlvl = Integer.MAX_VALUE, maxlvl = 0; + if (mc.peerColors != null) { + minlvl = Math.min(minlvl, mc.peerColors.maxLevel()); + maxlvl = Math.max(maxlvl, mc.peerColors.maxLevel()); + minlvl = Math.min(minlvl, mc.peerColors.minLevel()); + maxlvl = Math.max(maxlvl, mc.peerColors.minLevel()); + } + minlvl = Math.min(minlvl, mc.channelBgIconLevelMin); + maxlvl = Math.min(maxlvl, mc.channelBgIconLevelMin); + if (mc.profilePeerColors != null) { + minlvl = Math.min(minlvl, mc.profilePeerColors.maxLevel()); + maxlvl = Math.max(maxlvl, mc.profilePeerColors.maxLevel()); + minlvl = Math.min(minlvl, mc.profilePeerColors.minLevel()); + maxlvl = Math.max(maxlvl, mc.profilePeerColors.minLevel()); + } + minlvl = Math.min(minlvl, mc.channelProfileIconLevelMin); + maxlvl = Math.max(maxlvl, mc.channelProfileIconLevelMin); + minlvl = Math.min(minlvl, mc.channelEmojiStatusLevelMin); + maxlvl = Math.max(maxlvl, mc.channelEmojiStatusLevelMin); + minlvl = Math.min(minlvl, mc.channelWallpaperLevelMin); + maxlvl = Math.max(maxlvl, mc.channelWallpaperLevelMin); + minlvl = Math.min(minlvl, mc.channelCustomWallpaperLevelMin); + maxlvl = Math.max(maxlvl, mc.channelCustomWallpaperLevelMin); + TLRPC.Chat chat = mc.getChat(-dialogId); + int currentLevel = chat == null ? 0 : chat.level; + if (currentLevel < maxlvl) { + lock = new LevelLock(context, true, Math.max(currentLevel, minlvl), resourcesProvider); + } + } + if (isChannel && lock == null) { + button = TextCell.applyNewSpan(button); + } + buttonText = new Text(button, 16); updateColors(); } @@ -1385,7 +1444,8 @@ public void set(TLRPC.Chat chat, boolean divider) { userText.setColor(color); userTextBackgroundPaint.setColor(Theme.multAlpha(color, .10f)); - color1Drawable = color2Drawable = null; + color1Drawable = PeerColorDrawable.from(currentAccount, colorId).setRadius(dp(11)); + color2Drawable = ChatObject.getProfileColorId(chat) >= 0 ? PeerColorDrawable.fromProfile(currentAccount, ChatObject.getProfileColorId(chat)).setRadius(dp(11)) : null; } public void set(TLRPC.User user) { @@ -1442,13 +1502,18 @@ protected void dispatchDraw(Canvas canvas) { getMeasuredHeight() / 2 + drawable.getIntrinsicHeight() / 2 ); drawable.draw(canvas); - buttonText - .ellipsize(getMeasuredWidth() - dp(64 + 7 + 100)) - .draw(canvas, LocaleController.isRTL ? getMeasuredWidth() - buttonText.getWidth() - dp(64 + 7) : dp(64 + 7), getMeasuredHeight() / 2f); + buttonText.ellipsize(getMeasuredWidth() - dp(64 + 7 + 100) - (lock != null ? lock.getIntrinsicWidth() + dp(8) : 0)); + float textX = LocaleController.isRTL ? getMeasuredWidth() - buttonText.getWidth() - dp(64 + 7) : dp(64 + 7); + buttonText.draw(canvas, textX, getMeasuredHeight() / 2f); + if (lock != null) { + int x = (int) (textX + buttonText.getWidth() + dp(6)); + lock.setBounds(x, 0, x, getHeight()); + lock.draw(canvas); + } if (color1Drawable != null && color2Drawable != null) { - int x = getMeasuredWidth() - dp(16); + int x = LocaleController.isRTL ? dp(24 + 16 + 18) : getMeasuredWidth() - dp(24); color2Drawable.setBounds(x - dp(11), (getMeasuredHeight() - dp(11)) / 2, x, (getMeasuredHeight() + dp(11)) / 2); color2Drawable.stroke(dpf2(3), Theme.getColor(Theme.key_windowBackgroundWhite, resourcesProvider)); color2Drawable.draw(canvas); @@ -1459,7 +1524,7 @@ protected void dispatchDraw(Canvas canvas) { color1Drawable.draw(canvas); } else if (userText != null) { - final int maxWidth = (int) (getMeasuredWidth() - dp(64 + 7 + 15 + 9 + 9 + 12) - Math.min(buttonText.getWidth(), getMeasuredWidth() - dp(64 + 100))); + final int maxWidth = (int) (getMeasuredWidth() - dp(64 + 7 + 15 + 9 + 9 + 12) - Math.min(buttonText.getWidth() + (lock == null ? 0 : lock.getIntrinsicWidth() + dp(6 + 6)), getMeasuredWidth() - dp(64 + 100))); final int w = (int) Math.min(userText.getWidth(), maxWidth); AndroidUtilities.rectTmp.set( @@ -1485,7 +1550,8 @@ protected void dispatchDraw(Canvas canvas) { } } - public class PeerColorGrid extends View { + public static class PeerColorGrid extends View { + private final Theme.ResourcesProvider resourcesProvider; private final Paint backgroundPaint = new Paint(Paint.ANTI_ALIAS_FLAG); { backgroundPaint.setStyle(Paint.Style.STROKE); } @@ -1517,21 +1583,21 @@ public void set(MessagesController.PeerColor color) { if (color == null) { return; } - final boolean dark = isDark; + final boolean dark = resourcesProvider == null ? Theme.isCurrentThemeDark() : resourcesProvider.isDark(); if (type == PAGE_NAME) { if (dark && color.hasColor2() && !color.hasColor3()) { - paint1.setColor(color.getColor2(dark)); - paint2.setColor(color.getColor1(dark)); + paint1.setColor(color.getColor(1, resourcesProvider)); + paint2.setColor(color.getColor(0, resourcesProvider)); } else { - paint1.setColor(color.getColor1(dark)); - paint2.setColor(color.getColor2(dark)); + paint1.setColor(color.getColor(0, resourcesProvider)); + paint2.setColor(color.getColor(1, resourcesProvider)); } - paint3.setColor(color.getColor3(dark)); + paint3.setColor(color.getColor(2, resourcesProvider)); hasColor2 = color.hasColor2(dark); hasColor3 = color.hasColor3(dark); } else { - paint1.setColor(color.getColor1(dark)); - paint2.setColor(color.hasColor6(dark) ? color.getColor2(dark) : color.getColor1(dark)); + paint1.setColor(color.getColor(0, resourcesProvider)); + paint2.setColor(color.hasColor6(dark) ? color.getColor(1, resourcesProvider) : color.getColor(0, resourcesProvider)); hasColor2 = color.hasColor6(dark); hasColor3 = false; } @@ -1595,7 +1661,7 @@ protected void draw(Canvas canvas) { if (selectT > 0) { backgroundPaint.setStrokeWidth(dpf2(2)); - backgroundPaint.setColor(getThemedColor(Theme.key_windowBackgroundWhite)); + backgroundPaint.setColor(Theme.getColor(Theme.key_windowBackgroundWhite, resourcesProvider)); canvas.drawCircle( bounds.centerX(), bounds.centerY(), Math.min(bounds.height() / 2f, bounds.width() / 2f) + backgroundPaint.getStrokeWidth() * AndroidUtilities.lerp(.5f, -2f, selectT), @@ -1621,10 +1687,11 @@ public void setPressed(boolean pressed) { private ColorButton[] buttons; - public PeerColorGrid(Context context, int type, int currentAccount) { + public PeerColorGrid(Context context, int type, int currentAccount, Theme.ResourcesProvider resourcesProvider) { super(context); this.type = type; this.currentAccount = currentAccount; + this.resourcesProvider = resourcesProvider; } public void updateColors() { @@ -1634,9 +1701,9 @@ public void updateColors() { for (int i = 0; i < buttons.length; ++i) { if (i < 7 && type == PAGE_NAME) { buttons[i].id = order[i]; - buttons[i].set(Theme.getColor(Theme.keys_avatar_nameInMessage[order[i]], resourceProvider)); + buttons[i].set(Theme.getColor(Theme.keys_avatar_nameInMessage[order[i]], resourcesProvider)); } else { - final int id = i - (type == PAGE_NAME ? 7 : 0); + final int id = i; if (peerColors != null && id >= 0 && id < peerColors.colors.size()) { buttons[i].id = peerColors.colors.get(id).id; buttons[i].set(peerColors.colors.get(id)); @@ -1653,7 +1720,7 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { final MessagesController mc = MessagesController.getInstance(currentAccount); final MessagesController.PeerColors peerColors = type == PAGE_NAME ? mc.peerColors : mc.profilePeerColors; - final int colorsCount = 7 + (peerColors == null ? 0 : peerColors.colors.size()); + final int colorsCount = peerColors == null ? 0 : peerColors.colors.size(); final int columns = type == PAGE_NAME ? 7 : 8; final float iconSize = Math.min(dp(38 + 16), width / (columns + (columns + 1) * .28947f)); @@ -1669,15 +1736,9 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { buttons = new ColorButton[colorsCount]; for (int i = 0; i < colorsCount; ++i) { buttons[i] = new ColorButton(); - if (i < 7 && type == PAGE_NAME) { - buttons[i].id = order[i]; - buttons[i].set(Theme.getColor(Theme.keys_avatar_nameInMessage[order[i]])); - } else { - final int id = i - (type == PAGE_NAME ? 7 : 0); - if (peerColors != null && id >= 0 && id < peerColors.colors.size()) { - buttons[i].id = peerColors.colors.get(id).id; - buttons[i].set(peerColors.colors.get(id)); - } + if (peerColors != null && i >= 0 && i < peerColors.colors.size()) { + buttons[i].id = peerColors.colors.get(i).id; + buttons[i].set(peerColors.colors.get(i)); } } } @@ -1703,6 +1764,11 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { } private final Paint dividerPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + private boolean needDivider = true; + public void setDivider(boolean needDivider) { + this.needDivider = needDivider; + invalidate(); + } @Override protected void dispatchDraw(Canvas canvas) { @@ -1711,16 +1777,18 @@ protected void dispatchDraw(Canvas canvas) { buttons[i].draw(canvas); } } - dividerPaint.setColor(getThemedColor(Theme.key_divider)); - canvas.drawRect(dp(21), getMeasuredHeight() - 1, getMeasuredWidth() - dp(21), getMeasuredHeight(), dividerPaint); + if (needDivider) { + dividerPaint.setColor(Theme.getColor(Theme.key_divider, resourcesProvider)); + canvas.drawRect(dp(21), getMeasuredHeight() - 1, getMeasuredWidth() - dp(21), getMeasuredHeight(), dividerPaint); + } } private int selectedColorId = 0; - public void setSelected(int colorId) { + public void setSelected(int colorId, boolean animated) { selectedColorId = colorId; if (buttons != null) { for (int i = 0; i < buttons.length; ++i) { - buttons[i].setSelected(buttons[i].id == colorId, true); + buttons[i].setSelected(buttons[i].id == colorId, animated); } } } @@ -1785,6 +1853,37 @@ public boolean dispatchTouchEvent(MotionEvent event) { } } + public static class PeerColorSpan extends ReplacementSpan { + private int size = dp(21); + public PeerColorDrawable drawable; + + public PeerColorSpan(boolean profile, int currentAccount, int colorId) { + drawable = profile ? PeerColorDrawable.fromProfile(currentAccount, colorId) : PeerColorDrawable.from(currentAccount, colorId); + } + + public PeerColorSpan setSize(int sz) { + if (drawable != null) { + drawable.setRadius(sz / 2f); + size = sz; + } + return this; + } + + @Override + public int getSize(@NonNull Paint paint, CharSequence text, int start, int end, @Nullable Paint.FontMetricsInt fm) { + return dp(3) + size + dp(3); + } + + @Override + public void draw(@NonNull Canvas canvas, CharSequence text, int start, int end, float x, int top, int y, int bottom, @NonNull Paint paint) { + if (drawable != null) { + int cy = (top + bottom) / 2; + drawable.setBounds((int) (x + dp(3)), cy - size, (int) (x + dp(5) + size), cy + size); + drawable.draw(canvas); + } + } + } + public static class PeerColorDrawable extends Drawable { public static PeerColorDrawable from(int currentAccount, int colorId) { @@ -1894,37 +1993,39 @@ public int getIntrinsicWidth() { } } - private class ColoredActionBar extends View { + public static class ColoredActionBar extends View { private int defaultColor; - public ColoredActionBar(Context context) { + private final Theme.ResourcesProvider resourcesProvider; + + public ColoredActionBar(Context context, Theme.ResourcesProvider resourcesProvider) { super(context); - defaultColor = getThemedColor(Theme.key_actionBarDefault); - setColor(-1, false); + this.resourcesProvider = resourcesProvider; + defaultColor = Theme.getColor(Theme.key_actionBarDefault, resourcesProvider); + setColor(-1, -1, false); } - public void setColor(int colorId, boolean animated) { + public void setColor(int currentAccount, int colorId, boolean animated) { isDefault = false; - if (colorId < 0) { + if (colorId < 0 || currentAccount < 0) { isDefault = true; - color1 = color2 = getThemedColor(Theme.key_actionBarDefault); + color1 = color2 = Theme.getColor(Theme.key_actionBarDefault, resourcesProvider); } else { MessagesController.PeerColors peerColors = MessagesController.getInstance(currentAccount).profilePeerColors; MessagesController.PeerColor peerColor = peerColors == null ? null : peerColors.getColor(colorId); if (peerColor != null) { + final boolean isDark = resourcesProvider != null ? resourcesProvider.isDark() : Theme.isCurrentThemeDark(); color1 = peerColor.getBgColor1(isDark); color2 = peerColor.getBgColor2(isDark); } else { isDefault = true; - color1 = color2 = getThemedColor(Theme.key_actionBarDefault); + color1 = color2 = Theme.getColor(Theme.key_actionBarDefault, resourcesProvider); } } if (!animated) { color1Animated.set(color1, true); color2Animated.set(color2, true); } - updateLightStatusBar(); - updateActionBarButtonsColor(); invalidate(); } @@ -1932,9 +2033,7 @@ public void setColor(int colorId, boolean animated) { public void setProgressToGradient(float progress) { if (Math.abs(progressToGradient - progress) > 0.001f) { progressToGradient = progress; - updateTabsViewBackground(); - updateActionBarButtonsColor(); - updateLightStatusBar(); + onUpdateColor(); invalidate(); } } @@ -1948,6 +2047,10 @@ public void setProgressToGradient(float progress) { private LinearGradient backgroundGradient; private final Paint backgroundPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + protected void onUpdateColor() { + + } + @Override protected void dispatchDraw(Canvas canvas) { final int color1 = color1Animated.set(this.color1); @@ -1955,8 +2058,7 @@ protected void dispatchDraw(Canvas canvas) { if (backgroundGradient == null || backgroundGradientColor1 != color1 || backgroundGradientColor2 != color2 || backgroundGradientHeight != getHeight()) { backgroundGradient = new LinearGradient(0, 0, 0, backgroundGradientHeight = getHeight(), new int[] { backgroundGradientColor2 = color2, backgroundGradientColor1 = color1 }, new float[] { 0, 1 }, Shader.TileMode.CLAMP); backgroundPaint.setShader(backgroundGradient); - updateTabsViewBackground(); - updateLightStatusBar(); + onUpdateColor(); } if (progressToGradient < 1) { canvas.drawColor(defaultColor); @@ -1967,47 +2069,35 @@ protected void dispatchDraw(Canvas canvas) { } } + protected boolean ignoreMeasure; + @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { - super.onMeasure(widthMeasureSpec, MeasureSpec.makeMeasureSpec(AndroidUtilities.statusBarHeight + dp(144), MeasureSpec.EXACTLY)); + super.onMeasure(widthMeasureSpec, ignoreMeasure ? heightMeasureSpec : MeasureSpec.makeMeasureSpec(AndroidUtilities.statusBarHeight + dp(144), MeasureSpec.EXACTLY)); } public void updateColors() { - defaultColor = getThemedColor(Theme.key_actionBarDefault); - updateTabsViewBackground(); - updateActionBarButtonsColor(); - updateLightStatusBar(); + defaultColor = Theme.getColor(Theme.key_actionBarDefault, resourcesProvider); + onUpdateColor(); invalidate(); } - - private int lastBtnColor = 0; - public void updateActionBarButtonsColor() { - final int btnColor = ColorUtils.blendARGB(getThemedColor(Theme.key_actionBarDefaultIcon), isDefault ? getThemedColor(Theme.key_actionBarDefaultIcon) : Color.WHITE, progressToGradient); - if (lastBtnColor != btnColor) { - if (backButton != null) { - lastBtnColor = btnColor; - backButton.setColorFilter(new PorterDuffColorFilter(btnColor, PorterDuff.Mode.SRC_IN)); - } - if (dayNightItem != null) { - lastBtnColor = btnColor; - dayNightItem.setColorFilter(new PorterDuffColorFilter(btnColor, PorterDuff.Mode.SRC_IN)); - } - } - } public int getColor() { - return ColorUtils.blendARGB(getThemedColor(Theme.key_actionBarDefault), ColorUtils.blendARGB(color1Animated.get(), color2Animated.get(), .75f), progressToGradient); + return ColorUtils.blendARGB(Theme.getColor(Theme.key_actionBarDefault, resourcesProvider), ColorUtils.blendARGB(color1Animated.get(), color2Animated.get(), .75f), progressToGradient); } - private void updateTabsViewBackground() { - if (tabsView == null) return; - tabsView.setBackgroundColor( + public int getActionBarButtonColor() { + return ColorUtils.blendARGB(Theme.getColor(Theme.key_actionBarDefaultIcon, resourcesProvider), isDefault ? Theme.getColor(Theme.key_actionBarDefaultIcon, resourcesProvider) : Color.WHITE, progressToGradient); + } + + public int getTabsViewBackgroundColor() { + return ( ColorUtils.blendARGB( - AndroidUtilities.computePerceivedBrightness(getThemedColor(Theme.key_actionBarDefault)) > .721f ? - getThemedColor(Theme.key_actionBarDefaultIcon) : - Theme.adaptHSV(getThemedColor(Theme.key_actionBarDefault), +.08f, -.08f), + AndroidUtilities.computePerceivedBrightness(Theme.getColor(Theme.key_actionBarDefault, resourcesProvider)) > .721f ? + Theme.getColor(Theme.key_actionBarDefaultIcon, resourcesProvider) : + Theme.adaptHSV(Theme.getColor(Theme.key_actionBarDefault, resourcesProvider), +.08f, -.08f), AndroidUtilities.computePerceivedBrightness(ColorUtils.blendARGB(color1Animated.get(), color2Animated.get(), .75f)) > .721f ? - getThemedColor(Theme.key_windowBackgroundWhiteBlueIcon) : + Theme.getColor(Theme.key_windowBackgroundWhiteBlueIcon, resourcesProvider) : Theme.adaptHSV(ColorUtils.blendARGB(color1Animated.get(), color2Animated.get(), .75f), +.08f, -.08f), progressToGradient ) @@ -2015,20 +2105,47 @@ private void updateTabsViewBackground() { } } - private class ProfilePreview extends FrameLayout { + public static class ProfilePreview extends FrameLayout { + + private final Theme.ResourcesProvider resourcesProvider; + private final int currentAccount; + private final long dialogId; + private final boolean isChannel; private final ImageReceiver imageReceiver = new ImageReceiver(this); private final AvatarDrawable avatarDrawable = new AvatarDrawable(); private final SimpleTextView titleView, subtitleView; + private final AnimatedEmojiDrawable.SwapAnimatedEmojiDrawable statusEmoji; + private final AnimatedEmojiDrawable.SwapAnimatedEmojiDrawable emoji = new AnimatedEmojiDrawable.SwapAnimatedEmojiDrawable(this, false, dp(20), AnimatedEmojiDrawable.CACHE_TYPE_ALERT_PREVIEW_STATIC); private final StoriesUtilities.StoryGradientTools storyGradient = new StoriesUtilities.StoryGradientTools(this, false); - public ProfilePreview(Context context) { + public ProfilePreview(Context context, int currentAccount, long dialogId, Theme.ResourcesProvider resourcesProvider) { super(context); - titleView = new SimpleTextView(context); + this.currentAccount = currentAccount; + this.dialogId = dialogId; + this.resourcesProvider = resourcesProvider; + this.isChannel = dialogId < 0; + + titleView = new SimpleTextView(context) { + @Override + protected void onAttachedToWindow() { + super.onAttachedToWindow(); + statusEmoji.attach(); + } + + @Override + protected void onDetachedFromWindow() { + super.onDetachedFromWindow(); + statusEmoji.detach(); + } + }; + statusEmoji = new AnimatedEmojiDrawable.SwapAnimatedEmojiDrawable(titleView, true, dp(24)); + titleView.setDrawablePadding(dp(8)); + titleView.setRightDrawable(statusEmoji); titleView.setTextColor(0xFFFFFFFF); titleView.setTextSize(20); titleView.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); @@ -2092,18 +2209,26 @@ public ProfilePreview(Context context) { setWillNotDraw(false); } - public void updateAvatarDrawable(MessagesController.PeerColor profilePeerColor) { - if (isChannel) { - TLRPC.Chat chat = MessagesController.getInstance(currentAccount).getChat(-dialogId); - if (chat != null) { - avatarDrawable.setInfo(chat.id, chat.title, null, null, ChatObject.getColorId(chat), profilePeerColor); + public void overrideAvatarColor(int colorId) { + final int color1, color2; + if (colorId >= 14) { + MessagesController messagesController = MessagesController.getInstance(UserConfig.selectedAccount); + MessagesController.PeerColors peerColors = messagesController != null ? messagesController.peerColors : null; + MessagesController.PeerColor peerColor = peerColors != null ? peerColors.getColor(colorId) : null; + if (peerColor != null) { + final int peerColorValue = peerColor.getColor1(); + color1 = getThemedColor(Theme.keys_avatar_background[AvatarDrawable.getPeerColorIndex(peerColorValue)]); + color2 = getThemedColor(Theme.keys_avatar_background2[AvatarDrawable.getPeerColorIndex(peerColorValue)]); + } else { + color1 = getThemedColor(Theme.keys_avatar_background[AvatarDrawable.getColorIndex(colorId)]); + color2 = getThemedColor(Theme.keys_avatar_background2[AvatarDrawable.getColorIndex(colorId)]); } } else { - TLRPC.User user = UserConfig.getInstance(currentAccount).getCurrentUser(); - if (user != null) { - avatarDrawable.setInfo(user.id, user.first_name, user.last_name, null, UserObject.getColorId(user), profilePeerColor); - } + color1 = getThemedColor(Theme.keys_avatar_background[AvatarDrawable.getColorIndex(colorId)]); + color2 = getThemedColor(Theme.keys_avatar_background2[AvatarDrawable.getColorIndex(colorId)]); } + avatarDrawable.setColor(color1, color2); + invalidate(); } @Override @@ -2120,12 +2245,18 @@ protected void onDetachedFromWindow() { imageReceiver.onDetachedFromWindow(); } + private int getThemedColor(int key) { + return Theme.getColor(key, resourcesProvider); + } + private int lastColorId = -1; public void setColor(int colorId, boolean animated) { MessagesController.PeerColors peerColors = MessagesController.getInstance(currentAccount).profilePeerColors; MessagesController.PeerColor peerColor = peerColors == null ? null : peerColors.getColor(lastColorId = colorId); + final boolean isDark = resourcesProvider != null ? resourcesProvider.isDark() : Theme.isCurrentThemeDark(); if (peerColor != null) { emoji.setColor(adaptProfileEmojiColor(peerColor.getBgColor1(isDark))); + statusEmoji.setColor(ColorUtils.blendARGB(peerColor.getColor(1, resourcesProvider), peerColor.hasColor6(isDark) ? peerColor.getColor(4, resourcesProvider) : peerColor.getColor(2, resourcesProvider), .5f)); final int accentColor = ColorUtils.blendARGB(peerColor.getStoryColor1(isDark), peerColor.getStoryColor2(isDark), .5f); if (!Theme.hasHue(getThemedColor(Theme.key_actionBarDefault))) { subtitleView.setTextColor(accentColor); @@ -2134,17 +2265,18 @@ public void setColor(int colorId, boolean animated) { } titleView.setTextColor(Color.WHITE); } else { + final int emojiColor; if (AndroidUtilities.computePerceivedBrightness(getThemedColor(Theme.key_actionBarDefault)) > .8f) { - emoji.setColor(Theme.getColor(Theme.key_windowBackgroundWhiteBlueText, resourceProvider)); + emoji.setColor(getThemedColor(Theme.key_windowBackgroundWhiteBlueText)); } else if (AndroidUtilities.computePerceivedBrightness(getThemedColor(Theme.key_actionBarDefault)) < .2f) { - emoji.setColor(Theme.multAlpha(Theme.getColor(Theme.key_actionBarDefaultTitle, resourceProvider), .5f)); + emoji.setColor(Theme.multAlpha(getThemedColor(Theme.key_actionBarDefaultTitle), .5f)); } else { - emoji.setColor(adaptProfileEmojiColor(Theme.getColor(Theme.key_actionBarDefault, resourceProvider))); + emoji.setColor(adaptProfileEmojiColor(getThemedColor(Theme.key_actionBarDefault))); } - subtitleView.setTextColor(Theme.getColor(Theme.key_actionBarDefaultSubtitle, resourceProvider)); - titleView.setTextColor(Theme.getColor(Theme.key_actionBarDefaultTitle, resourceProvider)); + statusEmoji.setColor(Theme.getColor(Theme.key_profile_verifiedBackground, resourcesProvider)); + subtitleView.setTextColor(getThemedColor(Theme.key_actionBarDefaultSubtitle)); + titleView.setTextColor(getThemedColor(Theme.key_actionBarDefaultTitle)); } - updateAvatarDrawable(peerColor); storyGradient.setColorId(colorId, animated); invalidate(); @@ -2158,14 +2290,32 @@ public void setEmoji(long docId, boolean animated) { } MessagesController.PeerColors peerColors = MessagesController.getInstance(currentAccount).profilePeerColors; MessagesController.PeerColor peerColor = peerColors == null ? null : peerColors.getColor(lastColorId); + final boolean isDark = resourcesProvider != null ? resourcesProvider.isDark() : Theme.isCurrentThemeDark(); if (peerColor != null) { emoji.setColor(adaptProfileEmojiColor(peerColor.getBgColor1(isDark))); } else if (AndroidUtilities.computePerceivedBrightness(getThemedColor(Theme.key_actionBarDefault)) > .8f) { - emoji.setColor(Theme.getColor(Theme.key_windowBackgroundWhiteBlueText, resourceProvider)); + emoji.setColor(getThemedColor(Theme.key_windowBackgroundWhiteBlueText)); } else if (AndroidUtilities.computePerceivedBrightness(getThemedColor(Theme.key_actionBarDefault)) < .2f) { - emoji.setColor(Theme.multAlpha(Theme.getColor(Theme.key_actionBarDefaultTitle, resourceProvider), .5f)); + emoji.setColor(Theme.multAlpha(Theme.getColor(Theme.key_actionBarDefaultTitle), .5f)); + } else { + emoji.setColor(adaptProfileEmojiColor(Theme.getColor(Theme.key_actionBarDefault))); + } + if (peerColor != null) { + statusEmoji.setColor(ColorUtils.blendARGB(peerColor.getColor(1, resourcesProvider), peerColor.hasColor6(isDark) ? peerColor.getColor(4, resourcesProvider) : peerColor.getColor(2, resourcesProvider), .5f)); + } else { + statusEmoji.setColor(Theme.getColor(Theme.key_profile_verifiedBackground, resourcesProvider)); + } + } + + public void setStatusEmoji(long docId, boolean animated) { + statusEmoji.set(docId, animated); + MessagesController.PeerColors peerColors = MessagesController.getInstance(currentAccount).profilePeerColors; + MessagesController.PeerColor peerColor = peerColors == null ? null : peerColors.getColor(lastColorId); + final boolean isDark = resourcesProvider != null ? resourcesProvider.isDark() : Theme.isCurrentThemeDark(); + if (peerColor != null) { + statusEmoji.setColor(ColorUtils.blendARGB(peerColor.getColor2(isDark), peerColor.hasColor6(isDark) ? peerColor.getColor5(isDark) : peerColor.getColor3(isDark), .5f)); } else { - emoji.setColor(adaptProfileEmojiColor(Theme.getColor(Theme.key_actionBarDefault, resourceProvider))); + statusEmoji.setColor(Theme.getColor(Theme.key_profile_verifiedBackground, resourcesProvider)); } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/PhotoViewer.java b/TMessagesProj/src/main/java/org/telegram/ui/PhotoViewer.java index 55b7da1ca25..091747b1e05 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/PhotoViewer.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/PhotoViewer.java @@ -5981,7 +5981,7 @@ protected boolean customBlur() { } @Override - protected boolean ignoreTouches() { + protected boolean ignoreTouches(float x, float y) { return !keyboardShown && currentEditMode != EDIT_MODE_NONE; } @@ -6096,7 +6096,7 @@ public void setTranslationY(float translationY) { if (captionEdit.isCaptionOverLimit()) { AndroidUtilities.shakeViewSpring(captionEdit.limitTextView, shiftDp = -shiftDp); BotWebViewVibrationEffect.APP_ERROR.vibrate(); - if (!MessagesController.getInstance(currentAccount).premiumLocked && MessagesController.getInstance(currentAccount).captionLengthLimitPremium > captionEdit.getCodePointCount()) { + if (!MessagesController.getInstance(currentAccount).premiumFeaturesBlocked() && MessagesController.getInstance(currentAccount).captionLengthLimitPremium > captionEdit.getCodePointCount()) { showCaptionLimitBulletin(containerView); } return; @@ -10734,6 +10734,10 @@ public void onAnimationEnd(Animator animation) { updateMinMax(scale); padImageForHorizontalInsets = true; containerView.invalidate(); + + if (placeProvider == null || !placeProvider.closeKeyboard()) { + makeFocusable(); + } } }); imageMoveAnimation.start(); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/PopupNotificationActivity.java b/TMessagesProj/src/main/java/org/telegram/ui/PopupNotificationActivity.java index 9ec379ef491..6e2123e6879 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/PopupNotificationActivity.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/PopupNotificationActivity.java @@ -395,7 +395,7 @@ public void didPressAttachButton() { } @Override - public void needStartRecordVideo(int state, boolean notify, int scheduleDate) { + public void needStartRecordVideo(int state, boolean notify, int scheduleDate, int ttl) { } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/PremiumPreviewFragment.java b/TMessagesProj/src/main/java/org/telegram/ui/PremiumPreviewFragment.java index ad26d33ac63..8448234d9cd 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/PremiumPreviewFragment.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/PremiumPreviewFragment.java @@ -984,7 +984,7 @@ public boolean isSwipeBackEnabled(MotionEvent event) { @Override public boolean onFragmentCreate() { - if (getMessagesController().premiumLocked) { + if (getMessagesController().premiumFeaturesBlocked()) { return false; } NotificationCenter.getGlobalInstance().addObserver(this, NotificationCenter.billingProductDetailsUpdated); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/PrivacySettingsActivity.java b/TMessagesProj/src/main/java/org/telegram/ui/PrivacySettingsActivity.java index b0a23d6fd98..405d743e679 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/PrivacySettingsActivity.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/PrivacySettingsActivity.java @@ -647,7 +647,7 @@ private void updateRows(boolean notify) { callsRow = rowCount++; groupsRow = rowCount++; groupsDetailRow = -1; - if (!getMessagesController().premiumLocked || getUserConfig().isPremium()) { + if (!getMessagesController().premiumFeaturesBlocked() || getUserConfig().isPremium()) { voicesRow = rowCount++; } else { voicesRow = -1; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/ProfileActivity.java b/TMessagesProj/src/main/java/org/telegram/ui/ProfileActivity.java index a311fca7c98..7e8006925c2 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/ProfileActivity.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/ProfileActivity.java @@ -214,6 +214,7 @@ import org.telegram.ui.Components.Premium.PremiumGradient; import org.telegram.ui.Components.Premium.PremiumPreviewBottomSheet; import org.telegram.ui.Components.Premium.ProfilePremiumCell; +import org.telegram.ui.Components.Premium.boosts.UserSelectorBottomSheet; import org.telegram.ui.Components.ProfileGalleryView; import org.telegram.ui.Components.RLottieDrawable; import org.telegram.ui.Components.RLottieImageView; @@ -271,6 +272,7 @@ public class ProfileActivity extends BaseFragment implements NotificationCenter. private SearchAdapter searchAdapter; private SimpleTextView[] nameTextView = new SimpleTextView[2]; private String nameTextViewRightDrawableContentDescription = null; + private String nameTextViewRightDrawable2ContentDescription = null; private SimpleTextView[] onlineTextView = new SimpleTextView[4]; private AudioPlayerAlert.ClippingTextViewSwitcher mediaCounterTextView; private RLottieImageView writeButton; @@ -290,6 +292,7 @@ public class ProfileActivity extends BaseFragment implements NotificationCenter. private StickerEmptyView emptyView; private boolean sharedMediaLayoutAttached; private SharedMediaLayout.SharedMediaPreloader sharedMediaPreloader; + private boolean preloadedChannelEmojiStatuses; private View blurredView; @@ -529,6 +532,7 @@ public void setAlpha(int a) { private int addToGroupButtonRow; private int addToGroupInfoRow; private int premiumRow; + private int premiumGiftingRow; private int premiumSectionsRow; private int settingsTimerRow; @@ -3611,6 +3615,8 @@ public void openExceptions() { onWriteButtonClick(); } else if (position == premiumRow) { presentFragment(new PremiumPreviewFragment("settings")); + } else if (position == premiumGiftingRow) { + UserSelectorBottomSheet.open(); } else { processOnClickOrPress(position, view, x, y); } @@ -3655,7 +3661,8 @@ public boolean onItemClick(View view, int position) { BuildVars.DEBUG_VERSION ? (SharedConfig.useSurfaceInStories ? "back to TextureView in stories" : "use SurfaceView in stories") : null, BuildVars.DEBUG_PRIVATE_VERSION ? (SharedConfig.photoViewerBlur ? "do not blur in photoviewer" : "blur in photoviewer") : null, !SharedConfig.payByInvoice ? "Enable Invoice Payment" : "Disable Invoice Payment", - BuildVars.DEBUG_PRIVATE_VERSION ? "Update Attach Bots" : null + BuildVars.DEBUG_PRIVATE_VERSION ? "Update Attach Bots" : null, + BuildVars.DEBUG_PRIVATE_VERSION ? ((SharedConfig.forceLessData ? "Disable using less data" : "Use less data on stories") + (ApplicationLoader.isConnectionSlow() ? " (connection is already slow)" : "")) : null }; builder.setItems(items, (dialog, which) -> { @@ -3675,13 +3682,21 @@ public boolean onItemClick(View view, int position) { sharedPreferences.edit().putBoolean("logsEnabled", BuildVars.LOGS_ENABLED).commit(); updateRowsIds(); listAdapter.notifyDataSetChanged(); + if (BuildVars.LOGS_ENABLED) { + FileLog.d("app start time = " + ApplicationLoader.startTime); + try { + FileLog.d("buildVersion = " + ApplicationLoader.applicationContext.getPackageManager().getPackageInfo(ApplicationLoader.applicationContext.getPackageName(), 0).versionCode); + } catch (Exception e) { + FileLog.e(e); + } + } } else if (which == 5) { SharedConfig.toggleInappCamera(); } else if (which == 6) { getMessagesStorage().clearSentMedia(); SharedConfig.setNoSoundHintShowed(false); SharedPreferences.Editor editor = MessagesController.getGlobalMainSettings().edit(); - editor.remove("archivehint").remove("proximityhint").remove("archivehint_l").remove("speedhint").remove("gifhint").remove("reminderhint").remove("soundHint").remove("themehint").remove("bganimationhint").remove("filterhint").remove("n_0").remove("storyprvhint").remove("storyhint").remove("storyhint2").remove("storydualhint").remove("storysvddualhint").remove("stories_camera").remove("dualcam").remove("dualmatrix").remove("dual_available").remove("archivehint").remove("askNotificationsAfter").remove("askNotificationsDuration").remove("viewoncehint").remove("taptostorysoundhint").commit(); + editor.remove("archivehint").remove("proximityhint").remove("archivehint_l").remove("speedhint").remove("gifhint").remove("reminderhint").remove("soundHint").remove("themehint").remove("bganimationhint").remove("filterhint").remove("n_0").remove("storyprvhint").remove("storyhint").remove("storyhint2").remove("storydualhint").remove("storysvddualhint").remove("stories_camera").remove("dualcam").remove("dualmatrix").remove("dual_available").remove("archivehint").remove("askNotificationsAfter").remove("askNotificationsDuration").remove("viewoncehint").remove("taptostorysoundhint").remove("nothanos").remove("voiceoncehint").commit(); MessagesController.getEmojiSettings(currentAccount).edit().remove("featured_hidden").remove("emoji_featured_hidden").commit(); SharedConfig.textSelectionHintShows = 0; SharedConfig.lockRecordAudioVideoHint = 0; @@ -3701,6 +3716,7 @@ public boolean onItemClick(View view, int position) { getNotificationCenter().postNotificationName(NotificationCenter.newSuggestionsAvailable); RestrictedLanguagesSelectActivity.cleanup(); PersistColorPalette.getInstance(currentAccount).cleanup(); + getMessagesController().getMainSettings().edit().remove("peerColors").remove("profilePeerColors").remove("boostingappearance").commit(); } else if (which == 7) { VoIPHelper.showCallDebugSettings(getParentActivity()); } else if (which == 8) { @@ -3889,6 +3905,8 @@ protected void onSend(LongSparseArray dids, int count, TLRPC.TL_fo SharedConfig.togglePaymentByInvoice(); } else if (which == 26) { getMediaDataController().loadAttachMenuBots(false, true); + } else if (which == 27) { + SharedConfig.setForceLessData(!SharedConfig.forceLessData); } }); builder.setNegativeButton(LocaleController.getString("Cancel", R.string.Cancel), null); @@ -4187,7 +4205,7 @@ protected void dispatchDraw(Canvas canvas) { if (avatarBig != null) { return; } - if (isTopic && !getMessagesController().premiumLocked) { + if (isTopic && !getMessagesController().premiumFeaturesBlocked()) { ArrayList topics = getMessagesController().getTopicsController().getTopics(chatId); if (topics != null) { TLRPC.TL_forumTopic currentTopic = null; @@ -4311,8 +4329,17 @@ protected void setCustomAvatarProgress(float progress) { @Override public void onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo info) { super.onInitializeAccessibilityNodeInfo(info); - if (isFocusable() && nameTextViewRightDrawableContentDescription != null) { - info.setText(getText() + ", " + nameTextViewRightDrawableContentDescription); + if (isFocusable() && (nameTextViewRightDrawableContentDescription != null || nameTextViewRightDrawable2ContentDescription != null)) { + StringBuilder s = new StringBuilder(getText()); + if (nameTextViewRightDrawable2ContentDescription != null) { + if (s.length() > 0) s.append(", "); + s.append(nameTextViewRightDrawable2ContentDescription); + } + if (nameTextViewRightDrawableContentDescription != null) { + if (s.length() > 0) s.append(", "); + s.append(nameTextViewRightDrawableContentDescription); + } + info.setText(s); } } }; @@ -4397,7 +4424,7 @@ public void setTextColor(int color) { protected TextView createTextView() { TextView textView = new TextView(context); textView.setTextColor(getThemedColor(Theme.key_player_actionBarSubtitle)); - textView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 14); + textView.setTextSize(TypedValue.COMPLEX_UNIT_PX, AndroidUtilities.dp(14)); textView.setSingleLine(true); textView.setEllipsize(TextUtils.TruncateAt.END); textView.setGravity(Gravity.LEFT); @@ -4405,7 +4432,7 @@ protected TextView createTextView() { } }; mediaCounterTextView.setAlpha(0.0f); - avatarContainer2.addView(mediaCounterTextView, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.LEFT | Gravity.TOP, 118, 0, 8, 0)); + avatarContainer2.addView(mediaCounterTextView, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.LEFT | Gravity.TOP, 118.33f, -2, 8, 0)); storyView = new ProfileStoriesView(context, currentAccount, getDialogId(), avatarContainer, avatarImage, resourcesProvider) { @Override protected void onTap(StoryViewer.PlaceProvider provider) { @@ -5039,32 +5066,57 @@ public void showStatusSelect() { xoff = MathUtils.clamp(ecenter - popupWidth / 2, 0, AndroidUtilities.displaySize.x - popupWidth); ecenter -= xoff; boolean hasEmoji = emojiStatusDrawable[1] != null && emojiStatusDrawable[1].getDrawable() instanceof AnimatedEmojiDrawable; - SelectAnimatedEmojiDialog popupLayout = new SelectAnimatedEmojiDialog(this, getContext(), true, Math.max(0, ecenter), SelectAnimatedEmojiDialog.TYPE_EMOJI_STATUS, true, resourcesProvider, topMarginDp) { + SelectAnimatedEmojiDialog popupLayout = new SelectAnimatedEmojiDialog(this, getContext(), true, Math.max(0, ecenter), currentChat == null ? SelectAnimatedEmojiDialog.TYPE_EMOJI_STATUS : SelectAnimatedEmojiDialog.TYPE_EMOJI_STATUS_CHANNEL, true, resourcesProvider, topMarginDp) { @Override protected void onEmojiSelected(View emojiView, Long documentId, TLRPC.Document document, Integer until) { - TLRPC.TL_account_updateEmojiStatus req = new TLRPC.TL_account_updateEmojiStatus(); - if (documentId == null) { - req.emoji_status = new TLRPC.TL_emojiStatusEmpty(); - } else if (until != null) { - req.emoji_status = new TLRPC.TL_emojiStatusUntil(); - ((TLRPC.TL_emojiStatusUntil) req.emoji_status).document_id = documentId; - ((TLRPC.TL_emojiStatusUntil) req.emoji_status).until = until; + TLObject request; + if (currentChat == null) { + TLRPC.TL_account_updateEmojiStatus req = new TLRPC.TL_account_updateEmojiStatus(); + if (documentId == null) { + req.emoji_status = new TLRPC.TL_emojiStatusEmpty(); + } else if (until != null) { + req.emoji_status = new TLRPC.TL_emojiStatusUntil(); + ((TLRPC.TL_emojiStatusUntil) req.emoji_status).document_id = documentId; + ((TLRPC.TL_emojiStatusUntil) req.emoji_status).until = until; + } else { + req.emoji_status = new TLRPC.TL_emojiStatus(); + ((TLRPC.TL_emojiStatus) req.emoji_status).document_id = documentId; + } + TLRPC.User user = MessagesController.getInstance(currentAccount).getUser(UserConfig.getInstance(currentAccount).getClientUserId()); + if (user != null) { + user.emoji_status = req.emoji_status; + MessagesController.getInstance(currentAccount).updateEmojiStatusUntilUpdate(user.id, user.emoji_status); + NotificationCenter.getInstance(currentAccount).postNotificationName(NotificationCenter.userEmojiStatusUpdated, user); + } + request = req; } else { - req.emoji_status = new TLRPC.TL_emojiStatus(); - ((TLRPC.TL_emojiStatus) req.emoji_status).document_id = documentId; - } - TLRPC.User user = MessagesController.getInstance(currentAccount).getUser(UserConfig.getInstance(currentAccount).getClientUserId()); - if (user != null) { - user.emoji_status = req.emoji_status; - MessagesController.getInstance(currentAccount).updateEmojiStatusUntilUpdate(user.id, user.emoji_status); - NotificationCenter.getInstance(currentAccount).postNotificationName(NotificationCenter.userEmojiStatusUpdated, user); + TLRPC.TL_channels_updateEmojiStatus req = new TLRPC.TL_channels_updateEmojiStatus(); + req.channel = MessagesController.getInputChannel(currentChat); + if (documentId == null) { + req.emoji_status = new TLRPC.TL_emojiStatusEmpty(); + } else if (until != null) { + req.emoji_status = new TLRPC.TL_emojiStatusUntil(); + ((TLRPC.TL_emojiStatusUntil) req.emoji_status).document_id = documentId; + ((TLRPC.TL_emojiStatusUntil) req.emoji_status).until = until; + } else { + req.emoji_status = new TLRPC.TL_emojiStatus(); + ((TLRPC.TL_emojiStatus) req.emoji_status).document_id = documentId; + } + if (currentChat != null) { + currentChat.emoji_status = req.emoji_status; + MessagesController.getInstance(currentAccount).updateEmojiStatusUntilUpdate(-currentChat.id, currentChat.emoji_status); + NotificationCenter.getInstance(currentAccount).postNotificationName(NotificationCenter.updateInterfaces, MessagesController.UPDATE_MASK_EMOJI_STATUS); + } + request = req; } for (int a = 0; a < 2; ++a) { if (emojiStatusDrawable[a] != null) { - if (documentId == null) { + if (documentId == null && currentChat == null) { emojiStatusDrawable[a].set(getPremiumCrossfadeDrawable(a), true); - } else { + } else if (documentId != null) { emojiStatusDrawable[a].set(documentId, true); + } else { + emojiStatusDrawable[a].set((Drawable) null, true); } } } @@ -5073,11 +5125,7 @@ protected void onEmojiSelected(View emojiView, Long documentId, TLRPC.Document d } updateEmojiStatusDrawableColor(); updateEmojiStatusEffectPosition(); - ConnectionsManager.getInstance(currentAccount).sendRequest(req, (res, err) -> { - if (!(res instanceof TLRPC.TL_boolTrue)) { - // TODO: reject - } - }); + ConnectionsManager.getInstance(currentAccount).sendRequest(request, null); if (popup[0] != null) { selectAnimatedEmojiDialog = null; popup[0].dismiss(); @@ -5085,8 +5133,8 @@ protected void onEmojiSelected(View emojiView, Long documentId, TLRPC.Document d } }; TLRPC.User user = getMessagesController().getUser(userId); - if (user != null && user.emoji_status instanceof TLRPC.TL_emojiStatusUntil && ((TLRPC.TL_emojiStatusUntil) user.emoji_status).until > (int) (System.currentTimeMillis() / 1000)) { - popupLayout.setExpireDateHint(((TLRPC.TL_emojiStatusUntil) user.emoji_status).until); + if (user != null) { + popupLayout.setExpireDateHint(DialogObject.getEmojiStatusUntil(user.emoji_status)); } popupLayout.setSelected(emojiStatusDrawable[1] != null && emojiStatusDrawable[1].getDrawable() instanceof AnimatedEmojiDrawable ? ((AnimatedEmojiDrawable) emojiStatusDrawable[1].getDrawable()).getDocumentId() : null); popupLayout.setSaveState(3); @@ -5838,7 +5886,7 @@ public void setValue(ActionBar object, float value) { verifiedDrawable[0].setColorFilter(AndroidUtilities.getOffsetColor(color1, color2, value, 1.0f), PorterDuff.Mode.MULTIPLY); } if (verifiedDrawable[1] != null) { - color1 = applyPeerColor(getThemedColor(Theme.key_profile_verifiedBackground)); + color1 = peerColor != null ? Theme.adaptHSV(ColorUtils.blendARGB(peerColor.getColor2(), peerColor.hasColor6(Theme.isCurrentThemeDark()) ? peerColor.getColor5() : peerColor.getColor3(), .4f), +.1f, Theme.isCurrentThemeDark() ? -.1f : -.08f) : getThemedColor(Theme.key_profile_verifiedBackground); color2 = getThemedColor(Theme.key_player_actionBarTitle); verifiedDrawable[1].setColorFilter(AndroidUtilities.getOffsetColor(color1, color2, value, 1.0f), PorterDuff.Mode.MULTIPLY); } @@ -5849,7 +5897,7 @@ public void setValue(ActionBar object, float value) { verifiedCheckDrawable[0].setColorFilter(AndroidUtilities.getOffsetColor(color1, color2, value, 1.0f), PorterDuff.Mode.MULTIPLY); } if (verifiedCheckDrawable[1] != null) { - color1 = applyPeerColor(getThemedColor(Theme.key_profile_verifiedCheck)); + color1 = peerColor != null ? Color.WHITE : applyPeerColor(getThemedColor(Theme.key_profile_verifiedCheck)); color2 = getThemedColor(Theme.key_windowBackgroundWhite); verifiedCheckDrawable[1].setColorFilter(AndroidUtilities.getOffsetColor(color1, color2, value, 1.0f), PorterDuff.Mode.MULTIPLY); } @@ -6723,7 +6771,7 @@ public void didReceivedNotification(int id, int account, final Object... args) { } } } else if (chatId != 0) { - if ((mask & MessagesController.UPDATE_MASK_CHAT) != 0 || (mask & MessagesController.UPDATE_MASK_CHAT_AVATAR) != 0 || (mask & MessagesController.UPDATE_MASK_CHAT_NAME) != 0 || (mask & MessagesController.UPDATE_MASK_CHAT_MEMBERS) != 0 || (mask & MessagesController.UPDATE_MASK_STATUS) != 0) { + if ((mask & MessagesController.UPDATE_MASK_CHAT) != 0 || (mask & MessagesController.UPDATE_MASK_CHAT_AVATAR) != 0 || (mask & MessagesController.UPDATE_MASK_CHAT_NAME) != 0 || (mask & MessagesController.UPDATE_MASK_CHAT_MEMBERS) != 0 || (mask & MessagesController.UPDATE_MASK_STATUS) != 0 || (mask & MessagesController.UPDATE_MASK_EMOJI_STATUS) != 0) { if ((mask & MessagesController.UPDATE_MASK_CHAT) != 0) { updateListAnimated(true); } else { @@ -7691,6 +7739,7 @@ private void updateRowsIds() { notificationRow = -1; languageRow = -1; premiumRow = -1; + premiumGiftingRow = -1; premiumSectionsRow = -1; privacyRow = -1; dataRow = -1; @@ -7816,8 +7865,13 @@ private void updateRowsIds() { devicesRow = rowCount++; languageRow = rowCount++; devicesSectionRow = rowCount++; - if (!getMessagesController().premiumLocked) { + if (!getMessagesController().premiumFeaturesBlocked()) { premiumRow = rowCount++; + } + if (!getMessagesController().premiumPurchaseBlocked()) { + premiumGiftingRow = rowCount++; + } + if (premiumRow >= 0 || premiumGiftingRow >= 0) { premiumSectionsRow = rowCount++; } helpHeaderRow = rowCount++; @@ -8066,6 +8120,12 @@ private Drawable getVerifiedCrossfadeDrawable(int a) { if (verifiedCrossfadeDrawable[a] == null) { verifiedDrawable[a] = Theme.profile_verifiedDrawable.getConstantState().newDrawable().mutate(); verifiedCheckDrawable[a] = Theme.profile_verifiedCheckDrawable.getConstantState().newDrawable().mutate(); + if (a == 1 && peerColor != null) { + int color = Theme.adaptHSV(ColorUtils.blendARGB(peerColor.getColor2(), peerColor.hasColor6(Theme.isCurrentThemeDark()) ? peerColor.getColor5() : peerColor.getColor3(), .4f), +.1f, Theme.isCurrentThemeDark() ? -.1f : -.08f); + verifiedDrawable[1].setColorFilter(AndroidUtilities.getOffsetColor(color, getThemedColor(Theme.key_player_actionBarTitle), mediaHeaderAnimationProgress, 1.0f), PorterDuff.Mode.MULTIPLY); + color = Color.WHITE; + verifiedCheckDrawable[1].setColorFilter(AndroidUtilities.getOffsetColor(color, getThemedColor(Theme.key_windowBackgroundWhite), mediaHeaderAnimationProgress, 1.0f), PorterDuff.Mode.MULTIPLY); + } verifiedCrossfadeDrawable[a] = new CrossfadeDrawable( new CombinedDrawable(verifiedDrawable[a], verifiedCheckDrawable[a]), ContextCompat.getDrawable(getParentActivity(), R.drawable.verified_profile) @@ -8276,55 +8336,61 @@ private void updateProfileData(boolean reload) { onlineTextView[a].setText(newString2); } Drawable leftIcon = currentEncryptedChat != null ? getLockIconDrawable() : null; - Drawable rightIcon = null; boolean rightIconIsPremium = false, rightIconIsStatus = false; nameTextView[a].setRightDrawableOutside(a == 0); if (a == 0) { if (user.scam || user.fake) { - rightIcon = getScamDrawable(user.scam ? 0 : 1); - nameTextViewRightDrawableContentDescription = LocaleController.getString("ScamMessage", R.string.ScamMessage); + nameTextView[a].setRightDrawable2(getScamDrawable(user.scam ? 0 : 1)); + nameTextViewRightDrawable2ContentDescription = LocaleController.getString(R.string.ScamMessage); } else if (user.verified) { - rightIcon = getVerifiedCrossfadeDrawable(a); - nameTextViewRightDrawableContentDescription = LocaleController.getString("AccDescrVerified", R.string.AccDescrVerified); - } else if (user.emoji_status instanceof TLRPC.TL_emojiStatus || user.emoji_status instanceof TLRPC.TL_emojiStatusUntil && ((TLRPC.TL_emojiStatusUntil) user.emoji_status).until > (int) (System.currentTimeMillis() / 1000)) { + nameTextView[a].setRightDrawable2(getVerifiedCrossfadeDrawable(a)); + nameTextViewRightDrawable2ContentDescription = LocaleController.getString(R.string.AccDescrVerified); + } else if (getMessagesController().isDialogMuted(dialogId != 0 ? dialogId : userId, topicId)) { + nameTextView[a].setRightDrawable2(getThemedDrawable(Theme.key_drawable_muteIconDrawable)); + nameTextViewRightDrawable2ContentDescription = LocaleController.getString(R.string.NotificationsMuted); + } else { + nameTextView[a].setRightDrawable2(null); + nameTextViewRightDrawable2ContentDescription = null; + } + if (user.emoji_status instanceof TLRPC.TL_emojiStatus || user.emoji_status instanceof TLRPC.TL_emojiStatusUntil && ((TLRPC.TL_emojiStatusUntil) user.emoji_status).until > (int) (System.currentTimeMillis() / 1000)) { rightIconIsStatus = true; rightIconIsPremium = false; - rightIcon = getEmojiStatusDrawable(user.emoji_status, false, false, a); - nameTextViewRightDrawableContentDescription = LocaleController.getString("AccDescrPremium", R.string.AccDescrPremium); + nameTextView[a].setRightDrawable(getEmojiStatusDrawable(user.emoji_status, false, false, a)); + nameTextViewRightDrawableContentDescription = LocaleController.getString(R.string.AccDescrPremium); } else if (getMessagesController().isPremiumUser(user)) { rightIconIsStatus = false; rightIconIsPremium = true; - rightIcon = getEmojiStatusDrawable(null, false, false, a); - nameTextViewRightDrawableContentDescription = LocaleController.getString("AccDescrPremium", R.string.AccDescrPremium); - } else if (getMessagesController().isDialogMuted(dialogId != 0 ? dialogId : userId, topicId)) { - rightIcon = getThemedDrawable(Theme.key_drawable_muteIconDrawable); - nameTextViewRightDrawableContentDescription = LocaleController.getString("NotificationsMuted", R.string.NotificationsMuted); + nameTextView[a].setRightDrawable(getEmojiStatusDrawable(null, false, false, a)); + nameTextViewRightDrawableContentDescription = LocaleController.getString( R.string.AccDescrPremium); } else { - rightIcon = null; + nameTextView[a].setRightDrawable(null); nameTextViewRightDrawableContentDescription = null; } } else if (a == 1) { if (user.scam || user.fake) { - rightIcon = getScamDrawable(user.scam ? 0 : 1); + nameTextView[a].setRightDrawable2(getScamDrawable(user.scam ? 0 : 1)); } else if (user.verified) { - rightIcon = getVerifiedCrossfadeDrawable(a); - } else if (user.emoji_status instanceof TLRPC.TL_emojiStatus || user.emoji_status instanceof TLRPC.TL_emojiStatusUntil && ((TLRPC.TL_emojiStatusUntil) user.emoji_status).until > (int) (System.currentTimeMillis() / 1000)) { + nameTextView[a].setRightDrawable2(getVerifiedCrossfadeDrawable(a)); + } else { + nameTextView[a].setRightDrawable2(null); + } + if (user.emoji_status instanceof TLRPC.TL_emojiStatus || user.emoji_status instanceof TLRPC.TL_emojiStatusUntil && ((TLRPC.TL_emojiStatusUntil) user.emoji_status).until > (int) (System.currentTimeMillis() / 1000)) { rightIconIsStatus = true; rightIconIsPremium = false; - rightIcon = getEmojiStatusDrawable(user.emoji_status, true, true, a); + nameTextView[a].setRightDrawable(getEmojiStatusDrawable(user.emoji_status, true, true, a)); } else if (getMessagesController().isPremiumUser(user)) { rightIconIsStatus = false; rightIconIsPremium = true; - rightIcon = getEmojiStatusDrawable(null, true, true, a); + nameTextView[a].setRightDrawable(getEmojiStatusDrawable(null, true, true, a)); + } else { + nameTextView[a].setRightDrawable(null); } } nameTextView[a].setLeftDrawable(leftIcon); - nameTextView[a].setRightDrawable(rightIcon); if (a == 1 && (rightIconIsStatus || rightIconIsPremium)) { nameTextView[a].setRightDrawableOutside(true); } if (user.self && getMessagesController().isPremiumUser(user)) { - final SimpleTextView textView = nameTextView[a]; nameTextView[a].setRightDrawableOnClick(v -> { showStatusSelect(); }); @@ -8426,7 +8492,7 @@ private void updateProfileData(boolean reload) { final int colorId = ChatObject.getProfileColorId(chat); MessagesController.PeerColors peerColors = MessagesController.getInstance(currentAccount).profilePeerColors; final MessagesController.PeerColor wasPeerColor = peerColor; - peerColor = null; + peerColor = peerColors == null ? null : peerColors.getColor(colorId); if (wasPeerColor != peerColor) { updatedPeerColor(); } @@ -8541,24 +8607,45 @@ private void updateProfileData(boolean reload) { } nameTextView[a].setLeftDrawable(null); nameTextView[a].setRightDrawableOutside(a == 0); + nameTextView[a].setRightDrawableOnClick(null); if (a != 0) { if (chat.scam || chat.fake) { - nameTextView[a].setRightDrawable(getScamDrawable(chat.scam ? 0 : 1)); + nameTextView[a].setRightDrawable2(getScamDrawable(chat.scam ? 0 : 1)); nameTextViewRightDrawableContentDescription = LocaleController.getString("ScamMessage", R.string.ScamMessage); } else if (chat.verified) { - nameTextView[a].setRightDrawable(getVerifiedCrossfadeDrawable(a)); + nameTextView[a].setRightDrawable2(getVerifiedCrossfadeDrawable(a)); nameTextViewRightDrawableContentDescription = LocaleController.getString("AccDescrVerified", R.string.AccDescrVerified); } else { - nameTextView[a].setRightDrawable(null); + nameTextView[a].setRightDrawable2(null); nameTextViewRightDrawableContentDescription = null; } + if (DialogObject.getEmojiStatusDocumentId(chat.emoji_status) != 0) { + nameTextView[a].setRightDrawable(getEmojiStatusDrawable(chat.emoji_status, true, false, a)); + nameTextView[a].setRightDrawableOutside(true); + nameTextViewRightDrawableContentDescription = null; + if (ChatObject.canChangeChatInfo(chat)) { + nameTextView[a].setRightDrawableOnClick(v -> { + showStatusSelect(); + }); + if (preloadedChannelEmojiStatuses) { + preloadedChannelEmojiStatuses = true; + getMediaDataController().loadRestrictedStatusEmojis(); + } + } + } } else { if (chat.scam || chat.fake) { - nameTextView[a].setRightDrawable(getScamDrawable(chat.scam ? 0 : 1)); + nameTextView[a].setRightDrawable2(getScamDrawable(chat.scam ? 0 : 1)); } else if (chat.verified) { - nameTextView[a].setRightDrawable(getVerifiedCrossfadeDrawable(a)); + nameTextView[a].setRightDrawable2(getVerifiedCrossfadeDrawable(a)) ; } else if (getMessagesController().isDialogMuted(-chatId, topicId)) { - nameTextView[a].setRightDrawable(getThemedDrawable(Theme.key_drawable_muteIconDrawable)); + nameTextView[a].setRightDrawable2(getThemedDrawable(Theme.key_drawable_muteIconDrawable)); + } else { + nameTextView[a].setRightDrawable2(null); + } + if (DialogObject.getEmojiStatusDocumentId(chat.emoji_status) != 0) { + nameTextView[a].setRightDrawable(getEmojiStatusDrawable(chat.emoji_status, false, false, a)); + nameTextView[a].setRightDrawableOutside(true); } else { nameTextView[a].setRightDrawable(null); } @@ -8673,6 +8760,16 @@ private void updatedPeerColor() { final int iconColor = peerColor != null ? Color.WHITE : getThemedColor(Theme.key_actionBarDefaultIcon); actionBar.setItemsColor(ColorUtils.blendARGB(iconColor, color, avatarAnimationProgress), false); } + if (verifiedDrawable[1] != null) { + int color1 = peerColor != null ? Theme.adaptHSV(ColorUtils.blendARGB(peerColor.getColor2(), peerColor.hasColor6(Theme.isCurrentThemeDark()) ? peerColor.getColor5() : peerColor.getColor3(), .4f), +.1f, Theme.isCurrentThemeDark() ? -.1f : -.08f) : getThemedColor(Theme.key_profile_verifiedBackground); + int color2 = getThemedColor(Theme.key_player_actionBarTitle); + verifiedDrawable[1].setColorFilter(AndroidUtilities.getOffsetColor(color1, color2, mediaHeaderAnimationProgress, 1.0f), PorterDuff.Mode.MULTIPLY); + } + if (verifiedCheckDrawable[1] != null) { + int color1 = peerColor != null ? Color.WHITE : applyPeerColor(getThemedColor(Theme.key_profile_verifiedCheck)); + int color2 = getThemedColor(Theme.key_windowBackgroundWhite); + verifiedCheckDrawable[1].setColorFilter(AndroidUtilities.getOffsetColor(color1, color2, mediaHeaderAnimationProgress, 1.0f), PorterDuff.Mode.MULTIPLY); + } if (nameTextView[1] != null) { nameTextView[1].setTextColor(ColorUtils.blendARGB(peerColor != null ? Color.WHITE : getThemedColor(Theme.key_profile_title), Color.WHITE, currentExpandAnimatorValue)); } @@ -8800,7 +8897,7 @@ private void createActionBarMenu(boolean animated) { otherItem.addSubItem(delete_contact, R.drawable.msg_delete, LocaleController.getString("DeleteContact", R.string.DeleteContact)); } if (!UserObject.isDeleted(user) && !isBot && currentEncryptedChat == null && !userBlocked && userId != 333000 && userId != 777000 && userId != 42777) { - if (!user.premium && !BuildVars.IS_BILLING_UNAVAILABLE && !user.self && userInfo != null && !getMessagesController().premiumLocked && !userInfo.premium_gifts.isEmpty()) { + if (!user.premium && !BuildVars.IS_BILLING_UNAVAILABLE && !user.self && userInfo != null && !getMessagesController().premiumFeaturesBlocked() && !userInfo.premium_gifts.isEmpty()) { otherItem.addSubItem(gift_premium, R.drawable.msg_gift_premium, LocaleController.getString(R.string.GiftPremium)); } otherItem.addSubItem(start_secret_chat, R.drawable.msg_secret, LocaleController.getString("StartEncryptedChat", R.string.StartEncryptedChat)); @@ -10149,7 +10246,10 @@ public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) { textCell.setTextAndIcon(LocaleController.getString("AddToGroupOrChannel", R.string.AddToGroupOrChannel), R.drawable.msg_groups_create, false); textCell.setColors(Theme.key_windowBackgroundWhiteBlueIcon, Theme.key_windowBackgroundWhiteBlueButton); } else if (position == premiumRow) { - textCell.setTextAndIcon(LocaleController.getString("TelegramPremium", R.string.TelegramPremium), new AnimatedEmojiDrawable.WrapSizeDrawable(PremiumGradient.getInstance().premiumStarMenuDrawable, AndroidUtilities.dp(24), AndroidUtilities.dp(24)), false); + textCell.setTextAndIcon(LocaleController.getString("TelegramPremium", R.string.TelegramPremium), new AnimatedEmojiDrawable.WrapSizeDrawable(PremiumGradient.getInstance().premiumStarMenuDrawable, AndroidUtilities.dp(24), AndroidUtilities.dp(24)), true); + textCell.setImageLeft(23); + } else if (position == premiumGiftingRow) { + textCell.setTextAndIcon(TextCell.applyNewSpan(LocaleController.getString("GiftPremiumGifting", R.string.GiftPremiumGifting)), R.drawable.menu_gift, false); textCell.setImageLeft(23); } textCell.valueTextView.setTextColor(applyPeerColor(getThemedColor(Theme.key_windowBackgroundWhiteValueText), false)); @@ -10356,7 +10456,7 @@ public boolean isEnabled(RecyclerView.ViewHolder holder) { position == questionRow || position == devicesRow || position == filtersRow || position == stickersRow || position == faqRow || position == policyRow || position == sendLogsRow || position == sendLastLogsRow || position == clearLogsRow || position == switchBackendRow || position == setAvatarRow || - position == addToGroupButtonRow || position == premiumRow || position == liteModeRow; + position == addToGroupButtonRow || position == premiumRow || position == premiumGiftingRow || position == liteModeRow; } if (holder.itemView instanceof UserCell) { UserCell userCell = (UserCell) holder.itemView; @@ -10398,7 +10498,7 @@ public int getItemViewType(int position) { position == questionRow || position == devicesRow || position == filtersRow || position == stickersRow || position == faqRow || position == policyRow || position == sendLogsRow || position == sendLastLogsRow || position == clearLogsRow || position == switchBackendRow || position == setAvatarRow || position == addToGroupButtonRow || - position == addToContactsRow || position == liteModeRow) { + position == addToContactsRow || position == liteModeRow || position == premiumGiftingRow) { return VIEW_TYPE_TEXT; } else if (position == notificationsDividerRow) { return VIEW_TYPE_DIVIDER; @@ -10834,7 +10934,7 @@ private SearchResult[] onCreateSearchArray() { } private boolean isPremiumFeatureAvailable(int feature) { - if (getMessagesController().premiumLocked && !getUserConfig().isPremium()) { + if (getMessagesController().premiumFeaturesBlocked() && !getUserConfig().isPremium()) { return false; } @@ -11635,6 +11735,7 @@ public void fillPositions(SparseIntArray sparseIntArray) { put(++pointer, languageRow, sparseIntArray); put(++pointer, premiumRow, sparseIntArray); put(++pointer, premiumSectionsRow, sparseIntArray); + put(++pointer, premiumGiftingRow, sparseIntArray); put(++pointer, privacyRow, sparseIntArray); put(++pointer, dataRow, sparseIntArray); put(++pointer, liteModeRow, sparseIntArray); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/SecretMediaViewer.java b/TMessagesProj/src/main/java/org/telegram/ui/SecretMediaViewer.java index a05cf72cdbe..d91a3200a3a 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/SecretMediaViewer.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/SecretMediaViewer.java @@ -152,7 +152,6 @@ protected boolean drawChild(Canvas canvas, View child, long drawingTime) { return child != aspectRatioFrameLayout && super.drawChild(canvas, child, drawingTime); } - @Override protected void onAttachedToWindow() { super.onAttachedToWindow(); @@ -1387,9 +1386,7 @@ private void showSecretHint() { secretHint.setInnerPadding(12, 7, 11, 7); secretHint.setIconMargin(2); secretHint.setIconTranslate(0, 0); - RLottieDrawable icon = new RLottieDrawable(R.raw.fire_on, "" + R.raw.fire_on, dp(34), dp(34)); - icon.start(); - secretHint.setIcon(icon); + secretHint.setIcon(R.raw.fire_on); secretHint.show(); MessagesController.getGlobalMainSettings().edit().putInt("viewoncehint", MessagesController.getGlobalMainSettings().getInt("viewoncehint", 0) + 1).commit(); } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/SecretVoicePlayer.java b/TMessagesProj/src/main/java/org/telegram/ui/SecretVoicePlayer.java new file mode 100644 index 00000000000..31b0ec364bf --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/SecretVoicePlayer.java @@ -0,0 +1,623 @@ +package org.telegram.ui; + +import static org.telegram.messenger.AndroidUtilities.dp; + +import android.animation.Animator; +import android.animation.AnimatorListenerAdapter; +import android.animation.StateListAnimator; +import android.animation.ValueAnimator; +import android.app.Dialog; +import android.content.Context; +import android.content.DialogInterface; +import android.graphics.Bitmap; +import android.graphics.BitmapShader; +import android.graphics.Canvas; +import android.graphics.ColorMatrix; +import android.graphics.ColorMatrixColorFilter; +import android.graphics.Insets; +import android.graphics.Matrix; +import android.graphics.Paint; +import android.graphics.PixelFormat; +import android.graphics.Rect; +import android.graphics.RectF; +import android.graphics.Shader; +import android.graphics.SurfaceTexture; +import android.net.Uri; +import android.os.Build; +import android.os.Bundle; +import android.view.Gravity; +import android.view.KeyEvent; +import android.view.View; +import android.view.ViewGroup; +import android.view.Window; +import android.view.WindowInsets; +import android.view.WindowManager; +import android.widget.FrameLayout; +import android.widget.TextView; + +import androidx.annotation.NonNull; +import androidx.core.view.WindowInsetsCompat; + +import com.google.android.exoplayer2.ExoPlayer; +import com.google.android.exoplayer2.util.Log; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.FileLoader; +import org.telegram.messenger.LocaleController; +import org.telegram.messenger.MediaController; +import org.telegram.messenger.MessageObject; +import org.telegram.messenger.MessagesController; +import org.telegram.messenger.R; +import org.telegram.messenger.UserConfig; +import org.telegram.messenger.UserObject; +import org.telegram.messenger.Utilities; +import org.telegram.tgnet.TLRPC; +import org.telegram.ui.ActionBar.AlertDialog; +import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.Cells.ChatMessageCell; +import org.telegram.ui.Components.AnimatedFloat; +import org.telegram.ui.Components.AudioVisualizerDrawable; +import org.telegram.ui.Components.CubicBezierInterpolator; +import org.telegram.ui.Components.LayoutHelper; +import org.telegram.ui.Components.ScaleStateListAnimator; +import org.telegram.ui.Components.SeekBar; +import org.telegram.ui.Components.SeekBarWaveform; +import org.telegram.ui.Components.Text; +import org.telegram.ui.Components.ThanosEffect; +import org.telegram.ui.Components.VideoPlayer; +import org.telegram.ui.Stories.recorder.HintView2; +import org.telegram.ui.Stories.recorder.StoryRecorder; + +import java.io.File; + +public class SecretVoicePlayer extends Dialog { + + public final Context context; + +// WindowManager windowManager; +// private final WindowManager.LayoutParams windowLayoutParams; + private FrameLayout windowView; + private FrameLayout containerView; + + private ThanosEffect thanosEffect; + + private final Rect insets = new Rect(); + private Bitmap blurBitmap; + private BitmapShader blurBitmapShader; + private Paint blurBitmapPaint; + private Matrix blurMatrix; + + private boolean open; + private float openProgress; + private float openProgressLinear; + + private VideoPlayer player; + + private HintView2 hintView; + private TextView closeButton; + + public SecretVoicePlayer(Context context) { + super(context, R.style.TransparentDialog); + this.context = context; + + windowView = new FrameLayout(context) { + @Override + protected void dispatchDraw(Canvas canvas) { + if (openProgress > 0 && blurBitmapPaint != null) { + blurMatrix.reset(); + final float s = (float) getWidth() / blurBitmap.getWidth(); + blurMatrix.postScale(s, s); + blurBitmapShader.setLocalMatrix(blurMatrix); + + blurBitmapPaint.setAlpha((int) (0xFF * openProgress)); + canvas.drawRect(0, 0, getWidth(), getHeight(), blurBitmapPaint); + } + if (setCellInvisible && cell != null) { + cell.setVisibility(View.INVISIBLE); + setCellInvisible = false; + } + super.dispatchDraw(canvas); + } + + @Override + public boolean dispatchKeyEventPreIme(KeyEvent event) { + if (event != null && event.getKeyCode() == KeyEvent.KEYCODE_BACK && event.getAction() == KeyEvent.ACTION_UP) { + dismiss(); + return true; + } + return super.dispatchKeyEventPreIme(event); + } + + @Override + protected void onLayout(boolean changed, int left, int top, int right, int bottom) { + super.onLayout(changed, left, top, right, bottom); + setupTranslation(); + } + }; + windowView.setOnClickListener(v -> { + if (closeAction == null) { + dismiss(); + } + }); + containerView = new FrameLayout(context) { + @Override + protected boolean drawChild(Canvas canvas, View child, long drawingTime) { + if (child == myCell || child == hintView) { + canvas.save(); + canvas.clipRect(0, AndroidUtilities.lerp(clipTop, 0, openProgress), getWidth(), AndroidUtilities.lerp(clipBottom, getHeight(), openProgress)); + boolean r = super.drawChild(canvas, child, drawingTime); + canvas.restore(); + return r; + } + return super.drawChild(canvas, child, drawingTime); + } + }; + windowView.addView(containerView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT, Gravity.FILL)); + + if (Build.VERSION.SDK_INT >= 21) { + windowView.setFitsSystemWindows(true); + windowView.setOnApplyWindowInsetsListener(new View.OnApplyWindowInsetsListener() { + @NonNull + @Override + public WindowInsets onApplyWindowInsets(@NonNull View v, @NonNull WindowInsets insets) { + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) { + Insets r = insets.getInsets(WindowInsetsCompat.Type.displayCutout() | WindowInsetsCompat.Type.systemBars()); + SecretVoicePlayer.this.insets.set(r.left, r.top, r.right, r.bottom); + } else { + SecretVoicePlayer.this.insets.set(insets.getStableInsetLeft(), insets.getStableInsetTop(), insets.getStableInsetRight(), insets.getStableInsetBottom()); + } + containerView.setPadding(SecretVoicePlayer.this.insets.left, SecretVoicePlayer.this.insets.top, SecretVoicePlayer.this.insets.right, SecretVoicePlayer.this.insets.bottom); + windowView.requestLayout(); + if (Build.VERSION.SDK_INT >= 30) { + return WindowInsets.CONSUMED; + } else { + return insets.consumeSystemWindowInsets(); + } + } + }); + } + + } + + private void prepareBlur(View withoutView) { + if (withoutView != null) { + withoutView.setVisibility(View.INVISIBLE); + } + AndroidUtilities.makeGlobalBlurBitmap(bitmap -> { + if (withoutView != null) { + withoutView.setVisibility(View.VISIBLE); + } + blurBitmap = bitmap; + + blurBitmapPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + blurBitmapPaint.setShader(blurBitmapShader = new BitmapShader(blurBitmap, Shader.TileMode.CLAMP, Shader.TileMode.CLAMP)); + ColorMatrix colorMatrix = new ColorMatrix(); + AndroidUtilities.adjustSaturationColorMatrix(colorMatrix, Theme.isCurrentThemeDark() ? .05f : +.25f); + AndroidUtilities.adjustBrightnessColorMatrix(colorMatrix, Theme.isCurrentThemeDark() ? -.02f : -.04f); + blurBitmapPaint.setColorFilter(new ColorMatrixColorFilter(colorMatrix)); + blurMatrix = new Matrix(); + }, 14); + } + + @Override + protected void onCreate(Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + + Window window = getWindow(); + window.setWindowAnimations(R.style.DialogNoAnimation); + setContentView(windowView, new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT)); + + WindowManager.LayoutParams params = window.getAttributes(); + params.width = ViewGroup.LayoutParams.MATCH_PARENT; + params.height = ViewGroup.LayoutParams.MATCH_PARENT; + params.gravity = Gravity.FILL; + params.dimAmount = 0; + params.flags &= ~WindowManager.LayoutParams.FLAG_DIM_BEHIND; + params.softInputMode = WindowManager.LayoutParams.SOFT_INPUT_ADJUST_NOTHING; + params.flags |= WindowManager.LayoutParams.FLAG_ALT_FOCUSABLE_IM; + if (Build.VERSION.SDK_INT >= 21) { + params.flags |= WindowManager.LayoutParams.FLAG_LAYOUT_IN_SCREEN | + WindowManager.LayoutParams.FLAG_LAYOUT_INSET_DECOR | + WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS | + WindowManager.LayoutParams.FLAG_TRANSLUCENT_NAVIGATION; + } + params.flags |= WindowManager.LayoutParams.FLAG_SECURE; + params.flags |= WindowManager.LayoutParams.FLAG_FULLSCREEN; + params.flags |= WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON; + if (Build.VERSION.SDK_INT >= 28) { + params.layoutInDisplayCutoutMode = WindowManager.LayoutParams.LAYOUT_IN_DISPLAY_CUTOUT_MODE_SHORT_EDGES; + } + window.setAttributes(params); + + windowView.setSystemUiVisibility(View.SYSTEM_UI_FLAG_LAYOUT_STABLE | View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN | View.SYSTEM_UI_FLAG_FULLSCREEN); + AndroidUtilities.setLightNavigationBar(windowView, !Theme.isCurrentThemeDark()); + } + + private Theme.ResourcesProvider resourcesProvider; + private MessageObject messageObject; + private ChatMessageCell myCell; + private ChatMessageCell cell; + + private float tx, ty; + private boolean hasTranslation; + private float dtx, dty; + private boolean hasDestTranslation; + private void setupTranslation() { + if (hasTranslation || windowView.getWidth() <= 0) return; + if (cell != null) { + int[] loc = new int[2]; + cell.getLocationOnScreen(loc); + tx = loc[0] - insets.left - (windowView.getWidth() - insets.left - insets.right - cell.getWidth()) / 2f; + ty = loc[1] - insets.top - (windowView.getHeight() - insets.top - insets.bottom - cell.getHeight()) / 2f; + if (!hasDestTranslation) { + hasDestTranslation = true; + dtx = 0; + float cy = Utilities.clamp(loc[1] + cell.getHeight() / 2f, windowView.getHeight() * .7f, windowView.getHeight() * .3f); + dty = cy - cell.getHeight() / 2f - (windowView.getHeight() - cell.getHeight()) / 2f; + dty = AndroidUtilities.lerp(0, dty, .78f); + } + updateTranslation(); + } else { + tx = ty = 0; + } + hasTranslation = true; + } + private void updateTranslation() { + if (thanosEffect != null) return; + myCell.setTranslationX(AndroidUtilities.lerp(tx, dtx, openProgress)); + myCell.setTranslationY(AndroidUtilities.lerp(ty, dty, openProgress)); + if (hintView != null) { + hintView.setTranslationX(AndroidUtilities.lerp(tx, dtx, openProgress)); + hintView.setTranslationY(AndroidUtilities.lerp(ty, dty, openProgress)); + } + } + + private float clipTop = 0, clipBottom = 0; + + private AudioVisualizerDrawable audioVisualizerDrawable; + private boolean setCellInvisible; + private Runnable openAction, closeAction; + public void setCell(ChatMessageCell messageCell, Runnable openAction, Runnable closeAction) { + this.openAction = openAction; + this.closeAction = closeAction; + if (myCell != null) { + containerView.removeView(myCell); + myCell = null; + } + cell = messageCell; + messageObject = cell != null ? cell.getMessageObject() : null; + resourcesProvider = cell != null ? cell.getResourcesProvider() : null; + if (cell != null) { + + clipTop = messageCell.parentBoundsTop; + clipBottom = messageCell.parentBoundsBottom; + if (messageCell.getParent() instanceof View) { + View parent = (View) messageCell.getParent(); + clipTop += parent.getY(); + clipBottom += parent.getY(); + } + + myCell = new ChatMessageCell(getContext(), false, null, cell.getResourcesProvider()) { + @Override + public void setPressed(boolean pressed) {} + }; + myCell.setDelegate(new ChatMessageCell.ChatMessageCellDelegate() { + @Override + public boolean canPerformActions() { + return false; + } + }); + myCell.setMessageObject(messageObject, cell.getCurrentMessagesGroup(), cell.pinnedBottom, cell.pinnedTop); + audioVisualizerDrawable = new AudioVisualizerDrawable(); + audioVisualizerDrawable.setParentView(myCell); + myCell.overrideAudioVisualizer(audioVisualizerDrawable); + if (myCell.getSeekBarWaveform() != null) { + myCell.getSeekBarWaveform().setExplosionRate(openProgressLinear); + } + hasTranslation = false; + containerView.addView(myCell, new FrameLayout.LayoutParams(cell.getWidth(), cell.getHeight(), Gravity.CENTER)); + } + + MediaController.getInstance().pauseByRewind(); + + if (player != null) { + player.pause(); + player.releasePlayer(true); + player = null; + } + if (cell != null && cell.getMessageObject() != null) { + File file = FileLoader.getInstance(cell.getMessageObject().currentAccount).getPathToAttach(cell.getMessageObject().getDocument()); + if (file == null || !file.exists()) { + file = FileLoader.getInstance(cell.getMessageObject().currentAccount).getPathToMessage(cell.getMessageObject().messageOwner); + } + if ((file == null || !file.exists()) && (cell.getMessageObject().messageOwner.attachPath != null)) { + file = new File(cell.getMessageObject().messageOwner.attachPath); + } + if (file == null) { + return; + } + player = new VideoPlayer(); + player.setDelegate(new VideoPlayer.VideoPlayerDelegate() { + @Override + public void onStateChanged(boolean playWhenReady, int playbackState) { + if (playbackState == ExoPlayer.STATE_ENDED) { + dismiss(); + } else { + AndroidUtilities.cancelRunOnUIThread(checkTimeRunnable); + AndroidUtilities.runOnUIThread(checkTimeRunnable, 16); + } + } + + @Override + public void onError(VideoPlayer player, Exception e) { + + } + + @Override + public void onVideoSizeChanged(int width, int height, int unappliedRotationDegrees, float pixelWidthHeightRatio) { + + } + + @Override + public void onRenderedFirstFrame() { + + } + + @Override + public void onSurfaceTextureUpdated(SurfaceTexture surfaceTexture) { + + } + + @Override + public boolean onSurfaceDestroyed(SurfaceTexture surfaceTexture) { + return false; + } + }); + player.setAudioVisualizerDelegate(new VideoPlayer.AudioVisualizerDelegate() { + @Override + public void onVisualizerUpdate(boolean playing, boolean animate, float[] values) { + audioVisualizerDrawable.setWaveform(playing, animate, values); + } + + @Override + public boolean needUpdate() { + return audioVisualizerDrawable.getParentView() != null; + } + }); + player.preparePlayer(Uri.fromFile(file), "other"); + player.play(); + } + + if (hintView != null) { + containerView.removeView(hintView); + hintView = null; + } + final boolean isOut = messageObject != null && messageObject.isOutOwner(); + if (messageObject != null && messageObject.getDialogId() != UserConfig.getInstance(messageObject.currentAccount).getClientUserId()) { + hintView = new HintView2(context, HintView2.DIRECTION_BOTTOM); + hintView.setMultilineText(true); + if (isOut) { + String name = ""; + long did = messageObject.getDialogId(); + if (did > 0) { + TLRPC.User user = MessagesController.getInstance(messageObject.currentAccount).getUser(did); + if (user != null) { + name = UserObject.getFirstName(user); + } + } else { + TLRPC.Chat chat = MessagesController.getInstance(messageObject.currentAccount).getChat(-did); + if (chat != null) { + name = chat.title; + } + } + hintView.setText(AndroidUtilities.replaceTags(LocaleController.formatString(R.string.VoiceOnceOutHint, name))); + } else { + hintView.setText(AndroidUtilities.replaceTags(LocaleController.getString(R.string.VoiceOnceHint))); + } + hintView.setRounding(12); + hintView.setPadding(dp(!isOut && !cell.pinnedBottom ? 6 : 0), 0, 0, 0); + hintView.setJointPx(0, dp(34)); + hintView.setTextSize(14); + hintView.setMaxWidthPx(HintView2.cutInFancyHalf(hintView.getText(), hintView.getTextPaint())); + containerView.addView(hintView, LayoutHelper.createFrame((int) (cell.getWidth() / AndroidUtilities.density * .6f), 150, Gravity.CENTER, (cell.getWidth() * -(1f - .6f) / 2f + cell.getBoundsLeft()) / AndroidUtilities.density + 1, -75 - (cell.getHeight() / AndroidUtilities.density) / 2f - 8, 0, 0)); + hintView.show(); + } + + if (closeButton != null) { + containerView.removeView(closeButton); + closeButton = null; + } + closeButton = new TextView(context); + closeButton.setTextColor(0xFFFFFFFF); + closeButton.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); + if (Theme.isCurrentThemeDark()) { + closeButton.setBackground(Theme.createSimpleSelectorRoundRectDrawable(64, 0x20ffffff, 0x33ffffff)); + } else { + closeButton.setBackground(Theme.createSimpleSelectorRoundRectDrawable(64, 0x2e000000, 0x44000000)); + } + closeButton.setPadding(dp(12), dp(6), dp(12), dp(6)); + ScaleStateListAnimator.apply(closeButton); + closeButton.setText(LocaleController.getString(isOut ? R.string.VoiceOnceClose : R.string.VoiceOnceDeleteClose)); + closeButton.setOnClickListener(v -> { + dismiss(); + }); + containerView.addView(closeButton, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.BOTTOM | Gravity.CENTER, 0, 0, 0, 18)); + + if (!isOut && myCell != null && myCell.getMessageObject() != null && myCell.getMessageObject().messageOwner != null) { + myCell.getMessageObject().messageOwner.media_unread = false; + myCell.invalidate(); + } + } + + @Override + public void show() { + super.show(); + + prepareBlur(cell); + setCellInvisible = true; + animateOpenTo(open = true, null); + + if (this.openAction != null) { + AndroidUtilities.runOnUIThread(this.openAction); + this.openAction = null; + } + } + + private Runnable checkTimeRunnable = this::checkTime; + private void checkTime() { + if (player == null) { + return; + } + float progress = player.getCurrentPosition() / (float) player.getDuration(); + if (myCell != null) { + myCell.overrideDuration((player.getDuration() - player.getCurrentPosition()) / 1000L); + myCell.updatePlayingMessageProgress(); + SeekBarWaveform seekBarWaveform = myCell.getSeekBarWaveform(); + if (seekBarWaveform != null) { + seekBarWaveform.explodeAt(progress); + } + } + + if (player.isPlaying()) { + AndroidUtilities.cancelRunOnUIThread(checkTimeRunnable); + AndroidUtilities.runOnUIThread(checkTimeRunnable, 16); + } + } + + public boolean isShown() { + return !dismissing; + } + + private boolean dismissing = false; + + private AlertDialog backDialog; + @Override + public void onBackPressed() { + if (backDialog != null) { + backDialog.dismiss(); + backDialog = null; + return; + } + if (!dismissing && messageObject != null && !messageObject.isOutOwner()) { + backDialog = new AlertDialog.Builder(getContext(), resourcesProvider) + .setTitle(LocaleController.getString(R.string.VoiceOnceCloseTitle)) + .setMessage(LocaleController.getString(R.string.VoiceOnceCloseMessage)) + .setPositiveButton(LocaleController.getString(R.string.Continue), (di, w) -> { + if (backDialog != null) { + backDialog.dismiss(); + } + }) + .setNegativeButton(LocaleController.getString(R.string.Delete), (di, w) -> { + if (backDialog != null) { + backDialog.dismiss(); + backDialog = null; + } + dismiss(); + }) + .create(); + backDialog.show(); + TextView button = (TextView) backDialog.getButton(DialogInterface.BUTTON_NEGATIVE); + if (button != null) { + button.setTextColor(Theme.getColor(Theme.key_text_RedBold)); + } + return; + } + super.onBackPressed(); + } + + @Override + public void dismiss() { + if (dismissing) return; + if (backDialog != null) { + backDialog.dismiss(); + backDialog = null; + } + dismissing = true; + if (hintView != null) { + hintView.hide(); + } + if (player != null) { + player.pause(); + player.releasePlayer(true); + player = null; + } + if (myCell != null && myCell.getSeekBarWaveform() != null) { + myCell.getSeekBarWaveform().setExplosionRate(openProgressLinear); + } + hasTranslation = false; + setupTranslation(); + animateOpenTo(open = false, () -> { + if (thanosEffect == null) { + AndroidUtilities.runOnUIThread(super::dismiss); + if (cell != null) { + cell.invalidate(); + cell.setVisibility(View.VISIBLE); + } + } + + MediaController.getInstance().tryResumePausedAudio(); + }); + windowView.invalidate(); + + if (this.closeAction != null) { + AndroidUtilities.runOnUIThread(this.closeAction); + this.closeAction = null; + + thanosEffect = new ThanosEffect(context, null); + windowView.addView(thanosEffect, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT, Gravity.FILL)); + thanosEffect.animate(myCell, () -> { + super.dismiss(); + if (cell != null) { + cell.setVisibility(View.VISIBLE); + cell.invalidate(); + } + }); + + WindowManager.LayoutParams params = getWindow().getAttributes(); + params.flags |= WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE; + getWindow().setAttributes(params); + } + } + + private ValueAnimator openAnimator; + private void animateOpenTo(boolean open, Runnable after) { + if (openAnimator != null) { + openAnimator.cancel(); + } + setupTranslation(); + openAnimator = ValueAnimator.ofFloat(openProgressLinear, open ? 1 : 0); + openAnimator.addUpdateListener(anm -> { + openProgressLinear = (float) anm.getAnimatedValue(); + openProgress = open ? CubicBezierInterpolator.EASE_OUT_QUINT.getInterpolation(openProgressLinear) : 1f - CubicBezierInterpolator.EASE_OUT_QUINT.getInterpolation(1f - openProgressLinear); + windowView.invalidate(); + containerView.invalidate(); + updateTranslation(); + if (closeButton != null) { + closeButton.setAlpha(openProgress); + } + if (myCell != null && myCell.getSeekBarWaveform() != null) { + myCell.getSeekBarWaveform().setExplosionRate((open ? CubicBezierInterpolator.EASE_OUT : CubicBezierInterpolator.EASE_IN).getInterpolation(Utilities.clamp(openProgressLinear * 1.25f, 1f, 0f))); + } + }); + openAnimator.addListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + openProgress = openProgressLinear = open ? 1 : 0; + windowView.invalidate(); + containerView.invalidate(); + updateTranslation(); + if (closeButton != null) { + closeButton.setAlpha(openProgress); + } + if (myCell != null && myCell.getSeekBarWaveform() != null) { + myCell.getSeekBarWaveform().setExplosionRate(openProgressLinear); + } + if (after != null) { + after.run(); + } + } + }); + openAnimator.setDuration(!open && closeAction == null ? 330 : 520); + openAnimator.start(); + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/SelectAnimatedEmojiDialog.java b/TMessagesProj/src/main/java/org/telegram/ui/SelectAnimatedEmojiDialog.java index dbd631ca051..b81c9b8d3cc 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/SelectAnimatedEmojiDialog.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/SelectAnimatedEmojiDialog.java @@ -145,9 +145,11 @@ public class SelectAnimatedEmojiDialog extends FrameLayout implements Notificati public static final int TYPE_CHAT_REACTIONS = 6; public final static int TYPE_SET_REPLY_ICON_BOTTOM = 7; public final static int TYPE_EXPANDABLE_REACTIONS = 8; + public final static int TYPE_EMOJI_STATUS_CHANNEL = 9; + public final static int TYPE_EMOJI_STATUS_CHANNEL_TOP = 10; public boolean isBottom() { - return type == TYPE_SET_REPLY_ICON; + return type == TYPE_SET_REPLY_ICON || type == TYPE_EMOJI_STATUS_CHANNEL_TOP; } private final int SPAN_COUNT_FOR_EMOJI = 8; @@ -432,7 +434,7 @@ public SelectAnimatedEmojiDialog(BaseFragment baseFragment, Context context, boo this.type = type; this.includeEmpty = includeEmpty; this.baseFragment = baseFragment; - this.includeHint = MessagesController.getGlobalMainSettings().getInt("emoji" + (type == TYPE_EMOJI_STATUS ? "status" : "reaction") + "usehint", 0) < 3; + this.includeHint = MessagesController.getGlobalMainSettings().getInt("emoji" + (type == TYPE_EMOJI_STATUS || type == TYPE_EMOJI_STATUS_CHANNEL || type == TYPE_EMOJI_STATUS_CHANNEL_TOP ? "status" : "reaction") + "usehint", 0) < 3; this.accentColor = accentColor; selectorPaint.setColor(Theme.getColor(Theme.key_listSelector, resourcesProvider)); @@ -445,7 +447,7 @@ public SelectAnimatedEmojiDialog(BaseFragment baseFragment, Context context, boo setFocusableInTouchMode(true); - if (type == TYPE_EMOJI_STATUS || type == TYPE_SET_DEFAULT_REACTION || type == TYPE_SET_REPLY_ICON || type == TYPE_SET_REPLY_ICON_BOTTOM) { + if (type == TYPE_EMOJI_STATUS || type == TYPE_EMOJI_STATUS_CHANNEL || type == TYPE_EMOJI_STATUS_CHANNEL_TOP || type == TYPE_SET_DEFAULT_REACTION || type == TYPE_SET_REPLY_ICON || type == TYPE_SET_REPLY_ICON_BOTTOM) { topMarginDp = topPaddingDp; setPadding(AndroidUtilities.dp(4), AndroidUtilities.dp(4), AndroidUtilities.dp(4), AndroidUtilities.dp(4)); setOnTouchListener((v, e) -> { @@ -558,12 +560,12 @@ public void getOutline(View view, Outline outline) { } } - if (type == TYPE_EMOJI_STATUS || type == TYPE_SET_DEFAULT_REACTION || type == TYPE_SET_REPLY_ICON) { + if (type == TYPE_EMOJI_STATUS || type == TYPE_EMOJI_STATUS_CHANNEL || type == TYPE_EMOJI_STATUS_CHANNEL_TOP || type == TYPE_SET_DEFAULT_REACTION || type == TYPE_SET_REPLY_ICON) { contentView.setPadding(AndroidUtilities.dp(8), AndroidUtilities.dp(8), AndroidUtilities.dp(8), AndroidUtilities.dp(8)); } contentView.addView(backgroundView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT)); - addView(contentView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT, Gravity.FILL, 0, type == TYPE_EMOJI_STATUS || type == TYPE_SET_DEFAULT_REACTION || type == TYPE_SET_REPLY_ICON_BOTTOM ? 6 + topMarginDp : 0, 0, isBottom() ? 6 + topMarginDp : 0)); + addView(contentView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT, Gravity.FILL, 0, type == TYPE_EMOJI_STATUS || type == TYPE_EMOJI_STATUS_CHANNEL || type == TYPE_SET_DEFAULT_REACTION || type == TYPE_SET_REPLY_ICON_BOTTOM ? 6 + topMarginDp : 0, 0, isBottom() ? 6 + topMarginDp : 0)); if (bubbleX != null) { bubble2View = new View(context) { @@ -580,7 +582,7 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { addView(bubble2View, LayoutHelper.createFrame(17, 9, (isBottom() ? Gravity.BOTTOM : Gravity.TOP) | Gravity.LEFT, bubbleX / AndroidUtilities.density + (bubbleRight ? -25 : 10), isBottom() ? 0 : 5 + topMarginDp, 0, isBottom() ? 5 + topMarginDp + 9 : 0)); } - boolean showSettings = baseFragment != null && type != TYPE_TOPIC_ICON && type != TYPE_CHAT_REACTIONS && type != TYPE_SET_REPLY_ICON && type != TYPE_SET_REPLY_ICON_BOTTOM && type != TYPE_AVATAR_CONSTRUCTOR && shouldDrawBackground; + boolean showSettings = baseFragment != null && type != TYPE_TOPIC_ICON && type != TYPE_CHAT_REACTIONS && type != TYPE_SET_REPLY_ICON && type != TYPE_SET_REPLY_ICON_BOTTOM && type != TYPE_AVATAR_CONSTRUCTOR && type != TYPE_EMOJI_STATUS_CHANNEL && type != TYPE_EMOJI_STATUS_CHANNEL_TOP && shouldDrawBackground; for (int i = 0; i < 2; i++) { EmojiTabsStrip emojiTabs = new EmojiTabsStrip(context, resourcesProvider, true, false, true, type, showSettings ? () -> { search(null, false, false); @@ -684,7 +686,7 @@ public void onScrolled(int dx, int dy) { updateTabsPosition(layoutManager.findFirstCompletelyVisibleItemPosition()); } updateSearchBox(); - AndroidUtilities.updateViewVisibilityAnimated(emojiTabsShadow, emojiGridView.computeVerticalScrollOffset() != 0 || type == TYPE_EMOJI_STATUS || type == TYPE_REACTIONS || type == TYPE_CHAT_REACTIONS, 1f, true); + AndroidUtilities.updateViewVisibilityAnimated(emojiTabsShadow, emojiGridView.computeVerticalScrollOffset() != 0 || type == TYPE_EMOJI_STATUS || type == TYPE_EMOJI_STATUS_CHANNEL_TOP || type == TYPE_REACTIONS || type == TYPE_CHAT_REACTIONS, 1f, true); invalidateParent(); } @@ -818,7 +820,7 @@ public void onScrolled(int dx, int dy) { TextView emptyViewText = new TextView(context); if (type == TYPE_AVATAR_CONSTRUCTOR) { emptyViewText.setText(LocaleController.getString("NoEmojiOrStickersFound", R.string.NoEmojiOrStickersFound)); - } else if (type == TYPE_EMOJI_STATUS) { + } else if (type == TYPE_EMOJI_STATUS || type == TYPE_EMOJI_STATUS_CHANNEL || type == TYPE_EMOJI_STATUS_CHANNEL_TOP) { emptyViewText.setText(LocaleController.getString("NoEmojiFound", R.string.NoEmojiFound)); } else if (type == TYPE_REACTIONS || type == TYPE_SET_DEFAULT_REACTION) { emptyViewText.setText(LocaleController.getString("NoReactionsFound", R.string.NoReactionsFound)); @@ -919,7 +921,7 @@ public boolean onItemClick(View view, int position, float x, float y) { invalidateParent(); return true; } - if (view instanceof ImageViewEmoji && ((ImageViewEmoji) view).span != null && type == TYPE_EMOJI_STATUS) { + if (view instanceof ImageViewEmoji && ((ImageViewEmoji) view).span != null && (type == TYPE_EMOJI_STATUS || type == TYPE_EMOJI_STATUS_CHANNEL || type == TYPE_EMOJI_STATUS_CHANNEL_TOP)) { SelectStatusDurationDialog dialog = selectStatusDateDialog = new SelectStatusDurationDialog(context, dismiss, SelectAnimatedEmojiDialog.this, (ImageViewEmoji) view, resourcesProvider) { @Override protected boolean getOutBounds(Rect rect) { @@ -1097,6 +1099,7 @@ protected void onSettings() { } public void setExpireDateHint(int date) { + if (date <= 0) return; includeHint = true; hintExpireDate = date; updateRows(true, false); @@ -1227,7 +1230,7 @@ private void updateSearchBox() { private Drawable getPremiumStar() { if (premiumStar == null) { - if (type == TYPE_SET_REPLY_ICON || type == TYPE_SET_REPLY_ICON_BOTTOM) { + if (type == TYPE_SET_REPLY_ICON || type == TYPE_EMOJI_STATUS_CHANNEL || type == TYPE_EMOJI_STATUS_CHANNEL_TOP || type == TYPE_SET_REPLY_ICON_BOTTOM) { premiumStar = ApplicationLoader.applicationContext.getResources().getDrawable(R.drawable.msg_filled_blocked).mutate(); } else { premiumStar = ApplicationLoader.applicationContext.getResources().getDrawable(R.drawable.msg_settings_premium).mutate(); @@ -2222,7 +2225,7 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { textView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 13); if (type == TYPE_TOPIC_ICON) { textView.setText(LocaleController.getString("SelectTopicIconHint", R.string.SelectTopicIconHint)); - } else if (type == TYPE_EMOJI_STATUS) { + } else if (type == TYPE_EMOJI_STATUS || type == TYPE_EMOJI_STATUS_CHANNEL || type == TYPE_EMOJI_STATUS_CHANNEL_TOP) { textView.setText(LocaleController.getString("EmojiLongtapHint", R.string.EmojiLongtapHint)); } else { textView.setText(LocaleController.getString("ReactionsLongtapHint", R.string.ReactionsLongtapHint)); @@ -3113,7 +3116,7 @@ public void createPremiumLockView() { public void onEmojiClick(View view, AnimatedEmojiSpan span) { incrementHintUse(); - if (span == null || type == TYPE_EMOJI_STATUS && selectedDocumentIds.contains(span.documentId)) { + if (span == null || (type == TYPE_EMOJI_STATUS || type == TYPE_EMOJI_STATUS_CHANNEL || type == TYPE_EMOJI_STATUS_CHANNEL_TOP) && selectedDocumentIds.contains(span.documentId)) { onEmojiSelected(view, null, null, null); } else { TLRPC.TL_emojiStatus status = new TLRPC.TL_emojiStatus(); @@ -3121,10 +3124,10 @@ public void onEmojiClick(View view, AnimatedEmojiSpan span) { TLRPC.Document document = span.document == null ? AnimatedEmojiDrawable.findDocument(currentAccount, span.documentId) : span.document; if (view instanceof ImageViewEmoji) { - if (type == TYPE_EMOJI_STATUS) { + if (type == TYPE_EMOJI_STATUS || type == TYPE_EMOJI_STATUS_CHANNEL || type == TYPE_EMOJI_STATUS_CHANNEL_TOP) { MediaDataController.getInstance(currentAccount).pushRecentEmojiStatus(status); } - if (type == TYPE_EMOJI_STATUS || type == TYPE_SET_DEFAULT_REACTION) { + if (type == TYPE_EMOJI_STATUS || type == TYPE_EMOJI_STATUS_CHANNEL || type == TYPE_EMOJI_STATUS_CHANNEL_TOP || type == TYPE_SET_DEFAULT_REACTION) { animateEmojiSelect((ImageViewEmoji) view, () -> { onEmojiSelected(view, span.documentId, document, null); }); @@ -3141,7 +3144,7 @@ private void incrementHintUse() { if (type == TYPE_SET_DEFAULT_REACTION) { return; } - final String key = "emoji" + (type == TYPE_EMOJI_STATUS ? "status" : "reaction") + "usehint"; + final String key = "emoji" + (type == TYPE_EMOJI_STATUS || type == TYPE_EMOJI_STATUS_CHANNEL || type == TYPE_EMOJI_STATUS_CHANNEL_TOP ? "status" : "reaction") + "usehint"; final int value = MessagesController.getGlobalMainSettings().getInt(key, 0); if (value <= 3) { MessagesController.getGlobalMainSettings().edit().putInt(key, value + 1).apply(); @@ -3163,13 +3166,21 @@ public void preload(int type, int account) { MediaDataController.getInstance(account).checkStickers(MediaDataController.TYPE_EMOJIPACKS); if (type == TYPE_REACTIONS || type == TYPE_SET_DEFAULT_REACTION || type == TYPE_CHAT_REACTIONS) { MediaDataController.getInstance(account).checkReactions(); + } else if (type == TYPE_EMOJI_STATUS_CHANNEL || type == TYPE_EMOJI_STATUS_CHANNEL_TOP) { + if (MessagesController.getInstance(account).getMainSettings().getBoolean("resetemojipacks", true)) { + MediaDataController.getInstance(account).loadStickers(MediaDataController.TYPE_EMOJIPACKS, false, false); + MessagesController.getInstance(account).getMainSettings().edit().putBoolean("resetemojipacks", false).commit(); + } + MediaDataController.getInstance(account).fetchEmojiStatuses(2, false); + MediaDataController.getInstance(account).loadRestrictedStatusEmojis(); + MediaDataController.getInstance(account).getStickerSet(new TLRPC.TL_inputStickerSetEmojiChannelDefaultStatuses(), false); } else if (type == TYPE_EMOJI_STATUS) { MediaDataController.getInstance(account).fetchEmojiStatuses(0, true); MediaDataController.getInstance(account).getStickerSet(new TLRPC.TL_inputStickerSetEmojiDefaultStatuses(), false); } else if (type == TYPE_TOPIC_ICON) { MediaDataController.getInstance(account).checkDefaultTopicIcons(); } else if (type == TYPE_AVATAR_CONSTRUCTOR) { - MediaDataController.getInstance(currentAccount).loadRecents(MediaDataController.TYPE_IMAGE, false, true, false); + MediaDataController.getInstance(account).loadRecents(MediaDataController.TYPE_IMAGE, false, true, false); MediaDataController.getInstance(account).checkStickers(MediaDataController.TYPE_IMAGE); } } @@ -3233,7 +3244,7 @@ private void updateRows(boolean updateEmojipacks, boolean animated, boolean diff stickerSets.clear(); recentStickers.clear(); - if ((!installedEmojipacks.isEmpty() || type == TYPE_AVATAR_CONSTRUCTOR) && type != TYPE_SET_REPLY_ICON && type != TYPE_SET_REPLY_ICON_BOTTOM && type != TYPE_CHAT_REACTIONS && type != TYPE_EXPANDABLE_REACTIONS) { + if ((!installedEmojipacks.isEmpty() || type == TYPE_AVATAR_CONSTRUCTOR) && type != TYPE_SET_REPLY_ICON && type != TYPE_SET_REPLY_ICON_BOTTOM && type != TYPE_CHAT_REACTIONS && type != TYPE_EXPANDABLE_REACTIONS && type != TYPE_EMOJI_STATUS_CHANNEL && type != TYPE_EMOJI_STATUS_CHANNEL_TOP) { searchRow = totalCount++; rowHashCodes.add(9L); } else { @@ -3329,6 +3340,14 @@ private void updateRows(boolean updateEmojipacks, boolean animated, boolean diff rowHashCodes.add(6L); } + HashSet restricted = null; + if (type == TYPE_EMOJI_STATUS_CHANNEL || type == TYPE_EMOJI_STATUS_CHANNEL_TOP) { + TLRPC.TL_emojiList emojiList = MediaDataController.getInstance(currentAccount).restrictedStatusEmojis; + if (emojiList != null) { + restricted = new HashSet<>(); + restricted.addAll(emojiList.document_id); + } + } if (recentReactionsToSet != null) { topReactionsStartRow = totalCount; ArrayList tmp = new ArrayList<>(recentReactionsToSet); @@ -3373,9 +3392,9 @@ private void updateRows(boolean updateEmojipacks, boolean animated, boolean diff totalCount += recentReactions.size(); recentReactionsEndRow = totalCount; } - } else if (type == TYPE_EMOJI_STATUS) { + } else if (type == TYPE_EMOJI_STATUS || type == TYPE_EMOJI_STATUS_CHANNEL || type == TYPE_EMOJI_STATUS_CHANNEL_TOP) { ArrayList recentEmojiStatuses = MediaDataController.getInstance(currentAccount).getRecentEmojiStatuses(); - TLRPC.TL_messages_stickerSet defaultSet = MediaDataController.getInstance(currentAccount).getStickerSet(new TLRPC.TL_inputStickerSetEmojiDefaultStatuses(), true); + TLRPC.TL_messages_stickerSet defaultSet = MediaDataController.getInstance(currentAccount).getStickerSet(type == TYPE_EMOJI_STATUS ? new TLRPC.TL_inputStickerSetEmojiDefaultStatuses() : new TLRPC.TL_inputStickerSetEmojiChannelDefaultStatuses(), true); if (defaultSet == null) { defaultSetLoading = true; } else { @@ -3383,7 +3402,12 @@ private void updateRows(boolean updateEmojipacks, boolean animated, boolean diff totalCount++; rowHashCodes.add(2L); } - ArrayList defaultEmojiStatuses = MediaDataController.getInstance(currentAccount).getDefaultEmojiStatuses(); + ArrayList defaultEmojiStatuses; + if (type == TYPE_EMOJI_STATUS) { + defaultEmojiStatuses = MediaDataController.getInstance(currentAccount).getDefaultEmojiStatuses(); + } else { + defaultEmojiStatuses = MediaDataController.getInstance(currentAccount).getDefaultChannelEmojiStatuses(); + } final int maxrecentlen = SPAN_COUNT_FOR_EMOJI * (RECENT_MAX_LINES + 8); if (defaultSet.documents != null && !defaultSet.documents.isEmpty()) { for (int i = 0; i < Math.min(SPAN_COUNT_FOR_EMOJI - 1, defaultSet.documents.size()); ++i) { @@ -3393,7 +3417,7 @@ private void updateRows(boolean updateEmojipacks, boolean animated, boolean diff } } } - if (recentEmojiStatuses != null && !recentEmojiStatuses.isEmpty()) { + if (type == TYPE_EMOJI_STATUS && recentEmojiStatuses != null && !recentEmojiStatuses.isEmpty()) { for (TLRPC.EmojiStatus emojiStatus : recentEmojiStatuses) { Long did = UserObject.getEmojiStatusDocumentId(emojiStatus); if (did == null) { @@ -3466,6 +3490,9 @@ private void updateRows(boolean updateEmojipacks, boolean animated, boolean diff if ((type == TYPE_SET_REPLY_ICON || type == TYPE_SET_REPLY_ICON_BOTTOM) && !MessageObject.isTextColorSet(set)) { continue; } + if ((type == TYPE_EMOJI_STATUS_CHANNEL_TOP || type == TYPE_EMOJI_STATUS_CHANNEL) && !set.set.channel_emoji_status) { + continue; + } if ((set.set.emojis || showStickers) && !installedEmojiSets.contains(set.set.id)) { positionToSection.put(totalCount, packs.size()); sectionToPosition.put(packs.size(), totalCount); @@ -3478,7 +3505,7 @@ private void updateRows(boolean updateEmojipacks, boolean animated, boolean diff pack.expanded = true; pack.free = !MessageObject.isPremiumEmojiPack(set); pack.set = set.set; - pack.documents = set.documents; + pack.documents = filter(set.documents, restricted); pack.index = packs.size(); packs.add(pack); totalCount += pack.documents.size(); @@ -3525,6 +3552,9 @@ private void updateRows(boolean updateEmojipacks, boolean animated, boolean diff if ((type == TYPE_SET_REPLY_ICON || type == TYPE_SET_REPLY_ICON_BOTTOM) && (documents.isEmpty() || !MessageObject.isTextColorEmoji(documents.get(0)))) { continue; } + if ((type == TYPE_EMOJI_STATUS_CHANNEL_TOP || type == TYPE_EMOJI_STATUS_CHANNEL) && !set.channel_emoji_status) { + continue; + } positionToSection.put(totalCount, packs.size()); sectionToPosition.put(packs.size(), totalCount); @@ -3536,7 +3566,7 @@ private void updateRows(boolean updateEmojipacks, boolean animated, boolean diff pack.featured = true; pack.free = !isPremiumPack; pack.set = set; - pack.documents = documents; + pack.documents = filter(documents, restricted); pack.index = packs.size(); pack.expanded = expandedEmojiSets.contains(pack.set.id); @@ -3713,7 +3743,7 @@ private int getCacheType() { if (type == TYPE_TOPIC_ICON || type == TYPE_AVATAR_CONSTRUCTOR) { return AnimatedEmojiDrawable.CACHE_TYPE_ALERT_PREVIEW_STATIC; } - return type == TYPE_EMOJI_STATUS || type == TYPE_SET_DEFAULT_REACTION ? AnimatedEmojiDrawable.CACHE_TYPE_KEYBOARD : AnimatedEmojiDrawable.CACHE_TYPE_ALERT_PREVIEW; + return type == TYPE_EMOJI_STATUS || type == TYPE_EMOJI_STATUS_CHANNEL || type == TYPE_EMOJI_STATUS_CHANNEL_TOP || type == TYPE_SET_DEFAULT_REACTION ? AnimatedEmojiDrawable.CACHE_TYPE_KEYBOARD : AnimatedEmojiDrawable.CACHE_TYPE_ALERT_PREVIEW; } public class EmojiListView extends RecyclerListView { @@ -4018,7 +4048,7 @@ public void prepareDraw(long time) { ImageReceiver imageReceiver; if (imageView.empty) { Drawable drawable = getPremiumStar(); - float scale = type == TYPE_SET_REPLY_ICON || type == TYPE_SET_REPLY_ICON_BOTTOM ? 1.3f : 1f; + float scale = type == TYPE_SET_REPLY_ICON || type == TYPE_EMOJI_STATUS_CHANNEL_TOP || type == TYPE_EMOJI_STATUS_CHANNEL || type == TYPE_SET_REPLY_ICON_BOTTOM ? 1.3f : 1f; if (imageView.pressedProgress != 0 || imageView.selected) { scale *= 0.8f + 0.2f * (1f - (imageView.selected ? .7f : imageView.pressedProgress)); } @@ -4181,7 +4211,7 @@ protected void drawInUiThread(Canvas canvas, float alpha) { Drawable drawable = null; if (imageView.empty) { drawable = getPremiumStar(); - if (type == TYPE_SET_REPLY_ICON || type == TYPE_SET_REPLY_ICON_BOTTOM) { + if (type == TYPE_SET_REPLY_ICON || type == TYPE_EMOJI_STATUS_CHANNEL_TOP || type == TYPE_EMOJI_STATUS_CHANNEL || type == TYPE_SET_REPLY_ICON_BOTTOM) { AndroidUtilities.rectTmp2.inset((int) (-AndroidUtilities.rectTmp2.width() * .15f), (int) (-AndroidUtilities.rectTmp2.height() * .15f)); } drawable.setBounds(AndroidUtilities.rectTmp2); @@ -4460,7 +4490,7 @@ public void onAnimationEnd(Animator animation) { } }); - if (isFirstOpen && type != TYPE_SET_REPLY_ICON && type != TYPE_SET_REPLY_ICON_BOTTOM) { + if (isFirstOpen && type != TYPE_SET_REPLY_ICON && type != TYPE_EMOJI_STATUS_CHANNEL_TOP && type != TYPE_SET_REPLY_ICON_BOTTOM) { isFirstOpen = false; AnimatedEmojiDrawable.getDocumentFetcher(currentAccount).setUiDbCallback(() -> { HwEmojis.enableHw(); @@ -5770,4 +5800,16 @@ public SetTitleDocument(String title) { this.title = title; } } + + private ArrayList filter(ArrayList documents, HashSet restricted) { + if (restricted == null) return documents; + for (int i = 0; i < documents.size(); ++i) { + TLRPC.Document d = documents.get(i); + if (d == null || restricted.contains(d.id)) { + documents.remove(i); + i--; + } + } + return documents; + } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/DialogStoriesCell.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/DialogStoriesCell.java index d548f11b65d..37fd08fdc82 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/DialogStoriesCell.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/DialogStoriesCell.java @@ -471,7 +471,7 @@ public void updateItems(boolean animated, boolean force) { } if (!hasOverlayText) { - titleView.setText(currentTitle, animated); + titleView.setText(currentTitle, animated && !LocaleController.isRTL); } miniItems.clear(); @@ -911,12 +911,12 @@ public void setTitleOverlayText(String titleOverlayText, int textId) { textToSet = spannableString; } } - titleView.setText(textToSet, true); + titleView.setText(textToSet, !LocaleController.isRTL); } } else { hasOverlayText = false; overlayTextId = 0; - titleView.setText(currentTitle, true); + titleView.setText(currentTitle, !LocaleController.isRTL); } if (hasEllipsizedText) { ellipsizeSpanAnimator.addView(titleView); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/HwLayouts.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/HwLayouts.java index 32d8b23fafe..073ff798720 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/HwLayouts.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/HwLayouts.java @@ -5,6 +5,7 @@ import android.annotation.SuppressLint; import android.content.Context; +import android.util.Log; import android.view.TextureView; import android.view.View; import android.widget.FrameLayout; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/PeerStoriesView.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/PeerStoriesView.java index 50b7e2100d5..1aa07b023f3 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/PeerStoriesView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/PeerStoriesView.java @@ -35,7 +35,6 @@ import android.text.style.CharacterStyle; import android.text.style.ClickableSpan; import android.text.style.URLSpan; -import android.util.Log; import android.util.TypedValue; import android.view.Gravity; import android.view.HapticFeedbackConstants; @@ -319,7 +318,7 @@ public class PeerStoriesView extends SizeNotifierFrameLayout implements Notifica private AnimatedFloat linesAlpha = new AnimatedFloat(this); private float prevToHideProgress; public long videoDuration; - private boolean allowShare, allowShareLink; + private boolean allowShare, allowRepost, allowShareLink; public boolean forceUpdateOffsets; private HintView mediaBanTooltip; @@ -421,7 +420,7 @@ protected boolean setImageBitmapByKey(Drawable drawable, String key, int type, b this.resourcesProvider = resourcesProvider; setClipChildren(false); - storyAreasView = new StoryMediaAreasView(context, resourcesProvider) { + storyAreasView = new StoryMediaAreasView(context, storyContainer, resourcesProvider) { @Override protected void onHintVisible(boolean hintVisible) { if (delegate != null) { @@ -449,15 +448,20 @@ public void showEffect(StoryReactionWidgetView v) { v.playAnimation(); emojiAnimationsOverlay.showAnimationForWidget(v); } + + @Override + protected Bitmap getPlayingBitmap() { + return PeerStoriesView.this.getPlayingBitmap(); + } }; storyContainer = new HwFrameLayout(context) { - AnimatedFloat progressToAudio = new AnimatedFloat(this, 150, CubicBezierInterpolator.DEFAULT); - AnimatedFloat progressToFullBlackoutA = new AnimatedFloat(this, 150, CubicBezierInterpolator.DEFAULT); - CellFlickerDrawable loadingDrawable = new CellFlickerDrawable(32, 102, 240); - AnimatedFloat loadingDrawableAlpha2 = new AnimatedFloat(this); - AnimatedFloat loadingDrawableAlpha = new AnimatedFloat(this); + final AnimatedFloat progressToAudio = new AnimatedFloat(this, 150, CubicBezierInterpolator.DEFAULT); + final AnimatedFloat progressToFullBlackoutA = new AnimatedFloat(this, 150, CubicBezierInterpolator.DEFAULT); + final CellFlickerDrawable loadingDrawable = new CellFlickerDrawable(32, 102, 240); + final AnimatedFloat loadingDrawableAlpha2 = new AnimatedFloat(this); + final AnimatedFloat loadingDrawableAlpha = new AnimatedFloat(this); { loadingDrawableAlpha2.setDuration(500); loadingDrawableAlpha.setDuration(100); @@ -472,7 +476,7 @@ protected void dispatchDraw(Canvas canvas) { headerView.backupImageView.getImageReceiver().setVisible(true, true); } if (!unsupported) { - if (playerSharedScope.renderView != null) { + if (playerSharedScope.renderView != null || storyAreasView != null && (storyAreasView.hasSelectedForScale() || storyAreasView.parentHighlightScaleAlpha.isInProgress())) { invalidate(); } canvas.save(); @@ -896,6 +900,17 @@ public void onLinkLongPress(URLSpan span, View spoilersTextView, Runnable done) @Override public void onReplyClick(Reply reply) { if (reply == null) return; + if (reply.isRepostMessage && reply.peerId != null && reply.messageId != null) { + Bundle args = new Bundle(); + if (reply.peerId >= 0) { + args.putLong("user_id", reply.peerId); + } else { + args.putLong("chat_id", -reply.peerId); + } + args.putInt("message_id", reply.messageId); + storyViewer.presentFragment(new ChatActivity(args)); + return; + } if (reply.peerId == null || reply.storyId == null) { BulletinFactory.of(storyContainer, resourcesProvider) .createSimpleBulletin(R.raw.error, LocaleController.getString(R.string.StoryHidAccount)) @@ -1222,7 +1237,7 @@ protected void onCreate(ActionBarPopupWindow.ActionBarPopupWindowLayout popupLay time = playerSharedScope.player.currentPosition; } StoryEntry entry = MessagesController.getInstance(currentAccount).getStoriesController().getDraftsController().getForEdit(currentStory.storyItem.dialogId, currentStory.storyItem); - if (entry == null || entry.file == null || !entry.file.exists()) { + if (entry == null || entry.isRepostMessage || entry.file == null || !entry.file.exists()) { entry = StoryEntry.fromStoryItem(currentStory.getPath(), currentStory.storyItem); entry.editStoryPeerId = dialogId; } @@ -1326,7 +1341,7 @@ protected void onCreate(ActionBarPopupWindow.ActionBarPopupWindowLayout popupLay }); } - if (!MessagesController.getInstance(currentAccount).premiumLocked && !isChannel) { + if (!MessagesController.getInstance(currentAccount).premiumFeaturesBlocked() && !isChannel) { createStealthModeItem(popupLayout); } @@ -1445,7 +1460,7 @@ protected void onCreate(ActionBarPopupWindow.ActionBarPopupWindowLayout popupLay popupMenu.dismiss(); } }); - } else if (!MessagesController.getInstance(currentAccount).premiumLocked) { + } else if (!MessagesController.getInstance(currentAccount).premiumFeaturesBlocked()) { Drawable lockIcon = ContextCompat.getDrawable(context, R.drawable.msg_gallery_locked2); lockIcon.setColorFilter(new PorterDuffColorFilter(ColorUtils.blendARGB(Color.WHITE, Color.BLACK, 0.5f), PorterDuff.Mode.MULTIPLY)); CombinedDrawable combinedDrawable = new CombinedDrawable( @@ -1474,7 +1489,7 @@ public void setColorFilter(ColorFilter colorFilter) { } } - if (!MessagesController.getInstance(currentAccount).premiumLocked && !isChannel) { + if (!MessagesController.getInstance(currentAccount).premiumFeaturesBlocked() && !isChannel) { createStealthModeItem(popupLayout); } if (allowShareLink) { @@ -2208,13 +2223,13 @@ public void didPressAttachButton() { } @Override - public void needStartRecordVideo(int state, boolean notify, int scheduleDate) { + public void needStartRecordVideo(int state, boolean notify, int scheduleDate, int ttl) { checkInstantCameraView(); if (instantCameraView != null) { if (state == 0) { instantCameraView.showCamera(); } else if (state == 1 || state == 3 || state == 4) { - instantCameraView.send(state, notify, scheduleDate); + instantCameraView.send(state, notify, scheduleDate, ttl); } else if (state == 2 || state == 5) { instantCameraView.cancel(state == 2); } @@ -2612,6 +2627,11 @@ public void dismissInternal() { shareAlert = null; } + @Override + protected void onShareStory(View cell) { + tryToOpenRepostStory(); + } + @Override protected void onSend(LongSparseArray dids, int count, TLRPC.TL_forumTopic topic) { super.onSend(dids, count, topic); @@ -2619,9 +2639,7 @@ protected void onSend(LongSparseArray dids, int count, TLRPC.TL_fo if (bulletinFactory != null) { if (dids.size() == 1) { long did = dids.keyAt(0); - if (did == Long.MAX_VALUE) { - tryToOpenRepostStory(); - } else if (did == UserConfig.getInstance(currentAccount).clientUserId) { + if (did == UserConfig.getInstance(currentAccount).clientUserId) { bulletinFactory.createSimpleBulletin(R.raw.saved_messages, AndroidUtilities.replaceTags(LocaleController.formatString("StorySharedToSavedMessages", R.string.StorySharedToSavedMessages)), Bulletin.DURATION_PROLONG).hideAfterBottomSheet(false).show(); } else if (did < 0) { TLRPC.Chat chat = MessagesController.getInstance(currentAccount).getChat(-did); @@ -2837,10 +2855,9 @@ private void bindInternal(int startFromPosition) { isSelf = false; isChannel = true; - //TODO uncomment if server support views for channels -// if (storiesController.canEditStories(dialogId)) { -// userCanSeeViews = true; -// } + if (storiesController.canEditStories(dialogId) || BuildVars.DEBUG_PRIVATE_VERSION) { + userCanSeeViews = true; + } TLRPC.Chat chat = MessagesController.getInstance(currentAccount).getChat(-dialogId); avatarDrawable.setInfo(currentAccount, chat); headerView.backupImageView.getImageReceiver().setForUserOrChat(chat, avatarDrawable); @@ -3480,7 +3497,7 @@ private void updatePosition(boolean preload) { } currentStory.set(uploadingStory); storyAreasView.set(null, StoryMediaAreasView.getMediaAreasFor(uploadingStory.entry), emojiAnimationsOverlay); - allowShare = allowShareLink = false; + allowShare = allowRepost = allowShareLink = false; } else { isUploading = false; isEditing = false; @@ -3503,9 +3520,9 @@ private void updatePosition(boolean preload) { currentStory.set(editingStory); storyAreasView.set(null, StoryMediaAreasView.getMediaAreasFor(editingStory.entry), emojiAnimationsOverlay); currentStory.editingSourceItem = storyItem; - allowShare = allowShareLink = false; + allowShare = allowRepost = allowShareLink = false; } else { - boolean isVideo = storyItem.media != null && MessageObject.isVideoDocument(storyItem.media.document); + boolean isVideo = storyItem.media != null && MessageObject.isVideoDocument(storyItem.media.getDocument()); Drawable thumbDrawable = null; storyItem.dialogId = dialogId; imageReceiver.setCrossfadeWithOldImage(wasEditing); @@ -3518,7 +3535,7 @@ private void updatePosition(boolean preload) { } if (isVideo) { if (storyItem.media != null) { - thumbDrawable = ImageLoader.createStripedBitmap(storyItem.media.document.thumbs); + thumbDrawable = ImageLoader.createStripedBitmap(storyItem.media.getDocument().thumbs); } if (storyItem.firstFramePath != null && ImageLoader.getInstance().isInMemCache(ImageLocation.getForPath(storyItem.firstFramePath).getKey(null, null, false) + "@" + filter, false)) { imageReceiver.setImage(null, null, ImageLocation.getForPath(storyItem.firstFramePath), filter, null, null, thumbDrawable, 0, null, null, 0); @@ -3542,11 +3559,11 @@ private void updatePosition(boolean preload) { } storyItem.dialogId = dialogId; if (isVideo) { - TLRPC.PhotoSize size = FileLoader.getClosestPhotoSizeWithSize(storyItem.media.document.thumbs, 1000); + TLRPC.PhotoSize size = FileLoader.getClosestPhotoSizeWithSize(storyItem.media.getDocument().thumbs, 1000); if (thumbDrawable == null) { - thumbDrawable = ImageLoader.createStripedBitmap(storyItem.media.document.thumbs); + thumbDrawable = ImageLoader.createStripedBitmap(storyItem.media.getDocument().thumbs); } - imageReceiver.setImage(null, null, ImageLocation.getForDocument(storyItem.media.document), filter + "_pframe", ImageLocation.getForDocument(size, storyItem.media.document), filter, thumbDrawable, 0, null, storyItem, 0); + imageReceiver.setImage(null, null, ImageLocation.getForDocument(storyItem.media.getDocument()), filter + "_pframe", ImageLocation.getForDocument(size, storyItem.media.getDocument()), filter, thumbDrawable, 0, null, storyItem, 0); } else { TLRPC.Photo photo = storyItem.media != null ? storyItem.media.photo : null; if (photo != null && photo.sizes != null) { @@ -3572,6 +3589,11 @@ private void updatePosition(boolean preload) { if (allowShare) { allowShare = currentStory.allowScreenshots() && currentStory.storyItem.isPublic; } + allowRepost = allowShare; + if (allowRepost && isChannel) { + final TLRPC.Chat chat = MessagesController.getInstance(currentAccount).getChat(-dialogId); + allowRepost = chat != null && ChatObject.isPublic(chat); + } if (allowShareLink) { if (isChannel) { final TLRPC.Chat chat = MessagesController.getInstance(currentAccount).getChat(-dialogId); @@ -3640,23 +3662,24 @@ private void updatePosition(boolean preload) { } else if (currentStory.storyItem != null) { if (currentStory.storyItem.date == -1) { headerView.setSubtitle(LocaleController.getString("CachedStory", R.string.CachedStory)); - } else if (currentStory.storyItem.fwd_from != null) { + } else if (currentStory.getReply() != null) { + StoryCaptionView.Reply reply = currentStory.getReply(); + SpannableStringBuilder ssb = new SpannableStringBuilder(); SpannableString repostIcon = new SpannableString("r"); repostIcon.setSpan(new ColoredImageSpan(R.drawable.mini_repost_story), 0, repostIcon.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); ssb.append(repostIcon).append(" "); - if (currentStory.storyItem.fwd_from.from != null) { + if (reply.peerId != null) { AvatarSpan avatar = new AvatarSpan(headerView.subtitleView[0], currentAccount, 15); SpannableString avatarStr = new SpannableString("a"); avatarStr.setSpan(avatar, 0, 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); ssb.append(avatarStr).append(" "); - long did = DialogObject.getPeerDialogId(currentStory.storyItem.fwd_from.from); - if (did > 0) { - TLRPC.User user = MessagesController.getInstance(currentAccount).getUser(did); + if (reply.peerId > 0) { + TLRPC.User user = MessagesController.getInstance(currentAccount).getUser(reply.peerId); avatar.setUser(user); ssb.append(UserObject.getUserName(user)); } else { - TLRPC.Chat chat = MessagesController.getInstance(currentAccount).getChat(-did); + TLRPC.Chat chat = MessagesController.getInstance(currentAccount).getChat(-reply.peerId); avatar.setChat(chat); if (chat != null) { ssb.append(chat.title); @@ -3666,15 +3689,19 @@ private void updatePosition(boolean preload) { ssb.append(currentStory.storyItem.fwd_from.from_name); } headerView.setOnSubtitleClick(v -> { - if (currentStory.storyItem.fwd_from.from != null) { - long did = DialogObject.getPeerDialogId(currentStory.storyItem.fwd_from.from); + if (reply.peerId != null) { Bundle args = new Bundle(); - if (did >= 0) { - args.putLong("user_id", did); + if (reply.peerId >= 0) { + args.putLong("user_id", reply.peerId); + } else { + args.putLong("chat_id", -reply.peerId); + } + if (reply.isRepostMessage && reply.messageId != null) { + args.putInt("message_id", reply.messageId); + storyViewer.presentFragment(new ChatActivity(args)); } else { - args.putLong("chat_id", -did); + storyViewer.presentFragment(new ProfileActivity(args)); } - storyViewer.presentFragment(new ProfileActivity(args)); } else { BulletinFactory.of(storyContainer, resourcesProvider) .createSimpleBulletin(R.raw.error, LocaleController.getString(R.string.StoryHidAccount)) @@ -3730,7 +3757,7 @@ private void updatePosition(boolean preload) { createReplyDisabledView(); unsupportedContainer.setVisibility(View.VISIBLE); replyDisabledTextView.setVisibility(View.VISIBLE); - allowShare = allowShareLink = false; + allowShare = allowRepost = allowShareLink = false; if (chatActivityEnterView != null) { chatActivityEnterView.setVisibility(View.GONE); } @@ -3774,7 +3801,7 @@ private void updatePosition(boolean preload) { if (isChannel) { shareButton.setVisibility(allowShare ? View.VISIBLE : View.INVISIBLE); if (repostButtonContainer != null) { - repostButtonContainer.setVisibility(allowShare ? View.VISIBLE : View.INVISIBLE); + repostButtonContainer.setVisibility(allowRepost ? View.VISIBLE : View.GONE); } likeButtonContainer.setVisibility(isFailed ? View.GONE : View.VISIBLE); } else { @@ -3982,9 +4009,9 @@ private void updatePreloadImages() { storyItem.dialogId = dialogId; setStoryImage(storyItem, imageReceiver, filter); - boolean isVideo = storyItem.media != null && storyItem.media.document != null && MessageObject.isVideoDocument(storyItem.media.document); + boolean isVideo = storyItem.media != null && MessageObject.isVideoDocument(storyItem.media.getDocument()); if (isVideo) { - TLRPC.Document document = storyItem.media.document; + TLRPC.Document document = storyItem.media.getDocument(); if (storyItem.fileReference == 0) { storyItem.fileReference = FileLoader.getInstance(currentAccount).getFileReference(storyItem); } @@ -4032,7 +4059,7 @@ private void setStoryImage(TL_stories.StoryItem storyItem, ImageReceiver imageRe setStoryImage(editingStory, imageReceiver, filter); return; } - boolean isVideo = storyItem.media != null && storyItem.media.document != null && MessageObject.isVideoDocument(storyItem.media.document); + boolean isVideo = storyItem.media != null && MessageObject.isVideoDocument(storyItem.media.getDocument()); if (storyItem.attachPath != null) { if (storyItem.media == null) { @@ -4045,8 +4072,8 @@ private void setStoryImage(TL_stories.StoryItem storyItem, ImageReceiver imageRe } } else { if (isVideo) { - TLRPC.PhotoSize size = FileLoader.getClosestPhotoSizeWithSize(storyItem.media.document.thumbs, 1000); - imageReceiver.setImage(ImageLocation.getForDocument(storyItem.media.document), filter + "_pframe", ImageLocation.getForDocument(size, storyItem.media.document), filter, null, null, null, 0, null, storyItem, 0); + TLRPC.PhotoSize size = FileLoader.getClosestPhotoSizeWithSize(storyItem.media.getDocument().thumbs, 1000); + imageReceiver.setImage(ImageLocation.getForDocument(storyItem.media.getDocument()), filter + "_pframe", ImageLocation.getForDocument(size, storyItem.media.getDocument()), filter, null, null, null, 0, null, storyItem, 0); } else { TLRPC.Photo photo = storyItem.media != null ? storyItem.media.photo : null; if (photo != null && photo.sizes != null) { @@ -4214,7 +4241,7 @@ private void requestVideoPlayer(long t) { } else if (currentStory.storyItem != null) { currentStory.storyItem.dialogId = dialogId; try { - document = currentStory.storyItem.media.document; + document = currentStory.storyItem.media.getDocument(); if (currentStory.storyItem.fileReference == 0) { currentStory.storyItem.fileReference = FileLoader.getInstance(currentAccount).getFileReference(currentStory.storyItem); } @@ -4275,9 +4302,9 @@ public void setDelegate(Delegate delegate) { this.delegate = delegate; } - public void createBlurredBitmap(Canvas canvas, Bitmap bitmap) { + public void drawPlayingBitmap(int w, int h, Canvas canvas) { if (playerSharedScope.renderView != null && playerSharedScope.surfaceView != null) { - Bitmap surfaceBitmap = Bitmap.createBitmap(bitmap.getWidth(), bitmap.getHeight(), Bitmap.Config.ARGB_8888); + Bitmap surfaceBitmap = Bitmap.createBitmap(w, h, Bitmap.Config.ARGB_8888); if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.N) { AndroidUtilities.getBitmapFromSurface(playerSharedScope.surfaceView, surfaceBitmap); } @@ -4285,16 +4312,26 @@ public void createBlurredBitmap(Canvas canvas, Bitmap bitmap) { canvas.drawBitmap(surfaceBitmap, 0, 0, null); } } else if (playerSharedScope.renderView != null && playerSharedScope.textureView != null) { - Bitmap textureBitmap = playerSharedScope.textureView.getBitmap(bitmap.getWidth(), bitmap.getHeight()); + Bitmap textureBitmap = playerSharedScope.textureView.getBitmap(w, h); if (textureBitmap != null) { canvas.drawBitmap(textureBitmap, 0, 0, null); } } else { canvas.save(); - canvas.scale(bitmap.getWidth() / (float) storyContainer.getMeasuredWidth(), bitmap.getHeight() / (float) storyContainer.getMeasuredHeight()); + canvas.scale(w / (float) storyContainer.getMeasuredWidth(), h / (float) storyContainer.getMeasuredHeight()); imageReceiver.draw(canvas); canvas.restore(); } + } + + public Bitmap getPlayingBitmap() { + Bitmap bitmap = Bitmap.createBitmap(storyContainer.getWidth(), storyContainer.getHeight(), Bitmap.Config.ARGB_8888); + drawPlayingBitmap(bitmap.getWidth(), bitmap.getHeight(), new Canvas(bitmap)); + return bitmap; + } + + public void createBlurredBitmap(Canvas canvas, Bitmap bitmap) { + drawPlayingBitmap(bitmap.getWidth(), bitmap.getHeight(), canvas); int color = AndroidUtilities.getDominantColor(bitmap); float brightness = AndroidUtilities.computePerceivedBrightness(color); if (brightness < 0.15f) { @@ -5093,8 +5130,8 @@ private boolean isVideoInternal() { if (uploadingStory != null) { return uploadingStory.isVideo; } - if (storyItem != null && storyItem.media != null && storyItem.media.document != null) { - return storyItem.media.document != null && MessageObject.isVideoDocument(storyItem.media.document); + if (storyItem != null && storyItem.media != null && storyItem.media.getDocument() != null) { + return MessageObject.isVideoDocument(storyItem.media.getDocument()); } if (storyItem != null && storyItem.media == null && storyItem.attachPath != null) { return storyItem.attachPath.toLowerCase().endsWith(".mp4"); @@ -5168,9 +5205,10 @@ boolean hasSound() { if (!isVideo) { return false; } - if (storyItem != null && storyItem.media != null && storyItem.media.document != null) { - for (int i = 0; i < storyItem.media.document.attributes.size(); i++) { - TLRPC.DocumentAttribute attribute = storyItem.media.document.attributes.get(i); + TLRPC.Document document; + if (storyItem != null && storyItem.media != null && (document = storyItem.media.getDocument()) != null) { + for (int i = 0; i < document.attributes.size(); i++) { + TLRPC.DocumentAttribute attribute = document.attributes.get(i); if (attribute instanceof TLRPC.TL_documentAttributeVideo && attribute.nosound) { return false; } @@ -5206,8 +5244,8 @@ public File getPath() { return new File(getLocalPath()); } if (storyItem != null) { - if (storyItem.media != null && storyItem.media.document != null) { - return FileLoader.getInstance(currentAccount).getPathToAttach(storyItem.media.document); + if (storyItem.media != null && storyItem.media.getDocument() != null) { + return FileLoader.getInstance(currentAccount).getPathToAttach(storyItem.media.getDocument()); } else if (storyItem.media != null && storyItem.media.photo != null) { TLRPC.PhotoSize size = FileLoader.getClosestPhotoSizeWithSize(storyItem.media.photo.sizes, Integer.MAX_VALUE); File file = FileLoader.getInstance(currentAccount).getPathToAttach(size, true); @@ -5520,7 +5558,18 @@ private void updateViewOffsets() { } else { inputBackgroundPaint.setColor(ColorUtils.setAlphaComponent(Color.BLACK, (int) (140 * hideInterfaceAlpha * (1f - outT)))); } - if (forceUpdateOffsets || progressToReply != storyViewer.swipeToReplyProgress || progressToHideInterface.get() != prevToHideProgress || lastAnimatingKeyboardHeight != animatingKeyboardHeight || progressToKeyboardLocal != progressToKeyboard || progressToDismissLocal != progressToDismiss || progressToRecord != progressToRecording.get() || popupVisible || progressToStickerExpandedLocal != progressToStickerExpanded.get() || progressToText != progressToTextA.get()) { + if ( + forceUpdateOffsets || + progressToReply != storyViewer.swipeToReplyProgress || + progressToHideInterface.get() != prevToHideProgress || + lastAnimatingKeyboardHeight != animatingKeyboardHeight || + progressToKeyboardLocal != progressToKeyboard || + progressToDismissLocal != progressToDismiss || + progressToRecord != progressToRecording.get() || + popupVisible || + progressToStickerExpandedLocal != progressToStickerExpanded.get() || + progressToText != progressToTextA.get() + ) { forceUpdateOffsets = false; lastAnimatingKeyboardHeight = animatingKeyboardHeight; if (progressToHideInterface.get() != prevToHideProgress) { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/SelfStoryViewsPage.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/SelfStoryViewsPage.java index dcd21c654a5..e5995d34bae 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/SelfStoryViewsPage.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/SelfStoryViewsPage.java @@ -11,8 +11,11 @@ import android.graphics.drawable.ColorDrawable; import android.graphics.drawable.Drawable; import android.graphics.drawable.GradientDrawable; +import android.os.Bundle; import android.text.SpannableStringBuilder; import android.text.TextUtils; +import android.util.SparseArray; +import android.util.SparseIntArray; import android.util.TypedValue; import android.view.Gravity; import android.view.HapticFeedbackConstants; @@ -33,9 +36,11 @@ import org.telegram.messenger.AndroidUtilities; import org.telegram.messenger.BuildVars; import org.telegram.messenger.ContactsController; +import org.telegram.messenger.DialogObject; import org.telegram.messenger.FileLog; import org.telegram.messenger.LocaleController; import org.telegram.messenger.MessagesController; +import org.telegram.messenger.MessagesStorage; import org.telegram.messenger.NotificationCenter; import org.telegram.messenger.R; import org.telegram.messenger.UserConfig; @@ -51,6 +56,7 @@ import org.telegram.ui.ActionBar.Theme; import org.telegram.ui.Cells.FixedHeightEmptyCell; import org.telegram.ui.Cells.ReactedUserHolderView; +import org.telegram.ui.ChatActivity; import org.telegram.ui.Components.AnimatedEmojiDrawable; import org.telegram.ui.Components.Bulletin; import org.telegram.ui.Components.BulletinFactory; @@ -118,7 +124,6 @@ public class SelfStoryViewsPage extends FrameLayout implements NotificationCente SelfStoryViewsView.StoryItemInternal storyItem; ViewsModel currentModel; ViewsModel defaultModel; - ViewsModel filteredModel; private boolean isAttachedToWindow; RecyclerItemsEnterAnimator recyclerItemsEnterAnimator; StoryViewer storyViewer; @@ -136,7 +141,11 @@ public class SelfStoryViewsPage extends FrameLayout implements NotificationCente private boolean showServerErrorText; private long dialogId; - private boolean isStoryShownToUser(TL_stories.TL_storyView view) { + private boolean isStoryShownToUser(TL_stories.StoryView view) { + if (view == null) { + return true; + } + if (MessagesController.getInstance(currentAccount).getStoriesController().isBlocked(view)) { return false; } @@ -230,9 +239,38 @@ public void onScrollStateChanged(@NonNull RecyclerView recyclerView, int newStat if (position < 0 || position >= listAdapter.items.size()) { return; } - TL_stories.TL_storyView user = listAdapter.items.get(position).user; - if (user != null) { - storyViewer.presentFragment(ProfileActivity.of(user.user_id)); + Item item = listAdapter.items.get(position); + if (item.view instanceof TL_stories.TL_storyView) { + storyViewer.presentFragment(ProfileActivity.of(item.view.user_id)); + } else if (item.view instanceof TL_stories.TL_storyViewPublicRepost) { + if (storyViewer.fragment.getOrCreateOverlayStoryViewer().isShowing) { + return; + } + storyViewer.fragment.getOrCreateOverlayStoryViewer().open(getContext(), ((TL_stories.TL_storyViewPublicRepost) item.view).story, StoriesListPlaceProvider.of(recyclerListView)); + } else if (item.reaction instanceof TL_stories.TL_storyReaction) { + storyViewer.presentFragment(ProfileActivity.of(DialogObject.getPeerDialogId(item.reaction.peer_id))); + } else if (item.reaction instanceof TL_stories.TL_storyReactionPublicRepost) { + if (storyViewer.fragment.getOrCreateOverlayStoryViewer().isShowing) { + return; + } + storyViewer.fragment.getOrCreateOverlayStoryViewer().open(getContext(), ((TL_stories.TL_storyReactionPublicRepost) item.reaction).story, StoriesListPlaceProvider.of(recyclerListView)); + } else if (item.reaction instanceof TL_stories.TL_storyReactionPublicForward || item.view instanceof TL_stories.TL_storyViewPublicForward) { + TLRPC.Message message; + if (item.reaction instanceof TL_stories.TL_storyReactionPublicForward) { + message = item.reaction.message; + } else { + message = item.view.message; + } + Bundle args = new Bundle(); + long dialogId = DialogObject.getPeerDialogId(message.peer_id); + if (dialogId >= 0) { + args.putLong("user_id", dialogId); + } else { + args.putLong("chat_id", -dialogId); + } + args.putInt("message_id", message.id); + ChatActivity chatActivity = new ChatActivity(args); + storyViewer.presentFragment(chatActivity); } }); recyclerListView.setOnItemLongClickListener(new RecyclerListView.OnItemLongClickListener() { @@ -245,7 +283,7 @@ public boolean onItemClick(View view, int position) { if (storyViewer == null || storyViewer.containerView == null) { return false; } - TL_stories.TL_storyView viewUser = listAdapter.items.get(position).user; + TL_stories.StoryView viewUser = listAdapter.items.get(position).view; if (viewUser == null) { return false; } @@ -501,18 +539,25 @@ private void updateViewsVisibility() { if (serverItem.views != null) { showSearch = serverItem.views.views_count >= 15; showReactionsSort = serverItem.views.reactions_count >= (BuildVars.DEBUG_PRIVATE_VERSION ? 5 : 10); - showContactsFilter = serverItem.views.views_count >= 20 && !serverItem.contacts && !serverItem.close_friends && !serverItem.selected_contacts; + showContactsFilter = serverItem.dialogId >= 0 && serverItem.views.views_count >= 20 && !serverItem.contacts && !serverItem.close_friends && !serverItem.selected_contacts; } - defaultModel = MessagesController.getInstance(currentAccount).storiesController.selfViewsModel.get(serverItem.id); + SparseArray models = MessagesController.getInstance(currentAccount).storiesController.selfViewsModel.get(serverItem.dialogId); + defaultModel = models != null ? models.get(serverItem.id) : null; int totalCount = serverItem.views == null ? 0 : serverItem.views.views_count; - if (defaultModel == null || defaultModel.totalCount != totalCount) { + if (defaultModel == null || !defaultModel.isChannel && defaultModel.totalCount != totalCount) { if (defaultModel != null) { defaultModel.release(); } defaultModel = new ViewsModel(currentAccount, dialogId, serverItem, true); defaultModel.reloadIfNeed(state, showContactsFilter, showReactionsSort); defaultModel.loadNext(); - MessagesController.getInstance(currentAccount).storiesController.selfViewsModel.put(serverItem.id, defaultModel); + if (models != null) { + models.put(serverItem.id, defaultModel); + } else { + models = new SparseArray<>(); + models.put(serverItem.id, defaultModel); + MessagesController.getInstance(currentAccount).storiesController.selfViewsModel.put(serverItem.dialogId, models); + } } else { defaultModel.reloadIfNeed(state, showContactsFilter, showReactionsSort); } @@ -523,11 +568,11 @@ private void updateViewsVisibility() { if (currentModel != null && isAttachedToWindow) { currentModel.addListener(this); } - if ((currentModel.isExpiredViews && !UserConfig.getInstance(currentAccount).isPremium()) || (!currentModel.loading && !currentModel.hasNext && currentModel.views.isEmpty() && TextUtils.isEmpty(currentModel.state.searchQuery))) { + if ((currentModel != null && currentModel.isExpiredViews && !UserConfig.getInstance(currentAccount).isPremium()) || (!currentModel.loading && !currentModel.hasNext && currentModel.views.isEmpty() && currentModel.reactions.isEmpty() && TextUtils.isEmpty(currentModel.state.searchQuery))) { showSearch = false; showReactionsSort = false; showContactsFilter = false; - titleView.setText(LocaleController.getString("Viewers", R.string.Viewers)); + titleView.setText(LocaleController.getString(currentModel.isChannel ? R.string.Reactions : R.string.Viewers)); searchField.setVisibility(View.GONE); headerView.setVisibility(View.GONE); TOP_PADDING = 46; @@ -535,29 +580,29 @@ private void updateViewsVisibility() { showSearch = false; showReactionsSort = false; showContactsFilter = false; - titleView.setText(LocaleController.getString("Viewers", R.string.Viewers)); + titleView.setText(LocaleController.getString(currentModel.isChannel ? R.string.Reactions : R.string.Viewers)); searchField.setVisibility(View.GONE); headerView.setVisibility(View.GONE); TOP_PADDING = 46; } else { headerView.setVisibility(View.VISIBLE); if (currentModel.showReactionOnly) { - titleView.setText(LocaleController.formatPluralString("Likes", serverItem.views.reactions_count, serverItem.views.reactions_count)); + titleView.setText(LocaleController.getString(currentModel.isChannel ? R.string.Reactions : R.string.Viewers)); showSearch = false; showReactionsSort = false; showContactsFilter = false; } else { - if (currentModel.views.size() < 20 && currentModel.views.size() < serverItem.views.views_count && !currentModel.loading && !currentModel.hasNext) { + if (currentModel.getCount() < 20 && currentModel.getCount() < serverItem.views.views_count && !currentModel.loading && !currentModel.hasNext) { showSearch = false; showReactionsSort = false; showContactsFilter = false; showServerErrorText = true; } else { - showSearch = serverItem.views.views_count >= 15; + showSearch = !currentModel.isChannel && serverItem.views.views_count >= 15; showReactionsSort = serverItem.views.reactions_count >= (BuildVars.DEBUG_VERSION ? 5 : 10); - showContactsFilter = serverItem.views.views_count >= 20 && !serverItem.contacts && !serverItem.close_friends && !serverItem.selected_contacts; + showContactsFilter = serverItem.dialogId >= 0 && serverItem.views.views_count >= 20 && !serverItem.contacts && !serverItem.close_friends && !serverItem.selected_contacts; } - titleView.setText(LocaleController.getString("Viewers", R.string.Viewers)); + titleView.setText(LocaleController.getString(currentModel.isChannel ? R.string.Reactions : R.string.Viewers)); } searchField.setVisibility(showSearch ? View.VISIBLE : View.GONE); TOP_PADDING = showSearch ? 96 : 46; @@ -587,7 +632,8 @@ public static void preload(int currentAccount, long dialogId, TL_stories.StoryIt if (storyItem == null) { return; } - ViewsModel model = MessagesController.getInstance(currentAccount).storiesController.selfViewsModel.get(storyItem.id); + SparseArray models = MessagesController.getInstance(currentAccount).storiesController.selfViewsModel.get(storyItem.dialogId); + ViewsModel model = models == null ? null : models.get(storyItem.id); int totalCount = storyItem.views == null ? 0 : storyItem.views.views_count; if (model == null || model.totalCount != totalCount) { if (model != null) { @@ -595,7 +641,10 @@ public static void preload(int currentAccount, long dialogId, TL_stories.StoryIt } model = new ViewsModel(currentAccount, dialogId, storyItem, true); model.loadNext(); - MessagesController.getInstance(currentAccount).storiesController.selfViewsModel.put(storyItem.id, model); + if (models == null) { + MessagesController.getInstance(currentAccount).storiesController.selfViewsModel.put(storyItem.dialogId, models = new SparseArray<>()); + } + models.put(storyItem.id, model); } } @@ -674,7 +723,7 @@ public void didReceivedNotification(int id, int account, Object... args) { if (position < 0 || position >= listAdapter.items.size()) { continue; } - ((ReactedUserHolderView) child).animateAlpha(isStoryShownToUser(listAdapter.items.get(position).user) ? 1 : .5f, true); + ((ReactedUserHolderView) child).animateAlpha(isStoryShownToUser(listAdapter.items.get(position).view) ? 1 : .5f, true); } } } @@ -836,7 +885,7 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { emptyView.title.setVisibility(View.GONE); SpannableStringBuilder spannableStringBuilder = new SpannableStringBuilder(); spannableStringBuilder.append(AndroidUtilities.replaceTags(LocaleController.getString("ExpiredViewsStub", R.string.ExpiredViewsStub))); - if (!MessagesController.getInstance(currentAccount).premiumLocked) { + if (!MessagesController.getInstance(currentAccount).premiumFeaturesBlocked()) { spannableStringBuilder.append("\n\n"); spannableStringBuilder.append(AndroidUtilities.replaceSingleTag(LocaleController.getString("ExpiredViewsStubPremiumDescription", R.string.ExpiredViewsStubPremiumDescription), SelfStoryViewsPage.this::showPremiumAlert)); emptyView.createButtonLayout(LocaleController.getString("LearnMore", R.string.LearnMore), SelfStoryViewsPage.this::showPremiumAlert); @@ -844,8 +893,13 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { emptyView.subtitle.setText(spannableStringBuilder); } else { emptyView.title.setVisibility(View.VISIBLE); - emptyView.title.setText(LocaleController.getString("NoViews", R.string.NoViews)); - emptyView.setSubtitle(LocaleController.getString("NoViewsStub", R.string.NoViewsStub)); + if (defaultModel.isChannel) { + emptyView.title.setText(LocaleController.getString(R.string.NoReactions)); + emptyView.setSubtitle(LocaleController.getString(R.string.NoReactionsStub)); + } else { + emptyView.title.setText(LocaleController.getString(R.string.NoViews)); + emptyView.setSubtitle(LocaleController.getString(R.string.NoViewsStub)); + } } emptyView.showProgress(false, false); view = emptyView; @@ -857,21 +911,78 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { @Override public void onBindViewHolder(@NonNull RecyclerView.ViewHolder holder, int position) { if (holder.getItemViewType() == USER_ITEM) { + if (position < 0 || position >= items.size()) return; + final Item item = items.get(position); ReactedUserHolderView view = (ReactedUserHolderView) holder.itemView; - TLRPC.User user = MessagesController.getInstance(currentAccount).getUser(items.get(position).user.user_id); - - boolean animated = defaultModel.animateDateForUsers.remove(items.get(position).user.user_id); - boolean like = false; - if (items.get(position).user.reaction != null) { - ReactionsLayoutInBubble.VisibleReaction visibleReaction = ReactionsLayoutInBubble.VisibleReaction.fromTLReaction(items.get(position).user.reaction); - if (visibleReaction != null && visibleReaction.emojicon != null && visibleReaction.emojicon.equals("\u2764")) { - like = true; + + TLRPC.Peer peer = null; + if (item.view != null) { + if (item.view instanceof TL_stories.TL_storyViewPublicRepost) { + peer = item.view.peer_id; + } else if (item.view instanceof TL_stories.TL_storyViewPublicForward && item.view.message != null) { + peer = item.view.message.peer_id; + } else { + peer = new TLRPC.TL_peerUser(); + peer.user_id = item.view.user_id; + } + } else if (item.reaction != null) { + peer = item.reaction.peer_id; + if (item.reaction instanceof TL_stories.TL_storyReactionPublicForward && item.reaction.message != null) { + peer = item.reaction.message.peer_id; + } + } + long did = DialogObject.getPeerDialogId(peer); + TLRPC.User user = null; + TLRPC.Chat chat = null; + if (did >= 0) { + user = MessagesController.getInstance(currentAccount).getUser(did); + } else { + chat = MessagesController.getInstance(currentAccount).getChat(-did); + } + boolean animated = defaultModel.animateDateForUsers.remove(did); + + if (item.view != null) { + boolean like = false; + if (item.view.reaction != null) { + ReactionsLayoutInBubble.VisibleReaction visibleReaction = ReactionsLayoutInBubble.VisibleReaction.fromTLReaction(item.view.reaction); + if (visibleReaction != null && visibleReaction.emojicon != null && visibleReaction.emojicon.equals("\u2764")) { + like = true; + } + } + if (item.view instanceof TL_stories.TL_storyViewPublicRepost) { + view.setUserReaction(user, null, null, like, 0, item.view.story, false, true, animated); + } else if (item.view instanceof TL_stories.TL_storyViewPublicForward) { + view.setUserReaction(user, null, null, like, item.view.message != null ? item.view.message.date : 0, storyItem == null ? null : storyItem.storyItem, true, true, animated); + } else { + view.setUserReaction(user, null, like ? null : item.view.reaction, like, item.view.date, null, false, true, animated); + } + int nextItemType = position < items.size() - 1 ? items.get(position + 1).viewType : -1; + view.drawDivider = nextItemType == USER_ITEM || nextItemType == SUBSCRIBE_TO_PREMIUM_TEXT_HINT || nextItemType == SERVER_CANT_RETURN_TEXT_HINT; + view.animateAlpha(isStoryShownToUser(item.view) ? 1f : .5f, false); + } else if (item.reaction != null) { + TL_stories.StoryReaction peerReaction = item.reaction; + + if (peerReaction instanceof TL_stories.TL_storyReaction) { + TL_stories.TL_storyReaction reaction = (TL_stories.TL_storyReaction) peerReaction; + boolean like = false; + if (reaction.reaction != null) { + ReactionsLayoutInBubble.VisibleReaction visibleReaction = ReactionsLayoutInBubble.VisibleReaction.fromTLReaction(reaction.reaction); + if (visibleReaction != null && visibleReaction.emojicon != null && visibleReaction.emojicon.equals("\u2764")) { + like = true; + } + } + view.setUserReaction(user, chat, like ? null : reaction.reaction, like, reaction.date, null, false, true, animated); + } else if (peerReaction instanceof TL_stories.TL_storyReactionPublicRepost) { + TL_stories.TL_storyReactionPublicRepost repost = (TL_stories.TL_storyReactionPublicRepost) peerReaction; + view.setUserReaction(user, chat, null, false, 0, repost.story, false, true, animated); + } else if (peerReaction instanceof TL_stories.TL_storyReactionPublicForward) { + view.setUserReaction(user, chat, null, false, peerReaction.message != null ? peerReaction.message.date : 0, storyItem == null ? null : storyItem.storyItem, true, true, animated); } + + int nextItemType = position < items.size() - 1 ? items.get(position + 1).viewType : -1; + view.drawDivider = nextItemType == USER_ITEM || nextItemType == SUBSCRIBE_TO_PREMIUM_TEXT_HINT || nextItemType == SERVER_CANT_RETURN_TEXT_HINT; + view.animateAlpha(1f, false); } - view.setUserReaction(user, null, like ? null : items.get(position).user.reaction, like, items.get(position).user.date, true, animated); - int nextItemType = position < items.size() - 1 ? items.get(position + 1).viewType : -1; - view.drawDivider = nextItemType == USER_ITEM || nextItemType == SUBSCRIBE_TO_PREMIUM_TEXT_HINT || nextItemType == SERVER_CANT_RETURN_TEXT_HINT; - view.animateAlpha(isStoryShownToUser(items.get(position).user) ? 1f : .5f, false); } } @@ -893,7 +1004,7 @@ public void updateRows() { items.add(new Item(FLICKER_LOADING_ITEM_FULL)); } else { items.add(new Item(FIRST_PADDING_ITEM)); - if (model != null && model.views.isEmpty() && (model.isExpiredViews || (!model.loading && !model.hasNext))) { + if (model != null && model.getCount() <= 0 && (model.isExpiredViews || (!model.loading && !model.hasNext))) { if (!TextUtils.isEmpty(model.state.searchQuery)) { items.add(new Item(EMPTY_VIEW_SEARCH)); } else if (model.isExpiredViews) { @@ -907,19 +1018,25 @@ public void updateRows() { } } else { if (model != null) { - for (int i = 0; i < model.views.size(); i++) { - items.add(new Item(USER_ITEM, model.views.get(i))); + if (model.isChannel) { + for (int i = 0; i < model.reactions.size(); i++) { + items.add(new Item(USER_ITEM, model.reactions.get(i))); + } + } else { + for (int i = 0; i < model.views.size(); i++) { + items.add(new Item(USER_ITEM, model.views.get(i))); + } } } if (model != null && (model.loading || model.hasNext)) { - if (model.views.isEmpty()) { + if (model.getCount() <= 0) { items.add(new Item(FLICKER_LOADING_ITEM_FULL)); } else { items.add(new Item(FLICKER_LOADING_ITEM)); } } else if (model != null && model.showReactionOnly) { items.add(new Item(SUBSCRIBE_TO_PREMIUM_TEXT_HINT)); - } else if (model != null && model.views.size() < model.totalCount && TextUtils.isEmpty(model.state.searchQuery) && !model.state.contactsOnly) { + } else if (model != null && model.getCount() < model.totalCount && TextUtils.isEmpty(model.state.searchQuery) && !model.state.contactsOnly) { items.add(new Item(SERVER_CANT_RETURN_TEXT_HINT)); } } @@ -939,17 +1056,27 @@ private void showPremiumAlert() { sheet.show(); } - private class Item { + private static class Item { final int viewType; - TL_stories.TL_storyView user; + final TL_stories.StoryView view; + final TL_stories.StoryReaction reaction; private Item(int viewType) { this.viewType = viewType; + this.view = null; + this.reaction = null; } - private Item(int viewType, TL_stories.TL_storyView user) { + private Item(int viewType, TL_stories.StoryView view) { this.viewType = viewType; - this.user = user; + this.view = view; + this.reaction = null; + } + + private Item(int viewType, TL_stories.StoryReaction reaction) { + this.viewType = viewType; + this.view = null; + this.reaction = reaction; } } @@ -960,8 +1087,10 @@ public static class ViewsModel { private long dialogId; int currentAccount; boolean loading; - ArrayList views = new ArrayList<>(); - ArrayList originalViews = new ArrayList<>(); + public final boolean isChannel; + ArrayList views = new ArrayList<>(); + ArrayList originalViews = new ArrayList<>(); + ArrayList reactions = new ArrayList<>(); boolean isExpiredViews; boolean showReactionOnly; boolean initial; @@ -971,12 +1100,17 @@ public static class ViewsModel { HashSet animateDateForUsers = new HashSet<>(); boolean useLocalFilters; + public int getCount() { + return isChannel ? reactions.size() : views.size(); + } + ArrayList listeners = new ArrayList<>(); FiltersState state = new FiltersState(); public ViewsModel(int currentAccount, long dialogId, TL_stories.StoryItem storyItem, boolean isDefault) { this.currentAccount = currentAccount; this.storyItem = storyItem; + isChannel = dialogId < 0; this.dialogId = dialogId; this.totalCount = storyItem.views == null ? 0 : storyItem.views.views_count; if (totalCount < 200) { @@ -1008,100 +1142,186 @@ public void loadNext() { if (loading || !hasNext || isExpiredViews) { return; } - TL_stories.TL_stories_getStoryViewsList req = new TL_stories.TL_stories_getStoryViewsList(); - req.id = storyItem.id; - req.peer = MessagesController.getInstance(currentAccount).getInputPeer(dialogId); - if (useLocalFilters) { - req.q = ""; - req.just_contacts = false; - req.reactions_first = true; - } else { - req.q = state.searchQuery; - if (!TextUtils.isEmpty(req.q)) { + if (isChannel) { + TL_stories.TL_getStoryReactionsList req = new TL_stories.TL_getStoryReactionsList(); + req.forwards_first = state.sortByReactions; + req.id = storyItem.id; + req.peer = MessagesController.getInstance(currentAccount).getInputPeer(dialogId); + req.limit = (initial || reactions.size() < 20) ? 20 : 100; + req.offset = offset; + if (req.offset == null) { + req.offset = ""; + } else { req.flags |= 2; } - req.just_contacts = state.contactsOnly; - req.reactions_first = state.sortByReactions; - } - req.limit = (initial || views.size() < 20) ? 20 : 100; - req.offset = offset; - if (req.offset == null) { - req.offset = ""; - } - loading = true; - int[] localReqId = new int[1]; - FileLog.d("SelfStoryViewsPage load next " + storyItem.id + " " + initial + " offset=" + req.offset + " q" + req.q + " " + req.just_contacts + " " + req.reactions_first); - localReqId[0] = reqId = ConnectionsManager.getInstance(currentAccount).sendRequest(req, (response, error) -> AndroidUtilities.runOnUIThread(() -> { - if (localReqId[0] != reqId) { - FileLog.d("SelfStoryViewsPage " + storyItem.id + " localId != reqId"); - return; - } - loading = false; - reqId = -1; - if (response != null) { - TL_stories.TL_stories_storyViewsList res = (TL_stories.TL_stories_storyViewsList) response; - MessagesController.getInstance(currentAccount).getStoriesController().applyStoryViewsBlocked(res); - MessagesController.getInstance(currentAccount).putUsers(res.users, false); - if (initial) { - initial = false; - for (int i = 0; i < views.size(); i++) { - animateDateForUsers.add(views.get(i).user_id); - } - views.clear(); - originalViews.clear(); - } - if (useLocalFilters) { - originalViews.addAll(res.views); - applyLocalFilter(); - } else { - views.addAll(res.views); + loading = true; + int[] localReqId = new int[1]; + FileLog.d("SelfStoryViewsPage reactions load next " + storyItem.id + " " + initial + " offset=" + req.offset/* + " q" + req.q + " " + req.just_contacts + " " + req.reactions_first*/); + localReqId[0] = reqId = ConnectionsManager.getInstance(currentAccount).sendRequest(req, (response, error) -> AndroidUtilities.runOnUIThread(() -> { + if (localReqId[0] != reqId) { + FileLog.d("SelfStoryViewsPage reactions " + storyItem.id + " localId != reqId"); + return; } + loading = false; + reqId = -1; + if (response != null) { + TL_stories.TL_storyReactionsList res = (TL_stories.TL_storyReactionsList) response; + MessagesController.getInstance(currentAccount).putUsers(res.users, false); + MessagesController.getInstance(currentAccount).putChats(res.chats, false); + MessagesStorage.getInstance(currentAccount).putUsersAndChats(res.users, res.chats, true, false); + if (initial) { + initial = false; + for (int i = 0; i < reactions.size(); i++) { + animateDateForUsers.add(DialogObject.getPeerDialogId(reactions.get(i).peer_id)); + } + reactions.clear(); + originalViews.clear(); + } +// if (useLocalFilters) { +// originalReactions.addAll(res.reactions); +// applyLocalFilter(); +// } else { + reactions.addAll(res.reactions); +// } + + if (!res.reactions.isEmpty()) { + hasNext = true; + } else { + hasNext = false; + } + offset = res.next_offset; + if (TextUtils.isEmpty(offset)) { + hasNext = false; + } - if (!res.views.isEmpty()) { - hasNext = true; + if (storyItem.views == null) { + storyItem.views = new TL_stories.TL_storyViews(); + } + boolean counterUpdated = totalCount != res.count; + totalCount = res.count; + if (counterUpdated) { + NotificationCenter.getInstance(currentAccount).postNotificationName(NotificationCenter.storiesUpdated); + } } else { + if (error != null && "MSG_ID_INVALID".equals(error.text)) { + totalCount = 0; + } hasNext = false; } - offset = res.next_offset; - if (TextUtils.isEmpty(offset)) { - hasNext = false; + + FileLog.d("SelfStoryViewsPage reactions " + storyItem.id + " response totalItems " + reactions.size() + " has next " + hasNext); + for (int i = 0; i < listeners.size(); i++) { + listeners.get(i).onDataRecieved(this); } + if (reactions.size() < 20 && hasNext) { + loadNext(); + } + })); + } else { + TL_stories.TL_stories_getStoryViewsList req = new TL_stories.TL_stories_getStoryViewsList(); + req.id = storyItem.id; + req.peer = MessagesController.getInstance(currentAccount).getInputPeer(dialogId); + if (useLocalFilters) { + req.q = ""; + req.just_contacts = false; + req.reactions_first = true; + } else { + req.q = state.searchQuery; + if (!TextUtils.isEmpty(req.q)) { + req.flags |= 2; + } + req.just_contacts = state.contactsOnly; + req.reactions_first = state.sortByReactions; + } + req.limit = (initial || views.size() < 20) ? 20 : 100; + req.offset = offset; + if (req.offset == null) { + req.offset = ""; + } - if (storyItem.views == null) { - storyItem.views = new TL_stories.TL_storyViews(); + loading = true; + int[] localReqId = new int[1]; + FileLog.d("SelfStoryViewsPage load next " + storyItem.id + " " + initial + " offset=" + req.offset + " q" + req.q + " " + req.just_contacts + " " + req.reactions_first); + localReqId[0] = reqId = ConnectionsManager.getInstance(currentAccount).sendRequest(req, (response, error) -> AndroidUtilities.runOnUIThread(() -> { + if (localReqId[0] != reqId) { + FileLog.d("SelfStoryViewsPage " + storyItem.id + " localId != reqId"); + return; } - boolean counterUpdated = false; - if (res.count > storyItem.views.views_count) { - storyItem.views.recent_viewers.clear(); - for (int i = 0; i < (Math.min(3, res.users.size())); i++) { - storyItem.views.recent_viewers.add(res.users.get(i).id); + loading = false; + reqId = -1; + if (response != null) { + TL_stories.StoryViewsList res = (TL_stories.StoryViewsList) response; + MessagesController.getInstance(currentAccount).getStoriesController().applyStoryViewsBlocked(res); + MessagesController.getInstance(currentAccount).putUsers(res.users, false); + MessagesController.getInstance(currentAccount).putChats(res.chats, false); + MessagesStorage.getInstance(currentAccount).putUsersAndChats(res.users, res.chats, true, false); + if (initial) { + initial = false; + for (int i = 0; i < views.size(); i++) { + animateDateForUsers.add(views.get(i).user_id); + } + views.clear(); + originalViews.clear(); + } + if (useLocalFilters) { + originalViews.addAll(res.views); + applyLocalFilter(); + } else { + views.addAll(res.views); + } + + if (!res.views.isEmpty()) { + hasNext = true; + } else { + hasNext = false; + } + offset = res.next_offset; + if (TextUtils.isEmpty(offset)) { + hasNext = false; + } + + if (storyItem.views == null) { + storyItem.views = new TL_stories.TL_storyViews(); + } + boolean counterUpdated = false; + if (res.count > storyItem.views.views_count) { + storyItem.views.recent_viewers.clear(); + for (int i = 0; i < (Math.min(3, res.users.size())); i++) { + storyItem.views.recent_viewers.add(res.users.get(i).id); + } + storyItem.views.views_count = res.count; + counterUpdated = true; } - storyItem.views.views_count = res.count; - counterUpdated = true; + if (storyItem.views.reactions_count != res.reactions_count) { + storyItem.views.reactions_count = res.reactions_count; + counterUpdated = true; + } + if (counterUpdated) { + NotificationCenter.getInstance(currentAccount).postNotificationName(NotificationCenter.storiesUpdated); + } + } else { + if (error != null && "MSG_ID_INVALID".equals(error.text)) { + totalCount = 0; + } + hasNext = false; } - if (storyItem.views.reactions_count != res.reactions_count) { - storyItem.views.reactions_count = res.reactions_count; - counterUpdated = true; + + FileLog.d("SelfStoryViewsPage " + storyItem.id + " response totalItems " + views.size() + " has next " + hasNext); + for (int i = 0; i < listeners.size(); i++) { + listeners.get(i).onDataRecieved(this); } - if (counterUpdated) { - NotificationCenter.getInstance(currentAccount).postNotificationName(NotificationCenter.storiesUpdated); + if (views.size() < 20 && hasNext) { + loadNext(); } - } else { - hasNext = false; - } - - FileLog.d("SelfStoryViewsPage " + storyItem.id + " response totalItems " + views.size() + " has next " + hasNext); - for (int i = 0; i < listeners.size(); i++) { - listeners.get(i).onDataRecieved(this); - } - if (views.size() < 20 && hasNext) { - loadNext(); - } - })); + })); + } } private void applyLocalFilter() { + if (isChannel) { + return; + } views.clear(); if (state.contactsOnly || !TextUtils.isEmpty(state.searchQuery)) { String search1 = null; @@ -1165,7 +1385,7 @@ public void reloadIfNeed(FiltersState state, boolean showContactsFilter, boolean return; } this.state.set(localState); - if (useLocalFilters) { + if (!isChannel && useLocalFilters) { applyLocalFilter(); for (int i = 0; i < listeners.size(); i++) { listeners.get(i).onDataRecieved(this); @@ -1173,6 +1393,7 @@ public void reloadIfNeed(FiltersState state, boolean showContactsFilter, boolean } else { release(); views.clear(); + reactions.clear(); initial = true; loading = false; hasNext = true; @@ -1266,7 +1487,8 @@ public HeaderView(@NonNull Context context) { @Override protected void onCreate(ActionBarPopupWindow.ActionBarPopupWindowLayout popupLayout) { popupLayout.setBackgroundColor(ColorUtils.blendARGB(Color.BLACK, Color.WHITE, 0.18f)); - ActionBarMenuSubItem item = ActionBarMenuItem.addItem(popupLayout, state.sortByReactions ? R.drawable.menu_views_reactions2 : R.drawable.menu_views_reactions, LocaleController.getString("SortByReactions", R.string.SortByReactions), false, resourcesProvider); + final boolean isChannel = currentModel != null && currentModel.isChannel; + ActionBarMenuSubItem item = ActionBarMenuItem.addItem(popupLayout, isChannel ? R.drawable.menu_views_reposts : (state.sortByReactions ? R.drawable.menu_views_reactions2 : R.drawable.menu_views_reactions), LocaleController.getString(isChannel ? R.string.SortByReposts : R.string.SortByReactions), false, resourcesProvider); if (!state.sortByReactions) { item.setAlpha(0.5f); } @@ -1308,7 +1530,7 @@ protected void onCreate(ActionBarPopupWindow.ActionBarPopupWindowLayout popupLay ActionBarPopupWindow.GapView gap = new ActionBarPopupWindow.GapView(getContext(), resourcesProvider, Theme.key_actionBarDefaultSubmenuSeparator); gap.setTag(R.id.fit_width_tag, 1); popupLayout.addView(gap, LayoutHelper.createLinear(LayoutHelper.MATCH_PARENT, 8)); - ActionBarMenuItem.addText(popupLayout, LocaleController.getString("StoryViewsSortDescription", R.string.StoryViewsSortDescription), resourcesProvider); + ActionBarMenuItem.addText(popupLayout, LocaleController.getString(isChannel ? R.string.StoryReactionsSortDescription : R.string.StoryViewsSortDescription), resourcesProvider); } @Override @@ -1405,14 +1627,12 @@ private void reload() { private void updateViewState(boolean animated) { headerView.setState(state.contactsOnly, animated); - if (headerView.lastSortType != state.sortByReactions) { - headerView.lastSortType = state.sortByReactions; - headerView.replacableDrawable.setIcon(state.sortByReactions ? R.drawable.menu_views_reactions3 : R.drawable.menu_views_recent3, animated); - } + headerView.lastSortType = state.sortByReactions; + headerView.replacableDrawable.setIcon(state.sortByReactions ? (currentModel != null && currentModel.isChannel ? R.drawable.menu_views_reposts3 : R.drawable.menu_views_reactions3) : R.drawable.menu_views_recent3, animated); } public static class FiltersState { - boolean sortByReactions = true; + boolean sortByReactions = true; // converts to sortByForwards when showing channel reactions boolean contactsOnly; String searchQuery; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoriesController.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoriesController.java index 30dda1c2c7e..064d40da7ee 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoriesController.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoriesController.java @@ -118,7 +118,7 @@ public class StoriesController { private final DraftsController draftsController; - public SparseArray selfViewsModel = new SparseArray<>(); + public LongSparseArray> selfViewsModel = new LongSparseArray<>(); private String stateHidden; private boolean hasMoreHidden = true; private boolean firstLoad = true; @@ -558,12 +558,12 @@ private void preloadStory(long dialogId, TL_stories.StoryItem storyItem) { if (!canPreloadStories) { return; } - boolean isVideo = storyItem.media != null && MessageObject.isVideoDocument(storyItem.media.document); + boolean isVideo = storyItem.media != null && MessageObject.isVideoDocument(storyItem.media.getDocument()); storyItem.dialogId = dialogId; if (isVideo) { - TLRPC.PhotoSize size = FileLoader.getClosestPhotoSizeWithSize(storyItem.media.document.thumbs, 1000); - FileLoader.getInstance(currentAccount).loadFile(storyItem.media.document, storyItem, FileLoader.PRIORITY_LOW, 1); - FileLoader.getInstance(currentAccount).loadFile(ImageLocation.getForDocument(size, storyItem.media.document), storyItem, "jpg", FileLoader.PRIORITY_LOW, 1); + TLRPC.PhotoSize size = FileLoader.getClosestPhotoSizeWithSize(storyItem.media.getDocument().thumbs, 1000); + FileLoader.getInstance(currentAccount).loadFile(storyItem.media.getDocument(), storyItem, FileLoader.PRIORITY_LOW, 1); + FileLoader.getInstance(currentAccount).loadFile(ImageLocation.getForDocument(size, storyItem.media.getDocument()), storyItem, "jpg", FileLoader.PRIORITY_LOW, 1); } else { TLRPC.Photo photo = storyItem.media == null ? null : storyItem.media.photo; if (photo != null && photo.sizes != null) { @@ -2458,6 +2458,12 @@ private void preloadCache() { loadChatIds.add(-did); } } + for (int j = 0; j < storyItem.media_areas.size(); ++j) { + if (storyItem.media_areas.get(j) instanceof TL_stories.TL_mediaAreaChannelPost) { + long channel_id = ((TL_stories.TL_mediaAreaChannelPost) storyItem.media_areas.get(j)).channel_id; + loadChatIds.add(channel_id); + } + } msg.generateThumbs(false); cacheResult.add(msg); data.reuse(); @@ -2734,6 +2740,8 @@ public boolean load(boolean force, final int count, List ids) { FileLog.d("StoriesList " + type + "{"+ dialogId +"} loaded {" + storyItemMessageIds(newMessageObjects) + "}"); MessagesController.getInstance(currentAccount).putUsers(stories.users, false); + MessagesController.getInstance(currentAccount).putChats(stories.chats, false); + MessagesStorage.getInstance(currentAccount).putUsersAndChats(stories.users, stories.chats, true, true); loading = false; totalCount = stories.count; @@ -3120,7 +3128,7 @@ public void updateBlockedUsers(HashSet ids, Runnable done) { })); } - public boolean isBlocked(TL_stories.TL_storyView storyView) { + public boolean isBlocked(TL_stories.StoryView storyView) { if (storyView == null) { return false; } @@ -3143,12 +3151,12 @@ public boolean isBlocked(long did) { return blocklist.contains(did); } - public void applyStoryViewsBlocked(TL_stories.TL_stories_storyViewsList res) { + public void applyStoryViewsBlocked(TL_stories.StoryViewsList res) { if (res == null || res.views == null) { return; } for (int i = 0; i < res.views.size(); ++i) { - TL_stories.TL_storyView view = res.views.get(i); + TL_stories.StoryView view = res.views.get(i); if (blockedOverride.containsKey(view.user_id)) { blockedOverride.put(view.user_id, view.blocked_my_stories_from); } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoriesListPlaceProvider.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoriesListPlaceProvider.java index f54614683b4..427693c0b5c 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoriesListPlaceProvider.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoriesListPlaceProvider.java @@ -206,17 +206,31 @@ public boolean findView(long dialogId, int messageId, int storyId, int type, Sto } else if (child instanceof ReactedUserHolderView) { ReactedUserHolderView cell = (ReactedUserHolderView) child; if (cell.dialogId == dialogId) { - holder.view = cell.avatarView; - holder.params = cell.params; - holder.avatarImage = cell.avatarView.getImageReceiver(); - holder.clipParent = (View) cell.getParent(); - holder.alpha = cell.getAlpha() * cell.getAlphaInternal(); - if (holder.alpha < 1) { - holder.bgPaint = new Paint(Paint.ANTI_ALIAS_FLAG); - holder.bgPaint.setColor(Theme.getColor(Theme.key_dialogBackground, cell.getResourcesProvider())); + final boolean hasStoryPreview = cell.storyPreviewView != null && cell.storyPreviewView.getImageReceiver() != null && cell.storyPreviewView.getImageReceiver().getImageDrawable() != null; + if (cell.storyId == storyId && hasStoryPreview) { + holder.view = cell.storyPreviewView; + holder.storyImage = cell.storyPreviewView.getImageReceiver(); + holder.clipParent = (View) cell.getParent(); + holder.alpha = cell.getAlpha() * cell.getAlphaInternal(); + if (holder.alpha < 1) { + holder.bgPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + holder.bgPaint.setColor(Theme.getColor(Theme.key_dialogBackground, cell.getResourcesProvider())); + } + updateClip(holder); + return true; + } else if (!hasStoryPreview) { + holder.view = cell.avatarView; + holder.params = cell.params; + holder.avatarImage = cell.avatarView.getImageReceiver(); + holder.clipParent = (View) cell.getParent(); + holder.alpha = cell.getAlpha() * cell.getAlphaInternal(); + if (holder.alpha < 1) { + holder.bgPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + holder.bgPaint.setColor(Theme.getColor(Theme.key_dialogBackground, cell.getResourcesProvider())); + } + updateClip(holder); + return true; } - updateClip(holder); - return true; } } else if (child instanceof ProfileSearchCell) { ProfileSearchCell cell = (ProfileSearchCell) child; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoriesStorage.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoriesStorage.java index 555ededf812..936155b9917 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoriesStorage.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoriesStorage.java @@ -81,6 +81,13 @@ public void getAllStories(Consumer consumer) { if (storyItem.fwd_from != null && storyItem.fwd_from.from != null) { MessagesStorage.addLoadPeerInfo(storyItem.fwd_from.from, usersToLoad, chatsToLoad); } + for (int j = 0; j < storyItem.media_areas.size(); ++j) { + if (storyItem.media_areas.get(j) instanceof TL_stories.TL_mediaAreaChannelPost) { + long channel_id = ((TL_stories.TL_mediaAreaChannelPost) storyItem.media_areas.get(j)).channel_id; + if (!chatsToLoad.contains(channel_id)) + chatsToLoad.add(channel_id); + } + } StoryCustomParamsHelper.readLocalParams(storyItem, customData); storyItems.add(storyItem); data.reuse(); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoryCaptionView.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoryCaptionView.java index 57b35053fdc..7852c3a557d 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoryCaptionView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoryCaptionView.java @@ -77,6 +77,7 @@ import org.telegram.ui.Components.URLSpanMono; import org.telegram.ui.Components.spoilers.SpoilerEffect; import org.telegram.ui.Components.spoilers.SpoilersClickDetector; +import org.telegram.ui.Stories.recorder.StoryEntry; import java.lang.reflect.Field; import java.lang.reflect.Method; @@ -579,6 +580,9 @@ public static class Reply { public Long peerId; public Integer storyId; + public Integer messageId; + public boolean isRepostMessage; + private boolean small = true; private final AnimatedFloat animatedSmall = new AnimatedFloat(0, 350, CubicBezierInterpolator.EASE_OUT_QUINT); public final ButtonBounce bounce = new ButtonBounce(null); @@ -625,40 +629,84 @@ public void load() { } public static Reply from(int currentAccount, TL_stories.StoryItem storyItem) { - if (storyItem == null || storyItem.fwd_from == null) { + if (storyItem == null) { return null; } - Reply reply = new Reply(); - reply.currentAccount = currentAccount; - if (storyItem.fwd_from.from != null) { - long did = reply.peerId = DialogObject.getPeerDialogId(storyItem.fwd_from.from); - if (did >= 0) { - TLRPC.User user = MessagesController.getInstance(currentAccount).getUser(did); - reply.title = new SpannableStringBuilder(MessageObject.userSpan()).append(" ").append(UserObject.getUserName(user)); - } else { - TLRPC.Chat chat = MessagesController.getInstance(currentAccount).getChat(-did); - reply.title = new SpannableStringBuilder(ChatObject.isChannelAndNotMegaGroup(chat) ? MessageObject.channelSpan() : MessageObject.groupSpan()).append(" ").append(chat != null ? chat.title : ""); + if (storyItem.fwd_from != null) { + Reply reply = new Reply(); + reply.currentAccount = currentAccount; + if (storyItem.fwd_from.from != null) { + long did = reply.peerId = DialogObject.getPeerDialogId(storyItem.fwd_from.from); + if (did >= 0) { + TLRPC.User user = MessagesController.getInstance(currentAccount).getUser(did); + reply.title = new SpannableStringBuilder(MessageObject.userSpan()).append(" ").append(UserObject.getUserName(user)); + } else { + TLRPC.Chat chat = MessagesController.getInstance(currentAccount).getChat(-did); + reply.title = new SpannableStringBuilder(ChatObject.isChannelAndNotMegaGroup(chat) ? MessageObject.channelSpan() : MessageObject.groupSpan()).append(" ").append(chat != null ? chat.title : ""); + } + } else if (storyItem.fwd_from.from_name != null) { + reply.title = new SpannableStringBuilder(MessageObject.userSpan()).append(" ").append(storyItem.fwd_from.from_name); } - } else if (storyItem.fwd_from.from_name != null) { - reply.title = new SpannableStringBuilder(MessageObject.userSpan()).append(" ").append(storyItem.fwd_from.from_name); + reply.small = true; + if ((storyItem.fwd_from.flags & 4) != 0) { + reply.storyId = storyItem.fwd_from.story_id; + } + reply.load(); + return reply; } - reply.small = true; - if ((storyItem.fwd_from.flags & 4) != 0) { - reply.storyId = storyItem.fwd_from.story_id; + if (storyItem.media_areas != null) { + TL_stories.TL_mediaAreaChannelPost postArea = null; + for (int i = 0; i < storyItem.media_areas.size(); ++i) { + if (storyItem.media_areas.get(i) instanceof TL_stories.TL_mediaAreaChannelPost) { + postArea = (TL_stories.TL_mediaAreaChannelPost) storyItem.media_areas.get(i); + } + } + if (postArea != null) { + TLRPC.Chat chat = MessagesController.getInstance(currentAccount).getChat(postArea.channel_id); + if (chat != null) { + Reply reply = new Reply(); + reply.peerId = -chat.id; + reply.isRepostMessage = true; + reply.currentAccount = currentAccount; + reply.small = true; + reply.messageId = postArea.msg_id; + reply.title = new SpannableStringBuilder(ChatObject.isChannelAndNotMegaGroup(chat) ? MessageObject.channelSpan() : MessageObject.groupSpan()).append(" ").append(chat.title); + return reply; + } + } } - reply.load(); - return reply; + return null; } public static Reply from(StoriesController.UploadingStory uploadingStory) { - if (uploadingStory == null || uploadingStory.entry == null || !uploadingStory.entry.isRepost) { + if (uploadingStory == null || uploadingStory.entry == null) { return null; } - Reply reply = new Reply(); - reply.title = uploadingStory.entry.repostPeerName; - reply.text = uploadingStory.entry.repostCaption; - reply.small = TextUtils.isEmpty(reply.text); - return reply; + if (uploadingStory.entry.isRepost) { + Reply reply = new Reply(); + reply.title = uploadingStory.entry.repostPeerName; + reply.text = uploadingStory.entry.repostCaption; + reply.small = TextUtils.isEmpty(reply.text); + return reply; + } + if (uploadingStory.entry.isRepostMessage && uploadingStory.entry.messageObjects != null && uploadingStory.entry.messageObjects.size() > 0) { + MessageObject messageObject = uploadingStory.entry.messageObjects.get(0); + final long dialogId = StoryEntry.getRepostDialogId(messageObject); + if (dialogId < 0) { + TLRPC.Chat chat = MessagesController.getInstance(messageObject.currentAccount).getChat(-dialogId); + if (chat != null) { + Reply reply = new Reply(); + reply.peerId = dialogId; + reply.isRepostMessage = true; + reply.currentAccount = messageObject.currentAccount; + reply.small = true; + reply.messageId = StoryEntry.getRepostMessageId(messageObject); + reply.title = new SpannableStringBuilder(ChatObject.isChannelAndNotMegaGroup(chat) ? MessageObject.channelSpan() : MessageObject.groupSpan()).append(" ").append(chat.title); + return reply; + } + } + } + return null; } private final Paint backgroundPaint = new Paint(Paint.ANTI_ALIAS_FLAG); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoryMediaAreasView.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoryMediaAreasView.java index f9937130238..cced3d0c701 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoryMediaAreasView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoryMediaAreasView.java @@ -3,17 +3,22 @@ import static org.telegram.messenger.AndroidUtilities.dp; import android.content.Context; +import android.graphics.Bitmap; import android.graphics.Canvas; import android.graphics.LinearGradient; import android.graphics.Matrix; import android.graphics.Paint; +import android.graphics.Path; import android.graphics.PorterDuff; import android.graphics.PorterDuffXfermode; +import android.graphics.Rect; import android.graphics.RectF; import android.graphics.Shader; +import android.os.Bundle; import android.text.SpannableString; import android.text.SpannableStringBuilder; import android.text.Spanned; +import android.util.Log; import android.view.MotionEvent; import android.view.View; import android.view.animation.LinearInterpolator; @@ -28,8 +33,10 @@ import org.telegram.tgnet.tl.TL_stories; import org.telegram.ui.ActionBar.BaseFragment; import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.ChatActivity; import org.telegram.ui.Components.AnimatedFloat; import org.telegram.ui.Components.ColoredImageSpan; +import org.telegram.ui.Components.CubicBezierInterpolator; import org.telegram.ui.Components.LayoutHelper; import org.telegram.ui.Components.ScaleStateListAnimator; import org.telegram.ui.EmojiAnimationsOverlay; @@ -41,6 +48,7 @@ public class StoryMediaAreasView extends FrameLayout implements View.OnClickListener { + private AreaView lastSelectedArea = null; private AreaView selectedArea = null; private HintView2 hintView = null; @@ -50,11 +58,15 @@ public class StoryMediaAreasView extends FrameLayout implements View.OnClickList Matrix matrix = new Matrix(); float[] point = new float[2]; + private View parentView; private Theme.ResourcesProvider resourcesProvider; - public StoryMediaAreasView(Context context, Theme.ResourcesProvider resourcesProvider) { + public StoryMediaAreasView(Context context, View parentView, Theme.ResourcesProvider resourcesProvider) { super(context); + this.parentView = parentView; this.resourcesProvider = resourcesProvider; + parentHighlightAlpha = new AnimatedFloat(parentView, 0, 120, new LinearInterpolator()); + parentHighlightScaleAlpha = new AnimatedFloat(parentView, 0, 360, CubicBezierInterpolator.EASE_OUT_QUINT); setClipChildren(false); addView(hintsContainer = new FrameLayout(context)); } @@ -105,6 +117,7 @@ public void set(TL_stories.StoryItem storyItem, ArrayList } } selectedArea = null; + parentHighlightScaleAlpha.set(0, true); invalidate(); onHintVisible(false); malicious = false; @@ -131,7 +144,7 @@ public void set(TL_stories.StoryItem storyItem, ArrayList } ScaleStateListAnimator.apply(areaView); } else { - areaView = new AreaView(getContext(), this, mediaArea); + areaView = new AreaView(getContext(), parentView, mediaArea); } areaView.setOnClickListener(this); addView(areaView); @@ -139,7 +152,7 @@ public void set(TL_stories.StoryItem storyItem, ArrayList totalArea += (mediaArea.coordinates.w / 100f * W) * (mediaArea.coordinates.h / 100f * H); } } - malicious = totalArea > W * H * .33f; + malicious = false; // totalArea > W * H * .33f; hintsContainer.bringToFront(); } @@ -185,6 +198,17 @@ public void onClick(View v) { onHintVisible(false); }, 200); + if (selectedArea.mediaArea instanceof TL_stories.TL_mediaAreaChannelPost) { + Bundle args = new Bundle(); + args.putLong("chat_id", ((TL_stories.TL_mediaAreaChannelPost) selectedArea.mediaArea).channel_id); + args.putInt("message_id", ((TL_stories.TL_mediaAreaChannelPost) selectedArea.mediaArea).msg_id); + presentFragment(new ChatActivity(args)); + + selectedArea = null; + invalidate(); + return; + } + LocationActivity fragment = new LocationActivity(3) { @Override protected boolean disablePermissionCheck() { @@ -227,7 +251,7 @@ protected boolean disablePermissionCheck() { return; } - selectedArea = (AreaView) v; + selectedArea = lastSelectedArea = (AreaView) v; invalidate(); if (hintView != null) { hintView.hide(); @@ -236,7 +260,12 @@ protected boolean disablePermissionCheck() { boolean top = selectedArea.getTranslationY() < AndroidUtilities.dp(100); - SpannableStringBuilder text = new SpannableStringBuilder(LocaleController.getString("StoryViewLocation", R.string.StoryViewLocation)); + SpannableStringBuilder text = new SpannableStringBuilder(); + if (selectedArea.mediaArea instanceof TL_stories.TL_mediaAreaChannelPost) { + text.append(LocaleController.getString(R.string.StoryViewMessage)); + } else { + text.append(LocaleController.getString(R.string.StoryViewLocation)); + } SpannableString arrowRight = new SpannableString(">"); ColoredImageSpan imageSpan = new ColoredImageSpan(R.drawable.photos_arrow); imageSpan.translate(dp(2), dp(1)); @@ -261,7 +290,13 @@ protected boolean disablePermissionCheck() { onHintVisible(false); } }); - if (top) { + if (selectedArea.mediaArea instanceof TL_stories.TL_mediaAreaChannelPost && ( + top ? + selectedArea.getTranslationY() + selectedArea.getMeasuredHeight() / 2f > getMeasuredHeight() - dp(120) : + selectedArea.getTranslationY() - selectedArea.getMeasuredHeight() / 2f - dp(50) < dp(120) + )) { + hintView.setTranslationY(selectedArea.getTranslationY() - selectedArea.getMeasuredHeight() / 3f); + } else if (top) { hintView.setTranslationY(selectedArea.getTranslationY() + selectedArea.getMeasuredHeight() / 2f); } else { hintView.setTranslationY(selectedArea.getTranslationY() - selectedArea.getMeasuredHeight() / 2f - dp(50)); @@ -336,13 +371,15 @@ protected void onLayout(boolean changed, int left, int top, int right, int botto } + private final Rect rect = new Rect(); private final RectF rectF = new RectF(); private final Paint cutPaint = new Paint(Paint.ANTI_ALIAS_FLAG); { cutPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_OUT)); cutPaint.setColor(0xffffffff); } - public final AnimatedFloat parentHighlightAlpha = new AnimatedFloat(this, 0, 120, new LinearInterpolator()); + public final AnimatedFloat parentHighlightAlpha; + public final AnimatedFloat parentHighlightScaleAlpha; @Override protected boolean drawChild(Canvas canvas, View child, long drawingTime) { @@ -352,8 +389,13 @@ protected boolean drawChild(Canvas canvas, View child, long drawingTime) { return super.drawChild(canvas, child, drawingTime); } + private final Path clipPath = new Path(); + private final float[] radii = new float[8]; + private void drawHighlight(Canvas canvas) { - float parentAlpha = parentHighlightAlpha.set(selectedArea != null); + float parentAlpha = parentHighlightAlpha.set(selectedArea != null && selectedArea.supportsBounds && !selectedArea.scaleOnTap); + boolean scale = selectedArea != null && selectedArea.scaleOnTap; + float parentScale = parentHighlightScaleAlpha.set(scale); if (parentAlpha > 0) { canvas.saveLayerAlpha(0, 0, getMeasuredWidth(), getMeasuredHeight(), 0xFF, Canvas.ALL_SAVE_FLAG); canvas.drawColor(Theme.multAlpha(0x18000000, parentAlpha)); @@ -361,7 +403,7 @@ private void drawHighlight(Canvas canvas) { View child2 = getChildAt(i); if (child2 != hintsContainer) { AreaView areaView = (AreaView) child2; - float alpha = areaView.highlightAlpha.set(child2 == selectedArea); + float alpha = areaView.highlightAlpha.set(child2 == selectedArea && selectedArea.supportsBounds); if (alpha > 0) { canvas.save(); rectF.set(child2.getX(), child2.getY(), child2.getX() + child2.getMeasuredWidth(), child2.getY() + child2.getMeasuredHeight()); @@ -374,6 +416,49 @@ private void drawHighlight(Canvas canvas) { } canvas.restore(); } + if ((scale || parentScale > 0) && lastSelectedArea != null) { + if (parentBitmap == null) { + parentBitmap = getPlayingBitmap(); + } + if (parentBitmap != null) { + canvas.drawColor(Theme.multAlpha(0x30000000, parentScale)); + canvas.save(); + clipPath.rewind(); + rectF.set(lastSelectedArea.getX(), lastSelectedArea.getY(), lastSelectedArea.getX() + lastSelectedArea.getMeasuredWidth(), lastSelectedArea.getY() + lastSelectedArea.getMeasuredHeight()); + final float s = AndroidUtilities.lerp(1.0f, 1.05f, parentScale); + canvas.scale(s, s, rectF.centerX(), rectF.centerY()); + canvas.rotate(lastSelectedArea.getRotation(), rectF.centerX(), rectF.centerY()); + radii[0] = radii[1] = dp(16); + radii[2] = radii[3] = dp(16); + radii[4] = radii[5] = dp(16); + radii[6] = radii[7] = dp(8); + clipPath.addRoundRect(rectF, radii, Path.Direction.CW); + canvas.clipPath(clipPath); + AndroidUtilities.rectTmp.set(0, 0, getWidth(), getHeight()); + rect.set(0, 0, parentBitmap.getWidth(), parentBitmap.getHeight()); + canvas.rotate(-lastSelectedArea.getRotation(), rectF.centerX(), rectF.centerY()); + canvas.drawBitmap(parentBitmap, rect, AndroidUtilities.rectTmp, null); + canvas.restore(); + } + } else if (parentBitmap != null) { + parentBitmap.recycle(); + parentBitmap = null; + } + invalidate(); + } + + private Bitmap parentBitmap; + protected Bitmap getPlayingBitmap() { + return null; + } + + @Override + protected void onDetachedFromWindow() { + super.onDetachedFromWindow(); + if (parentBitmap != null) { + parentBitmap.recycle(); + parentBitmap = null; + } } private boolean shined = false; @@ -394,6 +479,10 @@ public boolean hasSelected() { return selectedArea != null; } + public boolean hasSelectedForScale() { + return selectedArea != null && (selectedArea.scaleOnTap || selectedArea.supportsBounds); + } + // returns true when widget that is drawn above the story (f.ex. reaction) is at these coordinates // used to detect that back gesture safety measure should not occur public boolean hasAreaAboveAt(float x, float y) { @@ -468,6 +557,9 @@ public static class AreaView extends View { public AreaView(Context context, View parent, TL_stories.MediaArea mediaArea) { super(context); this.mediaArea = mediaArea; + supportsBounds = mediaArea instanceof TL_stories.TL_mediaAreaGeoPoint || mediaArea instanceof TL_stories.TL_mediaAreaVenue; // || mediaArea instanceof TL_stories.TL_mediaAreaChannelPost; + supportsShining = mediaArea instanceof TL_stories.TL_mediaAreaGeoPoint || mediaArea instanceof TL_stories.TL_mediaAreaVenue; + scaleOnTap = false; // mediaArea instanceof TL_stories.TL_mediaAreaChannelPost; highlightAlpha = new AnimatedFloat(parent, 0, 120, new LinearInterpolator()); strokeGradientPaint.setStyle(Paint.Style.STROKE); } @@ -477,6 +569,9 @@ public AreaView(Context context, View parent, TL_stories.MediaArea mediaArea) { private LinearGradient gradient, strokeGradient; private final Matrix gradientMatrix = new Matrix(); + private boolean supportsBounds = false; + private boolean supportsShining = false; + private boolean scaleOnTap; private boolean shining = false; private long startTime; @@ -484,7 +579,7 @@ public AreaView(Context context, View parent, TL_stories.MediaArea mediaArea) { protected void onDraw(Canvas canvas) { super.onDraw(canvas); - if (shining && gradient != null) { + if (supportsShining && shining && gradient != null) { float w = getMeasuredWidth() * .7f; float t = (System.currentTimeMillis() - startTime) / 600f; float tx = t * (getMeasuredWidth() + w) - w; @@ -516,11 +611,17 @@ protected void onDraw(Canvas canvas) { private final Runnable shineRunnable = this::shineInternal; public void shine() { + if (!supportsShining) { + return; + } AndroidUtilities.cancelRunOnUIThread(shineRunnable); AndroidUtilities.runOnUIThread(shineRunnable, 400L); } private void shineInternal() { + if (!supportsShining) { + return; + } shining = true; startTime = System.currentTimeMillis(); gradient = new LinearGradient(0, 0, 40, 0, new int[] { 0x00ffffff, 0x2dffffff, 0x2dffffff, 0x00ffffff }, new float[] { 0, .4f, .6f, 1f }, Shader.TileMode.CLAMP ); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoryViewer.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoryViewer.java index c0ff28159ee..d1a9c3d8e92 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoryViewer.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/StoryViewer.java @@ -781,13 +781,6 @@ protected void dispatchDraw(Canvas canvas) { } } canvas.restore(); -// -// if (transitionViewHolder.avatarImage != null) { -// transitionViewHolder.avatarImage.setVisible(false, true); -// } -// if (transitionViewHolder.storyImage != null) { -// transitionViewHolder.storyImage.setVisible(false, true); -// } } if (runOpenAnimationAfterLayout) { @@ -2200,6 +2193,9 @@ public void instantClose() { } AndroidUtilities.hideKeyboard(windowView); isClosed = true; + fullyVisible = false; + progressToOpen = 0; + progressToDismiss = 0; updatePlayingMode(); fromX = fromY = 0; if (transitionViewHolder.avatarImage != null) { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/ButtonWithCounterView.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/ButtonWithCounterView.java index c58074d0337..4e5b8550963 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/ButtonWithCounterView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/ButtonWithCounterView.java @@ -145,10 +145,14 @@ public boolean isTimerActive() { } public void setText(CharSequence newText, boolean animated) { + setText(newText, animated, true); + } + + public void setText(CharSequence newText, boolean animated, boolean moveDown) { if (animated) { text.cancelAnimation(); } - text.setText(newText, animated); + text.setText(newText, animated, moveDown); setContentDescription(newText); invalidate(); } @@ -343,6 +347,10 @@ protected boolean drawChild(Canvas canvas, View child, long drawingTime) { private int globalAlpha = 255; private final int subTextAlpha = 200; + protected float calculateCounterWidth(float width, float percent) { + return width * percent; + } + @Override protected void onDraw(Canvas canvas) { rippleView.draw(canvas); @@ -370,7 +378,7 @@ protected void onDraw(Canvas canvas) { float countAlpha = countAlphaAnimated.set(this.countAlpha); float lightningWidth = withCounterIcon ? AndroidUtilities.dp(12) : 0; - float width = textWidth + lightningWidth + (dp(5.66f + 5 + 5) + countText.getCurrentWidth()) * countAlpha; + float width = textWidth + lightningWidth + calculateCounterWidth((dp(5.66f + 5 + 5) + countText.getCurrentWidth()), countAlpha); AndroidUtilities.rectTmp2.set( (int) ((getMeasuredWidth() - width - getWidth()) / 2f), (int) ((getMeasuredHeight() - text.getHeight()) / 2f - dp(1)), @@ -419,7 +427,8 @@ protected void onDraw(Canvas canvas) { canvas.drawRoundRect(AndroidUtilities.rectTmp, radius, radius, paint); } - AndroidUtilities.rectTmp2.offset(-dp(.3f), -dp(.4f)); + int countLength = countText.getText() != null ? countText.getText().length() : 0; + AndroidUtilities.rectTmp2.offset(-dp(countLength > 1 ? .3f : 0), -dp(.4f)); countText.setAlpha((int) (globalAlpha * (1f - loadingT) * countAlpha * (countFilled ? 1 : .5f))); countText.setBounds(AndroidUtilities.rectTmp2); canvas.save(); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/CaptionContainerView.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/CaptionContainerView.java index 68439590329..3ccbc71344d 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/CaptionContainerView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/CaptionContainerView.java @@ -279,7 +279,7 @@ public void afterTextChanged(Editable s) { limitTextView.cancelAnimation(); limitTextView.setText(limitText); limitTextView.setTextColor(codePointCount >= limit ? 0xffEC7777 : 0xffffffff); - if (codePointCount > limit && !UserConfig.getInstance(currentAccount).isPremium() && codePointCount < getCaptionPremiumLimit() && codePointCount > lastLength && (captionLimitToast() || MessagesController.getInstance(currentAccount).premiumLocked)) { + if (codePointCount > limit && !UserConfig.getInstance(currentAccount).isPremium() && codePointCount < getCaptionPremiumLimit() && codePointCount > lastLength && (captionLimitToast() || MessagesController.getInstance(currentAccount).premiumFeaturesBlocked())) { AndroidUtilities.shakeViewSpring(limitTextView, shiftDp = -shiftDp); BotWebViewVibrationEffect.APP_ERROR.vibrate(); } @@ -369,13 +369,13 @@ public void closeKeyboard() { public boolean ignoreTouches; - protected boolean ignoreTouches() { + protected boolean ignoreTouches(float x, float y) { return false; } @Override public boolean dispatchTouchEvent(MotionEvent ev) { - if (ignoreTouches || ignoreTouches() || !bounds.contains(ev.getX(), ev.getY()) && !keyboardShown) { + if (ignoreTouches || ev.getAction() == MotionEvent.ACTION_DOWN && ignoreTouches(ev.getX(), ev.getY()) || !bounds.contains(ev.getX(), ev.getY()) && !keyboardShown) { return false; } if (ev.getAction() == MotionEvent.ACTION_DOWN && !keyboardShown) { @@ -1236,7 +1236,7 @@ protected void onDetachedFromWindow() { public static class PeriodDrawable extends Drawable { - private final Paint strokePaint = new Paint(Paint.ANTI_ALIAS_FLAG); + public final Paint strokePaint = new Paint(Paint.ANTI_ALIAS_FLAG); private final Paint fillPaint = new Paint(Paint.ANTI_ALIAS_FLAG); public final AnimatedTextView.AnimatedTextDrawable textDrawable = new AnimatedTextView.AnimatedTextDrawable(true, false, false) { @Override @@ -1244,11 +1244,26 @@ public void invalidateSelf() { PeriodDrawable.this.invalidateSelf(); } }; + public final AnimatedTextView.AnimatedTextDrawable activeTextDrawable = new AnimatedTextView.AnimatedTextDrawable(true, false, false) { + @Override + public void invalidateSelf() { + PeriodDrawable.this.invalidateSelf(); + } + }; private boolean filled = false; private final AnimatedFloat fillT = new AnimatedFloat(this::invalidateSelf, 0, 350, CubicBezierInterpolator.EASE_OUT_QUINT); + private final Path activePath = new Path(); + private final int dashes; + public float diameterDp = 21; public PeriodDrawable() { + this(5); + } + + public PeriodDrawable(int dashes) { + this.dashes = dashes; + strokePaint.setStyle(Paint.Style.STROKE); strokePaint.setStrokeWidth(dpf2(1.66f)); strokePaint.setStrokeCap(Paint.Cap.ROUND); @@ -1258,33 +1273,74 @@ public PeriodDrawable() { textDrawable.setTextSize(dpf2(12)); textDrawable.setGravity(Gravity.CENTER); - updateColors(0xffffffff, 0xff1A9CFF); + activeTextDrawable.setAnimationProperties(.3f, 0, 250, CubicBezierInterpolator.EASE_OUT_QUINT); + activeTextDrawable.setTypeface(AndroidUtilities.getTypeface("fonts/num.otf")); + activeTextDrawable.setTextSize(dpf2(12)); + activeTextDrawable.setGravity(Gravity.CENTER); + + updateColors(0xffffffff, 0xff1A9CFF, 0xffffffff); + } + + public void setTextSize(float dp) { + activeTextDrawable.setTextSize(dpf2(dp)); + textDrawable.setTextSize(dpf2(dp)); } - public void updateColors(int strokeColor, int fillColor) { + public float textOffsetX, textOffsetY; + + public void updateColors(int strokeColor, int fillColor, int activeTextColor) { strokePaint.setColor(strokeColor); textDrawable.setTextColor(strokeColor); + activeTextDrawable.setTextColor(activeTextColor); fillPaint.setColor(fillColor); } + private boolean clear; + public void setClear(boolean clear) { + if (this.clear != clear) { + this.clear = clear; + strokePaint.setXfermode(clear ? new PorterDuffXfermode(PorterDuff.Mode.CLEAR) : null); + textDrawable.getPaint().setXfermode(clear ? new PorterDuffXfermode(PorterDuff.Mode.CLEAR) : null); + } + } + + private float cx, cy; + public void setCenterXY(float x, float y) { + this.cx = x; + this.cy = y; + } + @Override - public void draw(@NonNull Canvas canvas) { - final float cx = getBounds().centerY(); - final float cy = getBounds().centerY(); + public void setBounds(@NonNull Rect bounds) { + super.setBounds(bounds); + this.cx = getBounds().centerX(); + this.cy = getBounds().centerY(); + } - final float r = dpf2(21) / 2f; + @Override + public void setBounds(int left, int top, int right, int bottom) { + super.setBounds(left, top, right, bottom); + this.cx = getBounds().centerX(); + this.cy = getBounds().centerY(); + } + + @Override + public void draw(@NonNull Canvas canvas) { + draw(canvas, 1f); + } + public void draw(@NonNull Canvas canvas, float alpha) { + final float r = dpf2(diameterDp) / 2f; final float fillT = this.fillT.set(filled); if (fillT > 0) { - fillPaint.setAlpha((int) (0xFF * fillT)); + fillPaint.setAlpha((int) (0xFF * alpha * fillT)); canvas.drawCircle(cx, cy, dpf2(11.33f) * fillT, fillPaint); } - strokePaint.setAlpha((int) (0xFF * (1f - fillT))); + strokePaint.setAlpha((int) (0xFF * alpha * (1f - fillT))); AndroidUtilities.rectTmp.set(cx - r, cy - r, cx + r, cy + r); canvas.drawArc(AndroidUtilities.rectTmp, 90, 180, false, strokePaint); - final int dashes = 5; final int gaps = dashes + 1; final float dashWeight = 1f, gapWeight = 1.5f; final float dashSweep = dashWeight / (dashes * dashWeight + gaps * gapWeight) * 180; @@ -1296,7 +1352,7 @@ public void draw(@NonNull Canvas canvas) { } canvas.save(); - canvas.translate(0, -1); + canvas.translate(textOffsetX + 0, textOffsetY); AndroidUtilities.rectTmp2.set( (int) (cx - dp(20)), (int) (cy - dp(20)), @@ -1304,12 +1360,22 @@ public void draw(@NonNull Canvas canvas) { (int) (cy + dp(20)) ); textDrawable.setBounds(AndroidUtilities.rectTmp2); + textDrawable.setAlpha((int) (0xFF * alpha)); textDrawable.draw(canvas); + if (fillT > 0) { + activePath.rewind(); + activePath.addCircle(cx, cy + dp(1), dpf2(11.33f) * fillT, Path.Direction.CW); + canvas.clipPath(activePath); + activeTextDrawable.setBounds(AndroidUtilities.rectTmp2); + activeTextDrawable.setAlpha((int) (0xFF * alpha)); + activeTextDrawable.draw(canvas); + } canvas.restore(); } public void setValue(int num, boolean fill, boolean animated) { textDrawable.setText("" + num, animated); + activeTextDrawable.setText("" + num, animated); filled = fill; if (!animated) { fillT.set(filled, true); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/CaptionStory.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/CaptionStory.java index 82dd9c7f16b..4caa9ac4cf8 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/CaptionStory.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/CaptionStory.java @@ -67,7 +67,6 @@ public class CaptionStory extends CaptionContainerView { private boolean periodVisible = true; public static final int[] periods = new int[] { 6 * 3600, 12 * 3600, 86400, 2 * 86400 }; - public static final int[] periodDrawables = new int[] { R.drawable.msg_story_6h, R.drawable.msg_story_12h, R.drawable.msg_story_24h, R.drawable.msg_story_48h }; private int periodIndex = 0; private Drawable flipButton; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/DraftsController.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/DraftsController.java index 876204da078..092b4e722cf 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/DraftsController.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/DraftsController.java @@ -54,6 +54,7 @@ private void loadInternal(final boolean failed, Utilities.Callback todelete = new ArrayList<>(); cursor = database.queryFinalized("SELECT id, data, type FROM story_drafts WHERE type = " + (failed ? "2" : "0 OR type = 1") + " ORDER BY date DESC"); while (cursor.next()) { long id = cursor.longValue(0); @@ -65,10 +66,19 @@ private void loadInternal(final boolean failed, Utilities.Callback 0) { + for (int i = 0; i < todelete.size(); ++i) { + database.executeFast("DELETE FROM story_drafts WHERE id = " + todelete.get(i)).stepThis().dispose(); + } + } } catch (Exception e) { FileLog.e(e); } finally { @@ -253,7 +263,7 @@ private File prepareFile(File file) { } public void append(StoryEntry entry) { - if (entry == null) { + if (entry == null || entry.isRepostMessage) { return; } prepare(entry); @@ -330,7 +340,7 @@ public void deleteForEdit(long peerId, int storyId) { } public void saveForEdit(StoryEntry entry, long dialogId, TL_stories.StoryItem storyItem) { - if (entry == null || storyItem == null || storyItem.media == null) { + if (entry == null || entry.isRepostMessage || storyItem == null || storyItem.media == null) { return; } @@ -477,8 +487,6 @@ public static class StoryDraft { private int period; - private final ArrayList parts = new ArrayList<>(); - public boolean isEdit; public int editStoryId; public long editStoryPeerId; @@ -541,8 +549,6 @@ public StoryDraft(@NonNull StoryEntry entry) { this.filterFilePath = entry.filterFile == null ? "" : entry.filterFile.toString(); this.filterState = entry.filterState; this.period = entry.period; - this.parts.clear(); - this.parts.addAll(entry.parts); this.isError = entry.isError; this.error = entry.error; @@ -629,12 +635,6 @@ public StoryEntry toEntry() { } entry.filterState = filterState; entry.period = period; - entry.parts.clear(); - entry.parts.addAll(parts); - entry.partsMaxId = 0; - for (int i = 0; i < parts.size(); ++i) { - entry.partsMaxId = Math.max(entry.partsMaxId, parts.get(i).id); - } entry.isEdit = isEdit; entry.editStoryId = editStoryId; entry.editStoryPeerId = editStoryPeerId; @@ -732,10 +732,7 @@ public void toStream(AbstractSerializedData stream) { } stream.writeInt32(period); stream.writeInt32(0x1cb5c415); - stream.writeInt32(parts.size()); - for (int i = 0; i < parts.size(); ++i) { - parts.get(i).serializeToStream(stream); - } + stream.writeInt32(0); stream.writeBool(isEdit); stream.writeInt32(editStoryId); stream.writeInt64(editStoryPeerId); @@ -881,7 +878,7 @@ public StoryDraft(@NonNull AbstractSerializedData stream, boolean exception) { if (mediaEntities == null) { mediaEntities = new ArrayList<>(); } - mediaEntities.add(new VideoEditedInfo.MediaEntity(stream, true)); + mediaEntities.add(new VideoEditedInfo.MediaEntity(stream, true, exception)); } magic = stream.readInt32(exception); if (magic != 0x1cb5c415) { @@ -918,12 +915,6 @@ public StoryDraft(@NonNull AbstractSerializedData stream, boolean exception) { return; } count = stream.readInt32(exception); - parts.clear(); - for (int i = 0; i < count; ++i) { - StoryEntry.Part part = new StoryEntry.Part(); - part.readParams(stream, exception); - parts.add(part); - } } if (stream.remaining() > 0) { isEdit = stream.readBool(exception); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/EmojiBottomSheet.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/EmojiBottomSheet.java index 7a73bb6add7..04293febe79 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/EmojiBottomSheet.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/EmojiBottomSheet.java @@ -1216,12 +1216,12 @@ public int getItemCount() { } } - public void showPremiumBulletin(String str, int resId) { + public void showPremiumBulletin(String text) { container.performHapticFeedback(HapticFeedbackConstants.KEYBOARD_TAP); BulletinFactory.of(container, resourcesProvider).createSimpleBulletin( - ContextCompat.getDrawable(getContext(), R.drawable.msg_premium_normal), - LocaleController.getString("IncreaseLimit", R.string.IncreaseLimit), - premiumText(LocaleController.getString(str, resId)) + R.raw.star_premium_2, + LocaleController.getString(R.string.IncreaseLimit), + premiumText(text) ).show(true); } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/HintView2.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/HintView2.java index 3b7e3926298..eae8dba7696 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/HintView2.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/HintView2.java @@ -80,13 +80,13 @@ public class HintView2 extends View { private Drawable closeButtonDrawable; private boolean closeButton; - private float rounding = dp(8); + protected float rounding = dp(8); private final RectF innerPadding = new RectF(dp(11), dp(6), dp(11), dp(7)); private float closeButtonMargin = dp(2); private float arrowHalfWidth = dp(7); private float arrowHeight = dp(6); - private final Paint backgroundPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + protected final Paint backgroundPaint = new Paint(Paint.ANTI_ALIAS_FLAG); private CharSequence textToSet; private AnimatedTextView.AnimatedTextDrawable textDrawable; @@ -212,6 +212,12 @@ public HintView2 setMaxWidthPx(int widthPx) { return this; } + public HintView2 setIcon(int resId) { + RLottieDrawable icon = new RLottieDrawable(resId, "" + resId, dp(34), dp(34)); + icon.start(); + return setIcon(icon); + } + public HintView2 setIcon(Drawable icon) { if (this.icon != null) { this.icon.setCallback(null); @@ -260,6 +266,7 @@ private static float measureCorrectly(CharSequence text, TextPaint paint) { len += paint.measureText(spanned, s, e); paint.setTypeface(oldTypeface); } + s = e; } e = Math.max(s, text.length()); if (e - s > 0) { @@ -275,15 +282,16 @@ public static int cutInFancyHalf(CharSequence text, TextPaint paint) { float prevLeftWidth = 0; float prevRightWidth = Float.MAX_VALUE; + int dir = -1; for (int i = 0; i < 10; ++i) { // Adjust the mid to point to the nearest space on the left - while (mid > 0 && text.charAt(mid) != ' ') { - mid--; + while (mid > 0 && mid < text.length() && text.charAt(mid) != ' ') { + mid += dir; } - leftWidth = measureCorrectly(text.subSequence(0, mid).toString(), paint); - rightWidth = measureCorrectly(text.subSequence(mid, text.length()).toString().trim(), paint); + leftWidth = measureCorrectly(text.subSequence(0, mid), paint); + rightWidth = measureCorrectly(AndroidUtilities.getTrimmedString(text.subSequence(mid, text.length())), paint); // If we're not making progress, exit the loop. // (This is a basic way to ensure termination when we can't improve the result.) @@ -296,11 +304,13 @@ public static int cutInFancyHalf(CharSequence text, TextPaint paint) { // If left side is shorter, move midpoint to the right. if (leftWidth < rightWidth) { - mid++; + dir = +1; + mid += dir; } // If right side is shorter or equal, move midpoint to the left. else { - mid--; + dir = -1; + mid += dir; } // Ensure mid doesn't go out of bounds @@ -585,12 +595,16 @@ private void makeLayout(CharSequence text, int width) { private final Rect boundsWithArrow = new Rect(); private final RectF bounds = new RectF(); - private final Path path = new Path(); + protected final Path path = new Path(); private float arrowX, arrowY; private float pathLastWidth, pathLastHeight; private boolean pathSet; private boolean firstDraw = true; + protected void drawBgPath(Canvas canvas) { + canvas.drawPath(path, backgroundPaint); + } + @Override protected void dispatchDraw(Canvas canvas) { if (multiline && textLayout == null) { @@ -656,7 +670,7 @@ protected void dispatchDraw(Canvas canvas) { backgroundAlpha *= .2f; } backgroundPaint.setAlpha((int) (wasAlpha * backgroundAlpha)); - canvas.drawPath(path, backgroundPaint); + drawBgPath(canvas); backgroundPaint.setAlpha(wasAlpha); if (selectorDrawable != null) { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/IStoryPart.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/IStoryPart.java deleted file mode 100644 index 910e2c8a927..00000000000 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/IStoryPart.java +++ /dev/null @@ -1,11 +0,0 @@ -package org.telegram.ui.Stories.recorder; - -import android.graphics.Matrix; - -import java.io.File; - -public abstract class IStoryPart { - public int id; - public int width, height; - public final Matrix matrix = new Matrix(); -} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/PaintView.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/PaintView.java index 25ef77662fa..c07b4bedf22 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/PaintView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/PaintView.java @@ -42,6 +42,7 @@ import android.view.HapticFeedbackConstants; import android.view.KeyEvent; import android.view.MotionEvent; +import android.view.TextureView; import android.view.View; import android.view.ViewGroup; import android.view.WindowManager; @@ -66,10 +67,12 @@ import org.telegram.messenger.ApplicationLoader; import org.telegram.messenger.Bitmaps; import org.telegram.messenger.BuildVars; +import org.telegram.messenger.DialogObject; import org.telegram.messenger.DispatchQueue; import org.telegram.messenger.Emoji; import org.telegram.messenger.FileLoader; import org.telegram.messenger.FileLog; +import org.telegram.messenger.ImageReceiver; import org.telegram.messenger.LocaleController; import org.telegram.messenger.MediaController; import org.telegram.messenger.MessageObject; @@ -87,6 +90,7 @@ import org.telegram.ui.ActionBar.AlertDialog; import org.telegram.ui.ActionBar.Theme; import org.telegram.ui.BubbleActivity; +import org.telegram.ui.Cells.ChatMessageCell; import org.telegram.ui.ChatActivity; import org.telegram.ui.Components.AnimatedEmojiDrawable; import org.telegram.ui.Components.AnimatedEmojiSpan; @@ -112,6 +116,7 @@ import org.telegram.ui.Components.Paint.Views.EntitiesContainerView; import org.telegram.ui.Components.Paint.Views.EntityView; import org.telegram.ui.Components.Paint.Views.LocationView; +import org.telegram.ui.Components.Paint.Views.MessageEntityView; import org.telegram.ui.Components.Paint.Views.PaintCancelView; import org.telegram.ui.Components.Paint.Views.PaintColorsListView; import org.telegram.ui.Components.Paint.Views.PaintDoneView; @@ -161,6 +166,7 @@ public class PaintView extends SizeNotifierFrameLayoutPhoto implements IPhotoPai private boolean ignoreLayout; private float baseScale; private Size paintingSize; + public boolean drawForThemeToggle, clipVideoMessageForBitmap; private EntityView currentEntityView; private boolean editingText; @@ -272,11 +278,14 @@ public void set(float val) { private boolean reactionLayoutShowing; private boolean invalidateReactionPosition; private BlurringShader.BlurManager blurManager; + private PreviewView.TextureViewHolder videoTextureHolder; @SuppressLint("NotifyDataSetChanged") - public PaintView(Context context, boolean fileFromGallery, File file, boolean isVideo, StoryRecorder.WindowView parent, Activity activity, int currentAccount, Bitmap bitmap, Bitmap blurBitmap, Bitmap originalBitmap, int originalRotation, ArrayList entities, StoryEntry entry, int viewWidth, int viewHeight, MediaController.CropState cropState, Runnable onInit, BlurringShader.BlurManager blurManager, Theme.ResourcesProvider resourcesProvider) { + public PaintView(Context context, boolean fileFromGallery, File file, boolean isVideo, StoryRecorder.WindowView parent, Activity activity, int currentAccount, Bitmap bitmap, Bitmap blurBitmap, Bitmap originalBitmap, int originalRotation, ArrayList entities, StoryEntry entry, int viewWidth, int viewHeight, MediaController.CropState cropState, Runnable onInit, BlurringShader.BlurManager blurManager, Theme.ResourcesProvider resourcesProvider, PreviewView.TextureViewHolder videoTextureHolder) { super(context, activity, true); setDelegate(this); + this.blurManager = blurManager; + this.videoTextureHolder = videoTextureHolder; this.fileFromGallery = fileFromGallery; this.file = file; this.isVideo = isVideo; @@ -385,7 +394,7 @@ public ColorFilter getAnimatedEmojiColorFilter() { textDim.setBackgroundColor(0x4d000000); textDim.setAlpha(0f); - renderView = new RenderView(context, new Painting(getPaintingSize(), originalBitmap, originalRotation, blurManager), bitmapToEdit, blurBitmapToEdit, blurManager) { + renderView = new RenderView(context, new Painting(getPaintingSize(), originalBitmap, originalRotation, blurManager), bitmapToEdit, blurBitmapToEdit, entry != null && entry.isRepostMessage ? null : blurManager) { @Override public void selectBrush(Brush brush) { int index = 1 + Brush.BRUSHES_LIST.indexOf(brush); @@ -800,7 +809,7 @@ protected void onDraw(Canvas canvas) { bottomLayout.setBackground(new GradientDrawable(GradientDrawable.Orientation.TOP_BOTTOM, new int [] {0x00000000, 0x80000000} )); addView(bottomLayout, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 44 + 60, Gravity.BOTTOM)); - paintToolsView = new PaintToolsView(context, blurManager != null); + paintToolsView = new PaintToolsView(context, !entry.isRepostMessage && blurManager != null); paintToolsView.setPadding(dp(16), 0, dp(16), 0); paintToolsView.setDelegate(this); // paintToolsView.setSelectedIndex(MathUtils.clamp(palette.getCurrentBrush(), 0, Brush.BRUSHES_LIST.size()) + 1); @@ -1770,15 +1779,15 @@ public boolean canClickWidget(Integer widgetId) { widgetsCount++; } } - if (widgetsCount >= 1 && !UserConfig.getInstance(currentAccount).isPremium()) { - showPremiumBulletin("StoryPremiumWidgets", R.string.StoryPremiumWidgets); + if (widgetsCount >= MessagesController.getInstance(currentAccount).storiesSuggestedReactionsLimitDefault && !UserConfig.getInstance(currentAccount).isPremium()) { + showPremiumBulletin(LocaleController.formatPluralString("StoryPremiumWidgets2", MessagesController.getInstance(currentAccount).storiesSuggestedReactionsLimitPremium)); return false; } - if (widgetsCount >= 5) { + if (widgetsCount >= MessagesController.getInstance(currentAccount).storiesSuggestedReactionsLimitPremium) { container.performHapticFeedback(HapticFeedbackConstants.KEYBOARD_TAP); BulletinFactory.of(container, resourcesProvider).createSimpleBulletin(R.raw.chats_infotip, LocaleController.getString("LimitReached", R.string.LimitReached), - LocaleController.getString("StoryReactionsWidgetLimit", R.string.StoryReactionsWidgetLimit) + LocaleController.formatPluralString("StoryReactionsWidgetLimit2", MessagesController.getInstance(currentAccount).storiesSuggestedReactionsLimitPremium) ).show(true); return false; } @@ -2145,13 +2154,25 @@ private void setupEntities() { view = textPaintView; } else if (entity.type == VideoEditedInfo.MediaEntity.TYPE_PHOTO) { PhotoView photoView = createPhoto(entity.text, false); + photoView.preloadSegmented(entity.segmentedPath); if ((entity.subType & 2) != 0) { photoView.mirror(); } + if ((entity.subType & 16) != 0) { + photoView.toggleSegmented(false); + } view = photoView; ViewGroup.LayoutParams layoutParams = view.getLayoutParams(); layoutParams.width = entity.viewWidth; layoutParams.height = entity.viewHeight; + } else if (entity.type == VideoEditedInfo.MediaEntity.TYPE_MESSAGE) { + MessageEntityView messageView = createMessage(entry.messageObjects, false, entry.isVideo); + view = messageView; + if (entity.viewWidth > 0 && entity.viewHeight > 0) { + ViewGroup.LayoutParams layoutParams = view.getLayoutParams(); + layoutParams.width = entity.viewWidth; + layoutParams.height = entity.viewHeight; + } } else if (entity.type == VideoEditedInfo.MediaEntity.TYPE_LOCATION) { LocationView locationView = createLocationSticker(entity.mediaGeo, entity.mediaArea, false); locationView.setType(entity.subType, entity.color); @@ -2310,13 +2331,15 @@ public static boolean isVideoStickerDocument(TLRPC.Document document) { @Override public Bitmap getBitmap(ArrayList entities, Bitmap[] thumbBitmap) { - return getBitmap(entities, (int) paintingSize.width, (int) paintingSize.height, true, true, false, null); + return getBitmap(entities, (int) paintingSize.width, (int) paintingSize.height, true, true, false, false, null); } - public Bitmap getBitmap(ArrayList entities, int resultWidth, int resultHeight, boolean drawPaint, boolean drawEntities, boolean drawBlur, StoryEntry entry) { + public Bitmap getBitmap(ArrayList entities, int resultWidth, int resultHeight, boolean drawPaint, boolean drawEntities, boolean drawMessage, boolean drawBlur, StoryEntry entry) { Bitmap bitmap; if (drawPaint) { bitmap = renderView.getResultBitmap(false, drawBlur); + } else if (drawMessage) { + bitmap = Bitmap.createBitmap(Math.max(1, entitiesView.getMeasuredWidth()), Math.max(1, entitiesView.getMeasuredHeight()), Bitmap.Config.ARGB_8888); } else if (drawEntities) { Bitmap ref = renderView.getResultBitmap(false, false); if (ref != null) { @@ -2440,6 +2463,13 @@ public Bitmap getBitmap(ArrayList entities, int res if (photoView.isMirrored()) { mediaEntity.subType |= 2; } + if (photoView.hasSegmentedImage() && photoView.isSegmented()) { + File segmentedFile = photoView.saveSegmentedImage(currentAccount); + if (segmentedFile != null) { + mediaEntity.subType |= 16; + mediaEntity.segmentedPath = segmentedFile.getPath(); + } + } } else if (entity instanceof LocationView) { LocationView locationView = (LocationView) entity; mediaEntity.type = VideoEditedInfo.MediaEntity.TYPE_LOCATION; @@ -2491,6 +2521,44 @@ public Bitmap getBitmap(ArrayList entities, int res if (roundView.isMirrored()) { mediaEntity.subType |= 2; } + } else if (entity instanceof MessageEntityView) { + MessageEntityView messageView = (MessageEntityView) entity; + mediaEntity.type = VideoEditedInfo.MediaEntity.TYPE_MESSAGE; + mediaEntity.width = mediaEntity.viewWidth = messageView.getWidth(); + mediaEntity.height = mediaEntity.viewHeight = messageView.getHeight(); + mediaEntity.mediaArea = new TL_stories.TL_inputMediaAreaChannelPost(); + mediaEntity.mediaArea.coordinates = new TL_stories.TL_mediaAreaCoordinates(); + if (entry != null && entry.messageObjects != null) { + MessageObject messageObject = entry.messageObjects.get(0); + ((TL_stories.TL_inputMediaAreaChannelPost) mediaEntity.mediaArea).channel = MessagesController.getInstance(currentAccount).getInputChannel(-StoryEntry.getRepostDialogId(messageObject)); + ((TL_stories.TL_inputMediaAreaChannelPost) mediaEntity.mediaArea).msg_id = StoryEntry.getRepostMessageId(messageObject); + } + if (!drawMessage) { + skipDrawToBitmap = true; + } else if (entry != null && entry.isVideo) { + entry.matrix.reset(); + View child = null; + ImageReceiver photoImage = null; + if (messageView.listView.getChildCount() == 1) { + child = messageView.listView.getChildAt(0); + if (child instanceof ChatMessageCell) { + photoImage = ((ChatMessageCell) child).getPhotoImage(); + } + } + if (photoImage != null) { + float scale = Math.max(photoImage.getImageWidth() / Math.max(1, entry.width), photoImage.getImageHeight() / Math.max(1, entry.height)); + entry.matrix.postScale(scale, scale); + entry.matrix.postTranslate(photoImage.getCenterX() - entry.width * scale / 2f, photoImage.getCenterY() - entry.height * scale / 2f); + entry.matrix.postTranslate(messageView.container.getX(), messageView.container.getY()); + entry.matrix.postTranslate(messageView.listView.getX(), messageView.listView.getY()); + entry.matrix.postTranslate(child.getX(), child.getY()); + entry.matrix.postScale(messageView.getScaleX(), messageView.getScaleY(), messageView.getPivotX(), messageView.getPivotY()); + entry.matrix.postRotate(messageView.getRotation(), messageView.getPivotX(), messageView.getPivotY()); + entry.matrix.postTranslate(messageView.getX(), messageView.getY()); + entry.matrix.postScale(1f / entitiesView.getWidth(), 1f / entitiesView.getHeight()); + entry.matrix.postScale(entry.resultWidth, entry.resultHeight); + } + } } else { continue; } @@ -2513,7 +2581,17 @@ public Bitmap getBitmap(ArrayList entities, int res mediaEntity.textViewHeight = mediaEntity.viewHeight / (float) entitiesView.getMeasuredHeight(); mediaEntity.scale = scaleX; - if (entity instanceof StickerView) { + if (entity instanceof MessageEntityView) { + MessageEntityView mv = (MessageEntityView) entity; + mv.getBubbleBounds(AndroidUtilities.rectTmp); + AndroidUtilities.rectTmp.offset(mv.container.getX(), mv.container.getY()); + AndroidUtilities.rectTmp.offset(mv.listView.getX(), mv.listView.getY()); + mediaEntity.mediaArea.coordinates.x = (x + v.getWidth() / 2f - v.getWidth() / 2f * scaleX + AndroidUtilities.rectTmp.centerX() * scaleX) / entitiesView.getMeasuredWidth() * 100; + mediaEntity.mediaArea.coordinates.y = (y + v.getHeight() / 2f - v.getHeight() / 2f * scaleY + AndroidUtilities.rectTmp.centerY() * scaleY) / entitiesView.getMeasuredHeight() * 100; + mediaEntity.mediaArea.coordinates.w = AndroidUtilities.rectTmp.width() * scaleX / entitiesView.getMeasuredWidth() * 100; + mediaEntity.mediaArea.coordinates.h = AndroidUtilities.rectTmp.height() * scaleY / entitiesView.getMeasuredHeight() * 100; + mediaEntity.mediaArea.coordinates.rotation = -mediaEntity.rotation / Math.PI * 180; + } else if (entity instanceof StickerView) { final float a = ((StickerView) entity).centerImage.getImageAspectRatio(); final float cx = mediaEntity.x + mediaEntity.width / 2f; final float cy = mediaEntity.y + mediaEntity.height / 2f; @@ -2534,7 +2612,7 @@ public Bitmap getBitmap(ArrayList entities, int res if (entity instanceof LocationView) { mediaEntity.mediaArea.coordinates.w = (mediaEntity.width - 2 * ((LocationView) entity).marker.padx * scaleX / (float) entitiesView.getMeasuredWidth()) * 100; mediaEntity.mediaArea.coordinates.h = (mediaEntity.height - 2 * ((LocationView) entity).marker.pady * scaleY / (float) entitiesView.getMeasuredHeight()) * 100; - } else if (entity instanceof ReactionWidgetEntityView){ + } else if (entity instanceof ReactionWidgetEntityView) { float padW = 2 * ((ReactionWidgetEntityView) entity).getPadding() * scaleX / (float) entitiesView.getMeasuredWidth(); float padH = 2 * ((ReactionWidgetEntityView) entity).getPadding() * scaleX / (float) entitiesView.getMeasuredHeight(); mediaEntity.mediaArea.coordinates.w = (mediaEntity.width - padW) * 100; @@ -2543,7 +2621,7 @@ public Bitmap getBitmap(ArrayList entities, int res mediaEntity.mediaArea.coordinates.rotation = -mediaEntity.rotation / Math.PI * 180; } } - if (drawEntities && bitmap != null) { + if ((drawEntities || drawMessage && mediaEntity.type == VideoEditedInfo.MediaEntity.TYPE_MESSAGE) && bitmap != null) { canvas = new Canvas(bitmap); final float s = bitmap.getWidth() / (float) entitiesView.getMeasuredWidth(); for (int k = 0; k < 2; k++) { @@ -2572,7 +2650,14 @@ public Bitmap getBitmap(ArrayList entities, int res } b.recycle(); } else { - v.draw(currentCanvas); + if (v instanceof MessageEntityView) { + MessageEntityView mv = (MessageEntityView) v; + mv.prepareToDraw(true); + v.draw(currentCanvas); + mv.prepareToDraw(false); + } else { + v.draw(currentCanvas); + } } currentCanvas.restore(); } @@ -3392,6 +3477,14 @@ public void onAddButtonPressed(View btn) { } private void showMenuForEntity(final EntityView entityView) { + if (entityView instanceof MessageEntityView) { + if (popupWindow != null && popupWindow.isShowing()) { + popupWindow.dismiss(); + return; + } + return; + } + int[] pos = getCenterLocationInWindow(entityView); int x = pos[0]; int y = pos[1] - dp(32); @@ -3400,29 +3493,31 @@ private void showMenuForEntity(final EntityView entityView) { LinearLayout parent = new LinearLayout(getContext()); parent.setOrientation(LinearLayout.HORIZONTAL); - TextView deleteView = new TextView(getContext()); - deleteView.setTextColor(getThemedColor(Theme.key_actionBarDefaultSubmenuItem)); - deleteView.setGravity(Gravity.CENTER_VERTICAL); - deleteView.setLines(1); - deleteView.setSingleLine(); - deleteView.setEllipsize(TextUtils.TruncateAt.END); - deleteView.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); - deleteView.setPadding(dp(16), 0, dp(16), 0); - deleteView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 14); - deleteView.setTag(0); - deleteView.setText(LocaleController.getString("PaintDelete", R.string.PaintDelete)); - deleteView.setOnClickListener(v -> { - if (entityView instanceof RoundView) { - onTryDeleteRound(); - } else { - removeEntity(entityView); - } + if (!(entityView instanceof MessageEntityView)) { + TextView deleteView = new TextView(getContext()); + deleteView.setTextColor(getThemedColor(Theme.key_actionBarDefaultSubmenuItem)); + deleteView.setGravity(Gravity.CENTER_VERTICAL); + deleteView.setLines(1); + deleteView.setSingleLine(); + deleteView.setEllipsize(TextUtils.TruncateAt.END); + deleteView.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); + deleteView.setPadding(dp(16), 0, dp(16), 0); + deleteView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 14); + deleteView.setTag(0); + deleteView.setText(LocaleController.getString("PaintDelete", R.string.PaintDelete)); + deleteView.setOnClickListener(v -> { + if (entityView instanceof RoundView) { + onTryDeleteRound(); + } else { + removeEntity(entityView); + } - if (popupWindow != null && popupWindow.isShowing()) { - popupWindow.dismiss(true); - } - }); - parent.addView(deleteView, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, 44)); + if (popupWindow != null && popupWindow.isShowing()) { + popupWindow.dismiss(true); + } + }); + parent.addView(deleteView, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, 44)); + } if (entityView instanceof TextPaintView) { TextView editView = new TextView(getContext()); @@ -3494,7 +3589,23 @@ private void showMenuForEntity(final EntityView entityView) { parent.addView(flipView, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, 44)); } - if (!(entityView instanceof PhotoView) && !(entityView instanceof RoundView) && !(entityView instanceof LocationView) && !(entityView instanceof ReactionWidgetEntityView)) { + if (entityView instanceof PhotoView && ((PhotoView) entityView).hasSegmentedImage()) { + PhotoView photoView = (PhotoView) entityView; + TextView cutView = createActionLayoutButton(5, LocaleController.getString(photoView.isSegmented() ? R.string.SegmentationUndoCutOut : R.string.SegmentationCutOut)); + cutView.setOnClickListener(v -> { + photoView.toggleSegmented(true); + if (photoView.isSegmented()) { + onSwitchSegmentedAnimation(photoView); + } + if (popupWindow != null && popupWindow.isShowing()) { + popupWindow.dismiss(true); + } + }); + parent.addView(cutView, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, 44)); + photoView.highlightSegmented(); + } + + if (!(entityView instanceof PhotoView) && !(entityView instanceof MessageEntityView) && !(entityView instanceof RoundView) && !(entityView instanceof LocationView) && !(entityView instanceof ReactionWidgetEntityView)) { TextView duplicateView = new TextView(getContext()); duplicateView.setTextColor(getThemedColor(Theme.key_actionBarDefaultSubmenuItem)); duplicateView.setLines(1); @@ -3598,7 +3709,7 @@ private Point startPositionRelativeToEntity(EntityView entityView) { boolean occupied = false; for (int index = 0; index < entitiesView.getChildCount(); index++) { View view = entitiesView.getChildAt(index); - if (!(view instanceof EntityView)) + if (!(view instanceof EntityView) || view instanceof MessageEntityView) continue; Point location = ((EntityView) view).getPosition(); @@ -3933,6 +4044,10 @@ public void onDeleteRound() { } + public void onSwitchSegmentedAnimation(PhotoView photoView) { + + } + public void onDeselectRound(RoundView roundView) { } @@ -3978,6 +4093,33 @@ public RoundView createRound(String thumbPath, boolean select) { return view; } + public MessageEntityView createMessage(ArrayList messageObjects, boolean select, boolean hasVideo) { + forceChanges = true; + MessageEntityView view = new MessageEntityView(getContext(), centerPositionForEntity(), 0, 1f, messageObjects, blurManager, hasVideo, videoTextureHolder) { + @Override + public boolean drawForBitmap() { + return drawForThemeToggle; + } + }; + view.setDelegate(this); + entitiesView.addView(view); + if (select) { + registerRemovalUndo(view); + selectEntity(view); + } + return view; + } + + public MessageEntityView findMessageView() { + for (int i = 0; i < entitiesView.getChildCount(); ++i) { + View child = entitiesView.getChildAt(i); + if (child instanceof MessageEntityView) { + return (MessageEntityView) child; + } + } + return null; + } + public PhotoView createPhoto(TLObject obj, boolean select) { forceChanges = true; Size size = basePhotoSize(obj); @@ -4076,6 +4218,9 @@ private void removeEntity(EntityView entityView) { if (entityView != null) { undoStore.unregisterUndo(entityView.getUUID()); } + if (entityView instanceof PhotoView) { + ((PhotoView) entityView).deleteSegmentedFile(); + } weightChooserView.setValueOverride(weightDefaultValueOverride); weightChooserView.setShowPreview(true); @@ -4101,6 +4246,14 @@ public boolean onEntitySelected(EntityView entityView) { return selectEntity(entityView); } + public boolean isEntityDeletable() { + return isEntityDeletable(currentEntityView); + } + + public boolean isEntityDeletable(EntityView entityView) { + return !(entityView instanceof MessageEntityView); + } + @Override public void onEntityDragEnd(boolean delete) { updatePreviewViewTranslationY(); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/PreviewButtons.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/PreviewButtons.java index 2227f1f4dcd..c24eb0d645a 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/PreviewButtons.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/PreviewButtons.java @@ -55,7 +55,7 @@ public class PreviewButtons extends FrameLayout { private View shadowView; - private ArrayList buttons = new ArrayList<>(); + private ArrayList buttons = new ArrayList<>(); public ShareButtonView shareButton; private String shareText; @@ -79,6 +79,25 @@ public PreviewButtons(Context context) { updateAppearT(); } + public void setFiltersVisible(boolean visible) { + for (int i = 0; i < buttons.size(); ++i) { + ButtonView button = buttons.get(i); + if (button.id == BUTTON_ADJUST) { + button.setVisibility(visible ? View.VISIBLE : View.GONE); + } + } + } + + private boolean isFiltersVisible() { + for (int i = 0; i < buttons.size(); ++i) { + ButtonView button = buttons.get(i); + if (button.id == BUTTON_ADJUST) { + return button.getVisibility() == View.VISIBLE; + } + } + return false; + } + public void setShareText(String text) { if (TextUtils.equals(text, shareText)) { return; @@ -106,11 +125,19 @@ protected void onLayout(boolean changed, int left, int top, int right, int botto shareButton.layout(w - shareButton.getMeasuredWidth(), (h - shareButton.getMeasuredHeight()) / 2, w, (h + shareButton.getMeasuredHeight()) / 2); int W = w - dp(10 + 10 + 12.33f) - shareButton.getMeasuredWidth(); - int maxPossibleMargin = buttons.size() < 2 ? 0 : (W - buttons.size() * dp(40)) / (buttons.size() - 1); - int margin = Math.min(dp(20), maxPossibleMargin); + int visibleButtons = 0; + for (int i = 0; i < buttons.size(); ++i) { + ButtonView button = buttons.get(i); + if (button.getVisibility() == View.VISIBLE) { + visibleButtons++; + } + } + int maxPossibleMargin = visibleButtons < 2 ? 0 : (W - visibleButtons * dp(40)) / (visibleButtons - 1); + int margin = Math.min(dp(isFiltersVisible() ? 20 : 30), maxPossibleMargin); int t = (h - dp(40)) / 2, b = (h + dp(40)) / 2; - for (int i = 0, x = dp(12.33f); i < buttons.size(); ++i) { + for (int i = 0, x = dp(12.33f) + (!isFiltersVisible() ? (W - visibleButtons * dp(40) - (visibleButtons - 1) * margin) / 2 : 0); i < buttons.size(); ++i) { + if (buttons.get(i).getVisibility() != View.VISIBLE) continue; buttons.get(i).layout(x, t, x + dp(40), b); x += dp(40) + margin; } @@ -319,8 +346,10 @@ public void onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo info) { } private class ButtonView extends ImageView { + public final int id; public ButtonView(Context context, int id, int resId) { super(context); + this.id = id; setBackground(Theme.createSelectorDrawable(Theme.ACTION_BAR_WHITE_SELECTOR_COLOR)); setScaleType(ScaleType.CENTER); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/PreviewView.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/PreviewView.java index 7000412b2f5..e90825ba8e2 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/PreviewView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/PreviewView.java @@ -1,16 +1,26 @@ package org.telegram.ui.Stories.recorder; +import static org.telegram.messenger.AndroidUtilities.dp; + +import android.animation.ValueAnimator; +import android.app.Activity; import android.content.ContentUris; import android.content.Context; +import android.content.SharedPreferences; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.graphics.Canvas; +import android.graphics.Color; import android.graphics.LinearGradient; import android.graphics.Matrix; import android.graphics.Paint; +import android.graphics.Path; import android.graphics.PointF; import android.graphics.Shader; import android.graphics.SurfaceTexture; +import android.graphics.drawable.BitmapDrawable; +import android.graphics.drawable.ColorDrawable; +import android.graphics.drawable.Drawable; import android.net.Uri; import android.os.Build; import android.provider.MediaStore; @@ -18,25 +28,39 @@ import android.util.Log; import android.util.Pair; import android.util.Size; +import android.util.SparseIntArray; import android.view.Gravity; import android.view.HapticFeedbackConstants; import android.view.MotionEvent; import android.view.TextureView; import android.view.View; import android.view.ViewConfiguration; +import android.view.ViewGroup; +import android.view.ViewParent; import android.widget.FrameLayout; +import androidx.annotation.NonNull; + import com.google.android.exoplayer2.C; import com.google.zxing.common.detector.MathUtils; import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.ApplicationLoader; +import org.telegram.messenger.ChatThemeController; import org.telegram.messenger.MessageObject; +import org.telegram.messenger.MessagesController; +import org.telegram.messenger.Utilities; import org.telegram.tgnet.TLRPC; +import org.telegram.ui.ActionBar.EmojiThemes; +import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.ChatBackgroundDrawable; import org.telegram.ui.Components.AnimatedFloat; import org.telegram.ui.Components.BlurringShader; import org.telegram.ui.Components.ButtonBounce; import org.telegram.ui.Components.CubicBezierInterpolator; import org.telegram.ui.Components.LayoutHelper; +import org.telegram.ui.Components.MotionBackgroundDrawable; +import org.telegram.ui.Components.Paint.Texture; import org.telegram.ui.Components.Paint.Views.RoundView; import org.telegram.ui.Components.PhotoFilterView; import org.telegram.ui.Components.VideoEditTextureView; @@ -75,20 +99,19 @@ public class PreviewView extends FrameLayout { public static final long MAX_DURATION = 59_500L; public static final long MIN_DURATION = 1_000L; - private final HashMap partsBitmap = new HashMap<>(); - private final HashMap partsBounce = new HashMap<>(); - private final BlurringShader.BlurManager blurManager; + private final TextureViewHolder textureViewHolder; - public PreviewView(Context context, BlurringShader.BlurManager blurManager) { + public PreviewView(Context context, BlurringShader.BlurManager blurManager, TextureViewHolder textureViewHolder) { super(context); this.blurManager = blurManager; + this.textureViewHolder = textureViewHolder; - snapPaint.setStrokeWidth(AndroidUtilities.dp(1)); + snapPaint.setStrokeWidth(dp(1)); snapPaint.setStyle(Paint.Style.STROKE); snapPaint.setColor(0xffffffff); - snapPaint.setShadowLayer(AndroidUtilities.dp(3), 0, AndroidUtilities.dp(1), 0x40000000); + snapPaint.setShadowLayer(dp(3), 0, dp(1), 0x40000000); } protected void onTimeDrag(boolean dragStart, long time, boolean dragEnd) {} @@ -112,7 +135,7 @@ public void set(StoryEntry entry, Runnable whenReady, long seekTo) { if (entry == null) { setupVideoPlayer(null, whenReady, seekTo); setupImage(null); - setupParts(null); + setupWallpaper(null, false); gradientPaint.setShader(null); setupAudio((StoryEntry) null, false); setupRound(null, null, false); @@ -131,8 +154,36 @@ public void set(StoryEntry entry, Runnable whenReady, long seekTo) { setupImage(entry); setupGradient(); } - setupParts(entry); applyMatrix(); + setupWallpaper(entry, false); + setupAudio(entry, false); + setupRound(entry, null, false); + } + + // set without video for faster transition + public void preset(StoryEntry entry) { + this.entry = entry; + if (entry == null) { + setupImage(null); + setupWallpaper(null, false); + gradientPaint.setShader(null); + setupAudio((StoryEntry) null, false); + setupRound(null, null, false); + return; + } + if (entry.isVideo) { + setupImage(entry); + if (entry.gradientTopColor != 0 || entry.gradientBottomColor != 0) { + setupGradient(); + } else { + entry.setupGradient(this::setupGradient); + } + } else { + setupImage(entry); + setupGradient(); + } + applyMatrix(); + setupWallpaper(entry, false); setupAudio(entry, false); setupRound(entry, null, false); } @@ -178,7 +229,7 @@ public void onRenderedFirstFrame() { @Override public void onSurfaceTextureUpdated(SurfaceTexture surfaceTexture) { - + invalidateTextureViewHolder(); } @Override @@ -548,14 +599,16 @@ private void setupGradient() { invalidate(); } - private void setupVideoPlayer(StoryEntry entry, Runnable whenReady, long seekTo) { + public void setupVideoPlayer(StoryEntry entry, Runnable whenReady, long seekTo) { if (entry == null) { if (videoPlayer != null) { videoPlayer.pause(); videoPlayer.releasePlayer(true); videoPlayer = null; } - if (textureView != null) { + if (textureViewHolder != null && textureViewHolder.active) { + textureViewHolder.setTextureView(null); + } else if (textureView != null) { textureView.clearAnimation(); textureView.animate().alpha(0).withEndAction(() -> { if (textureView != null) { @@ -566,7 +619,7 @@ private void setupVideoPlayer(StoryEntry entry, Runnable whenReady, long seekTo) }).start(); } if (timelineView != null) { - timelineView.setVideo(null, 1, 0); + timelineView.setVideo(false, null, 1, 0); } AndroidUtilities.cancelRunOnUIThread(updateProgressRunnable); if (whenReady != null) { @@ -626,6 +679,9 @@ public void onVideoSizeChanged(int width, int height, int unappliedRotationDegre @Override public void onRenderedFirstFrame() { + if (textureViewHolder != null && textureViewHolder.active) { + textureViewHolder.activateTextureView(videoWidth, videoHeight); + } if (whenReadyFinal[0] != null) { post(whenReadyFinal[0]); whenReadyFinal[0] = null; @@ -637,7 +693,7 @@ public void onRenderedFirstFrame() { bitmap = null; invalidate(); } - } else if (textureView != null) { + } else if (textureView != null && !(textureViewHolder != null && textureViewHolder.active)) { textureView.animate().alpha(1f).setDuration(180).withEndAction(() -> { if (bitmap != null) { bitmap.recycle(); @@ -652,7 +708,9 @@ public void onRenderedFirstFrame() { } @Override - public void onSurfaceTextureUpdated(SurfaceTexture surfaceTexture) {} + public void onSurfaceTextureUpdated(SurfaceTexture surfaceTexture) { + invalidateTextureViewHolder(); + } @Override public boolean onSurfaceDestroyed(SurfaceTexture surfaceTexture) { @@ -669,11 +727,15 @@ public boolean onSurfaceDestroyed(SurfaceTexture surfaceTexture) { textureView = new VideoEditTextureView(getContext(), videoPlayer); blurManager.resetBitmap(); - textureView.updateUiBlurManager(blurManager); - textureView.setAlpha(whenReady == null ? 0f : 1f); + textureView.updateUiBlurManager(entry.isRepostMessage ? null : blurManager); textureView.setOpaque(false); applyMatrix(); - addView(textureView, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.TOP | Gravity.LEFT)); + if (textureViewHolder != null && textureViewHolder.active) { + textureViewHolder.setTextureView(textureView); + } else { + textureView.setAlpha(whenReady == null ? 0f : 1f); + addView(textureView, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.TOP | Gravity.LEFT)); + } entry.detectHDR((hdrInfo) -> { if (textureView != null) { @@ -694,7 +756,8 @@ public boolean onSurfaceDestroyed(SurfaceTexture surfaceTexture) { checkVolumes(); updateAudioPlayer(true); - timelineView.setVideo(entry.getOriginalFile().getAbsolutePath(), getDuration(), entry.videoVolume); + final boolean isRound = entry.isRepostMessage && entry.messageObjects != null && entry.messageObjects.size() == 1 && entry.messageObjects.get(0).type == MessageObject.TYPE_ROUND_VIDEO; + timelineView.setVideo(isRound, entry.getOriginalFile().getAbsolutePath(), getDuration(), entry.videoVolume); timelineView.setVideoLeft(entry.left); timelineView.setVideoRight(entry.right); if (timelineView != null && seekTo > 0) { @@ -703,6 +766,60 @@ public boolean onSurfaceDestroyed(SurfaceTexture surfaceTexture) { } } + public static class TextureViewHolder { + private TextureView textureView; + private Utilities.Callback whenTextureViewReceived; + private Utilities.Callback2 whenTextureViewActive; + public boolean textureViewActive; + public int videoWidth, videoHeight; + + public boolean active; + + public void setTextureView(TextureView textureView) { + if (this.textureView == textureView) return; + if (this.textureView != null) { + ViewParent parent = this.textureView.getParent(); + if (parent instanceof ViewGroup) { + ((ViewGroup) parent).removeView(this.textureView); + } + this.textureView = null; + } + textureViewActive = false; + this.textureView = textureView; + if (whenTextureViewReceived != null) { + whenTextureViewReceived.run(this.textureView); + } + } + + public void activateTextureView(int w, int h) { + textureViewActive = true; + videoWidth = w; + videoHeight = h; + if (whenTextureViewActive != null) { + whenTextureViewActive.run(videoWidth, videoHeight); + } + } + + public void takeTextureView(Utilities.Callback whenReceived, Utilities.Callback2 whenActive) { + whenTextureViewReceived = whenReceived; + whenTextureViewActive = whenActive; + if (textureView != null && whenTextureViewReceived != null) { + whenTextureViewReceived.run(textureView); + } + if (textureViewActive && whenTextureViewActive != null) { + whenTextureViewActive.run(videoWidth, videoHeight); + } + } + } + + @Override + protected boolean drawChild(Canvas canvas, View child, long drawingTime) { + if (child == textureView && entry != null && entry.isRepostMessage) { + return false; + } + return super.drawChild(canvas, child, drawingTime); + } + public void setupRound(StoryEntry entry, RoundView roundView, boolean animated) { if (entry == null || entry.round == null) { if (roundPlayer != null) { @@ -802,53 +919,6 @@ public long release() { return t; } - public void setupParts(StoryEntry entry) { - if (entry == null) { - for (Bitmap bitmap : partsBitmap.values()) { - if (bitmap != null) { - bitmap.recycle(); - } - } - partsBitmap.clear(); - partsBounce.clear(); - return; - } - final int rw = getMeasuredWidth() <= 0 ? AndroidUtilities.displaySize.x : getMeasuredWidth(); - final int rh = (int) (rw * 16 / 9f); - for (int i = 0; i < entry.parts.size(); ++i) { - StoryEntry.Part part = entry.parts.get(i); - if (part == null) { - continue; - } - Bitmap bitmap = partsBitmap.get(part.id); - if (bitmap != null) { - continue; - } - String path = part.file.getPath(); - BitmapFactory.Options options = new BitmapFactory.Options(); - options.inJustDecodeBounds = true; - BitmapFactory.decodeFile(path, options); - options.inJustDecodeBounds = false; - options.inSampleSize = StoryEntry.calculateInSampleSize(options, rw, rh); - bitmap = BitmapFactory.decodeFile(path, options); - partsBitmap.put(part.id, bitmap); - } - for (Iterator> i = partsBitmap.entrySet().iterator(); i.hasNext(); ) { - Map.Entry e = i.next(); - boolean found = false; - for (int j = 0; j < entry.parts.size(); ++j) { - if (entry.parts.get(j).id == e.getKey()) { - found = true; - break; - } - } - if (!found) { - i.remove(); - partsBounce.remove(e.getKey()); - } - } - } - public void setFilterTextureView(TextureView view, PhotoFilterView photoFilterView) { if (filterTextureView != null) { removeView(filterTextureView); @@ -1026,7 +1096,10 @@ public void checkVolumes() { } } + private AnimatedFloat wallpaperDrawableCrossfade = new AnimatedFloat(this, 0, 350, CubicBezierInterpolator.EASE_OUT_QUINT); private final Paint bitmapPaint = new Paint(Paint.ANTI_ALIAS_FLAG | Paint.FILTER_BITMAP_FLAG | Paint.DITHER_FLAG); + private Drawable lastWallpaperDrawable; + private Drawable wallpaperDrawable; private final Paint gradientPaint = new Paint(Paint.ANTI_ALIAS_FLAG); private int gradientTop, gradientBottom; @@ -1065,10 +1138,35 @@ public void setDraw(boolean draw) { } private final AnimatedFloat thumbAlpha = new AnimatedFloat(this, 0, 320, CubicBezierInterpolator.EASE_OUT); + public boolean drawForThemeToggle = false; @Override protected void dispatchDraw(Canvas canvas) { - canvas.drawRect(0, 0, getWidth(), getHeight(), gradientPaint); + if (wallpaperDrawable != null) { + if (drawForThemeToggle) { + Path path = new Path(); + AndroidUtilities.rectTmp.set(0, 0, getWidth(), getHeight()); + path.addRoundRect(AndroidUtilities.rectTmp, dp(12), dp(12), Path.Direction.CW); + canvas.save(); + canvas.clipPath(path); + } + boolean drawableReady = true; + if (wallpaperDrawable instanceof MotionBackgroundDrawable) { + drawableReady = ((MotionBackgroundDrawable) wallpaperDrawable).getPatternBitmap() != null; + } + float crossfadeAlpha = !drawableReady ? 0f : wallpaperDrawableCrossfade.set(1f); + if (lastWallpaperDrawable != null && crossfadeAlpha < 1) { + lastWallpaperDrawable.setAlpha((int) (0xFF * (1f - crossfadeAlpha))); + StoryEntry.drawBackgroundDrawable(canvas, lastWallpaperDrawable, getWidth(), getHeight()); + } + wallpaperDrawable.setAlpha((int) (0xFF * crossfadeAlpha)); + StoryEntry.drawBackgroundDrawable(canvas, wallpaperDrawable, getWidth(), getHeight()); + if (drawForThemeToggle) { + canvas.restore(); + } + } else { + canvas.drawRect(0, 0, getWidth(), getHeight(), gradientPaint); + } if (draw && entry != null) { float alpha = this.thumbAlpha.set(bitmap != null); if (thumbBitmap != null && (1f - alpha) > 0) { @@ -1087,46 +1185,16 @@ protected void dispatchDraw(Canvas canvas) { } } super.dispatchDraw(canvas); - if (draw && entry != null) { - float trash = trashT.set(!inTrash); - for (int i = 0; i < entry.parts.size(); ++i) { - StoryEntry.Part part = entry.parts.get(i); - if (part == null) { - continue; - } - Bitmap bitmap = partsBitmap.get(part.id); - if (bitmap == null) { - continue; - } - float scale = 1f; - ButtonBounce bounce = partsBounce.get(part.id); - if (bounce != null) { - scale = bounce.getScale(.05f); - } - matrix.set(part.matrix); - canvas.save(); - if (scale != 1) { - tempVertices[0] = part.width / 2f; - tempVertices[1] = part.height / 2f; - matrix.mapPoints(tempVertices); - canvas.scale(scale, scale, tempVertices[0] / entry.resultWidth * getWidth(), tempVertices[1] / entry.resultHeight * getHeight()); - } - if (trashPartIndex == part.id) { - float trashScale = AndroidUtilities.lerp(.2f, 1f, trash); - canvas.scale(trashScale, trashScale, trashCx, trashCy); - } - matrix.preScale((float) part.width / bitmap.getWidth(), (float) part.height / bitmap.getHeight()); - matrix.postScale((float) getWidth() / entry.resultWidth, (float) getHeight() / entry.resultHeight); - canvas.drawBitmap(bitmap, matrix, bitmapPaint); - canvas.restore(); - } - } } public VideoEditTextureView getTextureView() { return textureView; } + protected void invalidateTextureViewHolder() { + + } + public Pair getPaintSize() { if (entry == null) { return new Pair<>(1080, 1920); @@ -1151,7 +1219,7 @@ public void setVisibility(int visibility) { } public void applyMatrix() { - if (entry == null) { + if (entry == null || entry.isRepostMessage) { return; } if (textureView != null) { @@ -1185,16 +1253,7 @@ public void setAllowCropping(boolean allow) { private float rotationDiff; private boolean snappedToCenterAndScaled, snappedRotation; private boolean doNotSpanRotation; - private float[] tempPoint = new float[4]; - private IStoryPart activePart; - private boolean isPart; - private float Tx, Ty; - private boolean activePartPressed; - - private int trashPartIndex; - private boolean inTrash; - private AnimatedFloat trashT = new AnimatedFloat(this, 0, 280, CubicBezierInterpolator.EASE_OUT_QUINT); - private float trashCx, trashCy; + private boolean moving; public boolean additionalTouchEvent(MotionEvent ev) { return false; @@ -1229,33 +1288,15 @@ private boolean touchEvent(MotionEvent ev) { final float scale = (entry.resultWidth / (float) getWidth()); if (ev.getActionMasked() == MotionEvent.ACTION_DOWN) { - Tx = Ty = 0; rotationDiff = 0; snappedRotation = false; snappedToCenterAndScaled = false; doNotSpanRotation = false; - activePart = findPartAt(touch.x, touch.y); - if (isPart = (activePart instanceof StoryEntry.Part)) { - entry.parts.remove(activePart); - entry.parts.add((StoryEntry.Part) activePart); - trashPartIndex = activePart.id; - invalidate(); - allowWithSingleTouch = true; - ButtonBounce bounce = partsBounce.get(activePart.id); - if (bounce == null) { - partsBounce.put(activePart.id, bounce = new ButtonBounce(this)); - } - bounce.setPressed(true); - activePartPressed = true; - onEntityDragStart(); - onEntityDraggedTop(false); - onEntityDraggedBottom(false); - } else { - trashPartIndex = -1; - } - touchMatrix.set(activePart.matrix); + invalidate(); + moving = true; + touchMatrix.set(entry.matrix); } - if (ev.getActionMasked() == MotionEvent.ACTION_MOVE && activePart != null) { + if (ev.getActionMasked() == MotionEvent.ACTION_MOVE && moving && entry != null) { float tx = touch.x * scale, ty = touch.y * scale; float ltx = lastTouch.x * scale, lty = lastTouch.y * scale; if (ev.getPointerCount() > 1) { @@ -1285,15 +1326,6 @@ private boolean touchEvent(MotionEvent ev) { } if (ev.getPointerCount() > 1 || allowWithSingleTouch) { touchMatrix.postTranslate(tx - ltx, ty - lty); - Tx += (tx - ltx); - Ty += (ty - lty); - } - if (activePartPressed && MathUtils.distance(0, 0, Tx, Ty) > AndroidUtilities.touchSlop) { - ButtonBounce bounce = partsBounce.get(activePart.id); - if (bounce != null) { - bounce.setPressed(false); - } - activePartPressed = false; } finalMatrix.set(touchMatrix); matrix.set(touchMatrix); @@ -1312,41 +1344,19 @@ private boolean touchEvent(MotionEvent ev) { snappedRotation = false; } } - boolean trash = isPart && MathUtils.distance(touch.x, touch.y, getWidth() / 2f, getHeight() - AndroidUtilities.dp(76)) < AndroidUtilities.dp(35); - if (trash != inTrash) { - onEntityDragTrash(trash); - inTrash = trash; - } - if (trash) { - trashCx = touch.x; - trashCy = touch.y; - } - if (isPart) { - onEntityDraggedTop(cy - h / 2f < AndroidUtilities.dp(66) / (float) getHeight() * entry.resultHeight); - onEntityDraggedBottom(cy + h / 2f > entry.resultHeight - AndroidUtilities.dp(64 + 50) / (float) getHeight() * entry.resultHeight); - } - activePart.matrix.set(finalMatrix); + entry.matrix.set(finalMatrix); entry.editedMedia = true; applyMatrix(); invalidate(); } if (ev.getAction() == MotionEvent.ACTION_UP || ev.getAction() == MotionEvent.ACTION_CANCEL) { - Tx = Ty = 0; - if (!(activePart instanceof StoryEntry.Part) || ev.getPointerCount() <= 1) { - ButtonBounce bounce = partsBounce.get(activePart.id); - if (bounce != null) { - bounce.setPressed(false); - } - activePartPressed = false; + if (ev.getPointerCount() <= 1) { allowWithSingleTouch = false; - onEntityDragEnd(inTrash && ev.getAction() == MotionEvent.ACTION_UP); - activePart = null; - inTrash = false; onEntityDraggedTop(false); onEntityDraggedBottom(false); } - isPart = false; + moving = false; allowRotation = false; rotationDiff = 0; @@ -1361,39 +1371,10 @@ private boolean touchEvent(MotionEvent ev) { return true; } - public void deleteCurrentPart() { - if (activePart != null) { - entry.parts.remove(activePart); - setupParts(entry); - } - } - - private Matrix tempMatrix; - private float[] tempVertices = new float[2]; - private IStoryPart findPartAt(float x, float y) { - for (int i = entry.parts.size() - 1; i >= 0; --i) { - IStoryPart part = entry.parts.get(i); - tempVertices[0] = x / getWidth() * entry.resultWidth; - tempVertices[1] = y / getHeight() * entry.resultHeight; - if (tempMatrix == null) { - tempMatrix = new Matrix(); - } - part.matrix.invert(tempMatrix); - tempMatrix.mapPoints(tempVertices); - if (tempVertices[0] >= 0 && tempVertices[0] <= part.width && tempVertices[1] >= 0 && tempVertices[1] <= part.height) { - return part; - } - } - return entry; - } - private long tapTime; - private float tapX, tapY; private boolean tapTouchEvent(MotionEvent ev) { if (ev.getAction() == MotionEvent.ACTION_DOWN) { tapTime = System.currentTimeMillis(); - tapX = ev.getX(); - tapY = ev.getY(); return true; } else if (ev.getAction() == MotionEvent.ACTION_UP) { if (System.currentTimeMillis() - tapTime <= ViewConfiguration.getTapTimeout() && onTap != null) { @@ -1424,10 +1405,8 @@ public boolean onInterceptTouchEvent(MotionEvent ev) { @Override public boolean dispatchTouchEvent(MotionEvent ev) { boolean result = touchEvent(ev); - if (!(activePart instanceof StoryEntry.Part)) { - result = additionalTouchEvent(ev) || result; - tapTouchEvent(ev); - } + result = additionalTouchEvent(ev) || result; + tapTouchEvent(ev); if (result) { if (ev.getPointerCount() <= 1) { return super.dispatchTouchEvent(ev) || true; @@ -1479,4 +1458,202 @@ public boolean isPlaying() { public void play(boolean play) { updatePauseReason(-9982, !play); } + + public static Drawable getBackgroundDrawable(Drawable prevDrawable, int currentAccount, long dialogId, boolean isDark) { + if (dialogId == Long.MIN_VALUE) { + return null; + } + TLRPC.WallPaper wallpaper = null; + if (dialogId >= 0) { + TLRPC.UserFull userFull = MessagesController.getInstance(currentAccount).getUserFull(dialogId); + if (userFull != null) { + wallpaper = userFull.wallpaper; + } + } else { + TLRPC.ChatFull chatFull = MessagesController.getInstance(currentAccount).getChatFull(-dialogId); + if (chatFull != null) { + wallpaper = chatFull.wallpaper; + } + } + return getBackgroundDrawable(prevDrawable, currentAccount, wallpaper, isDark); + } + + public static Drawable getBackgroundDrawable(Drawable prevDrawable, int currentAccount, TLRPC.WallPaper wallpaper, boolean isDark) { + if (wallpaper != null && TextUtils.isEmpty(ChatThemeController.getWallpaperEmoticon(wallpaper))) { + return ChatBackgroundDrawable.getOrCreate(prevDrawable, wallpaper, isDark); + } + + EmojiThemes theme = null; + if (wallpaper != null && wallpaper.settings != null) { + theme = ChatThemeController.getInstance(currentAccount).getTheme(wallpaper.settings.emoticon); + } + if (theme != null) { + return getBackgroundDrawableFromTheme(currentAccount, theme, 0, isDark); + } + + SharedPreferences preferences = ApplicationLoader.applicationContext.getSharedPreferences("themeconfig", Activity.MODE_PRIVATE); + String dayThemeName = preferences.getString("lastDayTheme", "Blue"); + if (Theme.getTheme(dayThemeName) == null || Theme.getTheme(dayThemeName).isDark()) { + dayThemeName = "Blue"; + } + String nightThemeName = preferences.getString("lastDarkTheme", "Dark Blue"); + if (Theme.getTheme(nightThemeName) == null || !Theme.getTheme(nightThemeName).isDark()) { + nightThemeName = "Dark Blue"; + } + Theme.ThemeInfo themeInfo = Theme.getActiveTheme(); + if (dayThemeName.equals(nightThemeName)) { + if (themeInfo.isDark() || dayThemeName.equals("Dark Blue") || dayThemeName.equals("Night")) { + dayThemeName = "Blue"; + } else { + nightThemeName = "Dark Blue"; + } + } + if (isDark) { + themeInfo = Theme.getTheme(nightThemeName); + } else { + themeInfo = Theme.getTheme(dayThemeName); + } + SparseIntArray currentColors = new SparseIntArray(); + final String[] wallpaperLink = new String[1]; + final SparseIntArray themeColors; + if (themeInfo.assetName != null) { + themeColors = Theme.getThemeFileValues(null, themeInfo.assetName, wallpaperLink); + } else { + themeColors = Theme.getThemeFileValues(new File(themeInfo.pathToFile), null, wallpaperLink); + } + int[] defaultColors = Theme.getDefaultColors(); + if (defaultColors != null) { + for (int i = 0; i < defaultColors.length; ++i) { + currentColors.put(i, defaultColors[i]); + } + } + Theme.ThemeAccent accent = themeInfo.getAccent(false); + if (accent != null) { + accent.fillAccentColors(themeColors, currentColors); + } else if (themeColors != null) { + for (int i = 0; i < themeColors.size(); ++i) { + currentColors.put(themeColors.keyAt(i), themeColors.valueAt(i)); + } + } + Theme.BackgroundDrawableSettings bg = Theme.createBackgroundDrawable(themeInfo, currentColors, wallpaperLink[0], 0, true); + return bg.themedWallpaper != null ? bg.themedWallpaper : bg.wallpaper; + } + + public void setupWallpaper(StoryEntry entry, boolean animated) { + lastWallpaperDrawable = wallpaperDrawable; + if (wallpaperDrawable != null) { + wallpaperDrawable.setCallback(null); + } + if (entry == null) { + wallpaperDrawable = null; + return; + } + long dialogId = entry.backgroundWallpaperPeerId; + if (entry.backgroundWallpaperEmoticon != null) { + wallpaperDrawable = entry.backgroundDrawable = getBackgroundDrawableFromTheme(entry.currentAccount, entry.backgroundWallpaperEmoticon, entry.isDark); + } else if (dialogId != Long.MIN_VALUE) { + wallpaperDrawable = entry.backgroundDrawable = getBackgroundDrawable(wallpaperDrawable, entry.currentAccount, dialogId, entry.isDark); + } else { + wallpaperDrawable = null; + return; + } + if (lastWallpaperDrawable != wallpaperDrawable) { + if (animated) { + wallpaperDrawableCrossfade.set(0, true); + } else { + lastWallpaperDrawable = null; + } + } + if (wallpaperDrawable != null) { + wallpaperDrawable.setCallback(this); + } + if (blurManager != null) { + if (wallpaperDrawable != null) { + if (wallpaperDrawable instanceof BitmapDrawable) { + blurManager.setFallbackBlur(((BitmapDrawable) wallpaperDrawable).getBitmap(), 0); + } else { + int w = wallpaperDrawable.getIntrinsicWidth(); + int h = wallpaperDrawable.getIntrinsicHeight(); + if (w <= 0 || h <= 0) { + w = 1080; + h = 1920; + } + float scale = Math.max(100f / w, 100f / h); + if (scale > 1) { + w *= scale; + h *= scale; + } + Bitmap bitmap = Bitmap.createBitmap(w, h, Bitmap.Config.ARGB_8888); + wallpaperDrawable.setBounds(0, 0, w, h); + wallpaperDrawable.draw(new Canvas(bitmap)); + blurManager.setFallbackBlur(bitmap, 0, true); + } + } else { + blurManager.setFallbackBlur(null, 0); + } + } + invalidate(); + } + + public static Drawable getBackgroundDrawableFromTheme(int currentAccount, String emoticon, boolean isDark) { + return getBackgroundDrawableFromTheme(currentAccount, emoticon, isDark, false); + } + + public static Drawable getBackgroundDrawableFromTheme(int currentAccount, String emoticon, boolean isDark, boolean preview) { + EmojiThemes theme = ChatThemeController.getInstance(currentAccount).getTheme(emoticon); + if (theme == null) { + return Theme.getCachedWallpaper(); + } + return getBackgroundDrawableFromTheme(currentAccount, theme, 0, isDark, preview); + } + + public static Drawable getBackgroundDrawableFromTheme(int currentAccount, EmojiThemes chatTheme, int prevPhase, boolean isDark) { + return getBackgroundDrawableFromTheme(currentAccount, chatTheme, prevPhase, isDark, false); + } + + public static Drawable getBackgroundDrawableFromTheme(int currentAccount, EmojiThemes chatTheme, int prevPhase, boolean isDark, boolean preview) { + Drawable drawable; + if (chatTheme.showAsDefaultStub) { + Theme.ThemeInfo themeInfo = EmojiThemes.getDefaultThemeInfo(isDark); + SparseIntArray currentColors = chatTheme.getPreviewColors(currentAccount, isDark ? 1 : 0); + String wallpaperLink = chatTheme.getWallpaperLink(isDark ? 1 : 0); + Theme.BackgroundDrawableSettings settings = Theme.createBackgroundDrawable(themeInfo, currentColors, wallpaperLink, prevPhase, false); + drawable = settings.wallpaper; + drawable = new ColorDrawable(Color.BLACK); + } else { + SparseIntArray currentColors = chatTheme.getPreviewColors(currentAccount, isDark ? 1 : 0); + int backgroundColor = currentColors.get(Theme.key_chat_wallpaper, Theme.getColor(Theme.key_chat_wallpaper)); + int gradientColor1 = currentColors.get(Theme.key_chat_wallpaper_gradient_to1, Theme.getColor(Theme.key_chat_wallpaper_gradient_to1)); + int gradientColor2 = currentColors.get(Theme.key_chat_wallpaper_gradient_to2, Theme.getColor(Theme.key_chat_wallpaper_gradient_to2)); + int gradientColor3 = currentColors.get(Theme.key_chat_wallpaper_gradient_to3, Theme.getColor(Theme.key_chat_wallpaper_gradient_to3)); + + MotionBackgroundDrawable motionDrawable = new MotionBackgroundDrawable(); + motionDrawable.isPreview = preview; + motionDrawable.setPatternBitmap(chatTheme.getWallpaper(isDark ? 1 : 0).settings.intensity); + motionDrawable.setColors(backgroundColor, gradientColor1, gradientColor2, gradientColor3, 0,true); + motionDrawable.setPhase(prevPhase); + int patternColor = motionDrawable.getPatternColor(); + final boolean isDarkTheme = isDark; + chatTheme.loadWallpaper(isDark ? 1 : 0, pair -> { + if (pair == null) { + return; + } + long themeId = pair.first; + Bitmap bitmap = pair.second; + if (themeId == chatTheme.getTlTheme(isDark ? 1 : 0).id && bitmap != null) { + int intensity = chatTheme.getWallpaper(isDarkTheme ? 1 : 0).settings.intensity; + motionDrawable.setPatternBitmap(intensity, bitmap); + motionDrawable.setPatternColorFilter(patternColor); + motionDrawable.setPatternAlpha(1f); + } + }); + drawable = motionDrawable; + } + return drawable; + } + + @Override + protected boolean verifyDrawable(@NonNull Drawable who) { + return wallpaperDrawable == who || super.verifyDrawable(who); + } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/StoryEntry.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/StoryEntry.java index a0687561442..12fdbfb33b2 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/StoryEntry.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/StoryEntry.java @@ -8,16 +8,22 @@ import android.graphics.Matrix; import android.graphics.Paint; import android.graphics.Shader; +import android.graphics.drawable.BitmapDrawable; +import android.graphics.drawable.Drawable; import android.media.MediaExtractor; import android.media.MediaFormat; import android.os.Build; import android.provider.MediaStore; import android.text.SpannableString; +import android.text.TextUtils; import androidx.annotation.NonNull; import org.telegram.messenger.AndroidUtilities; import org.telegram.messenger.ApplicationLoader; +import org.telegram.messenger.BuildVars; +import org.telegram.messenger.ChatObject; +import org.telegram.messenger.DialogObject; import org.telegram.messenger.Emoji; import org.telegram.messenger.FileLoader; import org.telegram.messenger.FileLog; @@ -47,7 +53,7 @@ import java.util.ArrayList; import java.util.List; -public class StoryEntry extends IStoryPart { +public class StoryEntry { public final int currentAccount = UserConfig.selectedAccount; @@ -70,6 +76,9 @@ public class StoryEntry extends IStoryPart { public String repostCaption; public TLRPC.MessageMedia repostMedia; + public boolean isRepostMessage; + public ArrayList messageObjects; + public boolean isError; public TLRPC.TL_error error; @@ -90,21 +99,20 @@ public class StoryEntry extends IStoryPart { public String thumbPath; public Bitmap thumbPathBitmap; public float videoVolume = 1f; + public int orientation, invert; public boolean muted; public float left, right = 1; - public int orientation; - public int invert; - // public int width, height; public long duration; public int resultWidth = 720; public int resultHeight = 1280; - public int partsMaxId = 1; - public final ArrayList parts = new ArrayList<>(); + public int width, height; + // matrix describes transformations from width x height to resultWidth x resultHeight + public final Matrix matrix = new Matrix(); public File round; public String roundThumb; @@ -115,42 +123,10 @@ public class StoryEntry extends IStoryPart { public TLRPC.InputPeer peer; - public static class Part extends IStoryPart { - public File file; - public boolean fileDeletable; - public int orientantion, invert; - - public void readParams(AbstractSerializedData stream, boolean exception) { - width = stream.readInt32(exception); - height = stream.readInt32(exception); - file = new File(stream.readString(exception)); - fileDeletable = stream.readBool(exception); - orientantion = stream.readInt32(exception); - invert = stream.readInt32(exception); - float[] values = new float[9]; - for (int i = 0; i < 9; ++i) { - values[i] = stream.readFloat(exception); - } - matrix.setValues(values); - } - - public void serializeToStream(AbstractSerializedData stream) { - stream.writeInt32(width); - stream.writeInt32(height); - stream.writeString(file == null ? "" : file.getAbsolutePath()); - stream.writeBool(fileDeletable); - stream.writeInt32(orientantion); - stream.writeInt32(invert); - float[] values = new float[9]; - matrix.getValues(values); - for (int i = 0; i < 9; ++i) { - stream.writeFloat(values[i]); - } - } - } - - // matrix describes transformations from width x height to resultWidth x resultHeight -// public final Matrix matrix = new Matrix(); + public Drawable backgroundDrawable; + public boolean isDark = Theme.isCurrentThemeDark(); + public long backgroundWallpaperPeerId = Long.MIN_VALUE; // Long.MIN_VALUE = no wallpaper + public String backgroundWallpaperEmoticon; public int gradientTopColor, gradientBottomColor; public CharSequence caption; @@ -180,6 +156,9 @@ public void serializeToStream(AbstractSerializedData stream) { public ArrayList mediaEntities; public List stickers; public List editStickers; + public File messageFile; + public File messageVideoMaskFile; + public File backgroundFile; // filter public File filterFile; @@ -189,6 +168,10 @@ public void serializeToStream(AbstractSerializedData stream) { private boolean fromCamera; public boolean wouldBeVideo() { + return wouldBeVideo(mediaEntities); + } + + public boolean wouldBeVideo(ArrayList mediaEntities) { if (isVideo) { return true; } @@ -225,6 +208,23 @@ private boolean isAnimated(TLRPC.Document document, String path) { ); } + public static void drawBackgroundDrawable(Canvas canvas, Drawable drawable, int w, int h) { + if (drawable == null) { + return; + } + if (drawable instanceof BitmapDrawable) { + BitmapDrawable bd = (BitmapDrawable) drawable; + int bw = bd.getBitmap().getWidth(); + int bh = bd.getBitmap().getHeight(); + final float scale = Math.max(w / (float) bw, h / (float) bh); + drawable.setBounds(0, 0, (int) (bw * scale), (int) (bh * scale)); + drawable.draw(canvas); + } else { + drawable.setBounds(0, 0, w, h); + drawable.draw(canvas); + } + } + public Bitmap buildBitmap(float scale, Bitmap mainFileBitmap) { Matrix tempMatrix = new Matrix(); @@ -233,9 +233,36 @@ public Bitmap buildBitmap(float scale, Bitmap mainFileBitmap) { Bitmap finalBitmap = Bitmap.createBitmap(w, h, Bitmap.Config.ARGB_8888); Canvas canvas = new Canvas(finalBitmap); - Paint gradientPaint = new Paint(Paint.ANTI_ALIAS_FLAG); - gradientPaint.setShader(new LinearGradient(0, 0, 0, canvas.getHeight(), new int[] { gradientTopColor, gradientBottomColor }, new float[] {0, 1}, Shader.TileMode.CLAMP)); - canvas.drawRect(0, 0, canvas.getWidth(), canvas.getHeight(), gradientPaint); + if (backgroundFile != null) { + try { + Bitmap paintBitmap = getScaledBitmap(opts -> BitmapFactory.decodeFile(backgroundFile.getPath(), opts), w, h, false); + canvas.save(); + float s = resultWidth / (float) paintBitmap.getWidth(); + canvas.scale(s, s); + tempMatrix.postScale(scale, scale); + canvas.drawBitmap(paintBitmap, 0, 0, bitmapPaint); + canvas.restore(); + paintBitmap.recycle(); + } catch (Exception e) { + FileLog.e(e); + } + } else if (backgroundWallpaperEmoticon != null) { + Drawable drawable = backgroundDrawable; + if (drawable == null) { + drawable = PreviewView.getBackgroundDrawableFromTheme(currentAccount, backgroundWallpaperEmoticon, isDark); + } + drawBackgroundDrawable(canvas, drawable, canvas.getWidth(), canvas.getHeight()); + } else if (backgroundWallpaperPeerId != Long.MIN_VALUE) { + Drawable drawable = backgroundDrawable; + if (drawable == null) { + drawable = PreviewView.getBackgroundDrawable(null, currentAccount, backgroundWallpaperPeerId, isDark); + } + drawBackgroundDrawable(canvas, drawable, canvas.getWidth(), canvas.getHeight()); + } else { + Paint gradientPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + gradientPaint.setShader(new LinearGradient(0, 0, 0, canvas.getHeight(), new int[]{gradientTopColor, gradientBottomColor}, new float[]{0, 1}, Shader.TileMode.CLAMP)); + canvas.drawRect(0, 0, canvas.getWidth(), canvas.getHeight(), gradientPaint); + } tempMatrix.set(matrix); if (mainFileBitmap != null) { @@ -258,24 +285,24 @@ public Bitmap buildBitmap(float scale, Bitmap mainFileBitmap) { } } - for (int i = 0; i < parts.size(); ++i) { + if (paintFile != null) { try { - final Part part = parts.get(i); - Bitmap fileBitmap = getScaledBitmap(opts -> BitmapFactory.decodeFile(part.file.getPath(), opts), w, h, false); - final float s = (float) part.width / fileBitmap.getWidth(); - tempMatrix.set(part.matrix); - tempMatrix.preScale(s, s); + Bitmap paintBitmap = getScaledBitmap(opts -> BitmapFactory.decodeFile(paintFile.getPath(), opts), w, h, false); + canvas.save(); + float s = resultWidth / (float) paintBitmap.getWidth(); + canvas.scale(s, s); tempMatrix.postScale(scale, scale); - canvas.drawBitmap(fileBitmap, tempMatrix, bitmapPaint); - fileBitmap.recycle(); + canvas.drawBitmap(paintBitmap, 0, 0, bitmapPaint); + canvas.restore(); + paintBitmap.recycle(); } catch (Exception e) { FileLog.e(e); } } - if (paintFile != null) { + if (messageFile != null) { try { - Bitmap paintBitmap = getScaledBitmap(opts -> BitmapFactory.decodeFile(paintFile.getPath(), opts), w, h, false); + Bitmap paintBitmap = getScaledBitmap(opts -> BitmapFactory.decodeFile(messageFile.getPath(), opts), w, h, false); canvas.save(); float s = resultWidth / (float) paintBitmap.getWidth(); canvas.scale(s, s); @@ -483,6 +510,18 @@ public void clearPaint() { paintFile.delete(); paintFile = null; } + if (backgroundFile != null) { + backgroundFile.delete(); + backgroundFile = null; + } + if (messageFile != null) { + messageFile.delete(); + messageFile = null; + } + if (messageVideoMaskFile != null) { + messageVideoMaskFile.delete(); + messageVideoMaskFile = null; + } if (paintEntitiesFile != null) { paintEntitiesFile.delete(); paintEntitiesFile = null; @@ -513,11 +552,17 @@ public void destroy(boolean draft) { } thumbPath = null; } - for (Part part : parts) { - if (part.fileDeletable) { - part.file.delete(); + if (mediaEntities != null) { + for (VideoEditedInfo.MediaEntity entity : mediaEntities) { + if (entity.type == VideoEditedInfo.MediaEntity.TYPE_PHOTO && !TextUtils.isEmpty(entity.segmentedPath)) { + try { + new File(entity.segmentedPath).delete(); + } catch (Exception e) { + FileLog.e(e); + } + entity.segmentedPath = ""; + } } - part.file = null; } if (round != null && (!isEdit || editedMedia)) { round.delete(); @@ -590,6 +635,115 @@ public static StoryEntry repostStoryItem(File file, TL_stories.StoryItem storyIt return entry; } + public static boolean canRepostMessage(MessageObject messageObject) { + if (messageObject == null || messageObject.isSponsored()) { + return false; + } + if (messageObject.messageOwner != null && messageObject.messageOwner.noforwards) { + return false; + } + if (messageObject.type == MessageObject.TYPE_POLL || messageObject.type == MessageObject.TYPE_CONTACT) { + return false; + } + long dialogId = messageObject.getDialogId(); + TLRPC.Chat chat = MessagesController.getInstance(messageObject.currentAccount).getChat(-dialogId); + if (chat != null && chat.noforwards) { + return false; + } + if (dialogId >= 0 || !ChatObject.isChannelAndNotMegaGroup(chat)) { + if (messageObject.messageOwner.fwd_from != null && messageObject.messageOwner.fwd_from.from_id != null && (messageObject.messageOwner.fwd_from.flags & 4) != 0) { + dialogId = DialogObject.getPeerDialogId(messageObject.messageOwner.fwd_from.from_id); + chat = MessagesController.getInstance(messageObject.currentAccount).getChat(-dialogId); + if (dialogId >= 0 || chat != null && chat.noforwards || !ChatObject.isChannelAndNotMegaGroup(chat) || !ChatObject.isPublic(chat)) { + return false; + } + return true; + } + return false; + } + return true; + } + + public static Boolean useForwardForRepost(MessageObject messageObject) { + if (messageObject == null || messageObject.messageOwner == null) return null; + TLRPC.Peer peer = messageObject.messageOwner.peer_id; + long dialogId = DialogObject.getPeerDialogId(peer); + TLRPC.Chat chat = MessagesController.getInstance(messageObject.currentAccount).getChat(-dialogId); + if (chat != null && chat.noforwards || !ChatObject.isChannelAndNotMegaGroup(chat)) { + if (messageObject.messageOwner.fwd_from != null && messageObject.messageOwner.fwd_from.from_id != null && (messageObject.messageOwner.fwd_from.flags & 4) != 0) { + dialogId = DialogObject.getPeerDialogId(messageObject.messageOwner.fwd_from.from_id); + chat = MessagesController.getInstance(messageObject.currentAccount).getChat(-dialogId); + if (dialogId >= 0 || chat != null && chat.noforwards || !ChatObject.isChannelAndNotMegaGroup(chat)) { + return null; // no repost + } else { + return true; // repost of forward + } + } + return null; // no repost + } + return false; // repost + } + + public static long getRepostDialogId(MessageObject messageObject) { + Boolean useForward = useForwardForRepost(messageObject); + if (useForward == null) return 0; + if (useForward) { + return DialogObject.getPeerDialogId(messageObject.messageOwner.fwd_from.from_id); + } else { + return messageObject.getDialogId(); + } + } + + public static int getRepostMessageId(MessageObject messageObject) { + Boolean useForward = useForwardForRepost(messageObject); + if (useForward == null) return 0; + if (useForward) { + return messageObject.messageOwner.fwd_from.channel_post; + } else { + return messageObject.getId(); + } + } + + public static StoryEntry repostMessage(ArrayList messageObjects) { + StoryEntry entry = new StoryEntry(); + entry.isRepostMessage = true; + entry.messageObjects = messageObjects; + entry.resultWidth = 1080; + entry.resultHeight = 1920; + MessageObject msg = messageObjects.get(0); + entry.backgroundWallpaperPeerId = getRepostDialogId(msg); + + VideoEditedInfo.MediaEntity entity = new VideoEditedInfo.MediaEntity(); + entity.type = VideoEditedInfo.MediaEntity.TYPE_MESSAGE; + entity.x = 0.5f; + entity.y = 0.5f; + entry.mediaEntities = new ArrayList<>(); + entry.mediaEntities.add(entity); + + if (messageObjects.size() == 1) { + MessageObject messageObject = messageObjects.get(0); + if (messageObject != null && (messageObject.type == MessageObject.TYPE_GIF || messageObject.type == MessageObject.TYPE_VIDEO || messageObject.type == MessageObject.TYPE_ROUND_VIDEO)) { + if (messageObject.messageOwner != null && messageObject.messageOwner.attachPath != null) { + entry.file = new File(messageObject.messageOwner.attachPath); + } + if (entry.file == null || !entry.file.exists()) { + entry.file = FileLoader.getInstance(entry.currentAccount).getPathToMessage(messageObject.messageOwner); + } + if (entry.file != null && entry.file.exists()) { + entry.isVideo = true; + entry.fileDeletable = false; + entry.duration = (long) (messageObject.getDuration() * 1000); + entry.left = 0; + entry.right = Math.min(1, 59_500f / entry.duration); + } else { + entry.file = null; + } + } + } + + return entry; + } + public static StoryEntry fromStoryItem(File file, TL_stories.StoryItem storyItem) { StoryEntry entry = new StoryEntry(); entry.isEdit = true; @@ -842,114 +996,124 @@ public void getVideoEditedInfo(@NonNull Utilities.Callback when resultWidth = 720; resultHeight = 1280; } - final String videoPath = file.getAbsolutePath(); - Utilities.globalQueue.postRunnable(() -> { - final int[] params = new int[AnimatedFileDrawable.PARAM_NUM_COUNT]; - AnimatedFileDrawable.getVideoInfo(videoPath, params); - AndroidUtilities.runOnUIThread(() -> { - VideoEditedInfo info = new VideoEditedInfo(); - - info.isStory = true; - info.fromCamera = fromCamera; - info.originalWidth = width; - info.originalHeight = height; - info.resultWidth = resultWidth; - info.resultHeight = resultHeight; - info.paintPath = paintFile == null ? null : paintFile.getPath(); - - final int encoderBitrate = MediaController.extractRealEncoderBitrate(info.resultWidth, info.resultHeight, info.bitrate, true); - if (isVideo) { + final String videoPath = file == null ? null : file.getAbsolutePath(); + final int[] params = new int[AnimatedFileDrawable.PARAM_NUM_COUNT]; + Runnable fill = () -> { + VideoEditedInfo info = new VideoEditedInfo(); + + info.isStory = true; + info.fromCamera = fromCamera; + info.originalWidth = width; + info.originalHeight = height; + info.resultWidth = resultWidth; + info.resultHeight = resultHeight; + info.paintPath = paintFile == null ? null : paintFile.getPath(); + info.messagePath = messageFile == null ? null : messageFile.getPath(); + info.messageVideoMaskPath = messageVideoMaskFile == null ? null : messageVideoMaskFile.getPath(); + info.backgroundPath = backgroundFile == null ? null : backgroundFile.getPath(); + + final int encoderBitrate = MediaController.extractRealEncoderBitrate(info.resultWidth, info.resultHeight, info.bitrate, true); + if (isVideo && videoPath != null) { + info.originalPath = videoPath; + info.isPhoto = false; + info.framerate = Math.min(59, params[AnimatedFileDrawable.PARAM_NUM_FRAMERATE]); + int videoBitrate = MediaController.getVideoBitrate(videoPath); + info.originalBitrate = videoBitrate == -1 ? params[AnimatedFileDrawable.PARAM_NUM_BITRATE] : videoBitrate; + if (info.originalBitrate < 1_000_000 && (mediaEntities != null && !mediaEntities.isEmpty())) { + info.bitrate = 2_000_000; + info.originalBitrate = -1; + } else if (info.originalBitrate < 500_000) { + info.bitrate = 2_500_000; + info.originalBitrate = -1; + } else { + info.bitrate = Utilities.clamp(info.originalBitrate, 3_000_000, 500_000); + } + FileLog.d("story bitrate, original = " + info.originalBitrate + " => " + info.bitrate); + info.originalDuration = (duration = params[AnimatedFileDrawable.PARAM_NUM_DURATION]) * 1000L; + info.startTime = (long) (left * duration) * 1000L; + info.endTime = (long) (right * duration) * 1000L; + info.estimatedDuration = info.endTime - info.startTime; + info.muted = muted; + info.estimatedSize = (long) (params[AnimatedFileDrawable.PARAM_NUM_AUDIO_FRAME_SIZE] + params[AnimatedFileDrawable.PARAM_NUM_DURATION] / 1000.0f * encoderBitrate / 8); + info.estimatedSize = Math.max(file.length(), info.estimatedSize); + info.filterState = filterState; + info.blurPath = paintBlurFile == null ? null : paintBlurFile.getPath(); + } else { + if (filterFile != null) { + info.originalPath = filterFile.getAbsolutePath(); + } else { info.originalPath = videoPath; - info.isPhoto = false; - info.framerate = Math.min(59, params[AnimatedFileDrawable.PARAM_NUM_FRAMERATE]); - int videoBitrate = MediaController.getVideoBitrate(videoPath); - info.originalBitrate = videoBitrate == -1 ? params[AnimatedFileDrawable.PARAM_NUM_BITRATE] : videoBitrate; - if (info.originalBitrate < 1_000_000 && (mediaEntities != null && !mediaEntities.isEmpty())) { - info.bitrate = 2_000_000; - info.originalBitrate = -1; - } else if (info.originalBitrate < 500_000) { - info.bitrate = 2_500_000; - info.originalBitrate = -1; - } else { - info.bitrate = Utilities.clamp(info.originalBitrate, 3_000_000, 500_000); - } - FileLog.d("story bitrate, original = " + info.originalBitrate + " => " + info.bitrate); - info.originalDuration = (duration = params[AnimatedFileDrawable.PARAM_NUM_DURATION]) * 1000L; - info.startTime = (long) (left * duration) * 1000L; - info.endTime = (long) (right * duration) * 1000L; - info.estimatedDuration = info.endTime - info.startTime; - info.muted = muted; - info.estimatedSize = (long) (params[AnimatedFileDrawable.PARAM_NUM_AUDIO_FRAME_SIZE] + params[AnimatedFileDrawable.PARAM_NUM_DURATION] / 1000.0f * encoderBitrate / 8); - info.estimatedSize = Math.max(file.length(), info.estimatedSize); - info.filterState = filterState; - info.blurPath = paintBlurFile == null ? null : paintBlurFile.getPath(); + } + info.isPhoto = true; + if (round != null) { + info.estimatedDuration = info.originalDuration = duration = (long) ((roundRight - roundLeft) * roundDuration); + } else if (audioPath != null) { + info.estimatedDuration = info.originalDuration = duration = (long) ((audioRight - audioLeft) * audioDuration); } else { - if (filterFile != null) { - info.originalPath = filterFile.getAbsolutePath(); - } else { - info.originalPath = videoPath; - } - info.isPhoto = true; - if (round != null) { - info.estimatedDuration = info.originalDuration = duration = (long) ((roundRight - roundLeft) * roundDuration); - } else if (audioPath != null) { - info.estimatedDuration = info.originalDuration = duration = (long) ((audioRight - audioLeft) * audioDuration); - } else { - info.estimatedDuration = info.originalDuration = duration = averageDuration; - } - info.startTime = -1; - info.endTime = -1; - info.muted = true; - info.originalBitrate = -1; - info.bitrate = -1; - info.framerate = 30; - info.estimatedSize = (long) (duration / 1000.0f * encoderBitrate / 8); - info.filterState = null; + info.estimatedDuration = info.originalDuration = duration = averageDuration; } - info.avatarStartTime = -1; + info.startTime = -1; + info.endTime = -1; + info.muted = true; + info.originalBitrate = -1; + info.bitrate = -1; + info.framerate = 30; + info.estimatedSize = (long) (duration / 1000.0f * encoderBitrate / 8); + info.filterState = null; + } + info.account = currentAccount; + info.wallpaperPeerId = backgroundWallpaperPeerId; + info.isDark = isDark; + info.avatarStartTime = -1; - info.cropState = new MediaController.CropState(); - info.cropState.useMatrix = new Matrix(); - info.cropState.useMatrix.set(matrix); + info.cropState = new MediaController.CropState(); + info.cropState.useMatrix = new Matrix(); + info.cropState.useMatrix.set(matrix); - info.mediaEntities = mediaEntities; + info.mediaEntities = mediaEntities; - info.gradientTopColor = gradientTopColor; - info.gradientBottomColor = gradientBottomColor; - info.forceFragmenting = true; + info.gradientTopColor = gradientTopColor; + info.gradientBottomColor = gradientBottomColor; + info.forceFragmenting = true; - info.hdrInfo = hdrInfo; - info.parts = parts; + info.hdrInfo = hdrInfo; - info.mixedSoundInfos.clear(); - if (round != null) { - final MediaCodecVideoConvertor.MixedSoundInfo soundInfo = new MediaCodecVideoConvertor.MixedSoundInfo(round.getAbsolutePath()); - soundInfo.volume = roundVolume; - soundInfo.audioOffset = (long) (roundLeft * roundDuration) * 1000L; - if (isVideo) { - soundInfo.startTime = (long) (roundOffset - left * duration) * 1000L; - } else { - soundInfo.startTime = 0; - } - soundInfo.duration = (long) ((roundRight - roundLeft) * roundDuration) * 1000L; - info.mixedSoundInfos.add(soundInfo); + info.mixedSoundInfos.clear(); + if (round != null) { + final MediaCodecVideoConvertor.MixedSoundInfo soundInfo = new MediaCodecVideoConvertor.MixedSoundInfo(round.getAbsolutePath()); + soundInfo.volume = roundVolume; + soundInfo.audioOffset = (long) (roundLeft * roundDuration) * 1000L; + if (isVideo) { + soundInfo.startTime = (long) (roundOffset - left * duration) * 1000L; + } else { + soundInfo.startTime = 0; } - if (audioPath != null) { - final MediaCodecVideoConvertor.MixedSoundInfo soundInfo = new MediaCodecVideoConvertor.MixedSoundInfo(audioPath); - soundInfo.volume = audioVolume; - soundInfo.audioOffset = (long) (audioLeft * audioDuration) * 1000L; - if (isVideo) { - soundInfo.startTime = (long) (audioOffset - left * duration) * 1000L; - } else { - soundInfo.startTime = 0; - } - soundInfo.duration = (long) ((audioRight - audioLeft) * audioDuration) * 1000L; - info.mixedSoundInfos.add(soundInfo); + soundInfo.duration = (long) ((roundRight - roundLeft) * roundDuration) * 1000L; + info.mixedSoundInfos.add(soundInfo); + } + if (audioPath != null) { + final MediaCodecVideoConvertor.MixedSoundInfo soundInfo = new MediaCodecVideoConvertor.MixedSoundInfo(audioPath); + soundInfo.volume = audioVolume; + soundInfo.audioOffset = (long) (audioLeft * audioDuration) * 1000L; + if (isVideo) { + soundInfo.startTime = (long) (audioOffset - left * duration) * 1000L; + } else { + soundInfo.startTime = 0; } + soundInfo.duration = (long) ((audioRight - audioLeft) * audioDuration) * 1000L; + info.mixedSoundInfos.add(soundInfo); + } - whenDone.run(info); + whenDone.run(info); + }; + if (file == null) { + fill.run(); + } else { + Utilities.globalQueue.postRunnable(() -> { + AnimatedFileDrawable.getVideoInfo(videoPath, params); + AndroidUtilities.runOnUIThread(fill); }); - }); + } } public static File makeCacheFile(final int account, boolean video) { @@ -1166,9 +1330,6 @@ public StoryEntry copy() { newEntry.height = height; newEntry.resultWidth = resultWidth; newEntry.resultHeight = resultHeight; - newEntry.partsMaxId = partsMaxId; - newEntry.parts.clear(); - newEntry.parts.addAll(parts); newEntry.peer = peer; newEntry.invert = invert; newEntry.matrix.set(matrix); @@ -1189,6 +1350,8 @@ public StoryEntry copy() { newEntry.uploadThumbFile = uploadThumbFile; newEntry.draftThumbFile = draftThumbFile; newEntry.paintFile = paintFile; + newEntry.messageFile = messageFile; + newEntry.backgroundFile = backgroundFile; newEntry.paintBlurFile = paintBlurFile; newEntry.paintEntitiesFile = paintEntitiesFile; newEntry.averageDuration = averageDuration; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/StoryRecorder.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/StoryRecorder.java index 7163aaac328..55bbf5ee988 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/StoryRecorder.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/StoryRecorder.java @@ -20,6 +20,7 @@ import android.content.pm.PackageManager; import android.graphics.Bitmap; import android.graphics.BitmapFactory; +import android.graphics.BitmapShader; import android.graphics.Canvas; import android.graphics.Color; import android.graphics.Insets; @@ -31,6 +32,7 @@ import android.graphics.PixelFormat; import android.graphics.PorterDuff; import android.graphics.PorterDuffColorFilter; +import android.graphics.PorterDuffXfermode; import android.graphics.Rect; import android.graphics.RectF; import android.graphics.Shader; @@ -53,7 +55,6 @@ import android.text.style.ForegroundColorSpan; import android.text.style.ImageSpan; import android.text.style.URLSpan; -import android.util.Log; import android.util.Pair; import android.view.Gravity; import android.view.HapticFeedbackConstants; @@ -68,6 +69,7 @@ import android.view.WindowManager; import android.widget.FrameLayout; import android.widget.ImageView; +import android.widget.LinearLayout; import android.widget.TextView; import androidx.annotation.NonNull; @@ -82,7 +84,6 @@ import org.telegram.messenger.AnimationNotificationsLocker; import org.telegram.messenger.ApplicationLoader; import org.telegram.messenger.BotWebViewVibrationEffect; -import org.telegram.messenger.ChatObject; import org.telegram.messenger.DialogObject; import org.telegram.messenger.FileLoader; import org.telegram.messenger.FileLog; @@ -111,20 +112,25 @@ import org.telegram.ui.ActionBar.SimpleTextView; import org.telegram.ui.ActionBar.Theme; import org.telegram.ui.AvatarSpan; -import org.telegram.ui.BasePermissionsActivity; +import org.telegram.ui.Cells.ChatMessageCell; +import org.telegram.ui.Cells.ShareDialogCell; import org.telegram.ui.Components.AlertsCreator; -import org.telegram.ui.Components.BlobDrawable; +import org.telegram.ui.Components.BackupImageView; import org.telegram.ui.Components.BlurringShader; import org.telegram.ui.Components.Bulletin; import org.telegram.ui.Components.BulletinFactory; import org.telegram.ui.Components.CombinedDrawable; import org.telegram.ui.Components.CubicBezierInterpolator; +import org.telegram.ui.Components.Easings; import org.telegram.ui.Components.EmojiView; import org.telegram.ui.Components.FilterShaders; import org.telegram.ui.Components.GestureDetectorFixDoubleTap; import org.telegram.ui.Components.ItemOptions; import org.telegram.ui.Components.LayoutHelper; import org.telegram.ui.Components.Paint.RenderView; +import org.telegram.ui.Components.Paint.Views.EntityView; +import org.telegram.ui.Components.Paint.Views.MessageEntityView; +import org.telegram.ui.Components.Paint.Views.PhotoView; import org.telegram.ui.Components.Paint.Views.RoundView; import org.telegram.ui.Components.PhotoFilterBlurControl; import org.telegram.ui.Components.PhotoFilterCurvesControl; @@ -135,6 +141,7 @@ import org.telegram.ui.Components.RLottieImageView; import org.telegram.ui.Components.SizeNotifierFrameLayout; import org.telegram.ui.Components.TextStyleSpan; +import org.telegram.ui.Components.ThanosEffect; import org.telegram.ui.Components.URLSpanUserMention; import org.telegram.ui.Components.VideoEditTextureView; import org.telegram.ui.Components.WaveDrawable; @@ -172,6 +179,7 @@ public class StoryRecorder implements NotificationCenter.NotificationCenterDeleg private WindowView windowView; private ContainerView containerView; private FlashViews flashViews; + private ThanosEffect thanosEffect; private static StoryRecorder instance; private boolean wasSend; @@ -243,7 +251,9 @@ public static class SourceView { int type = 0; float rounding; RectF screenRect = new RectF(); + Drawable backgroundDrawable; ImageReceiver backgroundImageReceiver; + boolean hasShadow; Paint backgroundPaint; Drawable iconDrawable; int iconSize; @@ -343,6 +353,7 @@ protected void hide() { final View imageView = floatingButton.getChildAt(0); imageView.getLocationOnScreen(loc); src.screenRect.set(loc[0], loc[1], loc[0] + imageView.getWidth(), loc[1] + imageView.getHeight()); + src.hasShadow = true; src.backgroundPaint = new Paint(Paint.ANTI_ALIAS_FLAG); src.backgroundPaint.setColor(Theme.getColor(Theme.key_chats_actionBackground)); src.iconDrawable = floatingButton.getContext().getResources().getDrawable(R.drawable.story_camera).mutate(); @@ -351,6 +362,36 @@ protected void hide() { return src; } + public static SourceView fromShareCell(ShareDialogCell shareDialogCell) { + if (shareDialogCell == null) { + return null; + } + BackupImageView imageView = shareDialogCell.getImageView(); + SourceView src = new SourceView() { + @Override + protected void show() { + imageView.setVisibility(View.VISIBLE); + } + @Override + protected void hide() { + imageView.post(() -> { + imageView.setVisibility(View.GONE); + }); + } + }; + int[] loc = new int[2]; + imageView.getLocationOnScreen(loc); + src.screenRect.set(loc[0], loc[1], loc[0] + imageView.getWidth(), loc[1] + imageView.getHeight()); + src.backgroundDrawable = new ShareDialogCell.RepostStoryDrawable(imageView.getContext(), null, false, shareDialogCell.resourcesProvider); +// src.hasShadow = false; +// src.backgroundPaint = new Paint(Paint.ANTI_ALIAS_FLAG); +// src.backgroundPaint.setColor(Theme.getColor(Theme.key_chats_actionBackground)); +// src.iconDrawable = shareDialogCell.getContext().getResources().getDrawable(R.drawable.large_repost_story).mutate(); +// src.iconSize = AndroidUtilities.dp(30); + src.rounding = Math.max(src.screenRect.width(), src.screenRect.height()) / 2f; + return src; + } + public static SourceView fromStoryCell(DialogStoriesCell.StoryCell storyCell) { if (storyCell == null || storyCell.getRootView() == null) { return null; @@ -403,6 +444,7 @@ public void replaceSourceView(SourceView sourceView) { fromRect.set(sourceView.screenRect); fromRounding = sourceView.rounding; } else { + fromSourceView = null; openType = 0; fromRect.set(0, dp(100), AndroidUtilities.displaySize.x, dp(100) + AndroidUtilities.displaySize.y); fromRounding = dp(8); @@ -422,6 +464,7 @@ public void open(SourceView sourceView, boolean animated) { prepareClosing = false; // privacySelectorHintOpened = false; forceBackgroundVisible = false; + videoTextureHolder.active = false; if (windowManager != null && windowView != null && windowView.getParent() == null) { windowManager.addView(windowView, windowLayoutParams); @@ -479,6 +522,7 @@ public void openEdit(SourceView sourceView, StoryEntry entry, long time, boolean outputEntry = entry; isVideo = outputEntry != null && outputEntry.isVideo; + videoTextureHolder.active = false; if (sourceView != null) { fromSourceView = sourceView; @@ -513,6 +557,7 @@ public void openEdit(SourceView sourceView, StoryEntry entry, long time, boolean }, time); navigateTo(PAGE_PREVIEW, false); switchToEditMode(EDIT_MODE_NONE, false); + previewButtons.appear(false, false); addNotificationObservers(); } @@ -532,6 +577,7 @@ public void openForward(SourceView sourceView, StoryEntry entry, long time, bool outputEntry = entry; StoryPrivacySelector.applySaved(currentAccount, outputEntry); isVideo = outputEntry != null && outputEntry.isVideo; + videoTextureHolder.active = false; if (sourceView != null) { fromSourceView = sourceView; @@ -562,10 +608,64 @@ public void openForward(SourceView sourceView, StoryEntry entry, long time, bool navigateToPreviewWithPlayerAwait(() -> { animateOpenTo(1, animated, this::onOpenDone); - previewButtons.appear(true, true); }, time); + previewButtons.appear(true, false); + navigateTo(PAGE_PREVIEW, false); + switchToEditMode(EDIT_MODE_NONE, false); + + addNotificationObservers(); + } + + private static boolean firstOpen = true; + public void openRepost(SourceView sourceView, StoryEntry entry) { + if (isShown) { + return; + } + + prepareClosing = false; + forceBackgroundVisible = false; + + if (windowManager != null && windowView != null && windowView.getParent() == null) { + windowManager.addView(windowView, windowLayoutParams); + } + + outputEntry = entry; + StoryPrivacySelector.applySaved(currentAccount, outputEntry); + isVideo = outputEntry != null && outputEntry.isVideo; + videoTextureHolder.active = isVideo; + + if (sourceView != null) { + fromSourceView = sourceView; + openType = sourceView.type; + fromRect.set(sourceView.screenRect); + fromRounding = sourceView.rounding; + fromSourceView.hide(); + } else { + openType = 0; + fromRect.set(0, dp(100), AndroidUtilities.displaySize.x, dp(100) + AndroidUtilities.displaySize.y); + fromRounding = dp(8); + } + + containerView.updateBackground(); + previewContainer.setBackgroundColor(openType == 1 || openType == 0 ? 0 : 0xff1f1f1f); + + containerView.setTranslationX(0); + containerView.setTranslationY(0); + containerView.setTranslationY2(0); + containerView.setScaleX(1f); + containerView.setScaleY(1f); + dismissProgress = 0; + + AndroidUtilities.lockOrientation(activity, ActivityInfo.SCREEN_ORIENTATION_PORTRAIT); + + if (outputEntry != null) { + captionEdit.setText(outputEntry.caption); + } + + previewButtons.appear(true, false); navigateTo(PAGE_PREVIEW, false); switchToEditMode(EDIT_MODE_NONE, false); + animateOpenTo(1, true, this::onOpenDone); addNotificationObservers(); } @@ -662,6 +762,9 @@ public void onAnimationEnd(Animator animation) { onFullyOpenListener.run(); onFullyOpenListener = null; } + + containerView.invalidate(); + previewContainer.invalidate(); } }); if (value < 1 && wasSend) { @@ -669,7 +772,7 @@ public void onAnimationEnd(Animator animation) { openCloseAnimator.setInterpolator(CubicBezierInterpolator.EASE_OUT_QUINT); } else { if (value > 0 || containerView.getTranslationY1() < AndroidUtilities.dp(20)) { - openCloseAnimator.setDuration(270L); + openCloseAnimator.setDuration(300L); openCloseAnimator.setInterpolator(new FastOutSlowInInterpolator()); } else { openCloseAnimator.setDuration(400L); @@ -688,6 +791,9 @@ public void onAnimationEnd(Animator animation) { } checkBackgroundVisibility(); } + if (value > 0) { + firstOpen = false; + } } private void onOpenDone() { @@ -713,6 +819,11 @@ private void onOpenDone() { createPhotoPaintView(); hidePhotoPaintView(); createFilterPhotoView(); + } else if (outputEntry != null && outputEntry.isRepostMessage) { + if (outputEntry.isVideo) { + previewView.setupVideoPlayer(outputEntry, null, 0); + } + createFilterPhotoView(); } } @@ -814,7 +925,7 @@ public class WindowView extends SizeNotifierFrameLayout { private GestureDetectorFixDoubleTap gestureDetector; private ScaleGestureDetector scaleGestureDetector; - public WindowView(Context context) { + public WindowView(Context context) { super(context); gestureDetector = new GestureDetectorFixDoubleTap(context, new GestureListener()); scaleGestureDetector = new ScaleGestureDetector(context, new ScaleListener()); @@ -884,8 +995,14 @@ protected void dispatchDraw(Canvas canvas) { fromSourceView.backgroundImageReceiver.setAlpha(alpha); fromSourceView.backgroundImageReceiver.draw(canvas); fromSourceView.backgroundImageReceiver.setRoundRadius(prevRoundRadius); + } else if (fromSourceView.backgroundDrawable != null) { + fromSourceView.backgroundDrawable.setBounds((int) rectF.left, (int) rectF.top, (int) rectF.right, (int) rectF.bottom); + fromSourceView.backgroundDrawable.setAlpha((int) (0xFF * alpha * alpha * alpha)); + fromSourceView.backgroundDrawable.draw(canvas); } else if (fromSourceView.backgroundPaint != null) { - fromSourceView.backgroundPaint.setShadowLayer(dp(2), 0, dp(3), Theme.multAlpha(0x33000000, alpha)); + if (fromSourceView.hasShadow) { + fromSourceView.backgroundPaint.setShadowLayer(dp(2), 0, dp(3), Theme.multAlpha(0x33000000, alpha)); + } fromSourceView.backgroundPaint.setAlpha((int) (0xFF * alpha)); canvas.drawRoundRect(rectF, r, r, fromSourceView.backgroundPaint); } @@ -1194,6 +1311,21 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { MeasureSpec.makeMeasureSpec(W, MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec(H, MeasureSpec.EXACTLY) ); + if (thanosEffect != null) { + thanosEffect.measure( + MeasureSpec.makeMeasureSpec(W, MeasureSpec.EXACTLY), + MeasureSpec.makeMeasureSpec(H, MeasureSpec.EXACTLY) + ); + } + if (changeDayNightView != null) { + changeDayNightView.measure( + MeasureSpec.makeMeasureSpec(W, MeasureSpec.EXACTLY), + MeasureSpec.makeMeasureSpec(H, MeasureSpec.EXACTLY) + ); + } + if (themeSheet != null) { + themeSheet.measure(widthMeasureSpec, heightMeasureSpec); + } if (galleryListView != null) { galleryListView.measure(MeasureSpec.makeMeasureSpec(previewW, MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec(H, MeasureSpec.EXACTLY)); @@ -1276,10 +1408,19 @@ protected void onLayout(boolean changed, int left, int top, int right, int botto containerView.layout(l, t, r, b); flashViews.backgroundView.layout(0, 0, W, H); + if (thanosEffect != null) { + thanosEffect.layout(0, 0, W, H); + } + if (changeDayNightView != null) { + changeDayNightView.layout(0, 0, W, H); + } if (galleryListView != null) { galleryListView.layout((W - galleryListView.getMeasuredWidth()) / 2, 0, (W + galleryListView.getMeasuredWidth()) / 2, H); } + if (themeSheet != null) { + themeSheet.layout((W - themeSheet.getMeasuredWidth()) / 2, H - themeSheet.getMeasuredHeight(), (W + themeSheet.getMeasuredWidth()) / 2, H); + } if (captionEdit != null) { EmojiView emojiView = captionEdit.editText.getEmojiView(); @@ -1374,6 +1515,14 @@ public void updateBackground() { } } + @Override + public void invalidate() { + if (openCloseAnimator != null && openCloseAnimator.isRunning()) { + return; + } + super.invalidate(); + } + private float translationY1; private float translationY2; @@ -1496,7 +1645,7 @@ protected boolean drawChild(Canvas canvas, View child, long drawingTime) { topGradient = new LinearGradient(0, top, 0, top + dp(72), new int[] {0x40000000, 0x00000000}, new float[] { top / (top + dp(72)), 1 }, Shader.TileMode.CLAMP ); topGradientPaint.setShader(topGradient); } - topGradientPaint.setAlpha((int) (0xFF * openProgress)); + topGradientPaint.setAlpha(0xFF); AndroidUtilities.rectTmp.set(0, 0, getWidth(), dp(72 + 12) + top); canvas.drawRoundRect(AndroidUtilities.rectTmp, dp(12), dp(12), topGradientPaint); } @@ -1515,6 +1664,7 @@ protected boolean drawChild(Canvas canvas, View child, long drawingTime) { private FrameLayout previewContainer; private FrameLayout actionBarContainer; + private LinearLayout actionBarButtons; private FrameLayout controlContainer; private FrameLayout captionContainer; private FrameLayout navbarContainer; @@ -1523,6 +1673,7 @@ protected boolean drawChild(Canvas canvas, View child, long drawingTime) { private SimpleTextView titleTextView; private StoryPrivacyBottomSheet privacySheet; private BlurringShader.BlurManager blurManager; + private PreviewView.TextureViewHolder videoTextureHolder; private View captionEditOverlay; /* PAGE_CAMERA */ @@ -1542,6 +1693,7 @@ protected boolean drawChild(Canvas canvas, View child, long drawingTime) { private HintTextView hintTextView; private ZoomControlView zoomControlView; private HintView2 cameraHint; + private StoryThemeSheet themeSheet; /* PAGE_PREVIEW */ private PreviewView previewView; @@ -1553,6 +1705,8 @@ protected boolean drawChild(Canvas canvas, View child, long drawingTime) { private DownloadButton downloadButton; private RLottieDrawable muteButtonDrawable; private RLottieImageView muteButton; + private RLottieDrawable themeButtonDrawable; + private ImageView themeButton; private PlayPauseButton playButton; private HintView2 muteHint; private HintView2 dualHint; @@ -1682,10 +1836,19 @@ protected void onLayout(boolean changed, int left, int top, int right, int botto setSystemGestureExclusionRects(Arrays.asList(leftExclRect, rightExclRect)); } } + + @Override + public void invalidate() { + if (openCloseAnimator != null && openCloseAnimator.isRunning()) { + return; + } + super.invalidate(); + } }); containerView.addView(flashViews.foregroundView, new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT)); blurManager = new BlurringShader.BlurManager(previewContainer); + videoTextureHolder = new PreviewView.TextureViewHolder(); containerView.addView(actionBarContainer = new FrameLayout(context)); // 150dp containerView.addView(controlContainer = new FrameLayout(context)); // 220dp containerView.addView(captionContainer = new FrameLayout(context) { @@ -1739,7 +1902,7 @@ public void getOutline(View view, Outline outline) { previewContainer.setClipToOutline(true); } photoFilterEnhanceView = new PhotoFilterView.EnhanceView(context, this::createFilterPhotoView); - previewView = new PreviewView(context, blurManager) { + previewView = new PreviewView(context, blurManager, videoTextureHolder) { @Override public boolean additionalTouchEvent(MotionEvent ev) { if (captionEdit != null && captionEdit.isRecording()) { @@ -1774,9 +1937,6 @@ public void onEntityDragEnd(boolean delete) { trash.animate().alpha(0f).withEndAction(() -> { trash.setVisibility(View.GONE); }).setDuration(180).setInterpolator(CubicBezierInterpolator.EASE_OUT).setStartDelay(delete ? 500 : 0).start(); - if (delete) { - deleteCurrentPart(); - } super.onEntityDragEnd(delete); } @@ -1838,6 +1998,18 @@ public void onRoundRemove() { } } } + + @Override + protected void invalidateTextureViewHolder() { + if (outputEntry != null && outputEntry.isRepostMessage && outputEntry.isVideo && paintView != null && paintView.entitiesView != null) { + for (int i = 0; i < paintView.entitiesView.getChildCount(); ++i) { + View child = paintView.entitiesView.getChildAt(i); + if (child instanceof MessageEntityView) { + ((MessageEntityView) child).invalidateAll(); + } + } + } + } }; previewView.invalidateBlur = this::invalidateBlur; previewView.setOnTapListener(() -> { @@ -1864,6 +2036,29 @@ public void onRoundRemove() { previewContainer.addView(photoFilterEnhanceView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT, Gravity.FILL)); captionEdit = new CaptionStory(context, windowView, windowView, containerView, resourcesProvider, blurManager) { + @Override + protected boolean ignoreTouches(float x, float y) { + if (paintView == null || paintView.entitiesView == null || captionEdit.keyboardShown) return false; + x += captionEdit.getX(); + y += captionEdit.getY(); + x += captionContainer.getX(); + y += captionContainer.getY(); + x -= previewContainer.getX(); + y -= previewContainer.getY(); + + for (int i = 0; i < paintView.entitiesView.getChildCount(); ++i) { + View view = paintView.entitiesView.getChildAt(i); + if (view instanceof EntityView) { + org.telegram.ui.Components.Rect rect = ((EntityView) view).getSelectionBounds(); + AndroidUtilities.rectTmp.set(rect.x, rect.y, rect.x + rect.width, rect.y + rect.height); + if (AndroidUtilities.rectTmp.contains(x, y)) { + return true; + } + } + } + return false; + } + @Override protected void drawBlurBitmap(Bitmap bitmap, float amount) { windowView.drawBlurBitmap(bitmap, amount); @@ -1872,7 +2067,7 @@ protected void drawBlurBitmap(Bitmap bitmap, float amount) { @Override protected boolean captionLimitToast() { - if (MessagesController.getInstance(currentAccount).premiumLocked) { + if (MessagesController.getInstance(currentAccount).premiumFeaturesBlocked()) { return false; } Bulletin visibleBulletin = Bulletin.getVisibleBulletin(); @@ -2099,14 +2294,19 @@ protected void dispatchDraw(Canvas canvas) { titleTextView.setRightPadding(AndroidUtilities.dp(144)); actionBarContainer.addView(titleTextView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 56, Gravity.TOP | Gravity.FILL_HORIZONTAL, 71, 0, 0, 0)); + actionBarButtons = new LinearLayout(context); + actionBarButtons.setOrientation(LinearLayout.HORIZONTAL); + actionBarButtons.setGravity(Gravity.RIGHT); + actionBarContainer.addView(actionBarButtons, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 56, Gravity.RIGHT | Gravity.FILL_HORIZONTAL, 0, 0, 8, 0)); + downloadButton = new DownloadButton(context, done -> { applyPaint(); + applyPaintMessage(); applyFilter(done); }, currentAccount, windowView, resourcesProvider); - actionBarContainer.addView(downloadButton, LayoutHelper.createFrame(56, 56, Gravity.TOP | Gravity.RIGHT)); muteHint = new HintView2(activity, HintView2.DIRECTION_TOP) - .setJoint(1, -68) + .setJoint(1, -77 + 8 - 2) .setDuration(2000) .setBounce(false) .setAnimatedTextHacks(true, true, false); @@ -2137,7 +2337,6 @@ protected void dispatchDraw(Canvas canvas) { }); muteButton.setVisibility(View.GONE); muteButton.setAlpha(0f); - actionBarContainer.addView(muteButton, LayoutHelper.createFrame(56, 56, Gravity.TOP | Gravity.RIGHT, 0, 0, 48, 0)); playButton = new PlayPauseButton(context); playButton.setBackground(Theme.createSelectorDrawable(0x20ffffff)); @@ -2148,7 +2347,10 @@ protected void dispatchDraw(Canvas canvas) { previewView.play(!playing); playButton.drawable.setPause(!playing, true); }); - actionBarContainer.addView(playButton, LayoutHelper.createFrame(56, 56, Gravity.TOP | Gravity.RIGHT, 0, 0, 48 + 48, 0)); + + actionBarButtons.addView(playButton, LayoutHelper.createLinear(46, 56, Gravity.TOP | Gravity.RIGHT)); + actionBarButtons.addView(muteButton, LayoutHelper.createLinear(46, 56, Gravity.TOP | Gravity.RIGHT)); + actionBarButtons.addView(downloadButton, LayoutHelper.createFrame(46, 56, Gravity.TOP | Gravity.RIGHT)); flashButton = new ToggleButton2(context); flashButton.setBackground(Theme.createSelectorDrawable(0x20ffffff)); @@ -2492,12 +2694,10 @@ private void upload(boolean asStory) { return; } preparingUpload = true; - Utilities.globalQueue.postRunnable(() -> { - applyPaint(); - AndroidUtilities.runOnUIThread(() -> { - preparingUpload = false; - uploadInternal(asStory); - }); + applyPaintInBackground(() -> { + applyPaintMessage(); + preparingUpload = false; + uploadInternal(asStory); }); } @@ -2599,7 +2799,7 @@ private File prepareThumb(StoryEntry storyEntry, boolean forDraft) { final Paint bitmapPaint = new Paint(Paint.FILTER_BITMAP_FLAG); TextureView textureView = previewView.getTextureView(); - if (storyEntry.isVideo && textureView != null) { + if (storyEntry.isVideo && !storyEntry.isRepostMessage && textureView != null) { Bitmap previewTextureView = textureView.getBitmap(); Matrix matrix = textureView.getTransform(null); if (matrix != null) { @@ -2641,9 +2841,11 @@ private File prepareThumb(StoryEntry storyEntry, boolean forDraft) { if (paintView != null && paintView.entitiesView != null) { canvas.save(); canvas.scale(scale, scale); + paintView.drawForThemeToggle = true; paintView.entitiesView.drawForThumb = true; paintView.entitiesView.draw(canvas); paintView.entitiesView.drawForThumb = false; + paintView.drawForThemeToggle = false; canvas.restore(); } @@ -3166,6 +3368,9 @@ public void onAnimationEnd(Animator animation) { } public boolean onBackPressed() { + if (openCloseAnimator != null && openCloseAnimator.isRunning()) { + return false; + } if (captionEdit != null && captionEdit.stopRecording()) { return false; } @@ -3178,6 +3383,9 @@ public boolean onBackPressed() { } if (captionEdit.onBackPressed()) { return false; + } else if (themeSheet != null) { + themeSheet.dismiss(); + return false; } else if (galleryListView != null) { if (galleryListView.onBackPressed()) { return false; @@ -3190,10 +3398,10 @@ public boolean onBackPressed() { } else if (currentEditMode > EDIT_MODE_NONE) { switchToEditMode(EDIT_MODE_NONE, true); return false; - } else if (currentPage == PAGE_PREVIEW && (outputEntry == null || !outputEntry.isRepost) && (outputEntry == null || !outputEntry.isEdit || (paintView != null && paintView.hasChanges()) || outputEntry.editedMedia || outputEntry.editedCaption)) { + } else if (currentPage == PAGE_PREVIEW && (outputEntry == null || !outputEntry.isRepost && !outputEntry.isRepostMessage) && (outputEntry == null || !outputEntry.isEdit || (paintView != null && paintView.hasChanges()) || outputEntry.editedMedia || outputEntry.editedCaption)) { if (paintView != null && paintView.onBackPressed()) { return false; - } else if ((fromGallery && (paintView == null || !paintView.hasChanges()) && (outputEntry == null || outputEntry.filterFile == null) || !previewButtons.isShareEnabled()) && (outputEntry == null || !outputEntry.isEdit || !outputEntry.isRepost)) { + } else if ((fromGallery && (paintView == null || !paintView.hasChanges()) && (outputEntry == null || outputEntry.filterFile == null) || !previewButtons.isShareEnabled()) && (outputEntry == null || !outputEntry.isEdit || !outputEntry.isRepost && !outputEntry.isRepostMessage)) { navigateTo(PAGE_CAMERA, true); } else { showDismissEntry(); @@ -3239,6 +3447,9 @@ public void updateDrawState(TextPaint tp) { private Runnable afterPlayerAwait; private boolean previewAlreadySet; public void navigateToPreviewWithPlayerAwait(Runnable open, long seekTo) { + navigateToPreviewWithPlayerAwait(open, seekTo, 800); + } + public void navigateToPreviewWithPlayerAwait(Runnable open, long seekTo, long ms) { if (awaitingPlayer || outputEntry == null) { return; } @@ -3258,7 +3469,7 @@ public void navigateToPreviewWithPlayerAwait(Runnable open, long seekTo) { previewView.setVisibility(View.VISIBLE); previewView.set(outputEntry, afterPlayerAwait, seekTo); previewView.setupAudio(outputEntry, false); - AndroidUtilities.runOnUIThread(afterPlayerAwait, 800); + AndroidUtilities.runOnUIThread(afterPlayerAwait, ms); } private AnimatorSet pageAnimator; @@ -3276,7 +3487,7 @@ public void navigateTo(int page, boolean animated) { onNavigateStart(oldPage, page); if (previewButtons != null) { - previewButtons.appear(page == PAGE_PREVIEW && openProgress > 0, animated); + previewButtons.appear(page == PAGE_PREVIEW, animated); } showVideoTimer(page == PAGE_CAMERA && isVideo, animated); if (page != PAGE_PREVIEW) { @@ -3310,9 +3521,11 @@ public void navigateTo(int page, boolean animated) { animators.add(ObjectAnimator.ofFloat(timelineView, View.ALPHA, page == PAGE_PREVIEW ? 1f : 0)); animators.add(ObjectAnimator.ofFloat(muteButton, View.ALPHA, page == PAGE_PREVIEW && isVideo ? 1f : 0)); - ((ViewGroup.MarginLayoutParams) playButton.getLayoutParams()).rightMargin = dp(48 + (isVideo ? 48 : 0)); animators.add(ObjectAnimator.ofFloat(playButton, View.ALPHA, page == PAGE_PREVIEW && (isVideo || outputEntry != null && !TextUtils.isEmpty(outputEntry.audioPath)) ? 1f : 0)); animators.add(ObjectAnimator.ofFloat(downloadButton, View.ALPHA, page == PAGE_PREVIEW ? 1f : 0)); + if (themeButton != null) { + animators.add(ObjectAnimator.ofFloat(themeButton, View.ALPHA, page == PAGE_PREVIEW && (outputEntry != null && outputEntry.isRepostMessage) ? 1f : 0)); + } // animators.add(ObjectAnimator.ofFloat(privacySelector, View.ALPHA, page == PAGE_PREVIEW ? 1f : 0)); animators.add(ObjectAnimator.ofFloat(zoomControlView, View.ALPHA, 0)); @@ -3346,9 +3559,11 @@ public void onAnimationEnd(Animator animation) { captionContainer.setAlpha(page == PAGE_PREVIEW ? 1f : 0); captionContainer.setTranslationY(page == PAGE_PREVIEW ? 0 : dp(12)); muteButton.setAlpha(page == PAGE_PREVIEW && isVideo ? 1f : 0); - ((ViewGroup.MarginLayoutParams) playButton.getLayoutParams()).rightMargin = dp(48 + (isVideo ? 48 : 0)); playButton.setAlpha(page == PAGE_PREVIEW && (isVideo || outputEntry != null && !TextUtils.isEmpty(outputEntry.audioPath)) ? 1f : 0); downloadButton.setAlpha(page == PAGE_PREVIEW ? 1f : 0); + if (themeButton != null) { + themeButton.setAlpha(page == PAGE_PREVIEW && (outputEntry != null && outputEntry.isRepostMessage) ? 1f : 0); + } // privacySelector.setAlpha(page == PAGE_PREVIEW ? 1f : 0); timelineView.setAlpha(page == PAGE_PREVIEW ? 1f : 0); titleTextView.setAlpha(page == PAGE_PREVIEW ? 1f : 0f); @@ -3385,6 +3600,22 @@ public void onAnimationEnd(Animator animation) { containerViewBackAnimator.start(); } + private void openThemeSheet() { + if (themeSheet == null) { + themeSheet = new StoryThemeSheet(getContext(), currentAccount, resourcesProvider, () -> { + windowView.removeView(themeSheet); + themeSheet = null; + }) { + @Override + protected void updateWallpaper() { + previewView.setupWallpaper(outputEntry, true); + } + }; + windowView.addView(themeSheet); + } + themeSheet.open(outputEntry); + } + private Parcelable lastGalleryScrollPosition; private MediaController.AlbumEntry lastGallerySelectedAlbum; @@ -3672,6 +3903,12 @@ private void onNavigateStart(int fromPage, int toPage) { titleTextView.setRightPadding(AndroidUtilities.dp(48)); } downloadButton.setVisibility(View.VISIBLE); + if (outputEntry != null && outputEntry.isRepostMessage) { + getThemeButton().setVisibility(View.VISIBLE); + updateThemeButtonDrawable(false); + } else if (themeButton != null) { + themeButton.setVisibility(View.GONE); + } // privacySelector.setVisibility(View.VISIBLE); previewButtons.setVisibility(View.VISIBLE); previewView.setVisibility(View.VISIBLE); @@ -3680,7 +3917,7 @@ private void onNavigateStart(int fromPage, int toPage) { // privacySelector.setStoryPeriod(outputEntry == null || !UserConfig.getInstance(currentAccount).isPremium() ? 86400 : outputEntry.period); captionEdit.setPeriod(outputEntry == null ? 86400 : outputEntry.period, false); - captionEdit.setPeriodVisible(!MessagesController.getInstance(currentAccount).premiumLocked && (outputEntry == null || !outputEntry.isEdit)); + captionEdit.setPeriodVisible(!MessagesController.getInstance(currentAccount).premiumFeaturesBlocked() && (outputEntry == null || !outputEntry.isEdit)); captionEdit.setHasRoundVideo(outputEntry != null && outputEntry.round != null); setReply(); } @@ -3689,8 +3926,11 @@ private void onNavigateStart(int fromPage, int toPage) { previewButtons.setShareText(outputEntry != null && outputEntry.isEdit ? LocaleController.getString("Done", R.string.Done) : LocaleController.getString("Next", R.string.Next)); // privacySelector.set(outputEntry, false); if (!previewAlreadySet) { - previewView.set(outputEntry); - previewView.setupAudio(outputEntry, false); + if (outputEntry != null && outputEntry.isRepostMessage) { + previewView.preset(outputEntry); + } else { + previewView.set(outputEntry); + } } previewAlreadySet = false; captionEdit.editText.getEditText().setOnPremiumMenuLockClickListener(MessagesController.getInstance(currentAccount).storyEntitiesAllowed() ? null : () -> { @@ -3720,6 +3960,7 @@ private void onNavigateStart(int fromPage, int toPage) { } else { captionEdit.clear(); } + previewButtons.setFiltersVisible(outputEntry == null || !outputEntry.isRepostMessage || outputEntry.isVideo); previewButtons.setShareEnabled(!videoError && !captionEdit.isCaptionOverLimit() && (!MessagesController.getInstance(currentAccount).getStoriesController().hasStoryLimit() || (outputEntry != null && outputEntry.isEdit))); muteButton.setImageResource(outputEntry != null && outputEntry.muted ? R.drawable.media_unmute : R.drawable.media_mute); previewView.setVisibility(View.VISIBLE); @@ -3728,6 +3969,8 @@ private void onNavigateStart(int fromPage, int toPage) { titleTextView.setTranslationX(0); if (outputEntry != null && outputEntry.isEdit) { titleTextView.setText(LocaleController.getString(R.string.RecorderEditStory)); + } else if (outputEntry != null && outputEntry.isRepostMessage) { + titleTextView.setText(LocaleController.getString(R.string.RecorderRepost)); } else if (outputEntry != null && outputEntry.isRepost) { SpannableStringBuilder title = new SpannableStringBuilder(); AvatarSpan span = new AvatarSpan(titleTextView, currentAccount, 32); @@ -3794,6 +4037,9 @@ private void onNavigateEnd(int fromPage, int toPage) { muteButton.setVisibility(View.GONE); playButton.setVisibility(View.GONE); downloadButton.setVisibility(View.GONE); + if (themeButton != null) { + themeButton.setVisibility(View.GONE); + } // privacySelector.setVisibility(View.GONE); previewView.setVisibility(View.GONE); timelineView.setVisibility(View.GONE); @@ -3809,6 +4055,8 @@ private void onNavigateEnd(int fromPage, int toPage) { if (outputEntry == null || !outputEntry.isRepost) { createPhotoPaintView(); hidePhotoPaintView(); + } + if (outputEntry == null || !outputEntry.isRepost && !outputEntry.isRepostMessage) { createFilterPhotoView(); } if (photoFilterEnhanceView != null) { @@ -3912,6 +4160,9 @@ public void switchToEditMode(int editMode, boolean animated) { animators.add(ObjectAnimator.ofFloat(muteButton, View.ALPHA, editMode == EDIT_MODE_NONE && isVideo ? 1 : 0)); animators.add(ObjectAnimator.ofFloat(playButton, View.ALPHA, editMode == EDIT_MODE_NONE && (isVideo || outputEntry != null && !TextUtils.isEmpty(outputEntry.audioPath)) ? 1 : 0)); animators.add(ObjectAnimator.ofFloat(downloadButton, View.ALPHA, editMode == EDIT_MODE_NONE ? 1 : 0)); + if (themeButton != null) { + animators.add(ObjectAnimator.ofFloat(themeButton, View.ALPHA, editMode == EDIT_MODE_NONE && (outputEntry != null && outputEntry.isRepostMessage) ? 1f : 0)); + } // animators.add(ObjectAnimator.ofFloat(privacySelector, View.ALPHA, editMode == EDIT_MODE_NONE ? 1 : 0)); // animators.add(ObjectAnimator.ofFloat(videoTimelineView, View.ALPHA, currentPage == PAGE_PREVIEW && isVideo && editMode == EDIT_MODE_NONE ? 1f : 0f)); @@ -4032,7 +4283,8 @@ private void createPhotoPaintView() { new MediaController.CropState(), null, blurManager, - resourcesProvider + resourcesProvider, + videoTextureHolder ) { @Override public void onEntityDraggedTop(boolean value) { @@ -4055,13 +4307,12 @@ public void onEntityDraggedBottom(boolean value) { @Override public void onEntityDragEnd(boolean delete) { + if (!isEntityDeletable()) { + delete = false; + } captionContainer.clearAnimation(); captionContainer.animate().alpha(1f).setDuration(180).setInterpolator(CubicBezierInterpolator.EASE_OUT).start(); - trash.onDragInfo(false, delete); - trash.clearAnimation(); - trash.animate().alpha(0f).withEndAction(() -> { - trash.setVisibility(View.GONE); - }).setDuration(180).setInterpolator(CubicBezierInterpolator.EASE_OUT).setStartDelay(delete ? 500 : 0).start(); + showTrash(false, delete); if (delete) { removeCurrentEntity(); } @@ -4074,11 +4325,22 @@ public void onEntityDragStart() { paintView.showReactionsLayout(false); captionContainer.clearAnimation(); captionContainer.animate().alpha(0f).setDuration(180).setInterpolator(CubicBezierInterpolator.EASE_OUT).start(); + showTrash(isEntityDeletable(), false); + } - trash.setVisibility(View.VISIBLE); - trash.setAlpha(0f); - trash.clearAnimation(); - trash.animate().alpha(1f).setDuration(180).setInterpolator(CubicBezierInterpolator.EASE_OUT).start(); + public void showTrash(boolean show, boolean delete) { + if (show) { + trash.setVisibility(View.VISIBLE); + trash.setAlpha(0f); + trash.clearAnimation(); + trash.animate().alpha(1f).setDuration(180).setInterpolator(CubicBezierInterpolator.EASE_OUT).start(); + } else { + trash.onDragInfo(false, delete); + trash.clearAnimation(); + trash.animate().alpha(0f).withEndAction(() -> { + trash.setVisibility(View.GONE); + }).setDuration(180).setInterpolator(CubicBezierInterpolator.EASE_OUT).setStartDelay(delete ? 500 : 0).start(); + } } private boolean multitouch; @@ -4087,20 +4349,13 @@ public void onEntityDragStart() { public void onEntityDragMultitouchStart() { multitouch = true; paintView.showReactionsLayout(false); - trash.onDragInfo(false, false); - trash.clearAnimation(); - trash.animate().alpha(0f).withEndAction(() -> { - trash.setVisibility(View.GONE); - }).setDuration(180).setInterpolator(CubicBezierInterpolator.EASE_OUT).start(); + showTrash(false, false); } @Override public void onEntityDragMultitouchEnd() { multitouch = false; - trash.setVisibility(View.VISIBLE); - trash.setAlpha(0f); - trash.clearAnimation(); - trash.animate().alpha(1f).setDuration(180).setInterpolator(CubicBezierInterpolator.EASE_OUT).start(); + showTrash(isEntityDeletable(), false); previewHighlight.show(false, false, null); } @@ -4142,7 +4397,6 @@ protected void onAudioSelect(MessageObject messageObject) { if (outputEntry != null && !isVideo) { boolean appear = !TextUtils.isEmpty(outputEntry.audioPath); playButton.drawable.setPause(!previewView.isPlaying(), false); - ((MarginLayoutParams) playButton.getLayoutParams()).rightMargin = dp(48 + (isVideo ? 48 : 0)); playButton.setVisibility(View.VISIBLE); playButton.animate().alpha(appear ? 1 : 0).withEndAction(() -> { if (!appear) { @@ -4221,6 +4475,51 @@ public void onDeleteRound() { } } + @Override + public void onSwitchSegmentedAnimation(PhotoView photoView) { + if (photoView == null) { + return; + } + ThanosEffect thanosEffect = getThanosEffect(); + if (thanosEffect == null) { + photoView.onSwitchSegmentedAnimationStarted(false); + return; + } + Bitmap bitmap = photoView.getSegmentedOutBitmap(); + if (bitmap == null) { + photoView.onSwitchSegmentedAnimationStarted(false); + return; + } + Matrix matrix = new Matrix(); + float w = photoView.getWidth(), h = photoView.getHeight(); + float tx = 0, ty = 0; + if (photoView.getRotation() != 0) { + final float bw = bitmap.getWidth(); + final float bh = bitmap.getHeight(); + final float r = (float) Math.sqrt((bw / 2f) * (bw / 2f) + (bh / 2f) * (bh / 2f)); + final float d = 2 * r; + Bitmap newBitmap = Bitmap.createBitmap((int) d, (int) d, Bitmap.Config.ARGB_8888); + Canvas canvas = new Canvas(newBitmap); + canvas.save(); + canvas.rotate(photoView.getRotation(), r, r); + canvas.drawBitmap(bitmap, (d - bw) / 2, (d - bh) / 2, null); + bitmap.recycle(); + bitmap = newBitmap; + + final float pd = 2 * (float) Math.sqrt((w / 2f) * (w / 2f) + (h / 2f) * (h / 2f)); + tx = -(pd - w) / 2; + ty = -(pd - h) / 2; + w = pd; + h = pd; + } + matrix.postScale(w, h); + matrix.postScale(photoView.getScaleX(), photoView.getScaleY(), w / 2f, h / 2f); + matrix.postTranslate(containerView.getX() + previewContainer.getX() + photoView.getX() + tx, containerView.getY() + previewContainer.getY() + photoView.getY() + ty); + thanosEffect.animate(matrix, bitmap, () -> { + photoView.onSwitchSegmentedAnimationStarted(true); + }, () -> {}); + } + @Override public void onSelectRound(RoundView roundView) { if (timelineView != null) { @@ -4343,6 +4642,12 @@ private void onSwitchEditModeStart(int fromMode, int toMode) { paintView.clearSelection(); } downloadButton.setVisibility(View.VISIBLE); + if (outputEntry != null && outputEntry.isRepostMessage) { + getThemeButton().setVisibility(View.VISIBLE); + updateThemeButtonDrawable(false); + } else if (themeButton != null) { + themeButton.setVisibility(View.GONE); + } titleTextView.setVisibility(View.VISIBLE); // privacySelector.setVisibility(View.VISIBLE); if (isVideo) { @@ -4388,6 +4693,9 @@ private void onSwitchEditModeEnd(int fromMode, int toMode) { muteButton.setVisibility(View.GONE); playButton.setVisibility(View.GONE); downloadButton.setVisibility(View.GONE); + if (themeButton != null) { + themeButton.setVisibility(View.GONE); + } // privacySelector.setVisibility(View.GONE); timelineView.setVisibility(View.GONE); titleTextView.setVisibility(View.GONE); @@ -4400,6 +4708,193 @@ private void onSwitchEditModeEnd(int fromMode, int toMode) { photoFilterEnhanceView.setAllowTouch(toMode == EDIT_MODE_FILTER || toMode == EDIT_MODE_NONE); } } + private void applyPaintInBackground(Runnable whenDone) { + final PaintView paintView = this.paintView; + final StoryEntry outputEntry = this.outputEntry; + if (paintView == null || outputEntry == null) { + return; + } + + outputEntry.clearPaint(); + final boolean hasChanges = paintView.hasChanges(); + final boolean hasBlur = paintView.hasBlur(); + final int resultWidth = outputEntry.resultWidth; + final int resultHeight = outputEntry.resultHeight; + Utilities.searchQueue.postRunnable(() -> { + + ArrayList mediaEntities = new ArrayList<>(); + + paintView.getBitmap(mediaEntities, resultWidth, resultHeight, false, false, false, false, outputEntry); + if (!outputEntry.isVideo) { + outputEntry.averageDuration = Utilities.clamp(paintView.getLcm(), 7500L, 5000L); + } + List masks = paintView.getMasks(); + final List stickers = masks != null ? new ArrayList<>(masks) : null; + final boolean isVideo = outputEntry.isVideo; + final boolean wouldBeVideo = outputEntry.wouldBeVideo(); + + mediaEntities.clear(); + Bitmap bitmap = paintView.getBitmap(mediaEntities, resultWidth, resultHeight, true, false, false, !isVideo, outputEntry); + if (mediaEntities.isEmpty()) { + mediaEntities = null; + } + + final File paintFile = FileLoader.getInstance(currentAccount).getPathToAttach(ImageLoader.scaleAndSaveImage(bitmap, Bitmap.CompressFormat.PNG, outputEntry.resultWidth, outputEntry.resultHeight, 87, false, 101, 101), true); + if (bitmap != null && !bitmap.isRecycled()) { + bitmap.recycle(); + } + bitmap = null; + + final File backgroundFile; + if (outputEntry.isRepostMessage && outputEntry.backgroundWallpaperPeerId != Long.MIN_VALUE) { + Drawable drawable = outputEntry.backgroundDrawable; + if (drawable == null) { + drawable = PreviewView.getBackgroundDrawable(null, currentAccount, outputEntry.backgroundWallpaperPeerId, isDark); + } + if (drawable != null) { + backgroundFile = StoryEntry.makeCacheFile(currentAccount, "webp"); + bitmap = Bitmap.createBitmap(resultWidth, resultHeight, Bitmap.Config.ARGB_8888); + StoryEntry.drawBackgroundDrawable(new Canvas(bitmap), drawable, bitmap.getWidth(), bitmap.getHeight()); + try { + bitmap.compress(Bitmap.CompressFormat.WEBP, 100, new FileOutputStream(backgroundFile)); + } catch (Exception e) { + FileLog.e(e); + } finally { + if (bitmap != null && !bitmap.isRecycled()) { + bitmap.recycle(); + } + bitmap = null; + } + } else { + backgroundFile = null; + } + } else { + backgroundFile = null; + } + File messageVideoMaskFile = null; + if (outputEntry.isRepostMessage && outputEntry.isVideo) { + int videoWidth = outputEntry.width; + int videoHeight = outputEntry.height; + MessageEntityView messageEntityView = paintView.findMessageView(); + ImageReceiver photoImage = null; + if (messageEntityView != null && messageEntityView.listView.getChildCount() == 1 && videoWidth > 0 && videoHeight > 0) { + View child = messageEntityView.listView.getChildAt(0); + if (child instanceof ChatMessageCell) { + ChatMessageCell cell = (ChatMessageCell) messageEntityView.listView.getChildAt(0); + photoImage = cell.getPhotoImage(); + } + } + if (photoImage != null && (int) photoImage.getImageWidth() > 0 && (int) photoImage.getImageHeight() > 0) { + float scale = Math.max(photoImage.getImageWidth() / videoWidth, photoImage.getImageHeight() / videoHeight); + final float S = 2f; + int w = (int) (videoWidth * scale / S), h = (int) (videoHeight * scale / S); + Bitmap maskBitmap = Bitmap.createBitmap(w, h, Bitmap.Config.ARGB_8888); + float[] radii = new float[8]; + for (int a = 0; a < photoImage.getRoundRadius().length; a++) { + radii[a * 2] = photoImage.getRoundRadius()[a]; + radii[a * 2 + 1] = photoImage.getRoundRadius()[a]; + } + Canvas canvas = new Canvas(maskBitmap); + Path path = new Path(); + canvas.scale(1f / S, 1f / S); + AndroidUtilities.rectTmp.set( + w * S / 2f - photoImage.getImageWidth() / 2f, + h * S / 2f - photoImage.getImageHeight() / 2f, + w * S / 2f + photoImage.getImageWidth() / 2f, + h * S / 2f + photoImage.getImageHeight() / 2f + ); + path.addRoundRect(AndroidUtilities.rectTmp, radii, Path.Direction.CW); + Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG); + paint.setColor(Color.WHITE); + canvas.drawPath(path, paint); + try { + messageVideoMaskFile = StoryEntry.makeCacheFile(currentAccount, "webp"); + maskBitmap.compress(Bitmap.CompressFormat.WEBP, 100, new FileOutputStream(messageVideoMaskFile)); + } catch (Exception e) { + FileLog.e(e); + messageVideoMaskFile = null; + } + maskBitmap.recycle(); + } + } + final File finalMessageVideoMaskFile = messageVideoMaskFile; + + final File paintEntitiesFile; + if (!wouldBeVideo) { + bitmap = paintView.getBitmap(new ArrayList<>(), resultWidth, resultHeight, false, true, false, false, outputEntry); + paintEntitiesFile = FileLoader.getInstance(currentAccount).getPathToAttach(ImageLoader.scaleAndSaveImage(bitmap, Bitmap.CompressFormat.PNG, resultWidth, resultHeight, 87, false, 101, 101), true); + if (bitmap != null && !bitmap.isRecycled()) { + bitmap.recycle(); + } + bitmap = null; + } else { + paintEntitiesFile = null; + } + + final File paintBlurFile; + if (hasBlur) { + bitmap = paintView.getBlurBitmap(); + paintBlurFile = FileLoader.getInstance(currentAccount).getPathToAttach(ImageLoader.scaleAndSaveImage(bitmap, Bitmap.CompressFormat.PNG, resultWidth, resultHeight, 87, false, 101, 101), true); + if (bitmap != null && !bitmap.isRecycled()) { + bitmap.recycle(); + } + bitmap = null; + } else { + paintBlurFile = null; + } + + final ArrayList finalMediaEntities = mediaEntities; + AndroidUtilities.runOnUIThread(() -> { + try { + if (outputEntry.paintFile != null) { + outputEntry.paintFile.delete(); + } + } catch (Exception ignore) {} + try { + if (outputEntry.paintEntitiesFile != null) { + outputEntry.paintEntitiesFile.delete(); + } + } catch (Exception ignore) {} + try { + if (outputEntry.paintBlurFile != null) { + outputEntry.paintBlurFile.delete(); + } + } catch (Exception ignore) {} + outputEntry.paintFile = null; + outputEntry.paintEntitiesFile = null; + outputEntry.paintBlurFile = null; + if (outputEntry.backgroundFile != null) { + try { + outputEntry.backgroundFile.delete(); + } catch (Exception e) { + FileLog.e(e); + } + outputEntry.backgroundFile = null; + } + if (outputEntry.messageVideoMaskFile != null) { + try { + outputEntry.messageVideoMaskFile.delete(); + } catch (Exception e) { + FileLog.e(e); + } + outputEntry.messageVideoMaskFile = null; + } + + outputEntry.editedMedia |= hasChanges; + outputEntry.mediaEntities = finalMediaEntities; + outputEntry.paintFile = paintFile; + outputEntry.backgroundFile = backgroundFile; + outputEntry.paintEntitiesFile = paintEntitiesFile; + outputEntry.messageVideoMaskFile = finalMessageVideoMaskFile; + outputEntry.paintBlurFile = paintBlurFile; + outputEntry.stickers = stickers; + + if (whenDone != null) { + whenDone.run(); + } + }); + }); + } private void applyPaint() { if (paintView == null || outputEntry == null) { @@ -4414,7 +4909,7 @@ private void applyPaint() { } else { outputEntry.mediaEntities.clear(); } - paintView.getBitmap(outputEntry.mediaEntities, outputEntry.resultWidth, outputEntry.resultHeight, false, false, false, outputEntry); + paintView.getBitmap(outputEntry.mediaEntities, outputEntry.resultWidth, outputEntry.resultHeight, false, false, false, false, outputEntry); if (!outputEntry.isVideo) { outputEntry.averageDuration = Utilities.clamp(paintView.getLcm(), 7500L, 5000L); } @@ -4424,7 +4919,7 @@ private void applyPaint() { final boolean wouldBeVideo = outputEntry.wouldBeVideo(); outputEntry.mediaEntities = new ArrayList<>(); - Bitmap bitmap = paintView.getBitmap(outputEntry.mediaEntities, outputEntry.resultWidth, outputEntry.resultHeight, true, false, !isVideo, outputEntry); + Bitmap bitmap = paintView.getBitmap(outputEntry.mediaEntities, outputEntry.resultWidth, outputEntry.resultHeight, true, false, false, !isVideo, outputEntry); if (outputEntry.mediaEntities.isEmpty()) { outputEntry.mediaEntities = null; } @@ -4454,8 +4949,94 @@ private void applyPaint() { } bitmap = null; + if (outputEntry.isRepostMessage) { + if (outputEntry.backgroundFile != null) { + try { + outputEntry.backgroundFile.delete(); + } catch (Exception e) { + FileLog.e(e); + } + outputEntry.backgroundFile = null; + } + if (outputEntry.backgroundWallpaperPeerId != Long.MIN_VALUE) { + Drawable drawable = outputEntry.backgroundDrawable; + if (drawable == null) { + drawable = PreviewView.getBackgroundDrawable(null, currentAccount, outputEntry.backgroundWallpaperPeerId, isDark); + } + if (drawable != null) { + outputEntry.backgroundFile = StoryEntry.makeCacheFile(currentAccount, "webp"); + bitmap = Bitmap.createBitmap(outputEntry.resultWidth, outputEntry.resultHeight, Bitmap.Config.ARGB_8888); + StoryEntry.drawBackgroundDrawable(new Canvas(bitmap), drawable, bitmap.getWidth(), bitmap.getHeight()); + try { + bitmap.compress(Bitmap.CompressFormat.WEBP, 100, new FileOutputStream(outputEntry.backgroundFile)); + } catch (Exception e) { + FileLog.e(e); + } finally { + if (bitmap != null && !bitmap.isRecycled()) { + bitmap.recycle(); + } + bitmap = null; + } + } + } + } + if (outputEntry.isRepostMessage) { + if (outputEntry.messageVideoMaskFile != null) { + try { + outputEntry.messageVideoMaskFile.delete(); + } catch (Exception e) { + FileLog.e(e); + } + outputEntry.messageVideoMaskFile = null; + } + if (outputEntry.isRepostMessage && outputEntry.isVideo) { + int videoWidth = outputEntry.width; + int videoHeight = outputEntry.height; + MessageEntityView messageEntityView = paintView.findMessageView(); + if (messageEntityView != null && messageEntityView.listView.getChildCount() == 1 && videoWidth > 0 && videoHeight > 0) { + View child = messageEntityView.listView.getChildAt(0); + if (child instanceof ChatMessageCell) { + ChatMessageCell cell = (ChatMessageCell) messageEntityView.listView.getChildAt(0); + ImageReceiver photoImage = cell.getPhotoImage(); + if (photoImage != null && (int) photoImage.getImageWidth() > 0 && (int) photoImage.getImageHeight() > 0) { + float scale = Math.max(photoImage.getImageWidth() / videoWidth, photoImage.getImageHeight() / videoHeight); + final float S = 2f; + int w = (int) (videoWidth * scale / S), h = (int) (videoHeight * scale / S); + Bitmap maskBitmap = Bitmap.createBitmap(w, h, Bitmap.Config.ARGB_8888); + float[] radii = new float[8]; + for (int a = 0; a < photoImage.getRoundRadius().length; a++) { + radii[a * 2] = photoImage.getRoundRadius()[a]; + radii[a * 2 + 1] = photoImage.getRoundRadius()[a]; + } + Canvas canvas = new Canvas(maskBitmap); + Path path = new Path(); + canvas.scale(1f / S, 1f / S); + AndroidUtilities.rectTmp.set( + w * S / 2f - photoImage.getImageWidth() / 2f, + h * S / 2f - photoImage.getImageHeight() / 2f, + w * S / 2f + photoImage.getImageWidth() / 2f, + h * S / 2f + photoImage.getImageHeight() / 2f + ); + path.addRoundRect(AndroidUtilities.rectTmp, radii, Path.Direction.CW); + Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG); + paint.setColor(Color.WHITE); + canvas.drawPath(path, paint); + try { + outputEntry.messageVideoMaskFile = StoryEntry.makeCacheFile(currentAccount, "webp"); + maskBitmap.compress(Bitmap.CompressFormat.WEBP, 100, new FileOutputStream(outputEntry.messageVideoMaskFile)); + } catch (Exception e) { + FileLog.e(e); + outputEntry.messageVideoMaskFile = null; + } + maskBitmap.recycle(); + } + } + } + } + } + if (!wouldBeVideo) { - bitmap = paintView.getBitmap(new ArrayList<>(), outputEntry.resultWidth, outputEntry.resultHeight, false, true, false, outputEntry); + bitmap = paintView.getBitmap(new ArrayList<>(), outputEntry.resultWidth, outputEntry.resultHeight, false, true, false, false, outputEntry); outputEntry.paintEntitiesFile = FileLoader.getInstance(currentAccount).getPathToAttach(ImageLoader.scaleAndSaveImage(bitmap, Bitmap.CompressFormat.PNG, outputEntry.resultWidth, outputEntry.resultHeight, 87, false, 101, 101), true); if (bitmap != null && !bitmap.isRecycled()) { bitmap.recycle(); @@ -4473,6 +5054,42 @@ private void applyPaint() { } } + // separated to run on main thread (chatmessagecell uses global paints and resources) + private void applyPaintMessage() { + if (paintView == null || outputEntry == null) { + return; + } + + if (outputEntry.isRepostMessage) { + if (outputEntry.messageFile != null) { + try { + outputEntry.messageFile.delete(); + } catch (Exception e) { + FileLog.e(e); + } + outputEntry.messageFile = null; + } + outputEntry.messageFile = StoryEntry.makeCacheFile(currentAccount, "webp"); + Bitmap bitmap = paintView.getBitmap(outputEntry.mediaEntities, outputEntry.resultWidth, outputEntry.resultHeight, false, false, true, !isVideo, outputEntry); + try { + bitmap.compress(Bitmap.CompressFormat.WEBP, 100, new FileOutputStream(outputEntry.messageFile)); + } catch (Exception e) { + FileLog.e(e); + try { + outputEntry.messageFile.delete(); + } catch (Exception e2) { + FileLog.e(e2); + } + outputEntry.messageFile = null; + } finally { + if (bitmap != null && !bitmap.isRecycled()) { + bitmap.recycle(); + } + bitmap = null; + } + } + } + private void applyFilter(Runnable whenDone) { if (photoFilterView == null || outputEntry == null) { if (whenDone != null) { @@ -4809,6 +5426,7 @@ private void showDismissEntry() { showSavedDraftHint = !outputEntry.isDraft; applyFilter(null); applyPaint(); + applyPaintMessage(); destroyPhotoFilterView(); StoryEntry storyEntry = outputEntry; storyEntry.destroy(true); @@ -4825,11 +5443,11 @@ private void showDismissEntry() { }); } builder.setPositiveButton(outputEntry != null && outputEntry.isDraft && !outputEntry.isEdit ? LocaleController.getString("StoryDeleteDraft") : LocaleController.getString("Discard", R.string.Discard), (dialogInterface, i) -> { - if (outputEntry != null && !(outputEntry.isEdit || outputEntry.isRepost) && outputEntry.isDraft) { + if (outputEntry != null && !(outputEntry.isEdit || outputEntry.isRepost && !outputEntry.isRepostMessage) && outputEntry.isDraft) { MessagesController.getInstance(currentAccount).getStoriesController().getDraftsController().delete(outputEntry); outputEntry = null; } - if (outputEntry != null && (outputEntry.isEdit || outputEntry.isRepost)) { + if (outputEntry != null && (outputEntry.isEdit || outputEntry.isRepost && !outputEntry.isRepostMessage)) { close(true); } else { navigateTo(PAGE_CAMERA, true); @@ -5332,4 +5950,195 @@ public void draw(@NonNull Canvas canvas, CharSequence text, int start, int end, }, 0, 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); return cameraStr; } + + public ThanosEffect getThanosEffect() { + if (!ThanosEffect.supports()) { + return null; + } + if (thanosEffect == null) { + windowView.addView(thanosEffect = new ThanosEffect(getContext(), () -> { + ThanosEffect thisThanosEffect = thanosEffect; + if (thisThanosEffect != null) { + thanosEffect = null; + windowView.removeView(thisThanosEffect); + } + })); + } + return thanosEffect; + } + + private View changeDayNightView; + private float changeDayNightViewProgress; + private ValueAnimator changeDayNightViewAnimator; + + public ImageView getThemeButton() { + if (themeButton == null) { + themeButtonDrawable = new RLottieDrawable(R.raw.sun_outline, "" + R.raw.sun_outline, dp(28), dp(28), true, null); + themeButtonDrawable.setPlayInDirectionOfCustomEndFrame(true); + if (!(outputEntry != null && outputEntry.isDark)) { + themeButtonDrawable.setCustomEndFrame(0); + themeButtonDrawable.setCurrentFrame(0); + } else { + themeButtonDrawable.setCurrentFrame(35); + themeButtonDrawable.setCustomEndFrame(36); + } + themeButtonDrawable.beginApplyLayerColors(); + int color = Theme.getColor(Theme.key_chats_menuName, resourcesProvider); + themeButtonDrawable.setLayerColor("Sunny.**", color); + themeButtonDrawable.setLayerColor("Path 6.**", color); + themeButtonDrawable.setLayerColor("Path.**", color); + themeButtonDrawable.setLayerColor("Path 5.**", color); + themeButtonDrawable.commitApplyLayerColors(); + themeButton = new ImageView(getContext()); + themeButton.setScaleType(ImageView.ScaleType.CENTER); + themeButton.setColorFilter(new PorterDuffColorFilter(Color.WHITE, PorterDuff.Mode.MULTIPLY)); + themeButton.setBackground(Theme.createSelectorDrawable(0x20ffffff)); + themeButton.setOnClickListener(e -> { + toggleTheme(); + }); +// themeButton.setOnLongClickListener(e -> { +// openThemeSheet(); +// return true; +// }); + themeButton.setVisibility(View.GONE); + themeButton.setImageDrawable(themeButtonDrawable); + themeButton.setAlpha(0f); + actionBarButtons.addView(themeButton, 0, LayoutHelper.createLinear(46, 56, Gravity.TOP | Gravity.RIGHT)); + } + return themeButton; + } + + public void updateThemeButtonDrawable(boolean animated) { + if (themeButtonDrawable != null) { + if (animated) { + themeButtonDrawable.setCustomEndFrame(outputEntry != null && outputEntry.isDark ? themeButtonDrawable.getFramesCount() : 0); + if (themeButtonDrawable != null) { + themeButtonDrawable.start(); + } + } else { + int frame = outputEntry != null && outputEntry.isDark ? themeButtonDrawable.getFramesCount() - 1 : 0; + themeButtonDrawable.setCurrentFrame(frame, false, true); + themeButtonDrawable.setCustomEndFrame(frame); + if (themeButton != null) { + themeButton.invalidate(); + } + } + } + } + + public void toggleTheme() { + if (outputEntry == null || changeDayNightView != null || themeButton == null || changeDayNightViewAnimator != null && changeDayNightViewAnimator.isRunning()) { + return; + } + final boolean isDark = outputEntry.isDark; + + Bitmap bitmap = Bitmap.createBitmap(windowView.getWidth(), windowView.getHeight(), Bitmap.Config.ARGB_8888); + Canvas bitmapCanvas = new Canvas(bitmap); + themeButton.setAlpha(0f); + if (previewView != null) { + previewView.drawForThemeToggle = true; + } + if (paintView != null) { + paintView.drawForThemeToggle = true; + } + windowView.draw(bitmapCanvas); + if (previewView != null) { + previewView.drawForThemeToggle = false; + } + if (paintView != null) { + paintView.drawForThemeToggle = false; + } + themeButton.setAlpha(1f); + + Paint xRefPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + xRefPaint.setColor(0xff000000); + xRefPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.CLEAR)); + + Paint bitmapPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + bitmapPaint.setFilterBitmap(true); + int[] position = new int[2]; + themeButton.getLocationInWindow(position); + float x = position[0]; + float y = position[1]; + float cx = x + themeButton.getMeasuredWidth() / 2f; + float cy = y + themeButton.getMeasuredHeight() / 2f; + + float r = Math.max(bitmap.getHeight(), bitmap.getWidth()) + AndroidUtilities.navigationBarHeight; + + Shader bitmapShader = new BitmapShader(bitmap, Shader.TileMode.CLAMP, Shader.TileMode.CLAMP); + bitmapPaint.setShader(bitmapShader); + changeDayNightView = new View(getContext()) { + @Override + protected void onDraw(Canvas canvas) { + super.onDraw(canvas); + if (isDark) { + if (changeDayNightViewProgress > 0f) { + bitmapCanvas.drawCircle(cx, cy, r * changeDayNightViewProgress, xRefPaint); + } + canvas.drawBitmap(bitmap, 0, 0, bitmapPaint); + } else { + canvas.drawCircle(cx, cy, r * (1f - changeDayNightViewProgress), bitmapPaint); + } + canvas.save(); + canvas.translate(x, y); + themeButton.draw(canvas); + canvas.restore(); + } + }; + changeDayNightView.setOnTouchListener((v, event) -> true); + changeDayNightViewProgress = 0f; + changeDayNightViewAnimator = ValueAnimator.ofFloat(0, 1f); + changeDayNightViewAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() { + boolean changedNavigationBarColor = false; + + @Override + public void onAnimationUpdate(ValueAnimator valueAnimator) { + changeDayNightViewProgress = (float) valueAnimator.getAnimatedValue(); + if (changeDayNightView != null) { + changeDayNightView.invalidate(); + } + if (!changedNavigationBarColor && changeDayNightViewProgress > .5f) { + changedNavigationBarColor = true; + } + } + }); + changeDayNightViewAnimator.addListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + if (changeDayNightView != null) { + if (changeDayNightView.getParent() != null) { + ((ViewGroup) changeDayNightView.getParent()).removeView(changeDayNightView); + } + changeDayNightView = null; + } + changeDayNightViewAnimator = null; + super.onAnimationEnd(animation); + } + }); + changeDayNightViewAnimator.setStartDelay(80); + changeDayNightViewAnimator.setDuration(isDark ? 320 : 450); + changeDayNightViewAnimator.setInterpolator(isDark ? CubicBezierInterpolator.EASE_IN : CubicBezierInterpolator.EASE_OUT_QUINT); + changeDayNightViewAnimator.start(); + + windowView.addView(changeDayNightView, new ViewGroup.LayoutParams(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT)); + + AndroidUtilities.runOnUIThread(() -> { + if (outputEntry == null) { + return; + } + outputEntry.isDark = !outputEntry.isDark; + if (previewView != null) { + previewView.setupWallpaper(outputEntry, false); + } + if (paintView != null && paintView.entitiesView != null) { + for (int i = 0; i < paintView.entitiesView.getChildCount(); ++i) { + View child = paintView.entitiesView.getChildAt(i); + if (child instanceof MessageEntityView) { + ((MessageEntityView) child).setupTheme(outputEntry); + } + } + } + updateThemeButtonDrawable(true); + }); + } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/StoryThemeSheet.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/StoryThemeSheet.java new file mode 100644 index 00000000000..4c2a95c0543 --- /dev/null +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/StoryThemeSheet.java @@ -0,0 +1,154 @@ +package org.telegram.ui.Stories.recorder; + +import static org.telegram.messenger.AndroidUtilities.dp; + +import android.content.Context; +import android.text.TextUtils; +import android.util.TypedValue; +import android.view.Gravity; +import android.widget.FrameLayout; +import android.widget.ImageView; +import android.widget.TextView; + +import org.telegram.messenger.AndroidUtilities; +import org.telegram.messenger.ChatThemeController; +import org.telegram.messenger.LocaleController; +import org.telegram.messenger.MessagesController; +import org.telegram.messenger.R; +import org.telegram.tgnet.TLRPC; +import org.telegram.ui.ActionBar.BackDrawable; +import org.telegram.ui.ActionBar.Theme; +import org.telegram.ui.ChannelColorActivity; +import org.telegram.ui.Components.CubicBezierInterpolator; +import org.telegram.ui.Components.LayoutHelper; + +public class StoryThemeSheet extends FrameLayout { + + private final ImageView backButtonView; + private final BackDrawable backButtonDrawable; + + private final TextView titleView; + + private final ChannelColorActivity.ThemeChooser themeView; + + private final Theme.ResourcesProvider resourcesProvider; + private final Runnable whenDie; + + public StoryThemeSheet(Context context, int currentAccount, Theme.ResourcesProvider resourcesProvider, Runnable whenDie) { + super(context); + this.resourcesProvider = resourcesProvider; + this.whenDie = whenDie; + + setBackground(Theme.createRoundRectDrawable(dp(14),0, Theme.getColor(Theme.key_dialogBackground, resourcesProvider))); + + backButtonView = new ImageView(getContext()); + int padding = dp(10); + backButtonView.setPadding(padding, padding, padding, padding); + backButtonView.setBackground(Theme.createSelectorDrawable(Theme.getColor(Theme.key_listSelector, resourcesProvider), Theme.RIPPLE_MASK_CIRCLE_20DP)); + backButtonDrawable = new BackDrawable(true); + backButtonView.setImageDrawable(backButtonDrawable); + backButtonView.setOnClickListener(v -> { + dismiss(); + }); + addView(backButtonView, LayoutHelper.createFrame(44, 44, Gravity.TOP | Gravity.LEFT, 7, 8, 0, 0)); + + titleView = new TextView(context); + titleView.setTextColor(Theme.getColor(Theme.key_dialogTextBlack, resourcesProvider)); + titleView.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); + titleView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 20); + titleView.setText(LocaleController.getString(R.string.StorySetWallpaper)); + addView(titleView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.TOP | Gravity.LEFT, 54, 16, 24, 0)); + + themeView = new ChannelColorActivity.ThemeChooser(context, false, currentAccount, resourcesProvider) { + @Override + public boolean isDark() { + return currentEntry != null ? currentEntry.isDark : super.isDark(); + } + }; + themeView.setOnEmoticonSelected(emoticon -> { + if (currentEntry != null && !TextUtils.equals(currentEntry.backgroundWallpaperEmoticon, emoticon)) { + currentEntry.backgroundWallpaperEmoticon = emoticon; + updateWallpaper(); + } + }); + addView(themeView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.TOP, 0, 56, 0, 0)); + + } + + public void updateColors() { + themeView.updateColors(); + } + + protected void updateWallpaper() { + + } + + private boolean openWhenMeasured; + private StoryEntry currentEntry; + + public void open(StoryEntry entry) { + if (currentEntry != entry) { + currentEntry = entry; + themeView.updateColors(); + } + currentEntry = entry; + if (getMeasuredHeight() == 0) { + openWhenMeasured = true; + return; + } + + if (entry != null) { + TLRPC.WallPaper wallpaper = null; + if (entry.backgroundWallpaperPeerId != Long.MIN_VALUE) { + if (entry.backgroundWallpaperPeerId < 0) { + TLRPC.ChatFull chatFull = MessagesController.getInstance(entry.currentAccount).getChatFull(-entry.backgroundWallpaperPeerId); + if (chatFull != null) { + wallpaper = chatFull.wallpaper; + } + } else { + TLRPC.UserFull userFull = MessagesController.getInstance(entry.currentAccount).getUserFull(entry.backgroundWallpaperPeerId); + if (userFull != null) { + wallpaper = userFull.wallpaper; + } + } + } + themeView.setGalleryWallpaper(wallpaper); + if (entry.backgroundWallpaperEmoticon != null) { + themeView.setSelectedEmoticon(entry.backgroundWallpaperEmoticon, false); + } else if (!TextUtils.isEmpty(ChatThemeController.getWallpaperEmoticon(wallpaper))) { + themeView.setSelectedEmoticon(ChatThemeController.getWallpaperEmoticon(wallpaper), false); + } else { + themeView.setSelectedEmoticon(null, false); + } + } else { + themeView.setGalleryWallpaper(null); + themeView.setSelectedEmoticon(null, false); + } + + setTranslationY(getMeasuredHeight()); + animate().translationY(0).setInterpolator(CubicBezierInterpolator.EASE_OUT_QUINT).start(); + isOpen = true; + } + + public boolean isOpen; + + public void dismiss() { + animate().translationY(getMeasuredHeight()).withEndAction(() -> { + if (whenDie != null) { + whenDie.run(); + } else { + setVisibility(GONE); + } + }).setInterpolator(CubicBezierInterpolator.EASE_OUT_QUINT).start(); + isOpen = false; + } + + @Override + protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { + super.onMeasure(widthMeasureSpec, MeasureSpec.makeMeasureSpec(dp(235) + AndroidUtilities.navigationBarHeight, MeasureSpec.EXACTLY)); + if (openWhenMeasured) { + openWhenMeasured = false; + open(currentEntry); + } + } +} diff --git a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/TimelineView.java b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/TimelineView.java index 8c138ccce6a..25a1ff8fe69 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/TimelineView.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/Stories/recorder/TimelineView.java @@ -89,6 +89,7 @@ interface TimelineDelegate { private long scroll; private boolean hasVideo; + private boolean isMainVideoRound; private String videoPath; private long videoDuration; private float videoLeft; @@ -313,7 +314,7 @@ public void setDelegate(TimelineDelegate delegate) { this.delegate = delegate; } - public void setVideo(String videoPath, long videoDuration, float videoVolume) { + public void setVideo(boolean isRound, String videoPath, long videoDuration, float videoVolume) { if (TextUtils.equals(this.videoPath, videoPath)) { return; } @@ -321,6 +322,7 @@ public void setVideo(String videoPath, long videoDuration, float videoVolume) { thumbs.destroy(); thumbs = null; } + isMainVideoRound = isRound; if (videoPath != null) { scroll = 0; this.videoPath = videoPath; @@ -402,7 +404,7 @@ private void setupVideoThumbs() { if (getMeasuredWidth() <= 0 || this.thumbs != null) { return; } - this.thumbs = new VideoThumbsLoader(videoPath, w - px - px, dp(38), videoDuration > 2 ? videoDuration : null); + this.thumbs = new VideoThumbsLoader(isMainVideoRound, videoPath, w - px - px, dp(38), videoDuration > 2 ? videoDuration : null); if (this.thumbs.getDuration() > 0) { videoDuration = this.thumbs.getDuration(); } @@ -413,7 +415,7 @@ private void setupRoundThumbs() { if (getMeasuredWidth() <= 0 || this.roundThumbs != null || hasVideo && videoDuration < 1) { return; } - this.roundThumbs = new VideoThumbsLoader(roundPath, w - px - px, dp(38), roundDuration > 2 ? roundDuration : null, hasVideo ? videoDuration : MAX_SCROLL_DURATION); + this.roundThumbs = new VideoThumbsLoader(false, roundPath, w - px - px, dp(38), roundDuration > 2 ? roundDuration : null, hasVideo ? videoDuration : MAX_SCROLL_DURATION); if (this.roundThumbs.getDuration() > 0) { roundDuration = this.roundThumbs.getDuration(); } @@ -1403,7 +1405,7 @@ protected void dispatchDraw(Canvas canvas) { final int y = (int) videoBounds.top; boolean allLoaded = thumbs.frames.size() >= toFrame; - boolean fullyCovered = allLoaded; + boolean fullyCovered = allLoaded && !isMainVideoRound; if (fullyCovered) { for (int i = fromFrame; i < Math.min(thumbs.frames.size(), toFrame); ++i) { VideoThumbsLoader.BitmapFrame frame = thumbs.frames.get(i); @@ -1856,14 +1858,16 @@ private class VideoThumbsLoader { private final int frameWidth; private final int frameHeight; + private final boolean isRound; private boolean destroyed; - public VideoThumbsLoader(String path, int uiWidth, int uiHeight, Long overrideDuration) { - this(path, uiWidth, uiHeight, overrideDuration, MAX_SCROLL_DURATION); + public VideoThumbsLoader(boolean isRound, String path, int uiWidth, int uiHeight, Long overrideDuration) { + this(isRound, path, uiWidth, uiHeight, overrideDuration, MAX_SCROLL_DURATION); } - public VideoThumbsLoader(String path, int uiWidth, int uiHeight, Long overrideDuration, long maxDuration) { + public VideoThumbsLoader(boolean isRound, String path, int uiWidth, int uiHeight, Long overrideDuration, long maxDuration) { + this.isRound = isRound; metadataRetriever = new MediaMetadataRetriever(); long duration = MAX_SCROLL_DURATION; int width = 0; @@ -1952,6 +1956,7 @@ public void load() { private final Paint bitmapPaint = new Paint(Paint.ANTI_ALIAS_FLAG | Paint.FILTER_BITMAP_FLAG); + private Path clipPath; private void retrieveFrame() { if (metadataRetriever == null) { return; @@ -1971,6 +1976,14 @@ private void retrieveFrame() { (int) ((scaledBitmap.getWidth() + bitmap.getWidth() * scale) / 2f), (int) ((scaledBitmap.getHeight() + bitmap.getHeight() * scale) / 2f) ); + if (isRound) { + if (clipPath == null) { + clipPath = new Path(); + } + clipPath.rewind(); + clipPath.addCircle(frameWidth / 2f, frameHeight / 2f, Math.min(frameWidth, frameHeight) / 2f, Path.Direction.CW); + canvas.clipPath(clipPath); + } canvas.drawBitmap(bitmap, src, dest, bitmapPaint); bitmap.recycle(); bitmap = scaledBitmap; diff --git a/TMessagesProj/src/main/java/org/telegram/ui/ThemeActivity.java b/TMessagesProj/src/main/java/org/telegram/ui/ThemeActivity.java index 1f9e60c881d..a71957aa0da 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/ThemeActivity.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/ThemeActivity.java @@ -2192,7 +2192,7 @@ public boolean onInterceptTouchEvent(MotionEvent e) { view = new AppIconsSelectorCell(mContext, ThemeActivity.this, currentAccount); break; case TYPE_CHOOSE_COLOR: - view = new PeerColorActivity.ChangeNameColorCell(currentAccount, false, mContext, getResourceProvider()); + view = new PeerColorActivity.ChangeNameColorCell(currentAccount, 0, mContext, getResourceProvider()); break; } return new RecyclerListView.Holder(view); diff --git a/TMessagesProj/src/main/java/org/telegram/ui/ThemePreviewActivity.java b/TMessagesProj/src/main/java/org/telegram/ui/ThemePreviewActivity.java index ede8d96a5ed..34a7250267e 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/ThemePreviewActivity.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/ThemePreviewActivity.java @@ -19,7 +19,6 @@ import android.animation.ValueAnimator; import android.annotation.SuppressLint; import android.app.Activity; -import android.app.Dialog; import android.content.Context; import android.content.DialogInterface; import android.content.SharedPreferences; @@ -49,13 +48,12 @@ import android.graphics.drawable.Drawable; import android.graphics.drawable.GradientDrawable; import android.os.Build; +import android.os.Bundle; import android.os.SystemClock; -import android.text.SpannableString; import android.text.SpannableStringBuilder; import android.text.Spanned; import android.text.TextPaint; import android.text.TextUtils; -import android.util.Log; import android.util.SparseIntArray; import android.util.StateSet; import android.util.TypedValue; @@ -84,6 +82,7 @@ import org.telegram.messenger.AndroidUtilities; import org.telegram.messenger.ApplicationLoader; import org.telegram.messenger.BuildVars; +import org.telegram.messenger.ChatObject; import org.telegram.messenger.ChatThemeController; import org.telegram.messenger.DownloadController; import org.telegram.messenger.Emoji; @@ -110,17 +109,16 @@ import org.telegram.messenger.VideoEditedInfo; import org.telegram.tgnet.ConnectionsManager; import org.telegram.tgnet.TLRPC; +import org.telegram.tgnet.tl.TL_stories; import org.telegram.ui.ActionBar.ActionBar; import org.telegram.ui.ActionBar.ActionBarMenu; import org.telegram.ui.ActionBar.ActionBarMenuItem; import org.telegram.ui.ActionBar.AlertDialog; import org.telegram.ui.ActionBar.BackDrawable; import org.telegram.ui.ActionBar.BaseFragment; -import org.telegram.ui.ActionBar.INavigationLayout; import org.telegram.ui.ActionBar.MenuDrawable; import org.telegram.ui.ActionBar.Theme; import org.telegram.ui.ActionBar.ThemeDescription; -import org.telegram.ui.Cells.BrightnessControlCell; import org.telegram.ui.Cells.ChatActionCell; import org.telegram.ui.Cells.ChatMessageCell; import org.telegram.ui.Cells.DialogCell; @@ -128,21 +126,22 @@ import org.telegram.ui.Cells.LoadingCell; import org.telegram.ui.Cells.PatternCell; import org.telegram.ui.Components.AlertsCreator; +import org.telegram.ui.Components.AnimatedFloat; import org.telegram.ui.Components.BackgroundGradientDrawable; import org.telegram.ui.Components.BackupImageView; -import org.telegram.ui.Components.ButtonBounce; import org.telegram.ui.Components.CircularProgressDrawable; import org.telegram.ui.Components.ColorPicker; import org.telegram.ui.Components.ColoredImageSpan; import org.telegram.ui.Components.CombinedDrawable; import org.telegram.ui.Components.CubicBezierInterpolator; +import org.telegram.ui.Components.Easings; import org.telegram.ui.Components.GestureDetector2; import org.telegram.ui.Components.HintView; import org.telegram.ui.Components.LayoutHelper; import org.telegram.ui.Components.MotionBackgroundDrawable; +import org.telegram.ui.Components.Premium.LimitReachedBottomSheet; import org.telegram.ui.Components.Premium.PremiumFeatureBottomSheet; import org.telegram.ui.Components.RLottieDrawable; -import org.telegram.ui.Components.RadialProgressView; import org.telegram.ui.Components.RecyclerListView; import org.telegram.ui.Components.ScaleStateListAnimator; import org.telegram.ui.Components.SeekBarView; @@ -151,6 +150,7 @@ import org.telegram.ui.Components.UndoView; import org.telegram.ui.Components.WallpaperCheckBoxView; import org.telegram.ui.Components.WallpaperParallaxEffect; +import org.telegram.ui.Stories.recorder.PreviewView; import org.telegram.ui.Stories.recorder.SliderView; import java.io.File; @@ -163,7 +163,15 @@ public class ThemePreviewActivity extends BaseFragment implements DownloadController.FileDownloadProgressListener, NotificationCenter.NotificationCenterDelegate { - public final ThemeDelegate themeDelegate = new ThemeDelegate(); + public final ThemeDelegate themeDelegate = new ThemeDelegate() { + @Override + public boolean isDark() { + if (onSwitchDayNightDelegate != null) { + return onSwitchDayNightDelegate.isDark(); + } + return super.isDark(); + } + }; @Override public void setResourceProvider(Theme.ResourcesProvider resourceProvider) { @@ -302,6 +310,7 @@ public Theme.ResourcesProvider getResourceProvider() { private float dimAmount = 0f; private float progressToDarkTheme; DayNightSwitchDelegate onSwitchDayNightDelegate; + private ColoredImageSpan lockSpan; private AnimatorSet patternViewAnimation; @@ -343,8 +352,7 @@ public Theme.ResourcesProvider getResourceProvider() { private boolean shouldShowBrightnessControll; private RLottieDrawable sunDrawable; private ActionBarMenuItem dayNightItem; - private float changeDayNightViewProgress; - private ValueAnimator changeDayNightViewAnimator; + private ValueAnimator changeDayNightViewAnimator2; private FrameLayout dimmingSliderContainer; private SliderView dimmingSlider; @@ -399,6 +407,21 @@ public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float ve } }); + private boolean checkingBoostsLevel = false, checkedBoostsLevel = false; + public TL_stories.TL_premium_boostsStatus boostsStatus; + private void checkBoostsLevel() { + if (dialogId >= 0 || checkingBoostsLevel || checkedBoostsLevel || boostsStatus != null) { + return; + } + checkingBoostsLevel = true; + getMessagesController().getBoostsController().getBoostsStats(dialogId, boostsStatus -> { + this.boostsStatus = boostsStatus; + checkedBoostsLevel = true; + updateApplyButton1(true); + checkingBoostsLevel = false; + }); + } + float maxScrollOffset; float currentScrollOffset; float defaultScrollOffset; @@ -465,6 +488,11 @@ public boolean isDark() { return forceDark; } + @Override + public boolean supportsAnimation() { + return true; + } + @Override public void switchDayNight(boolean animated) { forceDark = !forceDark; @@ -490,7 +518,7 @@ public void setOnSwitchDayNightDelegate(DayNightSwitchDelegate delegate) { } public interface WallpaperActivityDelegate { - void didSetNewBackground(); + void didSetNewBackground(TLRPC.WallPaper wallpaper); } public ThemePreviewActivity(Object wallPaper, Bitmap bitmap) { @@ -761,7 +789,7 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { FrameLayout.LayoutParams layoutParams = (FrameLayout.LayoutParams) listView2.getLayoutParams(); layoutParams.topMargin = actionBarHeight; if (screenType == SCREEN_TYPE_CHANGE_BACKGROUND) { - listView2.setPadding(0, dp(4), 0, dp(12 + 48 + 12 + (!self ? 48 + 10 : 0)) - 12 + (insideBottomSheet() ? AndroidUtilities.navigationBarHeight : 0)); + listView2.setPadding(0, dp(4), 0, dp(12 + 48 + 12 + (!self && dialogId > 0 ? 48 + 10 : 0)) - 12 + (insideBottomSheet() ? AndroidUtilities.navigationBarHeight : 0)); } listView2.measure(MeasureSpec.makeMeasureSpec(widthSize, MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec(heightSize - layoutParams.bottomMargin, MeasureSpec.EXACTLY)); @@ -781,7 +809,7 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { if (bottomOverlayChat != null) { bottomOverlayChat.setPadding(dp(12), dp(12), dp(12), dp(12) + (insideBottomSheet() ? AndroidUtilities.navigationBarHeight : 0)); layoutParams = (FrameLayout.LayoutParams) bottomOverlayChat.getLayoutParams(); - layoutParams.height = dp(12 + 48 + 12 + (!self ? 48 + 10 : 0)) + (insideBottomSheet() ? AndroidUtilities.navigationBarHeight : 0); + layoutParams.height = dp(12 + 48 + 12 + (!self && dialogId > 0 ? 48 + 10 : 0)) + (insideBottomSheet() ? AndroidUtilities.navigationBarHeight : 0); measureChildWithMargins(bottomOverlayChat, widthMeasureSpec, 0, heightMeasureSpec, 0); } if (sheetDrawable != null) { @@ -929,47 +957,51 @@ protected void onSend(LongSparseArray dids, int count, TLRPC.TL_fo SharedConfig.increaseDayNightWallpaperSiwtchHint(); } boolean isDark = onSwitchDayNightDelegate.isDark(); - sunDrawable.setPlayInDirectionOfCustomEndFrame(true); - if (isDark) { - sunDrawable.setCustomEndFrame(0); - } else { - sunDrawable.setCustomEndFrame(36); - } - sunDrawable.start(); if (onSwitchDayNightDelegate != null) { - onSwitchDayNightDelegate.switchDayNight(true); - } - if (shouldShowBrightnessControll) { - if (onSwitchDayNightDelegate != null && onSwitchDayNightDelegate.isDark()) { - dimmingSlider.setVisibility(View.VISIBLE); - dimmingSlider.animateValueTo(dimAmount); + if (!onSwitchDayNightDelegate.supportsAnimation()) { + toggleTheme(); } else { - dimmingSlider.animateValueTo(0); - } - if (changeDayNightViewAnimator != null) { - changeDayNightViewAnimator.removeAllListeners(); - changeDayNightViewAnimator.cancel(); - } - changeDayNightViewAnimator = ValueAnimator.ofFloat(progressToDarkTheme, onSwitchDayNightDelegate.isDark() ? 1 : 0); - changeDayNightViewAnimator.addUpdateListener(animation -> { - progressToDarkTheme = (float) animation.getAnimatedValue(); - backgroundImage.invalidate(); - bottomOverlayChat.invalidate(); - dimmingSlider.setAlpha(progressToDarkTheme); - dimmingSliderContainer.invalidate(); - invalidateBlur(); - }); - changeDayNightViewAnimator.addListener(new AnimatorListenerAdapter() { - @Override - public void onAnimationEnd(Animator animation) { - if (!onSwitchDayNightDelegate.isDark()) { - dimmingSlider.setVisibility(View.GONE); + onSwitchDayNightDelegate.switchDayNight(true); + sunDrawable.setPlayInDirectionOfCustomEndFrame(true); + if (isDark) { + sunDrawable.setCustomEndFrame(0); + } else { + sunDrawable.setCustomEndFrame(36); + } + sunDrawable.start(); + if (shouldShowBrightnessControll) { + if (onSwitchDayNightDelegate != null && onSwitchDayNightDelegate.isDark()) { + dimmingSlider.setVisibility(View.VISIBLE); + dimmingSlider.animateValueTo(dimAmount); + } else { + dimmingSlider.animateValueTo(0); } + if (changeDayNightViewAnimator2 != null) { + changeDayNightViewAnimator2.removeAllListeners(); + changeDayNightViewAnimator2.cancel(); + } + changeDayNightViewAnimator2 = ValueAnimator.ofFloat(progressToDarkTheme, onSwitchDayNightDelegate.isDark() ? 1 : 0); + changeDayNightViewAnimator2.addUpdateListener(animation -> { + progressToDarkTheme = (float) animation.getAnimatedValue(); + backgroundImage.invalidate(); + bottomOverlayChat.invalidate(); + dimmingSlider.setAlpha(progressToDarkTheme); + dimmingSliderContainer.invalidate(); + invalidateBlur(); + }); + changeDayNightViewAnimator2.addListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + if (!onSwitchDayNightDelegate.isDark()) { + dimmingSlider.setVisibility(View.GONE); + } + } + }); + changeDayNightViewAnimator2.setDuration(250); + changeDayNightViewAnimator2.setInterpolator(CubicBezierInterpolator.DEFAULT); + changeDayNightViewAnimator2.start(); } - }); - changeDayNightViewAnimator.setDuration(250); - changeDayNightViewAnimator.setInterpolator(CubicBezierInterpolator.DEFAULT); - changeDayNightViewAnimator.start(); + } } } else if (id == OPTION_PHOTO_EDIT) { if (currentWallpaper instanceof WallpapersListActivity.FileWallpaper) { @@ -1319,7 +1351,7 @@ protected void onMoveAnimationUpdate(RecyclerView.ViewHolder holder) { listView2.setVerticalScrollBarEnabled(true); listView2.setOverScrollMode(RecyclerListView.OVER_SCROLL_NEVER); if (screenType == SCREEN_TYPE_CHANGE_BACKGROUND) { - listView2.setPadding(0, dp(4), 0, dp(12 + 48 + 12 + (!self ? 48 + 10 : 0)) - 12 + (insideBottomSheet() ? AndroidUtilities.navigationBarHeight : 0)); + listView2.setPadding(0, dp(4), 0, dp(12 + 48 + 12 + (!self && dialogId > 0 ? 48 + 10 : 0)) - 12 + (insideBottomSheet() ? AndroidUtilities.navigationBarHeight : 0)); } else if (screenType == SCREEN_TYPE_ACCENT_COLOR) { listView2.setPadding(0, dp(4), 0, dp(16)); } else { @@ -1451,14 +1483,10 @@ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { applyButton1 = new BlurButton(context); ScaleStateListAnimator.apply(applyButton1, 0.033f, 1.2f); - if (dialogId != 0) { - applyButton1.setText(LocaleController.getString(R.string.ApplyWallpaperForMe)); - } else { - applyButton1.setText(LocaleController.getString(R.string.ApplyWallpaper)); - } + updateApplyButton1(false); applyButton1.setOnClickListener(view -> applyWallpaperBackground(false)); - if (dialogId != 0 && !self && serverWallpaper == null) { + if (dialogId > 0 && !self && serverWallpaper == null) { applyButton2 = new BlurButton(context); ScaleStateListAnimator.apply(applyButton2, 0.033f, 1.2f); TLRPC.User user = getMessagesController().getUser(dialogId); @@ -1550,7 +1578,7 @@ public boolean dispatchTouchEvent(MotionEvent event) { TextPaint textPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG); textPaint.setTextSize(dp(14)); textPaint.setTypeface(AndroidUtilities.getTypeface("fonts/rmedium.ttf")); - { + if (!(currentWallpaper instanceof WallpapersListActivity.EmojiWallpaper)) { int textsCount; if (screenType == SCREEN_TYPE_ACCENT_COLOR || currentWallpaper instanceof WallpapersListActivity.ColorWallpaper) { textsCount = 3; @@ -2398,11 +2426,78 @@ private void updateIntensity() { invalidateBlur(); } + private void updateApplyButton1(boolean animated) { + if (dialogId > 0) { + applyButton1.setText(LocaleController.getString(R.string.ApplyWallpaperForMe)); + } else if (dialogId < 0) { + TLRPC.Chat chat = getMessagesController().getChat(-dialogId); + if (chat != null) { + applyButton1.setText(LocaleController.formatString(R.string.ApplyWallpaperForChannel, chat.title)); + if (boostsStatus != null && boostsStatus.level < getMessagesController().channelCustomWallpaperLevelMin) { + SpannableStringBuilder text = new SpannableStringBuilder("l"); + if (lockSpan == null) { + lockSpan = new ColoredImageSpan(R.drawable.mini_switch_lock); + lockSpan.setTopOffset(1); + } + text.setSpan(lockSpan, 0, 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); + text.append(" ").append(LocaleController.formatPluralString("ReactionLevelRequiredBtn", getMessagesController().channelCustomWallpaperLevelMin)); + applyButton1.setSubText(text, animated); + } else if (boostsStatus == null) { + checkBoostsLevel(); + } + } else { + applyButton1.setText(LocaleController.formatString(R.string.ApplyWallpaperForChannel, LocaleController.getString(R.string.AccDescrChannel).toLowerCase())); + } + } else { + applyButton1.setText(LocaleController.getString(R.string.ApplyWallpaper)); + } + } + private void applyWallpaperBackground(boolean forBoth) { + if (dialogId < 0) { + if (boostsStatus != null && boostsStatus.level < getMessagesController().channelCustomWallpaperLevelMin) { + getMessagesController().getBoostsController().userCanBoostChannel(dialogId, boostsStatus, canApplyBoost -> { + if (getContext() == null) { + return; + } + LimitReachedBottomSheet limitReachedBottomSheet = new LimitReachedBottomSheet(this, getContext(), LimitReachedBottomSheet.TYPE_BOOSTS_FOR_CUSTOM_WALLPAPER, currentAccount, getResourceProvider()); + limitReachedBottomSheet.setCanApplyBoost(canApplyBoost); + limitReachedBottomSheet.setBoostsStats(boostsStatus, true); + limitReachedBottomSheet.setDialogId(dialogId); + if (!insideBottomSheet()) { + limitReachedBottomSheet.showStatisticButtonInLink(() -> { + TLRPC.Chat chat = getMessagesController().getChat(-dialogId); + Bundle args = new Bundle(); + args.putLong("chat_id", -dialogId); + args.putBoolean("is_megagroup", chat.megagroup); + args.putBoolean("start_from_boosts", true); + TLRPC.ChatFull chatInfo = getMessagesController().getChatFull(-dialogId); + if (chatInfo == null || !chatInfo.can_view_stats) { + args.putBoolean("only_boosts", true); + } + ; + StatisticActivity fragment = new StatisticActivity(args); + presentFragment(fragment); + }); + } + showDialog(limitReachedBottomSheet); + }); + return; + } else if (boostsStatus == null) { + return; + } + } if (!getUserConfig().isPremium() && forBoth) { showDialog(new PremiumFeatureBottomSheet(this, PremiumPreviewFragment.PREMIUM_FEATURE_WALLPAPER, true)); return; } + if (currentWallpaper instanceof WallpapersListActivity.EmojiWallpaper) { + if (delegate != null) { + delegate.didSetNewBackground(null); + } + finishFragment(); + return; + } boolean done; boolean sameFile = false; @@ -2559,9 +2654,10 @@ private void applyWallpaperBackground(boolean forBoth) { int gradientColor3 = 0; File path = null; + TLRPC.WallPaper tlwallPaper = null; if (currentWallpaper instanceof TLRPC.TL_wallPaper) { - TLRPC.TL_wallPaper wallPaper = (TLRPC.TL_wallPaper) currentWallpaper; - slug = wallPaper.slug; + tlwallPaper = (TLRPC.TL_wallPaper) currentWallpaper; + slug = tlwallPaper.slug; } else if (currentWallpaper instanceof WallpapersListActivity.ColorWallpaper) { WallpapersListActivity.ColorWallpaper wallPaper = (WallpapersListActivity.ColorWallpaper) currentWallpaper; if (Theme.DEFAULT_BACKGROUND_SLUG.equals(wallPaper.slug)) { @@ -2648,12 +2744,12 @@ private void applyWallpaperBackground(boolean forBoth) { needFinishFragment = false; if (path != null && getMessagesController().uploadingWallpaperInfo == wallpaperInfo) { - TLRPC.WallPaper wallPaper = new TLRPC.TL_wallPaper(); - wallPaper.settings = new TLRPC.TL_wallPaperSettings(); - wallPaper.settings.intensity = (int) (wallpaperInfo.intensity * 100); - wallPaper.settings.blur = wallpaperInfo.isBlurred; - wallPaper.settings.motion = wallpaperInfo.isMotion; - wallPaper.uploadingImage = path.getAbsolutePath(); + tlwallPaper = new TLRPC.TL_wallPaper(); + tlwallPaper.settings = new TLRPC.TL_wallPaperSettings(); + tlwallPaper.settings.intensity = (int) (wallpaperInfo.intensity * 100); + tlwallPaper.settings.blur = wallpaperInfo.isBlurred; + tlwallPaper.settings.motion = wallpaperInfo.isMotion; + tlwallPaper.uploadingImage = path.getAbsolutePath(); Bitmap bitmap = Bitmap.createBitmap(50, 50, Bitmap.Config.ARGB_8888); Canvas canvas = new Canvas(bitmap); float s = Math.max(50 / (float) backgroundImage.getMeasuredWidth(), 50 / (float) backgroundImage.getMeasuredHeight()); @@ -2668,23 +2764,31 @@ private void applyWallpaperBackground(boolean forBoth) { backgroundImage.draw(canvas); dimAmount = currentDim; Utilities.blurBitmap(bitmap, 3, 1, bitmap.getWidth(), bitmap.getHeight(), bitmap.getRowBytes()); - wallPaper.stripedThumb = bitmap; + tlwallPaper.stripedThumb = bitmap; - createServiceMessageLocal(wallPaper, forBoth); + createServiceMessageLocal(tlwallPaper, forBoth); - TLRPC.UserFull fullUser = getMessagesController().getUserFull(dialogId); - if (fullUser != null) { - fullUser.wallpaper = wallPaper; - NotificationCenter.getInstance(currentAccount).postNotificationName(NotificationCenter.userInfoDidLoad, dialogId, fullUser); + if (dialogId >= 0) { + TLRPC.UserFull fullUser = getMessagesController().getUserFull(dialogId); + if (fullUser != null) { + fullUser.wallpaper = tlwallPaper; + NotificationCenter.getInstance(currentAccount).postNotificationName(NotificationCenter.userInfoDidLoad, dialogId, fullUser); + } + } else { + TLRPC.ChatFull fullChat = getMessagesController().getChatFull(-dialogId); + if (fullChat != null) { + fullChat.wallpaper = tlwallPaper; + NotificationCenter.getInstance(currentAccount).postNotificationName(NotificationCenter.chatInfoDidLoad, fullChat, 0, false, false); + } } } else { - ChatThemeController.getInstance(currentAccount).setWallpaperToUser(dialogId, null, wallpaperInfo, serverWallpaper, () -> { + ChatThemeController.getInstance(currentAccount).setWallpaperToPeer(dialogId, null, wallpaperInfo, serverWallpaper, () -> { }); } setupFinished = true; if (delegate != null) { - delegate.didSetNewBackground(); + delegate.didSetNewBackground(tlwallPaper); } finishFragment(); } else { @@ -2701,7 +2805,7 @@ private void applyWallpaperBackground(boolean forBoth) { } if (needFinishFragment) { if (delegate != null) { - delegate.didSetNewBackground(); + delegate.didSetNewBackground(tlwallPaper); } finishFragment(); } @@ -4457,6 +4561,11 @@ private void setCurrentImage(boolean setThumb) { } else { backgroundImage.setImage(wallPaper.imageUrl, imageFilter, wallPaper.thumbUrl, "100_100_b"); } + } else if (currentWallpaper instanceof WallpapersListActivity.EmojiWallpaper) { + final boolean isDark = onSwitchDayNightDelegate != null ? onSwitchDayNightDelegate.isDark() : Theme.isCurrentThemeDark(); + Drawable backgroundDrawable = PreviewView.getBackgroundDrawableFromTheme(currentAccount, ((WallpapersListActivity.EmojiWallpaper) currentWallpaper).emoticon, isDark); + backgroundImage.setBackground(backgroundDrawable); + themeDelegate.applyChatServiceMessageColor(AndroidUtilities.calcDrawableColor(backgroundDrawable), checkBlur(backgroundDrawable), backgroundDrawable, currentIntensity); } } else { if (backgroundGradientDisposable != null) { @@ -4768,50 +4877,86 @@ public MessagesAdapter(Context context) { TLRPC.Message message; MessageObject messageObject; if (screenType == SCREEN_TYPE_CHANGE_BACKGROUND) { - message = new TLRPC.TL_message(); - if (currentWallpaper instanceof WallpapersListActivity.ColorWallpaper) { - message.message = LocaleController.getString("BackgroundColorSinglePreviewLine2", R.string.BackgroundColorSinglePreviewLine2); - } else { - message.message = LocaleController.getString("BackgroundPreviewLine2", R.string.BackgroundPreviewLine2); + if (dialogId >= 0) { + message = new TLRPC.TL_message(); + if (currentWallpaper instanceof WallpapersListActivity.ColorWallpaper) { + message.message = LocaleController.getString(R.string.BackgroundColorSinglePreviewLine2); + } else { + message.message = LocaleController.getString(R.string.BackgroundPreviewLine2); + } + message.date = date + 60; + message.dialog_id = 1; + message.flags = 259; + message.id = 1; + message.media = new TLRPC.TL_messageMediaEmpty(); + message.out = true; + message.from_id = new TLRPC.TL_peerUser(); + message.from_id.user_id = UserConfig.getInstance(currentAccount).getClientUserId(); + message.peer_id = new TLRPC.TL_peerUser(); + message.peer_id.user_id = UserConfig.getInstance(currentAccount).getClientUserId(); + messageObject = new MessageObject(currentAccount, message, true, false) { + @Override + public boolean needDrawAvatar() { + return false; + } + }; + messageObject.eventId = 1; + messageObject.resetLayout(); + messages.add(messageObject); } - message.date = date + 60; - message.dialog_id = 1; - message.flags = 259; - message.from_id = new TLRPC.TL_peerUser(); - message.from_id.user_id = UserConfig.getInstance(currentAccount).getClientUserId(); - message.id = 1; - message.media = new TLRPC.TL_messageMediaEmpty(); - message.out = true; - message.peer_id = new TLRPC.TL_peerUser(); - message.peer_id.user_id = 0; - messageObject = new MessageObject(currentAccount, message, true, false); - messageObject.eventId = 1; - messageObject.resetLayout(); - messages.add(messageObject); + MessageObject replyMessageObject = null; message = new TLRPC.TL_message(); - if (dialogId != 0) { - message.message = LocaleController.getString("BackgroundColorSinglePreviewLine3", R.string.BackgroundColorSinglePreviewLine3); - } else if (currentWallpaper instanceof WallpapersListActivity.ColorWallpaper) { - message.message = LocaleController.getString("BackgroundColorSinglePreviewLine1", R.string.BackgroundColorSinglePreviewLine1); + TLRPC.Chat currentChat = dialogId < 0 ? getMessagesController().getChat(-dialogId) : null; + if (currentChat != null) { + message.message = LocaleController.getString(R.string.ChannelBackgroundMessagePreview); + + TLRPC.TL_message replyMessage = new TLRPC.TL_message(); + replyMessage.message = LocaleController.getString(R.string.ChannelBackgroundMessageReplyText); + replyMessageObject = new MessageObject(currentAccount, replyMessage, true, false) { + @Override + public boolean needDrawAvatar() { + return false; + } + }; + + message.from_id = new TLRPC.TL_peerChannel(); + message.from_id.channel_id = currentChat.id; + message.peer_id = new TLRPC.TL_peerChannel(); + message.peer_id.channel_id = currentChat.id; } else { - message.message = LocaleController.getString("BackgroundPreviewLine1", R.string.BackgroundPreviewLine1); + if (dialogId != 0) { + message.message = LocaleController.getString(R.string.BackgroundColorSinglePreviewLine3); + } else if (currentWallpaper instanceof WallpapersListActivity.ColorWallpaper) { + message.message = LocaleController.getString(R.string.BackgroundColorSinglePreviewLine1); + } else { + message.message = LocaleController.getString(R.string.BackgroundPreviewLine1); + } + message.from_id = new TLRPC.TL_peerUser(); + message.peer_id = new TLRPC.TL_peerUser(); + message.peer_id.user_id = UserConfig.getInstance(currentAccount).getClientUserId(); } message.date = date + 60; message.dialog_id = 1; message.flags = 257 + 8; - message.from_id = new TLRPC.TL_peerUser(); message.id = 1; message.media = new TLRPC.TL_messageMediaEmpty(); message.out = false; - message.peer_id = new TLRPC.TL_peerUser(); - message.peer_id.user_id = UserConfig.getInstance(currentAccount).getClientUserId(); - messageObject = new MessageObject(currentAccount, message, true, false); + messageObject = new MessageObject(currentAccount, message, replyMessageObject, true, false) { + @Override + public boolean needDrawAvatar() { + return false; + } + }; + if (replyMessageObject != null) { + messageObject.customReplyName = LocaleController.getString(R.string.ChannelBackgroundMessageReplyName); + } messageObject.eventId = 1; messageObject.resetLayout(); messages.add(messageObject); if (dialogId != 0 && serverWallpaper == null) { + TLRPC.User user = getMessagesController().getUser(dialogId); message = new TLRPC.TL_message(); message.message = ""; messageObject = new MessageObject(currentAccount, message, true, false); @@ -4820,9 +4965,12 @@ public MessagesAdapter(Context context) { messages.add(messageObject); message = new TLRPC.TL_message(); - TLRPC.User user = getMessagesController().getUser(dialogId); - String username = user == null ? "DELETED" : user.first_name; - message.message = LocaleController.formatString("ChatBackgroundHint", R.string.ChatBackgroundHint, username); + if (user != null) { + String username = UserObject.getFirstName(user); + message.message = LocaleController.formatString(R.string.ChatBackgroundHint, username); + } else { + message.message = LocaleController.getString(R.string.ChannelBackgroundHint); + } message.date = date + 60; message.dialog_id = 1; message.flags = 257 + 8; @@ -5181,7 +5329,7 @@ public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup viewGroup, int viewT view = new ChatMessageCell(mContext, false, null, new Theme.ResourcesProvider() { @Override public int getColor(int key) { - return getThemedColor(key); + return themeDelegate.getColor(key); } @Override @@ -5198,16 +5346,41 @@ public Drawable getDrawable(String drawableKey) { if (drawableKey.equals(Theme.key_drawable_msgOutMediaSelected)) { return msgOutMediaDrawableSelected; } - if (getResourceProvider() != null) { - return getResourceProvider().getDrawable(drawableKey); + if (themeDelegate != null) { + return themeDelegate.getDrawable(drawableKey); } return Theme.getThemeDrawable(drawableKey); } + @Override + public boolean isDark() { + return themeDelegate.isDark(); + } + + @Override + public int getCurrentColor(int key) { + return themeDelegate.getCurrentColor(key); + } + + @Override + public int getColorOrDefault(int key) { + return themeDelegate.getColorOrDefault(key); + } + + @Override + public Paint getPaint(String paintKey) { + return themeDelegate.getPaint(paintKey); + } + + @Override + public boolean hasGradientService() { + return themeDelegate.hasGradientService(); + } + @Override public void applyServiceShaderMatrix(int w, int h, float translationX, float translationY) { - if (getResourceProvider() != null) { - getResourceProvider().applyServiceShaderMatrix(w, h, translationX, translationY); + if (themeDelegate != null) { + themeDelegate.applyServiceShaderMatrix(w, h, translationX, translationY); } else { Theme.ResourcesProvider.super.applyServiceShaderMatrix(w, h, translationX, translationY); } @@ -5287,7 +5460,7 @@ public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) { } else { pinnedTop = false; } - messageCell.isChat = showSecretMessages; + messageCell.isChat = showSecretMessages || dialogId < 0; messageCell.setFullyDraw(true); messageCell.setMessageObject(message, null, pinnedBotton, pinnedTop); } else if (view instanceof ChatActionCell) { @@ -5609,11 +5782,19 @@ private void createServiceMessageLocal(TLRPC.WallPaper wallPaper, boolean forBot message.unread = true; message.out = true; message.local_id = message.id = getUserConfig().getNewMessageId(); - message.from_id = new TLRPC.TL_peerUser(); - message.from_id.user_id = getUserConfig().getClientUserId(); + TLRPC.Chat currentChat = getMessagesController().getChat(-dialogId); + if (ChatObject.isChannel(currentChat)) { + message.from_id = new TLRPC.TL_peerChannel(); + message.from_id.channel_id = currentChat.id; + message.peer_id = new TLRPC.TL_peerChannel(); + message.peer_id.channel_id = currentChat.id; + } else { + message.from_id = new TLRPC.TL_peerUser(); + message.from_id.user_id = getUserConfig().getClientUserId(); + message.peer_id = new TLRPC.TL_peerUser(); + message.peer_id.user_id = dialogId; + } message.flags |= 256; - message.peer_id = new TLRPC.TL_peerUser(); - message.peer_id.user_id = dialogId; message.date = getConnectionsManager().getCurrentTime(); TLRPC.TL_messageActionSetChatWallPaper setChatWallPaper = new TLRPC.TL_messageActionSetChatWallPaper(); message.action = setChatWallPaper; @@ -5631,6 +5812,8 @@ private void createServiceMessageLocal(TLRPC.WallPaper wallPaper, boolean forBot public interface DayNightSwitchDelegate { boolean isDark(); + boolean supportsAnimation(); + void switchDayNight(boolean animated); } @@ -5742,6 +5925,14 @@ public Drawable getBackground() { @Override public void setBackground(Drawable drawable) { background = drawable; + if (background != null) { + background.setCallback(this); + } + } + + @Override + protected boolean verifyDrawable(@NonNull Drawable who) { + return background == who || super.verifyDrawable(who); } } @@ -5770,6 +5961,9 @@ public void setTop(int top, int backgroundWidth, int backgroundHeight, int heigh private class BlurButton extends View { private Text text; + private Text subtext; + private boolean subtextShown; + private AnimatedFloat subtextShownT = new AnimatedFloat(this, 0, 350, CubicBezierInterpolator.EASE_OUT_QUINT); private final Drawable rippleDrawable = Theme.createRadSelectorDrawable(0x10ffffff, 8, 8); private final ColorFilter colorFilter; @@ -5779,7 +5973,7 @@ public BlurButton(Context context) { ColorMatrix colorMatrix = new ColorMatrix(); AndroidUtilities.adjustSaturationColorMatrix(colorMatrix, +.35f); - AndroidUtilities.multiplyBrightnessColorMatrix(colorMatrix, 0.75f); + AndroidUtilities.multiplyBrightnessColorMatrix(colorMatrix, 0.9f); colorFilter = new ColorMatrixColorFilter(colorMatrix); } @@ -5787,6 +5981,17 @@ public void setText(CharSequence text) { this.text = new Text(text, 14, AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); } + public void setSubText(CharSequence subtext, boolean animated) { + if (subtext != null) { + this.subtext = new Text(subtext, 12); + } + subtextShown = subtext != null; + if (!animated) { + subtextShownT.set(subtextShown, true); + } + invalidate(); + } + public CharSequence getText() { return this.text != null ? this.text.getText() : null; } @@ -5830,10 +6035,19 @@ protected void onDraw(Canvas canvas) { invalidate(); } + final float subtitleT = subtextShownT.set(subtextShown); if (loadingT < 1 && text != null) { text .ellipsize(getWidth() - dp(14)) - .draw(canvas, (getWidth() - text.getWidth()) / 2f, getHeight() / 2f + loadingT * dp(24), textColor, 1f - loadingT); + .draw(canvas, (getWidth() - text.getWidth()) / 2f, getHeight() / 2f + loadingT * dp(24) - dp(7) * subtitleT, textColor, 1f - loadingT); + } + if (loadingT < 1 && subtext != null) { + canvas.save(); + canvas.scale(subtitleT, subtitleT, getWidth() / 2f, getHeight() / 2f + dp(11)); + subtext + .ellipsize(getWidth() - dp(14)) + .draw(canvas, (getWidth() - subtext.getWidth()) / 2f, getHeight() / 2f + loadingT * dp(24) + dp(11), Theme.multAlpha(textColor, .75f), 1f - loadingT); + canvas.restore(); } rippleDrawable.setBounds(0, 0, getWidth(), getHeight()); @@ -5890,7 +6104,7 @@ protected boolean verifyDrawable(@NonNull Drawable who) { } } - public static class ThemeDelegate implements Theme.ResourcesProvider { + public class ThemeDelegate implements Theme.ResourcesProvider { public Theme.ResourcesProvider parentProvider; @@ -6006,6 +6220,22 @@ public void applyChatServiceMessageColor(int[] custom, Drawable wallpaperOverrid if (serviceBitmapShader != null && (currentColors.indexOfKey(Theme.key_chat_serviceBackground) < 0 || drawable instanceof MotionBackgroundDrawable || drawable instanceof BitmapDrawable)) { ColorMatrix colorMatrix = new ColorMatrix(); + if (drawable instanceof MotionBackgroundDrawable) { + float intensity = ((MotionBackgroundDrawable) drawable).getIntensity(); + if (intensity >= 0) { + colorMatrix.setSaturation(1.6f); + AndroidUtilities.multiplyBrightnessColorMatrix(colorMatrix, isDark() ? .97f : .92f); + AndroidUtilities.adjustBrightnessColorMatrix(colorMatrix, isDark() ? +.12f : -.06f); + } else { + colorMatrix.setSaturation(1.1f); + AndroidUtilities.multiplyBrightnessColorMatrix(colorMatrix, isDark() ? .4f : .8f); + AndroidUtilities.adjustBrightnessColorMatrix(colorMatrix, isDark() ? +.08f : -.06f); + } + } else { + colorMatrix.setSaturation(1.6f); + AndroidUtilities.multiplyBrightnessColorMatrix(colorMatrix, isDark() ? .9f : .84f); + AndroidUtilities.adjustBrightnessColorMatrix(colorMatrix, isDark() ? -.04f : +.06f); + } if (drawable instanceof MotionBackgroundDrawable) { float intensity = ((MotionBackgroundDrawable) drawable).getIntensity(); if (overrideIntensity != null) { @@ -6051,5 +6281,159 @@ public void applyServiceShaderMatrix(int w, int h, float translationX, float tra Theme.applyServiceShaderMatrix(serviceBitmap, serviceBitmapShader, serviceBitmapMatrix, w, h, translationX, translationY); } } + + @Override + public boolean isDark() { + return onSwitchDayNightDelegate != null ? onSwitchDayNightDelegate.isDark() : (parentProvider != null ? parentProvider.isDark() : Theme.isCurrentThemeDark()); + } + } + + private View changeDayNightView; + private float changeDayNightViewProgress; + private ValueAnimator changeDayNightViewAnimator; + + @SuppressLint("NotifyDataSetChanged") + public void toggleTheme() { + if (changeDayNightView != null) { + return; + } + + FrameLayout decorView1 = insideBottomSheet() ? (FrameLayout) parentLayout.getBottomSheet().getWindow().getDecorView() : (FrameLayout) getParentActivity().getWindow().getDecorView(); + Bitmap bitmap = Bitmap.createBitmap(decorView1.getWidth(), decorView1.getHeight(), Bitmap.Config.ARGB_8888); + Canvas bitmapCanvas = new Canvas(bitmap); + dayNightItem.setAlpha(0f); + decorView1.draw(bitmapCanvas); + dayNightItem.setAlpha(1f); + + Paint xRefPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + xRefPaint.setColor(0xff000000); + xRefPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.CLEAR)); + + Paint bitmapPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + bitmapPaint.setFilterBitmap(true); + int[] position = new int[2]; + dayNightItem.getLocationInWindow(position); + float x = position[0]; + float y = position[1]; + float cx = x + dayNightItem.getMeasuredWidth() / 2f; + float cy = y + dayNightItem.getMeasuredHeight() / 2f; + + float r = Math.max(bitmap.getHeight(), bitmap.getWidth()) + AndroidUtilities.navigationBarHeight; + + Shader bitmapShader = new BitmapShader(bitmap, Shader.TileMode.CLAMP, Shader.TileMode.CLAMP); + bitmapPaint.setShader(bitmapShader); + changeDayNightView = new View(getContext()) { + @Override + protected void onDraw(Canvas canvas) { + super.onDraw(canvas); + if (themeDelegate.isDark()) { + if (changeDayNightViewProgress > 0f) { + bitmapCanvas.drawCircle(cx, cy, r * changeDayNightViewProgress, xRefPaint); + } + canvas.drawBitmap(bitmap, 0, 0, bitmapPaint); + } else { + canvas.drawCircle(cx, cy, r * (1f - changeDayNightViewProgress), bitmapPaint); + } + canvas.save(); + canvas.translate(x, y); + dayNightItem.draw(canvas); + canvas.restore(); + } + }; + changeDayNightView.setOnTouchListener((v, event) -> true); + changeDayNightViewProgress = 0f; + changeDayNightViewAnimator = ValueAnimator.ofFloat(0, 1f); + changeDayNightViewAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() { + boolean changedNavigationBarColor = false; + + @Override + public void onAnimationUpdate(ValueAnimator valueAnimator) { + changeDayNightViewProgress = (float) valueAnimator.getAnimatedValue(); + changeDayNightView.invalidate(); + if (!changedNavigationBarColor && changeDayNightViewProgress > .5f) { + changedNavigationBarColor = true; + } + } + }); + changeDayNightViewAnimator.addListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + if (changeDayNightView != null) { + if (changeDayNightView.getParent() != null) { + ((ViewGroup) changeDayNightView.getParent()).removeView(changeDayNightView); + } + changeDayNightView = null; + } + changeDayNightViewAnimator = null; + super.onAnimationEnd(animation); + } + }); + changeDayNightViewAnimator.setDuration(400); + changeDayNightViewAnimator.setInterpolator(Easings.easeInOutQuad); + changeDayNightViewAnimator.start(); + + decorView1.addView(changeDayNightView, new ViewGroup.LayoutParams(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT)); + + AndroidUtilities.runOnUIThread(() -> { + onSwitchDayNightDelegate.switchDayNight(false); + setForceDark(themeDelegate.isDark(), true); + setCurrentImage(false); + invalidateBlur(); + updateBlurred(); + if (themeDescriptions != null) { + for (int i = 0; i < themeDescriptions.size(); ++i) { + themeDescriptions.get(i).setColor(getThemedColor(themeDescriptions.get(i).getCurrentKey()), false, false); + } + } + + if (shouldShowBrightnessControll) { + if (onSwitchDayNightDelegate != null && onSwitchDayNightDelegate.isDark()) { + dimmingSlider.setVisibility(View.VISIBLE); + dimmingSlider.animateValueTo(dimAmount); + } else { + dimmingSlider.animateValueTo(0); + } + if (changeDayNightViewAnimator2 != null) { + changeDayNightViewAnimator2.removeAllListeners(); + changeDayNightViewAnimator2.cancel(); + } + changeDayNightViewAnimator2 = ValueAnimator.ofFloat(progressToDarkTheme, onSwitchDayNightDelegate.isDark() ? 1 : 0); + changeDayNightViewAnimator2.addUpdateListener(animation -> { + progressToDarkTheme = (float) animation.getAnimatedValue(); + backgroundImage.invalidate(); + bottomOverlayChat.invalidate(); + dimmingSlider.setAlpha(progressToDarkTheme); + dimmingSliderContainer.invalidate(); + invalidateBlur(); + }); + changeDayNightViewAnimator2.addListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + if (!onSwitchDayNightDelegate.isDark()) { + dimmingSlider.setVisibility(View.GONE); + } + } + }); + changeDayNightViewAnimator2.setDuration(250); + changeDayNightViewAnimator2.setInterpolator(CubicBezierInterpolator.DEFAULT); + changeDayNightViewAnimator2.start(); + } + }); + } + + public void setForceDark(boolean isDark, boolean playAnimation) { + if (playAnimation) { + sunDrawable.setCustomEndFrame(isDark ? sunDrawable.getFramesCount() : 0); + if (sunDrawable != null) { + sunDrawable.start(); + } + } else { + int frame = isDark ? sunDrawable.getFramesCount() - 1 : 0; + sunDrawable.setCurrentFrame(frame, false, true); + sunDrawable.setCustomEndFrame(frame); + if (dayNightItem != null) { + dayNightItem.invalidate(); + } + } } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/VoIPFragment.java b/TMessagesProj/src/main/java/org/telegram/ui/VoIPFragment.java index a1e20aa4514..3fc5498e17c 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/VoIPFragment.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/VoIPFragment.java @@ -1,5 +1,7 @@ package org.telegram.ui; +import static org.telegram.messenger.AndroidUtilities.dp; +import static org.telegram.messenger.AndroidUtilities.isTablet; import static org.telegram.ui.GroupCallActivity.TRANSITION_DURATION; import android.Manifest; @@ -7,6 +9,7 @@ import android.animation.AnimatorListenerAdapter; import android.animation.AnimatorSet; import android.animation.ObjectAnimator; +import android.animation.PropertyValuesHolder; import android.animation.ValueAnimator; import android.annotation.SuppressLint; import android.annotation.TargetApi; @@ -23,13 +26,15 @@ import android.graphics.drawable.GradientDrawable; import android.os.Build; import android.os.PowerManager; +import android.text.Layout; +import android.text.Spannable; +import android.text.TextPaint; import android.text.TextUtils; import android.transition.ChangeBounds; import android.transition.TransitionManager; import android.transition.TransitionSet; import android.transition.TransitionValues; import android.transition.Visibility; -import android.util.Log; import android.util.TypedValue; import android.view.Gravity; import android.view.KeyEvent; @@ -59,11 +64,12 @@ import org.telegram.messenger.Emoji; import org.telegram.messenger.FileLog; import org.telegram.messenger.ImageLocation; -import org.telegram.messenger.ImageReceiver; import org.telegram.messenger.LocaleController; +import org.telegram.messenger.MediaDataController; import org.telegram.messenger.MessagesController; import org.telegram.messenger.NotificationCenter; import org.telegram.messenger.R; +import org.telegram.messenger.SharedConfig; import org.telegram.messenger.UserConfig; import org.telegram.messenger.UserObject; import org.telegram.messenger.Utilities; @@ -71,35 +77,51 @@ import org.telegram.messenger.voip.Instance; import org.telegram.messenger.voip.VideoCapturerDevice; import org.telegram.messenger.voip.VoIPService; +import org.telegram.messenger.voip.VoipAudioManager; import org.telegram.tgnet.TLRPC; import org.telegram.ui.ActionBar.ActionBar; import org.telegram.ui.ActionBar.AlertDialog; import org.telegram.ui.ActionBar.DarkAlertDialog; import org.telegram.ui.ActionBar.Theme; import org.telegram.ui.Components.AlertsCreator; +import org.telegram.ui.Components.AnimatedEmojiDrawable; +import org.telegram.ui.Components.AnimatedEmojiSpan; +import org.telegram.ui.Components.AvatarDrawable; import org.telegram.ui.Components.BackgroundGradientDrawable; import org.telegram.ui.Components.BackupImageView; import org.telegram.ui.Components.CubicBezierInterpolator; -import org.telegram.ui.Components.HintView; +import org.telegram.ui.Components.HideViewAfterAnimation; import org.telegram.ui.Components.LayoutHelper; import org.telegram.ui.Components.voip.AcceptDeclineView; -import org.telegram.ui.Components.voip.PrivateVideoPreviewDialog; +import org.telegram.ui.Components.voip.EmojiRationalLayout; +import org.telegram.ui.Components.voip.HideEmojiTextView; +import org.telegram.ui.Components.voip.ImageWithWavesView; +import org.telegram.ui.Components.voip.EndCloseLayout; +import org.telegram.ui.Components.voip.PrivateVideoPreviewDialogNew; +import org.telegram.ui.Components.voip.RateCallLayout; +import org.telegram.ui.Components.voip.VoIPBackgroundProvider; import org.telegram.ui.Components.voip.VoIPButtonsLayout; import org.telegram.ui.Components.voip.VoIPFloatingLayout; import org.telegram.ui.Components.voip.VoIPHelper; import org.telegram.ui.Components.voip.VoIPNotificationsLayout; -import org.telegram.ui.Components.voip.VoIPOverlayBackground; import org.telegram.ui.Components.voip.VoIPPiPView; import org.telegram.ui.Components.voip.VoIPStatusTextView; import org.telegram.ui.Components.voip.VoIPTextureView; import org.telegram.ui.Components.voip.VoIPToggleButton; import org.telegram.ui.Components.voip.VoIPWindowView; +import org.telegram.ui.Components.voip.VoIpGradientLayout; +import org.telegram.ui.Components.voip.VoIpHintView; +import org.telegram.ui.Components.voip.VoIpSnowView; +import org.telegram.ui.Components.voip.VoIpSwitchLayout; +import org.telegram.ui.Stories.recorder.HintView2; import org.webrtc.EglBase; import org.webrtc.GlRectDrawer; import org.webrtc.RendererCommon; import org.webrtc.TextureViewRenderer; import java.io.ByteArrayOutputStream; +import java.util.ArrayList; +import java.util.List; public class VoIPFragment implements VoIPService.StateListener, NotificationCenter.NotificationCenterDelegate { @@ -114,23 +136,34 @@ public class VoIPFragment implements VoIPService.StateListener, NotificationCent TLRPC.User currentUser; TLRPC.User callingUser; - VoIPToggleButton[] bottomButtons = new VoIPToggleButton[4]; + private VoIpSwitchLayout bottomSpeakerBtn; + private VoIpSwitchLayout bottomVideoBtn; + private VoIpSwitchLayout bottomMuteBtn; + private VoIPToggleButton bottomEndCallBtn; + private final VoIPBackgroundProvider backgroundProvider = new VoIPBackgroundProvider(); private ViewGroup fragmentView; - private VoIPOverlayBackground overlayBackground; - private BackupImageView callingUserPhotoView; - private BackupImageView callingUserPhotoViewMini; + private VoIpGradientLayout gradientLayout; + private VoIpSnowView voIpSnowView; + private ImageWithWavesView callingUserPhotoViewMini; private TextView callingUserTitle; private VoIPStatusTextView statusTextView; private ImageView backIcon; private ImageView speakerPhoneIcon; + private int selectedRating; LinearLayout emojiLayout; + FrameLayout hideEmojiLayout; + TextView hideEmojiTextView; + RateCallLayout rateCallLayout; + LinearLayout emojiRationalLayout; + TextView emojiRationalTopTextView; TextView emojiRationalTextView; - ImageView[] emojiViews = new ImageView[4]; + EndCloseLayout endCloseLayout; + BackupImageView[] emojiViews = new BackupImageView[4]; Emoji.EmojiDrawable[] emojiDrawables = new Emoji.EmojiDrawable[4]; LinearLayout statusLayout; private VoIPFloatingLayout currentUserCameraFloatingLayout; @@ -142,6 +175,7 @@ public class VoIPFragment implements VoIPService.StateListener, NotificationCent private VoIPTextureView currentUserTextureView; private AcceptDeclineView acceptDeclineView; + private boolean isNearEar; View bottomShadow; View topShadow; @@ -154,11 +188,12 @@ public class VoIPFragment implements VoIPService.StateListener, NotificationCent boolean callingUserIsVideo; boolean currentUserIsVideo; - private PrivateVideoPreviewDialog previewDialog; + private PrivateVideoPreviewDialogNew previewDialog; private int currentState; private int previousState; private WindowInsets lastInsets; + private boolean wasEstablished; float touchSlop; @@ -188,7 +223,8 @@ public class VoIPFragment implements VoIPService.StateListener, NotificationCent AnimationNotificationsLocker notificationsLocker = new AnimationNotificationsLocker(); VoIPNotificationsLayout notificationsLayout; - HintView tapToVideoTooltip; + HintView2 tapToVideoTooltip; + HintView2 encryptionTooltip; ValueAnimator uiVisibilityAnimator; ValueAnimator.AnimatorUpdateListener statusbarAnimatorListener = valueAnimator -> { @@ -197,23 +233,25 @@ public class VoIPFragment implements VoIPService.StateListener, NotificationCent }; float fillNaviagtionBarValue; - boolean fillNaviagtionBar; - ValueAnimator naviagtionBarAnimator; - ValueAnimator.AnimatorUpdateListener navigationBarAnimationListener = valueAnimator -> { - fillNaviagtionBarValue = (float) valueAnimator.getAnimatedValue(); - updateSystemBarColors(); - }; boolean hideUiRunnableWaiting; Runnable hideUIRunnable = () -> { hideUiRunnableWaiting = false; - if (canHideUI && uiVisible && !emojiExpanded) { + boolean tapToVideoTooltipVisible = tapToVideoTooltip != null && tapToVideoTooltip.shown(); + if (canHideUI && uiVisible && !emojiExpanded && !tapToVideoTooltipVisible) { lastContentTapTime = System.currentTimeMillis(); showUi(false); previousState = currentState; updateViewState(); } }; + + Runnable stopAnimatingBgRunnable = () -> { + if (currentState == VoIPService.STATE_ESTABLISHED) { + callingUserPhotoViewMini.setMute(true, false); + gradientLayout.pause(); + } + }; private boolean lockOnScreen; private boolean screenWasWakeup; private boolean isVideoCall; @@ -263,8 +301,8 @@ public static void show(Activity activity, boolean overlay, int account) { instance = fragment; VoIPWindowView windowView = new VoIPWindowView(activity, !transitionFromPip) { - private Path clipPath = new Path(); - private RectF rectF = new RectF(); + private final Path clipPath = new Path(); + private final RectF rectF = new RectF(); @Override public boolean dispatchKeyEvent(KeyEvent event) { @@ -378,7 +416,7 @@ private void onBackPressed() { if (emojiExpanded) { expandEmoji(false); } else { - if (emojiRationalTextView.getVisibility() != View.GONE) { + if (emojiRationalLayout.getVisibility() != View.GONE) { return; } if (canSwitchToPip && !lockOnScreen) { @@ -397,9 +435,6 @@ public static void clearInstance() { if (instance != null) { if (VoIPService.getSharedInstance() != null) { int h = instance.windowView.getMeasuredHeight(); - if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT_WATCH && instance.lastInsets != null) { - h -= instance.lastInsets.getSystemWindowInsetBottom(); - } if (instance.canSwitchToPip) { VoIPPiPView.show(instance.activity, instance.currentAccount, instance.windowView.getMeasuredWidth(), h, VoIPPiPView.ANIMATION_ENTER_TYPE_SCALE); if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT_WATCH && instance.lastInsets != null) { @@ -428,17 +463,14 @@ private void setInsets(WindowInsets windowInsets) { ((FrameLayout.LayoutParams) acceptDeclineView.getLayoutParams()).bottomMargin = lastInsets.getSystemWindowInsetBottom(); ((FrameLayout.LayoutParams) backIcon.getLayoutParams()).topMargin = lastInsets.getSystemWindowInsetTop(); ((FrameLayout.LayoutParams) speakerPhoneIcon.getLayoutParams()).topMargin = lastInsets.getSystemWindowInsetTop(); - ((FrameLayout.LayoutParams) topShadow.getLayoutParams()).topMargin = lastInsets.getSystemWindowInsetTop(); - ((FrameLayout.LayoutParams) statusLayout.getLayoutParams()).topMargin = AndroidUtilities.dp(68) + lastInsets.getSystemWindowInsetTop(); + ((FrameLayout.LayoutParams) statusLayout.getLayoutParams()).topMargin = AndroidUtilities.dp(135) + lastInsets.getSystemWindowInsetTop(); ((FrameLayout.LayoutParams) emojiLayout.getLayoutParams()).topMargin = AndroidUtilities.dp(17) + lastInsets.getSystemWindowInsetTop(); - ((FrameLayout.LayoutParams) callingUserPhotoViewMini.getLayoutParams()).topMargin = AndroidUtilities.dp(68) + lastInsets.getSystemWindowInsetTop(); - - ((FrameLayout.LayoutParams) currentUserCameraFloatingLayout.getLayoutParams()).bottomMargin = lastInsets.getSystemWindowInsetBottom(); + ((FrameLayout.LayoutParams) callingUserPhotoViewMini.getLayoutParams()).topMargin = AndroidUtilities.dp(93) + lastInsets.getSystemWindowInsetTop(); + ((FrameLayout.LayoutParams) hideEmojiLayout.getLayoutParams()).topMargin = lastInsets.getSystemWindowInsetTop(); + ((FrameLayout.LayoutParams) emojiRationalLayout.getLayoutParams()).topMargin = AndroidUtilities.dp(118) + lastInsets.getSystemWindowInsetTop(); + ((FrameLayout.LayoutParams) rateCallLayout.getLayoutParams()).topMargin = AndroidUtilities.dp(380) + lastInsets.getSystemWindowInsetTop(); ((FrameLayout.LayoutParams) callingUserMiniFloatingLayout.getLayoutParams()).bottomMargin = lastInsets.getSystemWindowInsetBottom(); - ((FrameLayout.LayoutParams) callingUserTextureView.getLayoutParams()).bottomMargin = lastInsets.getSystemWindowInsetBottom(); ((FrameLayout.LayoutParams) notificationsLayout.getLayoutParams()).bottomMargin = lastInsets.getSystemWindowInsetBottom(); - - ((FrameLayout.LayoutParams) bottomShadow.getLayoutParams()).bottomMargin = lastInsets.getSystemWindowInsetBottom(); currentUserCameraFloatingLayout.setInsets(lastInsets); callingUserMiniFloatingLayout.setInsets(lastInsets); fragmentView.requestLayout(); @@ -455,9 +487,11 @@ public VoIPFragment(int account) { isOutgoing = VoIPService.getSharedInstance().isOutgoing(); previousState = -1; currentState = VoIPService.getSharedInstance().getCallState(); + NotificationCenter.getGlobalInstance().addObserver(this, NotificationCenter.webRtcSpeakerAmplitudeEvent); NotificationCenter.getInstance(currentAccount).addObserver(this, NotificationCenter.voipServiceCreated); NotificationCenter.getGlobalInstance().addObserver(this, NotificationCenter.emojiLoaded); NotificationCenter.getGlobalInstance().addObserver(this, NotificationCenter.closeInCallActivity); + NotificationCenter.getGlobalInstance().addObserver(this, NotificationCenter.nearEarEvent); } private void destroy() { @@ -465,9 +499,11 @@ private void destroy() { if (service != null) { service.unregisterStateListener(this); } + NotificationCenter.getGlobalInstance().removeObserver(this, NotificationCenter.webRtcSpeakerAmplitudeEvent); NotificationCenter.getInstance(currentAccount).removeObserver(this, NotificationCenter.voipServiceCreated); NotificationCenter.getGlobalInstance().removeObserver(this, NotificationCenter.emojiLoaded); NotificationCenter.getGlobalInstance().removeObserver(this, NotificationCenter.closeInCallActivity); + NotificationCenter.getGlobalInstance().removeObserver(this, NotificationCenter.nearEarEvent); } @Override @@ -495,13 +531,34 @@ public void didReceivedNotification(int id, int account, Object... args) { updateKeyView(true); } else if (id == NotificationCenter.closeInCallActivity) { windowView.finish(); + } else if (id == NotificationCenter.webRtcSpeakerAmplitudeEvent) { + callingUserPhotoViewMini.setAmplitude((float) args[0] * 15.0f); + } else if (id == NotificationCenter.nearEarEvent) { + isNearEar = (boolean) args[0]; + if (isNearEar) { + callingUserPhotoViewMini.setMute(true, true); + } } } + private boolean signalBarWasReceived; + @Override public void onSignalBarsCountChanged(int count) { - if (statusTextView != null) { - statusTextView.setSignalBarCount(count); + if (count > 0) { + signalBarWasReceived = true; + } + if (statusTextView != null && gradientLayout != null && gradientLayout.isConnectedCalled() && signalBarWasReceived) { + AndroidUtilities.runOnUIThread(() -> { + statusTextView.setSignalBarCount(count); + if (count <= 1) { + gradientLayout.showToBadConnection(); + statusTextView.showBadConnection(true, true); + } else { + gradientLayout.hideBadConnection(); + statusTextView.showBadConnection(false, true); + } + }, 400); } } @@ -544,25 +601,35 @@ public View createView(Context context) { accessibilityManager = ContextCompat.getSystemService(context, AccessibilityManager.class); FrameLayout frameLayout = new FrameLayout(context) { - @Override - protected void dispatchDraw(Canvas canvas) { - super.dispatchDraw(canvas); - if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT_WATCH && lastInsets != null) { - canvas.drawRect(0, 0, getMeasuredWidth(), lastInsets.getSystemWindowInsetTop(), overlayPaint); - } - - if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT_WATCH && lastInsets != null) { - canvas.drawRect(0, getMeasuredHeight() - lastInsets.getSystemWindowInsetBottom(), getMeasuredWidth(), getMeasuredHeight(), overlayBottomPaint); - } - } float pressedX; float pressedY; boolean check; long pressedTime; + @Override + public boolean onInterceptTouchEvent(MotionEvent ev) { + if (ev.getActionMasked() == MotionEvent.ACTION_UP) { + callingUserPhotoViewMini.setMute(false, false); + gradientLayout.resume(); + AndroidUtilities.cancelRunOnUIThread(stopAnimatingBgRunnable); + if (currentState == VoIPService.STATE_ESTABLISHED) { + AndroidUtilities.runOnUIThread(stopAnimatingBgRunnable, 10000); + } + } + return super.onInterceptTouchEvent(ev); + } + @Override public boolean onTouchEvent(MotionEvent ev) { + if (ev.getActionMasked() == MotionEvent.ACTION_UP) { + callingUserPhotoViewMini.setMute(false, false); + gradientLayout.resume(); + AndroidUtilities.cancelRunOnUIThread(stopAnimatingBgRunnable); + if (currentState == VoIPService.STATE_ESTABLISHED) { + AndroidUtilities.runOnUIThread(stopAnimatingBgRunnable, 10000); + } + } /* === pinch to zoom === */ if (!canZoomGesture && !isInPinchToZoomTouchMode && !zoomStarted && ev.getActionMasked() != MotionEvent.ACTION_DOWN) { finishZoom(); @@ -666,6 +733,9 @@ public boolean onTouchEvent(MotionEvent ev) { } else if (canHideUI) { showUi(!uiVisible); previousState = currentState; + if (!uiVisible && tapToVideoTooltip != null && tapToVideoTooltip.shown()) { + tapToVideoTooltip.hide(); + } updateViewState(); } } @@ -678,11 +748,11 @@ public boolean onTouchEvent(MotionEvent ev) { @Override protected boolean drawChild(Canvas canvas, View child, long drawingTime) { - if (child == callingUserPhotoView && (currentUserIsVideo || callingUserIsVideo)) { + if (child == gradientLayout && (currentUserIsVideo || callingUserIsVideo)) { return false; } if ( - child == callingUserPhotoView || + child == gradientLayout || child == callingUserTextureView || (child == currentUserCameraFloatingLayout && currentUserCameraIsFullscreen) ) { @@ -704,16 +774,8 @@ protected boolean drawChild(Canvas canvas, View child, long drawingTime) { updateSystemBarColors(); fragmentView = frameLayout; frameLayout.setFitsSystemWindows(true); - callingUserPhotoView = new BackupImageView(context) { - - int blackoutColor = ColorUtils.setAlphaComponent(Color.BLACK, (int) (255 * 0.3f)); - @Override - protected void onDraw(Canvas canvas) { - super.onDraw(canvas); - canvas.drawColor(blackoutColor); - } - }; + gradientLayout = new VoIpGradientLayout(context, backgroundProvider); callingUserTextureView = new VoIPTextureView(context, false, true, false, false); callingUserTextureView.renderer.setScalingType(RendererCommon.ScalingType.SCALE_ASPECT_FIT); callingUserTextureView.renderer.setEnableHardwareScaler(true); @@ -721,30 +783,19 @@ protected void onDraw(Canvas canvas) { callingUserTextureView.scaleType = VoIPTextureView.SCALE_TYPE_FIT; // callingUserTextureView.attachBackgroundRenderer(); - frameLayout.addView(callingUserPhotoView); + frameLayout.addView(gradientLayout, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT)); + frameLayout.addView(voIpSnowView = new VoIpSnowView(context), LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 220)); frameLayout.addView(callingUserTextureView); - final BackgroundGradientDrawable gradientDrawable = new BackgroundGradientDrawable(GradientDrawable.Orientation.TOP_BOTTOM, new int[]{0xFF1b354e, 0xFF255b7d}); final BackgroundGradientDrawable.Sizes sizes = BackgroundGradientDrawable.Sizes.ofDeviceScreen(BackgroundGradientDrawable.Sizes.Orientation.PORTRAIT); gradientDrawable.startDithering(sizes, new BackgroundGradientDrawable.ListenerAdapter() { @Override public void onAllSizesReady() { - callingUserPhotoView.invalidate(); - } - }); - overlayBackground = new VoIPOverlayBackground(context); - overlayBackground.setVisibility(View.GONE); - - callingUserPhotoView.getImageReceiver().setDelegate((imageReceiver, set, thumb, memCache) -> { - ImageReceiver.BitmapHolder bmp = imageReceiver.getBitmapSafe(); - if (bmp != null) { - overlayBackground.setBackground(bmp); + gradientLayout.invalidate(); } }); - callingUserPhotoView.setImage(ImageLocation.getForUserOrChat(callingUser, ImageLocation.TYPE_BIG), null, gradientDrawable, callingUser); - currentUserCameraFloatingLayout = new VoIPFloatingLayout(context); currentUserCameraFloatingLayout.setDelegate((progress, value) -> currentUserTextureView.setScreenshareMiniProgress(progress, value)); currentUserCameraFloatingLayout.setRelativePosition(1f, 1f); @@ -778,7 +829,7 @@ public void onAllSizesReady() { View backgroundView = new View(context); backgroundView.setBackgroundColor(0xff1b1f23); - callingUserMiniFloatingLayout.addView(backgroundView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT)); + //callingUserMiniFloatingLayout.addView(backgroundView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.MATCH_PARENT)); callingUserMiniFloatingLayout.addView(callingUserMiniTextureRenderer, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER)); callingUserMiniFloatingLayout.setOnTapListener(view -> { if (cameraForceExpanded && System.currentTimeMillis() - lastContentTapTime > 500) { @@ -796,17 +847,14 @@ public void onAllSizesReady() { frameLayout.addView(currentUserCameraFloatingLayout, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT)); frameLayout.addView(callingUserMiniFloatingLayout); - frameLayout.addView(overlayBackground); - bottomShadow = new View(context); bottomShadow.setBackground(new GradientDrawable(GradientDrawable.Orientation.TOP_BOTTOM, new int[]{Color.TRANSPARENT, ColorUtils.setAlphaComponent(Color.BLACK, (int) (255 * 0.5f))})); - frameLayout.addView(bottomShadow, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 140, Gravity.BOTTOM)); + frameLayout.addView(bottomShadow, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 160, Gravity.BOTTOM)); topShadow = new View(context); topShadow.setBackground(new GradientDrawable(GradientDrawable.Orientation.TOP_BOTTOM, new int[]{ColorUtils.setAlphaComponent(Color.BLACK, (int) (255 * 0.4f)), Color.TRANSPARENT})); - frameLayout.addView(topShadow, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 140, Gravity.TOP)); - + frameLayout.addView(topShadow, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 160, Gravity.TOP)); emojiLayout = new LinearLayout(context) { @Override @@ -824,22 +872,60 @@ public void onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo info) { return; } lastContentTapTime = System.currentTimeMillis(); + if (emojiExpanded) return; if (emojiLoaded) { expandEmoji(!emojiExpanded); } }); - emojiRationalTextView = new TextView(context); - emojiRationalTextView.setText(LocaleController.formatString("CallEmojiKeyTooltip", R.string.CallEmojiKeyTooltip, UserObject.getFirstName(callingUser))); - emojiRationalTextView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 16); + hideEmojiTextView = new HideEmojiTextView(context, backgroundProvider); + hideEmojiLayout = new FrameLayout(context); + hideEmojiLayout.addView(hideEmojiTextView, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.LEFT, 0, 16, 0, 0)); + hideEmojiLayout.setVisibility(View.GONE); + hideEmojiLayout.setOnClickListener(v -> { + if (System.currentTimeMillis() - lastContentTapTime < 500) { + return; + } + lastContentTapTime = System.currentTimeMillis(); + if (emojiLoaded) { + expandEmoji(!emojiExpanded); + } + }); + + emojiRationalLayout = new EmojiRationalLayout(context, backgroundProvider); + emojiRationalLayout.setOrientation(LinearLayout.VERTICAL); + + emojiRationalTopTextView = new TextView(context); + emojiRationalTopTextView.setText(LocaleController.getString("VoipCallEncryptionEndToEnd", R.string.VoipCallEncryptionEndToEnd)); + emojiRationalTopTextView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 15); + emojiRationalTopTextView.setTypeface(AndroidUtilities.getTypeface(AndroidUtilities.TYPEFACE_ROBOTO_MEDIUM)); + emojiRationalTopTextView.setTextColor(Color.WHITE); + emojiRationalTopTextView.setGravity(Gravity.CENTER); + + emojiRationalTextView = new TextView(context) { + @Override + protected void onLayout(boolean changed, int left, int top, int right, int bottom) { + super.onLayout(changed, left, top, right, bottom); + if (changed) { + updateViewState(); + } + } + }; + emojiRationalTextView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 15); emojiRationalTextView.setTextColor(Color.WHITE); emojiRationalTextView.setGravity(Gravity.CENTER); - emojiRationalTextView.setVisibility(View.GONE); + CharSequence ellipsizeName = TextUtils.ellipsize(UserObject.getFirstName(callingUser), emojiRationalTextView.getPaint(), dp(300), TextUtils.TruncateAt.END); + emojiRationalTextView.setText(LocaleController.formatString("CallEmojiKeyTooltip", R.string.CallEmojiKeyTooltip, ellipsizeName)); + + emojiRationalLayout.setVisibility(View.GONE); + emojiRationalLayout.addView(emojiRationalTopTextView); + emojiRationalLayout.addView(emojiRationalTextView, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, 0, 8, 0, 0)); + emojiRationalLayout.setPadding(AndroidUtilities.dp(18), AndroidUtilities.dp(80), AndroidUtilities.dp(18), AndroidUtilities.dp(18)); for (int i = 0; i < 4; i++) { - emojiViews[i] = new ImageView(context); - emojiViews[i].setScaleType(ImageView.ScaleType.FIT_XY); - emojiLayout.addView(emojiViews[i], LayoutHelper.createLinear(22, 22, i == 0 ? 0 : 4, 0, 0, 0)); + emojiViews[i] = new BackupImageView(context); + emojiViews[i].getImageReceiver().setAspectFit(true); + emojiLayout.addView(emojiViews[i], LayoutHelper.createLinear(25, 25, i == 0 ? 0 : 6, 0, 0, 0)); } statusLayout = new LinearLayout(context) { @Override @@ -871,23 +957,26 @@ public void onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo info) { statusLayout.setFocusable(true); statusLayout.setFocusableInTouchMode(true); - callingUserPhotoViewMini = new BackupImageView(context); - callingUserPhotoViewMini.setImage(ImageLocation.getForUserOrChat(callingUser, ImageLocation.TYPE_SMALL), null, Theme.createCircleDrawable(AndroidUtilities.dp(135), 0xFF000000), callingUser); + callingUserPhotoViewMini = new ImageWithWavesView(context); + AvatarDrawable avatarDrawable = new AvatarDrawable(); + avatarDrawable.setInfo(callingUser); + callingUserPhotoViewMini.setImage(ImageLocation.getForUserOrChat(callingUser, ImageLocation.TYPE_BIG), null, avatarDrawable, callingUser); callingUserPhotoViewMini.setRoundRadius(AndroidUtilities.dp(135) / 2); - callingUserPhotoViewMini.setVisibility(View.GONE); callingUserTitle = new TextView(context); - callingUserTitle.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 24); + callingUserTitle.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 28); CharSequence name = ContactsController.formatName(callingUser.first_name, callingUser.last_name); name = Emoji.replaceEmoji(name, callingUserTitle.getPaint().getFontMetricsInt(), AndroidUtilities.dp(20), false); callingUserTitle.setText(name); + callingUserTitle.setMaxLines(2); + callingUserTitle.setEllipsize(TextUtils.TruncateAt.END); callingUserTitle.setShadowLayer(AndroidUtilities.dp(3), 0, AndroidUtilities.dp(.666666667f), 0x4C000000); callingUserTitle.setTextColor(Color.WHITE); callingUserTitle.setGravity(Gravity.CENTER_HORIZONTAL); callingUserTitle.setImportantForAccessibility(View.IMPORTANT_FOR_ACCESSIBILITY_NO); - statusLayout.addView(callingUserTitle, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_HORIZONTAL, 0, 0, 0, 6)); + statusLayout.addView(callingUserTitle, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_HORIZONTAL, 8, 0, 8, 6)); - statusTextView = new VoIPStatusTextView(context); + statusTextView = new VoIPStatusTextView(context, backgroundProvider); ViewCompat.setImportantForAccessibility(statusTextView, ViewCompat.IMPORTANT_FOR_ACCESSIBILITY_NO_HIDE_DESCENDANTS); statusLayout.addView(statusTextView, LayoutHelper.createLinear(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_HORIZONTAL, 0, 0, 0, 6)); @@ -895,18 +984,56 @@ public void onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo info) { statusLayout.setClipToPadding(false); statusLayout.setPadding(0, 0, 0, AndroidUtilities.dp(15)); - frameLayout.addView(callingUserPhotoViewMini, LayoutHelper.createFrame(135, 135, Gravity.CENTER_HORIZONTAL, 0, 68, 0, 0)); - frameLayout.addView(statusLayout, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, 0, 0, 68, 0, 0)); - frameLayout.addView(emojiLayout, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_HORIZONTAL, 0, 17, 0, 0)); - frameLayout.addView(emojiRationalTextView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER, 24, 32, 24, 0)); + endCloseLayout = new EndCloseLayout(context); + rateCallLayout = new RateCallLayout(context, backgroundProvider); + endCloseLayout.setAlpha(0f); + rateCallLayout.setVisibility(View.GONE); + + frameLayout.addView(callingUserPhotoViewMini, LayoutHelper.createFrame(204, 204, Gravity.CENTER_HORIZONTAL, 0, 93, 0, 0)); + frameLayout.addView(statusLayout, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, 0, 0, 135, 0, 0)); + frameLayout.addView(hideEmojiLayout, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_HORIZONTAL, 0, 0, 0, 0)); + frameLayout.addView(emojiRationalLayout, LayoutHelper.createFrame(304, LayoutHelper.WRAP_CONTENT, Gravity.TOP | Gravity.CENTER_HORIZONTAL, 0, 118, 0, 0)); + frameLayout.addView(emojiLayout, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_HORIZONTAL, 0, 0, 0, 0)); + frameLayout.addView(endCloseLayout, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 52, Gravity.RIGHT, 0, 0, 0, 0)); + frameLayout.addView(rateCallLayout, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.LEFT, 0, 380, 0, 0)); buttonsLayout = new VoIPButtonsLayout(context); - for (int i = 0; i < 4; i++) { - bottomButtons[i] = new VoIPToggleButton(context); - buttonsLayout.addView(bottomButtons[i]); - } + bottomSpeakerBtn = new VoIpSwitchLayout(context, backgroundProvider); + bottomVideoBtn = new VoIpSwitchLayout(context, backgroundProvider); + bottomMuteBtn = new VoIpSwitchLayout(context, backgroundProvider); + bottomEndCallBtn = new VoIPToggleButton(context) { + @Override + protected void dispatchSetPressed(boolean pressed) { + super.dispatchSetPressed(pressed); + setPressedBtn(pressed); + } + }; + int startDelay = 150; + bottomSpeakerBtn.setTranslationY(AndroidUtilities.dp(100)); + bottomSpeakerBtn.setScaleX(0f); + bottomSpeakerBtn.setScaleY(0f); + bottomSpeakerBtn.animate().setStartDelay(startDelay).translationY(0).scaleY(1f).scaleX(1f).setDuration(250).start(); + bottomVideoBtn.setTranslationY(AndroidUtilities.dp(100)); + bottomVideoBtn.setScaleX(0f); + bottomVideoBtn.setScaleY(0f); + bottomVideoBtn.animate().setStartDelay(startDelay + 16).translationY(0).scaleY(1f).scaleX(1f).setDuration(250).start(); + bottomMuteBtn.setTranslationY(AndroidUtilities.dp(100)); + bottomMuteBtn.setScaleX(0f); + bottomMuteBtn.setScaleY(0f); + bottomMuteBtn.animate().setStartDelay(startDelay + 32).translationY(0).scaleY(1f).scaleX(1f).setDuration(250).start(); + bottomEndCallBtn.setTranslationY(AndroidUtilities.dp(100)); + bottomEndCallBtn.setScaleX(0f); + bottomEndCallBtn.setScaleY(0f); + bottomEndCallBtn.animate().setStartDelay(startDelay + 48).translationY(0).scaleY(1f).scaleX(1f).setDuration(250).start(); + + buttonsLayout.addView(bottomSpeakerBtn); + buttonsLayout.addView(bottomVideoBtn); + buttonsLayout.addView(bottomMuteBtn); + buttonsLayout.addView(bottomEndCallBtn); + acceptDeclineView = new AcceptDeclineView(context); acceptDeclineView.setListener(new AcceptDeclineView.Listener() { + @Override public void onAccept() { if (currentState == VoIPService.STATE_BUSY) { @@ -927,10 +1054,14 @@ public void onAccept() { activity.requestPermissions(new String[]{Manifest.permission.RECORD_AUDIO}, 101); } else { if (VoIPService.getSharedInstance() != null) { - VoIPService.getSharedInstance().acceptIncomingCall(); - if (currentUserIsVideo) { - VoIPService.getSharedInstance().requestVideoCall(false); - } + runAcceptCallAnimation(() -> { + if (VoIPService.getSharedInstance() != null) { + VoIPService.getSharedInstance().acceptIncomingCall(); + if (currentUserIsVideo) { + VoIPService.getSharedInstance().requestVideoCall(false); + } + } + }); } } } @@ -947,14 +1078,16 @@ public void onDecline() { } } }); - acceptDeclineView.setScreenWasWakeup(screenWasWakeup); + acceptDeclineView.setScaleX(1.15f); + acceptDeclineView.setScaleY(1.15f); frameLayout.addView(buttonsLayout, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, LayoutHelper.WRAP_CONTENT, Gravity.BOTTOM)); - frameLayout.addView(acceptDeclineView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 186, Gravity.BOTTOM)); + int horizontalMargin = isTablet() ? 100 : 27; + frameLayout.addView(acceptDeclineView, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 186, Gravity.BOTTOM, horizontalMargin, 0, horizontalMargin, 0)); backIcon = new ImageView(context); backIcon.setBackground(Theme.createSelectorDrawable(ColorUtils.setAlphaComponent(Color.WHITE, (int) (255 * 0.3f)))); - backIcon.setImageResource(R.drawable.ic_ab_back); + backIcon.setImageResource(R.drawable.msg_call_minimize_shadow); backIcon.setPadding(AndroidUtilities.dp(16), AndroidUtilities.dp(16), AndroidUtilities.dp(16), AndroidUtilities.dp(16)); backIcon.setContentDescription(LocaleController.getString("Back", R.string.Back)); frameLayout.addView(backIcon, LayoutHelper.createFrame(56, 56, Gravity.TOP | Gravity.LEFT)); @@ -975,12 +1108,23 @@ public void onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo info) { speakerPhoneIcon.setBackground(Theme.createSelectorDrawable(ColorUtils.setAlphaComponent(Color.WHITE, (int) (255 * 0.3f)))); speakerPhoneIcon.setPadding(AndroidUtilities.dp(12), AndroidUtilities.dp(12), AndroidUtilities.dp(12), AndroidUtilities.dp(12)); frameLayout.addView(speakerPhoneIcon, LayoutHelper.createFrame(56, 56, Gravity.TOP | Gravity.RIGHT)); + speakerPhoneIcon.setAlpha(0f); speakerPhoneIcon.setOnClickListener(view -> { if (speakerPhoneIcon.getTag() == null) { return; } - if (VoIPService.getSharedInstance() != null) { - VoIPService.getSharedInstance().toggleSpeakerphoneOrShowRouteSheet(activity, false); + VoIPService service = VoIPService.getSharedInstance(); + if (service != null) { + startWaitingFoHideUi(); + final int selectedSpeaker; + if (service.isBluetoothOn()) { + selectedSpeaker = 2; + } else if (service.isSpeakerphoneOn()) { + selectedSpeaker = 0; + } else { + selectedSpeaker = 1; + } + service.toggleSpeakerphoneOrShowRouteSheet(activity, false, selectedSpeaker); } }); @@ -993,7 +1137,7 @@ public void onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo info) { backIcon.setVisibility(View.GONE); } - notificationsLayout = new VoIPNotificationsLayout(context); + notificationsLayout = new VoIPNotificationsLayout(context, backgroundProvider); notificationsLayout.setGravity(Gravity.BOTTOM); notificationsLayout.setOnViewsUpdated(() -> { previousState = currentState; @@ -1001,11 +1145,30 @@ public void onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo info) { }); frameLayout.addView(notificationsLayout, LayoutHelper.createFrame(LayoutHelper.MATCH_PARENT, 200, Gravity.BOTTOM, 16, 0, 16, 0)); - tapToVideoTooltip = new HintView(context, 4); + tapToVideoTooltip = new VoIpHintView(context, HintView2.DIRECTION_BOTTOM, backgroundProvider, true) + .setMultilineText(true) + .setTextAlign(Layout.Alignment.ALIGN_CENTER) + .setDuration(-1) + .setOnHiddenListener(this::startWaitingFoHideUi) + .setHideByTouch(true) + .setMaxWidth(320) + .useScale(true) + .setInnerPadding(10, 6, 10, 6) + .setRounding(8); tapToVideoTooltip.setText(LocaleController.getString("TapToTurnCamera", R.string.TapToTurnCamera)); - frameLayout.addView(tapToVideoTooltip, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.LEFT | Gravity.TOP, 19, 0, 19, 8)); - tapToVideoTooltip.setBottomOffset(AndroidUtilities.dp(4)); - tapToVideoTooltip.setVisibility(View.GONE); + frameLayout.addView(tapToVideoTooltip, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.BOTTOM, 19, 0, 19, 0)); + + encryptionTooltip = new VoIpHintView(context, HintView2.DIRECTION_TOP, backgroundProvider, false) + .setMultilineText(true) + .setTextAlign(Layout.Alignment.ALIGN_CENTER) + .setDuration(4000) + .setHideByTouch(true) + .setMaxWidth(320) + .useScale(true) + .setInnerPadding(10, 6, 10, 6) + .setRounding(8); + encryptionTooltip.setText(LocaleController.getString("VoipHintEncryptionKey", R.string.VoipHintEncryptionKey)); + frameLayout.addView(encryptionTooltip, LayoutHelper.createFrame(LayoutHelper.WRAP_CONTENT, LayoutHelper.WRAP_CONTENT, Gravity.CENTER_HORIZONTAL, 0, 0, 0, 0)); updateViewState(); @@ -1020,6 +1183,105 @@ public void onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo info) { return frameLayout; } + private void runAcceptCallAnimation(final Runnable after) { + if (bottomVideoBtn.getVisibility() == View.VISIBLE) { + int[] loc = new int[2]; + acceptDeclineView.getLocationOnScreen(loc); + //callingUserPhotoView.switchToCallConnected(loc[0] + AndroidUtilities.dp(82), loc[1] + AndroidUtilities.dp(74)); + acceptDeclineView.stopAnimations(); + after.run(); + return; + } + + bottomEndCallBtn.animate().cancel(); + bottomSpeakerBtn.animate().cancel(); + bottomMuteBtn.animate().cancel(); + bottomVideoBtn.animate().cancel(); + int[] loc = new int[2]; + acceptDeclineView.getLocationOnScreen(loc); + acceptDeclineView.stopAnimations(); + //callingUserPhotoView.switchToCallConnected(loc[0] + AndroidUtilities.dp(82), loc[1] + AndroidUtilities.dp(74)); + bottomEndCallBtn.setData(R.drawable.calls_decline, Color.WHITE, 0xFFF01D2C, LocaleController.getString("VoipEndCall2", R.string.VoipEndCall2), false, false); + bottomSpeakerBtn.setType(VoIpSwitchLayout.Type.SPEAKER, false); + bottomMuteBtn.setType(VoIpSwitchLayout.Type.MICRO, false); + bottomVideoBtn.setType(VoIpSwitchLayout.Type.VIDEO, true); + bottomEndCallBtn.setVisibility(View.VISIBLE); + bottomSpeakerBtn.setVisibility(View.VISIBLE); + bottomMuteBtn.setVisibility(View.VISIBLE); + bottomVideoBtn.setVisibility(View.VISIBLE); + bottomEndCallBtn.setAlpha(0f); + bottomSpeakerBtn.setAlpha(0f); + bottomMuteBtn.setAlpha(0f); + bottomVideoBtn.setAlpha(0f); + final ViewGroup.MarginLayoutParams lp = ((ViewGroup.MarginLayoutParams) acceptDeclineView.getLayoutParams()); + final int startMargin = lp.getMarginEnd(); + final int endMargin = AndroidUtilities.dp(52); + final int endMarginFinal = AndroidUtilities.dp(24); + final int transitionY = AndroidUtilities.dp(62); + + AnimatorSet animatorSet = new AnimatorSet(); + ValueAnimator marginAnimator = ValueAnimator.ofFloat(0f, 1f); + marginAnimator.addUpdateListener(valueAnimator -> { + float percent = (float) valueAnimator.getAnimatedValue(); + float diff = transitionY * percent; + acceptDeclineView.setTranslationY(diff); + diff = startMargin - ((startMargin + endMarginFinal) * percent); + lp.leftMargin = (int) diff; + lp.rightMargin = (int) diff; + acceptDeclineView.requestLayout(); + }); + final int totalDuration = 400; + ObjectAnimator scaleXAnimator = ObjectAnimator.ofFloat(acceptDeclineView, View.SCALE_X, acceptDeclineView.getScaleX(), 1f, 1f, 1f); + ObjectAnimator scaleYAnimator = ObjectAnimator.ofFloat(acceptDeclineView, View.SCALE_Y, acceptDeclineView.getScaleY(), 1f, 1f, 1f); + ObjectAnimator alphaAnimator = ObjectAnimator.ofFloat(acceptDeclineView, View.ALPHA, acceptDeclineView.getAlpha(), acceptDeclineView.getAlpha(), 0f, 0f); + animatorSet.playTogether(marginAnimator, scaleXAnimator, scaleYAnimator, alphaAnimator); + animatorSet.setDuration(totalDuration); + animatorSet.setInterpolator(new LinearInterpolator()); + animatorSet.addListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + after.run(); + acceptDeclineView.setScaleX(1.15f); + acceptDeclineView.setScaleY(1.15f); + final ViewGroup.MarginLayoutParams lp = ((ViewGroup.MarginLayoutParams) acceptDeclineView.getLayoutParams()); + lp.leftMargin = AndroidUtilities.dp(10); + lp.rightMargin = AndroidUtilities.dp(10); + acceptDeclineView.setVisibility(View.GONE); + } + }); + animatorSet.start(); + + AndroidUtilities.runOnUIThread(() -> { + int[] location = new int[2]; + acceptDeclineView.getLocationOnScreen(location); + int rootX = location[0]; + int rootY = location[1]; + bottomSpeakerBtn.getLocationOnScreen(location); + bottomSpeakerBtn.setTranslationX(rootX - location[0] + AndroidUtilities.dp(42)); + bottomSpeakerBtn.setTranslationY(rootY - location[1] + AndroidUtilities.dp(44)); + bottomMuteBtn.getLocationOnScreen(location); + bottomMuteBtn.setTranslationX(rootX - location[0] + AndroidUtilities.dp(42)); + bottomMuteBtn.setTranslationY(rootY - location[1] + AndroidUtilities.dp(44)); + bottomVideoBtn.getLocationOnScreen(location); + bottomVideoBtn.setTranslationX(rootX - location[0] + AndroidUtilities.dp(42)); + bottomVideoBtn.setTranslationY(rootY - location[1] + AndroidUtilities.dp(44)); + bottomEndCallBtn.getLocationOnScreen(location); + bottomEndCallBtn.setTranslationX(rootX + acceptDeclineView.getWidth() - location[0] - AndroidUtilities.dp(49) - AndroidUtilities.dp(60)); + bottomEndCallBtn.setTranslationY(rootY - location[1] + AndroidUtilities.dp(44)); + + bottomEndCallBtn.setAlpha(1f); + bottomSpeakerBtn.setAlpha(1f); + bottomMuteBtn.setAlpha(1f); + bottomVideoBtn.setAlpha(1f); + + int halfDuration = totalDuration / 2; + bottomEndCallBtn.animate().setStartDelay(0).translationY(0f).setInterpolator(new LinearInterpolator()).translationX(0f).setDuration(halfDuration).start(); + bottomSpeakerBtn.animate().setStartDelay(0).translationY(0f).setInterpolator(new LinearInterpolator()).translationX(0f).setDuration(halfDuration).start(); + bottomMuteBtn.animate().setStartDelay(0).translationY(0f).setInterpolator(new LinearInterpolator()).translationX(0f).setDuration(halfDuration).start(); + bottomVideoBtn.animate().setStartDelay(0).translationY(0f).setInterpolator(new LinearInterpolator()).translationX(0f).setDuration(halfDuration).start(); + }, totalDuration / 3); + } + private boolean checkPointerIds(MotionEvent ev) { if (ev.getPointerCount() < 2) { return false; @@ -1110,9 +1372,6 @@ public void switchToPip() { isFinished = true; if (VoIPService.getSharedInstance() != null) { int h = instance.windowView.getMeasuredHeight(); - if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT_WATCH && instance.lastInsets != null) { - h -= instance.lastInsets.getSystemWindowInsetBottom(); - } VoIPPiPView.show(instance.activity, instance.currentAccount, instance.windowView.getMeasuredWidth(), h, VoIPPiPView.ANIMATION_ENTER_TYPE_TRANSITION); if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT_WATCH && instance.lastInsets != null) { VoIPPiPView.topInset = instance.lastInsets.getSystemWindowInsetTop(); @@ -1186,7 +1445,6 @@ public void startTransitionFromPiP() { topShadow.setAlpha(0f); speakerPhoneIcon.setAlpha(0f); notificationsLayout.setAlpha(0f); - callingUserPhotoView.setAlpha(0f); currentUserCameraFloatingLayout.switchingToPip = true; AndroidUtilities.runOnUIThread(() -> { @@ -1201,7 +1459,6 @@ public void startTransitionFromPiP() { bottomShadow.animate().alpha(1f).setDuration(350).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); topShadow.animate().alpha(1f).setDuration(350).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); notificationsLayout.animate().alpha(1f).setDuration(350).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); - callingUserPhotoView.animate().alpha(1f).setDuration(350).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); animator.addListener(new AnimatorListenerAdapter() { @Override @@ -1285,12 +1542,6 @@ public Animator createPiPTransition(boolean enter) { callingUserTextureView.setTranslationX(callingUserToX); callingUserTextureView.setTranslationY(callingUserToY); callingUserTextureView.setRoundCorners(AndroidUtilities.dp(6) * 1f / callingUserToScale); - - callingUserPhotoView.setAlpha(0f); - callingUserPhotoView.setScaleX(callingUserToScale); - callingUserPhotoView.setScaleY(callingUserToScale); - callingUserPhotoView.setTranslationX(callingUserToX); - callingUserPhotoView.setTranslationY(callingUserToY); } ValueAnimator animator = ValueAnimator.ofFloat(enter ? 1f : 0, enter ? 0 : 1f); @@ -1327,11 +1578,6 @@ public Animator createPiPTransition(boolean enter) { currentUserTextureView.setScreenshareMiniProgress(v, false); } windowView.invalidate(); - callingUserPhotoView.setScaleX(callingUserScale); - callingUserPhotoView.setScaleY(callingUserScale); - callingUserPhotoView.setTranslationX(tx); - callingUserPhotoView.setTranslationY(ty); - callingUserPhotoView.setAlpha(1f - v); }); return animator; } @@ -1342,56 +1588,88 @@ private void expandEmoji(boolean expanded) { } emojiExpanded = expanded; if (expanded) { - AndroidUtilities.runOnUIThread(hideUIRunnable); - hideUiRunnableWaiting = false; - float s1 = emojiLayout.getMeasuredWidth(); - float s2 = windowView.getMeasuredWidth() - AndroidUtilities.dp(128); - float scale = s2 / s1; - emojiLayout.animate().scaleX(scale).scaleY(scale) - .translationY(windowView.getHeight() / 2f - emojiLayout.getBottom()) - .setInterpolator(CubicBezierInterpolator.EASE_OUT_QUINT) - .setDuration(250) - .start(); - emojiRationalTextView.animate().setListener(null).cancel(); - if (emojiRationalTextView.getVisibility() != View.VISIBLE) { - emojiRationalTextView.setVisibility(View.VISIBLE); - emojiRationalTextView.setAlpha(0f); + if (SharedConfig.callEncryptionHintDisplayedCount < 2) { + SharedConfig.incrementCallEncryptionHintDisplayed(2); } - emojiRationalTextView.animate().alpha(1f).setDuration(150).start(); + encryptionTooltip.hide(); + AndroidUtilities.cancelRunOnUIThread(hideUIRunnable); + hideUiRunnableWaiting = false; - overlayBackground.animate().setListener(null).cancel(); - if (overlayBackground.getVisibility() != View.VISIBLE) { - overlayBackground.setVisibility(View.VISIBLE); - overlayBackground.setAlpha(0f); - overlayBackground.setShowBlackout(currentUserIsVideo || callingUserIsVideo, false); + if (callingUserPhotoViewMini.getVisibility() == View.VISIBLE) { + callingUserPhotoViewMini.animate().setStartDelay(0).translationY(AndroidUtilities.dp(48)).scaleY(0.1f).scaleX(0.1f).alpha(0f).setDuration(200).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); } - overlayBackground.animate().alpha(1f).setDuration(150).start(); + + hideEmojiLayout.animate().setListener(null).cancel(); + hideEmojiLayout.setVisibility(View.VISIBLE); + hideEmojiLayout.setAlpha(0f); + hideEmojiLayout.setScaleX(0.3f); + hideEmojiLayout.setScaleY(0.3f); + hideEmojiLayout.animate().alpha(1f).scaleY(1f).scaleX(1f).setDuration(340).setInterpolator(CubicBezierInterpolator.EASE_OUT).start(); + + emojiLayout.animate().scaleX(1.72f).scaleY(1.72f) + .translationY(AndroidUtilities.dp(140)) + .setInterpolator(CubicBezierInterpolator.DEFAULT) + .setDuration(400) + .start(); + + emojiRationalLayout.animate().setListener(null).cancel(); + emojiRationalLayout.setVisibility(View.VISIBLE); + emojiRationalLayout.setTranslationY(-AndroidUtilities.dp(120)); + emojiRationalLayout.setScaleX(0.7f); + emojiRationalLayout.setScaleY(0.7f); + emojiRationalLayout.setAlpha(0f); + emojiRationalLayout.animate().alpha(1f).translationY(0).scaleX(1f).scaleY(1f).setDuration(400).setListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + for (BackupImageView emojiView : emojiViews) { + if (emojiView.animatedEmojiDrawable != null && emojiView.animatedEmojiDrawable.getImageReceiver() != null) { + emojiView.animatedEmojiDrawable.getImageReceiver().setAllowStartAnimation(true); + emojiView.animatedEmojiDrawable.getImageReceiver().startAnimation(); + } + } + } + }).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); } else { + if (callingUserPhotoViewMini.getVisibility() == View.VISIBLE) { + callingUserPhotoViewMini.animate().setStartDelay(50).translationY(0).scaleX(1f).scaleY(1f).alpha(1f).setDuration(250).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); + } + + hideEmojiLayout.animate().setListener(null).cancel(); + hideEmojiLayout.animate().alpha(0f).scaleY(0.3f).scaleX(0.3f).setDuration(230).setInterpolator(CubicBezierInterpolator.DEFAULT).setListener(new HideViewAfterAnimation(hideEmojiLayout)).start(); + emojiLayout.animate().scaleX(1f).scaleY(1f) .translationY(0) .setInterpolator(CubicBezierInterpolator.DEFAULT) - .setDuration(150) + .setDuration(280) .start(); - if (emojiRationalTextView.getVisibility() != View.GONE) { - emojiRationalTextView.animate().alpha(0).setListener(new AnimatorListenerAdapter() { - @Override - public void onAnimationEnd(Animator animation) { - VoIPService service = VoIPService.getSharedInstance(); - if (canHideUI && !hideUiRunnableWaiting && service != null && !service.isMicMute()) { - AndroidUtilities.runOnUIThread(hideUIRunnable, 3000); - hideUiRunnableWaiting = true; + emojiRationalLayout.animate().setListener(null).cancel(); + emojiRationalLayout.animate().alpha(0f).scaleY(0.7f).scaleX(0.7f).translationY(-AndroidUtilities.dp(120)).setListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + startWaitingFoHideUi(); + for (BackupImageView emojiView : emojiViews) { + if (emojiView.animatedEmojiDrawable != null && emojiView.animatedEmojiDrawable.getImageReceiver() != null) { + emojiView.animatedEmojiDrawable.getImageReceiver().setAllowStartAnimation(false); + emojiView.animatedEmojiDrawable.getImageReceiver().stopAnimation(); } - emojiRationalTextView.setVisibility(View.GONE); } - }).setDuration(150).start(); + emojiRationalLayout.setVisibility(View.GONE); + } + }).setDuration(250).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); + } + previousState = currentState; + updateViewState(); + } - overlayBackground.animate().alpha(0).setListener(new AnimatorListenerAdapter() { - @Override - public void onAnimationEnd(Animator animation) { - overlayBackground.setVisibility(View.GONE); - } - }).setDuration(150).start(); + private void startWaitingFoHideUi() { + VoIPService service = VoIPService.getSharedInstance(); + if (service != null) { + AndroidUtilities.cancelRunOnUIThread(hideUIRunnable); + hideUiRunnableWaiting = false; + if (canHideUI && uiVisible) { + AndroidUtilities.runOnUIThread(hideUIRunnable, 3000); + hideUiRunnableWaiting = true; } } } @@ -1405,26 +1683,19 @@ private void updateViewState() { boolean showAcceptDeclineView = false; boolean showTimer = false; boolean showReconnecting = false; - boolean showCallingAvatarMini = false; int statusLayoutOffset = 0; - VoIPService service = VoIPService.getSharedInstance(); + final VoIPService service = VoIPService.getSharedInstance(); switch (currentState) { case VoIPService.STATE_WAITING_INCOMING: showAcceptDeclineView = true; - lockOnScreen = true; - statusLayoutOffset = AndroidUtilities.dp(24); + lockOnScreen = false; acceptDeclineView.setRetryMod(false); if (service != null && service.privateCall.video) { - if (currentUserIsVideo && callingUser.photo != null) { - showCallingAvatarMini = true; - } else { - showCallingAvatarMini = false; - } - statusTextView.setText(LocaleController.getString("VoipInVideoCallBranding", R.string.VoipInVideoCallBranding), true, animated); + statusTextView.setText(LocaleController.getString("VoipInVideoCallBranding", R.string.VoipInVideoCallBranding), false, animated); acceptDeclineView.setTranslationY(-AndroidUtilities.dp(60)); } else { - statusTextView.setText(LocaleController.getString("VoipInCallBranding", R.string.VoipInCallBranding), true, animated); + statusTextView.setText(LocaleController.getString("VoipInCallBranding", R.string.VoipInCallBranding), false, animated); acceptDeclineView.setTranslationY(0); } break; @@ -1433,13 +1704,17 @@ private void updateViewState() { statusTextView.setText(LocaleController.getString("VoipConnecting", R.string.VoipConnecting), true, animated); break; case VoIPService.STATE_EXCHANGING_KEYS: - statusTextView.setText(LocaleController.getString("VoipExchangingKeys", R.string.VoipExchangingKeys), true, animated); + if (previousState != VoIPService.STATE_EXCHANGING_KEYS) { + statusTextView.setText(LocaleController.getString("VoipExchangingKeys", R.string.VoipExchangingKeys), true, animated); + } break; case VoIPService.STATE_WAITING: statusTextView.setText(LocaleController.getString("VoipWaiting", R.string.VoipWaiting), true, animated); break; case VoIPService.STATE_RINGING: - statusTextView.setText(LocaleController.getString("VoipRinging", R.string.VoipRinging), true, animated); + if (previousState != VoIPService.STATE_RINGING) { + statusTextView.setText(LocaleController.getString("VoipRinging", R.string.VoipRinging), true, animated); + } break; case VoIPService.STATE_REQUESTING: statusTextView.setText(LocaleController.getString("VoipRequesting", R.string.VoipRequesting), true, animated); @@ -1456,14 +1731,96 @@ private void updateViewState() { case VoIPService.STATE_ESTABLISHED: case VoIPService.STATE_RECONNECTING: updateKeyView(animated); - showTimer = true; if (currentState == VoIPService.STATE_RECONNECTING) { - showReconnecting = true; + showReconnecting = wasEstablished; + if (!wasEstablished && previousState != VoIPService.STATE_RECONNECTING) { + statusTextView.setText(LocaleController.getString("VoipConnecting", R.string.VoipConnecting), true, animated); + } + } else { + wasEstablished = true; + showTimer = true; } break; case VoIPService.STATE_ENDED: + boolean hasRate = service != null && service.hasRate(); currentUserTextureView.saveCameraLastBitmap(); - AndroidUtilities.runOnUIThread(() -> windowView.finish(), 200); + if (hasRate && !isFinished) { + final boolean uiVisibleLocal = uiVisible; + if (uiVisibleLocal) { + int[] locEndCall = new int[2]; + int w = AndroidUtilities.displaySize.x; + bottomEndCallBtn.getLocationOnScreen(locEndCall); + int marginCloseBtn = w - locEndCall[0] - (((bottomEndCallBtn.getMeasuredWidth() - AndroidUtilities.dp(52)) / 2)) - AndroidUtilities.dp(52); + ViewGroup.MarginLayoutParams lpCloseBtn = (ViewGroup.MarginLayoutParams) endCloseLayout.getLayoutParams(); + lpCloseBtn.rightMargin = marginCloseBtn; + lpCloseBtn.leftMargin = marginCloseBtn; + endCloseLayout.setTranslationY(locEndCall[1]); + endCloseLayout.setAlpha(1f); + endCloseLayout.setLayoutParams(lpCloseBtn); + buttonsLayout.animate().alpha(0f).setDuration(80).start(); + AndroidUtilities.runOnUIThread(() -> endCloseLayout.switchToClose(v -> { + AndroidUtilities.runOnUIThread(() -> windowView.finish()); + if (selectedRating > 0) { + service.sendCallRating(selectedRating); + } + }, true), 2); + } else { + buttonsLayout.setVisibility(View.GONE); + FrameLayout.LayoutParams lpCloseBtn = (FrameLayout.LayoutParams) endCloseLayout.getLayoutParams(); + lpCloseBtn.rightMargin = AndroidUtilities.dp(18); + lpCloseBtn.leftMargin = AndroidUtilities.dp(18); + lpCloseBtn.bottomMargin = AndroidUtilities.dp(36); + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT_WATCH && lastInsets != null) { + lpCloseBtn.bottomMargin += lastInsets.getSystemWindowInsetBottom(); + } + lpCloseBtn.gravity = Gravity.BOTTOM; + endCloseLayout.setLayoutParams(lpCloseBtn); + endCloseLayout.animate().alpha(1f).setDuration(250).start(); + endCloseLayout.switchToClose(v -> { + AndroidUtilities.runOnUIThread(() -> windowView.finish()); + if (selectedRating > 0) { + service.sendCallRating(selectedRating); + } + }, false); + } + + rateCallLayout.setVisibility(View.VISIBLE); + rateCallLayout.show(count -> selectedRating = count); + if (emojiExpanded) { + emojiExpanded = false; + hideEmojiLayout.animate().alpha(0f).scaleY(0.3f).scaleX(0.3f).setDuration(250).setInterpolator(CubicBezierInterpolator.DEFAULT).setListener(new HideViewAfterAnimation(hideEmojiLayout)).start(); + emojiLayout.animate().scaleX(1f).scaleY(1f).translationY(0).setInterpolator(CubicBezierInterpolator.DEFAULT).setDuration(250).start(); + emojiRationalLayout.animate().alpha(0f).scaleY(0.7f).scaleX(0.7f).translationY(-AndroidUtilities.dp(120)).setListener(new HideViewAfterAnimation(hideEmojiLayout)).setDuration(250).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); + } + for (BackupImageView emojiView : emojiViews) { + emojiView.animate().alpha(0f).scaleX(0f).scaleY(0f).setInterpolator(CubicBezierInterpolator.DEFAULT).setDuration(250).start(); + } + callingUserTitle.animate().alpha(0f).setDuration(70).setListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + callingUserTitle.setText(LocaleController.getString("VoipCallEnded", R.string.VoipCallEnded)); + callingUserTitle.animate().alpha(1f).setDuration(70).setListener(null).start(); + } + }).start(); + speakerPhoneIcon.animate().alpha(0f).setInterpolator(CubicBezierInterpolator.DEFAULT).setDuration(250).start(); + speakerPhoneIcon.setVisibility(View.GONE); + statusTextView.showReconnect(false, true); + statusTextView.showBadConnection(false, true); + statusTextView.setDrawCallIcon(); + callingUserPhotoViewMini.onNeedRating(); + updateButtons(true); + bottomEndCallBtn.setVisibility(View.INVISIBLE); + callingUserMiniFloatingLayout.setAlpha(0f); + callingUserMiniFloatingLayout.setVisibility(View.GONE); + currentUserCameraFloatingLayout.setAlpha(0f); + currentUserCameraFloatingLayout.setVisibility(View.GONE); + if (previewDialog != null) { + previewDialog.dismiss(false, false); + } + notificationsLayout.animate().alpha(0f).setDuration(250).start(); + } else { + AndroidUtilities.runOnUIThread(() -> windowView.finish(), 200); + } break; case VoIPService.STATE_FAILED: statusTextView.setText(LocaleController.getString("VoipFailed", R.string.VoipFailed), false, animated); @@ -1533,6 +1890,7 @@ private void updateViewState() { return; } + boolean wasVideo = callingUserIsVideo || currentUserIsVideo; if (service != null) { callingUserIsVideo = service.getRemoteVideoState() == Instance.VIDEO_STATE_ACTIVE; currentUserIsVideo = service.getVideoState(false) == Instance.VIDEO_STATE_ACTIVE || service.getVideoState(false) == Instance.VIDEO_STATE_PAUSED; @@ -1548,7 +1906,7 @@ private void updateViewState() { if (callingUserIsVideo) { if (!switchingToPip) { - callingUserPhotoView.setAlpha(1f); + gradientLayout.setAlpha(1f); } if (animated) { callingUserTextureView.animate().alpha(1f).setDuration(250).start(); @@ -1562,10 +1920,9 @@ private void updateViewState() { } if (currentUserIsVideo || callingUserIsVideo) { - fillNavigationBar(true, animated); + gradientLayout.setVisibility(View.INVISIBLE); } else { - fillNavigationBar(false, animated); - callingUserPhotoView.setVisibility(View.VISIBLE); + gradientLayout.setVisibility(View.VISIBLE); if (animated) { callingUserTextureView.animate().alpha(0f).setDuration(250).start(); } else { @@ -1580,8 +1937,8 @@ private void updateViewState() { boolean showCallingUserVideoMini = currentUserIsVideo && cameraForceExpanded; - showCallingUserAvatarMini(showCallingAvatarMini, animated); - statusLayoutOffset += callingUserPhotoViewMini.getTag() == null ? 0 : AndroidUtilities.dp(135) + AndroidUtilities.dp(12); + showCallingUserAvatarMini(animated, wasVideo); + statusLayoutOffset = callingUserPhotoViewMini.getTag() == null ? 0 : AndroidUtilities.dp(135) + AndroidUtilities.dp(12); showAcceptDeclineView(showAcceptDeclineView, animated); windowView.setLockOnScreen(lockOnScreen || deviceIsLocked); canHideUI = (currentState == VoIPService.STATE_ESTABLISHED) && (currentUserIsVideo || callingUserIsVideo); @@ -1589,26 +1946,24 @@ private void updateViewState() { showUi(true); } - if (uiVisible && canHideUI && !hideUiRunnableWaiting && service != null && !service.isMicMute()) { + if (uiVisible && canHideUI && !hideUiRunnableWaiting && service != null) { AndroidUtilities.runOnUIThread(hideUIRunnable, 3000); hideUiRunnableWaiting = true; - } else if (service != null && service.isMicMute()) { - AndroidUtilities.cancelRunOnUIThread(hideUIRunnable); - hideUiRunnableWaiting = false; - } - if (!uiVisible) { - statusLayoutOffset -= AndroidUtilities.dp(50); } if (animated) { - if (lockOnScreen || !uiVisible) { - if (backIcon.getVisibility() != View.VISIBLE) { - backIcon.setVisibility(View.VISIBLE); - backIcon.setAlpha(0f); - } - backIcon.animate().alpha(0f).start(); + if (currentState == VoIPService.STATE_ENDED) { + backIcon.animate().alpha(0f).setInterpolator(CubicBezierInterpolator.DEFAULT).setDuration(250).start(); } else { - backIcon.animate().alpha(1f).start(); + if (lockOnScreen || !uiVisible) { + if (backIcon.getVisibility() != View.VISIBLE) { + backIcon.setVisibility(View.VISIBLE); + backIcon.setAlpha(0f); + } + backIcon.animate().alpha(0f).start(); + } else { + backIcon.animate().alpha(1f).start(); + } } notificationsLayout.animate().translationY(-AndroidUtilities.dp(16) - (uiVisible ? AndroidUtilities.dp(80) : 0)).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); } else { @@ -1629,16 +1984,40 @@ private void updateViewState() { statusTextView.showReconnect(showReconnecting, animated); + if (callingUserPhotoViewMini.getVisibility() == View.VISIBLE && emojiExpanded) { + statusLayoutOffset += AndroidUtilities.dp(24); + Layout layout = emojiRationalTextView.getLayout(); + if (layout != null) { + int lines = layout.getLineCount(); + if (lines > 2) { + statusLayoutOffset += AndroidUtilities.dp(20) * (lines - 2); + } + } + } + + if (currentState == VoIPService.STATE_ENDED && (!currentUserIsVideo && !callingUserIsVideo)) { + statusLayoutOffset -= AndroidUtilities.dp(24); + } + + if (currentUserIsVideo || callingUserIsVideo) { + statusLayoutOffset -= AndroidUtilities.dp(60); + } + if (animated) { + if (emojiExpanded && (currentUserIsVideo || callingUserIsVideo)) { + statusLayout.animate().setStartDelay(0).alpha(0f).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); + } else { + statusLayout.animate().setStartDelay(250).alpha(1f).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); + } if (statusLayoutOffset != statusLayoutAnimateToOffset) { - statusLayout.animate().translationY(statusLayoutOffset).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); + statusLayout.animate().setStartDelay(currentState == VoIPService.STATE_ENDED ? 250 : 0).translationY(statusLayoutOffset).setDuration(200).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); } } else { statusLayout.setTranslationY(statusLayoutOffset); } statusLayoutAnimateToOffset = statusLayoutOffset; - overlayBackground.setShowBlackout(currentUserIsVideo || callingUserIsVideo, animated); - canSwitchToPip = (currentState != VoIPService.STATE_ENDED && currentState != VoIPService.STATE_BUSY) && (currentUserIsVideo || callingUserIsVideo); + boolean isScreencast = service != null && service.isScreencast(); + canSwitchToPip = (currentState != VoIPService.STATE_ENDED && currentState != VoIPService.STATE_BUSY) && ((currentUserIsVideo && !isScreencast) || callingUserIsVideo); int floatingViewsOffset; if (service != null) { @@ -1652,20 +2031,25 @@ private void updateViewState() { if (animated) { notificationsLayout.beforeLayoutChanges(); } + if (service.isMicMute()) { + notificationsLayout.addNotification(R.drawable.calls_mute_mini, LocaleController.getString(R.string.VoipMyMicrophoneState), "self-muted", animated); + } else { + notificationsLayout.removeNotification("self-muted"); + } if ((currentUserIsVideo || callingUserIsVideo) && (currentState == VoIPService.STATE_ESTABLISHED || currentState == VoIPService.STATE_RECONNECTING) && service.getCallDuration() > 500) { if (service.getRemoteAudioState() == Instance.AUDIO_STATE_MUTED) { - notificationsLayout.addNotification(R.drawable.calls_mute_mini, LocaleController.formatString("VoipUserMicrophoneIsOff", R.string.VoipUserMicrophoneIsOff, UserObject.getFirstName(callingUser)), "muted", animated); + notificationsLayout.addNotification(R.drawable.calls_mute_mini, LocaleController.formatString("VoipUserMicrophoneIsOff", R.string.VoipUserMicrophoneIsOff, notificationsLayout.ellipsize(UserObject.getFirstName(callingUser))), "muted", animated); } else { notificationsLayout.removeNotification("muted"); } if (service.getRemoteVideoState() == Instance.VIDEO_STATE_INACTIVE) { - notificationsLayout.addNotification(R.drawable.calls_camera_mini, LocaleController.formatString("VoipUserCameraIsOff", R.string.VoipUserCameraIsOff, UserObject.getFirstName(callingUser)), "video", animated); + notificationsLayout.addNotification(R.drawable.calls_camera_mini, LocaleController.formatString("VoipUserCameraIsOff", R.string.VoipUserCameraIsOff, notificationsLayout.ellipsize(UserObject.getFirstName(callingUser))), "video", animated); } else { notificationsLayout.removeNotification("video"); } } else { if (service.getRemoteAudioState() == Instance.AUDIO_STATE_MUTED) { - notificationsLayout.addNotification(R.drawable.calls_mute_mini, LocaleController.formatString("VoipUserMicrophoneIsOff", R.string.VoipUserMicrophoneIsOff, UserObject.getFirstName(callingUser)), "muted", animated); + notificationsLayout.addNotification(R.drawable.calls_mute_mini, LocaleController.formatString("VoipUserMicrophoneIsOff", R.string.VoipUserMicrophoneIsOff, notificationsLayout.ellipsize(UserObject.getFirstName(callingUser))), "muted", animated); } else { notificationsLayout.removeNotification("muted"); } @@ -1674,7 +2058,9 @@ private void updateViewState() { if (notificationsLayout.getChildCount() == 0 && callingUserIsVideo && service.privateCall != null && !service.privateCall.video && !service.sharedUIParams.tapToVideoTooltipWasShowed) { service.sharedUIParams.tapToVideoTooltipWasShowed = true; - tapToVideoTooltip.showForView(bottomButtons[1], true); + tapToVideoTooltip.setTranslationY(-(fragmentView.getMeasuredHeight() - buttonsLayout.getY() + AndroidUtilities.dp(6))); + tapToVideoTooltip.setJointPx(0, buttonsLayout.getX() + bottomVideoBtn.getX() + AndroidUtilities.dp(14)); + tapToVideoTooltip.show(); } else if (notificationsLayout.getChildCount() != 0) { tapToVideoTooltip.hide(); } @@ -1710,7 +2096,14 @@ private void updateViewState() { callingUserMiniFloatingLayout.setScaleY(0.5f); } callingUserMiniFloatingLayout.animate().setListener(null).cancel(); - callingUserMiniFloatingLayout.animate().alpha(1f).scaleX(1f).scaleY(1f).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).setStartDelay(150).start(); + callingUserMiniFloatingLayout.isAppearing = true; + callingUserMiniFloatingLayout.animate() + .alpha(1f).scaleX(1f).scaleY(1f) + .setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).setStartDelay(150) + .withEndAction(() -> { + callingUserMiniFloatingLayout.isAppearing = false; + callingUserMiniFloatingLayout.invalidate(); + }).start(); callingUserMiniFloatingLayout.setTag(1); } else if (!showCallingUserVideoMini && callingUserMiniFloatingLayout.getTag() != null) { callingUserMiniFloatingLayout.setIsActive(false); @@ -1729,29 +2122,42 @@ public void onAnimationEnd(Animator animation) { callingUserMiniFloatingLayout.restoreRelativePosition(); updateSpeakerPhoneIcon(); - } - private void fillNavigationBar(boolean fill, boolean animated) { - if (switchingToPip) { - return; + if (currentState == VoIPService.STATE_ESTABLISHED) { + callingUserPhotoViewMini.onConnected(); + if (!gradientLayout.isConnectedCalled()) { + int[] loc = new int[2]; + callingUserPhotoViewMini.getLocationOnScreen(loc); + boolean animatedSwitch = previousState != -1; + gradientLayout.switchToCallConnected(loc[0] + AndroidUtilities.dp(106), loc[1] + AndroidUtilities.dp(106), animatedSwitch); + } + } + boolean isVideoMode = currentUserIsVideo || callingUserIsVideo; + voIpSnowView.setState(isVideoMode); + backgroundProvider.setHasVideo(isVideoMode); + + if (callingUserIsVideo && !wasVideo && isNearEar) { + isNearEar = false; + if (service != null) { + service.playStartRecordSound(); + } + } + + if (!isVideoMode) { + if (topShadow.getVisibility() != View.INVISIBLE) { + topShadow.setVisibility(View.INVISIBLE); + bottomShadow.setVisibility(View.INVISIBLE); + } + } else { + if (topShadow.getVisibility() != View.VISIBLE) { + topShadow.setVisibility(View.VISIBLE); + bottomShadow.setVisibility(View.VISIBLE); + } + } + AndroidUtilities.cancelRunOnUIThread(stopAnimatingBgRunnable); + if (currentState == VoIPService.STATE_ESTABLISHED) { + AndroidUtilities.runOnUIThread(stopAnimatingBgRunnable, 10000); } - if (!animated) { - if (naviagtionBarAnimator != null) { - naviagtionBarAnimator.cancel(); - } - fillNaviagtionBarValue = fill ? 1 : 0; - overlayBottomPaint.setColor(ColorUtils.setAlphaComponent(Color.BLACK, (int) (255 * (fill ? 1f : 0.5f)))); - } else if (fill != fillNaviagtionBar) { - if (naviagtionBarAnimator != null) { - naviagtionBarAnimator.cancel(); - } - naviagtionBarAnimator = ValueAnimator.ofFloat(fillNaviagtionBarValue, fill ? 1 : 0); - naviagtionBarAnimator.addUpdateListener(navigationBarAnimationListener); - naviagtionBarAnimator.setDuration(300); - naviagtionBarAnimator.setInterpolator(new LinearInterpolator()); - naviagtionBarAnimator.start(); - } - fillNaviagtionBar = fill; } private void showUi(boolean show) { @@ -1759,12 +2165,15 @@ private void showUi(boolean show) { uiVisibilityAnimator.cancel(); } + int notificationsLayoutStartDelay = 0; if (!show && uiVisible) { - speakerPhoneIcon.animate().alpha(0).translationY(-AndroidUtilities.dp(50)).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); - backIcon.animate().alpha(0).translationY(-AndroidUtilities.dp(50)).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); - emojiLayout.animate().alpha(0).translationY(-AndroidUtilities.dp(50)).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); - statusLayout.animate().alpha(0).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); - buttonsLayout.animate().alpha(0).translationY(AndroidUtilities.dp(50)).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); + notificationsLayoutStartDelay = 150; + speakerPhoneIcon.animate().alpha(0).translationY(-AndroidUtilities.dp(10)).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); + backIcon.animate().alpha(0).translationY(-AndroidUtilities.dp(10)).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); + emojiLayout.animate().alpha(0).translationY(-AndroidUtilities.dp(10)).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); + callingUserTitle.animate().alpha(0).setDuration(150).translationY(-AndroidUtilities.dp(10)).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); + statusTextView.animate().alpha(0).setDuration(150).translationY(-AndroidUtilities.dp(10)).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); + buttonsLayout.animate().alpha(0).translationY(AndroidUtilities.dp(10)).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); bottomShadow.animate().alpha(0).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); topShadow.animate().alpha(0).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); uiVisibilityAnimator = ValueAnimator.ofFloat(uiVisibilityAlpha, 0); @@ -1774,12 +2183,15 @@ private void showUi(boolean show) { AndroidUtilities.cancelRunOnUIThread(hideUIRunnable); hideUiRunnableWaiting = false; buttonsLayout.setEnabled(false); + encryptionTooltip.hide(); } else if (show && !uiVisible) { tapToVideoTooltip.hide(); + encryptionTooltip.hide(); + callingUserTitle.animate().alpha(1f).setDuration(150).translationY(0).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); + statusTextView.animate().alpha(1f).setDuration(150).translationY(0).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); speakerPhoneIcon.animate().alpha(1f).translationY(0).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); backIcon.animate().alpha(1f).translationY(0).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); emojiLayout.animate().alpha(1f).translationY(0).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); - statusLayout.animate().alpha(1f).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); buttonsLayout.animate().alpha(1f).translationY(0).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); bottomShadow.animate().alpha(1f).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); topShadow.animate().alpha(1f).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); @@ -1792,7 +2204,7 @@ private void showUi(boolean show) { uiVisible = show; windowView.requestFullscreen(!show); - notificationsLayout.animate().translationY(-AndroidUtilities.dp(16) - (uiVisible ? AndroidUtilities.dp(80) : 0)).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); + notificationsLayout.animate().translationY(-AndroidUtilities.dp(16) - (uiVisible ? AndroidUtilities.dp(80) : 0)).setDuration(150).setStartDelay(notificationsLayoutStartDelay).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); } private void showFloatingLayout(int state, boolean animated) { @@ -1877,17 +2289,33 @@ public void onAnimationEnd(Animator animation) { currentUserCameraFloatingLayout.setTag(state); } - private void showCallingUserAvatarMini(boolean show, boolean animated) { + private void showCallingUserAvatarMini(boolean animated, boolean wasVideo) { + boolean noVideo = !currentUserIsVideo && !callingUserIsVideo; if (animated) { - if (show && callingUserPhotoViewMini.getTag() == null) { + if (noVideo && callingUserPhotoViewMini.getTag() == null) { callingUserPhotoViewMini.animate().setListener(null).cancel(); callingUserPhotoViewMini.setVisibility(View.VISIBLE); - callingUserPhotoViewMini.setAlpha(0); - callingUserPhotoViewMini.setTranslationY(-AndroidUtilities.dp(135)); - callingUserPhotoViewMini.animate().alpha(1f).translationY(0).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); - } else if (!show && callingUserPhotoViewMini.getTag() != null) { + if (!emojiExpanded) { + if (wasVideo) { + callingUserPhotoViewMini.setAlpha(0f); + callingUserPhotoViewMini.animate().alpha(1f).translationY(0).scaleY(1f).scaleX(1f).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); + } else { + callingUserPhotoViewMini.setAlpha(0); + callingUserPhotoViewMini.setTranslationY(-AndroidUtilities.dp(135)); + callingUserPhotoViewMini.animate().alpha(1f).translationY(0).scaleY(1f).scaleX(1f).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT).start(); + } + } else { + if (wasVideo) { + callingUserPhotoViewMini.setAlpha(0f); + callingUserPhotoViewMini.setTranslationY(AndroidUtilities.dp(48)); + callingUserPhotoViewMini.setScaleX(0.1f); + callingUserPhotoViewMini.setScaleY(0.1f); + } + } + } else if (!noVideo && callingUserPhotoViewMini.getTag() != null) { callingUserPhotoViewMini.animate().setListener(null).cancel(); - callingUserPhotoViewMini.animate().alpha(0).translationY(-AndroidUtilities.dp(135)).setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT) + callingUserPhotoViewMini.setTranslationY(0); + callingUserPhotoViewMini.animate().alpha(0).setDuration(150).scaleX(0.1f).scaleY(0.1f).setInterpolator(CubicBezierInterpolator.DEFAULT) .setListener(new AnimatorListenerAdapter() { @Override public void onAnimationEnd(Animator animation) { @@ -1899,9 +2327,11 @@ public void onAnimationEnd(Animator animation) { callingUserPhotoViewMini.animate().setListener(null).cancel(); callingUserPhotoViewMini.setTranslationY(0); callingUserPhotoViewMini.setAlpha(1f); - callingUserPhotoViewMini.setVisibility(show ? View.VISIBLE : View.GONE); + callingUserPhotoViewMini.setScaleX(1f); + callingUserPhotoViewMini.setScaleY(1f); + callingUserPhotoViewMini.setVisibility(noVideo ? View.VISIBLE : View.GONE); } - callingUserPhotoViewMini.setTag(show ? 1 : null); + callingUserPhotoViewMini.setTag(noVideo ? 1 : null); } private void updateKeyView(boolean animated) { @@ -1926,18 +2356,39 @@ private void updateKeyView(boolean animated) { } byte[] sha256 = Utilities.computeSHA256(auth_key, 0, auth_key.length); String[] emoji = EncryptionKeyEmojifier.emojifyForCall(sha256); + + List documents = new ArrayList<>(); + List drawables = new ArrayList<>(); for (int i = 0; i < 4; i++) { Emoji.preloadEmoji(emoji[i]); Emoji.EmojiDrawable drawable = Emoji.getEmojiDrawable(emoji[i]); if (drawable != null) { - drawable.setBounds(0, 0, AndroidUtilities.dp(22), AndroidUtilities.dp(22)); + drawable.setBounds(0, 0, AndroidUtilities.dp(40), AndroidUtilities.dp(40)); drawable.preload(); - emojiViews[i].setImageDrawable(drawable); - emojiViews[i].setContentDescription(emoji[i]); + int[] emojiOnly = new int[1]; + TextPaint paint = new TextPaint(Paint.ANTI_ALIAS_FLAG); + paint.setTextSize(AndroidUtilities.dp(28)); + CharSequence txt = emoji[i]; + txt = Emoji.replaceEmoji(txt, paint.getFontMetricsInt(), false, emojiOnly); + TLRPC.Document doc1 = replaceEmojiToLottieFrame(txt, emojiOnly); + drawables.add(drawable); + if (doc1 != null) { + documents.add(doc1); + } emojiViews[i].setVisibility(View.GONE); } emojiDrawables[i] = drawable; } + if (documents.size() == 4) { + for (int i = 0; i < documents.size(); i++) { + emojiViews[i].setAnimatedEmojiDrawable(new AnimatedEmojiDrawable(AnimatedEmojiDrawable.CACHE_TYPE_EMOJI_CALL, currentAccount, documents.get(i))); + emojiViews[i].getImageReceiver().clearImage(); + } + } else { + for (int i = 0; i < drawables.size(); i++) { + emojiViews[i].setImageDrawable(drawables.get(i)); + } + } checkEmojiLoaded(animated); } @@ -1957,11 +2408,19 @@ private void checkEmojiLoaded(boolean animated) { emojiViews[i].setVisibility(View.VISIBLE); if (animated) { emojiViews[i].setAlpha(0f); - emojiViews[i].setTranslationY(AndroidUtilities.dp(30)); - emojiViews[i].animate().alpha(1f).translationY(0f).setDuration(200).setStartDelay(20 * i).start(); + emojiViews[i].setScaleX(0f); + emojiViews[i].setScaleY(0f); + emojiViews[i].animate().alpha(1f).scaleX(1f).scaleY(1f).setInterpolator(CubicBezierInterpolator.EASE_OUT_BACK).setDuration(250).start(); } } } + encryptionTooltip.postDelayed(() -> { + if (SharedConfig.callEncryptionHintDisplayedCount < 2) { + SharedConfig.incrementCallEncryptionHintDisplayed(1); + encryptionTooltip.setTranslationY(emojiLayout.getY() + AndroidUtilities.dp(36)); + encryptionTooltip.show(); + } + }, 1000); } } @@ -2002,91 +2461,123 @@ private void updateButtons(boolean animated) { Visibility visibility = new Visibility() { @Override public Animator onAppear(ViewGroup sceneRoot, View view, TransitionValues startValues, TransitionValues endValues) { - ObjectAnimator animator = ObjectAnimator.ofFloat(view, View.TRANSLATION_Y, AndroidUtilities.dp(100), 0); + PropertyValuesHolder pvh1 = PropertyValuesHolder.ofFloat(View.TRANSLATION_Y, AndroidUtilities.dp(100), 0); + PropertyValuesHolder pvh2 = PropertyValuesHolder.ofFloat(View.SCALE_Y, 0f, 1f); + PropertyValuesHolder pvh3 = PropertyValuesHolder.ofFloat(View.SCALE_X, 0f, 1f); + ObjectAnimator animator = ObjectAnimator.ofPropertyValuesHolder(view, pvh1, pvh2, pvh3); if (view instanceof VoIPToggleButton) { view.setTranslationY(AndroidUtilities.dp(100)); + view.setScaleX(0f); + view.setScaleY(0f); animator.setStartDelay(((VoIPToggleButton) view).animationDelay); } + if (view instanceof VoIpSwitchLayout) { + view.setTranslationY(AndroidUtilities.dp(100)); + view.setScaleX(0f); + view.setScaleY(0f); + animator.setStartDelay(((VoIpSwitchLayout) view).animationDelay); + } return animator; } @Override public Animator onDisappear(ViewGroup sceneRoot, View view, TransitionValues startValues, TransitionValues endValues) { - return ObjectAnimator.ofFloat(view, View.TRANSLATION_Y, view.getTranslationY(), AndroidUtilities.dp(100)); + PropertyValuesHolder pvh1 = PropertyValuesHolder.ofFloat(View.TRANSLATION_Y, view.getTranslationY(), AndroidUtilities.dp(100)); + PropertyValuesHolder pvh2 = PropertyValuesHolder.ofFloat(View.SCALE_Y, view.getScaleY(), 0f); + PropertyValuesHolder pvh3 = PropertyValuesHolder.ofFloat(View.SCALE_X, view.getScaleX(), 0f); + return ObjectAnimator.ofPropertyValuesHolder(view, pvh1, pvh2, pvh3); } }; transitionSet - .addTransition(visibility.setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT)) - .addTransition(new ChangeBounds().setDuration(150).setInterpolator(CubicBezierInterpolator.DEFAULT)); + .addTransition(visibility.setDuration(250).setInterpolator(CubicBezierInterpolator.DEFAULT)) + .addTransition(new ChangeBounds().setDuration(250).setInterpolator(CubicBezierInterpolator.DEFAULT)); transitionSet.excludeChildren(VoIPToggleButton.class, true); + transitionSet.excludeChildren(VoIpSwitchLayout.class, true); TransitionManager.beginDelayedTransition(buttonsLayout, transitionSet); } + if (currentState == VoIPService.STATE_ENDED) { + bottomSpeakerBtn.setVisibility(View.GONE); + bottomVideoBtn.setVisibility(View.GONE); + bottomMuteBtn.setVisibility(View.GONE); + bottomEndCallBtn.setVisibility(View.GONE); + return; + } + if (currentState == VoIPService.STATE_WAITING_INCOMING || currentState == VoIPService.STATE_BUSY) { if (service.privateCall != null && service.privateCall.video && currentState == VoIPService.STATE_WAITING_INCOMING) { if (!service.isScreencast() && (currentUserIsVideo || callingUserIsVideo)) { - setFrontalCameraAction(bottomButtons[0], service, animated); + setFrontalCameraAction(bottomSpeakerBtn, service, animated); if (uiVisible) { speakerPhoneIcon.animate().alpha(1f).start(); } } else { - setSpeakerPhoneAction(bottomButtons[0], service, animated); + setSpeakerPhoneAction(bottomSpeakerBtn, service, animated); speakerPhoneIcon.animate().alpha(0).start(); } - setVideoAction(bottomButtons[1], service, animated); - setMicrophoneAction(bottomButtons[2], service, animated); + setVideoAction(bottomVideoBtn, service, false); + setMicrohoneAction(bottomMuteBtn, service, animated); } else { - bottomButtons[0].setVisibility(View.GONE); - bottomButtons[1].setVisibility(View.GONE); - bottomButtons[2].setVisibility(View.GONE); + bottomSpeakerBtn.setVisibility(View.GONE); + bottomVideoBtn.setVisibility(View.GONE); + bottomMuteBtn.setVisibility(View.GONE); } - bottomButtons[3].setVisibility(View.GONE); + bottomEndCallBtn.setVisibility(View.GONE); } else { if (instance == null) { return; } if (!service.isScreencast() && (currentUserIsVideo || callingUserIsVideo)) { - setFrontalCameraAction(bottomButtons[0], service, animated); + setFrontalCameraAction(bottomSpeakerBtn, service, animated); if (uiVisible) { speakerPhoneIcon.setTag(1); speakerPhoneIcon.animate().alpha(1f).start(); } } else { - setSpeakerPhoneAction(bottomButtons[0], service, animated); + setSpeakerPhoneAction(bottomSpeakerBtn, service, animated); speakerPhoneIcon.setTag(null); speakerPhoneIcon.animate().alpha(0f).start(); } - setVideoAction(bottomButtons[1], service, animated); - setMicrophoneAction(bottomButtons[2], service, animated); + setVideoAction(bottomVideoBtn, service, false); + setMicrohoneAction(bottomMuteBtn, service, animated); - bottomButtons[3].setData(R.drawable.calls_decline, Color.WHITE, 0xFFF01D2C, LocaleController.getString("VoipEndCall", R.string.VoipEndCall), false, animated); - bottomButtons[3].setOnClickListener(view -> { + bottomEndCallBtn.setData(R.drawable.calls_decline, Color.WHITE, 0xFFF01D2C, LocaleController.getString("VoipEndCall2", R.string.VoipEndCall2), false, animated); + bottomEndCallBtn.setOnClickListener(view -> { if (VoIPService.getSharedInstance() != null) { + AndroidUtilities.cancelRunOnUIThread(hideUIRunnable); + hideUiRunnableWaiting = false; VoIPService.getSharedInstance().hangUp(); } }); } int animationDelay = 0; - for (int i = 0; i < 4; i++) { - if (bottomButtons[i].getVisibility() == View.VISIBLE) { - bottomButtons[i].animationDelay = animationDelay; - animationDelay += 16; - } + if (bottomSpeakerBtn.getVisibility() == View.VISIBLE) { + bottomSpeakerBtn.animationDelay = animationDelay; + animationDelay += 16; + } + if (bottomVideoBtn.getVisibility() == View.VISIBLE) { + bottomVideoBtn.animationDelay = animationDelay; + animationDelay += 16; + } + if (bottomMuteBtn.getVisibility() == View.VISIBLE) { + bottomMuteBtn.animationDelay = animationDelay; + animationDelay += 16; + } + if (bottomEndCallBtn.getVisibility() == View.VISIBLE) { + bottomEndCallBtn.animationDelay = animationDelay; } updateSpeakerPhoneIcon(); } - private void setMicrophoneAction(VoIPToggleButton bottomButton, VoIPService service, boolean animated) { - if (service.isMicMute()) { - bottomButton.setData(R.drawable.calls_unmute, Color.BLACK, Color.WHITE, LocaleController.getString("VoipUnmute", R.string.VoipUnmute), true, animated); - } else { - bottomButton.setData(R.drawable.calls_unmute, Color.WHITE, ColorUtils.setAlphaComponent(Color.WHITE, (int) (255 * 0.12f)), LocaleController.getString("VoipMute", R.string.VoipMute), false, animated); - } + private void setMicrohoneAction(VoIpSwitchLayout bottomButton, VoIPService service, boolean animated) { + bottomButton.setType(VoIpSwitchLayout.Type.MICRO, service.isMicMute()); currentUserCameraFloatingLayout.setMuted(service.isMicMute(), animated); - bottomButton.setOnClickListener(view -> { + bottomButton.setOnBtnClickedListener((view) -> { final VoIPService serviceInstance = VoIPService.getSharedInstance(); if (serviceInstance != null) { + AndroidUtilities.cancelRunOnUIThread(hideUIRunnable); + hideUiRunnableWaiting = false; final boolean micMute = !serviceInstance.isMicMute(); if (accessibilityManager.isTouchExplorationEnabled()) { final String text; @@ -2104,7 +2595,7 @@ private void setMicrophoneAction(VoIPToggleButton bottomButton, VoIPService serv }); } - private void setVideoAction(VoIPToggleButton bottomButton, VoIPService service, boolean animated) { + private void setVideoAction(VoIpSwitchLayout bottomButton, VoIPService service, boolean fast) { boolean isVideoAvailable; if (currentUserIsVideo || callingUserIsVideo) { isVideoAvailable = true; @@ -2113,12 +2604,17 @@ private void setVideoAction(VoIPToggleButton bottomButton, VoIPService service, } if (isVideoAvailable) { if (currentUserIsVideo) { - bottomButton.setData(service.isScreencast() ? R.drawable.calls_sharescreen : R.drawable.calls_video, Color.WHITE, ColorUtils.setAlphaComponent(Color.WHITE, (int) (255 * 0.12f)), LocaleController.getString("VoipStopVideo", R.string.VoipStopVideo), false, animated); + if (service.isScreencast()) { + bottomButton.setType(VoIpSwitchLayout.Type.VIDEO, false, fast); + } else { + bottomButton.setType(VoIpSwitchLayout.Type.VIDEO, false, fast); + } } else { - bottomButton.setData(R.drawable.calls_video, Color.BLACK, Color.WHITE, LocaleController.getString("VoipStartVideo", R.string.VoipStartVideo), true, animated); + bottomButton.setType(VoIpSwitchLayout.Type.VIDEO, true, fast); } - bottomButton.setCrossOffset(-AndroidUtilities.dpf2(3.5f)); - bottomButton.setOnClickListener(view -> { + bottomButton.setOnBtnClickedListener(view -> { + AndroidUtilities.cancelRunOnUIThread(hideUIRunnable); + hideUiRunnableWaiting = false; if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M && activity.checkSelfPermission(Manifest.permission.CAMERA) != PackageManager.PERMISSION_GRANTED) { activity.requestPermissions(new String[]{Manifest.permission.CAMERA}, 102); } else { @@ -2138,7 +2634,7 @@ private void setVideoAction(VoIPToggleButton bottomButton, VoIPService service, }); bottomButton.setEnabled(true); } else { - bottomButton.setData(R.drawable.calls_video, ColorUtils.setAlphaComponent(Color.WHITE, (int) (255 * 0.5f)), ColorUtils.setAlphaComponent(Color.WHITE, (int) (255 * 0.12f)), "Video", false, animated); + bottomButton.setType(VoIpSwitchLayout.Type.VIDEO, true); bottomButton.setOnClickListener(null); bottomButton.setEnabled(false); } @@ -2149,9 +2645,10 @@ private void updateSpeakerPhoneIcon() { if (service == null) { return; } + VoipAudioManager vam = VoipAudioManager.get(); if (service.isBluetoothOn()) { speakerPhoneIcon.setImageResource(R.drawable.calls_bluetooth); - } else if (service.isSpeakerphoneOn()) { + } else if (vam.isSpeakerphoneOn()) { speakerPhoneIcon.setImageResource(R.drawable.calls_speaker); } else { if (service.isHeadsetPlugged()) { @@ -2162,42 +2659,47 @@ private void updateSpeakerPhoneIcon() { } } - private void setSpeakerPhoneAction(VoIPToggleButton bottomButton, VoIPService service, boolean animated) { + private void setSpeakerPhoneAction(VoIpSwitchLayout bottomButton, VoIPService service, boolean animated) { + final int selectedSpeaker; + VoipAudioManager vam = VoipAudioManager.get(); if (service.isBluetoothOn()) { - bottomButton.setData(R.drawable.calls_bluetooth, Color.WHITE, ColorUtils.setAlphaComponent(Color.WHITE, (int) (255 * 0.12f)), LocaleController.getString("VoipAudioRoutingBluetooth", R.string.VoipAudioRoutingBluetooth), false, animated); - bottomButton.setChecked(false, animated); - } else if (service.isSpeakerphoneOn()) { - bottomButton.setData(R.drawable.calls_speaker, Color.BLACK, Color.WHITE, LocaleController.getString("VoipSpeaker", R.string.VoipSpeaker), false, animated); - bottomButton.setChecked(true, animated); + selectedSpeaker = 2; + bottomButton.setType(VoIpSwitchLayout.Type.BLUETOOTH, false); + } else if (vam.isSpeakerphoneOn()) { + selectedSpeaker = 0; + bottomButton.setType(VoIpSwitchLayout.Type.SPEAKER, true); } else { - bottomButton.setData(R.drawable.calls_speaker, Color.WHITE, ColorUtils.setAlphaComponent(Color.WHITE, (int) (255 * 0.12f)), LocaleController.getString("VoipSpeaker", R.string.VoipSpeaker), false, animated); - bottomButton.setChecked(false, animated); + selectedSpeaker = 1; + bottomButton.setType(VoIpSwitchLayout.Type.SPEAKER, false); } - bottomButton.setCheckableForAccessibility(true); bottomButton.setEnabled(true); - bottomButton.setOnClickListener(view -> { + bottomButton.setOnBtnClickedListener(view -> { if (VoIPService.getSharedInstance() != null) { - VoIPService.getSharedInstance().toggleSpeakerphoneOrShowRouteSheet(activity, false); + AndroidUtilities.cancelRunOnUIThread(hideUIRunnable); + hideUiRunnableWaiting = false; + VoIPService.getSharedInstance().toggleSpeakerphoneOrShowRouteSheet(activity, false, selectedSpeaker); + setSpeakerPhoneAction(bottomButton, service, true); } }); } - private void setFrontalCameraAction(VoIPToggleButton bottomButton, VoIPService service, boolean animated) { + private void setFrontalCameraAction(VoIpSwitchLayout bottomButton, VoIPService service, boolean animated) { if (!currentUserIsVideo) { - bottomButton.setData(R.drawable.calls_flip, ColorUtils.setAlphaComponent(Color.WHITE, (int) (255 * 0.5f)), ColorUtils.setAlphaComponent(Color.WHITE, (int) (255 * 0.12f)), LocaleController.getString("VoipFlip", R.string.VoipFlip), false, animated); - bottomButton.setOnClickListener(null); + bottomButton.setType(VoIpSwitchLayout.Type.CAMERA, false); + bottomButton.setOnBtnClickedListener(null); bottomButton.setEnabled(false); } else { bottomButton.setEnabled(true); - if (!service.isFrontFaceCamera()) { - bottomButton.setData(R.drawable.calls_flip, Color.BLACK, Color.WHITE, LocaleController.getString("VoipFlip", R.string.VoipFlip), false, animated); + if (service.isFrontFaceCamera()) { + bottomButton.setType(VoIpSwitchLayout.Type.CAMERA, !service.isSwitchingCamera()); } else { - bottomButton.setData(R.drawable.calls_flip, Color.WHITE, ColorUtils.setAlphaComponent(Color.WHITE, (int) (255 * 0.12f)), LocaleController.getString("VoipFlip", R.string.VoipFlip), false, animated); + bottomButton.setType(VoIpSwitchLayout.Type.CAMERA, service.isSwitchingCamera()); } - - bottomButton.setOnClickListener(view -> { + bottomButton.setOnBtnClickedListener(view -> { final VoIPService serviceInstance = VoIPService.getSharedInstance(); if (serviceInstance != null) { + AndroidUtilities.cancelRunOnUIThread(hideUIRunnable); + hideUiRunnableWaiting = false; if (accessibilityManager.isTouchExplorationEnabled()) { final String text; if (service.isFrontFaceCamera()) { @@ -2207,6 +2709,7 @@ private void setFrontalCameraAction(VoIPToggleButton bottomButton, VoIPService s } view.announceForAccessibility(text); } + bottomButton.setType(VoIpSwitchLayout.Type.CAMERA, !service.isFrontFaceCamera()); serviceInstance.switchCamera(); } }); @@ -2240,7 +2743,9 @@ private void toggleCameraInput() { service.switchCamera(); } windowView.setLockOnScreen(true); - previewDialog = new PrivateVideoPreviewDialog(fragmentView.getContext(), false, true) { + int[] locVideoButton = new int[2]; + bottomVideoBtn.getLocationOnScreen(locVideoButton); + previewDialog = new PrivateVideoPreviewDialogNew(fragmentView.getContext(), locVideoButton[0], locVideoButton[1]) { @Override public void onDismiss(boolean screencast, boolean apply) { previewDialog = null; @@ -2251,6 +2756,10 @@ public void onDismiss(boolean screencast, boolean apply) { if (service != null && !screencast) { service.requestVideoCall(false); service.setVideoState(false, Instance.VIDEO_STATE_ACTIVE); + service.switchToSpeaker(); + } + if (service != null) { + setVideoAction(bottomVideoBtn, service, true); } } else { if (service != null) { @@ -2260,6 +2769,34 @@ public void onDismiss(boolean screencast, boolean apply) { previousState = currentState; updateViewState(); } + + @Override + protected void afterOpened() { + gradientLayout.lockDrawing = true; + gradientLayout.invalidate(); + } + + @Override + protected void beforeClosed() { + gradientLayout.lockDrawing = false; + gradientLayout.invalidate(); + } + + @Override + protected int[] getFloatingViewLocation() { + int[] loc = new int[2]; + int[] result = new int[3]; + currentUserCameraFloatingLayout.getLocationOnScreen(loc); + result[0] = loc[0]; + result[1] = loc[1]; + result[2] = currentUserCameraFloatingLayout.getMeasuredWidth(); + return result; + } + + @Override + protected boolean isHasVideoOnMainScreen() { + return callingUserIsVideo; + } }; if (lastInsets != null) { previewDialog.setBottomPadding(lastInsets.getSystemWindowInsetBottom()); @@ -2301,7 +2838,11 @@ private void onRequestPermissionsResultInternal(int requestCode, String[] permis return; } if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) { - VoIPService.getSharedInstance().acceptIncomingCall(); + runAcceptCallAnimation(() -> { + if (VoIPService.getSharedInstance() != null) { + VoIPService.getSharedInstance().acceptIncomingCall(); + } + }); } else { if (!activity.shouldShowRequestPermissionRationale(Manifest.permission.RECORD_AUDIO)) { VoIPService.getSharedInstance().declineIncomingCall(); @@ -2361,9 +2902,6 @@ public void onPauseInternal() { if (canSwitchToPip && hasPermissionsToPip) { int h = instance.windowView.getMeasuredHeight(); - if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT_WATCH && instance.lastInsets != null) { - h -= instance.lastInsets.getSystemWindowInsetBottom(); - } VoIPPiPView.show(instance.activity, instance.currentAccount, instance.windowView.getMeasuredWidth(), h, VoIPPiPView.ANIMATION_ENTER_TYPE_SCALE); if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT_WATCH && instance.lastInsets != null) { VoIPPiPView.topInset = instance.lastInsets.getSystemWindowInsetTop(); @@ -2419,4 +2957,22 @@ private void requestInlinePermissions() { }).show(); } } + + public TLRPC.Document replaceEmojiToLottieFrame(CharSequence text, int[] emojiOnly) { + if (!(text instanceof Spannable)) { + return null; + } + Spannable spannable = (Spannable) text; + Emoji.EmojiSpan[] spans = spannable.getSpans(0, spannable.length(), Emoji.EmojiSpan.class); + AnimatedEmojiSpan[] aspans = spannable.getSpans(0, spannable.length(), AnimatedEmojiSpan.class); + + if (spans == null || (emojiOnly == null ? 0 : emojiOnly[0]) - spans.length - (aspans == null ? 0 : aspans.length) > 0) { + return null; + } + + for (Emoji.EmojiSpan span : spans) { + return MediaDataController.getInstance(currentAccount).getEmojiAnimatedSticker(span.emoji); + } + return null; + } } diff --git a/TMessagesProj/src/main/java/org/telegram/ui/VoiceMessageEnterTransition.java b/TMessagesProj/src/main/java/org/telegram/ui/VoiceMessageEnterTransition.java index 7b17cdf7c59..3ae3356b55b 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/VoiceMessageEnterTransition.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/VoiceMessageEnterTransition.java @@ -147,30 +147,35 @@ public void onDraw(Canvas canvas) { canvas.drawCircle(cx, cy, radius, circlePaint); canvas.save(); - float scale = radius / toRadius; canvas.scale(scale, scale, cx, cy); - canvas.translate(cx - messageView.getRadialProgress().getProgressRect().centerX(), cy - messageView.getRadialProgress().getProgressRect().centerY()); - + float tx = cx - messageView.getRadialProgress().getProgressRect().centerX(); + float ty = cy - messageView.getRadialProgress().getProgressRect().centerY(); + canvas.translate(tx, ty); messageView.getRadialProgress().setOverrideAlpha(progress); messageView.getRadialProgress().setDrawBackground(false); - messageView.getRadialProgress().draw(canvas); + messageView.drawVoiceOnce(canvas, progress, () -> { + messageView.getRadialProgress().draw(canvas); + canvas.translate(-tx, -ty); + canvas.scale(1f / scale, 1f / scale, cx, cy); + if (recordCircle != null) { + recordCircle.drawIcon(canvas, (int) fromCx, (int) fromCy, 1f - moveProgress); + } + canvas.scale(scale, scale, cx, cy); + canvas.translate(tx, ty); + }); messageView.getRadialProgress().setDrawBackground(true); messageView.getRadialProgress().setOverrideAlpha(1f); canvas.restore(); - if (container.getMeasuredHeight() > 0) { - gradientMatrix.setTranslate(0, clipBottom); - gradientShader.setLocalMatrix(gradientMatrix); +// if (container.getMeasuredHeight() > 0) { +// gradientMatrix.setTranslate(0, clipBottom); +// gradientShader.setLocalMatrix(gradientMatrix); // canvas.drawRect(0, clipBottom, container.getMeasuredWidth(), container.getMeasuredHeight(), gradientPaint); - } +// } //restore clipRect // canvas.restore(); - - if (recordCircle != null) { - recordCircle.drawIcon(canvas, (int) fromCx, (int) fromCy, 1f - moveProgress); - } } private int getThemedColor(int key) { diff --git a/TMessagesProj/src/main/java/org/telegram/ui/WallpapersListActivity.java b/TMessagesProj/src/main/java/org/telegram/ui/WallpapersListActivity.java index ffcf9a5ee29..15bd5296c60 100644 --- a/TMessagesProj/src/main/java/org/telegram/ui/WallpapersListActivity.java +++ b/TMessagesProj/src/main/java/org/telegram/ui/WallpapersListActivity.java @@ -96,6 +96,8 @@ public class WallpapersListActivity extends BaseFragment implements Notification private int resetSectionRow; private int resetRow; private int resetInfoRow; + private int galleryRow; + private int galleryHintRow; private int currentType; private final long dialogId; @@ -272,6 +274,15 @@ public class WallpapersListActivity extends BaseFragment implements Notification public final static int TYPE_ALL = 0; public final static int TYPE_COLOR = 1; + public final static int TYPE_CHANNEL_PATTERNS = 2; + public final static int TYPE_CHANNEL_CUSTOM = 3; + + public static class EmojiWallpaper { + public final String emoticon; + public EmojiWallpaper(String emoticon) { + this.emoticon = emoticon; + } + } public static class ColorWallpaper { @@ -418,7 +429,7 @@ public WallpapersListActivity(int type, long dialogId) { @Override public boolean onFragmentCreate() { - if (currentType == TYPE_ALL) { + if (currentType == TYPE_ALL || currentType == TYPE_CHANNEL_PATTERNS) { NotificationCenter.getGlobalInstance().addObserver(this, NotificationCenter.wallpapersDidLoad); NotificationCenter.getGlobalInstance().addObserver(this, NotificationCenter.didSetNewWallpapper); NotificationCenter.getGlobalInstance().addObserver(this, NotificationCenter.wallpapersNeedReload); @@ -447,7 +458,7 @@ public static void fillDefaultColors(ArrayList wallPapers, boolean isDar @Override public void onFragmentDestroy() { - if (currentType == TYPE_ALL) { + if (currentType == TYPE_ALL || currentType == TYPE_CHANNEL_PATTERNS) { searchAdapter.onDestroy(); NotificationCenter.getGlobalInstance().removeObserver(this, NotificationCenter.wallpapersDidLoad); NotificationCenter.getGlobalInstance().removeObserver(this, NotificationCenter.didSetNewWallpapper); @@ -473,7 +484,7 @@ public void didSelectWallpaper(File file, Bitmap bitmap, boolean gallery) { ThemePreviewActivity themePreviewActivity = new ThemePreviewActivity(new FileWallpaper("", file, file), bitmap); themePreviewActivity.setDialogId(dialogId); if (dialogId != 0) { - themePreviewActivity.setDelegate(WallpapersListActivity.this::removeSelfFromStack); + themePreviewActivity.setDelegate(wallPaper -> WallpapersListActivity.this.removeSelfFromStack()); } presentFragment(themePreviewActivity, gallery); } @@ -488,9 +499,11 @@ public void needOpenColorPicker() { actionBar.setBackButtonImage(R.drawable.ic_ab_back); actionBar.setAllowOverlayTitle(true); if (currentType == TYPE_ALL) { - actionBar.setTitle(LocaleController.getString("ChatBackground", R.string.ChatBackground)); + actionBar.setTitle(LocaleController.getString(R.string.ChatBackground)); + } else if (currentType == TYPE_CHANNEL_PATTERNS) { + actionBar.setTitle("Channel Wallpaper"); } else if (currentType == TYPE_COLOR) { - actionBar.setTitle(LocaleController.getString("SelectColorTitle", R.string.SelectColorTitle)); + actionBar.setTitle(LocaleController.getString(R.string.SelectColorTitle)); } actionBar.setActionBarMenuOnItemClick(new ActionBar.ActionBarMenuOnItemClick() { @Override @@ -901,6 +914,9 @@ public void restoreSelfArgs(Bundle args) { } private boolean onItemLongClick(WallpaperCell view, Object object, int index) { + if (currentType == TYPE_CHANNEL_PATTERNS || currentType == TYPE_CHANNEL_CUSTOM) { + return false; + } Object originalObject = object; if (object instanceof ColorWallpaper) { ColorWallpaper colorWallpaper = (ColorWallpaper) object; @@ -970,7 +986,7 @@ public boolean insideBottomSheet() { } }; if (currentType == TYPE_COLOR || dialogId != 0) { - wallpaperActivity.setDelegate(WallpapersListActivity.this::removeSelfFromStack); + wallpaperActivity.setDelegate(wallPaper -> WallpapersListActivity.this.removeSelfFromStack()); } if (selectedBackgroundSlug.equals(slug)) { wallpaperActivity.setInitialModes(selectedBackgroundBlurred, selectedBackgroundMotion, selectedIntensity); @@ -1025,7 +1041,7 @@ public void didReceivedNotification(int id, int account, Object... args) { ArrayList arrayList = (ArrayList) args[0]; patterns.clear(); patternsDict.clear(); - if (currentType != TYPE_COLOR) { + if (currentType != TYPE_COLOR && currentType != TYPE_CHANNEL_PATTERNS) { wallPapers.clear(); localWallPapers.clear(); localDict.clear(); @@ -1045,7 +1061,7 @@ public void didReceivedNotification(int id, int account, Object... args) { patternsDict.put(wallPaper.document.id, wallPaper); } allWallPapersDict.put(wallPaper.slug, wallPaper); - if (currentType != TYPE_COLOR && (!wallPaper.pattern || wallPaper.settings != null && wallPaper.settings.background_color != 0)) { + if (currentType != TYPE_COLOR && (!wallPaper.pattern || wallPaper.settings != null && wallPaper.settings.background_color != 0) && (currentType != TYPE_CHANNEL_PATTERNS || wallPaper.pattern)) { if (!Theme.isCurrentThemeDark() && wallPaper.settings != null && wallPaper.settings.intensity < 0) { continue; } @@ -1124,7 +1140,7 @@ private void loadWallpapers(boolean force) { TLRPC.TL_account_wallPapers res = (TLRPC.TL_account_wallPapers) response; patterns.clear(); patternsDict.clear(); - if (currentType != TYPE_COLOR) { + if (currentType != TYPE_COLOR && currentType != TYPE_CHANNEL_PATTERNS) { wallPapers.clear(); allWallPapersDict.clear(); allWallPapers.clear(); @@ -1142,7 +1158,7 @@ private void loadWallpapers(boolean force) { patterns.add(wallPaper); patternsDict.put(wallPaper.document.id, wallPaper); } - if (currentType != TYPE_COLOR && (!wallPaper.pattern || wallPaper.settings != null && wallPaper.settings.background_color != 0)) { + if (currentType != TYPE_COLOR && (!wallPaper.pattern || wallPaper.settings != null && wallPaper.settings.background_color != 0) && (currentType != TYPE_CHANNEL_PATTERNS || wallPaper.pattern)) { if (!Theme.isCurrentThemeDark() && wallPaper.settings != null && wallPaper.settings.intensity < 0) { continue; } @@ -1179,7 +1195,7 @@ private void loadWallpapers(boolean force) { } private void fillWallpapersWithCustom() { - if (currentType != TYPE_ALL) { + if (currentType != TYPE_ALL && currentType != TYPE_CHANNEL_PATTERNS) { return; } SharedPreferences preferences = MessagesController.getGlobalMainSettings(); @@ -1374,10 +1390,20 @@ private void updateRows() { uploadImageRow = rowCount++; setColorRow = rowCount++; sectionRow = rowCount++; + galleryRow = -1; + galleryHintRow = -1; + } else if (currentType == TYPE_CHANNEL_PATTERNS) { + uploadImageRow = -1; + setColorRow = -1; + sectionRow = -1; + galleryRow = rowCount++; + galleryHintRow = rowCount++; } else { uploadImageRow = -1; setColorRow = -1; sectionRow = -1; + galleryRow = -1; + galleryHintRow = -1; } if (!wallPapers.isEmpty()) { totalWallpaperRows = (int) Math.ceil(wallPapers.size() / (float) columnsCount); @@ -1875,6 +1901,9 @@ public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) { textCell.setTextAndIcon(LocaleController.getString("SetColor", R.string.SetColor), R.drawable.msg_palette, true); } else if (position == resetRow) { textCell.setText(LocaleController.getString("ResetChatBackgrounds", R.string.ResetChatBackgrounds), false); + } else if (position == galleryRow) { + textCell.setTextAndIcon("Choose from gallery", R.drawable.msg_background, false); + textCell.setLockLevel(false, 10); } break; } @@ -1882,6 +1911,8 @@ public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) { TextInfoPrivacyCell cell = (TextInfoPrivacyCell) holder.itemView; if (position == resetInfoRow) { cell.setText(LocaleController.getString("ResetChatBackgroundsInfo", R.string.ResetChatBackgroundsInfo)); + } else if (position == galleryHintRow) { + cell.setText("Upload your own background for the channel."); } break; } @@ -1956,11 +1987,11 @@ public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) { @Override public int getItemViewType(int position) { - if (position == uploadImageRow || position == setColorRow || position == resetRow) { + if (position == uploadImageRow || position == galleryRow || position == setColorRow || position == resetRow) { return 0; } else if (position == sectionRow || position == resetSectionRow) { return 1; - } else if (position == resetInfoRow) { + } else if (position == resetInfoRow || position == galleryHintRow) { return 3; } else { return 2; diff --git a/TMessagesProj/src/main/res/drawable-hdpi/menu_feature_color_name.png b/TMessagesProj/src/main/res/drawable-hdpi/menu_feature_color_name.png new file mode 100644 index 0000000000000000000000000000000000000000..662f51cafae9ba6f3b982e9133250192d4a8e176 GIT binary patch literal 1116 zcmV-i1f%Px#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91B%lKT1ONa40RR91Bme*a04fgS`~Uz0#z{m$R9Fe^mrF=gQ5eUyhgo8w zP%0sT1d@v~Dhsk&WCbk}A}TNnifZAaMFbfbMrDhrP0I>55%{2tf}}->5fx+xEi8IK zFAIyv9%gFV@2}^KXRddg8Rxp|gP-5I-|L(?uX{~O`Y-=yAh8)pj+Q<+IJk(I&9DG6 zAZiXVUKzmf86HDza&q!}Z$Q*%29gO45VZWHU^)2yFkd6Va~SbGm~Te@6#W2R!=SOn zqb3tv0L3uRBwl5vH1pAg0o?}4H`p0HBen=?O@-so50d>+GYQ%_6gvE;G-|>)0w`T= z`uKs)9|l$*{+I%KC`i7FLu8Y$w6WUe}K?+dAUipl+R*+jO>v;3FpC z6f<+)ldKIc37=UKXF!h`t=`OGf;|gzqA<%%B6dA!)+Co9^o5}|oB%6!(8DFL?lYc0 zXnl+&B-B_4-%ZUH^fX(Lb{bqUKB;;b+0#7agWkT9b2f3jni_NAor)pPAaomNA*UFc zKqpimUv`!1)0fSo11CK|y?9gXR> zMnp}Jis@VWCA^Bj3+xrP2t`g=G2?pPwa_( z?&nO{#$YyqT_*kbUl2OhZU@+lKv#f7KOlIa@Ufxt!+&(E7y0#qT{p2VSjjDzu#K$u z&o%X8Px#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91B%lKT1ONa40RR91Bme*a04fgS`~Uz1LrFwIR9Fe^m}_X3Wf;fzs+g8C zokPjPoZ=CRC@L4EQi|edKgdvIUt%bw1SW(a76cWv6gqt|PbeM0LL4EUKB$O?DAcLx z+6>A=i4*fsvHAM@zuwoo>v{IRcJJN#AaucR_jSIXJaXdOw>PaQIWXXsu+)eR#8ImRe;i~Qf^pEVOHK$dv~T5-RC$pw`8eop;X{x(mvApUBr3S+IsN*U=<3)<5p_MY9h{G-RCO;7Txp9ova z+kXyT_VMoM^qJv$X_XC!O3&W|Y=xAZG4`oz7*aL+=1K~($ ztUENtx5FL{Cqn$7kLfP03jqy!Z?yTnlt}7^L9I4goeU)EH`AA6PlHP!xeZVFxGtz4 zY=E@rtW$K?PBd`_HQFvr7P^q;S_pSSP#5*Qk0*43dZz0^S8bwwBl_JUcbOG3uCfWx zb&`0;#dNBSJ5iE26y9~kYpij4B&P;mgSSC7w%4q;TzrqgH(>HLN`AFj-oK6%HH8j4 z)5-Y9z^)Ka52tYCJ`0Rh@S{t33q9^UWRp^$mAA(=X|R|^{DYu1#M>DU!ME^x@M)-r z7vQoCM6baD-zQ3^XfgZ+CTCEF`(o^mLQrSrX%<5t0H24LURkq%Ke zL88AH*23U)1AG@jQdc0Rk87)fL3C7!UKELb6joXn#si69Fz}CZImN!}8$yoWrGbU| zqCc{szo}=RmSC_cQf_T%DWjy*il+_m0UVeow;8ZV3~2nE?NLNJEAX4nhY!DL%h)1D?VTRFTBw`}Ir z*oIwK$~@iMx>GYSsUFvC(Eo}?z-?%PMpjiDW4GlVi>pIE`{B`C00000NkvXXu0mjf DpPMd3 literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-hdpi/menu_feature_cover.png b/TMessagesProj/src/main/res/drawable-hdpi/menu_feature_cover.png new file mode 100644 index 0000000000000000000000000000000000000000..0f7e5f628116774c5c9b7b54c46a9d700ae31c00 GIT binary patch literal 1051 zcmV+$1mydPP)Px#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91B%lKT1ONa40RR91Bme*a04fgS`~Uz0g-Jv~R9Fe^m(NR8K@`V**$*t0 zf`}BPHl-NS!d4Lx{R1sqgb=i_prEp%TH4y3&>~XFwX6P$aMMqvMH`JGi6XThMEUxB zpL2Y3@0~l|=X-D0fiH9Be9t*^?#$eods|xmOMC`OaeAFCm&@&F9YuU2J}aeCX*I&f zDT?R-kH8XWlqlB(=!=i;SR8vnzq+)SK(piziAO?3C6)}QDV*Av+VkjIrt9D{m<1m| zJ+@Ok2KX?Q3L|o-<4JKGGs1+Y=4WTvY2XPhL z4wB9YubHu8mgNuhw1RZmB=U*Q6ugQG7ZWSO=SZL>k!au}J||r^$I+BW)4bn@KL`%^ zwuBC;WSK#HCwSX4xig86W=gW)$EJ1or$X9Z=|kvM)Kd$=a|pZMCX>NnB8)g}x?%C@ z2EQD1XjWvKDYScmE>J;wQ^Xab(^;p#_Lwk>eGK&Y1!0eiyRF|lud@+v;}!!q{7K?2 zgN<2nx@^^1r@Ag+(ghh59O~ZGMqD7tLy3r7U3IKMxKv9`gH64tZ=q4Ea8Cvo3 z>lFQHeLb0;kbWbVY`@FDM*G#qO0*1W|6s!jqJ|DRbVliZiV8j#f#hATBcy)fB>DhP z9_6`FRe~7(T>b!~Rcls**N53FAbFVea2CiupkEWoG^nX4{PkTfnadjlPQ^Tt6QbAt zP@Zqaj>CvNooJ?pfxd5awHWsKNw8~XRLmz}GKn&f@;i-9Z=A^$pU>6tR#7%n7VNj$ zPx#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91B%lKT1ONa40RR91Bme*a04fgS`~Uz0gh@m}R9Fe^m&;4lQ4q(yVp^h$ zh{Q)lM41(?EFu#64}umh3(81>7E$e5M2n212SJO7ilT546%q9Zw2B~lEE2RL0+E!c zNPF1B+vj^db34D+z1RELIq-4j%*>fHp7VR%s;YU(y#uM-;n`?9ovy=L0?H4gB=gBO zNn^~wZ}^l-rKT-Vl%|O`xCle=2TGMk;YKKU4AzGIkugaJ;2V@vCMdSh7Q=3YY3Mhf zw6;1RB<#X*-8qNR1Mm#S!Hz-Nz|7dv68Y^2r5AvXdz54Qg3bu*^kCeAA^aO=y`jG( zt%hG9xd}C)l#++k)Xwz4D=;~Q(HSX@Qvu0KsEs6)=%D6t&@m;+o4^uey>prS0)a9O zsND|z;O~x>oPxWsGq4nn6L1f7n^#m=V*n&-W@$wiE2KD0bgCQVP9gav=de&6+=fY~ zBfF0OEd-CrUzuPRtnq!t+SKU=;!1d>xM(ujg9VY6t1sI%%IQfh7U^cPL z2DMwfMOy;p39JOYDojpdGzF5%JfMDZLt2XjZQ*wqDP?WuuY*01^SFuUZpbyI@1MiP zHpJW=VDBp39NkH^h~;l6X|nI9&|>(15xOA0EubBiSW_Rc+x+=v<1f7-ikyQP(4Dc{ zE3TL3N02-RjmI=Vo&BlMpCFx6atLKkh&*HILNP!gS7$h&C!6snq)i4OUcJzH5E5y|NW00tp zzO!T33lt!+K{K3mxk)vQ^>z0emIo%d0shPaaWV6e(+VS?TA3F*T3;3>*?-jLWHeFz zRFU*q09(5h{wlC=!CEy`%O+sk^)+F>h$iVONQPliB%w?W)pl#3(pM7866^r`m3jla z>`kdAsJ29Fb@&Zsx6qs<=+$xBjrCqT3r}Ip`HEGoOxy^EpxqTY!bjQ}XD2H)<+?{z zM6@$=duK91$ERz`E_BS2MYU8ja7_NzM2^f4noR-SmW?(pQcrR`Q$^05pO)Q$e=!ir Ue}b_k@Bjb+07*qoM6N<$f=cz#)&Kwi literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-hdpi/menu_feature_links.png b/TMessagesProj/src/main/res/drawable-hdpi/menu_feature_links.png new file mode 100644 index 0000000000000000000000000000000000000000..3758211e173d020747500ee29d8aa841a2ba7535 GIT binary patch literal 919 zcmV;I18Dq-P)Px#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91B%lKT1ONa40RR91Bme*a04fgS`~Uz00ZBwbR9Fe^m_KV3K@`Q`iYN)u z&WHxm7@NdzAgE>ZH3_5;i!3i~`~>q!c|Ms<7d-%sPkTn%=Q4|O9bso0i6g&#Q!w>K|e3)g~ zYs$Y5g6Y8ba3^#TOvEDERfcdrg!5bQq8sD~oZJRd#w^LWAHYdspSA8Y&ECeLg7kXKaJWE$hBTL zMX};L-seAr>KJLx=rxUn`ghUx#@VL(B?x~t-ANf7n<;o7PRI20Ycmzz`MUl7gI`BD z!t)b&8vcS0U=-M#LhuwGf!cIsUxwqZ<5QWS-;U!USZQf182AMWDs`Tfg8mf=!Oc>U zbs$>qNfVbg+QDFysZyXHhZ}B;*;1i85ZSwxe#adcO{HMm7&>MIZIHSpgrQC+q9~Ta zs?An3b%I%xbBE5Rw?GZij1wxI!GqGw=&ut@j-7URVcC3@mWXcjqywAPpaB{!ZGQqs z=Zx&p=g@-Dg3;9_4n(U^xEfQ^uYE>PP$|JE^L{XgV*dMm7_CELq2Cy&&_4|oRCH;Z zDLS2ff-;j9jEoV#k`=gY&YCFkrvq8Xk8Y6Zz`!=E>*-d_DhCf&eH{oF2(I>h8(Ym8 zy{7bo^Ut_je{nDg4Rc$2)1ENSp-tfVG^>aTXI&_07afgAw{@Y!7r=iE72A?GH>nIEXYjZd2Ta zhxH8Bp+4?()6*yNF?a@^g9o5+4=(T#sQ%_bPK@fS?n|sL1~73=M1J#6@6u# tg<8rwhdx97d7!L4CMz41d-(q{@Eek)kk8Qy){g)H002ovPDHLkV1iMViw^(* literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-hdpi/menu_feature_links2.png b/TMessagesProj/src/main/res/drawable-hdpi/menu_feature_links2.png new file mode 100644 index 0000000000000000000000000000000000000000..16e8585e1b8683e98b360de9eb1d54e414ca7cff GIT binary patch literal 1019 zcmVPx#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91B%lKT1ONa40RR91Bme*a04fgS`~Uz0Wl2OqR9Fe^m(5F6K^VsGjlMK< zX&-`6i9v1KwJ2;;xiM-HL0)VZwN*j?3fuMpk>*ChO_7_Tl0s+~wTLWgVYvxvAz9Sv z_nc;~4zoO&P*@6V7kIvu2X-ln>w|_yIPpJ8zM^7t&_q#4@G zD40qvLnA`3plqfli`aU#IO~H#;b5#oY_)tA2W)>`C0^R$7(7*%1B|fMGxZt@!}a=U zs2zcf#^FTKjO%!V+wj#f(leuFYWIz=q3@Nk&F_~W+}$*mXL4-1;4Ro4(lf5ZRCwd+ zHo6VBj)`H^D+%Tfx8uD5%s4oeIWh zBm~z|MRowuL@dp@wb3SnQKnLXz5_0sIl{XYkzHHqGf!YNg@RF(*yuGQ=z!EM!4C}@ z5k)laSFQA-sVA6Aa&FMS;>}Y-)N6!Fr}sr^%^0sI7#$xv;e`q7E44&)HI^n0tE2%M zZEg1f4n8xoM_++97;P9`UBW>01PUV|CF4401O=5R7-ilKW_!qgbqu3--!dwLOfLS4WGUC0XsgLP^JNNQ547ioGNn7PZ%MQ>^`g8E@q@PB*d1Sm5E*cL*Y1-5bxdu4Vr;1XmjOY%mPvk}T0xD~F`IpO-LEz^mfPx#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91B%lKT1ONa40RR91Bme*a04fgS`~Uz0O-V#SR9Fe^mrZO;Q545#2>nt- zv=xbkXi6kPXj6p5N^4`&g|8xE(-4bBqKQ};ED<)uM-vMY5@I0|EF^^ZTu6PSK}96> zm5?+xsPX&H-23`EbMMT&n&|>3`Ssp+&i|bA`rdu-zKO;D+uwKvlF8&asDX;8{~tLM zU>b~%nmbHSfr+pO9>Fh=X?zREVCk@g&;hX}a0=dlwLvCNK{d-l^88a0R>McgXIBu+ zv=uc6`SRILo0B=lGaXdgitjj#g#k0a#`qp4LnEv=BSY*fw1eVTp$25Va2j5~YnTcx zFb}%o@punoZsTMY!%y??-C?ZtWf0R<{A9G>83)X~pK&ATAM%>wkbrWhh=3MuFUTH3 z-UW=N0s6q&OD4fBbT%?hbj>EI)u|)cW&}SN=Q>3WCt5v0&*2%=IV4UO+Pxq<;uF}# zc+n}y;iTv+kwcOSp(%qMa0%?v(37j)ouD{2_f5|p#wHjO=s2h~K@D?)KnW-GSz~&3WVInsZh{)pmn}$Ox30IKCqV7?P@h#ES=xSWC>S;U-wy9nLG-oa$4;#W>yqgQlzYKW)s*;j!`QJ9&FJ zr{r@Qt_9cAsF&WRlAnO90gV@deo*RaiF?rO(vGT+lh7WOwZKR9B>tIZzX7%YQXK>) RKX(8C002ovPDHLkV1iwPtP=nL literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-hdpi/menu_feature_status.png b/TMessagesProj/src/main/res/drawable-hdpi/menu_feature_status.png new file mode 100644 index 0000000000000000000000000000000000000000..0b23197bc4351647db93bbd1eaab828a7d897d30 GIT binary patch literal 1136 zcmV-$1dscPP)Px#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91B%lKT1ONa40RR91Bme*a04fgS`~Uz0+DSw~R9Fe^ms^NUQ5eVPxDA>Z zF(hi-5|eU?lw4~PO<^A9&Fv9SgXBS^m=wt+9whQ0MZ<%bm?q?s%e)YBi%ISja=&lC z|FOPpwa-3R<3;}UpY^T(_y4}N&pvyteWs}BKmPI^2!f#fUtS`&_FTwVcnSW1Z{P-4 zoLluDx>#rm+yQsOX|Mu*f>G@H!=~la?E>Qtv<5IEVuG)YtDSV(=_b!^m68(C4#A@x9UjaN$L*t&~{OC#&+;D z6u!e=wkYvdiJ;jue?%V!D`9_M&nc#;4C-!Z7YX45OWYg2mI#hpLt=f!?Mr-?B9Fy@UNF_y#UFbaQ>$s~2LV)%)P*FoVLYy&&PJ5cxt725`AioJlSNP!+; z;Raj^FKGhwjPy1?@n(tOqNjCZZg+1Ev3ayJ`^v{S0vbA4ZQ2p%edn%n4LUNv?lk2R zPEQcubNVD}$b5yi7OuNX&~kyv-?iOVha^fgXP-cjkympe_v81uT^A(va} z>#;G$T4=Xfb?@LXGZU9WU8p>jZUKFDi(!f9+s0gYY?|}7MN_?y`BcZChfTXiXojzo z(XaQ`C~J(?2WoqNfCEjp5v`}oF$4=t*T{UdM4+>+ou5mIF|3voG`M1I+rvpE2);7< z52g;e7C0f=2yHDC&d21I{YnK#qPdhgM!TS6eiD5i)CKA8)%v-(7|%&4%#7yIjwUlf zuq2vGnPVIY?awd7g|jg83pLsxpE=FD!voNu-^`6r&zk-NxdIJm2nyd>d0v>l!gIEAWYk60!SEH7 zsdxDD%w)Px#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91B%lKT1ONa40RR91Bme*a04fgS`~Uz0+DSw~R9Fe!m|2L9Q5eU47)*A; z7;!-|LWr@Zv6L(q7-hMTBFiYQNXQL}lq6A>nhVAqN@-$*vSgVHWQ`gbTlR=)Y_H${ zbB@k@?>gU%lBa&2^FGi2dCvLH>wVutlP3R|rLtX*=XryuEr5K((^M+8JKKI))v<_C zunVdn$a>qxBp1Okm<3t?aNDnrr{MrN0h#uizzrye4Q>(Fcfi_x1`FJLR%4nph3)VI ztW`2t3%XsQMt%fb2iFQi3*r)VX%G74*7S>K#MNcyshJ4c-3!nqGoR?1!VYj7tZik` zEKzY|i!?EuGApUU&;JW^Y^3j1h`jCTc@gadYx5d=G3rP2n`QznDtzZj4uPLWyeV!m z==Aq&Ahe6qx_R_-!P;s!S7(gkv{C6%ZdxaqaY241)EfO4{5puNaa*80?$ow%bLlb3 z2}U3{*(Y!Us z1-@ZCGovK%5?v1XcU3`n39?mz)lN{Fw9@m=+c4KfP;KIJdm-26>*XmNsSd_^#TBFT zEl4%QJHb)YM^5iDeS{+oOXt(d41jA)I0qkS_9KdTAkjP0uoX3Aga4xz2Ql34DOcKM9(Q1)08Mv5P|Ed+dp& zU=4f)*+Wv;cu+dxlMtf89(d!VCFOmGTV8|M=^0Yt6fo)sJa3_r<{MGn#CgMN@~X`qV_q^j&D?c!*EGp+j3q0GCXG9^Gl7;w%&NYz}%b>{~p#lY~$OGVRow=!*C( zJ-S2mZbY@AxEt7=V(49l$fqx;A2mbaB*@BxG>Kz0XaNV*$*~H?OCT?*5&0Y_1^cfU zhTgHLn#4I8je!g5=$IyY0B4~LO5BEA4+d*b87zZDFHE;jPx#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91B%lKT1ONa40RR91Bme*a04fgS`~Uz0AW1|)R9Fe^mtAO2VI0TzmrX9b zjg66MUKSEcnF(u2QdWcuZqzOnrCcZ?#GvCla>z-es{0g}%z~`bE)9c*HpKm)&$XYz=Yo#ayEi#+qx*M>jJ+-Hg2|aYGa?f-PVVTj>ihH;=(} z(AWJ5)WQdt4R`U+M0+JsLlhiCwI;X-+KPSXP_T8*)@iZ?UceeYA5G{Q=mec8nqeID zlOzp|ri!+r8u~ydgu^s)7Ifz~VJfu2YMObE))}Z3rQ>fqOoO$MT`1T9k_I=Dxo4p# z2wIHRQMMKinz2-mu{+QeZVio@FNc>VPU=`RE|(Ruv2!W#1J-lbI>;Tu-VQp@>DAV= zqLtCx)lZmz$K)#bVx0OlN{hXSod}GfZE(fzxlVd@YQ_d&pU2k2y%QEPer|m}jbDeh zt3IE59%?f|+KNIFqsQ15d{f{w^S4Y~S}`ZTh%2QgF!L#rCgy;CO(flL62jF&9|ehC zTX$vSHW9{=UjcdsCCjrW6R;J0|7B{{7P*351(RT64wMnQ0bYUR9t?bQ1eL-@sEKyZ zSU-!sFci{Bk;-WOL^eg1KqGvHB=Ux&Gcdu|fbVVc#d+r_T8tx?8a+Jx-@L{0lQCD| YFWQ^Z3bAbeng9R*07*qoM6N<$g2$wkng9R* literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-hdpi/menu_gift.png b/TMessagesProj/src/main/res/drawable-hdpi/menu_gift.png new file mode 100644 index 0000000000000000000000000000000000000000..ce511f8cf7d768a50f9b5ccdb1d8b4d3b0558884 GIT binary patch literal 861 zcmeAS@N?(olHy`uVBq!ia0vp^Dj>|k1|%Oc%$NbB7>k44ofy`glX(f`xTHpSruq6Z zXaU(A42G$TlC0TWzSWdSpS4N|DZ_xKM31JiL&7srqY z&bKpmXNv@i94k-i>4>7%# z8W{^WOl@W3>{#hED`BxC@3D{H-`+FcS!~=NYqW;*4g39h#phqovozjpynE{{LFX%e z_JuO@mHrvYwAVjSkLX|IQN>%Y@AR)b;Fr6`|NNA#37aKE;t)l#qsn^$Sq#h z-cNe%)c`G zr~SM=UFM;f$h^aKs~hY%?mIMpaXf$je3a#(X*au5UTsr<$#Ss#A@7U%^FFURzF-6U ze&?t2nys>C_vFm#__OKc`dNph{#?5-Oa698ctPiS)~i0vX-XO53R%Y&*iJsE!gVY% zs)p<8TcbZb&R-HG8l3|sU&y>L(XQc;^NN$F)}K7NpRqi#gInUAgKt5frmZ*Yg$qry zPkNmS+@z3x;;H9~iPIeabe)dfpz?p!^`By^e(uY-ruBQv-kkTxB2G`Tdi7oX%EjK# zW|xI;CWz0Io6>*gMw89kqdJp*Uyh!>GCZc??Ou8Rg_-4zfB6k_=c;Vq%TrP$Tqyl& zu3LOgexl?#fd#k2FI+$B=dAs2M^|+9^@qZFKd0|}Xkz-(t)!qwGV;i_+S`+p*K18q zJgHnBST4=wAGqV4pT-;y6w*#ic+-5{mCs`;RN7;rb8l;FPB&esfA02a<8+nIH6Deg zZ?_)*rnS!Rl9h|Y z)a}~sx27(alQ|T5c;A%Q8GU1$`iW-;(I2% d+|k1|%Oc%$NbB7>k44ofy`glX(f`xTHpSruq6Z zXaU(A42G$TlC0TWzSWdSpS4N|DZ_xKM31Cy+$i(^Ox z=iBM~^_>Gnj@wS{3RM!ESh7HN37epDXWt`vgH0Q+`T(FF9u8y0KHJgr5l0ipT zt4lzoo%-I~`M=+-{{1N~dWrhaZ+ptC&V8?|uHU|XeR}L0#_OfQ3mA-Ir(Dd5>O8oU zU*YzjW#0ZBCo~Lq+*j6;cAJuXyK(~GBx@$NoNSv)qjs+Zi4cc#g>$%TESz10o*dNd zIJ~OsiP(IDm7IUJiueR=;9us``nF4wRWWS}TmJ!zf`fV%49_3PwwTtQm?pW&KU;`p zYU2&=h%I_1DnasDLBBN=Jeqf{{Vn9gqR_eTjpEIaEV+-@R$o{$x$5T9qif9FF3Zdc zc%t63`OJKcZEL%}KiKt2NU^Bn#%lAl8nulY_MF;M!bu#5_OjU>d0q0iYJu~6?k&oT zjn{9#uIsp~?^v~ez+!DCxqmMFvda6oj+Ec2`7Ka=nD6w)l&$Vt9M=_|@JSNTJl^nT z&q|4s9?ib=xu>r?S2gWXc6xWX-9axSDcV&0pRk@~Us~u>-wZZE<|jHr!tyWWS-wg; zEzyzn5^&u$!{fiqgVT&VFLg}yO=9vd=Gt@ki_1r@A7u`+ICwXzY|Pxi64tn-lKt6? z)!%>2=QZS1_#U_Mm+g*kVt+pib^Yo+q<(D2shY@-Eaw)SY?iEgX4SDt*ja6M4y$0k zqtcqkCAn)R-`e#stdlKp;>|2``}WL^6&C${WnO=u--+K|W1ZGnXLbja OU_4#@T-G@yGywpF$`UXD literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-hdpi/menu_views_reposts3.png b/TMessagesProj/src/main/res/drawable-hdpi/menu_views_reposts3.png new file mode 100644 index 0000000000000000000000000000000000000000..ac87d66d332eb85146e0c3aaced9078c3bf5659d GIT binary patch literal 669 zcmeAS@N?(olHy`uVBq!ia0vp^Dj>|k1|%Oc%$NbB7>k44ofy`glX(f`xTHpSruq6Z zXaU(A42G$TlC0TWzSWdSpS4N|DZ_xKM30~5Qai(^Ox z=iBLq`p$t8ZJ{b_SB0>6PFG`_EYj_%)N8+Rain_lLH+|rG~6~{^ti#}E3PDV(9cDO zSIF0P{hX{-+rQ6=&g;CRUs>e$zHS24yK{NoJdH)79bpG@S+lm@vRUJFNLM*U z)a}#)ii9s1rB|6%sR&Vs@lz*w^Hgy z)yMOapKeRU7seZxFQ^M&v{undIVRZ#3zXOJz2cI+BR*Sn{|8QCqow;zxURQb s#VnWbZ20%*Y`b-Eh;3A3{T8P8%ir0va-2JT9F#OXUHx3vIVCg!0JxbAjQ{`u literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-hdpi/mini_forward_story.png b/TMessagesProj/src/main/res/drawable-hdpi/mini_forward_story.png new file mode 100644 index 0000000000000000000000000000000000000000..46c001119bd98af88f7492afe96c3dc45415d76f GIT binary patch literal 609 zcmeAS@N?(olHy`uVBq!ia0vp^5+KaM1|%Pp+x`GjjKx9jP7LeL$-D$|Tv8)E(|mmy zw18|52FCVG1{RPKAeI7R1_tH@j10^`nh_+nfC(-uv49!D1}S`GTDlgf?6jwgV~7Xu z+bO;pE`cJ)(>t6LBRi53c_PmobDFY7RV*yb>{Q;VzA000bn|WuF!S<^&^mS8Z3=Iy z;}hdnrTgpdSDu^utoM6Op3a&vE!LdDrTk9?p${{L2j_ zdD!O1wB1+XS>F`BVdE*K@8-s_dwFJGI=T79x^>$xDoy{moU{2_Ne1gIcIjF23n$J$ z$9o}$SGj;!NYseuY)j=$1-*seqQw_36nLg!_e8Sgt-`gH1`FG)x179Y7t$kWaq*b; z7Wr2Swi_qzRgqxmOwaKBvZIJ6vVccucEBT}A1Xg~EorQ1D3YIDysf$K>sfSi-{G%Jfu{P4PVc(setOHk zCDpwd>Nn!A{eGAby<{PS-hnooEsu6o_3N%Ubz5xfSA9)h#R-nG6J+nZ%ws#O@ZY*9 hhWF#w|6lLhHiqYg#4poVt8D~Ducxb@%Q~loCIDGs`iKAk literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-hdpi/msg_call_bluetooth.png b/TMessagesProj/src/main/res/drawable-hdpi/msg_call_bluetooth.png new file mode 100644 index 0000000000000000000000000000000000000000..e467aa35d3c26514020787497fe4044333c83bae GIT binary patch literal 1044 zcmV+v1nc{WP)Px#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91B%lKT1ONa40RR91Bme*a04fgS`~Uz0en~_@R9Fe^mRpETVHC$_h8fq0 z$R%Y`LwHbTUTBobphogQBT14FrAc`r(|DLCqU1q&@}OpjJWL*>Oi3Z$zWl>!1>=)uOwo z-Q?vC&(t75SN$Dqi|Yp80wPfyVX$ZnrL^v0?A)+Fl+#qGmypM z8$lm1xsNf`i#Omu1WlmS_=1|`bwE#$NwZ%B89R`2h%Ev=R*2qbwb++E{x}i&Lkq?(mwJdCCm*2e3LUhT8*FTV*lq zQqT(|9mWqyQ#c-6Qim9M1-CqG2&YHfq|J{*UU7jQpC6J<16^6WYh>yr#=@`>T)ibY zo$_VHN0g4Scl)NAEp}p0uX0A>3im39`G0t-n46BsapEM4y6{ z;H@g1yfH(C*lWNiu|A2Kg00|(E7Y6+kQ*{@Ajj)n^}^*#T?ggJ?dhzOrFu93Q_$eppm{p*^fO;FoEt)jY2J+u~_1j&| zWH|fIQn(Z72uT)_DuKP1)VLQPESoUuwBJWp99fZgVrhk0O;Y^3+xOE+voZu z>hnT{kHC`LXvTxj83=zS(Ds}3W9*2l9TkqnvIAk)fD|y%2iUSvS$LqPEU3T6#GXR= z?I8ONJy89K^DL?_LtpgT$UC7&eBq>q4A2U65b9oAm6ThI6?b5)ci=y4rhi7>0wk6I O0000Px#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91B%lKT1ONa40RR91Bme*a04fgS`~Uz1B}qgPaYNDI zjteR{4@1O#cl`dLs;S$z@66>o5&{-{Q+2BTQ{A__PoJK$vXAyZy8`7+alUnRbpz1*5nXx+R%LFWtJq z@D0p{P9W3moWk5g^`nOV&}168bx&(441mWb*psdo8l!IytKqH9eS@D^NT1{p&9@11 zEs{bA$Zs)?Zi%i!VvH-H8hR$l5Rjh$k3gpReM31vL@+~2vu^o{vNCEjU4Z|>HN>r< zHq>qtuY&fjk!L&(bcJLWA?HU$t;8dxG8>aqv;np8F2Ij)4RH&&6>1*dB^cs5dB$4B zZa>6YPK=_ntMGGjiZ&qMFK8x54S8K9U4a#F2Hf=e80K2yR&d9PSMkZha!ukX{%gWb zNiqb~dRkp>MkwWk)SAOe&}%|g1E0Bwc#e{0Uwb~!BRFQF^La%Q)Z#N#ou!F6^b4(@ z+^=3E?iTV3JfG(gEegVZQ9xe5`kK}6Y8>lXF`8y@)aGbqwR9cvb<6MboP4u5Al?Oa zmYbTYY3xh-7hZ0!Q}w(dnztmkSwSl1ad zwGZk!kXFZ8JFxbjWGf-op7b)*8s8|J;%qtKC_AW}PVN)$OFRj5+hp3smqW5Zh&#d( z(3023-VRs_6G6{lX~yIY;I^M29`swr54?y!4%P58#5}_syU$3UBT28K&rXxzAh@lI zlSx*=WiMv<3*`O)-@Bq@YwT5YOb4yHe?q6Tw?^}fRw`yEG2)Pb07S4d% z$fzkgX7-nG%e&x@KxJ&eSYHCOU{K7bW6c9?BQos_r5Teyhd=p-;O+9h=yrk3H&`!~?rGKq zeuFm|PoDy9 z-#<|k1|%Oc%$NbB7>k44ofy`glX(f`xTHpSruq6Z zXaU(A42G$TlC0TWzSWdSpS4N|DZ_xKM`*(OgH$B+ol zx6}7}u^0**&p+y<%c9M#&F%O>F(lyr;ZHWv+SMOfYZx6jeAH#(fAePDg{TeRg!g7rFwhK0WbA8anX_#-NwXI*GzHi!v_Vjg%wc#Nr#2-uko4;Gl%=ER^ zzSu2N8a+9;s!#os=vaM?_d@2)SkF7wQLCqIw5VElW6mzs(0_QPq3odi}BfClW zrrbYglZ{t?T$1um-skjUs;!5&lNtA!t;-%tiE8X z|4U?cM=3rJo^gTe~ HDWM4fe|YQs literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-hdpi/msg_call_minimize_shadow.png b/TMessagesProj/src/main/res/drawable-hdpi/msg_call_minimize_shadow.png new file mode 100644 index 0000000000000000000000000000000000000000..409f662ea1387cd91846be1800afaa17ac465312 GIT binary patch literal 861 zcmV-j1ETziP)Px#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91B%lKT1ONa40RR91Bme*a04fgS`~Uy~$4Nv%R9Fe^SkJ3cK@|4hs|bO) z$zR)cX+eL2@QAkz-1J{Of_kl6)TTBCZ3P!D+J-G%7a_FM*Cr7`{81^0lA7Mz_lYBO z=UuN!bF;-ea5!^k&OPUwZ_dme4TY}c|7E}cG^$H3b+|@@d02&(8-EQAQP!-S`Fu*)aCX+8c z#&zM?#<6Cz*=Nk!p+zs55~4CgH=9hsLG}~=T5Nc+SbSwBT0{(QqV5BuGN}a&gLnZt z8kk*x6`-vw>CvC;J}%=#L#lN{6-LKC`SV0 zVn_x|$Ea{}oY&M%GnM~%nVHnqtXhxJ?LiQ$;$=gIC@e1Rv535c*UiAL48~f8&VlE^7*_~DwR$nkx1Gvlxdpwa5(&w%jLen&qcf6dxwsh zNb=0L8l^!L=B!q$-9!>~Vu$A4`v28^2*n#srBZiv0uomQfzT-w3Qw_QAJruT8%8V^ zJBi2R4|D<$iTy0;Cv&n!y4~($Zr9C<3G3i?cz@&T7xRzDyq1hRheb+d+rU@*ES*lj zV4Hp8@%TM#AmRl`Bw912+<~25ulEG^<`2S2x7`KE6U<;RILAr++On*`wz*o=obXOC z*ik&{xr~8{%FsSAfiWXyr~)$8n&C2Sm=Q%D;wdcxz-s?|h}Z^yMao;7cVY~x7b&A& zuiwSF^b>cSAC*ew27Jh@c98+?y91lmYBg0Zm#+hbRxoNX{t1xfRPiUk1Px#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91B%lKT1ONa40RR91Bme*a04fgS`~Uz0A4x<(R9Fe^m`i9BK@f&BYkZ3V zK@)r(RM3;hXjCL%E_xILdJKsdJw`+n!NaOIy(vWTrr-kvAt+wNfJ8iqcnN#j7$m+R zlK8*}$@=}9uCZrkb)98)K}f+*RabRa{oB*i-Lp+ibMl|fK#&#RQmM3uTsy%87z~1- zm{smnc_6YF+y=gUrEw@#qAWU7bOCtaF`4%4&)Dl#2i*d;f?O7LE9M}1Gq?*(9%8hC z1Li>YIYQ5z?-lwKSW+=*txgi=z;$3Ugb^0O{^0^=9id0AZE6f-U#;~M<487OnbG?7 z%l%HYjnFwqNbBq{cn(bTc#r$zYO+ak(K4c6f)(*Z_#%WXjz0iB;0rM6w|HtzhP-7# z7eH&JB={>5vcAN3gHJ#*1NK@hq6sVnr@<4T1?Xd(wn-6VZDg?5Aof>6>PnyOU=m1F zXPK9x+(B>6u%cC@NPI(NaI8T*K?q+LX!pC)?gb_%{FoRYwFr6yT#Mly@b zL7xaA+U-E1$CI})*a3I}Bzp7aSxkg($&0)J^TR+52IJ#|R^xjQOpaJu9R4hI_&et+ z*Qvlc^wS1pp~*K+o^4>%@QFY`SxGw{L8>)Ow zK^H1cn>uDG6xAYr3bwes7g)>_&ff2|)Z@@*P(AmdPV$@F#3*lo=tbbQ3zRoTe}=@a z8)yo3iVufuXN?)u2)M$pn~KG%HT8CBGn{1X4X?9&Yh3#>vFR`dfx4b$VIgNwuLDd0 zi4NDL@jhB$4LR+pvkMrRKVLr*3}X(G3n|yJSebN95~RK{GDgdx z4`hFZ6@3y90F!^YQ~UijF7r!%j-+Y6Ixac(TD&Jgasj7%P|4mT)nkZ%m$l;S1}433 zM%8yCsPD-@fofl_>ykgE&K^kVo9hnnMV~nbYbl-OzaL!?(7zua!98}}QS_W-J_El1 XLJPhkZ%RJS00000NkvXXu0mjfyLFA2 literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-mdpi/menu_feature_color_name.png b/TMessagesProj/src/main/res/drawable-mdpi/menu_feature_color_name.png new file mode 100644 index 0000000000000000000000000000000000000000..49a4e707a1ddb3e6c5fc8d275a674cfea8ce3328 GIT binary patch literal 762 zcmeAS@N?(olHy`uVBq!ia0vp^5+KaM1|%Pp+x`GjjKx9jP7LeL$-D$|Tv8)E(|mmy zw18|52FCVG1{RPKAeI7R1_tH@j10^`nh_+nfC(-uv49!D1}S`GTDq2jfhon)#WBQ# z_v+OB-l2&S$MzwDd*B34N9~%_N zFTmnxxYNo)BJ}am>%l9pgrDy>ldNHiJH+pyq%BhZ)L-u_v&fZ)=Xc*QsA4ReBfor= z$F>8;X)zBntGL7_*D%k2kaDoew&@FR!;`8NQyy$A(c<8mFZYUr{e^y`X+=)te^JvN z59X+9>KwOAscX6SXljmD`#RsE^XGRv{(UXrItEqg);^PCr@ch8(T zz$xpnMrO;O8Yh3(v&BES%NKARQ~dq$;S-sLcz){u50e+O{+8Y-v;F-k`5#}-$#02A z-zLm%Y>&rFwZ$CZoxNo!SV?< z41a~vTN~t_tG`f~|NY1Ikj{5ZC$nF)Yg(SLt>Cr__rHF>`|oph7N470EZq4&*6$AY zlAx7ERauGRQ=Az0G)}+3-rVG45lIUdXEEP{mqxI^$xA=iL{AA@1esLSIsxg?$e5TxscBxFUS&F^BB8T>Dvm zEx1_Z=$&i5r}<&RS+$#Pfu}$5bp7=`((W*8gU!FX87-6d#Tb69>b|VgI(q|)^(8Kr zC!ZXCZ#tg!!*fDjq~4WFAq@5p)|oz>-yqiAJS|G^%Au&$5A&N=8!)9OY>hr_$r02n zbR%@Z_KBx*nl!xQEi?G{9T3ntFf}JXU}Cy@>d%5_8rvSPW3COExF|fo@O<{Fgg@Cu z&pT(|l{uI>M6d%lz?NI%L@t*USS{G;A8rHS11Mf3$ z$=u~Coxp8!s9b`j*12Etm&fG=-Pv*_y6?aGuIv5H=6U+0P<#XH4&KUtQ7il(YrIfw zz19-XJgGXU=4V<_)#aGeN26w$A1E|UD(!xi^0M*q1T(=xdz;?5CTkiGhAvROaB>CP zcYU$dOKrG=YH$4L`Q#J8x$Dr<^B)-On)Xd!x5M*JyvX$)J177VgDDi?2VAeI9eJF7{dj^Lg9ido$-jF-&HMD?XB|yC&+^uGPD5GPgQotEyB7vO`Y+|W z_3HG6tR;GOyVz!$-aLJsrS1^N$&Jw}ymb+Lc_~`|ZvQEDo7IkKbh;$#CO`9<8PLv-O8!kMSjogaJ{~h?K#WVKlR~I688_SeVjAbS6_G0zW6q1(`Hp|qxOO~ zE^cd=oKK4Xn7-gf`J5we7dXYQ&TYBRmbGcsq3=l|*TXj~_xLDn$lr5OS3|iQl)OA$ L{an^LB{Ts5$&)TZ literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-mdpi/menu_feature_custombg.png b/TMessagesProj/src/main/res/drawable-mdpi/menu_feature_custombg.png new file mode 100644 index 0000000000000000000000000000000000000000..9ab59e359b74a45aaff5237a5af8e1b37a12c444 GIT binary patch literal 741 zcmeAS@N?(olHy`uVBq!ia0vp^5+KaM1|%Pp+x`GjjKx9jP7LeL$-D$|Tv8)E(|mmy zw18|52FCVG1{RPKAeI7R1_tH@j10^`nh_+nfC(-uv49!D1}S`GTDq2jfyv*~#WBQ# zckk5m-l2g4$Mzs>BFD?;2w(z7?;`akve%cAiG-dI-NIa=$&AfAI zqwtQ@IxC@!miPBGIv^6ETitT*LU_c=PNu-6#Mg|Drt%1+hB{9*;*GZ zy?rJgkzo=9+xZj1pMHL_kHg0yo#ZOxhmT%dh-Np zS0C?dvVV%zFFe-Hv{3O)THm~nedUJt2P-U0XJj_(zp^fAMqb4j<$6 zBzix7R9TiS@NpsUbGD4ul!PPKXZZ+)`Zxa(XTSXDoV07;o~G$@%6YbH@rGa6o%;Xi zoC6^NmxbQR{1%9~u-5fKO~cuDyYz3T7kA`N;F5YK6IZxDF_I;kG3R06%+Dv{lz!g{ zeR`9kl7-#@#PvUQup-hXoWFNJnWZ1a6&TeD{7 z%e$RsD?0K|@9;^=X6BAM`LpJuSd7s7X^+K==fwQ2Nj$-JL;LQej2&zijB``C>kB`G z?%_OnpkuvtKFgI!n)=6Bdv@N5{=}j71o@UYGU*VO!!>b4Q7td(N_}L;YjumxrJHkG{_5{(bX#q4qPU*T*mdKI;Vst0LoVXz5oCK literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-mdpi/menu_feature_links2.png b/TMessagesProj/src/main/res/drawable-mdpi/menu_feature_links2.png new file mode 100644 index 0000000000000000000000000000000000000000..07e4e8b6fc12f3b538225c0c1ccf6be4ac17fe60 GIT binary patch literal 682 zcmeAS@N?(olHy`uVBq!ia0vp^5+KaM1|%Pp+x`GjjKx9jP7LeL$-D$|Tv8)E(|mmy zw18|52FCVG1{RPKAeI7R1_tH@j10^`nh_+nfC(-uv49!D1}S`GTDq2jfl0*E#WBQ# z_vsW{Zy864hJ1cLVS_~qo=X*3bhk_v@Q^&=FL1{6ALIJ{?_SAhGEZX@X_=sQ{BL_i zi@=Giw^V0@>9Abw5(=C}S_cW%$;bJw!IXYsTjw%XIGU%;{dfp$s4 zKfj|22bgM@kLSNQEXVVJ$uHsW{z(sZu)O;w5;dWvF=G8e{~ay$-c8P8fA~LNH7Kxf zH)mD*sIpyO{Mh@45@ClLHd>m^J5p4d^lJsrV>Tc6se+2z?wxd8wkmz{hwjGI!sopz zp64E#T(NdH51+F1wS3L|uowTV^d7%yjhgo2+{FFN-hp0CXWm(LsW~KSOWon_y(UvB zE^{|YAcgbBp}mXFDgW+xQygu0OZS9!v$VipF^gFnS7&e$JdV{^0$yJ5PSP ztMuh(^`1M*Mv_L^?=+^WUX{LMy?IafDKF-uieI|9Qi>n6hOy3fo6Z`q_pAT9#LU`! z9S*e<#di<=V%>A4C7(Ix<a zP*kZNx2Lgx0lVF?!VecDcUtGSZ|Xe9xsUO@y3`l5gX|d8ZFEvX4JEa3Nm^OoSah|=YPQ4)vjpA}(NeRej$18{ wXM}Kgf0&fk8d&dBpKYQmP=EXD^t<~QwzBih|NLhboFyt=akR{08@7uVE_OC literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-mdpi/menu_feature_status.png b/TMessagesProj/src/main/res/drawable-mdpi/menu_feature_status.png new file mode 100644 index 0000000000000000000000000000000000000000..3db161145accf661d00a7168107719825d3fb23b GIT binary patch literal 737 zcmeAS@N?(olHy`uVBq!ia0vp^5+KaM1|%Pp+x`GjjKx9jP7LeL$-D$|Tv8)E(|mmy zw18|52FCVG1{RPKAeI7R1_tH@j10^`nh_+nfC(-uv49!D1}S`GTDq2jfyvv`#WBQ# z_wH0*50ONP(;P~%)yYy#w%hQ5`&P{ez5`4bq z`(&wW2R4}OSY(sUyhh>whHvd(Rblf*HU2MEC2zI&@Q317bFmegRZQ6%*i+5EXhwx4{p0v$ zo#G_s)0DklAT#l>e8g3j*K8Tpdd+nOxu@NCJIMVqFF9=R!aV4`@`SrwyAKGQWR+gD z_QHhk&%_sf4cYpPf5CRgYmZh=jMuY2=&_~I@1(41Wc~p?o>}XIzZ@;$t8%xC*>EFy zOP-SRzVjSnhpkS0QN9<>n9cg)kNGV5+L}L17tZ$VXuj~+;Q6n8Q;%rqIB}k7niZY< z!uHFa?Tqo&|9KZ3pL(@X{=(z6-I+qW4y2#hwlFrsHZOk73FQud$1iMa_Ho3AJ=N^? z7P_7&Z09h~H1oEYLY2+qtDRl5ZY!MjKNHomL~j#^c7Rlr#DoR=*i!a!|FA3Td;DU~ z{g(9x&V{uH)@+#Y`=tekcE?f4IgC{|`4-qtvv3T!&S-V%rTUbk!6M6?_ zwPBxY6=Uk^4{To;r|fO$c5FWXNB;BzFE-Nyp%Lwce)mOQ*JrsaU*7TKjFJ9!Q1bG0 L^>bP0l+XkKs?{!o literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-mdpi/menu_feature_stories.png b/TMessagesProj/src/main/res/drawable-mdpi/menu_feature_stories.png new file mode 100644 index 0000000000000000000000000000000000000000..aafa5eefa31b8673a389b664115dfb8eb3883e49 GIT binary patch literal 787 zcmeAS@N?(olHy`uVBq!ia0vp^5+KaM1|%Pp+x`GjjKx9jP7LeL$-D$|Tv8)E(|mmy zw18|52FCVG1{RPKAeI7R1_tH@j10^`nh_+nfC(-uv49!D1}S`GTDq2jfvMKh#WBQ# zcj~mW-XVb!$7J80RuSetrc+>)8|dWHD6orD{Gq&nZr1ZRa{l)_*2U*-lS4yU zmFJ(&Z+yiSX-&i|)&aK27Es={^jSn&YVEmrks^KWl zZX73E@}uU`ip=Y^%~7uvcXCPThUsu!xc<eF!|N%*0~#AndG|t+?H@`4_mqH zWZ#u97xGCpT5w-p@R8$Cs}ggiGLK_Ljlh}1O_2xYuDr=DFrQJj=2xHoj=223k5Ye{ z|JpJCvYyc-I3Zt(xIHH|#~Vz$RMI(%&%Uzc6kD{x(bcoBcFgDC*tPrq zg1HkVUX(6ipJlZo-7)=Q%%PSQ?3m z^Q+|JxtZZ(;fFk~$15|^4F4KW*yUQ={g=Ige|qZDc&kcKg7kFt Kb6Mw<&;$Tfg+Ryv literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-mdpi/menu_feature_wallpaper.png b/TMessagesProj/src/main/res/drawable-mdpi/menu_feature_wallpaper.png new file mode 100644 index 0000000000000000000000000000000000000000..2c8208d3f65b5e1a8af4c4492d295ce41ad1e7b1 GIT binary patch literal 637 zcmeAS@N?(olHy`uVBq!ia0vp^5+KaM1|%Pp+x`GjjKx9jP7LeL$-D$|Tv8)E(|mmy zw18|52FCVG1{RPKAeI7R1_tH@j10^`nh_+nfC(-uv49!D1}S`GTDlgf?763lV~7Xu z+o}EKIAwvBS$8#NvFi5`3KS?PjA&MKkLf>#B53hTtHE-lWE z5lz2n`bhi2lIQm?7~L%w)l}!7o0hMC=hoK~)ePl(yps=JJL8@bJt6-?eZynUXQeJS z@8dcSm`U_6t9;eZeqlqG+wlt=&IJx@n9gp<=S@f!kV^3WmP@}`7BJOzZo0~pY zF^}fm@`v#U_eRa0TKh@)_d~yk ztcLf}4iT{joO68aOae2^zh6Al9b+eby7=iy)g7;wIG=sL+@4`~i&uvBvJKBbQSa&M K=d#Wzp$Py+pa$ar literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-mdpi/menu_gift.png b/TMessagesProj/src/main/res/drawable-mdpi/menu_gift.png new file mode 100644 index 0000000000000000000000000000000000000000..02d9fa3e9586c0bb671b026a32a4062e7d5c314c GIT binary patch literal 601 zcmeAS@N?(olHy`uVBq!ia0vp^5+KaM1|%Pp+x`GjjKx9jP7LeL$-D$|Tv8)E(|mmy zw18|52FCVG1{RPKAeI7R1_tH@j10^`nh_+nfC(-uv49!D1}S`GTDlgf?69YcV~7Xu z+o`*~m>oshgiQsc6&W3K*5s%>QjUnc(K+FmmHtFOf%wBp9c7LIIcr#=4|3noXzG>V zZtaMek@ELn>AAE?IYQ1;Oyqu)ottxe&fS@JeO2649R-#o3cq{6doJOqVyxLuk;!s9 z{3m$@G{@d(Dqi7y_~oXP%V%72;GQsnQB*5Rsfue=-;{Lyq~M%mIsYsk`rFvP{L+@r zeWCS;+e@)z%T2CAk}nmyOPQC=EuUM+xbC6ovqUkK{sQh-Rf2g7cqXOa@K>9xKdXg1 zpoZ(O(6a~3K3_CSrgsG|FP+eHSmK3W)(1|t$r}6{jwoHdaDu6Ff=+bEosR!5M}D4J zCZ%xl^fEn_ZI0phFE3mdro1MTn=b8>$pr^q z)VGNmFXp!UYhH8l_HMQ)g=WbHq-+JYl$465EYmd#IdoLfp ztMdHSnFgJYMmh(VcrQ-c{^R+E!xr~H@3EcQy{kj(LW|?uZpQNl3s<+zYj|RFlDneS zcSlS4i&WN0%LAHt=RL3v^J#V~7dsj;(QAS30n?fTtGsr`y6_jgIOn!FWAz87&aU$c zBDeddb@eaUe!3^VEqTN51X-^Wvz4Ef>z%BN{Jbq_-GlstYdT-DshM4R9U;BP@T>X` z`KQV^c$0MPZ_csYyX?;MFSSj_Cf4%kPSkTNto#}kvzn7j`Xtji<qw;3yY=!eQ4~r`Db*oR?B#|1p1_+B5$p^D+(?C|y;#`u~np@2A5) c1v$xe3=5Bk_O$;?V+BQ(r>mdKI;Vst0M9$w`v3p{ literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-mdpi/menu_views_reposts3.png b/TMessagesProj/src/main/res/drawable-mdpi/menu_views_reposts3.png new file mode 100644 index 0000000000000000000000000000000000000000..e8c64ff6264fa6b469bf9c214b76d7459e715dd8 GIT binary patch literal 506 zcmeAS@N?(olHy`uVBq!ia0vp^5+KaM1|%Pp+x`GjjKx9jP7LeL$-D$|Tv8)E(|mmy zw18|52FCVG1{RPKAeI7R1_tH@j10^`nh_+nfC(-uv49!D1}S`GTDlgfEXC8sF~o!S z?NnQTCP#s`_YA5>nj(a%q?|q80-yN@u4$%)e2%{hCMamx`Sb<9J6_{lJ*lNadfh^8+4Hq4ju+J>b)0k9wmamHy2EZ}`GQB! zLu!t9z0Uo#g`d~HK9={-uI{Z@AN0N8i4$A3Y2OFG-3xkij#mj>Upwt>k7vTvv98?84#WZSa+181^M^xvKyG literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-mdpi/mini_forward_story.png b/TMessagesProj/src/main/res/drawable-mdpi/mini_forward_story.png new file mode 100644 index 0000000000000000000000000000000000000000..a871596cfd0cf9fa7ba9089f66d16e93c2b423f8 GIT binary patch literal 416 zcmeAS@N?(olHy`uVBq!ia0vp^0wB!61|;P_|4#%`jKx9jP7LeL$-D$|Tv8)E(|mmy zw18|52FCVG1{RPKAeI7R1_tH@j10^`nh_+nfC(-uuz(rC1}QWNE&K#j#^vea7-AuK z_Tt7YCP#r|h5QchDR;}p4zqm?PV*esh;H4qHNZMsTH8DLsi@1FwJl3L zmM-l~{QUnA>o-5n8%q+yzby^brNUgIH3{8pkG?rQ3zhlE^xxCt$cHvnxi@ar zK5yS&E-H(2mvPMZ|H0MB`PHsOJ~O`ah#a$OsEc^9%HBDLE4WOzH=Wg-#3lP)^4B?L zkzd?;C6&95w5`8+d#{J1Vaq)OH_k4pxP`v+^=9qmRT8=W!|D8~w==ZwP27JgM?F%6 j@66&STphQ5{+9aB@Za09O^ty^85BC6u6{1-oD!M<<0X|G literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-mdpi/msg_call_bluetooth.png b/TMessagesProj/src/main/res/drawable-mdpi/msg_call_bluetooth.png new file mode 100644 index 0000000000000000000000000000000000000000..dae338222da11e8612c162aa8248e19dc7302259 GIT binary patch literal 682 zcmeAS@N?(olHy`uVBq!ia0vp^5+KaM1|%Pp+x`GjjKx9jP7LeL$-D$|Tv8)E(|mmy zw18|52FCVG1{RPKAeI7R1_tH@j10^`nh_+nfC(-uv49!D1}S`GTDq2jfl0*E#WBQ# z_vw_g-XVb!?aLe4HpxsVxWL7ublkUF;$VQdYfyL8nL}rDmc(`|I11d7%sp~o?t(30 z7hJYSec|WO{iX72LQ$Y&_r4Z|BYo%JpQ~&)F3)Q<(mNtPuX6wW;&XG0^R>3>AF(VK ziz$?oS_HC=v(q(iGK-+X1UTFX|#>~q*xy5XaXy_l~QidoZq9{h4|e*Sdzh!&WzxFMXZEbdpP^*If8@u#jZ$b1l1*j2pz)3GXiQ4Q;7s zS6Tm{#Wd;ltgP?1B4!_5ZPKaugIRt-tY^tpCc8BCov;5YmELy@3R>gl@nY$u7mLfz zaBWyEc0itOZa{wZ^j8-qf8hJ~Z>7Rb#>W>@4%ZfMVfd({)AWS($>CkHw|O?5{~GP< zmZq@F)ml!<%qS-ySg>|>Udzs1H+k&XbxWmpEN9DQ{Cag=!|@J{bWl3+boFyt=akR{ E0JGu~y8r+H literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-mdpi/msg_call_earpiece.png b/TMessagesProj/src/main/res/drawable-mdpi/msg_call_earpiece.png new file mode 100644 index 0000000000000000000000000000000000000000..c4ed82f919faaf1059c6920e3672e87db1a3f2ea GIT binary patch literal 795 zcmeAS@N?(olHy`uVBq!ia0vp^5+KaM1|%Pp+x`GjjKx9jP7LeL$-D$|Tv8)E(|mmy zw18|52FCVG1{RPKAeI7R1_tH@j10^`nh_+nfC(-uv49!D1}S`GTDq2jfvMHg#WBQ# z_v%z{k5EU6Y^AP%$L0E1 zW`bUu=%Za0ukU1@7S85ee<(fvHAmLnC&vt{3z%2%M{zE`cU&a&Tzc;J8ufKrw@w~Oceu5h z?af7tqIYU$H8Yw%UdhsJyw7o4M{k43v&ORaYoXTKP08|4pI;KA%E$zJ^S91i8Ee)es!PsR_R}Zm*2hW<}BM;Aiku1gNdf!qYIZAC#~zb zv?wrrVXX3mylayu>{~ynbH!Wc%-BhDx(+w^esEHrcs5>?7 zV3oA(Lt%rLNtgd0wY~6GE0$aRYUkg4!;@TWZyVSmc(osL-)RVrxnK5I*suIAYjWsw U^Vp4CMWBS~>FVdQ&MBb@0NGkocK`qY literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-mdpi/msg_call_minimize.png b/TMessagesProj/src/main/res/drawable-mdpi/msg_call_minimize.png new file mode 100644 index 0000000000000000000000000000000000000000..c1a2ff8c953232097d1e5c8b8e4bb9da118e5199 GIT binary patch literal 466 zcmeAS@N?(olHy`uVBq!ia0vp^5+KaM1|%Pp+x`GjjKx9jP7LeL$-D$|Tv8)E(|mmy zw18|52FCVG1{RPKAeI7R1_tH@j10^`nh_+nfC(-uv49!D1}S`GTDlgf%*NBjF~oy6 zIYENeL-E7^|Mh+fdTdFqDv6h7IGpWqn8SC(=!8{c&4h>54H7I}6Q6n{F`LeA=>743 zzwk1_6Ygd*DV*v)eVoS>8q?ZWJ!QBUaM+A_<~fHw;vKb;l9KzGGYu4uGtG!=YHJn= zN@6q)u=ksOIMz{%SwiKS0}oTbf`~=Z!^FNkIqqEyoHdaPZlyc?Vd+@Tpv#=eD%jmn zSio^ak(+^c65AvDfa}SNpD9V7={Tb}jn^bpWJMTPchv;`b&Rzi*t=zNTOQak?BWuU zUr@59>!_CAf*UytPJ3-U@laGrZ6+6^?*sNn6?@bsYH@xMEc*7J_sM;wt2}dFUF2b( zsyn~ohngSn5|0E!nJ2=pA{x#XOnAH^iBa$H%1p);YD|n%KFMS`S_jPe#q( h;aFrrFMg{vd$@?2>@ITme>FQ literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-mdpi/msg_call_minimize_shadow.png b/TMessagesProj/src/main/res/drawable-mdpi/msg_call_minimize_shadow.png new file mode 100644 index 0000000000000000000000000000000000000000..3075eb64f63de51a0adc6828ed7d3364fbd6c307 GIT binary patch literal 622 zcmeAS@N?(olHy`uVBq!ia0vp^5+KaM1|%Pp+x`GjjKx9jP7LeL$-D$|Tv8)E(|mmy zw18|52FCVG1{RPKAeI7R1_tH@j10^`nh_+nfC(-uv49!D1}S`GTDlgf?1ra{V~7Xu z-q6!thaE)Ts)*kC;jnv(m-((Y{5v#W`Tr1I*R1!8y=liv>50E2^ITTGeza^h$4w5e z`IEL?Sl0R6Ln!UivXy34i8GGRtv)xm`uVvOqk5gHQ$Lt0X7#M}NjW;Fkl|3ASm4T# zYq#HSyO?3}Z9&JYr+YUh2Wm~-v^MOuhyp8T%LD1=gAXn?H$36lb>CvX;Id3PfhCt; z#xS)gh8^&9GU}8$eAG2!TB%D@-u7x<;dwRWu)B6cZPG_P}8N`TND2A6v-(t^4oD@$kp7xb@d}fBpJS zso|pSqPX?q4|X`ePw&jVU2ed|Z6WRIF{Aou(#2I*vv%){$y47Ie!X;i)Y^RswqN4- z1dn_DQDT^ve5;wHxgrfz zo#T3B@)B1A)_426v@->cG#anGX~JPrC*<>@Cp^&oXS~Vmv&9os@9kQ-;5Rpu{S`aU uv>Gu-#&g<-EMih^8KPdT)9(0h_><}P)v7S#Z!29uaqa2q=d#Wzp$Pyi6a9Dq literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-mdpi/msg_call_speaker.png b/TMessagesProj/src/main/res/drawable-mdpi/msg_call_speaker.png new file mode 100644 index 0000000000000000000000000000000000000000..8a16c35f2dc559d00b8b63f2f8d8fac3013ee7fd GIT binary patch literal 708 zcmeAS@N?(olHy`uVBq!ia0vp^5+KaM1|%Pp+x`GjjKx9jP7LeL$-D$|Tv8)E(|mmy zw18|52FCVG1{RPKAeI7R1_tH@j10^`nh_+nfC(-uv49!D1}S`GTDq2jfl1HP#WBQ# z_vw`V*}{nu$BR9ayRV3CoS4P!S9aoo%F+jyJOeKLV-}Svyx=HPlBu!9nSG&K20N=P zR}?$z(Mu~YtPI$2p=@Q^sSuU!(%^#^p8Tn~zx#WIu>Z+KGx;A+etxvGe^>kd)3Tud zQ-AEMm-5=c6mGSvQfgi^^8^#7r}5VW-faGF(!-X%fT>jD07u3Nqn@S|>2=N1GbC7< zKX6|8FirQET~ql5X6_HND)|c8pZ~4$eA8|_;ZE`7k7pUA4_|NI!*o(h=5o~Z$p#&@ z!fm-cwVZkfdY)Hy`~5MW{MC8Zy@ZGC8OzHTvOH7GtP+iSs@|A6`N}6}n_GkLWUn^ zvpskneAQI1lE3~B|IXSgFD7bDD1EW5`$igr74zp0yejz%_ntnN`R&RSHvhDH()(l$ zT4I}Jccf(tWpb5n(|1@jdsBwa{rWX-H`Ikqd%Tx*_B~3y5%J1+hk1TXPx#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91FrWhf1ONa40RR91FaQ7m0NXcg3;+NHQAtEWRA>e5nQMqtWf;e2GPS&+ zQIQ%eWEy28kxP-G2}Y0^VNxIw7}S@-%)TfQtn{G>q5?y|sAP5_!BR1yK+%+9MG0yn z(u_>o%G9#;iaP!N!+Fm<=bdw=nRm|4ChP+V_=TBhd@^_+{0(}O{vr7~&>q&Q7e%6h;B9ab zw3=2?@CvV`gjn+xih{Qgaj%Ig!S4k>f|EwyEPEvhZU9rjWhU8I@b~dr^%z}!EEn*e zcgsITZmTak%!~YEzvA!7MMyGYuzcB7oZSz4~(-N=NV>li;0(f^%}F{&Lg9 zcXg*FXuJtL0zb0u;2a&PI}H44dYR;>x0A)%W*A`u6X-7^w~UP@?s`*jy5Y-g)&-0; zj^7x*FW4?q)b>?YUv7++wXaO--RH3mC*T?z)CAwn-9DWWZ{Pi@WjpHv?Eav2KS4A< z6~*U3Yj+-WjGXOU!~s7Uy6xDWL_Zq5;oGw=!1wdzX(;k)Ub(LV?V#!cWaJ`NYfPii z3s^}q?U>`zk_PBB&o%_a6@o5+JMss`9tkPfNDQxzPQK5B2f(f1Fz5sw^l*qVod|pr zIC>)8t9P{&Zv&mK32!`wJYO9IJ%QeBy-=Vzt!`VvcvI^F{D4T-{L?PonY9m?xYn{X zrRFvQ*Zgt#hk=gO`XRLqhz5gY!B2ofA^Sj2G=It+_d}g|!lD1t`Gvkh&FlmBS6ZoX zH1sUeqKc-lm^=TGk;^Q8dh$0#`tzZ$Q0P8zg2Ek9@pz!A*$8jNib7!p&A(>kGK-() z$Gc!uWZNB$$pOc;z|pLvHD8G$Q$UX^#<&@{)g~^6PQTQ414rj!?uZ>6gYtS{6Xvzs zx)Sk7opInd;Ak(*u-GAD(6xiZ&WTa;W1;V-^>+Z+0UT|ra>_|Tfsvrg6=FOH^sOmD z=pF-h!8nOLC*kA-3@-ym3cAJF@*K%K2yAtQ81=15M}?S>i(o0Rq>H>gcCrkHY2bqK ze#+XL<;go-P#(rLrp`I|nKnQ25TK7+mb7$UXZ;d9#+}CdgM=|qAm433&Hia}>lLt6 zKGV1o7&Hg%2L<_RQPIwuO*ov9C!{-~T`ri2caPsbQRVhY=be}^-U+&mlP2qu*dc|9 ze|h1!5BrUgPZwj=&$pELi|DqqW-qoja=n6_&>Gg2*ZHh}mh2uWw%=l`;Sf>zr8qG>? zH={jSPDX#s@S4DVN8G`OfObkp3u<+Pla4J%oin3ezt$2(wds5aG?6W76{$Lu+3v7E zXac^rwwCcUu*o#x*k54u%_x(wj{x>*y_o6Tn+BS&XBx!P?_ff$n%GAHd*S+gA1}kT zVsp1q_-dea+maf32~4sDkp}}EjqUe-ZDRV7-giU;IvOtrX{3LdYN`Al*fYI+)%**6 z*SS}jW@Z5Y6p1R8^c3{1=l?}CrQ^sPx#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91FrWhf1ONa40RR91FaQ7m0NXcg3;+NH`bk7VRA>e5nQMquRTRe^H7%c& zJuDw-nwdqDS_Jm8(2;^DGcYQGK7>VnNGoa5rid)Y9zupi14B#}lwuL3mA$@XjATPp z5@ippQ8RHWv&WcDzrS<#z3c4R=iGbGz0-Wi1;5#QJ@(rBoU>;?W_tBHEQjB~|N9M8 zRr2vWIy!ojuR7v*%Htx`&mQ;zT0~V<)n3Y;7Y02H7Qh?u3lt;!33?wE!tfq>3Ob$) zjqoRwBYK#%aAB4QDR~Ofk*39RxG|-}efU~ocZ5QR!5MI@lmyv;y_V^)TG$6x^tGpC?tsR6^XqL!>Uvu1K&(irgdf!jc<)6I}&kx-5&!Aet7pNV!_Ut=>+-vTElHo$@3 z@A!eV_%t86R)UkdwAd~_S{(!W1$FW(mD9TfdInV+Iq@y&s6L34a*C`LAX zvq39c|D>3pod(Om%2mNY&q#wWZ6oR%y$}s*B>d&Y&Gq9l>m(ipc6VJ}Y%QuW!6c&+$7gFe_AQYh#vw|U{I(X_Tlh5SSjiYhw{a2WUPW8P3DGJ zK&0T;ZhI+|HaGL_XVUg#_zLa}_2F#z$HVRL4BQ8^oUcoXa~fE=*bl<_oC(lwIniZAbC=sGhnK5im+P}liBC+K?fzAVZGRH#A+8vH81BTv^VHYheg`aj2v z;dM@h2aT@k5Koud9bZ68>kX!_7pZHdS`2NGeJo{tRPQW!qRr+hYhz9nd^#6t+Bo@x z%0xe|9G!f(xa3h8+r+S)4?3|*bj!;NhUP=&&L0ouJ^320p|A*+LZt^ki=0~doHXWp z&ew2$;sWwkY8YR34oY1v{xhM!YH)jqhOq&T2n_)HRs~36espcyPV||k#{#-Zp`Vdp zm+#NeL%(l&$vP}L*%liTR&6qx*G{w}KMW_u0xPJHdoX+oPPS7S<#&NiE7SdO5S+~O z$8@5|p9u8lP)QK9|4Uea6r6!^~rO(`du zKH3Y47aE)lbtkJk1XlmJxm!zd3 zR(I+v;c0ji*261czv@fqxw))5JP1~{q1aPGTDR#w61{CQC*D zW$FW9P0rg{`G_L*Dw{sy$V(Ev&3T~t@8q#iW1KUWkqr(1YpsNz0{^pQ-2wvBMcCmHQq)cP5=g zbDBTK#W_2%y7)WM26=+>Riq@LmQMRm3)K_P zMfj$Jlh+Cjg20U~n6kQ1e8}(L$YL?v$l5X7WsJIg1v_E@gYnr5dW&h7RJH$|Dfq>b zk`%NvxVu4Tb!|lbTm$M_TIWDL*txHzc9M$=mG~!slb=E{-EO1zWo`tf)C#qrQ0K&~ zGi@3Q#dnLJ0lZ@dq3SNUx=?4t%vW$E`m@Pdh1ztBpN=mA9kQJ0?dzx;yenhZAB~&9 z%8GIZmpepdO~Od@_Fjc?UXCs3&9D;=zy#1=)wB>e`G86di^YCt=#i3#I2u-h-BD3f zWfk7sL9Ryz?sv+7Cc?9ze+z{~Q|v>i@!MwA%br4^jv1i#7oFdA;n7;Jf4gpnW~PhE aipjsk`rgr}@&JPX0000Px#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91FrWhf1ONa40RR91FaQ7m0NXcg3;+NHT1iAfRA>e5nN5gRRT#&8E2vOg zq*>|Eq!wwlsPSWw5@u4N;IwKLxVNuGB?PtTLo0$>MbN4?)2_&hNSrXUMPMiq2Bjcr zhA~YJ?Gr}zVFPq@PYq4=bY#Ne4KmFJ@>qKX3e@N7q381SD@9@ zyWMWj!Dl|Wq^WO$e-8UttJRuF@S2+``fe}+egji2%JCl9)V$IP4+1R*r@>6fmn2l3JPrz^!YX{xG%dL#Ks;Uk8#6 z#-_arYysEX*i>od055_6fQgn>^iJgUKs=gxy_FBnK!X}qfFoHgQhTFiKo|rs1Cu{! zJPfXiI!>XH+$({5CpMPSLD&HHgOL1#;bX9<fzvRC2s^Tw+2Eh(sVoPXl1wA-~^Dy`b{AGc%w#J5oju8WX$DCNR zBUGl^cBDKYzG7tb&Jfqgf1RC~t11^(i==k4!cW z(tHklH916!={C9pa>hqT2NSK)^@X5L2iv_s9FycW3Q^DHQFB0-Po$OuW8(?P7eBov zO;n&>S__@H+Ih5ny2aOik>U3ukQ}m@l^Q~~6VPsSSQ$LgZsYDaW)?TP`?v+dy#t&D zFSz5SHiWJwpg{O)67HR#TPeF4tzEnoiO&$w@A9xf=z0Pcl^C1BPT=lN+HEwJT=N#E zD@}r)Ct!(*r7d=ThRUtdxG7DHE5Soxag+<8Zxe_nYG|accY$OK6qZp>?}3JO^g_Qq zbJmZU>)b%K&-|{F+FXfiQSqO5zEik=iWl0+u(@C8BVOSFjI+;M@fiRn-}^pQ%VbwZQnE*Uhi>~+ zupHkB^rH=9xO{!)j|&Ml*>utRu(~4dQO`;o`|9^YJ&(>-eG7h?w8^SkzOdadVMpk% ztmPOX5#x|mfxb0D?U>-)1dcmCpQL?Z^2r8`#IGbKEyx@O`ej&RJU9`E$~bqW=e2 WIXrbM2r)?j0000Px#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91FrWhf1ONa40RR91FaQ7m0NXcg3;+NH6-h)vRA>e5ncIt3RTRhPH<^Zs zdF!I4<|QkI!n{Nmda5M5LG)nt2ZR;XgZbu*A_Z9xlo36O5g9>L4@yuOhIFx@=%q|D zEuxvpEGtXr=jZ$LteJK8*=J_|X3qKL!z}pBzO1#^TKk-R_SwHPU0q``b_GUt1sd&+ zZZ?}2FlHKbgT|MEYCxyDmnIF-4=3R$^fnp|^;It|>W^`N^I7nFC(DwAWBSH~^u=_THLcm}?OGC7Ij19+^E1)ZZ2d#O@u8svx2}DBEU^Ofr${ziPAXzt*T}O2!b2I#= ztPnkKo5~VE_XOxJ%r3JT=*Q?h7=E1%j2Kyk7+-=HO(GhX{_${4QX~vt0lj%^eGc%i zcT6a^CM8*Bgtv3SZtEjuc2)EfN2gmaI_XXsm2Vu}=REGCuESO>g6|oy6Ml#Na8p?z zbQi*ta9zQMPObU1@ET~J^`aNaM!2Fdu!9}Me=|$!Ps&NT&~$9eltUGNp$w1JIX`ta zi2b=sZoZE-j>d~{F|3C3Aru{9w}XE6oxQ>gwUl0PGyR$R(z79B@+iFnx%@EHo6%m| zTXM<@8u*?7%b@%`ABlr~+5|KfQ_VrzLp~Ge(jt*b@b0cD2M21O@eYFXi_G+3$0@_ zXe@)po%Kok80y%zecgHpSU$tf!L+IU~mA6|(2z255B zRWkt`T?uwtrP|yLdYfxWEeEfo*^rlV zE8(aaP(B+=+V^2I9`?C0-&3C)*d$NjGUsT2;N$_P`;~fb$1z*_m%z)g4x(GD*;u*)kq;RYkiEedl3r_M;Vx%S6!%{h3q95xXy8?Awfz$W*-0gE9ltTaj002ovPDHLk FV1lP6o%R3# literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-xhdpi/menu_feature_links.png b/TMessagesProj/src/main/res/drawable-xhdpi/menu_feature_links.png new file mode 100644 index 0000000000000000000000000000000000000000..2bf1895d1561660c057fbe87f2a4c28971a4f342 GIT binary patch literal 1201 zcmV;i1Wx;jP)Px#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91FrWhf1ONa40RR91FaQ7m0NXcg3;+NG8%ab#RA>e5m_JVyK@`P(LPMiL zpdcCwOcY8{6r-V_GiV~gFCcsa6Hvc^7)ygTs0l#}DYP-hLJKQMEQ}Nw3md_|kQl|p z!YKaE%ifpSW#8MG_fHbvO-^QZ=iYPA%+Aiv?2a7`YmC7EJp#FUbboVmvxz*%!7%6n zonRMO10TT)@FEqZ{EVer96b&Ag-_qzXCM@W7N{7wja z$MrNA1;;=y7zF3ReefN)w$8;+T{}U-9JuMKpuPgefZl_mMtdum0zZM-FDmuv1c}nb zKjr?OqN-CG@b3e9T4s6@wVh5RNE1KD{pJ*nnYW1R1a2o5GlME|AerX2YZAY+LQXvi zcRJN(bs|ZRtDJhp;;rLw>76iJvinsnVb{yI5oMXBtZ3w2aS^w>IvsS|t5zpbJ2+G| zLWc<6d0=+IY06itGFof_*1^jpd6ll8I6VDMJzov(07IbPn z1YZQMl_TD!jq%)~;&qV1uL-PJAmxdzDvR8~S^If+OI%m+l12V;VD^Pd$sKK>D>&*T z1qEjgv8@hM*o5^>TA)(6XL-r-jZ@scv>EnbcUxLep8zQM#* zI_HxkFY)^FhZ#W+6&cy6Pc**$77&%*OQrD>H<#e>9z5g7myj$tL<6zLA{Y2~0c{Ut zGr((|H#+=v`f{y2@gjs}*V((!Ajw6X&J%Vhx}Izgx`yaEFdGaCuE3du zI@txExYD-M`d4YT-zv2(IS_p<=ig{r2lS)W%zyp02#=DH)|@ZvJdsJS#|X@Hpl;0! z@^pYpOQo$gKdQ+clK}A&Fq?`smU#i0#NRX0s0^O-Ug;qpaT4k0QlD2~5hvo)#8xR&)8#8xrBhK>I`4YKeO(f|C`mt)JP8^OoWnAE|EwQ_+ z6j$xVn=X)U~7&nojN>$sN_Kx(*{X`JMz10z%G$S6Paq_Ro|P zpd~m_KtSdcO5;M*0q&m${XnZz2lxg)f!E+MSYXO%>NKn|0*w)<;RyT%U_+d3oPcbO P00000NkvXXu0mjfATbxd literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-xhdpi/menu_feature_links2.png b/TMessagesProj/src/main/res/drawable-xhdpi/menu_feature_links2.png new file mode 100644 index 0000000000000000000000000000000000000000..6f7f647994602c7720279bad0dabf6e603029ddc GIT binary patch literal 1350 zcmV-M1-bf(P)Px#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91FrWhf1ONa40RR91FaQ7m0NXcg3;+NGut`KgRA>e5nagWkRTRc^n}SUh zB({OXVFrDOib<(WsZ_|!c2w#_ut<_WKoD9xFEphilaLlVYtuogq7yA5I&jc7P(kd# z$3Rfyi`aogL7}bJ-*?Tf!9*P+thv znZD5nHW8A)4esWI{X}~a90n6$C)f|Bz-6!oyxpQP+0#joumsL{71US2G|+R<&=?;C zv)~_KcAH9XCPAW*_)qD7s-fz@0RH1ZlQz?x=+fhu0zG_xXNk^JwsqybxQyGs zowoV$j@3!D6^wO^&=J9_SFG7vZm3f2$kk#Ca0`5yC+|@EW5=`G)hpHH32-U2<_Ggc zU2K2vxJO*QQjMMfqV=j=AnF^SU)8O7vY0k;&jD@bV(ky-IOOW_RT75Y^T}%-o;e`< z7eq}0KW#hAuE(p+Jc$g6Kc7kj&%CL~%4cIZ@O{ed1+Mi&1Z!96trOzs>NACVNZb!roPd2{&pCb7-*(jn7SoySNukDer5tFe*pu{ zbXUf4><9h~I~^66IT72gc;&wc%(RU4F(A%xe#s&!^@A0+?x4rF)8ZxNSe)|AivS(# z%obyV39uh2i%nhg`71A;==@ck1btu%C=XW$3M%+%d*Cfeyr>Dj;0dUwi_eIzJEDmK z-PIxdhk>>SvgjWvBOqqH5_$hmYvQvBe0`s%tgmQMMvNv=xIJL}1@L^HkUpI!>`*kH zA5UsS^f@rwpA_808PfN_yCFCFb+X$L8cXVfqWQJPyWuTvVN)NuW>MW?D|#D-}N2 zjQ0bH(!;pDC>i4`Al7ylaRtX<)+GF$P%^F0%U~M344wgcE$RTSe`0u1Zw868PJCYj zSAaM}c8AUz;3TNIqeZ+mV+3kB`DcP=w4JLVYtVpAhnOQk54za}Dus8u0bRHoGQFMr zE7lT)%2hg#f}~$v{I%v~y7EoMxAaR0vDWoLpf#zt(-2q#zkzSTM_`%nrFSWl4DRCu z%z`h7Phay%*}CA&1a<-0i7wFhh7$2Fg1AF`ZzQc}X?O+x2lW4_ABjE+oB#j-07*qo IM6N<$g5#-Zs{jB1 literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-xhdpi/menu_feature_reactions.png b/TMessagesProj/src/main/res/drawable-xhdpi/menu_feature_reactions.png new file mode 100644 index 0000000000000000000000000000000000000000..4203ca32d6b14eda06b86bf70ac4577f61a46099 GIT binary patch literal 1352 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 RhD`tf002ovPDHLkV1fsa&@2D| literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-xhdpi/menu_feature_wallpaper.png b/TMessagesProj/src/main/res/drawable-xhdpi/menu_feature_wallpaper.png new file mode 100644 index 0000000000000000000000000000000000000000..2c8ad96ea0e0d0f20d96eab2408d07e8cacccde0 GIT binary patch literal 1263 zcmVPx#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91FrWhf1ONa40RR91FaQ7m0NXcg3;+NGSxH1eRA>e5nQMqtWf;e2w7itC zxp@gJw2RQDNG9rjSZl*1u@%`qi1cL<3;JgG!B2|e7a_Y7o{ z8vfwm;0$6t4;}@T#2vuDA9Q82+5h0xB?@lv@qVG9`-@xc0P39#0Yj6x`8 z8O1#a-UqjVQMdO(dDuGOpMr;df2A_<>cIhEw6Eltcx*H;$G^7#=w>;6A9A9_yF3$# zeKyQB0DsK-3UZ7lgPWm~R^UCb1UzI;$Zf!Ai|Nbd1j>R5QU2)LEfM_He47#Z3fl94 z_MzxQ_T5_W4Y)d_3O;pI8fnj$O%$bPsSorc{|%nV3+Chdyn$KN0G@OK{&KPx{lj0$ z1FM*ufE|?@%z7yu-FC9i8>piO;Ee{IbG5|$S*?$gC~5$4{FAWVN)TJco5kF4{ZP2Hocz^-lTkael`61*rSHUN^fkm)O+I~_fYd(mxx=ZJPQjnBcTPum} zWHKW|9?6nefwzckGyfy53*?@x)-FK;y7=S&D zDn2)>6rojk9r#)Pm-Zt1G|NQ^n=2UTx#}%uSndGEj$?I+LTwFaQ4-yhbuV(>WjXx)jzoZnZFMU+1B}*3#mwItIUBq*+IK)Rm;`j=)( z<<%C5K=6@Z4d@2?7bOYO^K}%wSs-mrU2qHKEMf>G)Hm{YP)S^EzjKLiml)%ynGUNN ZxC7YvplC5zW#|9^002ovPDHLkV1htyM2Y|a literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-xhdpi/menu_gift.png b/TMessagesProj/src/main/res/drawable-xhdpi/menu_gift.png new file mode 100644 index 0000000000000000000000000000000000000000..ddec205892b6bb6137b7b2dd9e1611a130836b2b GIT binary patch literal 1099 zcmeAS@N?(olHy`uVBq!ia0vp^1|ZDA1|-9oezpTC#^NA%Cx&(BWL^R}E~ycoX}-P; zT0k}j17mw80}DtA5K93u0|WB{Mh0de%?J`(zyy~ySip>6gB0F2&*1?o-|Fe&7?Q#I zcBXfZPoTu{<+rXVdo8)*~{na!fJZ%tCzc;UvzbmeP{?g|K9I-;E! zc<9ifQ_4CaAua*jO1WECUVCr9H~rJunLEpuEiS&T_TBrR?fswU{#VZ_erH?$E>~;n z37NRY&d$!X3#|JZ0zWW+XB_rB8N|2-Q?Gx<7Fk6a#<_y{B(ieUIqdmuGFxXD&BEB=H}ugQXOh{Z_5pi zdy8}$@APtUJ?~V$WT$uFs_la%mR+y=&wl1*{Uw_AXZ^*`D}|3`lqR2B@mNxK>L$yc zN~w@(8N3D4FBtr~R+G5#b=(iOIW?}H@xSflf153c5IP$7w)o#_{{vq9+h15;oLMJy zh4JIGhGqx#F8xii439dr<$r3mO`fe0`O`~lo`~$yJ~iH-K5sZzF8e%j~~5OZ+qOg z@IPZ;|14$Gh$re7&+gm2Xwjp}UC;j~GEG0@RI(ua_nZxOch2O?7rvNpxbfE7`RBJL z@8jDpRWd6=T=)3eEt_)-x!zW$S!Mm2sCCEbc-FD{u9&L@ijU`b-IOw(-X&cgz3$%z z<8w`6(-%y3^$UFczCm`jaFtQx%zJ9G^^Gy><~M!%tG4~-@^$scQq7NlaDP6_luyNz1`&nu09>4#0;5l!u`iq4>;v02$B`SS;d0pu2IoYH3Up(iWKDYJ3 zeXcnrPmI)bPZ~;XGb+3}S+*m4>ETZ*)&hihi^~qw3H3jkx;pIl!ZX(n;4e3asafE;$4f)sRtjb)zhF#WaA%Rs-32R}564Qyo&33* y<*NF0wh3QbE$nxzyjF<&>soK6sd-eV_!qm_*L#ycyYxzcGP0+upUXO@geCybH}Ho5 literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-xhdpi/menu_views_reposts.png b/TMessagesProj/src/main/res/drawable-xhdpi/menu_views_reposts.png new file mode 100644 index 0000000000000000000000000000000000000000..68b8b4279b3c88517137dd07b84cc5b1b5f0e82b GIT binary patch literal 911 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 ZAN*66TxE)6%G(7>pq{RNF6*2UngD2TM1BAO literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-xhdpi/msg_call_bluetooth.png b/TMessagesProj/src/main/res/drawable-xhdpi/msg_call_bluetooth.png new file mode 100644 index 0000000000000000000000000000000000000000..9453e9f6eef7b96ae09948ca3cc2aeb931c1b8b4 GIT binary patch literal 1399 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 literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-xxhdpi/menu_feature_color_profile.png b/TMessagesProj/src/main/res/drawable-xxhdpi/menu_feature_color_profile.png new file mode 100644 index 0000000000000000000000000000000000000000..e7bd4f1d09d918150c672ae7985ab514842b5916 GIT binary patch literal 2573 zcmV+o3i9=dP)Px#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91NT34%1ONa40RR91NB{r;0FW_8?*IS^cS%G+RCodHoC&BFRTRg4?v=J@ zmYJrRSs^J|TB&VF`=+g0m;@GURprHlo zYYSBE&ct`@*wKXg4}yao_UCvrY=B?jo2sg+Rcs5FLK}z%X~7wu0{v<^mF+%|M%H5V zCCq`DG?@15ztAWxVJLh6e?wmKBY`GE?|RuvBz`130_&jCGLH&Iz`lEYW07PRm;wJn z(XyTj$3UyR8Y;Iwh@_g4)Ln2Nw9Js8I$xpAIA9a|x*F7h9|=8RKS(QzGY%%vS8Yq{ zR53dmr#CEsH1aJ*&%hAqkR~bYs-QF60J=IiLmK%UqZ8BAQ4t#|SK<%DHt@+N3Mav_ zc_c!d7BCRrf~2~)V)Fnr&ZF+sF{rQ^JP$t6)p9fB)+e|L61yWzgulQiuTj|A^{>b= z6}5u5J=rMPv7mmb5TUNaKn=A8;$#s8M^rQ+Drf_`x8vkp3XZO55}x4HeFDshOAh@) zIKUI4^dbT`hflx~^#!+;9z8!#@=)J+JE;G$ z&?}2?T3h1who8Zco6^QfWk(IetXj}*1kJ5d3-;&31_(BLKmB!EZq&Y`4SNhG}*cfttc%u;l8j5ecH( zDo$o6#l!Jrcn^MsuiHV8}&JH za(b!xl;lSiw;7q}sg-ylB~_ThiMrJxTe1jc^AsU6n>6)!3^K}|V0~q^p)X3M2SX^;)a(;f6H`~-ju?g%81D~1xlXmmLqev7 z?NWDZ_{zv>Il9PD1>J$*p2PY`!Wi`gSKf#iItm>6z{-%LM%)iMW`!J5uO`@Ug>ngY z^>mGWvQlIy82~(gYOIPMb3=IWN>c7r{K@dA6FkEw0QMXWmNxB`E#tjvFGZ9Im3O;;2f#>J2tK)|l=w+LRA@bp zmB*uTzLaS(Pj`VOn^2w`=UbMZz-NN)SxfX{Pc!9`1^-siL0bi3+Y9~xSF{pnJ|xh$ zfjO>2?eX_)W9MciC+<2@$@QSi)RjwITi-Fx%|NkU0sXRJ(XllU)RU-%6&HPXSOJ#I zLV2L&R}meRSrf$6_ci{P!&1m1i!s)Ro)JNR3FwJt$rO~Ob;gmJQYffQk6iuguH2pE zj)Q#1yN)T3pTaJ3YoQwr+l?5y%10*UlxsN^c7s`9N#eK&rzb7P47%~b3E`v8`1u&$ zjpzpiwiVw}nZ`%@1+}hgi8?^DZwQj3e3<2kPUDoP;abpw`(V&*@jF-|yF=jhN zb$;lyEGS!y6$V->$Ck#Myi}ep&lerpK@*f^o8d0=?2r z6ztpub$#8DJV7Kfc89|Th!frR75Q4tdty}DBPJtV;u`*^`n9lYYs5)>;)RpL|9&He zRBE_$!nTF&Q}Nl#z!A-DdSpog`;(m*I_#M-#%V?j4GOztiIb}>mG1}YfL&RF!n-&X zP6XS5ZUBx`-CjW)>^m7TGL77WSiWth88>fNHxX=_5%6eq{ulmro%+^2Z1gY3>l|Iu zq4i|?G07=wd8P)c4L+Lf5~HH{<;jpSzxu@-x*g~jG`_ym$s(N$<5h;Pcoiqv9+pEO zy7$dLcthU=jsx9RdW>5#H0xOq|UHiRs zHSV%fec;aaU)*I;5B4~LmY^wvPZm*Vj+%X^h|feL=#@U7J+6!sIL0`Mu2HOY-_yX*&33_xc-+nXnwzfW}1n{~bqx7Sy$l zV=bL5Lf?lWg8t7({S_KXuXwe-lc&rhnD^~zy=yT}s6Whwa%DOJ_NjLbkS1YHOzSy= zA?dvuCK1t9bAFn7{uef!Y!}em&e@<(YqmjH^f+Gzli(aEe573QPCCmkNdss;cL?a? jfnQ+_9k3>{yG{NB3YP0a7pVF100000NkvXXu0mjfRWz@C literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-xxhdpi/menu_feature_cover.png b/TMessagesProj/src/main/res/drawable-xxhdpi/menu_feature_cover.png new file mode 100644 index 0000000000000000000000000000000000000000..d330b2a1cfcbfcd3eb805848fc1c0295f5205ddd GIT binary patch literal 2126 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 literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-xxhdpi/menu_feature_reactions.png b/TMessagesProj/src/main/res/drawable-xxhdpi/menu_feature_reactions.png new file mode 100644 index 0000000000000000000000000000000000000000..3950e404e3d7f347fdebd94f11374cd092d1b20c GIT binary patch literal 1945 zcmV;K2WI$*P)Px#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91NT34%1ONa40RR91NB{r;0FW_8?*IS?14%?dRCodHoO_5BRUF4%^O2UX zOnOkQm06aFr5Ry~u00hkqYz8NiYx-fpd$Py=s|zLo?_F z87mZ;S(=(@wYq)2y0e^n?wQ#+bI-kd?{$9gY3BUS`902W=bkxt=32LIx#ShdE09+p zuRvabyaIJvfre1?YHMrjj;{eQ2nNDV__?8>;ULGd32_xmG5bOv*hY^fFijzH$JA_u*3Cpx8`+ zH$gspbUok+aD8-9XOfc&f~07qTBJ#3GQO5WzZjo8(655^um|?SU>FV~VIuT`IQfi@ zSuC#GHGp`U$lEYM?HJjQejTiV7T67~Fci*#X>bNavKIg2qeieJ$F}U%u4MC%PVYa+KW;< zC}ALRo`NS!aN#t?XD=8JkHT)SMIZ5JVKO9cAG$HXb$7YI89A_$*xf}axA+(*R+n%bck3f9`6Xm4Q3(!x=-(_#{%Aj=h`Cl%Ke z%@k!L*z!=(z#3C9p|oqzm5hyFgXp}(t`6-RYpfoL=`tU$yO_No9&&g;;Lz2qif{B^ zko+cN0=qb0X0!`V?E>jBg)L>>|JcDQ)NVH7tfpwL3*+#RGzBX8IkJ&VwVXB^iJ0ex z`pBvM2vV1us%aLk@(;a>(hRO>#-kqW^@Np>URcEJebCzU*0nQ|d7chKy)(w>MKDD> z%ii9O>cqhOC`D7!DsPIOB~LdMNA)>Z%c)@Rru9HRm#6|1XeF3(h{jn_e?1(Tk6sYy zIh6l42BYa<%PaLz7|nr+>2-M|KXGz`*W57BOFR43i(c}L_HtB(gPFr{&=X5j?)6R; z>$#?f_VZ7@9aWhD^Z39qui4&d;^Y|k#^G$7x3elSU@kW}=CcLev6)iD@H&UXU@z}7 zC6&>Ld7KEY-xkcw=<up?GRlb&@1R*FZb*irMAiKJ+P&@RKHsBwP5RY z@m=6t?a?m=$&!)-mcapeJ#aX?!vE}}d3G7X6}lvdUQz2cu0+SgA2JkLm%l|mlfd5i zqH3F0{^&#?&`3oyL_c?_3L&QvpcjL-tUytjKO}+XThZrNLv_gMWN5K+Q*$#6s925! z(_h0B|99Pivn84Spx+VMqM!1N3LGxp9l`XXZ+SlL$&ssd5J#%a9l%DO1x zU+=zLUn5YrJzV1S-J=7b9*zSPd@e`rX=t1; z?99F@`bk*(8_D9hgy_@bT3ht(dSZgjIQsvvjcDsitcu4fak&@hNwyDY#~r%~q;D$N z@1sy1{wEJbF9OhC1$#BXl2+Qc7bH+G4JM`{#r@-zx9FP%GeLWqXuaaJL-b3No9d+= z&UrAw39!3k%IQz^kiGb)xy~pxu4JMYgS}`cmFiDAFR$@w`=!r>&xDOhRcY`0j$`UA zfnOF?KAMU5guB7DJN_2EHlf*G3kRCJbI%gUx0f3VF$9Cgb|d{SEw+0($Oq>Y$SaUn fAg{pDSAqWl#jcBnI(B)j00000NkvXXu0mjfr$CgL literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-xxhdpi/menu_feature_status.png b/TMessagesProj/src/main/res/drawable-xxhdpi/menu_feature_status.png new file mode 100644 index 0000000000000000000000000000000000000000..07cdfec3d4efb756f06e8d493415cf16fa1dd3d1 GIT binary patch literal 2438 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 literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-xxhdpi/menu_views_reposts.png b/TMessagesProj/src/main/res/drawable-xxhdpi/menu_views_reposts.png new file mode 100644 index 0000000000000000000000000000000000000000..c61144e1833c295d9ddbc34d1cf20b3c5e9f5eb2 GIT binary patch literal 1142 zcmV-+1d02JP)Px#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91NT34%1ONa40RR91NB{r;0FW_8?*IS;;7LS5RCodHnonyKF%-bNwiE=t zSw!25h57|Nh$r_0D1zX{(;l=3kJ6(T!Ry{^FWs`9JSYg_7w{j5U!j5*v7(+t(Sz0X z_ap<`G_%>vW+v(OJ$TIIWsdM2~<8o?-^hZ7J-Kuh^cUZ_yp}2il}WU!;SamuZE%WC!Ca znrt1S?X)k9IIxKRL~VJ2;Y1MW$Pd$gS~$`loaotPNWVnqBlkC`I@-?y&N7bdqQfH7 zN+ZWa&)P5+YP+NvQS zf~v$P*N$SdaLh=Zl3}qV`SdcrrQfs%%sC5=BrgRGca0C@OVF`T=<{CJsj!XO()Je2 z@Bun89C7PWT%^*e07O4ri@Nu~G5BLO3iULV$kvQ4vZ*&x4ilh}Aj&coP#GCxS3nBL zL%@@*5u|`T1U%UqK?=x2z>}>J6akRykj-WDG<7huy%EGY(mT2)2ZV)e7+aCns}L|B zd)|+`TxpejbYJxOJND7@VT~H^6!Q@-%g-?4T)*OcFY~{!M+XELIj53y`)yI~?-z|F zzPh$Hqw#+?M2P7Y=P%PW>hwJwQrn)7YpJ&ZL0wZ5EZIs$qFm+L_oHv^olo zOt++hfAV96Eo+T9MXRIWG=ENI^&0;c&`%Z}Ip;fLJsLkrBe}zw$4B|W!j@iq8di<8 zm^&Pa92XqA!SBx?bfMJbC+mh{|uZwmiph5B+0j z)SNy>{qml39=(h%pf2h1MNseJ%ouE%kvM3VE)EC3_XI@{4JNt4ATI*{03fO7AEoS}FaQ7m07*qo IM6N<$f|yYAZU6uP literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-xxhdpi/menu_views_reposts3.png b/TMessagesProj/src/main/res/drawable-xxhdpi/menu_views_reposts3.png new file mode 100644 index 0000000000000000000000000000000000000000..d939853aa69b0b69b814ecdb6e8d9b176348816e GIT binary patch literal 1255 zcmVPx#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91NT34%1ONa40RR91NB{r;0FW_8?*IS;?zrd)CM1iX|NxJIO5F%Q&%8&>O%}9zCrG07H z@3kvVbI-l^+;irPr+epx$2;@R%=^6acpT4}Ik%|DnNz?i;1qBQI0c*nP64NYQ@|-u zAPTIlt(AgWu)RR0?0=ewQV$k^*mF?je{=!#M36~QMzJ}{N1~egwx|MIjC&BSX+)~F z$QfTMglGWEK)1IzYT}!@#mo?*1uOyG7I4&BOeMZl2+<6_=n7GP2kJ~^o-h?CQydcB zO?z>sJ^|Yr8FvdveAV7lIhjZ#W>sg9vT=bBLxeaDIzS6J0QLaewg_P(BE7LqoR$x9 zI>9W6i`__@tj`7`mpy9igl`lK_~fHhTT@D*Fv_K?q z9e9cJ75^RZ1jwSf0!&*O0WX3>THs&Q2Uvd|YJR&_uAad(i_JCs`=ZQ7;sbdW(`^+; zUzFUwVv|G(nAfwTKI|(%h)!@+f*B9)6@0xRLYo^(B10Ytj+kTcjs#}~c;Ue<;`Dfr z+5#`hOlV1dpe}1dqz1>R#w%43rz=w8T!WL4{=j%8*BC+i!#f33`$?rUNUX@lkWjgr zVY1vE8KDvwu6~94J7u|JkC&0uaV5!(Vw0^bA#4J}AnlVoZSCM?679BfTjzmP_GENx zu$n{$G7>8T>JlL3K?nFkbrk0=r9sN#%h~c(Ag_6O`LYTYXi(OudDOuPl;xM{ z1abmpQ6NjRyMmlRSro|99P|qM#J zh*a0@Cbt%$PFX_>ggC9Fau)=;EB`x>NPa#hY8wK53lrVMq%M$vi8x{Ap^qKI(Hd49 z32I_uae7FVSMJUt%z0qs&e_DD*kO`-3L^h61&3}+I2OP(xDD6+o0?Ad9!WPwmVp;1qBQI0c*nP64NYQ@|Px#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91FrWhf1ONa40RR91FaQ7m0NXcg3;+NFu}MThRA>e5n9pwwQ547fT9u{| zX+(lHkt(8CNF|L*v=|Iwfqyxp+fpj`u z33|X?@C7^qhd@;^)|*umMb?4mz$R}nHfIGClR|N&V8~XvVf)5>Q!&+@sfEJU7=`*x zf3nz~35+-?jTq+_)~FV*pjU%#WiLbTWfNV>{(n^plA=uMa4W*N1B{!C&Rk+5H8WC> zTTs9;?fRussguk%f-%?TjPh4H$0jP2eezV2&j7oC+%8SY`Jc1c1U{Nh&U}z@a&NOF zTnkQv@4zNG8<4i*SPx!W0;Bhl0o!xq1NIeuv7y%>ECZ*2mpoyi6SRQoK@;Fx0WJV7Uoor?%H!(? z(Nu6=XK?gB#+AMVc@~M6=A3Z&%t3Cut$D)pFGF(1WOe3puGx}5I=BaX`z(3hVVe`2 zMR^h|w?{3{Fw9l1v8eENfL|)%kx1l_ECu?4l{?3*NQ41{;MYxi5HK{`4Wa>H$|y=* zwpaWbIi{IOnU99fLQu0J5p`>5q6h)x*{O05ho*txx+3WdO%fr1oVIJ=SSTt;mZ;)I zozSf^NCD*Nw&ay_)65|886N_=6wyFtu>cct`8@|VhfWFMW3_?pekAxdg2UhzkmrM3 z>~b8)`y|Du0>t+uhHN1)=?NtyyuUATa1kOAK zNi5%V20^V#Ag_*>z-?vEQQ>I9coXR2hp@E-Ek1|XHIU#vFlC;sVORz9u|oXh>)YR7 zjj>k@hGhMPux|izOx7YN-FV)KA$qOYXQMZOPs%RpQBZDdUT>hoSZl*~T!NGwNB8s+aUk| literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-xxhdpi/msg_call_bluetooth.png b/TMessagesProj/src/main/res/drawable-xxhdpi/msg_call_bluetooth.png new file mode 100644 index 0000000000000000000000000000000000000000..dba97c207ec7793635d95ce39b57e21c37a4ec16 GIT binary patch literal 2124 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 LNkvXXu0mjfIDU@Q literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-xxhdpi/msg_call_minimize.png b/TMessagesProj/src/main/res/drawable-xxhdpi/msg_call_minimize.png new file mode 100644 index 0000000000000000000000000000000000000000..6ca69d6746d38466ab58f959b45669bdacc0485f GIT binary patch literal 979 zcmeAS@N?(olHy`uVBq!ia0vp^9w5xY1|&n@ZgvM!jKx9jP7LeL$-D$|Tv8)E(|mmy zw18|52FCVG1{RPKAeI7R1_tH@j10^`nh_+nfC(<^v49!D1}U6i==L6{+}6{@F{Fa= z?d-i-qK+aB#l0HK1de!Z-{|3%>~6O#E@W^gI2D&X>{_ANPQmdZERo#pz6 z#Nvm??e0GGmY&bT_x0PwC6B*$^)3jzD4(HNwC7c7%b_bPcNZU7SF+t^w?yB24;Imz z-xytDrDb+*{v^_5z2d;c%gom^G^AP1NnU4T*GQk$G~<%rPPGD$HPXfZCTu@h@$FHF ztFzOo2A+4fCOaLPc+HJf!(>h%V;5t(Nkh^>N!A533O2EeNc&xD_|-63cfq~~DV!_( zEtoVS8INRd(R%NpvHtm#&#Jcf%eC)oJ8n>Os6XxEb^PLkCuWn)e!l*i_q2MU5AUyA z5}Rs5EQDrUzWq^vWl8tNcg=0voXZ5lL zI+@!~#6M@c!+6Qw`{t7S&-`=Fm7mz}d-t{8*I>D3yB#bS7M?lJSK7YbOMFL1om=vW zb=fT^`7Q}Dq=fyb?78RQeCO?@Tz<*wxR0wBu}o`hG>CR@yFXoSr#GctAxnzAnzO1j5lkp;toesIT z<_F7t*siow;7HCCZS8^&>+#E z?>tkrkU}#-CwnRJKX*2gRw1~QlPxPoW>n6{50otbx=xeLm ze&N09?0ieR@XIy7PFmzMUE%T*)HTyz+<)PBM#ugMT$lg;7iuUm{8_SVf?vU}{sqPx#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91NT34%1ONa40RR91NB{r;0FW_8?*IS=rAb6VRCodHn@>m_R~*MlZS$Uzc<^dz+A5D8)u zB8Wv4ThZ#O=95slCcJP*fb=!e1f-y{&n!QUVvAGpHrq4IIHvh^DmWG- z^rT4$%fma$jp7iaqV&e}^z`SOo14S*3^UG7b2J8NvSS`=SfS_K?da%ujQ?3a_RfMc zJ3IU3@$vDt7z+$D4%P`sDnlv5ZCDABe43b(ZXjR1?a-w-rlp*fEWjZ3JO3$;ZPQPT5RAzSF3jr&~pbs&yjw?&cnR2 zO9dPn%;6IhP-@`R11ME+f&xk%oFITw2`303U+WzERUXT?6=66zuSvRyLR>65?XBb& zpP58B1m*?kBjhR4c+ikHo~eN22T=Rc($f93!chX+-Q9hG@aB%P%o7Rdv zmZWfE=Nv+jo$DL~7*z>~983)yf>HsJp@^`_ISJ+XcH^-HU;mV$@?1mFd%9R3Ac0VI zadB~|tE=mGSSohqt~V6O!?a`gs{O>nisRGK`J(7pvT>1Y`8;;>Pm0CjO+_`eN~Q7- z=A?292%U_Pk&#aZ1_tis^Z9F-h_!n=V3~mP7p{PQ zgE>d=e4{BRX`xK>%=cH5(2G!n7jgb1IOnp4FOmqx6And$Q7Pp#$_f3%Q@0+ux5!~# zx#=zMr)A_ka#)v|+=6mgS8k}~y$lxV^8QA|=_o=VftLI_mFblAvllZaCnrDa>+Aag z3#=|wMp(MeaavF4#&Ml9JUo1v^}vVG zGn>kZ3LDGI%eQnvsLHReuYZqvBKviP_To#o$+!rF4C)cd9^?h6Haa@`2Zq91T@Y^N z?GdCYrBdmepp!JtY0evlaivfwe1Uqr8p4fV527;cMFhl?H+rw3z@E~EGvZsF zoti)&xg525vm>V7JUq<(y;-;BwhWhr^f~1e_e;xPDEF zF;(Rma-?ZED5nd^7aY`E!R^DhJYaXwA+b?}!#T$DD+~MH>Qk)9-=ECRb|C7LbiJ+T z8;<=63g2!Yrv*?@IDCb3D{(Omfa-z6VMt$0I-q*tupD0oi3X$t2W*lJNEc3`0fhl4 zv4FyYlSn{>!-+K?($}P7lK$W-d24IyXVzoahetdd0r3a0H?ZIh<5OD#D(vs?KRvyY zURzuHj&;Ga&e%9Ao9FL2Pa9U~@W-0tb=;o%yOMPda9`dSJb{K+(0`$)7np7p??JURi1y@f-H zU_{S}B_OtV8zFJ1(c3?O<%B8>Lol!-4A}=3(98m=mJ>!06oJr*>tdb^x}7ds8I$ry t1&B7frKbW;QW`n4BO{Oz$Ot5s!2dz8_+gt2JGKA-002ovPDHLkV1k|U?ArhU literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable-xxhdpi/msg_call_speaker.png b/TMessagesProj/src/main/res/drawable-xxhdpi/msg_call_speaker.png new file mode 100644 index 0000000000000000000000000000000000000000..cb7dc121d6a7fdc7f5f9c65e954272e59289eb95 GIT binary patch literal 1883 zcmV-h2c-CkP)Px#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91NT34%1ONa40RR91NB{r;0FW_8?*IS>#Ysd#RCodHoNH_qMHI&=6nRwv zr6#rnk!TFTx7Mg=()f(e7!5{)2|-N6FKVKon1}?c2CDc)d?kJn_}~W;gCA7TsHjnc ziJ}1%#S~hjKnVeP1S+k+bG=iyr)T!=ZTH^p-n%FHGc#w-%$)P@?(FRBT!szHBx4|B zAY&k7AY&k7AZ7-7dU{5J1~4|}uG8BcWYmKB;A?OKIQf;#^WfC$3H~hbBM6gy zRNgqqq0-l!pw9(=g)tMT>mq+$`l1}5#{@eSv)*@od!WwYC`ajNxXrU&g4o5#^Q@Qw3ar82jefuKL4Y^B({Mk2q< zs1K@Sa$f<)Cz+fvIQUu6>p)&^iHm&BQ+Fs}>$2mVJ&L)&H(xn>Njt!0u3chMoH_}7 z04(_%@#;;FXaAtTE2>l2 z7EO@ro3@R>_h68DYiMpa29~FWCGsMAzY8Eriv55IlKyEw8iZZR*F^CnHXEE-3ViZV zu>;Z3pa^o~(NC+#so&A}E*qSEjm!8_8i~yf;8q|DxG3w0N!Xd|8T60W)YPmXElTj? z5Dj`C>-olY^mVwHj`H!Wp^bH0wbEboIf>4i!1o|7d8U|Pg0?si%M~Pj(w9u$8;zi3 zF?n%tTyGx3+kmVW;*w?F!6H=PL~I1faP%kpPlt`*J}T%AjbwjD7H@HgOvr;?Z_r01 zg0OoM`~)_DE5m}NegUw=H$D8<9WB;btp=dmOq-cuPDZVxpMnHo^PuwaIORutV^CKM zwy8E!e9g7#H%U|3KxGHeb(_(>1a^W#(uF|>6Q1-Fi{MwmNgA7X8SEZoz<2Ovj-i)Z zZ9e+fRh>-yjkGzS^tpR;HjT9GL{ZnzxQq(Mk8%Y)s~r!NpRnAeejJb&Wi5A${cxkv zWJC8JY|39S=^|ISZ<2ix2KwFZ#3fhDY9p{_(OCqIF^u;cy2PpS!QO&n&w(E0U@c|e zlk{U{E5&B{$0A7dsTD}-28#(`i4nw1t|IFRwGrEHWj_PkNfk#>AjlT-n*wE*o#!$x zg;2RewzzUWm50={iX}*{Bm#Ux1uezP4V|cV8T$H)B`6>4dL!s5hE7y_4SmHE^d~{< zxuuc0#kw%6k05Td6A&be)}V}_AWyVNVlf%1VH%9&DTJS+l&^ar?zqxPvEI<_t!RP> z_);V2c0(tssw8NxLB`L9PE^+9-C40$5VzM&1U<_jW0lrnI}zP^N-rz%-zt(I0=}By zWLs6o*M@Eb&>|c5diP=rbJafGfo_<(GUPml)1QLV4CiIfS-+!+ne4f1`CJr{A9uQS z63dV?8^2!h__h{udN;zH4Q=E@fb|I~N04;|^*L6O??~>^G{_wuC%2=2z*8sWOi*>Q zwUgU6u?Xr=#!|c!riXOH%<#!l?J$%dqPvQ#=b)}5Cb@$i)q2XSoV}iyWD7FQTdv~? z?tLJW?lJHJXdzhrx`8^kd%A#Rd9Kj+pBc2-+E*1QK~gF=?Oh1?KLwoP%2rDKRA7Df zv7T~$yP#oeKl}8J#4ZuvC}5p?>_UH`-AZXOtjG_C#8m^<083gG4dF$4i2Kv?;_wmu zwfIu-tA@Yz|26cp++qe@8>rsH3j{B3^=|`ND`}Al{dy+mFLg!9$~csm<1hF!8GHpS zson!rurU(I_xG@K22YplDv8oD&XxTL9%IVa(s^Jj2$LmIVkoH%-pL`()gCWnJ;ccv zXM$Hjb_^I2MC3&?C*-Gtuvj6b^Hx=ZrR@?0g!J1){Et)LCo?B)$(Km${?eVjx5(b4_xeMsc%Fu(JOY^=IL7a`oD{eMo$N1T6=<<$QCPbW&M; z|FMAk>k+)1TYeaP3xadccFHZl`qt%|S62!d@+K}%urz#4sjfdx7H!5r#z5H^_!kjW Vk?%x*bH4xp002ovPDHLkV1jzBW+(sv literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/drawable/icon_2_background.xml b/TMessagesProj/src/main/res/drawable/icon_2_background.xml index d1440713e55..279e6595ba3 100644 --- a/TMessagesProj/src/main/res/drawable/icon_2_background.xml +++ b/TMessagesProj/src/main/res/drawable/icon_2_background.xml @@ -1,9 +1,5 @@ - - - - - - + + \ No newline at end of file diff --git a/TMessagesProj/src/main/res/drawable/icon_2_background_round.xml b/TMessagesProj/src/main/res/drawable/icon_2_background_round.xml index 5a8cc155f00..be67c479040 100644 --- a/TMessagesProj/src/main/res/drawable/icon_2_background_round.xml +++ b/TMessagesProj/src/main/res/drawable/icon_2_background_round.xml @@ -1,9 +1,5 @@ - - - - - - + + \ No newline at end of file diff --git a/TMessagesProj/src/main/res/drawable/icon_2_background_sa.png b/TMessagesProj/src/main/res/drawable/icon_2_background_sa.png deleted file mode 100644 index 424fc20ae4e7645e474aece96d91e3f0d596cc27..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 48746 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 OqpyAX - - - - - - + + \ No newline at end of file diff --git a/TMessagesProj/src/main/res/drawable/icon_3_background_round.xml b/TMessagesProj/src/main/res/drawable/icon_3_background_round.xml index 3e2c38cff49..b0d5d41e7ab 100644 --- a/TMessagesProj/src/main/res/drawable/icon_3_background_round.xml +++ b/TMessagesProj/src/main/res/drawable/icon_3_background_round.xml @@ -1,9 +1,5 @@ - - - - - - + + \ No newline at end of file diff --git a/TMessagesProj/src/main/res/drawable/icon_3_background_sa.png b/TMessagesProj/src/main/res/drawable/icon_3_background_sa.png deleted file mode 100644 index 72b85d15d646b8126c9227cd0e5cec4316da9be0..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 21353 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

uHV<|<4G>xPdS1KWK%v4 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 + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/TMessagesProj/src/main/res/drawable/icon_4_background.xml b/TMessagesProj/src/main/res/drawable/icon_4_background.xml index b65238aa664..0e4598e96ed 100644 --- a/TMessagesProj/src/main/res/drawable/icon_4_background.xml +++ b/TMessagesProj/src/main/res/drawable/icon_4_background.xml @@ -1,9 +1,5 @@ - - - - - - + + \ No newline at end of file diff --git a/TMessagesProj/src/main/res/drawable/icon_4_background_round.xml b/TMessagesProj/src/main/res/drawable/icon_4_background_round.xml index b2ccb6d31a3..b75253ac36e 100644 --- a/TMessagesProj/src/main/res/drawable/icon_4_background_round.xml +++ b/TMessagesProj/src/main/res/drawable/icon_4_background_round.xml @@ -1,9 +1,5 @@ - - - - - - + + \ No newline at end of file diff --git a/TMessagesProj/src/main/res/drawable/icon_4_background_sa.png b/TMessagesProj/src/main/res/drawable/icon_4_background_sa.png deleted file mode 100644 index f89e3fc0aae6c633c02bfd19fee1e64412c94eee..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 70958 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 + + + \ No newline at end of file diff --git a/TMessagesProj/src/main/res/drawable/icon_5_background.xml b/TMessagesProj/src/main/res/drawable/icon_5_background.xml index 9df3154c3b2..25ea137141a 100644 --- a/TMessagesProj/src/main/res/drawable/icon_5_background.xml +++ b/TMessagesProj/src/main/res/drawable/icon_5_background.xml @@ -1,9 +1,5 @@ - - - - - - + + \ No newline at end of file diff --git a/TMessagesProj/src/main/res/drawable/icon_5_background_round.xml b/TMessagesProj/src/main/res/drawable/icon_5_background_round.xml index 7c2df592b10..430c0847680 100644 --- a/TMessagesProj/src/main/res/drawable/icon_5_background_round.xml +++ b/TMessagesProj/src/main/res/drawable/icon_5_background_round.xml @@ -1,9 +1,5 @@ - - - - - - + + \ No newline at end of file diff --git a/TMessagesProj/src/main/res/drawable/icon_5_background_sa.png b/TMessagesProj/src/main/res/drawable/icon_5_background_sa.png deleted file mode 100644 index 862664b5eb090a02e10b29935ff387b1cd900475..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 61301 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 hs%hyAFO56@GD@CS + + + \ No newline at end of file diff --git a/TMessagesProj/src/main/res/drawable/icon_6_background.xml b/TMessagesProj/src/main/res/drawable/icon_6_background.xml index 2630ccb36b7..749c55699ff 100644 --- a/TMessagesProj/src/main/res/drawable/icon_6_background.xml +++ b/TMessagesProj/src/main/res/drawable/icon_6_background.xml @@ -1,9 +1,5 @@ - - - - - - + + \ No newline at end of file diff --git a/TMessagesProj/src/main/res/drawable/icon_6_background_round.xml b/TMessagesProj/src/main/res/drawable/icon_6_background_round.xml index deab3c4cbc2..c0765ce063d 100644 --- a/TMessagesProj/src/main/res/drawable/icon_6_background_round.xml +++ b/TMessagesProj/src/main/res/drawable/icon_6_background_round.xml @@ -1,9 +1,5 @@ - - - - - - + + \ No newline at end of file diff --git a/TMessagesProj/src/main/res/drawable/icon_6_background_sa.png b/TMessagesProj/src/main/res/drawable/icon_6_background_sa.png deleted file mode 100644 index 6e3cfe2d8c56d76215518231b67d86f8dc112153..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 7328 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 JmaAKb{1189-l_lq diff --git a/TMessagesProj/src/main/res/drawable/icon_6_background_sa.xml b/TMessagesProj/src/main/res/drawable/icon_6_background_sa.xml new file mode 100644 index 00000000000..d5756c75252 --- /dev/null +++ b/TMessagesProj/src/main/res/drawable/icon_6_background_sa.xml @@ -0,0 +1,8 @@ + + + + \ No newline at end of file diff --git a/TMessagesProj/src/main/res/drawable/icon_background.xml b/TMessagesProj/src/main/res/drawable/icon_background.xml index 27feec4283f..a2d43618a59 100644 --- a/TMessagesProj/src/main/res/drawable/icon_background.xml +++ b/TMessagesProj/src/main/res/drawable/icon_background.xml @@ -1,9 +1,5 @@ - - - - - - + + \ No newline at end of file diff --git a/TMessagesProj/src/main/res/drawable/icon_background_round.xml b/TMessagesProj/src/main/res/drawable/icon_background_round.xml index 444b6f188d7..4de8cca774a 100644 --- a/TMessagesProj/src/main/res/drawable/icon_background_round.xml +++ b/TMessagesProj/src/main/res/drawable/icon_background_round.xml @@ -1,9 +1,5 @@ - - - - - - + + \ No newline at end of file diff --git a/TMessagesProj/src/main/res/drawable/icon_background_sa.png b/TMessagesProj/src/main/res/drawable/icon_background_sa.png deleted file mode 100644 index 686a7d70e93585c2a308b1600289528cd311ed0a..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 7149 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 CWU^8K diff --git a/TMessagesProj/src/main/res/drawable/icon_background_sa.xml b/TMessagesProj/src/main/res/drawable/icon_background_sa.xml new file mode 100644 index 00000000000..d3a1ae4cc99 --- /dev/null +++ b/TMessagesProj/src/main/res/drawable/icon_background_sa.xml @@ -0,0 +1,8 @@ + + + + \ No newline at end of file diff --git a/TMessagesProj/src/main/res/drawable/icon_plane.xml b/TMessagesProj/src/main/res/drawable/icon_plane.xml index acbeff5e614..8afd524eb5d 100644 --- a/TMessagesProj/src/main/res/drawable/icon_plane.xml +++ b/TMessagesProj/src/main/res/drawable/icon_plane.xml @@ -1,10 +1,10 @@ + android:width="90dp" + android:height="90dp" + android:viewportWidth="90" + android:viewportHeight="90"> sP&a|v zN-eZ1r6`C+DG^Fk5;dSeSQH8*CN|DuJI}kHxA^&Or}ulm-N2Rz@_ePmdfSe#L7g% z8Z1gcMMErvQADl~jqx{;31A+P#`tqaNjJbQqKS9fLOt9?r+x>H-y%r^;Cw_{4v&U;KAZ~<2%_*Ak&pZK!zoaC&r({CaZ-OV?CjQzvC6%bwh|fA{-^lYCeWKp=jfF`D^^L zniTzJk&&dgqLe|RjVs3QSsu35=c@0{eS?&((t|o(*%yk~3jv(E)@GNov{2U~Lp&sm zhr0A`(Th$cV46)5o)-P>K)@(R3gZkG`rM6r`q3NL4!!W?hCBpXs~YTAuDcTzm7B?? zvrR;?k|zMUL&NgpY?m86ST;T49RV+!hTtzl*~`kNvj8GnUHHxt`Q3@(Y+6)P!4Kx! z6WLuw3KpLjD8EP3(5IVCyZjyeVj`_RSH{e(_$r1VvuQ!o%cjwgkq2142EWY$k4;{Z zT=VWiQU<10DIpsqH%1RX^^IO{0Eq+}&O0f6GIAxD0z4HLU$`9NWQ0`q-=p|3q)+c| z09^dbcS2L2tD*2V4{HJ7)4QR<`D*qOnoj^H&eo~R)6z>cGSoH61)@>@oI0_Vym#D= zzpltyC&j(ax6h`9KaN_Kd?cHOlf1~X5&+^@0cq0p(q_{|#kPA{Z3D(r_{#vK*V-I^ zIDRznbNY&3qLJP)0fJt}?VvC`#P*GX&GNrVRy&*K06a#%K8a(`d&dNZC%~V$KG7%( zpD>#SjG}7;132TCw{M#JMW=yza4CnH^i@`8^)Xl9&8CssE}$?Z&zhM6WkX?^B%Y=m zW3*0W0yD^-4-i0o{89oWcBUS_2z*ggtqN$X6HYd5FsaJG53NEl-NJ;B-x?1V%GJ~! zA-$EVsMTckO1M@?TqkJE2$b#cuUX3GBQ@D93BY2QG1C}NUPRUzKsXKb)YDFn+vFHn zPUqZcX|w5yzn@JT*JQB0^LS={aWq^@qn;zF^bhe%ZT?M)tT6se| zQ8Jf=hW$G9m49bKnXBg$3e$^ORpbIHQu&p^@fH#3?z}^*BuT?ZD+BLHHm$fABh(UV z4X@5oYpZM;)Woyx*}D5YV%)D|Ac7CMbDNdi?CMNQNS{3ZE>pz&1T^{UxsAfLobTQk zQ5HzjT_I`mA{dH}WUp;DO$IaRz9E0|ug{B|_!HU*Lx<#^b(xBpUy{+9Wz#`Fvw`7Z zlR`Mf5m1y(vko+hA2yq&Ae%1yA!l?%pMt;k?zHf)`W)M;k-#If>1vpn1T>gUmj+_^ z+n#YICzrflru7~X+3S*NHZ?qJBDJy{5H+;;tGQ;=r2;ma)(xzdR5s0uYq~>s14xj= zS$kvIwB-jE4~#B7HOOq5N_UIhdOHiVK1<4*{wlNSRUIhARhUhOpmas$6J^0sC_-4D zwjkd!n~s!DB-)F|ZQLc0rL(7^acV-W&#(+-uVjN!lfKVChU3$(cqpq1CXtQC#Z8B@ zNXE5t;$9tGF6@$E-SEn0(@V?Cyc;m{2g`E*aw-zo4#`QpHnZaDGf}XF z(_rOr#1SaLIlE@LBjwq&JFy{u-9zFpM_eP>bm%!MZYVs&;N)!D^^QuP7}RIwPl8P^ zo9>2dqqfXHvFkeOq%4wpmKk_Q858A`>MP@|0FlfQLA1e8L(WKdH4ykAxhBt{z%>k2 zZ489r&(%ei;IARESP{Aa?$H?|;?U}Tk#u988%Lk6C_4e3*d?2$=G_T+`s9oXOr1^V zhcUqR$Aq38M%gqOOp6Q>PiV_k7uj=^jW$8~4Zt?pbTs zNYu93^l+#a*IWV_GI48l#Pbc@od|w%q)t|>q8#@JfP_Ww@0>vcgAypXJ|F54$qXr5 zL~<$vEf)4#50R+$yyO`vF>5$HG6rreu5vX!4fvxqq>ki64$4 zc=VIYX(Ox9SA8~1aM(PtS(OTk0M#s%LpAkt;+8AA! zqQQC@k(aSdPE!lqDCwd{A!Ryv=1cyhw8fvJv(+9M5g5*Md6OO77JO*gy-`!BixH(YxS4j(?`lDD=95&vG{*=};V zOQo>}hm&on*{=M~*(5=mScg_PSJjPdT64)i#2GLe;-B@#7hH?J$A8t%Mypsao34D% zFPq-Idk1ORZ0uQI@2qwqPZ7^X0lsKmpbmb%%sn{EyB*zZdifSvl<=*b|3aSR-pFbuvlW{KGVFFt$_`}XYCj9>=lDq@)zZw8XpXA?K`jys<6eBt?OrWb*) z7$Y#8O>g+mp7i*mkKw7OPvFiAZ^5VUdq4@FEHM>hl-^?-V{G9;QC`bqndP4XW)2=& z=*>0^{!h)##uy4w^ovnA#>g>Et@|(Bhr@>s;(~p9@WfNk;Ki3tIX#kQpoTRNMTvkB zV_b^!gm!Dv>MC~B4Y`w3`=L%bb^0{!D@K@QSxn@#j+Y=TtQB_IB}W$uJ$>VRe40wD z_2%iv2Eg$VC-td0uXe3)VE=`i2()kS-nB9A?b^!Hb-M=gW~{}ZCXdL8mFjt%Qq}XP z@{Y?!t{!N|CM)^RoK8IVyib>d8OcY8DvKH}QWL+$GkL)yM-Jh-S6_(}&pd}yXHFyk zuqL1lC$q^RCBYDt3f8cQ)g~Ew$Ewh!q2pZ?e_H7EXwRHEi!VO*ASu> zcEh!}?b!9W;DQTq*B!UuvWu%{l+>Kf(o(Y5P84Bwd2}5X|Mqu%Gt{UsmQ9l+Q1m0& z^i-ZW@$A|NProABbR9vhk?P?sL80jhZE`l`LVAo(Q~NT`_g-Tln2{%!n!5ZaZ1+9z zd3^Q@596h^clfh=@5f_LJhf>{sK3^=9e-URHT2G}{wjRyd*6kl&@pGE3`g?kpaC>noD-v(`3G7n``x)5K6n7% z{!Q=3EjQn|PP#6^rALq8!7m=imtH)X2Y|Tn{c7rvkLk0_i*7FJv3z#Q6<_kYr>A`TahsB?1NY0f=uk8VQ13&;X{Y8v$JdCy{5ANqP5prt#ILm z7nOaWx1Wm<)*eqk^DO?;fB$8C-P_-?Is5(aM?Ws*2R!R&s(0*bZhIPH7fI1%ZKk8I zzLRpM*3NE{HwslPO+k<>(Q7E1-jIk64NQwHMP9il-*hxM2alLQwVHjyR~}K}pKhD{ z>`%Q9U-QYX{sQM5M%*LL)e656JomWz@>gN{z2b+CKMpexHUTNZpJayX zt3P=_75)V3*Z@K5b0@C~Z^=lfrviuj*ae0CWs_VD{!LxqyF>_yq*g3$$D=!Zyq=ui zRj?uQv7bJD1|Ry!@8HVIFT=g}Kd?5!g#%d+Qs%`eIYHC$;Gstzg}bQK1c-UzbqkLA zkgXk|=tZ^)s5;6^sC+u=$#cJ>%u5cIg~>hXX?|Q(3|WMo*VjX;~y&;zgz|Z`_{lZc(nJecDFc=i`Su96O@~( zZqY1NwM*;NF8{Q4T%yO2l4YpoD5bA-x&?eyd7jO38YX(5g`+3e_sYUD5F6JLVvpt4G^>VdbrUs1!6wy!shkAia z@#omc5uOx(J1_jkcY?!0@M-8v;qjWYytI(Ui-12xJY0sqK5 zQb_*Vn{UE*f7>^TmXWjRHk`ylb?>>x5&kiLpPIJhAMhbxE&}~fWGEf)$f>9QR^sL> zWN;9Bg=P0FZg#>0v8bFoN&%osn4X6d17d3;hr6bM#P<|9;YL<|(qtn48}IyU_`dJ_ zRu(9NvuW3Qq6wjLGdS~2kovUvfo;y#vwo65LkneIP)oZJti7Xy2UeeCSiMqb)AN@c zJytycCTj};0Iub^00_I+$6dFs&qni+SCP5Ur{xu~u5)}D+4NVv<@I>&tvBJ~qlaCZ zpzBaWIoyZ5Ax;sGCVs0=o=sP(UN#NEqbY7cpFEo$_2@l3%R?FoV&lp-udY*IJg7NX zvX|@6BZm&)hU>4zMHgL&V>e!pgX=?k!+z^*S{c4mF1%nL{`%Lv6?fh7+V#2b&L%ux zeZ~4%OiioHNK{kEA3*UKX~A`VcLS_GTlpGiLtSAbfLg6I;3~p_37T~Dnmg2{AZA<* zv+k?XU})i02M-*;)mL4COD{Q!$DVi+$B#dT6YGO}Og&aU3c7yANhOV zi#ObTCoZ}82=?r*FAh!b%G~>z2k;B;|8>0o*FMPjr^Zk$(p|fD;pcwlU*qrp!+(nB zU-%LuX`JnH=Eb@O_32-39;B!by!u@`E@q*x`qW#D+LO&o>cMa6)BB|d?&3&()==Ha zrb}JzQNFML|Lk+m;WPI=fajinVe@(7nP&{OV1aG2Y3qfbe)?Jb@t=MgpZokn;sMNw z^ob{*#;5-LK78@`qiXU-JmQS>(xXT5t?zv|-ujhqz>odN58%32Uo8f}Ph35}khAH& zpJEzfIJn-fS$p_H`}dn(rHknSH1!F)hy-r*vS~Oy?N`Sma_KdZnIGY@r6w^7yZ(f=bveZho53+QQ1#JgWUUk_e_~`%s#f@kD{vZBveBd{K%a9wduRJ;<+LUj4 z&pYuw-|;QD=goKHfzLgJpZ~>Q#()0#Uovoz?1N--w0-;%hy*=dPR}?#ec~B>?!n^* zzpv(wcMR}&hB=c*HW0*v<u6y4dtVyHI1eC1Q2B3jHqGRv=)L9*9k~GXUSY1A z{eEH-3e!vNG&m!d*23A`x=1rkcb-ZI+w|KX{{tLY?^9h z;wRTe_#k}zG}~9Q*>o~JNVzYsRDEtvCf0W!cGmmUY3p(J+&S#oS#1ooJa`)6FJC$( z@X#`BQzb>@(VdB>jYqtlnyWr5gc&AZKYR#@xst$+C3BBa2^UU}1aWLvWQHP=ba4yo z*IaWYuDkXceDq_#r^u7!pSV{f7|#(@_wgsxW7CFT=dQWqu%^acyArwO=16a#x0iCET;i($Q>#M%!t176aLI zVK?Gkk|CEXzF{_Q+-vJI4dGv~4dQ?GRhMr({nKk>bmv7&NOJ$(578(S;LeMdl`xb| zuVz9*2?tTxkSKR#vgzucmoEhfBZqZ^N;b(lZ48`7WYg3bfRjxtW(+im8^<=AHVhPg zLR*+|79*0;jeMhJz!bv7-(unF@}F`apkxDC{^^{!zp?=`20 zz^f+Lks0b`@B<%g77^fk-DP>Igg zDOe!B-NpLkFP;-)5SZ^EesV7*8o6MTOCMT0BMHVO_)?dCr%=1zC@4B+siPpXj;aGK zcHZ=kO*|6ZbW{9_%P(6W7ha5WbZ+h9S9tXCCzh9!8pThK9Xyg8$}!myr)ITRW_8xz zFC_S$CV3T2{xebWc*!zyvuVvg^$Y@1pZxU7;F0_@`HHs_M?<_L!R43Z)dE0MCkC=A zyH2Qu$7Y&=m3V6B>1y)}354Y-o>cggI--*J*??1OMj%N}j6cWEfx^cxIfnc_<&lN$Q$ovk-Zpij+)6!}hC$oO6%F@oANGY-&b zg{XZaO7jv1bT%EWX1#|Kc&gy(M<`qR;7?DB!d;vI_2U=NgCa`AU1QCOnla{&)hlPpE|&(6 zk-?MXE`fdOzb|6cXtL?lG!`$L_9oNCpU~B(n8P&Vhq@b(*$BeHY+B=~!GQmCQZg{f z2ouDrEo&JSfzjw4mxP2)mz^;Njd!k}vqFho5vcp=IJZ_@1%g=oka>r~2h$ zn*Lkq!MNdB$u#?i!ZPUKRU;ygo;4)Dr(u-b2JIy+_ZI%pwK(J?a)Ca7G?I&2a-R6| zvS~EdZLny?_pW^q$G#S5LY^dpI5`zr`h|LY?J~vg!q<@F3gRqb)4T% zO3END9_R~M!~_65I?f)EhXm67YKX3mp8+5(1u+O=v@NV8__w_yqyod(A<2aTndGp8 zp3ty3*)-{HW+Uks01e`&GJT;t&;}OZ?$HZJ!JA#d9evsf2>xC(Tdcion%H%-2xZGv z?1sw41O4FN`ELBpcfF&*)Z_2b)XM4o5>mQvmh;J`4S%xN$**?42>w&IYp%Q;H{Wor zp%JDa_PNxUF9(leG?q=HQHw|xEP&+^u6S{aVU&y@zJ10r%NDFQrc;@ZQt_u!c`aTzo*mv=2 z{!l184pRHN?q8=&rh^ds`n&GH|NO829=G0n6K=fzHTcRm-HqS=-QVB5D(QIkA^eju z-}B}-;B|N2h64vK+PvlV{0lD%V=P;Ot?&cr;3>9S_}6y61xg|Q#6%}taoHuf`NnH; z)7l$OW4Y+U3pN6td*Msqul54oG^TKVTlit{+t*_g?-ta^%S7PiSa1f-g_LM#|#hqU_i` zn{LpXfi3ioM`W%1Yj)GDg|Fho#E8{{#!BF;g?u-6>eLzh@t=Ge$B#eic-@ep{kr~q z@~P=p9qz}m8?VFv_@!UN!(V))_GAj=UcH@7k9)=Zs;b5>^yVqrc&+(9;^R*~g}?ah z=W+Pp!OdSQ_|4z?7#?}-@y)9OSwKquBUu$cI$lxXfu+|!L{Or$D>xxiQaXI)ZPX2O z0WMN>@i+#I5gj@#gBJ8Lr3-YSzv5Mw;rOGEyS*~Kgx$iQ>PD#-&XEkcZ@aV79a zpE7y!;Pm#3sW(Pi$f=OS#jh1DtNH9sPpD5u$YE4CN$9Z5OJT$DB<)!u_?2gYsk7-M z)I5?+Z+5-PHWiEwS}DEv!*uwf{Q%v>hy)$lVtECFiLXC z#rjx)@_Tlyx)RGy4xD4^shj**l)xFHHQY10euSrHFzFjqEpp;$tvKo3DdjfiA)8Y)YtbTz~F|_!r zBIShg17#5ETP`JwD{{786UCt(nvi#DEcI`-2wAn%QwCP;}uNO)D)n&> zc^7hRRxTUTU7GtXM+7rk=6Fy?n>cat8Usa6JmbRVMK-7vKL1mm(AHMfhheqG7pkTq zj>A>pz{G|{oq>+bzX!qKsgtGG(BL%H4qcMA6Cn0)xl~bkx|Q>`d)^=`Q5TVTa!1WN z=^n&ddR|do%4%X)G5v~(W>HG2-p{5PO_IF5Y?>QsU2NQQbV8jFUwO`4I@xqjk)EY7 z{Cu)9{+xh8P&Dv&vuUO)g{NxAX<*gwefK-?gWvm|K;RB^HeIEv_nPy+quDg1fSOHf z{z>SH-@;;hM8z=gCz!&ORC(zz8p@{0S?%5cb7?=^CJScPYx zL{6v$Mzd+oM+jx)Qf6xMiS$Yhxnb7kcZeFqx8gbcVd2Y{P0RLMZoD2}amTGVeCQy) z_HA$3oIk^%>%r0UE$5{XwK@W-$Mv^;>ByD05(i)m57niy+e(~Z{ z_uhw(ef$sc>mU3u{`AxL!80*2tYsi#T%uR1D=NMJMoIP04XY81P00XC1zkLVyHH|gCL+5N;%O1mKT)2$xY-9Xe&9 zwVtSGPn&!?e8nxqf90ObNUn&~@hpyWPu`e2XP}{v_Qa+DP2&tS=rK}NxjXnry*fmP zP=@DHbi|tefYH~#>ud3E{^dVf8{~!f7eDng_;3H~SMba;&#@>osLz!3=FCg3+45{% zGpZ=+yexmOdGS=yezEXPNP1K1Qf82+rO1^B1keR8oB}I0gRRd`&)n?PX}euZ_`TS1VYD)Ym6b5)aA<&g-^M{Qf8L(#cbJ z_W9@Qmc+1)W$@D@QvEDC9;)>?6gI|RsckV4S`3kuOc(DRNwJrMkeuRyM~l;D)1iRM zUX^K&G#TUcyUgeGJx3zf$iuDaAkf6g;W$$YlD*;dH5ER-J_#TA&`0nmpZ<*H1Iwp5 zFD{(uF2u;;hZc|wn#~^35GQ~?8s)839^+q5Hf<}MY#MdZ^%R&^E9~2|XY)Jy9O^2HZVZvF@kz{mAUWYd;TpcGDGV7j)sKxZfh#BkK)lKEXI-}R#q>yw*xP| zbZT>US&UHOC&7Ql&l8x*gHC_U{LuQ8V)_-@DXdR_Ywz^*msB-SsR+(hRM!M1b@7{w z#7vRYr%Aqx{sW+;nskRIocfc;6~9<;OrL_Mi$4kcJ1s9GzzQFtSc+`Az=vPh3MSK1 zrcyWLEr48sfQS3ksh8%(qrxFM*Ygnh&*P=d9`mC0{kZ9vY_%dB+<(!=2xmjoo&6CA zqa--lG+Vf;m&iiBc~LS&Hf``!1F8IBZM30JkxeTJF1jK7B*=H-G*t^+!n0aEd^LFG z70ad(z;ssg^h)K~vnMeE$gJjdnf|EabI(1GE3UYFLx22{M?FUHfl7Vq@LcCVM~~`V z%w!fhEwPnPq)-2DfK!JQc8IpPk$2xVo8GC^?bRp$^n#_9=2gRw#mHA44bQC&@|>9gjbDjZ`lO-}jX(dm=g4$11L^8>sa3Cb%USWCK8Znz!-^lXz0^fWRd-HdI_knPpTI)oD@)JC&NY`804xW z$t|-o{l}3KxUzfMbi%P?H)8ir`}sLB{?vnRM>fc1XGs-D5xls z;*!fQ#SXt;#~A@OtU;fFLf~oPpH}aM7wi*&k`#yWvuW~y>gIx4R&Rf{uEy2GBez90 zLU}MF{E;7LaKIyAFfY7AaIbXbRq`o!8da6N@$IdK`c7?aoqjOr&O7he7~{>yZrVK3 zG(D&=cd)BD6;9RzOV-rHKm8EXJHPgALLr{XZn&^)y2Lq`Qe7HF_;|8m$OK>Fy5-mn zY-nsnlhwAyC9++23JI>Fx4@sP^$mO1ba^(d^tn3UCyM#&oGu&nnadMTJc-}?y-#eO z68WP)`Xv7R&+o^XGpE7LZUjygK0Q3sFKFF)`)#=VvdeJmEyr;1z=2ReUD*+T`Te?U zufgB?=Jx=D`qW-nw@mn_pAvcF>+is2mt2f)p_osk&oqFuUmuZ=&&yO+-PirhYVa77 z*|2h_&Jzs&a~snQ=La8rXmhJX_Fx=zb!nCPXhE5yOxbYHEz?cv+it#T{ojq7Qm7_EkyO_N8CcYO6d_?~b777wVxKXc_rW}beox%vvc?VdN{=;0%qH}h_~ z@jCO9b2GajtU4nQurYT3mPBtyMlL*zT?KZ9mY5CbS@1+6MP&&6f z^2j6W%X3bzA7pJyaE?qco9O7VFa`K z!RCUj&sMBTpIMU*i2)k8mLdPZTH4 zZ!Rys_#$3B`4S#`>(hI3-p-g}W7<3R zFi4#aU4D$Xji;tDc3iU{aTIKNrs^V_O7!tm1tzwilObUa7dBMK_-h9~K@v^cJZ|;R z|H+SSUS7KR=;6&>04zg!;e{9QXZPNRANb)PU!Tf6raO1_$spj(Z@e3y_=8WP`ScFc zc>&STXG)SpT%(SWpY{6_*~UM2m1pZdo9UgEwISL?Ydgt{?lLbxBWhCI`Zsmepp&ev zR4fF+&XM&F5&qOcGf&WEmtKN9UV96E`X~P}u3qms|J6_bEZ+a?zk!DyI?kT_NgjhH z9y6{~I}M=k;y3abx#ZuP-x$A#C%OEW=0mSKM$+i>VAT`3#*7UjB{%;@o?M+Cm`o3c ze$P6wnA)ZXT*bic5?)#@e&1OP8e#{3_8kA=r~Vn<^RBPKvDbaIdh{$%Tv+Y@IAHy= zyuNoC#jl782=ix)p$)Nzzpc+6{v9uSN^n+!+(fW1zVMn|@;|@Jeq&j2slHtDnx44c zwcc~RRQ5*;or`D2KciRPV^OfYi5yNN>fPU@_)D?b;6M7YkK@3B{m%1`AyBIbjGs*# zN8#d6Lq@oftJE*;!2osex4mNTYH>~Yo0-mDk$XqZoSH@mc>YufUA!WH(mNN>X zbW(G{zP(P1)Th|afNO`_1FWre^vTbr0rq#gk^eerQ69;8B^fNl74b>snhz$F>IF9t za!IKud-@%hlP6z7%v9lrUJ_D*Hb%q8aL84jCM1XYeh+y+iEe*ul z|E$kxtJ6L%tP745er=2{OwC8>lLS49t}9K zhM!efk`-ND9_?ZVIp2Yo*>o`@9XCWn3UCv&fEwO3n~W!QhLcw_s}f>!SXFzV9DZut zll%<+9F~&eFU9h+>BRII_fXt&z;4g0|)Tn!^d&v>{-O6 zj{nP(O&dcbMNp4{Dn;Y=PV!_mCRAF0n3|+y&bAvFBbUJp(p?AgNpF`023B*B9!v9* z`b>jQR!25EH&uh=#yEYOvt3_D(`}{cNvAWX&(xQ|y82usoL4r@D`Ii077*?+7F%Ev zCe)Ee65vz^C)? zXP$iyW5w4HM>(LK22WW6LtLCMpY+5BUI!~yAV--7fE50V_hp^9vdhG>>^n92vjm1Q zl3|m3ceZX4w#%kdGRCOYkQAP+vS|kbv2g%$k>&?sh1?K78%MKLdl1_hR7>D^9H4@a z$6fy=Zq_q*TK?nnO5Pf&KuGg!Wz&p?KXOirw=hP=0~Rw!c>gGg{RYJ~@r zIggy(C2({hxhqdZF<1Vr%$D(>31K$9;oq4TC1EI=)?FPwym9(>MPWK58INVtNmp0d z9sVV91$Y9n1Ktr$Jlaco{D+EAkYvMEuc7kcCWR|1R`dnOmZhu2Y@t`_uw&6FYagMqW@4gUsBk(W@P#OAfu9z-`l?jx8!%U=~h zPkzB$0mvXS3Lsa8OB-^A$YoR1n@>BM_{qJL6nR49u6o2iqrh{(H|6&IQFrCIv>tvc z{71baVLVhz@)XhNsz9G5SrM;@Hoc47uuANF$DpKA1SS$ijGjU^$!VYte#1ymkVmcP z9Ro3K4Q*aYz~ zpu&FEQrqn?`pmyN{iN8?LPr`8BK&0VNW-WF7_nr3A&kQ}J!;H*x9xQkIIj=&AmsYcnzv=$99 z!(I%|^D-O{ID5@Z4~+EiNQ}@-8TW`}6r|`B20_2=dbZpn2l^Y zIs9}9o|KlJ!w!w(S7``BMZl}}8>07)!!npnCq|x}mI#mQ5yf#-XfTz2cKxZ&ORB43 zD4P}&>TJ#h{XC%&8=Xx<&c9;J+&E{n;2p}QDPxW-fus=t^f1eWLG_w`8v&MdiYW`yxjyQlm@3a2{( zkou2_47oKQ33Sy`-gq{xFq&*Si&STz91D+8zV;3MST>#Tva{(_aaa7wz$9V$jlzwP zz1XBjtBt&!O~?2vHHlBozc`Ns6Qaq(FT71umU%J!t2?TAW=1Qq@#z?)c;vtG<5b|G z!q2Ajh05BvE6ceDex6Y7u^@&uyJ+u4a?6;xamDDB%cd0wmQClLd~fGv(-wY$ zP_)KxBrrG%|5!F1D$(HTWz#DIxJnA-(-vNlm;y`HB69xMoPoLuB(x_K)o4j@bB09& z`crYx(T_HGPF@8!;}`2-0r6oGmleC{8TU}Ljs z-&Pzvsp22crqR}?$fi*Md1`>*Pvv9_Ti`GIK_x+N`?kuadB?CUv(3qub-ofy1PXA; zreoH{BPNVx(+oh4dbMd5tZz36i|XK+Jsio&Y-H0sRC5JY!5lwT530PR%%;f>eNdmp zgF1jXf1#VDVf1c*kj3gUHK3*hDx0POuOpb~imFd?J)+lVuz^1bJmSiuaBi~E+OQQK zg#6W_&2K7FAM;EPH)ir|S~XN785vmNPZ-Xo6Z4KE`~gQ1x!R|X_**=sL&rN-@`-Fm zvQ!G>)kQHWV3-)-6b>{u)SsXPbE~P5JBE2$oHl>_dK1cD>_`-jGhcN?=h1sh(o1GKBET7$;Gtk!-r+NsVk;^OvB#Y?{-P5v&lvUcY@GPGo`FkBm z;Eu9s?PVk$4CB$+bhyTHxKm!hVkh}kd6{|=ldo0Rl=afVPml^?fVkLKKAW~DRh*_~ z(}b#Li4MNhpv%z;l59mCvTtYKLC1sfh=JVyoj@yeur7w9UZ8UZ$~YS_Gee zHl26tKK(0|O^dhqX7ANyS@-{Z4-@-c!M|@f2|CG>E7CawO|ag&g{RQ7_j4JFYB)1y zo<$GsyIlP8VTm$YBeHIvgm0$5&T_L8T3U!LBEjCFZ|IYU=)%Cq zUk)2c8NAC>iS}c$?YYpWq-h!+-B}eYpEfeahH^SZY-&fsowui8m!x>cIv}r7DZ7p> zX2bXl!1ox{`=WE@7=EV+&ku}b(`BCsjCu8K^qG`;$VI{E)y99pWafXSxoB~x^6NC@@G*KuAOC7X7ljTtD*rbCaInaOb7@h)L#kNM~C4wb;ojgg0o zel`u=$;Bh^Q{ExkHJ1MnIzJYCIob65QSp;(V8%{;qu(Z*HvA#rQ0WU8e{jPNwVkL_Ui&tge@=j2Jz<{l&bJJiKH z4)+N3+03Sg*fcz%rS(E786b&Yh&U4$?HwHo2yk6$(b;rJsrff( fcDlN%%NDz|h^900000NkvXXu0mjfir%Lk literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/mipmap-hdpi/icon_background_clip.png b/TMessagesProj/src/main/res/mipmap-hdpi/icon_background_clip.png new file mode 100644 index 0000000000000000000000000000000000000000..02fb01faca149049562c547ddf47525858482b71 GIT binary patch literal 2578 zcmd5;`#%$m7akUpd4+^UZ@dz@6edy1{j%h`xol&|)QpsPi-=zBt6WAGWJ!|j!sgP( zyVqq+y4Z?jEHYn0*<3=V<@T!niT8(do^zgap7S}sopasyq&rASM+pD`fIK~1{5MhX z!xZH={q^6?W;S7Kf=5Uq0I*H{hsgj+?`dt4GKv1~P5|1QT~nJ0Fvii_5df$sY+H@k zvMH$cba4#4C^IwcLxRCHUd%5&Ja#+8*U7iU3lr?y^Ngdfk7sk0k88=w1NG6zfB&z;11=~o!3jdMBiTK&tNHG@)F>i}nngbR zR|^|-e-yctR$9UGOUl7AvH93GkvZSG!y|9a3?H%L#6U3I89__01p>ht@b3`nu`(Xa z%-1aND2es%Kp+rjdwKjld9WfSJi;#0;KAejWUd-NiqhGio2BT&9{ZNI=D~1fcvpPy z40U;(-fGSb^dalP^R(sjGeE+Fd^d}0B7_)_lsR-E1SOL1V5`xh1sAekS@Njf*7 zCuKdK!7r-TEynqq1-+1V^c0_`7aoE#k2tsJmoh*KNb!M2(`!2x+p$97V6` z24C+$`HgU1mYZ@?Ewxs9^PfTE)l#`hi^92R69@PuA7-~nnl?&o`p{xN#IbqoPHIc9 z2?`F2W%rXtj~!BW4Z*@EY9jLp*=;82##Z9tW3v6P(?FdlbMm z7)#;s^{UMJ5nHfYPcM&fE`>C~#gqoa4?#tjM>=)f>KIwnG%sQeW4wWYZrg>v;wx^h_e3|CbilPvzbhqrBVItW$bVZPaLz{c%s; z-t4lwD($|I8uaE91$DT5Iu?3S5#7Ii+L^=Gt*Uis$ZC+s$5B!^uITdh=a*{ zE%(CTJDvYvr$MH!MiJ$Rnlw&is#G>Ax=hY2Fw>dN#wx;2)0CrWex*lO2=(0SSP#Zt3hQ|v!C0)UV(Wx z0xk2HiNuFa`)f}1Zj>UJxKJUYWYArl3i zKJu#qIaT=B%xcSY^PU{*xVy$^Tz8TCPwlOr+FJLQvd)FZ=qUn+?D3P|5BfEN82?h= z8-Y<0C_8JO8MF*w$6NmzWxC{f{f_6mqPwy-Lj|2UsD1!@Qmet^Co%G4u~YBBPn3wl zC&hi+q5HoFl{;*Kt@bUMMcSB96xhX^fluY6988smumy4amRoDw+B{<_^aXwj_xH&Q&) zt1lT)erOs4xzH7TI1WB5#ZdPlY0uTo*DBj~f;uLbTUdmM*i@I*0b{;%O5iF_^(1OK z9pwN=Mcg4n&~J+UhN%U*R#|?_3rYADQ|XPoFsuV`7<`#%$L8ZzCxr2hUL6xym6FAy z-OY(^%LKoz1f=MuVFFL#utA|SH;!Br`*59~{~2-41Vue9pF zExjKEnfDVv>%M)vklWEYCVDs-@+{1Td_Umjy5-l})z=e*Yq`jnJ?NcG6vKbdN=JfU zG_%LZnzBI@BY`zd!<6V2BVzES?umeTShuEx^N0IcVT6P%HyqRa;QZNDL(5T7W`;K} zydeEy{#Uhk-(tSCeo1|JB|UOSyUVNsA$x78B84UKL3J4!5%U6Y-#Lo?$L4$$u}V8wkdy{f~xo^>#=92HN1Yms%7mT-%O2-%>df^dhwucdY2<25#Ldw9yeX zS_*WlBQK`y9H7`lzjth&ZWdeQhSw`G;W;jB`4{aLLl07*j`k6{fhv(-TF2JJQ5Xe< zj118I_7dS|{+-2FT~Z_!T6Hu#LPIflS37+-g-K#I&G0^--`ijjw{XjFcm1PJiFrCQ zV3m=Oi`9_1USW_RU!X`L$A}u9m7bAw!e336SGus>qL52uECEXwfv?Rz0Tq<;W3oR{ zG)bBiJ>TJIEfd*(6$j(73i}@y8uxno5nbmn#g4Y1-F0hXW98*XtF(78tH3rECuRjh zog_^n)E}S07ykG?!mQ;o6}42eGi3hzur5>m=a5oK&=J$k7Y^X*deVjFguePOTT7tQ literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/mipmap-hdpi/icon_background_clip_round.png b/TMessagesProj/src/main/res/mipmap-hdpi/icon_background_clip_round.png new file mode 100644 index 0000000000000000000000000000000000000000..2b6dbe8d4d5a50fa45705ec17182af954d54db13 GIT binary patch literal 2205 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 literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/mipmap-xhdpi/icon_2_background_sa.png b/TMessagesProj/src/main/res/mipmap-xhdpi/icon_2_background_sa.png new file mode 100644 index 0000000000000000000000000000000000000000..c54375e52bcb6b0eea65e4b2ff6d088bde0b20e7 GIT binary patch literal 22932 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 P00000NkvXXu0mjfZ+&g1 literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/mipmap-xhdpi/icon_background_clip.png b/TMessagesProj/src/main/res/mipmap-xhdpi/icon_background_clip.png new file mode 100644 index 0000000000000000000000000000000000000000..02fb01faca149049562c547ddf47525858482b71 GIT binary patch literal 2578 zcmd5;`#%$m7akUpd4+^UZ@dz@6edy1{j%h`xol&|)QpsPi-=zBt6WAGWJ!|j!sgP( zyVqq+y4Z?jEHYn0*<3=V<@T!niT8(do^zgap7S}sopasyq&rASM+pD`fIK~1{5MhX z!xZH={q^6?W;S7Kf=5Uq0I*H{hsgj+?`dt4GKv1~P5|1QT~nJ0Fvii_5df$sY+H@k zvMH$cba4#4C^IwcLxRCHUd%5&Ja#+8*U7iU3lr?y^Ngdfk7sk0k88=w1NG6zfB&z;11=~o!3jdMBiTK&tNHG@)F>i}nngbR zR|^|-e-yctR$9UGOUl7AvH93GkvZSG!y|9a3?H%L#6U3I89__01p>ht@b3`nu`(Xa z%-1aND2es%Kp+rjdwKjld9WfSJi;#0;KAejWUd-NiqhGio2BT&9{ZNI=D~1fcvpPy z40U;(-fGSb^dalP^R(sjGeE+Fd^d}0B7_)_lsR-E1SOL1V5`xh1sAekS@Njf*7 zCuKdK!7r-TEynqq1-+1V^c0_`7aoE#k2tsJmoh*KNb!M2(`!2x+p$97V6` z24C+$`HgU1mYZ@?Ewxs9^PfTE)l#`hi^92R69@PuA7-~nnl?&o`p{xN#IbqoPHIc9 z2?`F2W%rXtj~!BW4Z*@EY9jLp*=;82##Z9tW3v6P(?FdlbMm z7)#;s^{UMJ5nHfYPcM&fE`>C~#gqoa4?#tjM>=)f>KIwnG%sQeW4wWYZrg>v;wx^h_e3|CbilPvzbhqrBVItW$bVZPaLz{c%s; z-t4lwD($|I8uaE91$DT5Iu?3S5#7Ii+L^=Gt*Uis$ZC+s$5B!^uITdh=a*{ zE%(CTJDvYvr$MH!MiJ$Rnlw&is#G>Ax=hY2Fw>dN#wx;2)0CrWex*lO2=(0SSP#Zt3hQ|v!C0)UV(Wx z0xk2HiNuFa`)f}1Zj>UJxKJUYWYArl3i zKJu#qIaT=B%xcSY^PU{*xVy$^Tz8TCPwlOr+FJLQvd)FZ=qUn+?D3P|5BfEN82?h= z8-Y<0C_8JO8MF*w$6NmzWxC{f{f_6mqPwy-Lj|2UsD1!@Qmet^Co%G4u~YBBPn3wl zC&hi+q5HoFl{;*Kt@bUMMcSB96xhX^fluY6988smumy4amRoDw+B{<_^aXwj_xH&Q&) zt1lT)erOs4xzH7TI1WB5#ZdPlY0uTo*DBj~f;uLbTUdmM*i@I*0b{;%O5iF_^(1OK z9pwN=Mcg4n&~J+UhN%U*R#|?_3rYADQ|XPoFsuV`7<`#%$L8ZzCxr2hUL6xym6FAy z-OY(^%LKoz1f=MuVFFL#utA|SH;!Br`*59~{~2-41Vue9pF zExjKEnfDVv>%M)vklWEYCVDs-@+{1Td_Umjy5-l})z=e*Yq`jnJ?NcG6vKbdN=JfU zG_%LZnzBI@BY`zd!<6V2BVzES?umeTShuEx^N0IcVT6P%HyqRa;QZNDL(5T7W`;K} zydeEy{#Uhk-(tSCeo1|JB|UOSyUVNsA$x78B84UKL3J4!5%U6Y-#Lo?$L4$$u}V8wkdy{f~xo^>#=92HN1Yms%7mT-%O2-%>df^dhwucdY2<25#Ldw9yeX zS_*WlBQK`y9H7`lzjth&ZWdeQhSw`G;W;jB`4{aLLl07*j`k6{fhv(-TF2JJQ5Xe< zj118I_7dS|{+-2FT~Z_!T6Hu#LPIflS37+-g-K#I&G0^--`ijjw{XjFcm1PJiFrCQ zV3m=Oi`9_1USW_RU!X`L$A}u9m7bAw!e336SGus>qL52uECEXwfv?Rz0Tq<;W3oR{ zG)bBiJ>TJIEfd*(6$j(73i}@y8uxno5nbmn#g4Y1-F0hXW98*XtF(78tH3rECuRjh zog_^n)E}S07ykG?!mQ;o6}42eGi3hzur5>m=a5oK&=J$k7Y^X*deVjFguePOTT7tQ literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/mipmap-xhdpi/icon_background_clip_round.png b/TMessagesProj/src/main/res/mipmap-xhdpi/icon_background_clip_round.png new file mode 100644 index 0000000000000000000000000000000000000000..9796f81b0a7091f53737440453d061f504f5afe1 GIT binary patch literal 2716 zcmd5;`#;nBAD^g%>7++tsOac23Ar_n>s(@P%`P=$>MLwcoa35HCd@6VLxvKXOJ;Ut zrd*p#a#&ON47n`ih&9VW$7SQF@A)In53l$0{d&IMkJt0@dc5E7$K#pq;pPaIgUEqE zAh6Q~JLC?Id<)qsLDq2c7;hTmYKJ$Xb$57jbBvPwTfcI`nL?Af-1 z#OaUhCH?`Q1lI@M%S~=6j5=E8dQ@o|9yGtUmv!g1ZQ4E?nSJ*^bXhxU-9B~7(|*&d zH5QlSq8`v9;QGWSSk<0i);b>`~tznt1DDyY|Rv{f8i?OMhaK zMn=z}lzaP?(G)b+dBQ{DWsKW;a|Un$f`Jy~GCC8LiAJFM`ue5GL587Ng;t{{X6S|b zh0km@+gQGY$9?CQLSPYGt5V31HStSvJXon2-K=M{htB$Hmf zC^xHk|NF*6+X!Ie9?lAzibai~^k~;rnleZc!^5@5sU=e`D3Crd^)t}dJ<)So6Ym*@ zVtKW13_(gX|Dr33s(5NAQb?{`EX_xbCq=wR?!B%tiM8;LK1Tl-t#67>@Tdti_GX9| zw^|E}Gar)o86>%KLhvn&_K3+ZRuB_J#0YU*wo&hTZE>X6Y6U@KYUA0jZoJEL>Juh+ zGF5})__PM%4@Ku(F2H*z`1rDzjxQD*yw#t@h8QKrUMv@I7CbT^Pa;Y98UpXd;CH0P ztNFRVavLtqsZSb+l__Z&Tc>&y!M|mL_q)Ou&>CA;iyR4s4r~six&i9_TEU0vH=CKd zO^e|`crolY&!3SmEKS3U$UH+cb(F=8ECLMQNu#{jeY85xH?qsj9cx+%kk&b?KTR!U zDb?lhKDh9veeAY5tn|=TH`wr&EGL5KLX=u=PVrjj0I{==LRfD53*RBx;^qR4aotCj zv*V&#a@$GbX?)0%#qf%bMRXtJU9YR6ZwFI*b4JWMHKrtnvqquAw(Pj6Mdv)LRb{X_ z|IDg)gVTe9oZ#t;WgFdXlrOFBk{A5f@Jg= z6Em6@x1g{@|yeWLGu~ZdOu|42ds+|+D=UU~{uVtTMQt-nv^i*1Ou2jx% zb#tm#N$cBD49g@RLnC-+_+7V7jlFc7HK&N09%gKM0JH}6O_2M3~?CIb{y^ahF8rpbf)|FdPsd?1&i=VOWu@GY=% zpYIzg8B;EZ7wXyUtw48^%E!erKKH-%H5vTYw_l}p^!tHs(i_8isE=r$$~jHEg+Bk^ zqW!-`U6Xpkvv2AMh5xI2-T0=?{8;5q|e&BE9wD!BU%!;7V{RAn_SclP>%3Hk@~HF zZ&KVPG8!UOYAaIuER!04MQPEdc%d|0_-PB2vEBSr2yJu=2P{$Oh zk>8`<4MNTycunW|PX~}!jDEzJbauciyENbEJ}7Z=_xfOdUk5 zh0aD33fo&WMXGhnM=%G@V#frTJb~Va>u~4tC_#;uv5(f>b!V0>C8K4Z;Q=r_A*}Z@ z;EA~p@nz3}-~0D;$70Ygyt54Ro2z1Q%#oXL{@@SZv`$rkiYdVPY6`O6d*zLlWIs3} zWXW~T5^gyz;{cI@gd1DWgQ9mnTK*W#cLBMSTPh0tw2ldWjn2MQPS~HniD4l%o3DOS zgG6qfYt=)!UwTDovs0e|i%I?0B-E8@>FS@n87U;#_O6^AqBHhg*KN|_o!Q4PN^q2n zqiPjd4i6S%NGa5>T8X)+#`@=;w}z8XU3~TI1n7khM>Oc#_^%!lGAdABOlk; zkRfYxWJC0g+uMvSbHQtyZbHhXzIiInq`iR2Hc>=tm_aE$jx)z*Tnf zVvo~2%f-f30f(B1hplUhsGV{E0VqVugDv3fisIsJKL7@#{IzTk_yZ4+rgWxXf8e^Q zrC^OR$wV(ZzrpI(aXYiV8;WtsLkAfJ*)HWb|2b#0LX-{{h#2`{V!s literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/mipmap-xxhdpi/icon_2_background_sa.png b/TMessagesProj/src/main/res/mipmap-xxhdpi/icon_2_background_sa.png new file mode 100644 index 0000000000000000000000000000000000000000..0eee843d283a578878719db19840fc695aed02c5 GIT binary patch literal 60781 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 literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/mipmap-xxhdpi/icon_background_clip.png b/TMessagesProj/src/main/res/mipmap-xxhdpi/icon_background_clip.png new file mode 100644 index 0000000000000000000000000000000000000000..7974edca0b4551b672a7eb2e1deb4cd8d8265186 GIT binary patch literal 4523 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 literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/mipmap-xxhdpi/icon_background_clip_round.png b/TMessagesProj/src/main/res/mipmap-xxhdpi/icon_background_clip_round.png new file mode 100644 index 0000000000000000000000000000000000000000..8e1d96da7b2a308f498b8c9636f44f2131c71219 GIT binary patch literal 4821 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ⅅ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 Q3GvZOV literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/mipmap-xxxhdpi/icon_background_clip.png b/TMessagesProj/src/main/res/mipmap-xxxhdpi/icon_background_clip.png new file mode 100644 index 0000000000000000000000000000000000000000..c10fc6ed462ad84970b71e64e49db486b502f6d2 GIT binary patch literal 5492 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 literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/mipmap-xxxhdpi/icon_background_clip_round.png b/TMessagesProj/src/main/res/mipmap-xxxhdpi/icon_background_clip_round.png new file mode 100644 index 0000000000000000000000000000000000000000..3a22d7ff678a66c257bc761bad2b38595ddfa753 GIT binary patch literal 5846 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 literal 0 HcmV?d00001 diff --git a/TMessagesProj/src/main/res/raw/bt_to_speaker.json b/TMessagesProj/src/main/res/raw/bt_to_speaker.json new file mode 100644 index 00000000000..369a9132fef --- /dev/null +++ b/TMessagesProj/src/main/res/raw/bt_to_speaker.json @@ -0,0 +1 @@ +{"v":"5.10.1","fr":60,"ip":0,"op":60,"w":156,"h":156,"nm":"bt_to_speaker","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":3,"nm":"Top 3","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.35]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"i":{"x":[0.833],"y":[0.855]},"o":{"x":[0.167],"y":[0.096]},"t":1,"s":[0.042]},{"i":{"x":[0.833],"y":[0.721]},"o":{"x":[0.167],"y":[0.195]},"t":2,"s":[0.328]},{"i":{"x":[0.833],"y":[0.829]},"o":{"x":[0.167],"y":[0.119]},"t":3,"s":[0.54]},{"i":{"x":[0.833],"y":[0.839]},"o":{"x":[0.167],"y":[0.162]},"t":4,"s":[1.038]},{"i":{"x":[0.833],"y":[0.898]},"o":{"x":[0.167],"y":[0.172]},"t":5,"s":[1.564]},{"i":{"x":[0.833],"y":[1.207]},"o":{"x":[0.167],"y":[0.458]},"t":6,"s":[2.057]},{"i":{"x":[0.833],"y":[0.827]},"o":{"x":[0.167],"y":[0.059]},"t":7,"s":[2.167]},{"i":{"x":[0.833],"y":[0.697]},"o":{"x":[0.167],"y":[0.161]},"t":8,"s":[1.785]},{"i":{"x":[0.833],"y":[0.821]},"o":{"x":[0.167],"y":[0.115]},"t":9,"s":[1.374]},{"i":{"x":[0.833],"y":[0.875]},"o":{"x":[0.167],"y":[0.156]},"t":10,"s":[0.289]},{"i":{"x":[0.833],"y":[0.76]},"o":{"x":[0.167],"y":[0.249]},"t":11,"s":[-0.957]},{"i":{"x":[0.833],"y":[0.845]},"o":{"x":[0.167],"y":[0.128]},"t":12,"s":[-1.583]},{"i":{"x":[0.833],"y":[0.881]},"o":{"x":[0.167],"y":[0.181]},"t":13,"s":[-2.759]},{"i":{"x":[0.833],"y":[0.782]},"o":{"x":[0.167],"y":[0.281]},"t":14,"s":[-3.767]},{"i":{"x":[0.833],"y":[0.859]},"o":{"x":[0.167],"y":[0.135]},"t":15,"s":[-4.193]},{"i":{"x":[0.833],"y":[0.888]},"o":{"x":[0.167],"y":[0.205]},"t":16,"s":[-4.879]},{"i":{"x":[0.833],"y":[0.814]},"o":{"x":[0.167],"y":[0.325]},"t":17,"s":[-5.351]},{"i":{"x":[0.833],"y":[0.937]},"o":{"x":[0.167],"y":[0.151]},"t":18,"s":[-5.513]},{"i":{"x":[0.833],"y":[0.375]},"o":{"x":[0.167],"y":[-0.262]},"t":19,"s":[-5.713]},{"i":{"x":[0.833],"y":[0.626]},"o":{"x":[0.167],"y":[0.096]},"t":20,"s":[-5.665]},{"i":{"x":[0.833],"y":[0.806]},"o":{"x":[0.167],"y":[0.107]},"t":21,"s":[-5.352]},{"i":{"x":[0.833],"y":[0.872]},"o":{"x":[0.167],"y":[0.146]},"t":22,"s":[-4.261]},{"i":{"x":[0.833],"y":[0.753]},"o":{"x":[0.167],"y":[0.24]},"t":23,"s":[-2.819]},{"i":{"x":[0.833],"y":[0.842]},"o":{"x":[0.167],"y":[0.126]},"t":24,"s":[-2.05]},{"i":{"x":[0.833],"y":[0.88]},"o":{"x":[0.167],"y":[0.176]},"t":25,"s":[-0.539]},{"i":{"x":[0.833],"y":[0.778]},"o":{"x":[0.167],"y":[0.274]},"t":26,"s":[0.821]},{"i":{"x":[0.833],"y":[0.859]},"o":{"x":[0.167],"y":[0.133]},"t":27,"s":[1.416]},{"i":{"x":[0.833],"y":[0.898]},"o":{"x":[0.167],"y":[0.205]},"t":28,"s":[2.408]},{"i":{"x":[0.833],"y":[0.93]},"o":{"x":[0.167],"y":[0.46]},"t":29,"s":[3.089]},{"i":{"x":[0.833],"y":[-0.224]},"o":{"x":[0.167],"y":[-0.436]},"t":30,"s":[3.24]},{"i":{"x":[0.833],"y":[0.857]},"o":{"x":[0.167],"y":[0.089]},"t":31,"s":[3.216]},{"i":{"x":[0.833],"y":[0.724]},"o":{"x":[0.167],"y":[0.199]},"t":32,"s":[2.885]},{"i":{"x":[0.833],"y":[0.83]},"o":{"x":[0.167],"y":[0.119]},"t":33,"s":[2.647]},{"i":{"x":[0.833],"y":[0.877]},"o":{"x":[0.167],"y":[0.163]},"t":34,"s":[2.097]},{"i":{"x":[0.833],"y":[0.766]},"o":{"x":[0.167],"y":[0.257]},"t":35,"s":[1.524]},{"i":{"x":[0.833],"y":[0.848]},"o":{"x":[0.167],"y":[0.129]},"t":36,"s":[1.249]},{"i":{"x":[0.833],"y":[0.883]},"o":{"x":[0.167],"y":[0.185]},"t":37,"s":[0.751]},{"i":{"x":[0.833],"y":[0.788]},"o":{"x":[0.167],"y":[0.288]},"t":38,"s":[0.344]},{"i":{"x":[0.833],"y":[0.863]},"o":{"x":[0.167],"y":[0.137]},"t":39,"s":[0.178]},{"i":{"x":[0.833],"y":[0.89]},"o":{"x":[0.167],"y":[0.213]},"t":40,"s":[-0.079]},{"i":{"x":[0.833],"y":[0.83]},"o":{"x":[0.167],"y":[0.347]},"t":41,"s":[-0.243]},{"i":{"x":[0.833],"y":[0.914]},"o":{"x":[0.167],"y":[0.164]},"t":42,"s":[-0.295]},{"i":{"x":[0.833],"y":[1.491]},"o":{"x":[0.167],"y":[2.434]},"t":43,"s":[-0.349]},{"i":{"x":[0.833],"y":[0.632]},"o":{"x":[0.167],"y":[0.071]},"t":44,"s":[-0.351]},{"i":{"x":[0.833],"y":[0.808]},"o":{"x":[0.167],"y":[0.108]},"t":45,"s":[-0.338]},{"i":{"x":[0.833],"y":[0.872]},"o":{"x":[0.167],"y":[0.147]},"t":46,"s":[-0.293]},{"i":{"x":[0.833],"y":[0.753]},"o":{"x":[0.167],"y":[0.24]},"t":47,"s":[-0.234]},{"i":{"x":[0.833],"y":[0.842]},"o":{"x":[0.167],"y":[0.126]},"t":48,"s":[-0.203]},{"i":{"x":[0.833],"y":[0.88]},"o":{"x":[0.167],"y":[0.176]},"t":49,"s":[-0.142]},{"i":{"x":[0.833],"y":[0.778]},"o":{"x":[0.167],"y":[0.274]},"t":50,"s":[-0.087]},{"i":{"x":[0.833],"y":[0.856]},"o":{"x":[0.167],"y":[0.133]},"t":51,"s":[-0.063]},{"i":{"x":[0.833],"y":[0.886]},"o":{"x":[0.167],"y":[0.198]},"t":52,"s":[-0.023]},{"i":{"x":[0.833],"y":[0.804]},"o":{"x":[0.167],"y":[0.311]},"t":53,"s":[0.006]},{"i":{"x":[0.833],"y":[0.878]},"o":{"x":[0.167],"y":[0.145]},"t":54,"s":[0.017]},{"i":{"x":[0.833],"y":[0.904]},"o":{"x":[0.167],"y":[0.265]},"t":55,"s":[0.031]},{"i":{"x":[0.833],"y":[1.045]},"o":{"x":[0.167],"y":[0.647]},"t":56,"s":[0.038]},{"i":{"x":[0.833],"y":[0.658]},"o":{"x":[0.167],"y":[0.029]},"t":57,"s":[0.038]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.11]},"t":58,"s":[0.037]},{"t":59,"s":[0.032]}],"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.713},"o":{"x":0.167,"y":0.167},"t":0,"s":[77.722,79.526,0],"to":[0,0.053,0],"ti":[0,-0.183,0]},{"i":{"x":0.833,"y":0.864},"o":{"x":0.167,"y":0.117},"t":1,"s":[77.722,79.844,0],"to":[0,0.183,0],"ti":[0,-0.211,0]},{"i":{"x":0.833,"y":0.736},"o":{"x":0.167,"y":0.216},"t":2,"s":[77.722,80.623,0],"to":[0,0.211,0],"ti":[0,-0.258,0]},{"i":{"x":0.833,"y":0.849},"o":{"x":0.167,"y":0.122},"t":3,"s":[77.722,81.112,0],"to":[0,0.258,0],"ti":[0,-0.32,0]},{"i":{"x":0.833,"y":0.881},"o":{"x":0.167,"y":0.186},"t":4,"s":[77.722,82.17,0],"to":[0,0.32,0],"ti":[0,-0.087,0]},{"i":{"x":0.833,"y":0.796},"o":{"x":0.167,"y":0.276},"t":5,"s":[77.722,83.03,0],"to":[0,0.087,0],"ti":[0,0.147,0]},{"i":{"x":0.833,"y":0.679},"o":{"x":0.167,"y":0.141},"t":6,"s":[77.722,82.69,0],"to":[0,-0.147,0],"ti":[0,0.348,0]},{"i":{"x":0.833,"y":0.868},"o":{"x":0.167,"y":0.112},"t":7,"s":[77.722,82.15,0],"to":[0,-0.348,0],"ti":[0,0.409,0]},{"i":{"x":0.833,"y":0.743},"o":{"x":0.167,"y":0.225},"t":8,"s":[77.722,80.604,0],"to":[0,-0.409,0],"ti":[0,0.468,0]},{"i":{"x":0.833,"y":0.837},"o":{"x":0.167,"y":0.123},"t":9,"s":[77.722,79.694,0],"to":[0,-0.468,0],"ti":[0,0.62,0]},{"i":{"x":0.833,"y":0.879},"o":{"x":0.167,"y":0.17},"t":10,"s":[77.722,77.793,0],"to":[0,-0.62,0],"ti":[0,0.442,0]},{"i":{"x":0.833,"y":0.772},"o":{"x":0.167,"y":0.266},"t":11,"s":[77.722,75.972,0],"to":[0,-0.442,0],"ti":[0,0.378,0]},{"i":{"x":0.833,"y":0.852},"o":{"x":0.167,"y":0.131},"t":12,"s":[77.722,75.142,0],"to":[0,-0.378,0],"ti":[0,0.425,0]},{"i":{"x":0.833,"y":0.884},"o":{"x":0.167,"y":0.191},"t":13,"s":[77.722,73.703,0],"to":[0,-0.425,0],"ti":[0,0.257,0]},{"i":{"x":0.833,"y":0.795},"o":{"x":0.167,"y":0.298},"t":14,"s":[77.722,72.593,0],"to":[0,-0.257,0],"ti":[0,0.176,0]},{"i":{"x":0.833,"y":0.87},"o":{"x":0.167,"y":0.14},"t":15,"s":[77.722,72.163,0],"to":[0,-0.176,0],"ti":[0,0.164,0]},{"i":{"x":0.833,"y":0.868},"o":{"x":0.167,"y":0.231},"t":16,"s":[77.722,71.535,0],"to":[0,-0.164,0],"ti":[0,0.042,0]},{"i":{"x":0.833,"y":0.396},"o":{"x":0.167,"y":0.227},"t":17,"s":[77.722,71.181,0],"to":[0,-0.042,0],"ti":[0,-0.232,0]},{"i":{"x":0.833,"y":0.774},"o":{"x":0.167,"y":0.097},"t":18,"s":[77.722,71.285,0],"to":[0,0.232,0],"ti":[0,-0.581,0]},{"i":{"x":0.833,"y":0.868},"o":{"x":0.167,"y":0.132},"t":19,"s":[77.722,72.57,0],"to":[0,0.581,0],"ti":[0,-0.579,0]},{"i":{"x":0.833,"y":0.744},"o":{"x":0.167,"y":0.227},"t":20,"s":[77.722,74.769,0],"to":[0,0.579,0],"ti":[0,-0.653,0]},{"i":{"x":0.833,"y":0.838},"o":{"x":0.167,"y":0.124},"t":21,"s":[77.722,76.045,0],"to":[0,0.653,0],"ti":[0,-0.859,0]},{"i":{"x":0.833,"y":0.879},"o":{"x":0.167,"y":0.171},"t":22,"s":[77.722,78.687,0],"to":[0,0.859,0],"ti":[0,-0.608,0]},{"i":{"x":0.833,"y":0.773},"o":{"x":0.167,"y":0.267},"t":23,"s":[77.722,81.197,0],"to":[0,0.608,0],"ti":[0,-0.517,0]},{"i":{"x":0.833,"y":0.853},"o":{"x":0.167,"y":0.132},"t":24,"s":[77.722,82.334,0],"to":[0,0.517,0],"ti":[0,-0.578,0]},{"i":{"x":0.833,"y":0.89},"o":{"x":0.167,"y":0.192},"t":25,"s":[77.722,84.298,0],"to":[0,0.578,0],"ti":[0,-0.332,0]},{"i":{"x":0.833,"y":0.878},"o":{"x":0.167,"y":0.341},"t":26,"s":[77.722,85.804,0],"to":[0,0.332,0],"ti":[0,-0.115,0]},{"i":{"x":0.833,"y":0.716},"o":{"x":0.167,"y":0.261},"t":27,"s":[77.722,86.29,0],"to":[0,0.115,0],"ti":[0,0.058,0]},{"i":{"x":0.833,"y":0.847},"o":{"x":0.167,"y":0.118},"t":28,"s":[77.722,86.491,0],"to":[0,-0.058,0],"ti":[0,0.168,0]},{"i":{"x":0.833,"y":0.712},"o":{"x":0.167,"y":0.183},"t":29,"s":[77.722,85.944,0],"to":[0,-0.168,0],"ti":[0,0.264,0]},{"i":{"x":0.833,"y":0.826},"o":{"x":0.167,"y":0.117},"t":30,"s":[77.722,85.485,0],"to":[0,-0.264,0],"ti":[0,0.392,0]},{"i":{"x":0.833,"y":0.876},"o":{"x":0.167,"y":0.16},"t":31,"s":[77.722,84.362,0],"to":[0,-0.392,0],"ti":[0,0.305,0]},{"i":{"x":0.833,"y":0.763},"o":{"x":0.167,"y":0.253},"t":32,"s":[77.722,83.135,0],"to":[0,-0.305,0],"ti":[0,0.285,0]},{"i":{"x":0.833,"y":0.847},"o":{"x":0.167,"y":0.128},"t":33,"s":[77.722,82.533,0],"to":[0,-0.285,0],"ti":[0,0.34,0]},{"i":{"x":0.833,"y":0.882},"o":{"x":0.167,"y":0.183},"t":34,"s":[77.722,81.423,0],"to":[0,-0.34,0],"ti":[0,0.219,0]},{"i":{"x":0.833,"y":0.785},"o":{"x":0.167,"y":0.284},"t":35,"s":[77.722,80.493,0],"to":[0,-0.219,0],"ti":[0,0.166,0]},{"i":{"x":0.833,"y":0.861},"o":{"x":0.167,"y":0.136},"t":36,"s":[77.722,80.107,0],"to":[0,-0.166,0],"ti":[0,0.169,0]},{"i":{"x":0.833,"y":0.889},"o":{"x":0.167,"y":0.209},"t":37,"s":[77.722,79.497,0],"to":[0,-0.169,0],"ti":[0,0.09,0]},{"i":{"x":0.833,"y":0.821},"o":{"x":0.167,"y":0.334},"t":38,"s":[77.722,79.09,0],"to":[0,-0.09,0],"ti":[0,0.048,0]},{"i":{"x":0.833,"y":0.899},"o":{"x":0.167,"y":0.156},"t":39,"s":[77.722,78.956,0],"to":[0,-0.048,0],"ti":[0,0.031,0]},{"i":{"x":0.833,"y":0.868},"o":{"x":0.167,"y":0.478},"t":40,"s":[77.722,78.801,0],"to":[0,-0.031,0],"ti":[0,0.002,0]},{"i":{"x":0.833,"y":0.553},"o":{"x":0.167,"y":0.229},"t":41,"s":[77.722,78.768,0],"to":[0,-0.002,0],"ti":[0,-0.017,0]},{"i":{"x":0.833,"y":0.795},"o":{"x":0.167,"y":0.103},"t":42,"s":[77.722,78.786,0],"to":[0,0.017,0],"ti":[0,-0.034,0]},{"i":{"x":0.833,"y":0.871},"o":{"x":0.167,"y":0.141},"t":43,"s":[77.722,78.868,0],"to":[0,0.034,0],"ti":[0,-0.031,0]},{"i":{"x":0.833,"y":0.749},"o":{"x":0.167,"y":0.234},"t":44,"s":[77.722,78.987,0],"to":[0,0.031,0],"ti":[0,-0.033,0]},{"i":{"x":0.833,"y":0.84},"o":{"x":0.167,"y":0.125},"t":45,"s":[77.722,79.053,0],"to":[0,0.033,0],"ti":[0,-0.042,0]},{"i":{"x":0.833,"y":0.88},"o":{"x":0.167,"y":0.174},"t":46,"s":[77.722,79.186,0],"to":[0,0.042,0],"ti":[0,-0.029,0]},{"i":{"x":0.833,"y":0.775},"o":{"x":0.167,"y":0.271},"t":47,"s":[77.722,79.308,0],"to":[0,0.029,0],"ti":[0,-0.024,0]},{"i":{"x":0.833,"y":0.854},"o":{"x":0.167,"y":0.133},"t":48,"s":[77.722,79.362,0],"to":[0,0.024,0],"ti":[0,-0.027,0]},{"i":{"x":0.833,"y":0.885},"o":{"x":0.167,"y":0.195},"t":49,"s":[77.722,79.454,0],"to":[0,0.027,0],"ti":[0,-0.016,0]},{"i":{"x":0.833,"y":0.8},"o":{"x":0.167,"y":0.305},"t":50,"s":[77.722,79.523,0],"to":[0,0.016,0],"ti":[0,-0.01,0]},{"i":{"x":0.833,"y":0.874},"o":{"x":0.167,"y":0.143},"t":51,"s":[77.722,79.549,0],"to":[0,0.01,0],"ti":[0,-0.009,0]},{"i":{"x":0.833,"y":0.899},"o":{"x":0.167,"y":0.247},"t":52,"s":[77.722,79.585,0],"to":[0,0.009,0],"ti":[0,-0.004,0]},{"i":{"x":0.833,"y":0.884},"o":{"x":0.167,"y":0.484},"t":53,"s":[77.722,79.603,0],"to":[0,0.004,0],"ti":[0,-0.001,0]},{"i":{"x":0.833,"y":0.492},"o":{"x":0.167,"y":0.278},"t":54,"s":[77.722,79.607,0],"to":[0,0.001,0],"ti":[0,0.001,0]},{"i":{"x":0.833,"y":0.857},"o":{"x":0.167,"y":0.099},"t":55,"s":[77.722,79.607,0],"to":[0,-0.001,0],"ti":[0,0.002,0]},{"i":{"x":0.833,"y":0.724},"o":{"x":0.167,"y":0.199},"t":56,"s":[77.722,79.598,0],"to":[0,-0.002,0],"ti":[0,0.003,0]},{"i":{"x":0.833,"y":0.83},"o":{"x":0.167,"y":0.119},"t":57,"s":[77.722,79.592,0],"to":[0,-0.003,0],"ti":[0,0.005,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.163},"t":58,"s":[77.722,79.579,0],"to":[0,-0.005,0],"ti":[0,0.002,0]},{"t":59,"s":[77.722,79.564,0]}],"ix":2,"l":2},"a":{"a":0,"k":[-0.918,5,0],"ix":1,"l":2},"s":{"a":0,"k":[30.469,30.469,100],"ix":6,"l":2}},"ao":0,"ip":0,"op":60,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"Bluetooth Outlines","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":-0.058,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.2,"y":1},"o":{"x":0.167,"y":0.167},"t":3,"s":[-0.011,-0.207,0],"to":[-9.333,0,0],"ti":[9.333,0,0]},{"t":20,"s":[-56.011,-0.207,0]}],"ix":2,"l":2},"a":{"a":0,"k":[78,78,0],"ix":1,"l":2},"s":{"a":1,"k":[{"i":{"x":[0.2,0.2,0.2],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0]},"t":3,"s":[328.205,328.205,100]},{"t":20,"s":[0,0,100]}],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-17.75,20.5],[17.75,-15.5],[2.25,-31.5],[2.25,31.5],[17.75,15.5],[-17.75,-20.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.2],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":3,"s":[0]},{"t":20,"s":[50]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.2],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":3,"s":[100]},{"t":20,"s":[50]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":7,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[69.25,78],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"wave2","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":-0.058,"ix":10},"p":{"a":0,"k":[-0.011,-0.207,0],"ix":2,"l":2},"a":{"a":0,"k":[78,78,0],"ix":1,"l":2},"s":{"a":0,"k":[328.205,328.205,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.2,"y":1},"o":{"x":0.167,"y":0.167},"t":3,"s":[{"i":[[0,0],[0,-6.48],[4.153,-6.175]],"o":[[4.153,5.794],[0,6.48],[0,0]],"v":[[-3.115,-18.698],[3.115,-0.286],[-3.115,18.698]],"c":false}]},{"t":20,"s":[{"i":[[0,0],[0,-12.307],[12.759,-6.613]],"o":[[12.759,6.613],[0,12.307],[0,0]],"v":[[-19.177,-28.707],[-0.039,-0.329],[-19.177,28.05]],"c":false}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":7.5,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[108.267,78.286],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":4,"ty":4,"nm":"wave1","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":-0.058,"ix":10},"p":{"a":0,"k":[-0.011,-0.207,0],"ix":2,"l":2},"a":{"a":0,"k":[78,78,0],"ix":1,"l":2},"s":{"a":0,"k":[328.205,328.205,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.2,"y":1},"o":{"x":0.167,"y":0.167},"t":3,"s":[{"i":[[1.297,-1.308],[0,0],[-1.076,-1.085],[0,0],[-0.856,1.669],[-0.026,1.81],[0.896,1.763]],"o":[[0,0],[-1.076,1.085],[0,0],[1.297,1.308],[0.843,-1.646],[0.027,-1.924],[-0.856,-1.586]],"v":[[16.736,-6.202],[12.679,-2.112],[12.679,1.811],[16.736,5.901],[21.124,5.261],[22.426,0.002],[21.124,-5.617]],"c":true}]},{"t":20,"s":[{"i":[[0.5,0],[0,0],[0,0],[0,0],[-2.401,2.725],[0,5.188],[2.401,2.725]],"o":[[0,0],[0,0],[0,0],[0.5,0],[2.401,-2.725],[0,-5.188],[-2.401,-2.725]],"v":[[7,-16.704],[7,-8.362],[7,3.554],[7,15.471],[12.427,11.318],[17.103,-0.617],[12.427,-12.552]],"c":true}]}],"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"mm","mm":1,"nm":"Merge Paths 1","mn":"ADBE Vector Filter - Merge","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[78,78],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":5,"ty":4,"nm":"Speaker Base","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":-0.058,"ix":10},"p":{"a":0,"k":[-0.011,-0.207,0],"ix":2,"l":2},"a":{"a":0,"k":[78,78,0],"ix":1,"l":2},"s":{"a":1,"k":[{"i":{"x":[0.2,0.2,0.2],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0]},"t":0,"s":[0,0,100]},{"t":20,"s":[328.205,328.205,100]}],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,-2.198],[0,0],[-2.213,0],[0,0],[0,0],[0,3.557],[0,0],[2.534,-2.518],[0,0],[0,0]],"o":[[0,0],[0,2.198],[0,0],[0,0],[2.534,2.518],[0,0],[0,-3.557],[0,0],[0,0],[-2.213,0]],"v":[[-34,-8.59],[-34,7.397],[-29.977,11.394],[-19.908,11.394],[-6.672,24.544],[-1,22.914],[-1,-24.147],[-6.672,-25.777],[-19.908,-12.587],[-29.977,-12.587]],"c":true},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"mm","mm":1,"nm":"Merge Paths 1","mn":"ADBE Vector Filter - Merge","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[78,78],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 3","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60,"st":0,"ct":1,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/TMessagesProj/src/main/res/raw/call_accept.json b/TMessagesProj/src/main/res/raw/call_accept.json new file mode 100644 index 00000000000..6d377511334 --- /dev/null +++ b/TMessagesProj/src/main/res/raw/call_accept.json @@ -0,0 +1 @@ +{"v":"5.10.1","fr":60,"ip":0,"op":180,"w":156,"h":156,"nm":"call_accept","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":4,"nm":"Shape Layer 2","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.6],"y":[0]},"t":29,"s":[50]},{"t":45,"s":[0]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[78,78,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.6,0.6],"y":[1,1]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":16,"s":[32,32]},{"t":55,"s":[128,128]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[12.5]},{"t":25,"s":[0]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[12.5]},{"t":25,"s":[25]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.6],"y":[0]},"t":29,"s":[7]},{"t":45,"s":[0]}],"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":15,"op":195,"st":15,"ct":1,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"Shape Layer 1","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.6],"y":[0]},"t":14,"s":[50]},{"t":30,"s":[0]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[78,78,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.6,0.6],"y":[1,1]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":1,"s":[32,32]},{"t":40,"s":[128,128]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":1,"s":[12.5]},{"t":10,"s":[0]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":1,"s":[12.5]},{"t":10,"s":[25]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.6],"y":[0]},"t":14,"s":[7]},{"t":30,"s":[0]}],"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":180,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"Call Accept Outlines","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.696]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"i":{"x":[0.833],"y":[0.861]},"o":{"x":[0.167],"y":[0.115]},"t":1,"s":[0.556]},{"i":{"x":[0.833],"y":[0.726]},"o":{"x":[0.167],"y":[0.209]},"t":2,"s":[2.029]},{"i":{"x":[0.833],"y":[0.829]},"o":{"x":[0.167],"y":[0.12]},"t":3,"s":[3.008]},{"i":{"x":[0.833],"y":[0.849]},"o":{"x":[0.167],"y":[0.163]},"t":4,"s":[5.243]},{"i":{"x":[0.833],"y":[0.911]},"o":{"x":[0.167],"y":[0.187]},"t":5,"s":[7.585]},{"i":{"x":[0.833],"y":[1.753]},"o":{"x":[0.167],"y":[1.232]},"t":6,"s":[9.473]},{"i":{"x":[0.833],"y":[0.834]},"o":{"x":[0.167],"y":[0.075]},"t":7,"s":[9.609]},{"i":{"x":[0.833],"y":[0.691]},"o":{"x":[0.167],"y":[0.167]},"t":8,"s":[8.236]},{"i":{"x":[0.833],"y":[0.817]},"o":{"x":[0.167],"y":[0.114]},"t":9,"s":[6.866]},{"i":{"x":[0.833],"y":[0.879]},"o":{"x":[0.167],"y":[0.153]},"t":10,"s":[3.16]},{"i":{"x":[0.833],"y":[0.82]},"o":{"x":[0.167],"y":[0.267]},"t":11,"s":[-1.291]},{"i":{"x":[0.833],"y":[0.908]},"o":{"x":[0.167],"y":[0.155]},"t":12,"s":[-3.311]},{"i":{"x":[0.833],"y":[1.123]},"o":{"x":[0.167],"y":[0.93]},"t":13,"s":[-5.651]},{"i":{"x":[0.833],"y":[0.592]},"o":{"x":[0.167],"y":[0.05]},"t":14,"s":[-5.882]},{"i":{"x":[0.833],"y":[0.796]},"o":{"x":[0.167],"y":[0.105]},"t":15,"s":[-5.312]},{"i":{"x":[0.833],"y":[0.877]},"o":{"x":[0.167],"y":[0.141]},"t":16,"s":[-3.09]},{"i":{"x":[0.833],"y":[0.827]},"o":{"x":[0.167],"y":[0.259]},"t":17,"s":[0.138]},{"i":{"x":[0.833],"y":[0.921]},"o":{"x":[0.167],"y":[0.161]},"t":18,"s":[1.674]},{"i":{"x":[0.833],"y":[0.321]},"o":{"x":[0.167],"y":[-1.569]},"t":19,"s":[3.324]},{"i":{"x":[0.833],"y":[0.63]},"o":{"x":[0.167],"y":[0.095]},"t":20,"s":[3.241]},{"i":{"x":[0.833],"y":[0.802]},"o":{"x":[0.167],"y":[0.108]},"t":21,"s":[2.646]},{"i":{"x":[0.833],"y":[0.875]},"o":{"x":[0.167],"y":[0.144]},"t":22,"s":[0.599]},{"i":{"x":[0.833],"y":[0.793]},"o":{"x":[0.167],"y":[0.249]},"t":23,"s":[-2.217]},{"i":{"x":[0.833],"y":[0.873]},"o":{"x":[0.167],"y":[0.139]},"t":24,"s":[-3.63]},{"i":{"x":[0.833],"y":[0.902]},"o":{"x":[0.167],"y":[0.242]},"t":25,"s":[-5.732]},{"i":{"x":[0.833],"y":[1.024]},"o":{"x":[0.167],"y":[0.56]},"t":26,"s":[-6.835]},{"i":{"x":[0.833],"y":[0.603]},"o":{"x":[0.167],"y":[0.019]},"t":27,"s":[-7.028]},{"i":{"x":[0.833],"y":[0.858]},"o":{"x":[0.167],"y":[0.105]},"t":28,"s":[-6.78]},{"i":{"x":[0.833],"y":[0.907]},"o":{"x":[0.167],"y":[0.202]},"t":29,"s":[-5.846]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.167],"y":[0.176]},"t":30,"s":[-5.192]},{"t":45,"s":[0]}],"ix":10},"p":{"a":0,"k":[78,78,0],"ix":2,"l":2},"a":{"a":0,"k":[78,78,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[3.741,0.448],[0,0],[1.594,-1.594],[0.848,-0.848],[0.692,0.392],[4.998,9.038],[-0.735,0.735],[-2.213,2.213],[0.264,2.25],[0,0],[3.767,0.003],[0,0],[-0.265,-4.185],[-31.573,-1.991],[0.004,4.185],[0,0]],"o":[[0,0],[-2.25,-0.264],[-2.141,2.141],[-1.002,1.002],[-8.957,-5.066],[-0.311,-0.562],[0.884,-0.884],[1.594,-1.594],[0,0],[-0.448,-3.741],[0,0],[-4.185,-0.004],[1.991,31.573],[4.185,0.265],[0,0],[0.023,-3.74]],"v":[[26.828,12.017],[17.437,10.936],[11.344,13.049],[6.86,17.534],[3.285,17.873],[-18.107,-3.69],[-17.676,-6.73],[-13.03,-11.377],[-10.917,-17.47],[-11.998,-26.808],[-19.354,-33.38],[-25.736,-33.386],[-33.131,-25.729],[25.723,33.125],[33.38,25.729],[33.374,19.347]],"c":true},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":1,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"mm","mm":4,"nm":"Merge Paths 1","mn":"ADBE Vector Filter - Merge","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[79.13,79.376],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":180,"st":0,"ct":1,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/TMessagesProj/src/main/res/raw/call_mute.json b/TMessagesProj/src/main/res/raw/call_mute.json new file mode 100644 index 00000000000..2154708412b --- /dev/null +++ b/TMessagesProj/src/main/res/raw/call_mute.json @@ -0,0 +1 @@ +{"v":"5.10.1","fr":60,"ip":0,"op":60,"w":156,"h":156,"nm":"call_mute","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":3,"nm":"Top 2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[1.004]},"o":{"x":[0.167],"y":[0]},"t":0,"s":[0]},{"i":{"x":[0.833],"y":[0.689]},"o":{"x":[0.167],"y":[0.083]},"t":1,"s":[0]},{"i":{"x":[0.833],"y":[0.638]},"o":{"x":[0.167],"y":[0.114]},"t":2,"s":[-0.042]},{"i":{"x":[0.833],"y":[0.808]},"o":{"x":[0.167],"y":[0.108]},"t":3,"s":[-0.157]},{"i":{"x":[0.833],"y":[0.829]},"o":{"x":[0.167],"y":[0.148]},"t":4,"s":[-0.54]},{"i":{"x":[0.833],"y":[0.876]},"o":{"x":[0.167],"y":[0.162]},"t":5,"s":[-1.038]},{"i":{"x":[0.833],"y":[0.802]},"o":{"x":[0.167],"y":[0.256]},"t":6,"s":[-1.564]},{"i":{"x":[0.833],"y":[0.942]},"o":{"x":[0.167],"y":[0.144]},"t":7,"s":[-1.817]},{"i":{"x":[0.833],"y":[0.368]},"o":{"x":[0.167],"y":[-0.195]},"t":8,"s":[-2.167]},{"i":{"x":[0.833],"y":[0.785]},"o":{"x":[0.167],"y":[0.096]},"t":9,"s":[-2.062]},{"i":{"x":[0.833],"y":[0.869]},"o":{"x":[0.167],"y":[0.136]},"t":10,"s":[-1.374]},{"i":{"x":[0.833],"y":[0.746]},"o":{"x":[0.167],"y":[0.23]},"t":11,"s":[-0.289]},{"i":{"x":[0.833],"y":[0.839]},"o":{"x":[0.167],"y":[0.124]},"t":12,"s":[0.327]},{"i":{"x":[0.833],"y":[0.879]},"o":{"x":[0.167],"y":[0.172]},"t":13,"s":[1.583]},{"i":{"x":[0.833],"y":[0.774]},"o":{"x":[0.167],"y":[0.269]},"t":14,"s":[2.759]},{"i":{"x":[0.833],"y":[0.854]},"o":{"x":[0.167],"y":[0.132]},"t":15,"s":[3.288]},{"i":{"x":[0.833],"y":[0.885]},"o":{"x":[0.167],"y":[0.193]},"t":16,"s":[4.193]},{"i":{"x":[0.833],"y":[0.834]},"o":{"x":[0.167],"y":[0.302]},"t":17,"s":[4.879]},{"i":{"x":[0.833],"y":[1.096]},"o":{"x":[0.167],"y":[0.167]},"t":18,"s":[5.14]},{"i":{"x":[0.833],"y":[0.84]},"o":{"x":[0.167],"y":[0.045]},"t":19,"s":[5.401]},{"i":{"x":[0.833],"y":[0.706]},"o":{"x":[0.167],"y":[0.173]},"t":20,"s":[4.839]},{"i":{"x":[0.833],"y":[0.824]},"o":{"x":[0.167],"y":[0.116]},"t":21,"s":[4.319]},{"i":{"x":[0.833],"y":[0.875]},"o":{"x":[0.167],"y":[0.158]},"t":22,"s":[3.002]},{"i":{"x":[0.833],"y":[0.762]},"o":{"x":[0.167],"y":[0.251]},"t":23,"s":[1.531]},{"i":{"x":[0.833],"y":[0.846]},"o":{"x":[0.167],"y":[0.128]},"t":24,"s":[0.802]},{"i":{"x":[0.833],"y":[0.882]},"o":{"x":[0.167],"y":[0.182]},"t":25,"s":[-0.555]},{"i":{"x":[0.833],"y":[0.791]},"o":{"x":[0.167],"y":[0.282]},"t":26,"s":[-1.705]},{"i":{"x":[0.833],"y":[0.89]},"o":{"x":[0.167],"y":[0.139]},"t":27,"s":[-2.186]},{"i":{"x":[0.833],"y":[0.93]},"o":{"x":[0.167],"y":[0.342]},"t":28,"s":[-2.912]},{"i":{"x":[0.833],"y":[0.264]},"o":{"x":[0.167],"y":[-0.443]},"t":29,"s":[-3.145]},{"i":{"x":[0.833],"y":[0.781]},"o":{"x":[0.167],"y":[0.094]},"t":30,"s":[-3.108]},{"i":{"x":[0.833],"y":[0.869]},"o":{"x":[0.167],"y":[0.135]},"t":31,"s":[-2.818]},{"i":{"x":[0.833],"y":[0.746]},"o":{"x":[0.167],"y":[0.229]},"t":32,"s":[-2.348]},{"i":{"x":[0.833],"y":[0.838]},"o":{"x":[0.167],"y":[0.124]},"t":33,"s":[-2.08]},{"i":{"x":[0.833],"y":[0.879]},"o":{"x":[0.167],"y":[0.172]},"t":34,"s":[-1.528]},{"i":{"x":[0.833],"y":[0.774]},"o":{"x":[0.167],"y":[0.268]},"t":35,"s":[-1.009]},{"i":{"x":[0.833],"y":[0.853]},"o":{"x":[0.167],"y":[0.132]},"t":36,"s":[-0.775]},{"i":{"x":[0.833],"y":[0.885]},"o":{"x":[0.167],"y":[0.193]},"t":37,"s":[-0.372]},{"i":{"x":[0.833],"y":[0.797]},"o":{"x":[0.167],"y":[0.301]},"t":38,"s":[-0.066]},{"i":{"x":[0.833],"y":[0.871]},"o":{"x":[0.167],"y":[0.141]},"t":39,"s":[0.051]},{"i":{"x":[0.833],"y":[0.897]},"o":{"x":[0.167],"y":[0.237]},"t":40,"s":[0.219]},{"i":{"x":[0.833],"y":[0.89]},"o":{"x":[0.167],"y":[0.431]},"t":41,"s":[0.31]},{"i":{"x":[0.833],"y":[1.258]},"o":{"x":[0.167],"y":[0.345]},"t":42,"s":[0.332]},{"i":{"x":[0.833],"y":[0.85]},"o":{"x":[0.167],"y":[0.063]},"t":43,"s":[0.339]},{"i":{"x":[0.833],"y":[0.715]},"o":{"x":[0.167],"y":[0.187]},"t":44,"s":[0.31]},{"i":{"x":[0.833],"y":[0.827]},"o":{"x":[0.167],"y":[0.118]},"t":45,"s":[0.287]},{"i":{"x":[0.833],"y":[0.876]},"o":{"x":[0.167],"y":[0.16]},"t":46,"s":[0.232]},{"i":{"x":[0.833],"y":[0.764]},"o":{"x":[0.167],"y":[0.254]},"t":47,"s":[0.172]},{"i":{"x":[0.833],"y":[0.847]},"o":{"x":[0.167],"y":[0.129]},"t":48,"s":[0.143]},{"i":{"x":[0.833],"y":[0.882]},"o":{"x":[0.167],"y":[0.183]},"t":49,"s":[0.089]},{"i":{"x":[0.833],"y":[0.786]},"o":{"x":[0.167],"y":[0.285]},"t":50,"s":[0.045]},{"i":{"x":[0.833],"y":[0.862]},"o":{"x":[0.167],"y":[0.136]},"t":51,"s":[0.026]},{"i":{"x":[0.833],"y":[0.889]},"o":{"x":[0.167],"y":[0.21]},"t":52,"s":[-0.003]},{"i":{"x":[0.833],"y":[0.823]},"o":{"x":[0.167],"y":[0.337]},"t":53,"s":[-0.022]},{"i":{"x":[0.833],"y":[0.902]},"o":{"x":[0.167],"y":[0.157]},"t":54,"s":[-0.029]},{"i":{"x":[0.833],"y":[0.981]},"o":{"x":[0.167],"y":[0.549]},"t":55,"s":[-0.036]},{"i":{"x":[0.833],"y":[0.564]},"o":{"x":[0.167],"y":[-0.024]},"t":56,"s":[-0.037]},{"i":{"x":[0.833],"y":[0.799]},"o":{"x":[0.167],"y":[0.103]},"t":57,"s":[-0.036]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.142]},"t":58,"s":[-0.032]},{"t":59,"s":[-0.026]}],"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.713},"o":{"x":0.167,"y":0.167},"t":0,"s":[77.722,79.526,0],"to":[0,-0.053,0],"ti":[0,0.183,0]},{"i":{"x":0.833,"y":0.864},"o":{"x":0.167,"y":0.117},"t":1,"s":[77.722,79.208,0],"to":[0,-0.183,0],"ti":[0,0.211,0]},{"i":{"x":0.833,"y":0.736},"o":{"x":0.167,"y":0.216},"t":2,"s":[77.722,78.429,0],"to":[0,-0.211,0],"ti":[0,0.258,0]},{"i":{"x":0.833,"y":0.834},"o":{"x":0.167,"y":0.122},"t":3,"s":[77.722,77.94,0],"to":[0,-0.258,0],"ti":[0,0.351,0]},{"i":{"x":0.833,"y":0.856},"o":{"x":0.167,"y":0.168},"t":4,"s":[77.722,76.881,0],"to":[0,-0.351,0],"ti":[0,0.302,0]},{"i":{"x":0.833,"y":0.894},"o":{"x":0.167,"y":0.197},"t":5,"s":[77.722,75.834,0],"to":[0,-0.302,0],"ti":[0,0.118,0]},{"i":{"x":0.833,"y":0.567},"o":{"x":0.167,"y":0.388},"t":6,"s":[77.722,75.068,0],"to":[0,-0.118,0],"ti":[0,-0.156,0]},{"i":{"x":0.833,"y":0.849},"o":{"x":0.167,"y":0.103},"t":7,"s":[77.722,75.125,0],"to":[0,0.156,0],"ti":[0,-0.265,0]},{"i":{"x":0.833,"y":0.715},"o":{"x":0.167,"y":0.187},"t":8,"s":[77.722,76.004,0],"to":[0,0.265,0],"ti":[0,-0.404,0]},{"i":{"x":0.833,"y":0.827},"o":{"x":0.167,"y":0.118},"t":9,"s":[77.722,76.712,0],"to":[0,0.404,0],"ti":[0,-0.595,0]},{"i":{"x":0.833,"y":0.876},"o":{"x":0.167,"y":0.16},"t":10,"s":[77.722,78.426,0],"to":[0,0.595,0],"ti":[0,-0.46,0]},{"i":{"x":0.833,"y":0.763},"o":{"x":0.167,"y":0.254},"t":11,"s":[77.722,80.28,0],"to":[0,0.46,0],"ti":[0,-0.428,0]},{"i":{"x":0.833,"y":0.847},"o":{"x":0.167,"y":0.129},"t":12,"s":[77.722,81.184,0],"to":[0,0.428,0],"ti":[0,-0.508,0]},{"i":{"x":0.833,"y":0.882},"o":{"x":0.167,"y":0.183},"t":13,"s":[77.722,82.846,0],"to":[0,0.508,0],"ti":[0,-0.327,0]},{"i":{"x":0.833,"y":0.785},"o":{"x":0.167,"y":0.285},"t":14,"s":[77.722,84.232,0],"to":[0,0.327,0],"ti":[0,-0.246,0]},{"i":{"x":0.833,"y":0.875},"o":{"x":0.167,"y":0.136},"t":15,"s":[77.722,84.806,0],"to":[0,0.246,0],"ti":[0,-0.226,0]},{"i":{"x":0.833,"y":0.873},"o":{"x":0.167,"y":0.25},"t":16,"s":[77.722,85.708,0],"to":[0,0.226,0],"ti":[0,-0.042,0]},{"i":{"x":0.833,"y":0.524},"o":{"x":0.167,"y":0.244},"t":17,"s":[77.722,86.16,0],"to":[0,0.042,0],"ti":[0,0.217,0]},{"i":{"x":0.833,"y":0.79},"o":{"x":0.167,"y":0.101},"t":18,"s":[77.722,85.96,0],"to":[0,-0.217,0],"ti":[0,0.464,0]},{"i":{"x":0.833,"y":0.87},"o":{"x":0.167,"y":0.138},"t":19,"s":[77.722,84.857,0],"to":[0,-0.464,0],"ti":[0,0.437,0]},{"i":{"x":0.833,"y":0.748},"o":{"x":0.167,"y":0.232},"t":20,"s":[77.722,83.177,0],"to":[0,-0.437,0],"ti":[0,0.475,0]},{"i":{"x":0.833,"y":0.839},"o":{"x":0.167,"y":0.124},"t":21,"s":[77.722,82.236,0],"to":[0,-0.475,0],"ti":[0,0.614,0]},{"i":{"x":0.833,"y":0.879},"o":{"x":0.167,"y":0.173},"t":22,"s":[77.722,80.328,0],"to":[0,-0.614,0],"ti":[0,0.428,0]},{"i":{"x":0.833,"y":0.775},"o":{"x":0.167,"y":0.27},"t":23,"s":[77.722,78.552,0],"to":[0,-0.428,0],"ti":[0,0.358,0]},{"i":{"x":0.833,"y":0.858},"o":{"x":0.167,"y":0.132},"t":24,"s":[77.722,77.757,0],"to":[0,-0.358,0],"ti":[0,0.386,0]},{"i":{"x":0.833,"y":0.897},"o":{"x":0.167,"y":0.201},"t":25,"s":[77.722,76.403,0],"to":[0,-0.386,0],"ti":[0,0.197,0]},{"i":{"x":0.833,"y":0.886},"o":{"x":0.167,"y":0.441},"t":26,"s":[77.722,75.443,0],"to":[0,-0.197,0],"ti":[0,0.038,0]},{"i":{"x":0.833,"y":0.495},"o":{"x":0.167,"y":0.307},"t":27,"s":[77.722,75.22,0],"to":[0,-0.038,0],"ti":[0,-0.07,0]},{"i":{"x":0.833,"y":0.855},"o":{"x":0.167,"y":0.1},"t":28,"s":[77.722,75.218,0],"to":[0,0.07,0],"ti":[0,-0.122,0]},{"i":{"x":0.833,"y":0.722},"o":{"x":0.167,"y":0.196},"t":29,"s":[77.722,75.639,0],"to":[0,0.122,0],"ti":[0,-0.173,0]},{"i":{"x":0.833,"y":0.829},"o":{"x":0.167,"y":0.119},"t":30,"s":[77.722,75.95,0],"to":[0,0.173,0],"ti":[0,-0.249,0]},{"i":{"x":0.833,"y":0.877},"o":{"x":0.167,"y":0.162},"t":31,"s":[77.722,76.677,0],"to":[0,0.249,0],"ti":[0,-0.189,0]},{"i":{"x":0.833,"y":0.765},"o":{"x":0.167,"y":0.256},"t":32,"s":[77.722,77.443,0],"to":[0,0.189,0],"ti":[0,-0.173,0]},{"i":{"x":0.833,"y":0.848},"o":{"x":0.167,"y":0.129},"t":33,"s":[77.722,77.812,0],"to":[0,0.173,0],"ti":[0,-0.204,0]},{"i":{"x":0.833,"y":0.883},"o":{"x":0.167,"y":0.185},"t":34,"s":[77.722,78.483,0],"to":[0,0.204,0],"ti":[0,-0.13,0]},{"i":{"x":0.833,"y":0.787},"o":{"x":0.167,"y":0.287},"t":35,"s":[77.722,79.035,0],"to":[0,0.13,0],"ti":[0,-0.096,0]},{"i":{"x":0.833,"y":0.863},"o":{"x":0.167,"y":0.137},"t":36,"s":[77.722,79.261,0],"to":[0,0.096,0],"ti":[0,-0.096,0]},{"i":{"x":0.833,"y":0.89},"o":{"x":0.167,"y":0.212},"t":37,"s":[77.722,79.612,0],"to":[0,0.096,0],"ti":[0,-0.05,0]},{"i":{"x":0.833,"y":0.828},"o":{"x":0.167,"y":0.344},"t":38,"s":[77.722,79.839,0],"to":[0,0.05,0],"ti":[0,-0.025,0]},{"i":{"x":0.833,"y":0.904},"o":{"x":0.167,"y":0.162},"t":39,"s":[77.722,79.912,0],"to":[0,0.025,0],"ti":[0,-0.014,0]},{"i":{"x":0.833,"y":0.807},"o":{"x":0.167,"y":0.617},"t":40,"s":[77.722,79.989,0],"to":[0,0.014,0],"ti":[0,0.002,0]},{"i":{"x":0.833,"y":0.617},"o":{"x":0.167,"y":0.143},"t":41,"s":[77.722,79.996,0],"to":[0,-0.002,0],"ti":[0,0.012,0]},{"i":{"x":0.833,"y":0.805},"o":{"x":0.167,"y":0.107},"t":42,"s":[77.722,79.979,0],"to":[0,-0.012,0],"ti":[0,0.023,0]},{"i":{"x":0.833,"y":0.872},"o":{"x":0.167,"y":0.146},"t":43,"s":[77.722,79.921,0],"to":[0,-0.023,0],"ti":[0,0.02,0]},{"i":{"x":0.833,"y":0.753},"o":{"x":0.167,"y":0.239},"t":44,"s":[77.722,79.844,0],"to":[0,-0.02,0],"ti":[0,0.021,0]},{"i":{"x":0.833,"y":0.841},"o":{"x":0.167,"y":0.126},"t":45,"s":[77.722,79.802,0],"to":[0,-0.021,0],"ti":[0,0.026,0]},{"i":{"x":0.833,"y":0.88},"o":{"x":0.167,"y":0.176},"t":46,"s":[77.722,79.72,0],"to":[0,-0.026,0],"ti":[0,0.018,0]},{"i":{"x":0.833,"y":0.777},"o":{"x":0.167,"y":0.273},"t":47,"s":[77.722,79.646,0],"to":[0,-0.018,0],"ti":[0,0.014,0]},{"i":{"x":0.833,"y":0.856},"o":{"x":0.167,"y":0.133},"t":48,"s":[77.722,79.614,0],"to":[0,-0.014,0],"ti":[0,0.016,0]},{"i":{"x":0.833,"y":0.886},"o":{"x":0.167,"y":0.197},"t":49,"s":[77.722,79.56,0],"to":[0,-0.016,0],"ti":[0,0.009,0]},{"i":{"x":0.833,"y":0.803},"o":{"x":0.167,"y":0.31},"t":50,"s":[77.722,79.52,0],"to":[0,-0.009,0],"ti":[0,0.006,0]},{"i":{"x":0.833,"y":0.877},"o":{"x":0.167,"y":0.145},"t":51,"s":[77.722,79.505,0],"to":[0,-0.006,0],"ti":[0,0.005,0]},{"i":{"x":0.833,"y":0.903},"o":{"x":0.167,"y":0.26},"t":52,"s":[77.722,79.485,0],"to":[0,-0.005,0],"ti":[0,0.002,0]},{"i":{"x":0.833,"y":0.835},"o":{"x":0.167,"y":0.56},"t":53,"s":[77.722,79.476,0],"to":[0,-0.002,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.613},"o":{"x":0.167,"y":0.164},"t":54,"s":[77.722,79.475,0],"to":[0,0,0],"ti":[0,-0.001,0]},{"i":{"x":0.833,"y":0.861},"o":{"x":0.167,"y":0.106},"t":55,"s":[77.722,79.476,0],"to":[0,0.001,0],"ti":[0,-0.002,0]},{"i":{"x":0.833,"y":0.731},"o":{"x":0.167,"y":0.209},"t":56,"s":[77.722,79.482,0],"to":[0,0.002,0],"ti":[0,-0.002,0]},{"i":{"x":0.833,"y":0.832},"o":{"x":0.167,"y":0.121},"t":57,"s":[77.722,79.486,0],"to":[0,0.002,0],"ti":[0,-0.003,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.166},"t":58,"s":[77.722,79.495,0],"to":[0,0.003,0],"ti":[0,-0.001,0]},{"t":59,"s":[77.722,79.504,0]}],"ix":2,"l":2},"a":{"a":0,"k":[-0.918,5,0],"ix":1,"l":2},"s":{"a":0,"k":[30.469,30.469,100],"ix":6,"l":2}},"ao":0,"ip":0,"op":60,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"Call Unmute Outlines","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[-0.006,-0.008,0],"ix":2,"l":2},"a":{"a":0,"k":[78,78,0],"ix":1,"l":2},"s":{"a":0,"k":[328.205,328.205,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-28.5,-28.5],[28.5,28.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":3,"s":[0]},{"t":17,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":7,"ix":5},"lc":2,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[78.5,79.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 2","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"Call Mute Outlines","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[-0.006,-0.008,0],"ix":2,"l":2},"a":{"a":0,"k":[78,78,0],"ix":1,"l":2},"s":{"a":0,"k":[328.205,328.205,100],"ix":6,"l":2}},"ao":0,"hasMask":true,"masksProperties":[{"inv":false,"mode":"a","pt":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":3,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[-2.745,2.55],[0,0],[2.734,2.734],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[2.671,2.672],[0,0],[2.734,-2.734],[0,0],[0,0],[0,0]],"v":[[156.5,0.5],[156.5,156.5],[0.5,156.5],[0.5,21.5],[10.5,21.398],[47.55,58.45],[57.261,58.632],[57.45,58.45],[57.45,48.55],[10.5,1.601],[10.5,0.5]],"c":true}]},{"t":17,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[-2.745,2.55],[0,0],[2.734,2.734],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[2.671,2.672],[0,0],[2.734,-2.734],[0,0],[0,0],[0,0]],"v":[[156.5,0.5],[156.5,156.5],[0.5,156.5],[0.5,21.5],[10.5,21.398],[102.55,113.45],[112.261,113.632],[112.45,113.45],[112.45,103.55],[10.5,1.601],[10.5,0.5]],"c":true}]}],"ix":1},"o":{"a":0,"k":100,"ix":3},"x":{"a":0,"k":0,"ix":4},"nm":"Mask 1"}],"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-6.817,0],[0,-6.831],[0,0],[6.817,0],[0,6.831],[0,0]],"o":[[6.817,0],[0,0],[0,6.831],[-6.817,0],[0,0],[0,-6.831]],"v":[[0.007,-33],[12.349,-20.632],[12.349,1.047],[0.007,13.416],[-12.336,1.047],[-12.336,-20.632]],"c":true},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[-0.275,-1.973],[-9.95,0],[-1.477,10.291],[-1.968,-0.284],[0.283,-1.972],[11.947,-1.642],[0,0],[1.988,0],[0,1.992],[0,0],[1.757,12.587],[-1.969,0.276]],"o":[[1.442,10.33],[9.92,0],[0.283,-1.972],[1.968,0.284],[-1.799,12.54],[0,0],[0,1.992],[-1.988,0],[0,0],[-11.981,-1.651],[-0.275,-1.973],[1.969,-0.276]],"v":[[-19.835,3.117],[0.007,21.154],[19.836,3.2],[23.912,0.143],[26.963,4.227],[3.603,28.124],[3.603,35.393],[0.003,39],[-3.597,35.393],[-3.597,28.123],[-26.965,4.116],[-23.898,0.044]],"c":true},"ix":2},"nm":"Path 3","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"mm","mm":1,"nm":"Merge Paths 1","mn":"ADBE Vector Filter - Merge","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[78,78],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":4,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60,"st":0,"ct":1,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/TMessagesProj/src/main/res/raw/call_unmute.json b/TMessagesProj/src/main/res/raw/call_unmute.json new file mode 100644 index 00000000000..fee66f9404f --- /dev/null +++ b/TMessagesProj/src/main/res/raw/call_unmute.json @@ -0,0 +1 @@ +{"v":"5.10.1","fr":60,"ip":0,"op":60,"w":156,"h":156,"nm":"call_unmute","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":3,"nm":"Top 3","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.35]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"i":{"x":[0.833],"y":[0.855]},"o":{"x":[0.167],"y":[0.096]},"t":1,"s":[0.042]},{"i":{"x":[0.833],"y":[0.721]},"o":{"x":[0.167],"y":[0.195]},"t":2,"s":[0.328]},{"i":{"x":[0.833],"y":[0.829]},"o":{"x":[0.167],"y":[0.119]},"t":3,"s":[0.54]},{"i":{"x":[0.833],"y":[0.839]},"o":{"x":[0.167],"y":[0.162]},"t":4,"s":[1.038]},{"i":{"x":[0.833],"y":[0.898]},"o":{"x":[0.167],"y":[0.172]},"t":5,"s":[1.564]},{"i":{"x":[0.833],"y":[1.207]},"o":{"x":[0.167],"y":[0.458]},"t":6,"s":[2.057]},{"i":{"x":[0.833],"y":[0.827]},"o":{"x":[0.167],"y":[0.059]},"t":7,"s":[2.167]},{"i":{"x":[0.833],"y":[0.697]},"o":{"x":[0.167],"y":[0.161]},"t":8,"s":[1.785]},{"i":{"x":[0.833],"y":[0.821]},"o":{"x":[0.167],"y":[0.115]},"t":9,"s":[1.374]},{"i":{"x":[0.833],"y":[0.875]},"o":{"x":[0.167],"y":[0.156]},"t":10,"s":[0.289]},{"i":{"x":[0.833],"y":[0.76]},"o":{"x":[0.167],"y":[0.249]},"t":11,"s":[-0.957]},{"i":{"x":[0.833],"y":[0.845]},"o":{"x":[0.167],"y":[0.128]},"t":12,"s":[-1.583]},{"i":{"x":[0.833],"y":[0.881]},"o":{"x":[0.167],"y":[0.181]},"t":13,"s":[-2.759]},{"i":{"x":[0.833],"y":[0.782]},"o":{"x":[0.167],"y":[0.281]},"t":14,"s":[-3.767]},{"i":{"x":[0.833],"y":[0.859]},"o":{"x":[0.167],"y":[0.135]},"t":15,"s":[-4.193]},{"i":{"x":[0.833],"y":[0.888]},"o":{"x":[0.167],"y":[0.205]},"t":16,"s":[-4.879]},{"i":{"x":[0.833],"y":[0.814]},"o":{"x":[0.167],"y":[0.325]},"t":17,"s":[-5.351]},{"i":{"x":[0.833],"y":[0.937]},"o":{"x":[0.167],"y":[0.151]},"t":18,"s":[-5.513]},{"i":{"x":[0.833],"y":[0.375]},"o":{"x":[0.167],"y":[-0.262]},"t":19,"s":[-5.713]},{"i":{"x":[0.833],"y":[0.626]},"o":{"x":[0.167],"y":[0.096]},"t":20,"s":[-5.665]},{"i":{"x":[0.833],"y":[0.806]},"o":{"x":[0.167],"y":[0.107]},"t":21,"s":[-5.352]},{"i":{"x":[0.833],"y":[0.872]},"o":{"x":[0.167],"y":[0.146]},"t":22,"s":[-4.261]},{"i":{"x":[0.833],"y":[0.753]},"o":{"x":[0.167],"y":[0.24]},"t":23,"s":[-2.819]},{"i":{"x":[0.833],"y":[0.842]},"o":{"x":[0.167],"y":[0.126]},"t":24,"s":[-2.05]},{"i":{"x":[0.833],"y":[0.88]},"o":{"x":[0.167],"y":[0.176]},"t":25,"s":[-0.539]},{"i":{"x":[0.833],"y":[0.778]},"o":{"x":[0.167],"y":[0.274]},"t":26,"s":[0.821]},{"i":{"x":[0.833],"y":[0.859]},"o":{"x":[0.167],"y":[0.133]},"t":27,"s":[1.416]},{"i":{"x":[0.833],"y":[0.898]},"o":{"x":[0.167],"y":[0.205]},"t":28,"s":[2.408]},{"i":{"x":[0.833],"y":[0.93]},"o":{"x":[0.167],"y":[0.46]},"t":29,"s":[3.089]},{"i":{"x":[0.833],"y":[-0.224]},"o":{"x":[0.167],"y":[-0.436]},"t":30,"s":[3.24]},{"i":{"x":[0.833],"y":[0.857]},"o":{"x":[0.167],"y":[0.089]},"t":31,"s":[3.216]},{"i":{"x":[0.833],"y":[0.724]},"o":{"x":[0.167],"y":[0.199]},"t":32,"s":[2.885]},{"i":{"x":[0.833],"y":[0.83]},"o":{"x":[0.167],"y":[0.119]},"t":33,"s":[2.647]},{"i":{"x":[0.833],"y":[0.877]},"o":{"x":[0.167],"y":[0.163]},"t":34,"s":[2.097]},{"i":{"x":[0.833],"y":[0.766]},"o":{"x":[0.167],"y":[0.257]},"t":35,"s":[1.524]},{"i":{"x":[0.833],"y":[0.848]},"o":{"x":[0.167],"y":[0.129]},"t":36,"s":[1.249]},{"i":{"x":[0.833],"y":[0.883]},"o":{"x":[0.167],"y":[0.185]},"t":37,"s":[0.751]},{"i":{"x":[0.833],"y":[0.788]},"o":{"x":[0.167],"y":[0.288]},"t":38,"s":[0.344]},{"i":{"x":[0.833],"y":[0.863]},"o":{"x":[0.167],"y":[0.137]},"t":39,"s":[0.178]},{"i":{"x":[0.833],"y":[0.89]},"o":{"x":[0.167],"y":[0.213]},"t":40,"s":[-0.079]},{"i":{"x":[0.833],"y":[0.83]},"o":{"x":[0.167],"y":[0.347]},"t":41,"s":[-0.243]},{"i":{"x":[0.833],"y":[0.914]},"o":{"x":[0.167],"y":[0.164]},"t":42,"s":[-0.295]},{"i":{"x":[0.833],"y":[1.491]},"o":{"x":[0.167],"y":[2.434]},"t":43,"s":[-0.349]},{"i":{"x":[0.833],"y":[0.632]},"o":{"x":[0.167],"y":[0.071]},"t":44,"s":[-0.351]},{"i":{"x":[0.833],"y":[0.808]},"o":{"x":[0.167],"y":[0.108]},"t":45,"s":[-0.338]},{"i":{"x":[0.833],"y":[0.872]},"o":{"x":[0.167],"y":[0.147]},"t":46,"s":[-0.293]},{"i":{"x":[0.833],"y":[0.753]},"o":{"x":[0.167],"y":[0.24]},"t":47,"s":[-0.234]},{"i":{"x":[0.833],"y":[0.842]},"o":{"x":[0.167],"y":[0.126]},"t":48,"s":[-0.203]},{"i":{"x":[0.833],"y":[0.88]},"o":{"x":[0.167],"y":[0.176]},"t":49,"s":[-0.142]},{"i":{"x":[0.833],"y":[0.778]},"o":{"x":[0.167],"y":[0.274]},"t":50,"s":[-0.087]},{"i":{"x":[0.833],"y":[0.856]},"o":{"x":[0.167],"y":[0.133]},"t":51,"s":[-0.063]},{"i":{"x":[0.833],"y":[0.886]},"o":{"x":[0.167],"y":[0.198]},"t":52,"s":[-0.023]},{"i":{"x":[0.833],"y":[0.804]},"o":{"x":[0.167],"y":[0.311]},"t":53,"s":[0.006]},{"i":{"x":[0.833],"y":[0.878]},"o":{"x":[0.167],"y":[0.145]},"t":54,"s":[0.017]},{"i":{"x":[0.833],"y":[0.904]},"o":{"x":[0.167],"y":[0.265]},"t":55,"s":[0.031]},{"i":{"x":[0.833],"y":[1.045]},"o":{"x":[0.167],"y":[0.647]},"t":56,"s":[0.038]},{"i":{"x":[0.833],"y":[0.658]},"o":{"x":[0.167],"y":[0.029]},"t":57,"s":[0.038]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.11]},"t":58,"s":[0.037]},{"t":59,"s":[0.032]}],"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.713},"o":{"x":0.167,"y":0.167},"t":0,"s":[77.722,79.526,0],"to":[0,0.053,0],"ti":[0,-0.183,0]},{"i":{"x":0.833,"y":0.864},"o":{"x":0.167,"y":0.117},"t":1,"s":[77.722,79.844,0],"to":[0,0.183,0],"ti":[0,-0.211,0]},{"i":{"x":0.833,"y":0.736},"o":{"x":0.167,"y":0.216},"t":2,"s":[77.722,80.623,0],"to":[0,0.211,0],"ti":[0,-0.258,0]},{"i":{"x":0.833,"y":0.849},"o":{"x":0.167,"y":0.122},"t":3,"s":[77.722,81.112,0],"to":[0,0.258,0],"ti":[0,-0.32,0]},{"i":{"x":0.833,"y":0.881},"o":{"x":0.167,"y":0.186},"t":4,"s":[77.722,82.17,0],"to":[0,0.32,0],"ti":[0,-0.087,0]},{"i":{"x":0.833,"y":0.796},"o":{"x":0.167,"y":0.276},"t":5,"s":[77.722,83.03,0],"to":[0,0.087,0],"ti":[0,0.147,0]},{"i":{"x":0.833,"y":0.679},"o":{"x":0.167,"y":0.141},"t":6,"s":[77.722,82.69,0],"to":[0,-0.147,0],"ti":[0,0.348,0]},{"i":{"x":0.833,"y":0.868},"o":{"x":0.167,"y":0.112},"t":7,"s":[77.722,82.15,0],"to":[0,-0.348,0],"ti":[0,0.409,0]},{"i":{"x":0.833,"y":0.743},"o":{"x":0.167,"y":0.225},"t":8,"s":[77.722,80.604,0],"to":[0,-0.409,0],"ti":[0,0.468,0]},{"i":{"x":0.833,"y":0.837},"o":{"x":0.167,"y":0.123},"t":9,"s":[77.722,79.694,0],"to":[0,-0.468,0],"ti":[0,0.62,0]},{"i":{"x":0.833,"y":0.879},"o":{"x":0.167,"y":0.17},"t":10,"s":[77.722,77.793,0],"to":[0,-0.62,0],"ti":[0,0.442,0]},{"i":{"x":0.833,"y":0.772},"o":{"x":0.167,"y":0.266},"t":11,"s":[77.722,75.972,0],"to":[0,-0.442,0],"ti":[0,0.378,0]},{"i":{"x":0.833,"y":0.852},"o":{"x":0.167,"y":0.131},"t":12,"s":[77.722,75.142,0],"to":[0,-0.378,0],"ti":[0,0.425,0]},{"i":{"x":0.833,"y":0.884},"o":{"x":0.167,"y":0.191},"t":13,"s":[77.722,73.703,0],"to":[0,-0.425,0],"ti":[0,0.257,0]},{"i":{"x":0.833,"y":0.795},"o":{"x":0.167,"y":0.298},"t":14,"s":[77.722,72.593,0],"to":[0,-0.257,0],"ti":[0,0.176,0]},{"i":{"x":0.833,"y":0.87},"o":{"x":0.167,"y":0.14},"t":15,"s":[77.722,72.163,0],"to":[0,-0.176,0],"ti":[0,0.164,0]},{"i":{"x":0.833,"y":0.868},"o":{"x":0.167,"y":0.231},"t":16,"s":[77.722,71.535,0],"to":[0,-0.164,0],"ti":[0,0.042,0]},{"i":{"x":0.833,"y":0.396},"o":{"x":0.167,"y":0.227},"t":17,"s":[77.722,71.181,0],"to":[0,-0.042,0],"ti":[0,-0.232,0]},{"i":{"x":0.833,"y":0.774},"o":{"x":0.167,"y":0.097},"t":18,"s":[77.722,71.285,0],"to":[0,0.232,0],"ti":[0,-0.581,0]},{"i":{"x":0.833,"y":0.868},"o":{"x":0.167,"y":0.132},"t":19,"s":[77.722,72.57,0],"to":[0,0.581,0],"ti":[0,-0.579,0]},{"i":{"x":0.833,"y":0.744},"o":{"x":0.167,"y":0.227},"t":20,"s":[77.722,74.769,0],"to":[0,0.579,0],"ti":[0,-0.653,0]},{"i":{"x":0.833,"y":0.838},"o":{"x":0.167,"y":0.124},"t":21,"s":[77.722,76.045,0],"to":[0,0.653,0],"ti":[0,-0.859,0]},{"i":{"x":0.833,"y":0.879},"o":{"x":0.167,"y":0.171},"t":22,"s":[77.722,78.687,0],"to":[0,0.859,0],"ti":[0,-0.608,0]},{"i":{"x":0.833,"y":0.773},"o":{"x":0.167,"y":0.267},"t":23,"s":[77.722,81.197,0],"to":[0,0.608,0],"ti":[0,-0.517,0]},{"i":{"x":0.833,"y":0.853},"o":{"x":0.167,"y":0.132},"t":24,"s":[77.722,82.334,0],"to":[0,0.517,0],"ti":[0,-0.578,0]},{"i":{"x":0.833,"y":0.89},"o":{"x":0.167,"y":0.192},"t":25,"s":[77.722,84.298,0],"to":[0,0.578,0],"ti":[0,-0.332,0]},{"i":{"x":0.833,"y":0.878},"o":{"x":0.167,"y":0.341},"t":26,"s":[77.722,85.804,0],"to":[0,0.332,0],"ti":[0,-0.115,0]},{"i":{"x":0.833,"y":0.716},"o":{"x":0.167,"y":0.261},"t":27,"s":[77.722,86.29,0],"to":[0,0.115,0],"ti":[0,0.058,0]},{"i":{"x":0.833,"y":0.847},"o":{"x":0.167,"y":0.118},"t":28,"s":[77.722,86.491,0],"to":[0,-0.058,0],"ti":[0,0.168,0]},{"i":{"x":0.833,"y":0.712},"o":{"x":0.167,"y":0.183},"t":29,"s":[77.722,85.944,0],"to":[0,-0.168,0],"ti":[0,0.264,0]},{"i":{"x":0.833,"y":0.826},"o":{"x":0.167,"y":0.117},"t":30,"s":[77.722,85.485,0],"to":[0,-0.264,0],"ti":[0,0.392,0]},{"i":{"x":0.833,"y":0.876},"o":{"x":0.167,"y":0.16},"t":31,"s":[77.722,84.362,0],"to":[0,-0.392,0],"ti":[0,0.305,0]},{"i":{"x":0.833,"y":0.763},"o":{"x":0.167,"y":0.253},"t":32,"s":[77.722,83.135,0],"to":[0,-0.305,0],"ti":[0,0.285,0]},{"i":{"x":0.833,"y":0.847},"o":{"x":0.167,"y":0.128},"t":33,"s":[77.722,82.533,0],"to":[0,-0.285,0],"ti":[0,0.34,0]},{"i":{"x":0.833,"y":0.882},"o":{"x":0.167,"y":0.183},"t":34,"s":[77.722,81.423,0],"to":[0,-0.34,0],"ti":[0,0.219,0]},{"i":{"x":0.833,"y":0.785},"o":{"x":0.167,"y":0.284},"t":35,"s":[77.722,80.493,0],"to":[0,-0.219,0],"ti":[0,0.166,0]},{"i":{"x":0.833,"y":0.861},"o":{"x":0.167,"y":0.136},"t":36,"s":[77.722,80.107,0],"to":[0,-0.166,0],"ti":[0,0.169,0]},{"i":{"x":0.833,"y":0.889},"o":{"x":0.167,"y":0.209},"t":37,"s":[77.722,79.497,0],"to":[0,-0.169,0],"ti":[0,0.09,0]},{"i":{"x":0.833,"y":0.821},"o":{"x":0.167,"y":0.334},"t":38,"s":[77.722,79.09,0],"to":[0,-0.09,0],"ti":[0,0.048,0]},{"i":{"x":0.833,"y":0.899},"o":{"x":0.167,"y":0.156},"t":39,"s":[77.722,78.956,0],"to":[0,-0.048,0],"ti":[0,0.031,0]},{"i":{"x":0.833,"y":0.868},"o":{"x":0.167,"y":0.478},"t":40,"s":[77.722,78.801,0],"to":[0,-0.031,0],"ti":[0,0.002,0]},{"i":{"x":0.833,"y":0.553},"o":{"x":0.167,"y":0.229},"t":41,"s":[77.722,78.768,0],"to":[0,-0.002,0],"ti":[0,-0.017,0]},{"i":{"x":0.833,"y":0.795},"o":{"x":0.167,"y":0.103},"t":42,"s":[77.722,78.786,0],"to":[0,0.017,0],"ti":[0,-0.034,0]},{"i":{"x":0.833,"y":0.871},"o":{"x":0.167,"y":0.141},"t":43,"s":[77.722,78.868,0],"to":[0,0.034,0],"ti":[0,-0.031,0]},{"i":{"x":0.833,"y":0.749},"o":{"x":0.167,"y":0.234},"t":44,"s":[77.722,78.987,0],"to":[0,0.031,0],"ti":[0,-0.033,0]},{"i":{"x":0.833,"y":0.84},"o":{"x":0.167,"y":0.125},"t":45,"s":[77.722,79.053,0],"to":[0,0.033,0],"ti":[0,-0.042,0]},{"i":{"x":0.833,"y":0.88},"o":{"x":0.167,"y":0.174},"t":46,"s":[77.722,79.186,0],"to":[0,0.042,0],"ti":[0,-0.029,0]},{"i":{"x":0.833,"y":0.775},"o":{"x":0.167,"y":0.271},"t":47,"s":[77.722,79.308,0],"to":[0,0.029,0],"ti":[0,-0.024,0]},{"i":{"x":0.833,"y":0.854},"o":{"x":0.167,"y":0.133},"t":48,"s":[77.722,79.362,0],"to":[0,0.024,0],"ti":[0,-0.027,0]},{"i":{"x":0.833,"y":0.885},"o":{"x":0.167,"y":0.195},"t":49,"s":[77.722,79.454,0],"to":[0,0.027,0],"ti":[0,-0.016,0]},{"i":{"x":0.833,"y":0.8},"o":{"x":0.167,"y":0.305},"t":50,"s":[77.722,79.523,0],"to":[0,0.016,0],"ti":[0,-0.01,0]},{"i":{"x":0.833,"y":0.874},"o":{"x":0.167,"y":0.143},"t":51,"s":[77.722,79.549,0],"to":[0,0.01,0],"ti":[0,-0.009,0]},{"i":{"x":0.833,"y":0.899},"o":{"x":0.167,"y":0.247},"t":52,"s":[77.722,79.585,0],"to":[0,0.009,0],"ti":[0,-0.004,0]},{"i":{"x":0.833,"y":0.884},"o":{"x":0.167,"y":0.484},"t":53,"s":[77.722,79.603,0],"to":[0,0.004,0],"ti":[0,-0.001,0]},{"i":{"x":0.833,"y":0.492},"o":{"x":0.167,"y":0.278},"t":54,"s":[77.722,79.607,0],"to":[0,0.001,0],"ti":[0,0.001,0]},{"i":{"x":0.833,"y":0.857},"o":{"x":0.167,"y":0.099},"t":55,"s":[77.722,79.607,0],"to":[0,-0.001,0],"ti":[0,0.002,0]},{"i":{"x":0.833,"y":0.724},"o":{"x":0.167,"y":0.199},"t":56,"s":[77.722,79.598,0],"to":[0,-0.002,0],"ti":[0,0.003,0]},{"i":{"x":0.833,"y":0.83},"o":{"x":0.167,"y":0.119},"t":57,"s":[77.722,79.592,0],"to":[0,-0.003,0],"ti":[0,0.005,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.163},"t":58,"s":[77.722,79.579,0],"to":[0,-0.005,0],"ti":[0,0.002,0]},{"t":59,"s":[77.722,79.564,0]}],"ix":2,"l":2},"a":{"a":0,"k":[-0.918,5,0],"ix":1,"l":2},"s":{"a":0,"k":[30.469,30.469,100],"ix":6,"l":2}},"ao":0,"ip":0,"op":60,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"Call Unmute Outlines","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[-0.006,-0.008,0],"ix":2,"l":2},"a":{"a":0,"k":[78,78,0],"ix":1,"l":2},"s":{"a":0,"k":[328.205,328.205,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-28.5,-28.5],[28.5,28.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":3,"s":[100]},{"t":17,"s":[0]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":7,"ix":5},"lc":2,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[78.5,79.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 2","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"Call Mute Outlines","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[-0.006,-0.008,0],"ix":2,"l":2},"a":{"a":0,"k":[78,78,0],"ix":1,"l":2},"s":{"a":0,"k":[328.205,328.205,100],"ix":6,"l":2}},"ao":0,"hasMask":true,"masksProperties":[{"inv":false,"mode":"a","pt":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":3,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[-2.745,2.55],[0,0],[2.734,2.734],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[2.671,2.672],[0,0],[2.734,-2.734],[0,0],[0,0],[0,0]],"v":[[156.5,0.5],[156.5,156.5],[0.5,156.5],[0.5,21.5],[10.5,21.398],[102.55,113.45],[112.261,113.632],[112.45,113.45],[112.45,103.55],[10.5,1.601],[10.5,0.5]],"c":true}]},{"t":17,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[-2.745,2.55],[0,0],[2.734,2.734],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[2.671,2.672],[0,0],[2.734,-2.734],[0,0],[0,0],[0,0]],"v":[[156.5,0.5],[156.5,156.5],[0.5,156.5],[0.5,21.5],[10.5,21.398],[47.55,58.45],[57.261,58.632],[57.45,58.45],[57.45,48.55],[10.5,1.601],[10.5,0.5]],"c":true}]}],"ix":1},"o":{"a":0,"k":100,"ix":3},"x":{"a":0,"k":0,"ix":4},"nm":"Mask 1"}],"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-6.817,0],[0,-6.831],[0,0],[6.817,0],[0,6.831],[0,0]],"o":[[6.817,0],[0,0],[0,6.831],[-6.817,0],[0,0],[0,-6.831]],"v":[[0.007,-33],[12.349,-20.632],[12.349,1.047],[0.007,13.416],[-12.336,1.047],[-12.336,-20.632]],"c":true},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[-0.275,-1.973],[-9.95,0],[-1.477,10.291],[-1.968,-0.284],[0.283,-1.972],[11.947,-1.642],[0,0],[1.988,0],[0,1.992],[0,0],[1.757,12.587],[-1.969,0.276]],"o":[[1.442,10.33],[9.92,0],[0.283,-1.972],[1.968,0.284],[-1.799,12.54],[0,0],[0,1.992],[-1.988,0],[0,0],[-11.981,-1.651],[-0.275,-1.973],[1.969,-0.276]],"v":[[-19.835,3.117],[0.007,21.154],[19.836,3.2],[23.912,0.143],[26.963,4.227],[3.603,28.124],[3.603,35.393],[0.003,39],[-3.597,35.393],[-3.597,28.123],[-26.965,4.116],[-23.898,0.044]],"c":true},"ix":2},"nm":"Path 3","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"mm","mm":1,"nm":"Merge Paths 1","mn":"ADBE Vector Filter - Merge","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[78,78],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":4,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60,"st":0,"ct":1,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/TMessagesProj/src/main/res/raw/camera_flip2.json b/TMessagesProj/src/main/res/raw/camera_flip2.json new file mode 100644 index 00000000000..1d7dcf49bd1 --- /dev/null +++ b/TMessagesProj/src/main/res/raw/camera_flip2.json @@ -0,0 +1 @@ +{"v":"5.10.1","fr":60,"ip":0,"op":60,"w":156,"h":156,"nm":"camera_flip","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":4,"nm":"Camera Flip Outlines","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[78,78,0],"ix":2,"l":2},"a":{"a":0,"k":[78,78,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[-1.106,-1.407],[0,0],[-1.789,0],[0,0],[0,-3.84],[0,0],[3.85,0],[0,0],[0,3.84],[0,0],[-3.85,0],[0,0],[-1.106,1.406],[0,0],[-1.789,0]],"o":[[1.789,0],[0,0],[1.106,1.406],[0,0],[3.85,0],[0,0],[0,3.84],[0,0],[-3.85,0],[0,0],[0,-3.84],[0,0],[1.789,0],[0,0],[1.106,-1.407],[0,0]],"v":[[7.666,-33],[12.252,-30.772],[15.153,-27.082],[19.739,-24.855],[26.833,-24.855],[35,-16.709],[35,22.855],[26.833,31],[-26.833,31],[-35,22.855],[-35,-16.709],[-26.833,-24.855],[-19.739,-24.855],[-15.153,-27.082],[-12.252,-30.772],[-7.666,-33]],"c":true},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0.471,-0.453],[0,0],[0,-0.331],[-0.672,0],[0,0],[-1.382,-2.963],[-9.51,4.435],[0.584,1.251],[1.251,-0.584],[3.268,7.008],[-0.233,2.449],[0,0],[-0.229,0.238],[0.484,0.466]],"o":[[-0.471,-0.453],[0,0],[-0.238,0.229],[0,0.672],[0,0],[-0.222,3.173],[4.435,9.51],[1.251,-0.584],[-0.584,-1.251],[-7.008,3.268],[-1.07,-2.294],[0,0],[0.331,0],[0.466,-0.484],[0,0]],"v":[[62.669,69.1],[60.983,69.1],[52.673,77.096],[52.3,77.972],[53.516,79.189],[59.04,79.188],[60.778,88.527],[86.028,97.717],[87.237,94.395],[83.915,93.186],[65.31,86.414],[64.057,79.189],[70.136,79.189],[71.012,78.816],[70.979,77.096]],"c":true},"ix":2},"nm":"Path 3","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[1.382,2.963],[9.51,-4.435],[-0.584,-1.251],[-1.251,0.584],[-3.268,-7.008],[0.233,-2.449],[0,0],[0.229,-0.238],[-0.484,-0.466],[0,0],[-0.471,0.453],[0,0],[0,0.331],[0.672,0],[0,0]],"o":[[-4.435,-9.51],[-1.251,0.584],[0.584,1.251],[7.008,-3.268],[1.07,2.294],[0,0],[-0.331,0],[-0.466,0.484],[0,0],[0.471,0.453],[0,0],[0.238,-0.229],[0,-0.672],[0,0],[0.222,-3.173]],"v":[[95.22,72.47],[69.97,63.28],[68.761,66.602],[72.083,67.812],[90.688,74.583],[91.942,81.809],[85.862,81.809],[84.986,82.182],[85.019,83.902],[93.329,91.898],[95.015,91.898],[103.325,83.902],[103.698,83.025],[102.482,81.809],[96.958,81.809]],"c":true},"ix":2},"nm":"Path 4","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tr","p":{"a":0,"k":[0,2.5],"ix":2},"a":{"a":0,"k":[77.999,80.499],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.79]},"o":{"x":[0.167],"y":[0]},"t":0,"s":[0]},{"i":{"x":[0.833],"y":[0.689]},"o":{"x":[0.167],"y":[0.083]},"t":1,"s":[0]},{"i":{"x":[0.833],"y":[0.638]},"o":{"x":[0.167],"y":[0.114]},"t":2,"s":[2.522]},{"i":{"x":[0.833],"y":[0.808]},"o":{"x":[0.167],"y":[0.108]},"t":3,"s":[9.401]},{"i":{"x":[0.833],"y":[0.829]},"o":{"x":[0.167],"y":[0.148]},"t":4,"s":[32.408]},{"i":{"x":[0.833],"y":[0.876]},"o":{"x":[0.167],"y":[0.162]},"t":5,"s":[62.283]},{"i":{"x":[0.833],"y":[0.765]},"o":{"x":[0.167],"y":[0.256]},"t":6,"s":[93.835]},{"i":{"x":[0.833],"y":[0.88]},"o":{"x":[0.167],"y":[0.129]},"t":7,"s":[109.046]},{"i":{"x":[0.833],"y":[0.778]},"o":{"x":[0.167],"y":[0.275]},"t":8,"s":[136.729]},{"i":{"x":[0.833],"y":[0.856]},"o":{"x":[0.167],"y":[0.133]},"t":9,"s":[148.796]},{"i":{"x":[0.833],"y":[0.886]},"o":{"x":[0.167],"y":[0.198]},"t":10,"s":[168.86]},{"i":{"x":[0.833],"y":[0.805]},"o":{"x":[0.167],"y":[0.312]},"t":11,"s":[183.4]},{"i":{"x":[0.833],"y":[0.879]},"o":{"x":[0.167],"y":[0.145]},"t":12,"s":[188.706]},{"i":{"x":[0.833],"y":[0.905]},"o":{"x":[0.167],"y":[0.267]},"t":13,"s":[195.829]},{"i":{"x":[0.833],"y":[1.066]},"o":{"x":[0.167],"y":[0.676]},"t":14,"s":[199.054]},{"i":{"x":[0.833],"y":[0.673]},"o":{"x":[0.167],"y":[0.037]},"t":15,"s":[199.508]},{"i":{"x":[0.833],"y":[0.863]},"o":{"x":[0.167],"y":[0.112]},"t":16,"s":[198.696]},{"i":{"x":[0.833],"y":[0.734]},"o":{"x":[0.167],"y":[0.212]},"t":17,"s":[196.321]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.121]},"t":18,"s":[194.788]},{"i":{"x":[0.833],"y":[0.878]},"o":{"x":[0.167],"y":[0.166]},"t":19,"s":[191.419]},{"i":{"x":[0.833],"y":[0.769]},"o":{"x":[0.167],"y":[0.261]},"t":20,"s":[188.042]},{"i":{"x":[0.833],"y":[0.85]},"o":{"x":[0.167],"y":[0.13]},"t":21,"s":[186.461]},{"i":{"x":[0.833],"y":[0.883]},"o":{"x":[0.167],"y":[0.188]},"t":22,"s":[183.653]},{"i":{"x":[0.833],"y":[0.791]},"o":{"x":[0.167],"y":[0.292]},"t":23,"s":[181.413]},{"i":{"x":[0.833],"y":[0.866]},"o":{"x":[0.167],"y":[0.138]},"t":24,"s":[180.519]},{"i":{"x":[0.833],"y":[0.892]},"o":{"x":[0.167],"y":[0.22]},"t":25,"s":[179.168]},{"i":{"x":[0.833],"y":[0.844]},"o":{"x":[0.167],"y":[0.366]},"t":26,"s":[178.344]},{"i":{"x":[0.833],"y":[0.942]},"o":{"x":[0.167],"y":[0.179]},"t":27,"s":[178.101]},{"i":{"x":[0.833],"y":[0.781]},"o":{"x":[0.167],"y":[-0.192]},"t":28,"s":[177.889]},{"i":{"x":[0.833],"y":[0.678]},"o":{"x":[0.167],"y":[0.134]},"t":29,"s":[177.953]},{"i":{"x":[0.833],"y":[0.816]},"o":{"x":[0.167],"y":[0.112]},"t":30,"s":[178.058]},{"i":{"x":[0.833],"y":[0.874]},"o":{"x":[0.167],"y":[0.153]},"t":31,"s":[178.359]},{"i":{"x":[0.833],"y":[0.757]},"o":{"x":[0.167],"y":[0.246]},"t":32,"s":[178.722]},{"i":{"x":[0.833],"y":[0.844]},"o":{"x":[0.167],"y":[0.127]},"t":33,"s":[178.908]},{"i":{"x":[0.833],"y":[0.881]},"o":{"x":[0.167],"y":[0.179]},"t":34,"s":[179.263]},{"i":{"x":[0.833],"y":[0.781]},"o":{"x":[0.167],"y":[0.278]},"t":35,"s":[179.574]},{"i":{"x":[0.833],"y":[0.858]},"o":{"x":[0.167],"y":[0.134]},"t":36,"s":[179.707]},{"i":{"x":[0.833],"y":[0.887]},"o":{"x":[0.167],"y":[0.202]},"t":37,"s":[179.924]},{"i":{"x":[0.833],"y":[0.81]},"o":{"x":[0.167],"y":[0.319]},"t":38,"s":[180.077]},{"i":{"x":[0.833],"y":[0.884]},"o":{"x":[0.167],"y":[0.148]},"t":39,"s":[180.132]},{"i":{"x":[0.833],"y":[0.913]},"o":{"x":[0.167],"y":[0.299]},"t":40,"s":[180.201]},{"i":{"x":[0.833],"y":[2.16]},"o":{"x":[0.167],"y":[2.203]},"t":41,"s":[180.228]},{"i":{"x":[0.833],"y":[0.751]},"o":{"x":[0.167],"y":[0.078]},"t":42,"s":[180.229]},{"i":{"x":[0.833],"y":[0.866]},"o":{"x":[0.167],"y":[0.125]},"t":43,"s":[180.213]},{"i":{"x":[0.833],"y":[0.74]},"o":{"x":[0.167],"y":[0.222]},"t":44,"s":[180.182]},{"i":{"x":[0.833],"y":[0.836]},"o":{"x":[0.167],"y":[0.123]},"t":45,"s":[180.163]},{"i":{"x":[0.833],"y":[0.878]},"o":{"x":[0.167],"y":[0.169]},"t":46,"s":[180.123]},{"i":{"x":[0.833],"y":[0.771]},"o":{"x":[0.167],"y":[0.265]},"t":47,"s":[180.084]},{"i":{"x":[0.833],"y":[0.852]},"o":{"x":[0.167],"y":[0.131]},"t":48,"s":[180.066]},{"i":{"x":[0.833],"y":[0.884]},"o":{"x":[0.167],"y":[0.19]},"t":49,"s":[180.035]},{"i":{"x":[0.833],"y":[0.794]},"o":{"x":[0.167],"y":[0.296]},"t":50,"s":[180.01]},{"i":{"x":[0.833],"y":[0.868]},"o":{"x":[0.167],"y":[0.14]},"t":51,"s":[180.001]},{"i":{"x":[0.833],"y":[0.894]},"o":{"x":[0.167],"y":[0.227]},"t":52,"s":[179.987]},{"i":{"x":[0.833],"y":[0.862]},"o":{"x":[0.167],"y":[0.391]},"t":53,"s":[179.979]},{"i":{"x":[0.833],"y":[1.001]},"o":{"x":[0.167],"y":[0.21]},"t":54,"s":[179.977]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.001]},"t":55,"s":[179.975]},{"i":{"x":[0.833],"y":[0.701]},"o":{"x":[0.167],"y":[0.166]},"t":56,"s":[179.977]},{"i":{"x":[0.833],"y":[0.822]},"o":{"x":[0.167],"y":[0.115]},"t":57,"s":[179.978]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.157]},"t":58,"s":[179.982]},{"t":59,"s":[179.986]}],"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":2,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"mm","mm":1,"nm":"Merge Paths 1","mn":"ADBE Vector Filter - Merge","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[78,78],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":4,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60,"st":0,"ct":1,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/TMessagesProj/src/main/res/raw/fire_once.json b/TMessagesProj/src/main/res/raw/fire_once.json new file mode 100644 index 00000000000..478c9905646 --- /dev/null +++ b/TMessagesProj/src/main/res/raw/fire_once.json @@ -0,0 +1 @@ +{"tgs":1,"v":"5.5.2","fr":60,"ip":60,"op":180,"w":512,"h":512,"nm":"Fire 1 Fire","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":3,"nm":"SCALE ALL","sr":1,"ks":{"o":{"a":0,"k":0},"p":{"a":0,"k":[256,256,0]},"s":{"a":0,"k":[95,95,100]}},"ao":0,"ip":60,"op":180,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":3,"nm":"NULL SCALE FIRE","parent":1,"sr":1,"ks":{"o":{"a":0,"k":0},"p":{"a":0,"k":[-11,222,0]},"s":{"a":1,"k":[{"i":{"x":[0.7,0.7,0.7],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0]},"t":60,"s":[109.815,97.926,100]},{"i":{"x":[0.7,0.7,0.7],"y":[1,1,1]},"o":{"x":[0.3,0.3,0.3],"y":[0,0,0]},"t":65,"s":[115,93,100]},{"i":{"x":[0.7,0.7,0.7],"y":[1,1,1]},"o":{"x":[0.3,0.3,0.3],"y":[0,0,0]},"t":78,"s":[90,110,100]},{"i":{"x":[0.7,0.7,0.7],"y":[1,1,1]},"o":{"x":[0.3,0.3,0.3],"y":[0,0,0]},"t":95,"s":[118,83,100]},{"i":{"x":[0.7,0.7,0.7],"y":[1,1,1]},"o":{"x":[0.3,0.3,0.3],"y":[0,0,0]},"t":110,"s":[94,106,100]},{"i":{"x":[0.7,0.7,0.7],"y":[1,1,1]},"o":{"x":[0.3,0.3,0.3],"y":[0,0,0]},"t":125,"s":[112,90,100]},{"i":{"x":[0.7,0.7,0.7],"y":[1,1,1]},"o":{"x":[0.3,0.3,0.3],"y":[0,0,0]},"t":140,"s":[92,110,100]},{"i":{"x":[0.7,0.7,0.7],"y":[1,1,1]},"o":{"x":[0.3,0.3,0.3],"y":[0,0,0]},"t":155,"s":[115,90,100]},{"i":{"x":[0.7,0.7,0.7],"y":[1,1,1]},"o":{"x":[0.3,0.3,0.3],"y":[0,0,0]},"t":168,"s":[93,108,100]},{"t":179,"s":[109.815,97.926,100]}]}},"ao":0,"ip":60,"op":180,"st":-60,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"Main","parent":2,"sr":1,"ks":{"a":{"a":0,"k":[-11,192,0]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":59,"s":[{"i":[[-125.024,61.75],[-2.13,4.684],[48.466,105.584]],"o":[[-7.082,-3.904],[8.978,-20.285],[-16.933,71.067]],"v":[[100.957,152.217],[94.735,139.709],[126.467,-21.933]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":60,"s":[{"i":[[-48.957,-22.25],[0.197,0.442],[-70.695,21.726]],"o":[[-0.82,-0.452],[-1.61,-3.615],[-90.667,23]],"v":[[15.207,182.25],[13.658,180.925],[103.667,151.286]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":75,"s":[{"i":[[-48.957,-22.25],[0.213,1.609],[-75.448,37.427]],"o":[[-0.82,-0.452],[-1.741,-13.168],[-90.667,23]],"v":[[58.957,165.75],[57.379,162.616],[151.667,63]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":89,"s":[{"i":[[-125.024,61.75],[-2.13,4.684],[48.466,105.584]],"o":[[-7.082,-3.904],[8.978,-20.285],[-16.933,71.067]],"v":[[100.957,152.217],[94.735,139.709],[126.467,-21.933]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":90,"s":[{"i":[[-48.957,-22.25],[0.197,0.442],[-70.695,21.726]],"o":[[-0.82,-0.452],[-1.61,-3.615],[-90.667,23]],"v":[[15.207,182.25],[13.658,180.925],[103.667,151.286]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":105,"s":[{"i":[[-48.957,-22.25],[0.213,1.609],[-75.448,37.427]],"o":[[-0.82,-0.452],[-1.741,-13.168],[-90.667,23]],"v":[[58.957,165.75],[57.379,162.616],[151.667,63]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":119,"s":[{"i":[[-125.024,61.75],[-2.13,4.684],[48.466,105.584]],"o":[[-7.082,-3.904],[8.978,-20.285],[-16.933,71.067]],"v":[[100.957,152.217],[94.735,139.709],[126.467,-21.933]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":120,"s":[{"i":[[-48.957,-22.25],[0.197,0.442],[-70.695,21.726]],"o":[[-0.82,-0.452],[-1.61,-3.615],[-90.667,23]],"v":[[15.207,182.25],[13.658,180.925],[103.667,151.286]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":135,"s":[{"i":[[-48.957,-22.25],[0.213,1.609],[-75.448,37.427]],"o":[[-0.82,-0.452],[-1.741,-13.168],[-90.667,23]],"v":[[58.957,165.75],[57.379,162.616],[151.667,63]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":149,"s":[{"i":[[-125.024,61.75],[-2.13,4.684],[48.466,105.584]],"o":[[-7.082,-3.904],[8.978,-20.285],[-16.933,71.067]],"v":[[100.957,152.217],[94.735,139.709],[126.467,-21.933]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":150,"s":[{"i":[[-48.957,-22.25],[0.197,0.442],[-70.695,21.726]],"o":[[-0.82,-0.452],[-1.61,-3.615],[-90.667,23]],"v":[[15.207,182.25],[13.658,180.925],[103.667,151.286]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":165,"s":[{"i":[[-48.957,-22.25],[0.213,1.609],[-75.448,37.427]],"o":[[-0.82,-0.452],[-1.741,-13.168],[-90.667,23]],"v":[[58.957,165.75],[57.379,162.616],[151.667,63]],"c":true}]},{"t":179,"s":[{"i":[[-125.024,61.75],[-2.13,4.684],[52.039,111.96]],"o":[[-7.082,-3.904],[7.326,-24.421],[-16.933,71.067]],"v":[[101.676,152.754],[95.693,139.709],[125.748,-25.158]],"c":true}]}]},"nm":"Path 7","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[-11.062,192.375]},"a":{"a":0,"k":[-11.062,191.875]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 15","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":59,"s":[{"i":[[57.96,124.758],[13.259,3.326],[135.474,53.272]],"o":[[-3.77,13.222],[-23.379,-6.387],[-14.8,286.022]],"v":[[126.24,-22.025],[96.068,1.911],[-10.424,-191.94]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":60,"s":[{"i":[[-130.457,67.75],[-2.298,4.904],[57.317,110.452]],"o":[[-7.529,-4.15],[9.743,-20.793],[-11.667,74.5]],"v":[[103.957,151.25],[97.403,138.073],[124.667,-28]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":75,"s":[{"i":[[-132.207,149.375],[10.866,5.715],[-5.26,21.308]],"o":[[-18.645,9.327],[-46.078,-24.234],[-13,24.333]],"v":[[151.207,61.625],[107.41,65.591],[82,-113.333]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":89,"s":[{"i":[[57.96,124.758],[13.259,3.326],[135.474,53.272]],"o":[[-3.77,13.222],[-23.379,-6.387],[-14.8,286.022]],"v":[[126.24,-22.025],[96.068,1.911],[-10.424,-191.94]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":90,"s":[{"i":[[-130.457,67.75],[-2.298,4.904],[57.317,110.452]],"o":[[-7.529,-4.15],[9.743,-20.793],[-11.667,74.5]],"v":[[103.957,151.25],[97.403,138.073],[124.667,-28]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":105,"s":[{"i":[[-132.207,149.375],[10.866,5.715],[-5.26,21.308]],"o":[[-18.645,9.327],[-46.078,-24.234],[-13,24.333]],"v":[[151.207,61.625],[107.41,65.591],[82,-113.333]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":119,"s":[{"i":[[57.96,124.758],[13.259,3.326],[135.474,53.272]],"o":[[-3.77,13.222],[-23.379,-6.387],[-14.8,286.022]],"v":[[126.24,-22.025],[96.068,1.911],[-10.424,-191.94]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":120,"s":[{"i":[[-130.457,67.75],[-2.298,4.904],[57.317,110.452]],"o":[[-7.529,-4.15],[9.743,-20.793],[-11.667,74.5]],"v":[[103.957,151.25],[97.403,138.073],[124.667,-28]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":135,"s":[{"i":[[-132.207,149.375],[10.866,5.715],[-5.26,21.308]],"o":[[-18.645,9.327],[-46.078,-24.234],[-13,24.333]],"v":[[151.207,61.625],[107.41,65.591],[82,-113.333]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":149,"s":[{"i":[[57.96,124.758],[13.259,3.326],[135.474,53.272]],"o":[[-3.77,13.222],[-23.379,-6.387],[-14.8,286.022]],"v":[[126.24,-22.025],[96.068,1.911],[-10.424,-191.94]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":150,"s":[{"i":[[-130.457,67.75],[-2.298,4.904],[57.317,110.452]],"o":[[-7.529,-4.15],[9.743,-20.793],[-11.667,74.5]],"v":[[103.957,151.25],[97.403,138.073],[124.667,-28]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":165,"s":[{"i":[[-132.207,149.375],[10.866,5.715],[-5.26,21.308]],"o":[[-18.645,9.327],[-46.078,-24.234],[-13,24.333]],"v":[[151.207,61.625],[107.41,65.591],[82,-113.333]],"c":true}]},{"t":179,"s":[{"i":[[57.96,124.758],[13.259,3.326],[135.474,53.272]],"o":[[-3.77,13.222],[-23.379,-6.387],[-14.8,286.022]],"v":[[125.522,-25.25],[95.828,-1.314],[-10.424,-191.94]],"c":true}]}]},"nm":"Path 7","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[-11.062,192.375]},"a":{"a":0,"k":[-11.062,191.875]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 14","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":59,"s":[{"i":[[55.126,64.65],[0.301,0.099],[0.831,2.772]],"o":[[-0.349,-0.086],[-2.963,-0.975],[-3.627,85.303]],"v":[[-10.876,-192.4],[-11.85,-192.677],[-16.373,-198.053]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":60,"s":[{"i":[[71.543,123],[13.429,3.155],[145.526,55.555]],"o":[[-2.707,13.5],[-21.758,-5.112],[-14.929,304.714]],"v":[[124.457,-28],[95.258,-2.638],[-17.026,-197.555]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":75,"s":[{"i":[[-25.457,62],[5.632,-3.644],[-17.612,44.304]],"o":[[-6.17,6.45],[-55.496,35.902],[-124.4,236.8]],"v":[[82.457,-114],[64.744,-98.951],[-1.6,-198.8]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":89,"s":[{"i":[[55.126,64.65],[0.301,0.099],[0.831,2.772]],"o":[[-0.349,-0.086],[-2.963,-0.975],[-3.627,85.303]],"v":[[-10.876,-192.4],[-11.85,-192.677],[-16.373,-198.053]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":90,"s":[{"i":[[71.543,123],[13.429,3.155],[145.526,55.555]],"o":[[-2.707,13.5],[-21.758,-5.112],[-14.929,304.714]],"v":[[124.457,-28],[95.258,-2.638],[-17.026,-197.555]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":105,"s":[{"i":[[-25.457,62],[5.632,-3.644],[-17.612,44.304]],"o":[[-6.17,6.45],[-55.496,35.902],[-124.4,236.8]],"v":[[82.457,-114],[64.744,-98.951],[-1.6,-198.8]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":119,"s":[{"i":[[55.126,64.65],[0.301,0.099],[0.831,2.772]],"o":[[-0.349,-0.086],[-2.963,-0.975],[-3.627,85.303]],"v":[[-10.876,-192.4],[-11.85,-192.677],[-16.373,-198.053]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":120,"s":[{"i":[[71.543,123],[13.429,3.155],[145.526,55.555]],"o":[[-2.707,13.5],[-21.758,-5.112],[-14.929,304.714]],"v":[[124.457,-28],[95.258,-2.638],[-17.026,-197.555]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":135,"s":[{"i":[[-25.457,62],[5.632,-3.644],[-17.612,44.304]],"o":[[-6.17,6.45],[-55.496,35.902],[-124.4,236.8]],"v":[[82.457,-114],[64.744,-98.951],[-1.6,-198.8]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":149,"s":[{"i":[[55.126,64.65],[0.301,0.099],[0.831,2.772]],"o":[[-0.349,-0.086],[-2.963,-0.975],[-3.627,85.303]],"v":[[-10.876,-192.4],[-11.85,-192.677],[-16.373,-198.053]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":150,"s":[{"i":[[71.543,123],[13.429,3.155],[145.526,55.555]],"o":[[-2.707,13.5],[-21.758,-5.112],[-14.929,304.714]],"v":[[124.457,-28],[95.258,-2.638],[-17.026,-197.555]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":165,"s":[{"i":[[-25.457,62],[5.632,-3.644],[-17.612,44.304]],"o":[[-6.17,6.45],[-55.496,35.902],[-124.4,236.8]],"v":[[82.457,-114],[64.744,-98.951],[-1.6,-198.8]],"c":true}]},{"t":179,"s":[{"i":[[55.126,64.65],[0.301,0.099],[0.831,2.772]],"o":[[-0.349,-0.086],[-2.963,-0.975],[1.252,76.857]],"v":[[-10.876,-192.4],[-11.85,-192.677],[-16.373,-198.053]],"c":true}]}]},"nm":"Path 6","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 2","hd":false},{"ty":"tr","p":{"a":0,"k":[-11.062,192.375]},"a":{"a":0,"k":[-11.062,191.875]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 13","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":59,"s":[{"i":[[-20.057,-2.894],[40.208,25.5]],"o":[[2.443,9.106],[7.458,3.75]],"v":[[-39.943,164.894],[-43.958,166.25]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":60,"s":[{"i":[[-20.057,-2.894],[40.208,25.5]],"o":[[2.443,9.106],[7.458,3.75]],"v":[[-63.443,163.894],[-132.708,147.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":74,"s":[{"i":[[85.814,67.785],[-77.508,98.3]],"o":[[53.247,14.918],[-19.958,48.317]],"v":[[-128.31,148.728],[-137.925,-14.583]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":75,"s":[{"i":[[-20.057,-2.894],[40.208,25.5]],"o":[[2.443,9.106],[7.458,3.75]],"v":[[-39.943,164.894],[-43.958,166.25]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":89,"s":[{"i":[[-20.057,-2.894],[40.208,25.5]],"o":[[2.443,9.106],[7.458,3.75]],"v":[[-39.943,164.894],[-43.958,166.25]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":90,"s":[{"i":[[-20.057,-2.894],[40.208,25.5]],"o":[[2.443,9.106],[7.458,3.75]],"v":[[-63.443,163.894],[-132.708,147.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":104,"s":[{"i":[[85.814,67.785],[-77.508,98.3]],"o":[[53.247,14.918],[-19.958,48.317]],"v":[[-128.31,148.728],[-137.925,-14.583]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":105,"s":[{"i":[[-20.057,-2.894],[40.208,25.5]],"o":[[2.443,9.106],[7.458,3.75]],"v":[[-39.943,164.894],[-43.958,166.25]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":119,"s":[{"i":[[-20.057,-2.894],[40.208,25.5]],"o":[[2.443,9.106],[7.458,3.75]],"v":[[-39.943,164.894],[-43.958,166.25]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":120,"s":[{"i":[[-20.057,-2.894],[40.208,25.5]],"o":[[2.443,9.106],[7.458,3.75]],"v":[[-63.443,163.894],[-132.708,147.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":134,"s":[{"i":[[85.814,67.785],[-77.508,98.3]],"o":[[53.247,14.918],[-19.958,48.317]],"v":[[-128.31,148.728],[-137.925,-14.583]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":135,"s":[{"i":[[-20.057,-2.894],[40.208,25.5]],"o":[[2.443,9.106],[7.458,3.75]],"v":[[-39.943,164.894],[-43.958,166.25]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":149,"s":[{"i":[[-20.057,-2.894],[40.208,25.5]],"o":[[2.443,9.106],[7.458,3.75]],"v":[[-39.943,164.894],[-43.958,166.25]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":150,"s":[{"i":[[-20.057,-2.894],[40.208,25.5]],"o":[[2.443,9.106],[7.458,3.75]],"v":[[-63.443,163.894],[-132.708,147.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":164,"s":[{"i":[[85.814,67.785],[-77.508,98.3]],"o":[[53.247,14.918],[-19.958,48.317]],"v":[[-128.31,148.728],[-137.925,-14.583]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":165,"s":[{"i":[[-20.057,-2.894],[40.208,25.5]],"o":[[2.443,9.106],[7.458,3.75]],"v":[[-39.943,164.894],[-43.958,166.25]],"c":true}]},{"t":179,"s":[{"i":[[-20.057,-2.894],[40.208,25.5]],"o":[[2.443,9.106],[7.458,3.75]],"v":[[-39.943,164.894],[-43.958,166.25]],"c":true}]}]},"nm":"Path 4","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[-11.062,192.375]},"a":{"a":0,"k":[-11.062,191.875]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 12","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":59,"s":[{"i":[[85.814,67.785],[-77.508,98.3]],"o":[[53.247,14.918],[-19.958,48.317]],"v":[[-128.31,148.728],[-134.925,-15.833]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":60,"s":[{"i":[[93.376,72.833],[-85.917,103.5]],"o":[[56.876,15.333],[-21.917,51.5]],"v":[[-132.943,147.644],[-135.083,-27.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":74,"s":[{"i":[[-22.202,124.789],[-49.9,85.833]],"o":[[22.57,8.324],[22.6,258.333]],"v":[[-138.57,-14.824],[-63.6,-109.333]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":75,"s":[{"i":[[-20.057,-2.894],[40.208,25.5]],"o":[[2.443,9.106],[7.458,3.75]],"v":[[-63.443,163.894],[-132.708,147.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":89,"s":[{"i":[[85.814,67.785],[-77.508,98.3]],"o":[[53.247,14.918],[-19.958,48.317]],"v":[[-128.31,148.728],[-134.925,-15.833]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":90,"s":[{"i":[[93.376,72.833],[-85.917,103.5]],"o":[[56.876,15.333],[-21.917,51.5]],"v":[[-132.943,147.644],[-135.083,-27.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":104,"s":[{"i":[[-22.202,124.789],[-49.9,85.833]],"o":[[22.57,8.324],[22.6,258.333]],"v":[[-138.57,-14.824],[-63.6,-109.333]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":105,"s":[{"i":[[-20.057,-2.894],[40.208,25.5]],"o":[[2.443,9.106],[7.458,3.75]],"v":[[-63.443,163.894],[-132.708,147.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":119,"s":[{"i":[[85.814,67.785],[-77.508,98.3]],"o":[[53.247,14.918],[-19.958,48.317]],"v":[[-128.31,148.728],[-134.925,-15.833]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":120,"s":[{"i":[[93.376,72.833],[-85.917,103.5]],"o":[[56.876,15.333],[-21.917,51.5]],"v":[[-132.943,147.644],[-135.083,-27.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":134,"s":[{"i":[[-22.202,124.789],[-49.9,85.833]],"o":[[22.57,8.324],[22.6,258.333]],"v":[[-138.57,-14.824],[-63.6,-109.333]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":135,"s":[{"i":[[-20.057,-2.894],[40.208,25.5]],"o":[[2.443,9.106],[7.458,3.75]],"v":[[-63.443,163.894],[-132.708,147.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":149,"s":[{"i":[[85.814,67.785],[-77.508,98.3]],"o":[[53.247,14.918],[-19.958,48.317]],"v":[[-128.31,148.728],[-134.925,-15.833]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":150,"s":[{"i":[[93.376,72.833],[-85.917,103.5]],"o":[[56.876,15.333],[-21.917,51.5]],"v":[[-132.943,147.644],[-135.083,-27.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":164,"s":[{"i":[[-22.202,124.789],[-49.9,85.833]],"o":[[22.57,8.324],[22.6,258.333]],"v":[[-138.57,-14.824],[-63.6,-109.333]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":165,"s":[{"i":[[-20.057,-2.894],[40.208,25.5]],"o":[[2.443,9.106],[7.458,3.75]],"v":[[-63.443,163.894],[-132.708,147.5]],"c":true}]},{"t":179,"s":[{"i":[[85.814,67.785],[-78.857,106.935]],"o":[[53.247,14.918],[-19.958,48.317]],"v":[[-131.185,149.265],[-136.123,-24.97]],"c":true}]}]},"nm":"Path 4","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[-11.062,192.375]},"a":{"a":0,"k":[-11.062,191.875]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 10","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":59,"s":[{"i":[[-39.468,100.989],[-12.802,19.228],[-35.608,63.001]],"o":[[-2.466,24.131],[15.371,-24.636],[49.872,187.567]],"v":[[-134.903,-15.824],[-92.33,17.485],[-62.206,-107.767]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":60,"s":[{"i":[[-48.957,103],[-14.374,19.634],[-34,62.5]],"o":[[-3.957,25.5],[17.842,-24.371],[47,189.5]],"v":[[-135.043,-27.5],[-89.892,9.213],[-57,-113.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":74,"s":[{"i":[[-37.507,99.825],[-6.531,6.628],[-11.668,33.483]],"o":[[14.235,-1.998],[13.653,-13.856],[-19.75,203]],"v":[[-62.993,-109.325],[-33.016,-122.287],[-5.75,-193.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":75,"s":[{"i":[[93.376,72.833],[9.198,13.557],[-58.116,70.01]],"o":[[18.404,4.962],[-19.228,-28.339],[90.083,160.5]],"v":[[-132.943,147.644],[-126.463,133.299],[-135.083,-27.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":89,"s":[{"i":[[-39.468,100.989],[-12.802,19.228],[-35.608,63.001]],"o":[[-2.466,24.131],[15.371,-24.636],[49.872,187.567]],"v":[[-134.903,-15.824],[-92.33,17.485],[-62.206,-107.767]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":90,"s":[{"i":[[-48.957,103],[-14.374,19.634],[-34,62.5]],"o":[[-3.957,25.5],[17.842,-24.371],[47,189.5]],"v":[[-135.043,-27.5],[-89.892,9.213],[-57,-113.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":104,"s":[{"i":[[-37.507,99.825],[-6.531,6.628],[-11.668,33.483]],"o":[[14.235,-1.998],[13.653,-13.856],[-19.75,203]],"v":[[-62.993,-109.325],[-33.016,-122.287],[-5.75,-193.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":105,"s":[{"i":[[93.376,72.833],[9.198,13.557],[-58.116,70.01]],"o":[[18.404,4.962],[-19.228,-28.339],[90.083,160.5]],"v":[[-132.943,147.644],[-126.463,133.299],[-135.083,-27.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":119,"s":[{"i":[[-39.468,100.989],[-12.802,19.228],[-35.608,63.001]],"o":[[-2.466,24.131],[15.371,-24.636],[49.872,187.567]],"v":[[-134.903,-15.824],[-92.33,17.485],[-62.206,-107.767]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":120,"s":[{"i":[[-48.957,103],[-14.374,19.634],[-34,62.5]],"o":[[-3.957,25.5],[17.842,-24.371],[47,189.5]],"v":[[-135.043,-27.5],[-89.892,9.213],[-57,-113.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":134,"s":[{"i":[[-37.507,99.825],[-6.531,6.628],[-11.668,33.483]],"o":[[14.235,-1.998],[13.653,-13.856],[-19.75,203]],"v":[[-62.993,-109.325],[-33.016,-122.287],[-5.75,-193.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":135,"s":[{"i":[[93.376,72.833],[9.198,13.557],[-58.116,70.01]],"o":[[18.404,4.962],[-19.228,-28.339],[90.083,160.5]],"v":[[-132.943,147.644],[-126.463,133.299],[-135.083,-27.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":149,"s":[{"i":[[-39.468,100.989],[-12.802,19.228],[-35.608,63.001]],"o":[[-2.466,24.131],[15.371,-24.636],[49.872,187.567]],"v":[[-134.903,-15.824],[-92.33,17.485],[-62.206,-107.767]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":150,"s":[{"i":[[-48.957,103],[-14.374,19.634],[-34,62.5]],"o":[[-3.957,25.5],[17.842,-24.371],[47,189.5]],"v":[[-135.043,-27.5],[-89.892,9.213],[-57,-113.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":164,"s":[{"i":[[-37.507,99.825],[-6.531,6.628],[-11.668,33.483]],"o":[[14.235,-1.998],[13.653,-13.856],[-19.75,203]],"v":[[-62.993,-109.325],[-33.016,-122.287],[-5.75,-193.5]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":165,"s":[{"i":[[93.376,72.833],[9.198,13.557],[-58.116,70.01]],"o":[[18.404,4.962],[-19.228,-28.339],[90.083,160.5]],"v":[[-132.943,147.644],[-126.463,133.299],[-135.083,-27.5]],"c":true}]},{"t":179,"s":[{"i":[[-39.468,100.989],[-14.812,23.095],[-33.434,58.859]],"o":[[-2.466,24.131],[14.663,-22.858],[49.872,187.567]],"v":[[-136.101,-24.961],[-87.298,8.348],[-57.653,-112.066]],"c":true}]}]},"nm":"Path 3","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[-11.062,192.375]},"a":{"a":0,"k":[-11.062,191.875]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 11","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":59,"s":[{"i":[[-2.709,2.512],[0.244,-41.155]],"o":[[-46.792,-1.888],[70.621,0]],"v":[[100.925,152.255],[-11.052,191.948]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":60,"s":[{"i":[[-0.38,0.778],[0.056,-9.392]],"o":[[-10.717,-0.056],[16.117,0]],"v":[[14.938,182.198],[-10.618,191.257]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":75,"s":[{"i":[[-16.729,-11.318],[0.153,-25.821]],"o":[[-26.979,-24.818],[44.309,0]],"v":[[58.479,166.318],[-11.778,191.222]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":89,"s":[{"i":[[-2.709,2.512],[0.244,-41.155]],"o":[[-46.792,-1.888],[70.621,0]],"v":[[100.925,152.255],[-11.052,191.948]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":90,"s":[{"i":[[-0.38,0.778],[0.056,-9.392]],"o":[[-10.717,-0.056],[16.117,0]],"v":[[14.938,182.198],[-10.618,191.257]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":105,"s":[{"i":[[-16.729,-11.318],[0.153,-25.821]],"o":[[-26.979,-24.818],[44.309,0]],"v":[[58.479,166.318],[-11.778,191.222]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":119,"s":[{"i":[[-2.709,2.512],[0.244,-41.155]],"o":[[-46.792,-1.888],[70.621,0]],"v":[[100.925,152.255],[-11.052,191.948]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":120,"s":[{"i":[[-0.38,0.778],[0.056,-9.392]],"o":[[-10.717,-0.056],[16.117,0]],"v":[[14.938,182.198],[-10.618,191.257]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":135,"s":[{"i":[[-16.729,-11.318],[0.153,-25.821]],"o":[[-26.979,-24.818],[44.309,0]],"v":[[58.479,166.318],[-11.778,191.222]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":149,"s":[{"i":[[-2.709,2.512],[0.244,-41.155]],"o":[[-46.792,-1.888],[70.621,0]],"v":[[100.925,152.255],[-11.052,191.948]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":150,"s":[{"i":[[-0.38,0.778],[0.056,-9.392]],"o":[[-10.717,-0.056],[16.117,0]],"v":[[14.938,182.198],[-10.618,191.257]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":165,"s":[{"i":[[-16.729,-11.318],[0.153,-25.821]],"o":[[-26.979,-24.818],[44.309,0]],"v":[[58.479,166.318],[-11.778,191.222]],"c":true}]},{"t":179,"s":[{"i":[[-2.709,2.512],[0.244,-41.155]],"o":[[-46.792,-1.888],[70.621,0]],"v":[[101.644,152.792],[-11.052,191.948]],"c":true}]}]},"nm":"Path 8","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[-11.062,192.375]},"a":{"a":0,"k":[-11.062,191.875]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 8","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":59,"s":[{"i":[[64.493,117.425],[0.008,-189.908]],"o":[[-16.374,70.358],[87.933,-23.333]],"v":[[126.24,-22.025],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":60,"s":[{"i":[[-1.707,3.5],[0.25,-42.25]],"o":[[-48.207,-0.25],[72.5,0]],"v":[[103.957,151.25],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":75,"s":[{"i":[[-34.207,39.375],[0.125,-118.625]],"o":[[-43.207,-8.625],[143,0]],"v":[[151.207,61.625],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":89,"s":[{"i":[[64.493,117.425],[0.008,-189.908]],"o":[[-16.374,70.358],[87.933,-23.333]],"v":[[126.24,-22.025],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":90,"s":[{"i":[[-1.707,3.5],[0.25,-42.25]],"o":[[-48.207,-0.25],[72.5,0]],"v":[[103.957,151.25],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":105,"s":[{"i":[[-34.207,39.375],[0.125,-118.625]],"o":[[-43.207,-8.625],[143,0]],"v":[[151.207,61.625],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":119,"s":[{"i":[[64.493,117.425],[0.008,-189.908]],"o":[[-16.374,70.358],[87.933,-23.333]],"v":[[126.24,-22.025],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":120,"s":[{"i":[[-1.707,3.5],[0.25,-42.25]],"o":[[-48.207,-0.25],[72.5,0]],"v":[[103.957,151.25],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":135,"s":[{"i":[[-34.207,39.375],[0.125,-118.625]],"o":[[-43.207,-8.625],[143,0]],"v":[[151.207,61.625],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":149,"s":[{"i":[[64.493,117.425],[0.008,-189.908]],"o":[[-16.374,70.358],[87.933,-23.333]],"v":[[126.24,-22.025],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":150,"s":[{"i":[[-1.707,3.5],[0.25,-42.25]],"o":[[-48.207,-0.25],[72.5,0]],"v":[[103.957,151.25],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":165,"s":[{"i":[[-34.207,39.375],[0.125,-118.625]],"o":[[-43.207,-8.625],[143,0]],"v":[[151.207,61.625],[-11,192]],"c":true}]},{"t":179,"s":[{"i":[[65.925,115.008],[0.008,-189.908]],"o":[[-16.374,70.358],[87.933,-23.333]],"v":[[125.522,-25.25],[-11,192]],"c":true}]}]},"nm":"Path 7","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[-11.062,192.375]},"a":{"a":0,"k":[-11.062,191.875]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 7","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":59,"s":[{"i":[[135.076,57.333],[-186.567,-166]],"o":[[21.276,184.067],[131.267,-187.467]],"v":[[-10.876,-192.4],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":60,"s":[{"i":[[71.543,123],[0,-195]],"o":[[-14.457,76],[84,-25]],"v":[[124.457,-28],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":75,"s":[{"i":[[-25.457,62],[-96.5,-180]],"o":[[-66.457,59],[219,-96]],"v":[[82.457,-114],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":89,"s":[{"i":[[135.076,57.333],[-186.567,-166]],"o":[[21.276,184.067],[131.267,-187.467]],"v":[[-10.876,-192.4],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":90,"s":[{"i":[[71.543,123],[0,-195]],"o":[[-14.457,76],[84,-25]],"v":[[124.457,-28],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":105,"s":[{"i":[[-25.457,62],[-96.5,-180]],"o":[[-66.457,59],[219,-96]],"v":[[82.457,-114],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":119,"s":[{"i":[[135.076,57.333],[-186.567,-166]],"o":[[21.276,184.067],[131.267,-187.467]],"v":[[-10.876,-192.4],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":120,"s":[{"i":[[71.543,123],[0,-195]],"o":[[-14.457,76],[84,-25]],"v":[[124.457,-28],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":135,"s":[{"i":[[-25.457,62],[-96.5,-180]],"o":[[-66.457,59],[219,-96]],"v":[[82.457,-114],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":149,"s":[{"i":[[135.076,57.333],[-186.567,-166]],"o":[[21.276,184.067],[131.267,-187.467]],"v":[[-10.876,-192.4],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":150,"s":[{"i":[[71.543,123],[0,-195]],"o":[[-14.457,76],[84,-25]],"v":[[124.457,-28],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":165,"s":[{"i":[[-25.457,62],[-96.5,-180]],"o":[[-66.457,59],[219,-96]],"v":[[82.457,-114],[-11,192]],"c":true}]},{"t":179,"s":[{"i":[[135.076,57.333],[-186.567,-166]],"o":[[21.276,184.067],[131.267,-187.467]],"v":[[-10.876,-192.4],[-11,192]],"c":true}]}]},"nm":"Path 6","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[-11.062,192.375]},"a":{"a":0,"k":[-11.062,191.875]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 6","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":59,"s":[{"i":[[16.876,-12.167],[-54,0]],"o":[[21.876,-0.167],[-12,-27]],"v":[[-63.376,164.167],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":60,"s":[{"i":[[16.876,-12.167],[-54,0]],"o":[[21.876,-0.167],[-12,-27]],"v":[[-63.376,164.167],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":74,"s":[{"i":[[33.676,11.167],[-54,0]],"o":[[16.276,22.233],[-8.733,-68.067]],"v":[[-128.243,148.767],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":75,"s":[{"i":[[16.876,-12.167],[-54,0]],"o":[[21.876,-0.167],[-12,-27]],"v":[[-63.376,164.167],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":89,"s":[{"i":[[16.876,-12.167],[-54,0]],"o":[[21.876,-0.167],[-12,-27]],"v":[[-63.376,164.167],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":90,"s":[{"i":[[16.876,-12.167],[-54,0]],"o":[[21.876,-0.167],[-12,-27]],"v":[[-63.376,164.167],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":104,"s":[{"i":[[33.676,11.167],[-54,0]],"o":[[16.276,22.233],[-8.733,-68.067]],"v":[[-128.243,148.767],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":105,"s":[{"i":[[16.876,-12.167],[-54,0]],"o":[[21.876,-0.167],[-12,-27]],"v":[[-63.376,164.167],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":119,"s":[{"i":[[16.876,-12.167],[-54,0]],"o":[[21.876,-0.167],[-12,-27]],"v":[[-63.376,164.167],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":120,"s":[{"i":[[16.876,-12.167],[-54,0]],"o":[[21.876,-0.167],[-12,-27]],"v":[[-63.376,164.167],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":134,"s":[{"i":[[33.676,11.167],[-54,0]],"o":[[16.276,22.233],[-8.733,-68.067]],"v":[[-128.243,148.767],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":135,"s":[{"i":[[16.876,-12.167],[-54,0]],"o":[[21.876,-0.167],[-12,-27]],"v":[[-63.376,164.167],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":149,"s":[{"i":[[16.876,-12.167],[-54,0]],"o":[[21.876,-0.167],[-12,-27]],"v":[[-63.376,164.167],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":150,"s":[{"i":[[16.876,-12.167],[-54,0]],"o":[[21.876,-0.167],[-12,-27]],"v":[[-63.376,164.167],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":164,"s":[{"i":[[33.676,11.167],[-54,0]],"o":[[16.276,22.233],[-8.733,-68.067]],"v":[[-128.243,148.767],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":165,"s":[{"i":[[16.876,-12.167],[-54,0]],"o":[[21.876,-0.167],[-12,-27]],"v":[[-63.376,164.167],[-11,192]],"c":true}]},{"t":179,"s":[{"i":[[16.876,-12.167],[-54,0]],"o":[[21.876,-0.167],[-12,-27]],"v":[[-63.376,164.167],[-11,192]],"c":true}]}]},"nm":"Path 5","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[-11.062,192.375]},"a":{"a":0,"k":[-11.062,191.875]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 5","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":59,"s":[{"i":[[33.676,11.167],[-54,0]],"o":[[16.276,22.233],[-8.733,-68.067]],"v":[[-128.243,148.767],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":60,"s":[{"i":[[34.876,12.833],[-54,0]],"o":[[15.876,23.833],[-8.5,-71]],"v":[[-132.876,147.667],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":74,"s":[{"i":[[-5.102,104.922],[-107.857,0]],"o":[[-59.568,63.656],[85.767,-205.867]],"v":[[-138.47,-15.108],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":75,"s":[{"i":[[16.876,-12.167],[-54,0]],"o":[[21.876,-0.167],[-12,-27]],"v":[[-63.376,164.167],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":89,"s":[{"i":[[33.676,11.167],[-54,0]],"o":[[16.276,22.233],[-8.733,-68.067]],"v":[[-128.243,148.767],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":90,"s":[{"i":[[34.876,12.833],[-54,0]],"o":[[15.876,23.833],[-8.5,-71]],"v":[[-132.876,147.667],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":104,"s":[{"i":[[-5.102,104.922],[-107.857,0]],"o":[[-59.568,63.656],[85.767,-205.867]],"v":[[-138.47,-15.108],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":105,"s":[{"i":[[16.876,-12.167],[-54,0]],"o":[[21.876,-0.167],[-12,-27]],"v":[[-63.376,164.167],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":119,"s":[{"i":[[33.676,11.167],[-54,0]],"o":[[16.276,22.233],[-8.733,-68.067]],"v":[[-128.243,148.767],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":120,"s":[{"i":[[34.876,12.833],[-54,0]],"o":[[15.876,23.833],[-8.5,-71]],"v":[[-132.876,147.667],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":134,"s":[{"i":[[-5.102,104.922],[-107.857,0]],"o":[[-59.568,63.656],[85.767,-205.867]],"v":[[-138.47,-15.108],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":135,"s":[{"i":[[16.876,-12.167],[-54,0]],"o":[[21.876,-0.167],[-12,-27]],"v":[[-63.376,164.167],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":149,"s":[{"i":[[33.676,11.167],[-54,0]],"o":[[16.276,22.233],[-8.733,-68.067]],"v":[[-128.243,148.767],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":150,"s":[{"i":[[34.876,12.833],[-54,0]],"o":[[15.876,23.833],[-8.5,-71]],"v":[[-132.876,147.667],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":164,"s":[{"i":[[-5.102,104.922],[-107.857,0]],"o":[[-59.568,63.656],[85.767,-205.867]],"v":[[-138.47,-15.108],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":165,"s":[{"i":[[16.876,-12.167],[-54,0]],"o":[[21.876,-0.167],[-12,-27]],"v":[[-63.376,164.167],[-11,192]],"c":true}]},{"t":179,"s":[{"i":[[33.676,11.167],[-54,0]],"o":[[16.276,22.233],[-8.733,-68.067]],"v":[[-131.119,149.304],[-11,192]],"c":true}]}]},"nm":"Path 4","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[-11.062,192.375]},"a":{"a":0,"k":[-11.062,191.875]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 4","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":59,"s":[{"i":[[-5.102,104.922],[-103.467,0]],"o":[[-71.702,70.189],[85.767,-205.867]],"v":[[-134.898,-15.822],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":60,"s":[{"i":[[-7.957,111.5],[-107,0]],"o":[[-77.957,73.5],[92.5,-215.5]],"v":[[-135.043,-27.5],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":74,"s":[{"i":[[-11.007,99.075],[-95.125,-121.625]],"o":[[-30.132,71.2],[92.5,-215.5]],"v":[[-62.993,-109.325],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":75,"s":[{"i":[[34.876,12.833],[-54,0]],"o":[[15.876,23.833],[-8.5,-71]],"v":[[-132.876,147.667],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":89,"s":[{"i":[[-5.102,104.922],[-103.467,0]],"o":[[-71.702,70.189],[85.767,-205.867]],"v":[[-134.898,-15.822],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":90,"s":[{"i":[[-7.957,111.5],[-107,0]],"o":[[-77.957,73.5],[92.5,-215.5]],"v":[[-135.043,-27.5],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":104,"s":[{"i":[[-11.007,99.075],[-95.125,-121.625]],"o":[[-30.132,71.2],[92.5,-215.5]],"v":[[-62.993,-109.325],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":105,"s":[{"i":[[34.876,12.833],[-54,0]],"o":[[15.876,23.833],[-8.5,-71]],"v":[[-132.876,147.667],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":119,"s":[{"i":[[-5.102,104.922],[-103.467,0]],"o":[[-71.702,70.189],[85.767,-205.867]],"v":[[-134.898,-15.822],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":120,"s":[{"i":[[-7.957,111.5],[-107,0]],"o":[[-77.957,73.5],[92.5,-215.5]],"v":[[-135.043,-27.5],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":134,"s":[{"i":[[-11.007,99.075],[-95.125,-121.625]],"o":[[-30.132,71.2],[92.5,-215.5]],"v":[[-62.993,-109.325],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":135,"s":[{"i":[[34.876,12.833],[-54,0]],"o":[[15.876,23.833],[-8.5,-71]],"v":[[-132.876,147.667],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":149,"s":[{"i":[[-5.102,104.922],[-103.467,0]],"o":[[-71.702,70.189],[85.767,-205.867]],"v":[[-134.898,-15.822],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":150,"s":[{"i":[[-7.957,111.5],[-107,0]],"o":[[-77.957,73.5],[92.5,-215.5]],"v":[[-135.043,-27.5],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":164,"s":[{"i":[[-11.007,99.075],[-95.125,-121.625]],"o":[[-30.132,71.2],[92.5,-215.5]],"v":[[-62.993,-109.325],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":165,"s":[{"i":[[34.876,12.833],[-54,0]],"o":[[15.876,23.833],[-8.5,-71]],"v":[[-132.876,147.667],[-11,192]],"c":true}]},{"t":179,"s":[{"i":[[-5.102,104.922],[-103.467,0]],"o":[[-76.007,69.57],[85.767,-205.867]],"v":[[-136.097,-24.959],[-11,192]],"c":true}]}]},"nm":"Path 3","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[-11.062,192.375]},"a":{"a":0,"k":[-11.062,191.875]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 3","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":59,"s":[{"i":[[-47.49,91.167],[-5.346,71.297]],"o":[[14.643,7.767],[18.687,238.863]],"v":[[-62.243,-108.7],[-16.324,-197.878]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":60,"s":[{"i":[[-48.707,88.5],[6.4,70.3]],"o":[[12.543,7.5],[24.9,235.3]],"v":[[-57.043,-114.5],[-17.368,-197.816]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":74,"s":[{"i":[[-6.84,0.7],[-1.405,1.232]],"o":[[-1.007,-1.383],[-2.946,1.899]],"v":[[-5.243,-193.367],[-2.606,-198.922]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":75,"s":[{"i":[[-30.457,128.5],[-169.792,85.25]],"o":[[44.043,11.5],[-68.292,288.75]],"v":[[-135.043,-27.5],[-1.708,-198.75]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":89,"s":[{"i":[[-47.49,91.167],[-5.346,71.297]],"o":[[14.643,7.767],[18.687,238.863]],"v":[[-62.243,-108.7],[-16.324,-197.878]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":90,"s":[{"i":[[-48.707,88.5],[6.4,70.3]],"o":[[12.543,7.5],[24.9,235.3]],"v":[[-57.043,-114.5],[-17.368,-197.816]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":104,"s":[{"i":[[-6.84,0.7],[-1.405,1.232]],"o":[[-1.007,-1.383],[-2.946,1.899]],"v":[[-5.243,-193.367],[-2.606,-198.922]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":105,"s":[{"i":[[-30.457,128.5],[-169.792,85.25]],"o":[[44.043,11.5],[-68.292,288.75]],"v":[[-135.043,-27.5],[-1.708,-198.75]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":119,"s":[{"i":[[-47.49,91.167],[-5.346,71.297]],"o":[[14.643,7.767],[18.687,238.863]],"v":[[-62.243,-108.7],[-16.324,-197.878]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":120,"s":[{"i":[[-48.707,88.5],[6.4,70.3]],"o":[[12.543,7.5],[24.9,235.3]],"v":[[-57.043,-114.5],[-17.368,-197.816]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":134,"s":[{"i":[[-6.84,0.7],[-1.405,1.232]],"o":[[-1.007,-1.383],[-2.946,1.899]],"v":[[-5.243,-193.367],[-2.606,-198.922]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":135,"s":[{"i":[[-30.457,128.5],[-169.792,85.25]],"o":[[44.043,11.5],[-68.292,288.75]],"v":[[-135.043,-27.5],[-1.708,-198.75]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":149,"s":[{"i":[[-47.49,91.167],[-5.346,71.297]],"o":[[14.643,7.767],[18.687,238.863]],"v":[[-62.243,-108.7],[-16.324,-197.878]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":150,"s":[{"i":[[-48.707,88.5],[6.4,70.3]],"o":[[12.543,7.5],[24.9,235.3]],"v":[[-57.043,-114.5],[-17.368,-197.816]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":164,"s":[{"i":[[-6.84,0.7],[-1.405,1.232]],"o":[[-1.007,-1.383],[-2.946,1.899]],"v":[[-5.243,-193.367],[-2.606,-198.922]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":165,"s":[{"i":[[-30.457,128.5],[-169.792,85.25]],"o":[[44.043,11.5],[-68.292,288.75]],"v":[[-135.043,-27.5],[-1.708,-198.75]],"c":true}]},{"t":179,"s":[{"i":[[-42.264,87.741],[1.921,63.246]],"o":[[23.157,5.509],[18.687,238.863]],"v":[[-57.69,-113],[-16.324,-197.878]],"c":true}]}]},"nm":"Path 2","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[-11.062,192.375]},"a":{"a":0,"k":[-11.062,191.875]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 9","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":59,"s":[{"i":[[-19.157,109.633],[-87.4,-117.6]],"o":[[-37.89,72.1],[92.5,-215.5]],"v":[[-62.243,-108.7],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":60,"s":[{"i":[[-19.957,109.5],[-86,-126]],"o":[[-35.957,72.5],[92.5,-215.5]],"v":[[-57.043,-114.5],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":74,"s":[{"i":[[-31.757,107.242],[-162.125,-256.25]],"o":[[-110.882,74.992],[178.833,-193.567]],"v":[[-5.243,-193.367],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":75,"s":[{"i":[[-7.957,111.5],[-107,0]],"o":[[-64.957,66.5],[92.5,-215.5]],"v":[[-135.043,-27.5],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":89,"s":[{"i":[[-19.157,109.633],[-87.4,-117.6]],"o":[[-37.89,72.1],[92.5,-215.5]],"v":[[-62.243,-108.7],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":90,"s":[{"i":[[-19.957,109.5],[-86,-126]],"o":[[-35.957,72.5],[92.5,-215.5]],"v":[[-57.043,-114.5],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":104,"s":[{"i":[[-31.757,107.242],[-162.125,-256.25]],"o":[[-110.882,74.992],[178.833,-193.567]],"v":[[-5.243,-193.367],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":105,"s":[{"i":[[-7.957,111.5],[-107,0]],"o":[[-64.957,66.5],[92.5,-215.5]],"v":[[-135.043,-27.5],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":119,"s":[{"i":[[-19.157,109.633],[-87.4,-117.6]],"o":[[-37.89,72.1],[92.5,-215.5]],"v":[[-62.243,-108.7],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":120,"s":[{"i":[[-19.957,109.5],[-86,-126]],"o":[[-35.957,72.5],[92.5,-215.5]],"v":[[-57.043,-114.5],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":134,"s":[{"i":[[-31.757,107.242],[-162.125,-256.25]],"o":[[-110.882,74.992],[178.833,-193.567]],"v":[[-5.243,-193.367],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":135,"s":[{"i":[[-7.957,111.5],[-107,0]],"o":[[-64.957,66.5],[92.5,-215.5]],"v":[[-135.043,-27.5],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":149,"s":[{"i":[[-19.157,109.633],[-87.4,-117.6]],"o":[[-37.89,72.1],[92.5,-215.5]],"v":[[-62.243,-108.7],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":150,"s":[{"i":[[-19.957,109.5],[-86,-126]],"o":[[-35.957,72.5],[92.5,-215.5]],"v":[[-57.043,-114.5],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":164,"s":[{"i":[[-31.757,107.242],[-162.125,-256.25]],"o":[[-110.882,74.992],[178.833,-193.567]],"v":[[-5.243,-193.367],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":165,"s":[{"i":[[-7.957,111.5],[-107,0]],"o":[[-64.957,66.5],[92.5,-215.5]],"v":[[-135.043,-27.5],[-11,192]],"c":true}]},{"t":179,"s":[{"i":[[-19.157,109.633],[-87.4,-117.6]],"o":[[-34.595,67.317],[92.5,-215.5]],"v":[[-57.69,-113],[-11,192]],"c":true}]}]},"nm":"Path 2","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[-11.062,192.375]},"a":{"a":0,"k":[-11.062,191.875]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 2","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":59,"s":[{"i":[[133.943,60.867],[-190.8,-171.267]],"o":[[17.01,185.067],[129,-193.867]],"v":[[-16.476,-198.067],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":60,"s":[{"i":[[146.543,57],[-193,-165]],"o":[[27.543,193],[125,-194]],"v":[[-17.543,-198],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":75,"s":[{"i":[[-42.457,115],[-160,-259]],"o":[[-130.457,74],[185,-192]],"v":[[-1.543,-199],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":89,"s":[{"i":[[133.943,60.867],[-190.8,-171.267]],"o":[[17.01,185.067],[129,-193.867]],"v":[[-16.476,-198.067],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":90,"s":[{"i":[[146.543,57],[-193,-165]],"o":[[27.543,193],[125,-194]],"v":[[-17.543,-198],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":105,"s":[{"i":[[-42.457,115],[-160,-259]],"o":[[-130.457,74],[185,-192]],"v":[[-1.543,-199],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":119,"s":[{"i":[[133.943,60.867],[-190.8,-171.267]],"o":[[17.01,185.067],[129,-193.867]],"v":[[-16.476,-198.067],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":120,"s":[{"i":[[146.543,57],[-193,-165]],"o":[[27.543,193],[125,-194]],"v":[[-17.543,-198],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":135,"s":[{"i":[[-42.457,115],[-160,-259]],"o":[[-130.457,74],[185,-192]],"v":[[-1.543,-199],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":149,"s":[{"i":[[133.943,60.867],[-190.8,-171.267]],"o":[[17.01,185.067],[129,-193.867]],"v":[[-16.476,-198.067],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":150,"s":[{"i":[[146.543,57],[-193,-165]],"o":[[27.543,193],[125,-194]],"v":[[-17.543,-198],[-11,192]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":165,"s":[{"i":[[-42.457,115],[-160,-259]],"o":[[-130.457,74],[185,-192]],"v":[[-1.543,-199],[-11,192]],"c":true}]},{"t":179,"s":[{"i":[[141.782,61.822],[-190.8,-171.267]],"o":[[17.01,185.067],[129,-193.867]],"v":[[-16.476,-198.067],[-11,192]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[-11.062,192.375]},"a":{"a":0,"k":[-11.062,191.875]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 1","bm":0,"hd":false}],"ip":60,"op":180,"st":-60,"bm":0},{"ddd":0,"ind":4,"ty":4,"nm":"Piece 4","parent":1,"sr":1,"ks":{"p":{"a":0,"k":[-11,222,0]},"a":{"a":0,"k":[-11,192,0]},"s":{"a":0,"k":[98.84,101.698,100]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":120,"s":[{"i":[[10.48,12.585],[-33.087,-22.551]],"o":[[-3.995,13.156],[23.136,-22.387]],"v":[[128.427,-21.321],[134.072,41.111]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":122,"s":[{"i":[[12.25,13.389],[-33.921,-22.103]],"o":[[-3.743,14.769],[22.497,-25.684]],"v":[[119.32,-43.981],[132.393,22.16]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":132,"s":[{"i":[[9.594,11.45],[-23.16,-13.705]],"o":[[-8.131,21.979],[23.699,-13.705]],"v":[[85.054,-164.743],[92.514,-104.173]],"c":true}]},{"t":144,"s":[{"i":[[2.816,3.361],[-9.455,-5.147]],"o":[[-2.387,6.452],[10.868,-4.166]],"v":[[75.998,-245.014],[78.188,-227.233]],"c":true}]}]},"nm":"Path 7","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[-11.062,192.375]},"a":{"a":0,"k":[-11.062,191.875]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 7","bm":0,"hd":false}],"ip":120,"op":145,"st":35,"bm":0},{"ddd":0,"ind":5,"ty":4,"nm":"Piece 3","parent":1,"sr":1,"ks":{"p":{"a":0,"k":[-11,222,0]},"a":{"a":0,"k":[-11,192,0]},"s":{"a":0,"k":[110.403,97.367,100]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":61,"s":[{"i":[[27.136,38.976],[-12.621,-31.262]],"o":[[-21.202,24.423],[37.371,-16.641]],"v":[[-133.771,-29.001],[-144.924,50.399]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":70,"s":[{"i":[[8.658,18.337],[-21.412,-22.32]],"o":[[-8.749,10.943],[20.557,-5.204]],"v":[[-123.117,-143.606],[-131.301,-87.647]],"c":true}]},{"t":78,"s":[{"i":[[3.391,9.518],[-10.686,-10.248]],"o":[[-4.212,5.533],[11.468,-1.997]],"v":[[-111.837,-221.363],[-116.47,-198.856]],"c":true}]}]},"nm":"Path 2","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[-11.062,192.375]},"a":{"a":0,"k":[-11.062,191.875]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 2","bm":0,"hd":false}],"ip":61,"op":79,"st":-61,"bm":0},{"ddd":0,"ind":6,"ty":4,"nm":"Piece 2","parent":1,"sr":1,"ks":{"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":84,"s":[-11,222,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":91,"s":[-0.474,227.262,0],"to":[0,0,0],"ti":[0,0,0]},{"t":104,"s":[-11,200.947,0]}]},"a":{"a":0,"k":[-11,192,0]},"s":{"a":0,"k":[102.5,101.5,100]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":84,"s":[{"i":[[58.734,99.094],[-11.201,-85.802],[-36.604,18.513]],"o":[[-22.389,106.542],[5.23,33.981],[25.831,-61.058]],"v":[[-12.209,-195.788],[-78.412,-17.643],[13.229,13.899]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":91,"s":[{"i":[[42.547,47.82],[-4.687,-26.202],[-10.699,6.86]],"o":[[-10.235,53.653],[10.717,19.948],[22.677,-23.215]],"v":[[-24.023,-245.345],[-38.713,-168.504],[2.215,-149.194]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":94,"s":[{"i":[[35.046,27.614],[-22.831,-23.088],[-11.72,3.749]],"o":[[-7.533,26.642],[5.028,3.435],[13.467,-22.774]],"v":[[-21.149,-259.133],[-24.024,-188.828],[3.409,-184.694]],"c":true}]},{"t":104,"s":[{"i":[[5.778,9.464],[-8.704,-9.785],[0.29,-0.255]],"o":[[-2.694,5.315],[0.305,-0.316],[7.97,-3.241]],"v":[[-19.654,-258.358],[-22.117,-233.924],[-22.102,-233.986]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[-11.062,192.375]},"a":{"a":0,"k":[-11.062,191.875]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 1","bm":0,"hd":false}],"ip":84,"op":105,"st":14,"bm":0}]} \ No newline at end of file diff --git a/TMessagesProj/src/main/res/raw/giveaway_results.json b/TMessagesProj/src/main/res/raw/giveaway_results.json new file mode 100644 index 00000000000..8f2d79a64cc --- /dev/null +++ b/TMessagesProj/src/main/res/raw/giveaway_results.json @@ -0,0 +1 @@ +{"tgs":1,"v":"5.5.2","fr":60,"ip":0,"op":180,"w":512,"h":512,"nm":"boomstick","ddd":0,"assets":[{"id":"comp_0","layers":[{"ddd":0,"ind":1,"ty":3,"nm":"rl 2","sr":1,"ks":{"o":{"a":0,"k":0},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[176,6,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":66,"s":[136,6,0],"to":[0,0,0],"ti":[0,0,0]},{"t":131,"s":[216,6,0]}]}},"ao":0,"ip":0,"op":103,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":3,"nm":"fall 2","parent":1,"sr":1,"ks":{"o":{"a":0,"k":0},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":0,"s":[0,-2,0],"to":[0,0,0],"ti":[0,0,0]},{"t":110,"s":[2,501,0]}]}},"ao":0,"ip":0,"op":103,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"cube pinky","parent":2,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"t":118,"s":[-385]}]},"a":{"a":0,"k":[-58.749,-145.954,0]},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":0,"s":[0,0,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":16,"s":[100,100,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":87,"s":[100,100,100]},{"t":107,"s":[0,0,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[-1.559,-0.779],[0,0],[0.779,-1.559],[0,0],[1.559,0.779],[0,0],[-0.779,1.559],[0,0]],"o":[[0,0],[1.559,0.779],[0,0],[-0.779,1.559],[0,0],[-1.559,-0.779],[0,0],[0.779,-1.559]],"v":[[-61.835,-162.247],[-46.149,-154.412],[-44.736,-150.178],[-52.571,-134.493],[-56.805,-133.08],[-72.49,-140.915],[-73.903,-145.149],[-66.068,-160.834]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.976470589638,0.345098048449,0.564705908298,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[-1.559,-0.779],[0,0],[0.779,-1.559],[0,0],[1.559,0.779],[0,0],[-0.779,1.559],[0,0]],"o":[[0,0],[1.559,0.779],[0,0],[-0.779,1.559],[0,0],[-1.559,-0.779],[0,0],[0.779,-1.559]],"v":[[-61.835,-162.247],[-43.869,-153.273],[-42.455,-149.039],[-51.43,-131.073],[-55.663,-129.66],[-73.629,-138.635],[-75.042,-142.868],[-66.068,-160.834]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.807843148708,0.098039217293,0.301960796118,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 3","bm":0,"hd":false}],"ip":0,"op":103,"st":0,"bm":0}]},{"id":"comp_1","layers":[{"ddd":0,"ind":1,"ty":3,"nm":"rl 3","sr":1,"ks":{"o":{"a":0,"k":0},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[176,6,0],"to":[0,0,0],"ti":[0,0,0]},{"t":77,"s":[276,6,0]}]}},"ao":0,"ip":0,"op":103,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":3,"nm":"fall 3","parent":1,"sr":1,"ks":{"o":{"a":0,"k":0},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":0,"s":[0,-2,0],"to":[0,0,0],"ti":[0,0,0]},{"t":110,"s":[2,501,0]}]}},"ao":0,"ip":0,"op":103,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"cube yellow","parent":2,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"t":108,"s":[551]}]},"p":{"a":0,"k":[80,0,0]},"a":{"a":0,"k":[159.318,54.557,0]},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":0,"s":[0,0,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":16,"s":[100,100,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":87,"s":[100,100,100]},{"t":107,"s":[0,0,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[-0.762,-1.568],[0,0],[1.568,-0.762],[0,0],[0.762,1.568],[0,0],[-1.568,0.762],[0,0]],"o":[[0,0],[0.762,1.568],[0,0],[-1.568,0.762],[0,0],[-0.762,-1.568],[0,0],[1.568,-0.762]],"v":[[166.802,39.758],[174.463,55.53],[173.003,59.748],[157.232,67.409],[153.014,65.949],[145.353,50.178],[146.813,45.96],[162.584,38.299]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.972549021244,0.96862745285,0.290196090937,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[-0.762,-1.568],[0,0],[1.568,-0.762],[0,0],[0.762,1.568],[0,0],[-1.568,0.762],[0,0]],"o":[[0,0],[0.762,1.568],[0,0],[-1.568,0.762],[0,0],[-0.762,-1.568],[0,0],[1.568,-0.762]],"v":[[166.802,39.758],[175.577,57.822],[174.117,62.04],[156.053,70.815],[151.835,69.356],[143.06,51.291],[144.52,47.074],[162.584,38.299]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.737254917622,0.737254917622,0.258823543787,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 8","bm":0,"hd":false}],"ip":0,"op":103,"st":0,"bm":0}]},{"id":"comp_2","layers":[{"ddd":0,"ind":1,"ty":3,"nm":"rl 4","sr":1,"ks":{"o":{"a":0,"k":0},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[176,6,0],"to":[0,0,0],"ti":[0,0,0]},{"t":118,"s":[70,8,0]}]}},"ao":0,"ip":0,"op":103,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":3,"nm":"fall 4","parent":1,"sr":1,"ks":{"o":{"a":0,"k":0},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":0,"s":[0,-2,0],"to":[0,0,0],"ti":[0,0,0]},{"t":110,"s":[2,501,0]}]}},"ao":0,"ip":0,"op":103,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"cube red","parent":2,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"t":117,"s":[-115]}]},"p":{"a":0,"k":[80,0,0]},"a":{"a":0,"k":[-158.763,14.8,0]},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":0,"s":[0,0,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":16,"s":[100,100,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":87,"s":[100,100,100]},{"t":107,"s":[0,0,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[1.055,-1.387],[0,0],[1.387,1.055],[0,0],[-1.055,1.387],[0,0],[-1.387,-1.055],[0,0]],"o":[[0,0],[-1.055,1.387],[0,0],[-1.387,-1.055],[0,0],[1.055,-1.387],[0,0],[1.387,1.055]],"v":[[-142.18,14.8],[-152.797,28.754],[-157.22,29.355],[-171.173,18.737],[-171.774,14.315],[-161.157,0.361],[-156.734,-0.24],[-142.78,10.378]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.839215695858,0.184313729405,0.345098048449,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[1.055,-1.387],[0,0],[1.387,1.055],[0,0],[-1.055,1.387],[0,0],[-1.387,-1.055],[0,0]],"o":[[0,0],[-1.055,1.387],[0,0],[-1.387,-1.055],[0,0],[1.055,-1.387],[0,0],[1.387,1.055]],"v":[[-142.18,14.8],[-154.341,30.783],[-158.763,31.383],[-174.745,19.222],[-175.346,14.8],[-163.185,-1.183],[-158.762,-1.783],[-142.78,10.378]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.807843148708,0.098039217293,0.301960796118,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 6","bm":0,"hd":false}],"ip":0,"op":103,"st":0,"bm":0}]},{"id":"comp_3","layers":[{"ddd":0,"ind":1,"ty":3,"nm":"rl 5","sr":1,"ks":{"o":{"a":0,"k":0},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[176,6,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":66,"s":[70,8,0],"to":[0,0,0],"ti":[0,0,0]},{"t":136,"s":[232,8,0]}]}},"ao":0,"ip":0,"op":103,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":3,"nm":"fall 5","parent":1,"sr":1,"ks":{"o":{"a":0,"k":0},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":0,"s":[0,-2,0],"to":[0,0,0],"ti":[0,0,0]},{"t":110,"s":[2,501,0]}]}},"ao":0,"ip":0,"op":103,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"cube blue","parent":2,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":0,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":48,"s":[-68]},{"t":125,"s":[72]}]},"p":{"a":0,"k":[80,0,0]},"a":{"a":0,"k":[12.04,-164.683,0]},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":0,"s":[0,0,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":16,"s":[100,100,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":87,"s":[100,100,100]},{"t":107,"s":[0,0,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[1.541,-0.815],[0,0],[0.815,1.541],[0,0],[-1.541,0.815],[0,0],[-0.815,-1.541],[0,0]],"o":[[0,0],[-1.541,0.815],[0,0],[-0.815,-1.541],[0,0],[1.541,-0.815],[0,0],[0.815,1.541]],"v":[[27.089,-157.715],[11.592,-149.513],[7.326,-150.827],[-0.876,-166.324],[0.437,-170.589],[15.934,-178.791],[20.2,-177.478],[28.402,-161.981]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.172549024224,0.403921574354,0.917647063732,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[1.541,-0.815],[0,0],[0.815,1.541],[0,0],[-1.541,0.815],[0,0],[-0.815,-1.541],[0,0]],"o":[[0,0],[-1.541,0.815],[0,0],[-0.815,-1.541],[0,0],[1.541,-0.815],[0,0],[0.815,1.541]],"v":[[27.089,-157.715],[9.339,-148.321],[5.073,-149.634],[-4.321,-167.384],[-3.008,-171.65],[14.742,-181.044],[19.008,-179.731],[28.402,-161.981]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.0941176489,0.301960796118,0.607843160629,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 10","bm":0,"hd":false}],"ip":0,"op":103,"st":0,"bm":0}]},{"id":"comp_4","layers":[{"ddd":0,"ind":1,"ty":3,"nm":"rl","sr":1,"ks":{"o":{"a":0,"k":0},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[256,6,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":66,"s":[216,6,0],"to":[0,0,0],"ti":[0,0,0]},{"t":131,"s":[296,6,0]}]}},"ao":0,"ip":0,"op":103,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":3,"nm":"fall","parent":1,"sr":1,"ks":{"o":{"a":0,"k":0},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":0,"s":[0,-2,0],"to":[0,0,0],"ti":[0,0,0]},{"t":110,"s":[2,501,0]}]}},"ao":0,"ip":0,"op":103,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"cube orange","parent":2,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"t":114,"s":[235]}]},"a":{"a":0,"k":[165.82,119.688,0]},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":0,"s":[0,0,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":16,"s":[100,100,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":87,"s":[100,100,100]},{"t":107,"s":[0,0,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[0.758,-1.57],[0,0],[1.57,0.758],[0,0],[-0.758,1.57],[0,0],[-1.57,-0.758],[0,0]],"o":[[0,0],[-0.758,1.57],[0,0],[-1.57,-0.758],[0,0],[0.758,-1.57],[0,0],[1.57,0.758]],"v":[[182.071,116.386],[174.445,132.174],[170.231,133.643],[154.442,126.018],[152.973,121.803],[160.598,106.015],[164.813,104.545],[180.601,112.171]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.988235294819,0.596078455448,0.149019613862,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[0.758,-1.57],[0,0],[1.57,0.758],[0,0],[-0.758,1.57],[0,0],[-1.57,-0.758],[0,0]],"o":[[0,0],[-0.758,1.57],[0,0],[-1.57,-0.758],[0,0],[0.758,-1.57],[0,0],[1.57,0.758]],"v":[[182.071,116.386],[173.337,134.469],[169.122,135.939],[151.038,127.205],[149.569,122.99],[158.303,104.906],[162.518,103.437],[180.601,112.171]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.819607853889,0.537254929543,0.152941182256,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false}],"ip":0,"op":103,"st":0,"bm":0}]}],"layers":[{"ddd":0,"ind":2,"ty":3,"nm":"NULL SCALE ALL","sr":1,"ks":{"o":{"a":0,"k":0},"p":{"a":0,"k":[261,239,0]},"s":{"a":1,"k":[{"i":{"x":[0.1,0.1,0.1],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,1.136]},"t":74,"s":[108,108,100]},{"i":{"x":[0.12,0.12,0.12],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":84,"s":[105,105,100]},{"t":179,"s":[108,108,100]}]}},"ao":0,"ip":0,"op":180,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"cube blue 5","parent":2,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.2],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":99,"s":[0]},{"t":179,"s":[-67]}]},"p":{"a":1,"k":[{"i":{"x":0.2,"y":1},"o":{"x":0.31,"y":0},"t":99,"s":[-23.65,-220.624,0],"to":[0,0,0],"ti":[0,0,0]},{"t":179,"s":[-79.875,-44.5,0]}]},"a":{"a":0,"k":[12.04,-164.683,0]},"s":{"a":1,"k":[{"i":{"x":[0.4,0.4,0.4],"y":[1,1,1]},"o":{"x":[0.31,0.31,0.31],"y":[0,0,0]},"t":99,"s":[0,0,100]},{"t":115,"s":[100,100,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[1.541,-0.815],[0,0],[0.815,1.541],[0,0],[-1.541,0.815],[0,0],[-0.815,-1.541],[0,0]],"o":[[0,0],[-1.541,0.815],[0,0],[-0.815,-1.541],[0,0],[1.541,-0.815],[0,0],[0.815,1.541]],"v":[[27.089,-157.715],[11.592,-149.513],[7.326,-150.827],[-0.876,-166.324],[0.437,-170.589],[15.934,-178.791],[20.2,-177.478],[28.402,-161.981]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.172549024224,0.403921574354,0.917647063732,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[1.541,-0.815],[0,0],[0.815,1.541],[0,0],[-1.541,0.815],[0,0],[-0.815,-1.541],[0,0]],"o":[[0,0],[-1.541,0.815],[0,0],[-0.815,-1.541],[0,0],[1.541,-0.815],[0,0],[0.815,1.541]],"v":[[27.089,-157.715],[9.339,-148.321],[5.073,-149.634],[-4.321,-167.384],[-3.008,-171.65],[14.742,-181.044],[19.008,-179.731],[28.402,-161.981]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.0941176489,0.301960796118,0.607843160629,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 10","bm":0,"hd":false}],"ip":99,"op":237,"st":134,"bm":0},{"ddd":0,"ind":4,"ty":4,"nm":"cube red 5","parent":2,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.2],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":127,"s":[0]},{"t":179,"s":[-14]}]},"p":{"a":1,"k":[{"i":{"x":0.2,"y":1},"o":{"x":0.31,"y":0},"t":127,"s":[-94.196,-229.91,0],"to":[0,0,0],"ti":[0,0,0]},{"t":179,"s":[-103.25,-183.241,0]}]},"a":{"a":0,"k":[-158.763,14.8,0]},"s":{"a":1,"k":[{"i":{"x":[0.4,0.4,0.4],"y":[1,1,1]},"o":{"x":[0.31,0.31,0.31],"y":[0,0,0]},"t":127,"s":[0,0,100]},{"t":179,"s":[80,80,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[1.055,-1.387],[0,0],[1.387,1.055],[0,0],[-1.055,1.387],[0,0],[-1.387,-1.055],[0,0]],"o":[[0,0],[-1.055,1.387],[0,0],[-1.387,-1.055],[0,0],[1.055,-1.387],[0,0],[1.387,1.055]],"v":[[-142.18,14.8],[-152.797,28.754],[-157.22,29.355],[-171.173,18.737],[-171.774,14.315],[-161.157,0.361],[-156.734,-0.24],[-142.78,10.378]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.839215695858,0.184313729405,0.345098048449,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[1.055,-1.387],[0,0],[1.387,1.055],[0,0],[-1.055,1.387],[0,0],[-1.387,-1.055],[0,0]],"o":[[0,0],[-1.055,1.387],[0,0],[-1.387,-1.055],[0,0],[1.055,-1.387],[0,0],[1.387,1.055]],"v":[[-142.18,14.8],[-154.341,30.783],[-158.763,31.383],[-174.745,19.222],[-175.346,14.8],[-163.185,-1.183],[-158.762,-1.783],[-142.78,10.378]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.807843148708,0.098039217293,0.301960796118,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 6","bm":0,"hd":false}],"ip":138,"op":269,"st":166,"bm":0},{"ddd":0,"ind":5,"ty":4,"nm":"cube yellow 5","parent":2,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.2],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":102,"s":[-219]},{"t":179,"s":[-166]}]},"p":{"a":1,"k":[{"i":{"x":0.2,"y":1},"o":{"x":0.31,"y":0},"t":102,"s":[67.765,-216.675,0],"to":[0,0,0],"ti":[0,0,0]},{"t":179,"s":[110.125,-77.625,0]}]},"a":{"a":0,"k":[159.318,54.557,0]},"s":{"a":1,"k":[{"i":{"x":[0.4,0.4,0.4],"y":[1,1,1]},"o":{"x":[0.31,0.31,0.31],"y":[0,0,0]},"t":102,"s":[0,0,100]},{"t":118,"s":[100,100,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[-0.762,-1.568],[0,0],[1.568,-0.762],[0,0],[0.762,1.568],[0,0],[-1.568,0.762],[0,0]],"o":[[0,0],[0.762,1.568],[0,0],[-1.568,0.762],[0,0],[-0.762,-1.568],[0,0],[1.568,-0.762]],"v":[[166.802,39.758],[174.463,55.53],[173.003,59.748],[157.232,67.409],[153.014,65.949],[145.353,50.178],[146.813,45.96],[162.584,38.299]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.972549021244,0.96862745285,0.290196090937,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[-0.762,-1.568],[0,0],[1.568,-0.762],[0,0],[0.762,1.568],[0,0],[-1.568,0.762],[0,0]],"o":[[0,0],[0.762,1.568],[0,0],[-1.568,0.762],[0,0],[-0.762,-1.568],[0,0],[1.568,-0.762]],"v":[[166.802,39.758],[175.577,57.822],[174.117,62.04],[156.053,70.815],[151.835,69.356],[143.06,51.291],[144.52,47.074],[162.584,38.299]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.737254917622,0.737254917622,0.258823543787,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 8","bm":0,"hd":false}],"ip":106,"op":244,"st":141,"bm":0},{"ddd":0,"ind":6,"ty":4,"nm":"cube pinky 4","parent":2,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.2],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":106,"s":[-65]},{"t":179,"s":[-112]}]},"p":{"a":1,"k":[{"i":{"x":0.2,"y":1},"o":{"x":0.31,"y":0},"t":103,"s":[-142.855,-222.223,0],"to":[0,0,0],"ti":[0,0,0]},{"t":179,"s":[-160,-96.5,0]}]},"a":{"a":0,"k":[-58.749,-145.954,0]},"s":{"a":1,"k":[{"i":{"x":[0.4,0.4,0.4],"y":[1,1,1]},"o":{"x":[0.31,0.31,0.31],"y":[0,0,0]},"t":105,"s":[0,0,100]},{"t":121,"s":[100,100,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[-1.559,-0.779],[0,0],[0.779,-1.559],[0,0],[1.559,0.779],[0,0],[-0.779,1.559],[0,0]],"o":[[0,0],[1.559,0.779],[0,0],[-0.779,1.559],[0,0],[-1.559,-0.779],[0,0],[0.779,-1.559]],"v":[[-61.835,-162.247],[-46.149,-154.412],[-44.736,-150.178],[-52.571,-134.493],[-56.805,-133.08],[-72.49,-140.915],[-73.903,-145.149],[-66.068,-160.834]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.976470589638,0.345098048449,0.564705908298,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[-1.559,-0.779],[0,0],[0.779,-1.559],[0,0],[1.559,0.779],[0,0],[-0.779,1.559],[0,0]],"o":[[0,0],[1.559,0.779],[0,0],[-0.779,1.559],[0,0],[-1.559,-0.779],[0,0],[0.779,-1.559]],"v":[[-61.835,-162.247],[-43.869,-153.273],[-42.455,-149.039],[-51.43,-131.073],[-55.663,-129.66],[-73.629,-138.635],[-75.042,-142.868],[-66.068,-160.834]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.807843148708,0.098039217293,0.301960796118,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 3","bm":0,"hd":false}],"ip":103,"op":248,"st":145,"bm":0},{"ddd":0,"ind":7,"ty":4,"nm":"cube orange 4","parent":2,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.2],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":112,"s":[-16]},{"t":179,"s":[58]}]},"p":{"a":1,"k":[{"i":{"x":0.2,"y":1},"o":{"x":0.31,"y":0},"t":110,"s":[121.25,-243.25,0],"to":[0,0,0],"ti":[0,0,0]},{"t":179,"s":[105.25,-124.25,0]}]},"a":{"a":0,"k":[165.82,119.688,0]},"s":{"a":1,"k":[{"i":{"x":[0.4,0.4,0.4],"y":[1,1,1]},"o":{"x":[0.31,0.31,0.31],"y":[0,0,0]},"t":112,"s":[0,0,100]},{"t":167,"s":[100,100,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[0.758,-1.57],[0,0],[1.57,0.758],[0,0],[-0.758,1.57],[0,0],[-1.57,-0.758],[0,0]],"o":[[0,0],[-0.758,1.57],[0,0],[-1.57,-0.758],[0,0],[0.758,-1.57],[0,0],[1.57,0.758]],"v":[[182.071,116.386],[174.445,132.174],[170.231,133.643],[154.442,126.018],[152.973,121.803],[160.598,106.015],[164.813,104.545],[180.601,112.171]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.988235294819,0.596078455448,0.149019613862,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[0.758,-1.57],[0,0],[1.57,0.758],[0,0],[-0.758,1.57],[0,0],[-1.57,-0.758],[0,0]],"o":[[0,0],[-0.758,1.57],[0,0],[-1.57,-0.758],[0,0],[0.758,-1.57],[0,0],[1.57,0.758]],"v":[[182.071,116.386],[173.337,134.469],[169.122,135.939],[151.038,127.205],[149.569,122.99],[158.303,104.906],[162.518,103.437],[180.601,112.171]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.819607853889,0.537254929543,0.152941182256,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false}],"ip":124,"op":254,"st":151,"bm":0},{"ddd":0,"ind":8,"ty":4,"nm":"cube yellow 4","parent":2,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.2],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":119,"s":[0]},{"t":179,"s":[-145]}]},"p":{"a":1,"k":[{"i":{"x":0.2,"y":1},"o":{"x":0.31,"y":0},"t":117,"s":[55.007,-214.337,0],"to":[0,0,0],"ti":[0,0,0]},{"t":179,"s":[41.25,-155.875,0]}]},"a":{"a":0,"k":[159.318,54.557,0]},"s":{"a":1,"k":[{"i":{"x":[0.4,0.4,0.4],"y":[1,1,1]},"o":{"x":[0.31,0.31,0.31],"y":[0,0,0]},"t":118,"s":[0,0,100]},{"t":138,"s":[100,100,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[-0.762,-1.568],[0,0],[1.568,-0.762],[0,0],[0.762,1.568],[0,0],[-1.568,0.762],[0,0]],"o":[[0,0],[0.762,1.568],[0,0],[-1.568,0.762],[0,0],[-0.762,-1.568],[0,0],[1.568,-0.762]],"v":[[166.802,39.758],[174.463,55.53],[173.003,59.748],[157.232,67.409],[153.014,65.949],[145.353,50.178],[146.813,45.96],[162.584,38.299]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.972549021244,0.96862745285,0.290196090937,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[-0.762,-1.568],[0,0],[1.568,-0.762],[0,0],[0.762,1.568],[0,0],[-1.568,0.762],[0,0]],"o":[[0,0],[0.762,1.568],[0,0],[-1.568,0.762],[0,0],[-0.762,-1.568],[0,0],[1.568,-0.762]],"v":[[166.802,39.758],[175.577,57.822],[174.117,62.04],[156.053,70.815],[151.835,69.356],[143.06,51.291],[144.52,47.074],[162.584,38.299]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.737254917622,0.737254917622,0.258823543787,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 8","bm":0,"hd":false}],"ip":116,"op":261,"st":158,"bm":0},{"ddd":0,"ind":9,"ty":4,"nm":"cube pinky","parent":2,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.2],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":126,"s":[0]},{"t":179,"s":[82]}]},"p":{"a":1,"k":[{"i":{"x":0.2,"y":1},"o":{"x":0.31,"y":0},"t":124,"s":[2.074,-234.746,0],"to":[0,0,0],"ti":[0,0,0]},{"t":179,"s":[24.5,-188,0]}]},"a":{"a":0,"k":[-58.749,-145.954,0]},"s":{"a":1,"k":[{"i":{"x":[0.4,0.4,0.4],"y":[1,1,1]},"o":{"x":[0.31,0.31,0.31],"y":[0,0,0]},"t":126,"s":[0,0,100]},{"t":179,"s":[89,89,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[-1.559,-0.779],[0,0],[0.779,-1.559],[0,0],[1.559,0.779],[0,0],[-0.779,1.559],[0,0]],"o":[[0,0],[1.559,0.779],[0,0],[-0.779,1.559],[0,0],[-1.559,-0.779],[0,0],[0.779,-1.559]],"v":[[-61.835,-162.247],[-46.149,-154.412],[-44.736,-150.178],[-52.571,-134.493],[-56.805,-133.08],[-72.49,-140.915],[-73.903,-145.149],[-66.068,-160.834]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.976470589638,0.345098048449,0.564705908298,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[-1.559,-0.779],[0,0],[0.779,-1.559],[0,0],[1.559,0.779],[0,0],[-0.779,1.559],[0,0]],"o":[[0,0],[1.559,0.779],[0,0],[-0.779,1.559],[0,0],[-1.559,-0.779],[0,0],[0.779,-1.559]],"v":[[-61.835,-162.247],[-43.869,-153.273],[-42.455,-149.039],[-51.43,-131.073],[-55.663,-129.66],[-73.629,-138.635],[-75.042,-142.868],[-66.068,-160.834]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.807843148708,0.098039217293,0.301960796118,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 3","bm":0,"hd":false}],"ip":139,"op":188,"st":165,"bm":0},{"ddd":0,"ind":10,"ty":4,"nm":"red 2","parent":2,"sr":1,"ks":{"o":{"a":1,"k":[{"t":0,"s":[100],"h":1},{"t":35,"s":[0],"h":1},{"t":116,"s":[100],"h":1}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[0,0],[131.695,3.355],[50.863,-19.845],[-24.402,-18.879],[62.018,-68.887],[-15.209,-21.02]],"o":[[0,0],[-52.28,-1.332],[-74.021,28.88],[24.216,18.735],[-57.31,63.657],[10.153,14.033]],"v":[[186.378,-220.364],[150.138,-113.293],[167.021,-55.88],[165.868,-2.012],[162.31,115.343],[172.209,230.02]],"c":false}},"nm":"Path 1","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":0,"s":[27.419]},{"i":{"x":[0.69],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":45,"s":[100]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":127,"s":[4]},{"t":179,"s":[27.419]}]},"e":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":0,"s":[58.065]},{"i":{"x":[0.69],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":26,"s":[100]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":101,"s":[0]},{"t":179,"s":[58.065]}]},"o":{"a":0,"k":0},"m":1,"nm":"Trim Paths 1","hd":false},{"ty":"st","c":{"a":0,"k":[0.98431372549,0.286274509804,0.517647058824,1]},"o":{"a":0,"k":100},"w":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":0,"s":[15]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":26,"s":[15]},{"i":{"x":[0.69],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":45,"s":[0]},{"i":{"x":[0.69],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":108,"s":[0]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":127,"s":[15]},{"t":179,"s":[15]}]},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 2","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[0,0],[124.956,3.082],[50.863,-19.845],[-24.402,-18.879],[62.018,-68.887],[-15.209,-21.02]],"o":[[0,0],[-52.281,-1.289],[-74.021,28.88],[24.216,18.735],[-57.31,63.657],[10.153,14.033]],"v":[[186.378,-220.364],[160.138,-109.293],[174.021,-52.88],[170.868,-6.012],[162.31,117.343],[172.209,230.02]],"c":false}},"nm":"Path 1","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":0,"s":[29]},{"i":{"x":[0.69],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":45,"s":[100]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":127,"s":[4]},{"t":179,"s":[29]}]},"e":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":0,"s":[58.065]},{"i":{"x":[0.69],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":26,"s":[100]},{"i":{"x":[0.673],"y":[0.495]},"o":{"x":[0.397],"y":[0]},"t":101,"s":[0]},{"i":{"x":[0.484],"y":[1]},"o":{"x":[0.219],"y":[0.478]},"t":127,"s":[24]},{"t":179,"s":[58.065]}]},"o":{"a":0,"k":0},"m":1,"nm":"Trim Paths 1","hd":false},{"ty":"st","c":{"a":0,"k":[0.756862745098,0.129411764706,0.290196078431,1]},"o":{"a":0,"k":100},"w":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":0,"s":[15]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":18,"s":[15]},{"i":{"x":[0.69],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":42,"s":[0]},{"i":{"x":[0.69],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":108,"s":[0]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":127,"s":[15]},{"t":179,"s":[15]}]},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 3","bm":0,"hd":false}],"ip":0,"op":180,"st":45,"bm":0},{"ddd":0,"ind":11,"ty":4,"nm":"red","parent":2,"sr":1,"ks":{},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[0,0],[146.924,4.823],[3.791,-21.02]],"o":[[0,0],[-145.802,-4.786],[-3.074,17.045]],"v":[[-7,-9.5],[-64.13,-204.527],[-185.791,-76.98]],"c":false}},"nm":"Path 1","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.34],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":86,"s":[0]},{"t":116,"s":[100]}]},"e":{"a":1,"k":[{"i":{"x":[0.34],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":78,"s":[0]},{"t":107,"s":[100]}]},"o":{"a":0,"k":0},"m":1,"nm":"Trim Paths 1","hd":false},{"ty":"st","c":{"a":0,"k":[0.964705882353,0.109803921569,0.333333333333,1]},"o":{"a":0,"k":100},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":78,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":95,"s":[15]},{"t":116,"s":[0]}]},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[0,0],[146.924,4.823],[3.791,-21.02]],"o":[[0,0],[-145.802,-4.786],[-3.074,17.045]],"v":[[-6,-12],[-62.862,-199.293],[-185.791,-76.98]],"c":false}},"nm":"Path 1","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.34],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":86,"s":[0]},{"t":116,"s":[100]}]},"e":{"a":1,"k":[{"i":{"x":[0.34],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":78,"s":[0]},{"t":107,"s":[100]}]},"o":{"a":0,"k":0},"m":1,"nm":"Trim Paths 1","hd":false},{"ty":"st","c":{"a":0,"k":[0.807843137255,0.098039215686,0.301960784314,1]},"o":{"a":0,"k":100},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":78,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":95,"s":[15]},{"t":116,"s":[0]}]},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 2","bm":0,"hd":false}],"ip":78,"op":117,"st":6,"bm":0},{"ddd":0,"ind":12,"ty":4,"nm":"blue","parent":2,"sr":1,"ks":{},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[0,0],[-16.123,-146.117],[6,-116]],"o":[[0,0],[16,145],[-4.865,94.066]],"v":[[29,28.5],[198,23],[112,240]],"c":false}},"nm":"Path 1","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.34],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":86,"s":[0]},{"t":116,"s":[100]}]},"e":{"a":1,"k":[{"i":{"x":[0.34],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":78,"s":[0]},{"t":107,"s":[100]}]},"o":{"a":0,"k":0},"m":1,"nm":"Trim Paths 1","hd":false},{"ty":"st","c":{"a":0,"k":[0.282352941176,0.462745098039,0.996078431373,1]},"o":{"a":0,"k":100},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":78,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":95,"s":[15]},{"t":116,"s":[0]}]},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[0,0],[-16.123,-146.117],[6,-116]],"o":[[0,0],[16,145],[-4.865,94.066]],"v":[[30,26],[203,21],[112,240]],"c":false}},"nm":"Path 1","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.34],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":86,"s":[0]},{"t":116,"s":[100]}]},"e":{"a":1,"k":[{"i":{"x":[0.34],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":78,"s":[0]},{"t":107,"s":[100]}]},"o":{"a":0,"k":0},"m":1,"nm":"Trim Paths 1","hd":false},{"ty":"st","c":{"a":0,"k":[0.203921571374,0.349019616842,0.952941179276,1]},"o":{"a":0,"k":100},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":78,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":95,"s":[15]},{"t":116,"s":[0]}]},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 2","bm":0,"hd":false}],"ip":78,"op":117,"st":6,"bm":0},{"ddd":0,"ind":13,"ty":4,"nm":"cube orange 3","parent":39,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":82,"s":[190.298]},{"t":102,"s":[18.298]}]},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":82,"s":[56.488,37.669,0],"to":[0,0,0],"ti":[0,0,0]},{"t":102,"s":[324.523,-2.914,0]}]},"a":{"a":0,"k":[165.82,119.688,0]},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":82,"s":[100,100,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":87,"s":[100,100,100]},{"t":102,"s":[0,0,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[0.758,-1.57],[0,0],[1.57,0.758],[0,0],[-0.758,1.57],[0,0],[-1.57,-0.758],[0,0]],"o":[[0,0],[-0.758,1.57],[0,0],[-1.57,-0.758],[0,0],[0.758,-1.57],[0,0],[1.57,0.758]],"v":[[182.071,116.386],[174.445,132.174],[170.231,133.643],[154.442,126.018],[152.973,121.803],[160.598,106.015],[164.813,104.545],[180.601,112.171]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.988235294819,0.596078455448,0.149019613862,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[0.758,-1.57],[0,0],[1.57,0.758],[0,0],[-0.758,1.57],[0,0],[-1.57,-0.758],[0,0]],"o":[[0,0],[-0.758,1.57],[0,0],[-1.57,-0.758],[0,0],[0.758,-1.57],[0,0],[1.57,0.758]],"v":[[182.071,116.386],[173.337,134.469],[169.122,135.939],[151.038,127.205],[149.569,122.99],[158.303,104.906],[162.518,103.437],[180.601,112.171]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.819607853889,0.537254929543,0.152941182256,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false}],"ip":82,"op":102,"st":12,"bm":0},{"ddd":0,"ind":14,"ty":4,"nm":"cube pinky 3","parent":39,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":81,"s":[-182.39]},{"t":101,"s":[37.61]}]},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":81,"s":[49.386,-41.663,0],"to":[0,0,0],"ti":[0,0,0]},{"t":101,"s":[229.281,-97.221,0]}]},"a":{"a":0,"k":[-58.749,-145.954,0]},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":81,"s":[100,100,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":86,"s":[100,100,100]},{"t":101,"s":[0,0,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[-1.559,-0.779],[0,0],[0.779,-1.559],[0,0],[1.559,0.779],[0,0],[-0.779,1.559],[0,0]],"o":[[0,0],[1.559,0.779],[0,0],[-0.779,1.559],[0,0],[-1.559,-0.779],[0,0],[0.779,-1.559]],"v":[[-61.835,-162.247],[-46.149,-154.412],[-44.736,-150.178],[-52.571,-134.493],[-56.805,-133.08],[-72.49,-140.915],[-73.903,-145.149],[-66.068,-160.834]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.976470589638,0.345098048449,0.564705908298,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[-1.559,-0.779],[0,0],[0.779,-1.559],[0,0],[1.559,0.779],[0,0],[-0.779,1.559],[0,0]],"o":[[0,0],[1.559,0.779],[0,0],[-0.779,1.559],[0,0],[-1.559,-0.779],[0,0],[0.779,-1.559]],"v":[[-61.835,-162.247],[-43.869,-153.273],[-42.455,-149.039],[-51.43,-131.073],[-55.663,-129.66],[-73.629,-138.635],[-75.042,-142.868],[-66.068,-160.834]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.807843148708,0.098039217293,0.301960796118,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 3","bm":0,"hd":false}],"ip":81,"op":101,"st":11,"bm":0},{"ddd":0,"ind":15,"ty":4,"nm":"cube yellow 3","parent":39,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":80,"s":[403.13]},{"t":100,"s":[18.13]}]},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":80,"s":[39.167,6.777,0],"to":[0,0,0],"ti":[0,0,0]},{"t":100,"s":[261.728,-62.182,0]}]},"a":{"a":0,"k":[159.318,54.557,0]},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":80,"s":[100,100,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":85,"s":[100,100,100]},{"t":100,"s":[0,0,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[-0.762,-1.568],[0,0],[1.568,-0.762],[0,0],[0.762,1.568],[0,0],[-1.568,0.762],[0,0]],"o":[[0,0],[0.762,1.568],[0,0],[-1.568,0.762],[0,0],[-0.762,-1.568],[0,0],[1.568,-0.762]],"v":[[166.802,39.758],[174.463,55.53],[173.003,59.748],[157.232,67.409],[153.014,65.949],[145.353,50.178],[146.813,45.96],[162.584,38.299]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.972549021244,0.96862745285,0.290196090937,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[-0.762,-1.568],[0,0],[1.568,-0.762],[0,0],[0.762,1.568],[0,0],[-1.568,0.762],[0,0]],"o":[[0,0],[0.762,1.568],[0,0],[-1.568,0.762],[0,0],[-0.762,-1.568],[0,0],[1.568,-0.762]],"v":[[166.802,39.758],[175.577,57.822],[174.117,62.04],[156.053,70.815],[151.835,69.356],[143.06,51.291],[144.52,47.074],[162.584,38.299]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.737254917622,0.737254917622,0.258823543787,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 8","bm":0,"hd":false}],"ip":80,"op":100,"st":10,"bm":0},{"ddd":0,"ind":16,"ty":4,"nm":"cube red 3","parent":39,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":79,"s":[-22.803]},{"t":99,"s":[1498.197]}]},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":79,"s":[53.321,105.546,0],"to":[0,0,0],"ti":[0,0,0]},{"t":99,"s":[173.945,218.194,0]}]},"a":{"a":0,"k":[-158.763,14.8,0]},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":79,"s":[100,100,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":84,"s":[100,100,100]},{"t":99,"s":[0,0,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[1.055,-1.387],[0,0],[1.387,1.055],[0,0],[-1.055,1.387],[0,0],[-1.387,-1.055],[0,0]],"o":[[0,0],[-1.055,1.387],[0,0],[-1.387,-1.055],[0,0],[1.055,-1.387],[0,0],[1.387,1.055]],"v":[[-142.18,14.8],[-152.797,28.754],[-157.22,29.355],[-171.173,18.737],[-171.774,14.315],[-161.157,0.361],[-156.734,-0.24],[-142.78,10.378]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.839215695858,0.184313729405,0.345098048449,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[1.055,-1.387],[0,0],[1.387,1.055],[0,0],[-1.055,1.387],[0,0],[-1.387,-1.055],[0,0]],"o":[[0,0],[-1.055,1.387],[0,0],[-1.387,-1.055],[0,0],[1.055,-1.387],[0,0],[1.387,1.055]],"v":[[-142.18,14.8],[-154.341,30.783],[-158.763,31.383],[-174.745,19.222],[-175.346,14.8],[-163.185,-1.183],[-158.762,-1.783],[-142.78,10.378]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.807843148708,0.098039217293,0.301960796118,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 6","bm":0,"hd":false}],"ip":79,"op":99,"st":9,"bm":0},{"ddd":0,"ind":17,"ty":4,"nm":"cube blue 3","parent":39,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":78,"s":[5.755]},{"t":98,"s":[-342.245]}]},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":78,"s":[50.896,64.98,0],"to":[0,0,0],"ti":[0,0,0]},{"t":98,"s":[349.995,36.419,0]}]},"a":{"a":0,"k":[12.04,-164.683,0]},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":78,"s":[100,100,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":83,"s":[100,100,100]},{"t":98,"s":[0,0,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[1.541,-0.815],[0,0],[0.815,1.541],[0,0],[-1.541,0.815],[0,0],[-0.815,-1.541],[0,0]],"o":[[0,0],[-1.541,0.815],[0,0],[-0.815,-1.541],[0,0],[1.541,-0.815],[0,0],[0.815,1.541]],"v":[[27.089,-157.715],[11.592,-149.513],[7.326,-150.827],[-0.876,-166.324],[0.437,-170.589],[15.934,-178.791],[20.2,-177.478],[28.402,-161.981]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.172549024224,0.403921574354,0.917647063732,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[1.541,-0.815],[0,0],[0.815,1.541],[0,0],[-1.541,0.815],[0,0],[-0.815,-1.541],[0,0]],"o":[[0,0],[-1.541,0.815],[0,0],[-0.815,-1.541],[0,0],[1.541,-0.815],[0,0],[0.815,1.541]],"v":[[27.089,-157.715],[9.339,-148.321],[5.073,-149.634],[-4.321,-167.384],[-3.008,-171.65],[14.742,-181.044],[19.008,-179.731],[28.402,-161.981]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.0941176489,0.301960796118,0.607843160629,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 10","bm":0,"hd":false}],"ip":78,"op":98,"st":8,"bm":0},{"ddd":0,"ind":18,"ty":4,"nm":"cube orange 2","parent":39,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":77,"s":[190.298]},{"t":97,"s":[18.298]}]},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":77,"s":[68.022,25.141,0],"to":[0,0,0],"ti":[0,0,0]},{"t":97,"s":[322.591,26.755,0]}]},"a":{"a":0,"k":[165.82,119.688,0]},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":77,"s":[100,100,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":82,"s":[100,100,100]},{"t":97,"s":[0,0,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[0.758,-1.57],[0,0],[1.57,0.758],[0,0],[-0.758,1.57],[0,0],[-1.57,-0.758],[0,0]],"o":[[0,0],[-0.758,1.57],[0,0],[-1.57,-0.758],[0,0],[0.758,-1.57],[0,0],[1.57,0.758]],"v":[[182.071,116.386],[174.445,132.174],[170.231,133.643],[154.442,126.018],[152.973,121.803],[160.598,106.015],[164.813,104.545],[180.601,112.171]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.988235294819,0.596078455448,0.149019613862,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[0.758,-1.57],[0,0],[1.57,0.758],[0,0],[-0.758,1.57],[0,0],[-1.57,-0.758],[0,0]],"o":[[0,0],[-0.758,1.57],[0,0],[-1.57,-0.758],[0,0],[0.758,-1.57],[0,0],[1.57,0.758]],"v":[[182.071,116.386],[173.337,134.469],[169.122,135.939],[151.038,127.205],[149.569,122.99],[158.303,104.906],[162.518,103.437],[180.601,112.171]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.819607853889,0.537254929543,0.152941182256,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false}],"ip":77,"op":97,"st":7,"bm":0},{"ddd":0,"ind":19,"ty":4,"nm":"cube pinky 2","parent":39,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":77,"s":[-182.39]},{"t":97,"s":[37.61]}]},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":77,"s":[52.242,-2.716,0],"to":[0,0,0],"ti":[0,0,0]},{"t":97,"s":[254.86,-63.536,0]}]},"a":{"a":0,"k":[-58.749,-145.954,0]},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":77,"s":[100,100,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":82,"s":[100,100,100]},{"t":97,"s":[0,0,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[-1.559,-0.779],[0,0],[0.779,-1.559],[0,0],[1.559,0.779],[0,0],[-0.779,1.559],[0,0]],"o":[[0,0],[1.559,0.779],[0,0],[-0.779,1.559],[0,0],[-1.559,-0.779],[0,0],[0.779,-1.559]],"v":[[-61.835,-162.247],[-46.149,-154.412],[-44.736,-150.178],[-52.571,-134.493],[-56.805,-133.08],[-72.49,-140.915],[-73.903,-145.149],[-66.068,-160.834]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.976470589638,0.345098048449,0.564705908298,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[-1.559,-0.779],[0,0],[0.779,-1.559],[0,0],[1.559,0.779],[0,0],[-0.779,1.559],[0,0]],"o":[[0,0],[1.559,0.779],[0,0],[-0.779,1.559],[0,0],[-1.559,-0.779],[0,0],[0.779,-1.559]],"v":[[-61.835,-162.247],[-43.869,-153.273],[-42.455,-149.039],[-51.43,-131.073],[-55.663,-129.66],[-73.629,-138.635],[-75.042,-142.868],[-66.068,-160.834]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.807843148708,0.098039217293,0.301960796118,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 3","bm":0,"hd":false}],"ip":77,"op":97,"st":7,"bm":0},{"ddd":0,"ind":20,"ty":4,"nm":"cube yellow 2","parent":39,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":77,"s":[403.13]},{"t":97,"s":[18.13]}]},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":77,"s":[69.883,70.255,0],"to":[0,0,0],"ti":[0,0,0]},{"t":97,"s":[195.07,191.252,0]}]},"a":{"a":0,"k":[159.318,54.557,0]},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":77,"s":[100,100,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":82,"s":[100,100,100]},{"t":97,"s":[0,0,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[-0.762,-1.568],[0,0],[1.568,-0.762],[0,0],[0.762,1.568],[0,0],[-1.568,0.762],[0,0]],"o":[[0,0],[0.762,1.568],[0,0],[-1.568,0.762],[0,0],[-0.762,-1.568],[0,0],[1.568,-0.762]],"v":[[166.802,39.758],[174.463,55.53],[173.003,59.748],[157.232,67.409],[153.014,65.949],[145.353,50.178],[146.813,45.96],[162.584,38.299]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.972549021244,0.96862745285,0.290196090937,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[-0.762,-1.568],[0,0],[1.568,-0.762],[0,0],[0.762,1.568],[0,0],[-1.568,0.762],[0,0]],"o":[[0,0],[0.762,1.568],[0,0],[-1.568,0.762],[0,0],[-0.762,-1.568],[0,0],[1.568,-0.762]],"v":[[166.802,39.758],[175.577,57.822],[174.117,62.04],[156.053,70.815],[151.835,69.356],[143.06,51.291],[144.52,47.074],[162.584,38.299]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.737254917622,0.737254917622,0.258823543787,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 8","bm":0,"hd":false}],"ip":77,"op":97,"st":7,"bm":0},{"ddd":0,"ind":21,"ty":4,"nm":"cube red 2","parent":39,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":77,"s":[-22.803]},{"t":97,"s":[1498.197]}]},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":77,"s":[50.342,47.012,0],"to":[0,0,0],"ti":[0,0,0]},{"t":97,"s":[289.484,81.596,0]}]},"a":{"a":0,"k":[-158.763,14.8,0]},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":77,"s":[100,100,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":82,"s":[100,100,100]},{"t":97,"s":[0,0,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[1.055,-1.387],[0,0],[1.387,1.055],[0,0],[-1.055,1.387],[0,0],[-1.387,-1.055],[0,0]],"o":[[0,0],[-1.055,1.387],[0,0],[-1.387,-1.055],[0,0],[1.055,-1.387],[0,0],[1.387,1.055]],"v":[[-142.18,14.8],[-152.797,28.754],[-157.22,29.355],[-171.173,18.737],[-171.774,14.315],[-161.157,0.361],[-156.734,-0.24],[-142.78,10.378]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.839215695858,0.184313729405,0.345098048449,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[1.055,-1.387],[0,0],[1.387,1.055],[0,0],[-1.055,1.387],[0,0],[-1.387,-1.055],[0,0]],"o":[[0,0],[-1.055,1.387],[0,0],[-1.387,-1.055],[0,0],[1.055,-1.387],[0,0],[1.387,1.055]],"v":[[-142.18,14.8],[-154.341,30.783],[-158.763,31.383],[-174.745,19.222],[-175.346,14.8],[-163.185,-1.183],[-158.762,-1.783],[-142.78,10.378]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.807843148708,0.098039217293,0.301960796118,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 6","bm":0,"hd":false}],"ip":77,"op":97,"st":7,"bm":0},{"ddd":0,"ind":22,"ty":4,"nm":"cube blue 2","parent":39,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":77,"s":[5.755]},{"t":97,"s":[-342.245]}]},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":77,"s":[64.809,-34.33,0],"to":[0,0,0],"ti":[0,0,0]},{"t":97,"s":[174.346,-151.23,0]}]},"a":{"a":0,"k":[12.04,-164.683,0]},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":77,"s":[100,100,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":82,"s":[100,100,100]},{"t":97,"s":[0,0,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[1.541,-0.815],[0,0],[0.815,1.541],[0,0],[-1.541,0.815],[0,0],[-0.815,-1.541],[0,0]],"o":[[0,0],[-1.541,0.815],[0,0],[-0.815,-1.541],[0,0],[1.541,-0.815],[0,0],[0.815,1.541]],"v":[[27.089,-157.715],[11.592,-149.513],[7.326,-150.827],[-0.876,-166.324],[0.437,-170.589],[15.934,-178.791],[20.2,-177.478],[28.402,-161.981]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.172549024224,0.403921574354,0.917647063732,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[1.541,-0.815],[0,0],[0.815,1.541],[0,0],[-1.541,0.815],[0,0],[-0.815,-1.541],[0,0]],"o":[[0,0],[-1.541,0.815],[0,0],[-0.815,-1.541],[0,0],[1.541,-0.815],[0,0],[0.815,1.541]],"v":[[27.089,-157.715],[9.339,-148.321],[5.073,-149.634],[-4.321,-167.384],[-3.008,-171.65],[14.742,-181.044],[19.008,-179.731],[28.402,-161.981]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.0941176489,0.301960796118,0.607843160629,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 10","bm":0,"hd":false}],"ip":77,"op":97,"st":7,"bm":0},{"ddd":0,"ind":23,"ty":0,"nm":"cube pink","parent":2,"refId":"comp_0","sr":1,"ks":{"p":{"a":0,"k":[-60,0,0]},"a":{"a":0,"k":[256,256,0]},"s":{"a":0,"k":[-100,100,100]}},"ao":0,"w":512,"h":512,"ip":0,"op":89,"st":-14,"bm":0},{"ddd":0,"ind":24,"ty":0,"nm":"cube yellow","parent":2,"refId":"comp_1","sr":1,"ks":{"p":{"a":0,"k":[60,0,0]},"a":{"a":0,"k":[256,256,0]},"s":{"a":0,"k":[-100,100,100]}},"ao":0,"w":512,"h":512,"ip":0,"op":82,"st":-21,"bm":0},{"ddd":0,"ind":25,"ty":0,"nm":"cube red","parent":2,"refId":"comp_2","sr":1,"ks":{"p":{"a":0,"k":[-100,0,0]},"a":{"a":0,"k":[256,256,0]},"s":{"a":0,"k":[-100,100,100]}},"ao":0,"w":512,"h":512,"ip":0,"op":75,"st":-28,"bm":0},{"ddd":0,"ind":26,"ty":0,"nm":"cube blue","parent":2,"refId":"comp_3","sr":1,"ks":{"a":{"a":0,"k":[256,256,0]},"s":{"a":0,"k":[-100,100,100]}},"ao":0,"w":512,"h":512,"ip":0,"op":87,"st":-16,"bm":0},{"ddd":0,"ind":27,"ty":0,"nm":"cube orange","parent":2,"refId":"comp_4","sr":1,"ks":{"p":{"a":0,"k":[120,0,0]},"a":{"a":0,"k":[256,256,0]}},"ao":0,"w":512,"h":512,"ip":0,"op":75,"st":-28,"bm":0},{"ddd":0,"ind":28,"ty":0,"nm":"cube pink","parent":2,"refId":"comp_0","sr":1,"ks":{"p":{"a":0,"k":[-60,0,0]},"a":{"a":0,"k":[256,256,0]}},"ao":0,"w":512,"h":512,"ip":0,"op":69,"st":-34,"bm":0},{"ddd":0,"ind":29,"ty":0,"nm":"cube yellow","parent":2,"refId":"comp_1","sr":1,"ks":{"p":{"a":0,"k":[60,0,0]},"a":{"a":0,"k":[256,256,0]}},"ao":0,"w":512,"h":512,"ip":0,"op":65,"st":-38,"bm":0},{"ddd":0,"ind":30,"ty":0,"nm":"cube red","parent":2,"refId":"comp_2","sr":1,"ks":{"p":{"a":0,"k":[-102.778,-178.704,0]},"a":{"a":0,"k":[253.222,68.037,0]}},"ao":0,"w":512,"h":512,"ip":0,"op":90,"st":-13,"bm":0},{"ddd":0,"ind":31,"ty":0,"nm":"cube blue","parent":2,"refId":"comp_3","sr":1,"ks":{"a":{"a":0,"k":[256,256,0]}},"ao":0,"w":512,"h":512,"ip":0,"op":58,"st":-45,"bm":0},{"ddd":0,"ind":32,"ty":4,"nm":"cube red 4","parent":2,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.2],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":110,"s":[196]},{"t":180,"s":[223]}]},"p":{"a":1,"k":[{"i":{"x":0.2,"y":1},"o":{"x":0.31,"y":0},"t":110,"s":[-91.5,-223.781,0],"to":[0,0,0],"ti":[0,0,0]},{"t":179,"s":[-85.5,-123.75,0]}]},"a":{"a":0,"k":[-158.763,14.8,0]},"s":{"a":1,"k":[{"i":{"x":[0.7,0.7,0.7],"y":[1,1,1]},"o":{"x":[0.31,0.31,0.31],"y":[0,0,0]},"t":110,"s":[0,0,100]},{"i":{"x":[0.4,0.4,0.4],"y":[1,1,1]},"o":{"x":[0.3,0.3,0.3],"y":[0,0,0]},"t":132,"s":[100,100,100]},{"t":180,"s":[98,98,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[1.055,-1.387],[0,0],[1.387,1.055],[0,0],[-1.055,1.387],[0,0],[-1.387,-1.055],[0,0]],"o":[[0,0],[-1.055,1.387],[0,0],[-1.387,-1.055],[0,0],[1.055,-1.387],[0,0],[1.387,1.055]],"v":[[-142.18,14.8],[-152.797,28.754],[-157.22,29.355],[-171.173,18.737],[-171.774,14.315],[-161.157,0.361],[-156.734,-0.24],[-142.78,10.378]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.839215695858,0.184313729405,0.345098048449,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[1.055,-1.387],[0,0],[1.387,1.055],[0,0],[-1.055,1.387],[0,0],[-1.387,-1.055],[0,0]],"o":[[0,0],[-1.055,1.387],[0,0],[-1.387,-1.055],[0,0],[1.055,-1.387],[0,0],[1.387,1.055]],"v":[[-142.18,14.8],[-154.341,30.783],[-158.763,31.383],[-174.745,19.222],[-175.346,14.8],[-163.185,-1.183],[-158.762,-1.783],[-142.78,10.378]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.807843148708,0.098039217293,0.301960796118,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 6","bm":0,"hd":false}],"ip":116,"op":255,"st":152,"bm":0},{"ddd":0,"ind":33,"ty":4,"nm":"cube blue 4","parent":2,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.1],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":121,"s":[-239]},{"t":179,"s":[-197]}]},"p":{"a":1,"k":[{"i":{"x":0.2,"y":1},"o":{"x":0.31,"y":0},"t":121,"s":[5.5,-244.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":179,"s":[15.5,-178.5,0]}]},"a":{"a":0,"k":[12.04,-164.683,0]},"s":{"a":1,"k":[{"i":{"x":[0.4,0.4,0.4],"y":[1,1,1]},"o":{"x":[0.4,0.4,0.4],"y":[0,0,0]},"t":121,"s":[0,0,100]},{"t":179,"s":[100,100,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[1.541,-0.815],[0,0],[0.815,1.541],[0,0],[-1.541,0.815],[0,0],[-0.815,-1.541],[0,0]],"o":[[0,0],[-1.541,0.815],[0,0],[-0.815,-1.541],[0,0],[1.541,-0.815],[0,0],[0.815,1.541]],"v":[[27.089,-157.715],[11.592,-149.513],[7.326,-150.827],[-0.876,-166.324],[0.437,-170.589],[15.934,-178.791],[20.2,-177.478],[28.402,-161.981]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.172549024224,0.403921574354,0.917647063732,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[1.541,-0.815],[0,0],[0.815,1.541],[0,0],[-1.541,0.815],[0,0],[-0.815,-1.541],[0,0]],"o":[[0,0],[-1.541,0.815],[0,0],[-0.815,-1.541],[0,0],[1.541,-0.815],[0,0],[0.815,1.541]],"v":[[27.089,-157.715],[9.339,-148.321],[5.073,-149.634],[-4.321,-167.384],[-3.008,-171.65],[14.742,-181.044],[19.008,-179.731],[28.402,-161.981]],"c":true}},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.0941176489,0.301960796118,0.607843160629,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 10","bm":0,"hd":false}],"ip":137,"op":266,"st":163,"bm":0},{"ddd":0,"ind":34,"ty":3,"nm":"rl","parent":2,"sr":1,"ks":{"o":{"a":0,"k":0},"p":{"a":1,"k":[{"i":{"x":0.518,"y":1},"o":{"x":0.22,"y":0.004},"t":0,"s":[-6.666,0,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.11,"y":1},"o":{"x":0.33,"y":0},"t":10,"s":[0,0,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.2,"y":1},"o":{"x":0.8,"y":0},"t":63,"s":[16.667,0,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.397,"y":1},"o":{"x":0.434,"y":0},"t":98,"s":[-20,0,0],"to":[0,0,0],"ti":[0,0,0]},{"t":179,"s":[-6.666,0,0]}]}},"ao":0,"ip":0,"op":180,"st":45,"bm":0},{"ddd":0,"ind":35,"ty":3,"nm":"updown","parent":34,"sr":1,"ks":{"o":{"a":0,"k":0},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0.006},"t":0,"s":[-375,101.805,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.667,"y":0.667},"o":{"x":0.333,"y":0.333},"t":11,"s":[-375,83,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":75,"s":[-375,83,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.667,"y":0.966},"o":{"x":0.333,"y":0},"t":171,"s":[-375,103,0],"to":[0,0,0],"ti":[0,0,0]},{"t":179,"s":[-375,101.805,0]}]}},"ao":0,"ip":0,"op":180,"st":45,"bm":0},{"ddd":0,"ind":36,"ty":4,"nm":"fx bands inside 2","parent":2,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.657],"y":[0.961]},"o":{"x":[0.167],"y":[0.167]},"t":78,"s":[96]},{"t":110,"s":[-54.658]}]},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":78,"s":[-8.461,-33.424,0],"to":[0,0,0],"ti":[0,0,0]},{"t":110,"s":[118.062,-223.746,0]}]},"a":{"a":0,"k":[-77.47,-46.162,0]},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0]},"t":78,"s":[0,0,100]},{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":84,"s":[100,100,100]},{"t":110,"s":[0,0,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":78,"s":[{"i":[[0,0],[-7.469,-11.599]],"o":[[0,0],[0,0]],"v":[[-42.231,-180.359],[-24.142,-162.38]],"c":false}]},{"t":110,"s":[{"i":[[0,0],[-12.932,4.804]],"o":[[0,0],[0,0]],"v":[[-123.422,-71.409],[-101.979,-85.217]],"c":false}]}]},"nm":"Path 1","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"w":{"a":0,"k":6},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":20},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 3","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":78,"s":[{"i":[[0,0],[-0.284,-1.099]],"o":[[0.362,1.046],[0,0]],"v":[[-19.226,-152.377],[-18.253,-149.159]],"c":false}]},{"t":110,"s":[{"i":[[0,0],[-1.135,0.042]],"o":[[1.1,-0.129],[0,0]],"v":[[-91.153,-87.87],[-87.801,-88.129]],"c":false}]}]},"nm":"Path 1","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"w":{"a":0,"k":6},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":20},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 4","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":78,"s":[{"i":[[21.89,-42.85],[0,0],[-1.41,24.23],[12.04,9.62],[-1.59,8.11],[-0.77,2.09],[-5.47,-4.24],[-0.34,-39.6]],"o":[[0,0],[16.79,-34.12],[2.31,-39.72],[-12.04,-9.62],[0.6,-3.05],[-0.12,0.58],[5.66,4.37],[0.35,39.6]],"v":[[-47.853,-1.193],[-52.783,-5.323],[-23.083,-98.603],[-44.633,-164.823],[-55.653,-180.543],[-53.783,-188.473],[-50.053,-174.783],[-15.983,-112.203]],"c":true}]},{"t":110,"s":[{"i":[[21.89,-42.85],[0,0],[23.362,6.579],[11.981,-9.694],[7.577,3.299],[1.876,1.201],[-5.307,4.442],[-38.751,-8.161]],"o":[[0,0],[16.79,-34.12],[-38.298,-10.785],[-11.981,9.694],[-2.85,-1.241],[0.541,0.242],[5.483,-4.59],[38.751,8.161]],"v":[[-47.853,-1.193],[-52.783,-5.323],[-39.461,-72.557],[-108.764,-65.728],[-126.484,-58.34],[-133.827,-61.869],[-119.656,-62.573],[-51.22,-82.412]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.807843148708,0.098039217293,0.301960796118,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 7","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":78,"s":[{"i":[[38.37,-65.76],[0,0],[0,0],[-1.41,24.23],[12.04,9.62],[-1.59,8.11],[-0.77,2.09],[0,0],[-4.36,-2.27],[-8.91,-34.63]],"o":[[0,0],[0,0],[16.79,-34.12],[2.31,-39.72],[-12.04,-9.62],[0.6,-3.05],[0,-0.02],[1.26,-3.44],[8.55,4.45],[8.72,33.86]],"v":[[-42.313,3.437],[-47.853,-1.193],[-52.783,-5.323],[-23.083,-98.603],[-44.633,-164.823],[-55.653,-180.543],[-53.783,-188.473],[-53.783,-188.503],[-46.183,-191.213],[-8.443,-139.753]],"c":true}]},{"t":110,"s":[{"i":[[56.487,-85.424],[0,0],[0,0],[23.362,6.579],[11.981,-9.694],[7.577,3.299],[1.876,1.201],[0,0],[-3.153,3.771],[-35.735,1.275]],"o":[[0,0],[0,0],[16.79,-34.12],[-38.298,-10.785],[-11.981,9.694],[-2.85,-1.241],[-0.02,-0.004],[-3.089,-1.969],[6.182,-7.395],[34.943,-1.246]],"v":[[-42.313,3.437],[-47.853,-1.193],[-52.783,-5.323],[-39.461,-72.557],[-108.764,-65.728],[-126.484,-58.34],[-133.827,-61.869],[-133.856,-61.876],[-134.871,-69.88],[-76.508,-95.691]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.964705884457,0.109803922474,0.333333343267,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 8","bm":0,"hd":false}],"ip":78,"op":104,"st":8,"bm":0},{"ddd":0,"ind":37,"ty":4,"nm":"fx bands inside","parent":2,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":82,"s":[-30.85]},{"t":115,"s":[174.15]}]},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":82,"s":[29.858,19.098,0],"to":[0,0,0],"ti":[0,0,0]},{"t":115,"s":[233.52,-196.875,0]}]},"a":{"a":0,"k":[104.079,-13.343,0]},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0]},"t":82,"s":[0,0,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":93,"s":[100,100,100]},{"t":113,"s":[0,0,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":82,"s":[{"i":[[0,0],[-3.306,0.599]],"o":[[0,0],[0,0]],"v":[[134.547,-38.045],[139.743,-39.06]],"c":false}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":90,"s":[{"i":[[0,0],[-2.381,-0.26]],"o":[[0,0],[0,0]],"v":[[130.387,-31.028],[134.151,-30.668]],"c":false}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":98,"s":[{"i":[[0,0],[-1.455,-1.119]],"o":[[0,0],[0,0]],"v":[[126.226,-24.012],[128.559,-22.276]],"c":false}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":108,"s":[{"i":[[0,0],[-0.619,-1.895]],"o":[[0,0],[0,0]],"v":[[122.467,-17.672],[123.505,-14.692]],"c":false}]},{"t":115,"s":[{"i":[[0,0],[-0.317,-2.175]],"o":[[0,0],[0,0]],"v":[[121.112,-15.388],[121.685,-11.96]],"c":false}]}]},"nm":"Path 1","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"w":{"a":0,"k":10},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":20},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":82,"s":[{"i":[[0,0],[-11.649,-0.644]],"o":[[9.919,-0.974],[0,0]],"v":[[161.787,-42.118],[195.004,-43.043]],"c":false}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":90,"s":[{"i":[[0,0],[-7.542,-2.706]],"o":[[6.889,1.315],[0,0]],"v":[[149.736,-28.326],[172.09,-22.404]],"c":false}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":98,"s":[{"i":[[0,0],[-3.435,-4.768]],"o":[[3.86,3.604],[0,0]],"v":[[137.685,-14.534],[149.175,-1.766]],"c":false}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":108,"s":[{"i":[[0,0],[0.276,-6.632]],"o":[[1.122,5.672],[0,0]],"v":[[126.794,-2.071],[128.467,16.886]],"c":false}]},{"t":115,"s":[{"i":[[0,0],[1.613,-7.303]],"o":[[0.136,6.418],[0,0]],"v":[[122.87,2.419],[121.008,23.605]],"c":false}]}]},"nm":"Path 1","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"w":{"a":0,"k":10},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":20},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":82,"s":[{"i":[[-3.59,0.71],[0,0],[0.01,-0.01],[-6.33,4.68],[-58.63,12.74]],"o":[[-89.46,31.09],[0,0],[-2.89,-5.01],[0,0],[3.76,-0.82]],"v":[[117.777,-45.713],[2.577,24.667],[2.567,24.677],[4.677,4.857],[106.737,-43.423]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":90,"s":[{"i":[[-2.603,-0.244],[0,0],[0.01,-0.01],[-1.007,7.808],[-43.4,-7.891]],"o":[[-65.941,6.335],[0,0],[-2.89,-5.01],[0,0],[2.732,0.209]],"v":[[121.599,-39.305],[63.607,38.204],[63.597,38.214],[61.919,16.171],[113.561,-39.985]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":98,"s":[{"i":[[-1.616,-1.197],[0,0],[0.01,-0.01],[-6.33,4.68],[-28.17,-28.522]],"o":[[-42.421,-18.419],[0,0],[-2.89,-5.01],[0,0],[1.703,1.238]],"v":[[125.421,-32.896],[31.387,6.859],[31.377,6.869],[33.487,-12.951],[120.384,-36.546]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":108,"s":[{"i":[[-0.725,-2.059],[0,0],[0.01,-0.01],[-6.33,4.68],[-14.406,-47.166]],"o":[[-21.166,-40.79],[0,0],[-2.89,-5.01],[0,0],[0.774,2.168]],"v":[[128.874,-27.105],[44.405,-1.188],[44.395,-1.178],[46.505,-20.998],[126.551,-33.439]],"c":true}]},{"t":115,"s":[{"i":[[-0.403,-2.369],[0,0],[0.01,-0.01],[-6.33,4.68],[-9.448,-53.882]],"o":[[-13.51,-48.849],[0,0],[-2.89,-5.01],[0,0],[0.439,2.503]],"v":[[130.119,-25.019],[49.095,-4.086],[49.085,-4.077],[51.195,-23.896],[128.772,-32.319]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.203921571374,0.349019616842,0.952941179276,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 5","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":82,"s":[{"i":[[-9.004,6.666],[-58.629,12.747],[-2.161,-3.086],[3.086,-3.429],[58.451,-17.592],[4.457,-1.714]],"o":[[0,0],[58.629,-12.747],[2.161,3.086],[-3.086,3.429],[-70.629,21.257],[-4.457,1.714]],"v":[[4.678,4.855],[106.734,-43.427],[206.744,-52.402],[205.134,-31.488],[99.191,-17.773],[6.619,27.484]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":90,"s":[{"i":[[-1.425,11.112],[-42.867,-3.23],[-0.488,-2.429],[3.103,-1.622],[40.454,0.651],[4.457,-1.714]],"o":[[0,0],[42.867,3.23],[0.488,2.429],[-3.103,1.622],[-48.842,4.158],[-4.457,1.714]],"v":[[61.918,16.17],[113.56,-39.988],[182.763,-26.18],[175.272,-12.906],[103.511,-24.067],[67.649,41.021]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":98,"s":[{"i":[[-9.004,6.666],[-27.105,-19.207],[1.184,-1.773],[3.12,0.185],[22.457,18.893],[4.457,-1.714]],"o":[[0,0],[27.105,19.207],[-1.184,1.773],[-3.12,-0.185],[-27.056,-12.941],[-4.457,1.714]],"v":[[33.488,-12.953],[120.385,-36.548],[158.782,0.041],[145.411,5.675],[107.831,-30.36],[35.43,9.676]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":108,"s":[{"i":[[-9.004,6.666],[-12.861,-33.645],[2.696,-1.18],[3.135,1.817],[6.193,35.379],[4.457,-1.714]],"o":[[0,0],[12.861,33.645],[-2.696,1.18],[-3.135,-1.817],[-7.367,-28.394],[-4.457,1.714]],"v":[[46.506,-20.999],[126.554,-33.441],[137.11,23.738],[118.425,22.467],[115.897,-25.076],[48.448,1.629]],"c":true}]},{"t":115,"s":[{"i":[[-9.004,6.666],[-7.73,-38.847],[3.24,-0.966],[3.141,2.405],[0.335,41.318],[4.457,-1.714]],"o":[[0,0],[7.73,38.847],[-3.24,0.967],[-3.141,-2.405],[-0.275,-33.961],[-4.457,1.714]],"v":[[51.195,-23.898],[128.776,-32.321],[129.303,32.274],[108.704,28.516],[107.826,-10.964],[53.137,-1.269]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.282352954149,0.46274510026,0.996078431606,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 6","bm":0,"hd":false}],"ip":82,"op":111,"st":12,"bm":0},{"ddd":0,"ind":38,"ty":3,"nm":"masterboom","parent":35,"sr":1,"ks":{"o":{"a":0,"k":0},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":74,"s":[0.364]},{"i":{"x":[0.5],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":77,"s":[-6]},{"t":93,"s":[0.364]}]},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":74,"s":[304,6,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.5,"y":1},"o":{"x":0.333,"y":0},"t":77,"s":[288.259,21.741,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.3,"y":1},"o":{"x":0.5,"y":0},"t":93,"s":[313.259,-3.259,0],"to":[0,0,0],"ti":[0,0,0]},{"t":179,"s":[304,6,0]}]}},"ao":0,"ip":0,"op":180,"st":45,"bm":0},{"ddd":0,"ind":39,"ty":4,"nm":"base","parent":38,"sr":1,"ks":{"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0.016]},"t":0,"s":[-46.934]},{"i":{"x":[0.565],"y":[1]},"o":{"x":[0.422],"y":[0]},"t":9,"s":[-50]},{"i":{"x":[0.843],"y":[1]},"o":{"x":[0.331],"y":[0]},"t":31,"s":[-33]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":75,"s":[-46]},{"i":{"x":[0.434],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":116,"s":[-40]},{"t":179,"s":[-46.934]}]},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":42,"s":[9.315,-21.341,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":43,"s":[11.526,-25.295,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":44,"s":[13.354,-19.188,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":45,"s":[9.261,-15.248,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":46,"s":[5.296,-19.297,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":47,"s":[9.28,-15.277,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":48,"s":[9.315,-21.341,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":49,"s":[11.526,-25.295,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":50,"s":[13.354,-19.188,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":51,"s":[9.261,-15.248,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":52,"s":[5.296,-19.297,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":53,"s":[9.28,-15.277,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":54,"s":[9.315,-21.341,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":55,"s":[11.526,-25.295,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":56,"s":[13.354,-19.188,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":57,"s":[9.261,-15.248,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":58,"s":[11.526,-25.295,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":59,"s":[13.354,-19.188,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":60,"s":[9.261,-15.248,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":61,"s":[5.296,-19.297,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":62,"s":[9.28,-15.277,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":63,"s":[9.315,-21.341,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":64,"s":[11.526,-25.295,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":65,"s":[13.354,-19.188,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":66,"s":[9.261,-15.248,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":67,"s":[5.296,-19.297,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":68,"s":[9.28,-15.277,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":69,"s":[13.354,-19.188,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":70,"s":[9.261,-15.248,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":71,"s":[5.296,-19.297,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":72,"s":[9.28,-15.277,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":73,"s":[7.297,-31.279,0],"to":[0,0,0],"ti":[0,0,0]},{"t":75,"s":[9.315,-21.341,0]}]},"a":{"a":0,"k":[-69.685,34.659,0]},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":28,"s":[100,100,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":74,"s":[95,105,100]},{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0]},"t":77,"s":[120,77,100]},{"i":{"x":[0.34,0.34,0.34],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":86,"s":[95,105,100]},{"t":107,"s":[100,100,100]}]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[{"i":[[21.831,-0.431],[0,0],[0.007,0.008],[1.315,65.982],[-17.495,13.858],[0,0],[-1.514,-76.122]],"o":[[0,0],[-0.007,-0.008],[-18.016,-13.169],[-1.316,-66.003],[0,0],[21.831,-0.431],[1.514,76.122]],"v":[[60.509,172.465],[51.987,169.476],[51.973,169.461],[18.248,35.42],[46.625,-99.876],[55.02,-103.21],[97.288,33.844]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[-9.914,-0.431],[0,0],[-0.003,0.008],[-0.597,65.982],[7.945,13.858],[0,0],[0.688,-76.122]],"o":[[0,0],[0.003,-0.008],[8.181,-13.169],[0.597,-66.003],[0,0],[-9.914,-0.431],[-0.688,76.122]],"v":[[58.512,168.459],[61.398,169.476],[61.404,169.461],[88.725,35.41],[63.833,-99.876],[61.149,-98.971],[40.826,33.844]],"c":true}]},{"t":68,"s":[{"i":[[21.831,-0.431],[0,0],[0.007,0.008],[1.315,65.982],[-17.495,13.858],[0,0],[-1.514,-76.122]],"o":[[0,0],[-0.007,-0.008],[-18.016,-13.169],[-1.316,-66.003],[0,0],[21.831,-0.431],[1.514,76.122]],"v":[[60.509,172.465],[51.987,169.476],[51.973,169.461],[18.248,35.42],[46.625,-99.876],[55.02,-103.21],[97.288,33.844]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"st","c":{"a":0,"k":[0.607843137255,0.360784313725,0,1]},"o":{"a":0,"k":100},"w":{"a":0,"k":8},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Outline Back","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":0,"s":[{"i":[[8.052,65.502],[-14.663,13.559]],"o":[[-6.079,-49.448],[0,0]],"v":[[18.304,51.304],[38.424,-86.054]],"c":false}]},{"i":{"x":0.4,"y":1},"o":{"x":0.167,"y":0.167},"t":20,"s":[{"i":[[1.292,65.848],[-0.327,13.749]],"o":[[1.821,-68.499],[0,0]],"v":[[65.713,40.606],[63.234,-92.033]],"c":false}]},{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[-1.814,65.959],[7.945,13.858]],"o":[[1.898,-69.024],[0,0]],"v":[[92.548,36.026],[68.561,-98.932]],"c":false}]},{"t":68,"s":[{"i":[[6.674,65.656],[-14.663,13.559]],"o":[[-6.242,-61.406],[0,0]],"v":[[19.198,48.545],[42.122,-89.555]],"c":false}]}]},"nm":"Path 1","hd":false},{"ty":"st","c":{"a":0,"k":[0.957647106694,0.769865088369,0.226079664043,1]},"o":{"a":0,"k":100},"w":{"a":0,"k":8},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","hd":false},{"ty":"tr","p":{"a":0,"k":[-4.63,0.029]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Outline Back 2","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[55.248,34.627]},"a":{"a":0,"k":[55.248,34.627]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":1,"k":[{"t":0,"s":[100],"h":1},{"t":20,"s":[0],"h":1},{"t":51,"s":[100],"h":1}]},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[{"i":[[0,0],[0,0],[0,0],[0,0]],"o":[[-3.825,-4.794],[0,0],[0,0],[0,0]],"v":[[-30.087,144.858],[-37.312,133.839],[-7.801,144.212],[-2.591,154.551]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[0,0],[0,0],[0,0],[0,0]],"o":[[0.01,-3.796],[0,0],[0,0],[0,0]],"v":[[-27.409,143.767],[-26.033,135.167],[3.526,145.754],[2.153,153.46]],"c":true}]},{"t":68,"s":[{"i":[[0,0],[0,0],[0,0],[0,0]],"o":[[-3.825,-4.794],[0,0],[0,0],[0,0]],"v":[[-30.087,144.858],[-37.312,133.839],[-7.801,144.212],[-2.591,154.551]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.266666680574,0,0.419607847929,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[{"i":[[0,0],[0,0],[0,0],[0,0]],"o":[[-3.804,-5.112],[0,0],[0,0],[0,0]],"v":[[-135.783,107.71],[-141.997,97.046],[-112.822,107.3],[-109.06,117.102]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[0,0],[0,0],[0,0],[0,0]],"o":[[0.195,-5.112],[0,0],[0,0],[0,0]],"v":[[-128.346,108.916],[-127.81,99.402],[-100.659,108.557],[-102.11,118.252]],"c":true}]},{"t":68,"s":[{"i":[[0,0],[0,0],[0,0],[0,0]],"o":[[-3.804,-5.112],[0,0],[0,0],[0,0]],"v":[[-135.783,107.71],[-141.997,97.046],[-112.822,107.3],[-109.06,117.102]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.266666680574,0,0.419607847929,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[{"i":[[0,0],[2.339,6.696],[-2.582,-0.907],[0,0],[0,0],[0,0]],"o":[[0,0],[-0.049,-0.139],[0,0],[0,0],[0,0],[-1.298,-0.459]],"v":[[-212.274,80.797],[-218.458,69.563],[-207.482,74.03],[-195.267,78.321],[-190.947,88.297],[-210.134,81.577]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[0,0],[-0.014,5.676],[-2.582,-0.769],[0,0],[0,0],[0,0]],"o":[[0,0],[0,-0.118],[0,0],[0,0],[0,0],[-1.298,-0.389]],"v":[[-210.865,77.289],[-211.022,68.264],[-200.539,72.051],[-189.638,75.688],[-189.808,84.144],[-203.133,79.698]],"c":true}]},{"t":68,"s":[{"i":[[0,0],[2.339,6.696],[-2.582,-0.907],[0,0],[0,0],[0,0]],"o":[[0,0],[-0.049,-0.139],[0,0],[0,0],[0,0],[-1.298,-0.459]],"v":[[-212.274,80.797],[-218.458,69.563],[-207.482,74.03],[-195.267,78.321],[-190.947,88.297],[-210.134,81.577]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.266666680574,0,0.419607847929,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 3","bm":0,"hd":false},{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[{"i":[[0,0],[0,0],[-3.357,1.333],[0,0],[0,0],[0,0]],"o":[[2.526,-8.134],[0,0],[0,0],[0,0],[0,0],[-2.78,1.104]],"v":[[-219.733,10.98],[-215.121,-0.215],[-211.793,-1.618],[-193.568,-8.902],[-197.239,1.212],[-208.842,5.817]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[0,0],[0,0],[-3.357,1.13],[0,0],[0,0],[0,0]],"o":[[-0.015,-6.895],[0,0],[0,0],[0,0],[0,0],[-2.78,0.936]],"v":[[-210.75,10.979],[-211.027,3.364],[-204.837,0.925],[-189.69,-5.25],[-189.546,3.324],[-201.51,7.727]],"c":true}]},{"t":68,"s":[{"i":[[0,0],[0,0],[-3.357,1.333],[0,0],[0,0],[0,0]],"o":[[2.526,-8.134],[0,0],[0,0],[0,0],[0,0],[-2.78,1.104]],"v":[[-219.733,10.98],[-215.121,-0.215],[-211.793,-1.618],[-193.568,-8.902],[-197.239,1.212],[-208.842,5.817]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.309803932905,0,0.509803950787,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 4","bm":0,"hd":false},{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[{"i":[[0,0],[-2.389,3.005],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0]],"v":[[-144.166,-19.851],[-139.213,-30.423],[-114.082,-40.398],[-117.203,-30.552]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[0,0],[0.48,2.968],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0]],"v":[[-127.177,-23.855],[-128.249,-32.738],[-101.835,-42.056],[-100.689,-33.723]],"c":true}]},{"t":68,"s":[{"i":[[0,0],[-2.389,3.005],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0]],"v":[[-144.166,-19.851],[-139.213,-30.423],[-114.082,-40.398],[-117.203,-30.552]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.309803932905,0,0.509803950787,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 5","bm":0,"hd":false},{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[{"i":[[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0]],"v":[[-42.534,-60.187],[-37.905,-70.631],[-4.329,-83.957],[-9.292,-73.38]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0]],"v":[[-24.308,-62.213],[-26.899,-69.976],[2.38,-81.552],[4.964,-73.531]],"c":true}]},{"t":68,"s":[{"i":[[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0]],"v":[[-42.534,-60.187],[-37.905,-70.631],[-4.329,-83.957],[-9.292,-73.38]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.309803932905,0,0.509803950787,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 6","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[-111.162,35.297]},"a":{"a":0,"k":[-111.162,35.297]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Dark Pieces","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[{"i":[[-18.016,-13.169],[0,0],[4.345,2.865],[-13.583,12.599],[-5.582,2.217],[0,0],[-1.31,-66.003]],"o":[[0,0],[-5.405,-1.899],[-14.247,-9.397],[3.89,-3.608],[0,0],[-17.495,13.858],[1.309,65.982]],"v":[[51.973,169.461],[-208.808,77.803],[-225.061,70.431],[-227.862,11.791],[-210.316,2.099],[46.625,-99.876],[18.248,35.42]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[8.181,-13.169],[0,0],[1.296,5.041],[-1.023,17.729],[-5.582,2.217],[0,0],[0.595,-66.003]],"o":[[0,0],[-5.405,-1.899],[-3.634,-14.133],[0.306,-5.297],[0,0],[7.945,13.858],[-0.595,65.982]],"v":[[61.404,169.461],[-208.808,77.803],[-224.92,68.639],[-225.481,10.798],[-210.316,2.099],[63.833,-99.876],[88.725,35.41]],"c":true}]},{"t":68,"s":[{"i":[[-18.016,-13.169],[0,0],[4.345,2.865],[-13.583,12.599],[-5.582,2.217],[0,0],[-1.31,-66.003]],"o":[[0,0],[-5.405,-1.899],[-14.247,-9.397],[3.89,-3.608],[0,0],[-17.495,13.858],[1.309,65.982]],"v":[[51.973,169.461],[-208.808,77.803],[-225.061,70.431],[-227.862,11.791],[-210.316,2.099],[46.625,-99.876],[18.248,35.42]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"st","c":{"a":0,"k":[0.607843160629,0.360784322023,0,1]},"o":{"a":0,"k":100},"w":{"a":0,"k":8},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Outline Main","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[{"i":[[-0.678,-1.171],[0,0],[0,0],[2.695,1.741],[0.766,2.385],[0.62,3.653],[0.111,5.601],[-4.493,10.136],[-2.952,1.171],[0,0],[0,0],[0.538,-1.556],[-0.3,-15.515],[-0.828,-5.816],[-0.362,-1.874],[-0.073,-0.356],[-0.622,-2.176],[-0.943,-2.204]],"o":[[0,0],[0,0],[-3.297,-1.155],[-1.027,-2.09],[-1.053,-3.28],[-0.906,-5.34],[-0.266,-13.468],[2.439,-1.651],[0,0],[0,0],[-0.606,1.287],[-3.257,9.425],[0.129,6.663],[0.287,2.013],[0.071,0.371],[0.51,2.474],[0.789,2.761],[0.619,1.445]],"v":[[-192.84,83.418],[-195.864,82.351],[-208.808,77.803],[-216.725,73.569],[-219.412,66.84],[-221.906,56.395],[-223.393,39.804],[-217.745,5.914],[-210.316,2.099],[-198.41,-2.626],[-195.25,-3.882],[-196.965,0.399],[-201.814,39.279],[-200.302,58.1],[-199.323,63.946],[-199.101,65.037],[-197.393,72.02],[-194.782,79.49]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[0.027,-0.993],[0,0],[0,0],[-0.005,1.476],[-0.004,2.022],[-0.004,3.097],[-0.001,4.748],[0.087,8.592],[-2.952,0.993],[0,0],[0,0],[-0.021,-1.319],[0.012,-13.152],[0.033,-4.931],[0.014,-1.589],[0.003,-0.302],[0.024,-1.844],[0.037,-1.868]],"o":[[0,0],[0,0],[-3.297,-0.979],[0.006,-1.772],[0.006,-2.78],[0.005,-4.527],[0.002,-11.417],[-0.014,-1.399],[0,0],[0,0],[0.024,1.091],[0.128,7.989],[-0.005,5.648],[-0.011,1.706],[-0.003,0.314],[-0.02,2.097],[-0.031,2.34],[-0.024,1.225]],"v":[[-189.733,80.009],[-188.922,79.104],[-201.865,75.25],[-211.032,71.66],[-211.016,65.956],[-210.888,54.352],[-210.935,39.788],[-211.012,7.31],[-203.36,4.076],[-191.453,0.07],[-189.624,-0.994],[-189.557,2.635],[-189.322,39.343],[-189.276,56.547],[-189.42,60.252],[-189.487,64.427],[-189.554,70.347],[-189.657,76.679]],"c":true}]},{"t":68,"s":[{"i":[[-0.678,-1.171],[0,0],[0,0],[2.695,1.741],[0.766,2.385],[0.62,3.653],[0.111,5.601],[-4.493,10.136],[-2.952,1.171],[0,0],[0,0],[0.538,-1.556],[-0.3,-15.515],[-0.828,-5.816],[-0.362,-1.874],[-0.073,-0.356],[-0.622,-2.176],[-0.943,-2.204]],"o":[[0,0],[0,0],[-3.297,-1.155],[-1.027,-2.09],[-1.053,-3.28],[-0.906,-5.34],[-0.266,-13.468],[2.439,-1.651],[0,0],[0,0],[-0.606,1.287],[-3.257,9.425],[0.129,6.663],[0.287,2.013],[0.071,0.371],[0.51,2.474],[0.789,2.761],[0.619,1.445]],"v":[[-192.84,83.418],[-195.864,82.351],[-208.808,77.803],[-216.725,73.569],[-219.412,66.84],[-221.906,56.395],[-223.393,39.804],[-217.745,5.914],[-210.316,2.099],[-198.41,-2.626],[-195.25,-3.882],[-196.965,0.399],[-201.814,39.279],[-200.302,58.1],[-199.323,63.946],[-199.101,65.037],[-197.393,72.02],[-194.782,79.49]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"gf","o":{"a":0,"k":100},"r":1,"bm":0,"g":{"p":5,"k":{"a":0,"k":[0.27,0.541,0.282,0.745,0.699,0.717,0.42,0.85,0.88,0.892,0.558,0.955,0.918,0.637,0.337,0.718,0.99,0.381,0.117,0.481]}},"s":{"a":0,"k":[-210.134,19.783]},"e":{"a":0,"k":[-220.763,71.277]},"t":1,"nm":"Gradient Fill 3","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Line 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[{"i":[[-3.101,-5.676],[0,0],[1.656,6.449],[1.055,7.814],[0.159,7.867],[-6.646,17.181],[0,0],[-0.585,-29.699],[-1.067,-8.644],[-0.076,-0.591],[-0.251,-1.732],[-0.192,-1.132],[-0.592,-2.753],[-0.177,-0.778],[-0.426,-1.623]],"o":[[0,0],[-2.046,-5.319],[-1.763,-6.865],[-0.989,-7.325],[-0.498,-24.593],[0,0],[-6.722,16.259],[0.189,9.609],[0.076,0.611],[0.231,1.792],[0.169,1.167],[0.5,2.948],[0.162,0.792],[0.389,1.713],[2.289,8.71]],"v":[[-110.325,112.418],[-140.235,101.9],[-145.82,84.174],[-150.097,62.066],[-151.87,39.216],[-142.367,-24.868],[-115.135,-35.675],[-125.325,36.66],[-123.406,64.165],[-123.176,65.97],[-122.424,71.263],[-121.872,74.708],[-120.228,83.267],[-119.707,85.621],[-118.484,90.612]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[0.842,-5.614],[0,0],[-0.626,6.419],[-0.71,7.782],[-0.008,7.867],[2.39,17.018],[0,0],[0.001,-29.699],[0.265,-8.64],[0.018,-0.591],[0.053,-1.731],[0.035,-1.132],[0.229,-2.744],[0.065,-0.775],[0.33,-1.589]],"o":[[0,0],[0.551,-5.291],[0.666,-6.833],[0.665,-7.295],[0.026,-24.593],[0,0],[3.002,16.879],[0,9.609],[-0.019,0.611],[-0.055,1.791],[-0.036,1.166],[-0.091,2.946],[-0.066,0.789],[-0.142,1.707],[-1.782,8.583]],"v":[[-101.417,113.619],[-128.089,103.122],[-126.208,88.33],[-123.669,60.587],[-123.347,39.44],[-127.868,-28.825],[-101.708,-38.333],[-95.058,36.963],[-95.879,64.483],[-95.935,66.288],[-96.316,71.692],[-96.422,75.136],[-97.418,87.38],[-97.614,89.727],[-98.031,91.512]],"c":true}]},{"t":68,"s":[{"i":[[-3.101,-5.676],[0,0],[1.656,6.449],[1.055,7.814],[0.159,7.867],[-6.646,17.181],[0,0],[-0.585,-29.699],[-1.067,-8.644],[-0.076,-0.591],[-0.251,-1.732],[-0.192,-1.132],[-0.592,-2.753],[-0.177,-0.778],[-0.426,-1.623]],"o":[[0,0],[-2.046,-5.319],[-1.763,-6.865],[-0.989,-7.325],[-0.498,-24.593],[0,0],[-6.722,16.259],[0.189,9.609],[0.076,0.611],[0.231,1.792],[0.169,1.167],[0.5,2.948],[0.162,0.792],[0.389,1.713],[2.289,8.71]],"v":[[-110.325,112.418],[-140.235,101.9],[-145.82,84.174],[-150.097,62.066],[-151.87,39.216],[-142.367,-24.868],[-115.135,-35.675],[-125.325,36.66],[-123.406,64.165],[-123.176,65.97],[-122.424,71.263],[-121.872,74.708],[-120.228,83.267],[-119.707,85.621],[-118.484,90.612]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"gf","o":{"a":0,"k":100},"r":1,"bm":0,"g":{"p":5,"k":{"a":0,"k":[0.27,0.541,0.282,0.745,0.617,0.771,0.476,0.869,0.854,1,0.671,0.992,0.919,0.691,0.394,0.736,0.99,0.381,0.117,0.481]}},"s":{"a":0,"k":[-136.13,-9.791]},"e":{"a":0,"k":[-159.542,95.879]},"t":1,"nm":"Gradient Fill 3","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Line 2","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[{"i":[[-13.149,-20.596],[0,0],[0.816,41.184],[-9.351,23.054],[0,0],[-1.071,-53.108]],"o":[[0,0],[-16.2,-29.348],[-0.824,-41.609],[0,0],[-14.297,18.654],[0.981,48.655]],"v":[[-4.731,149.532],[-34.218,139.734],[-56.766,37.215],[-40.59,-65.262],[-5.691,-79.119],[-28.402,38.265]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[5.38,-19.243],[0,0],[-0.017,41.184],[6.665,20.226],[0,0],[0.079,-53.108]],"o":[[0,0],[4.578,-26.923],[0.017,-41.609],[0,0],[6.709,18.798],[-0.073,48.655]],"v":[[1.932,150.441],[-27.333,140.643],[-13.795,36.908],[-25.486,-66.496],[3.625,-77.324],[17.936,37.911]],"c":true}]},{"t":68,"s":[{"i":[[-13.149,-20.596],[0,0],[0.816,41.184],[-9.351,23.054],[0,0],[-1.071,-53.108]],"o":[[0,0],[-16.2,-29.348],[-0.824,-41.609],[0,0],[-14.297,18.654],[0.981,48.655]],"v":[[-4.731,149.532],[-34.218,139.734],[-56.766,37.215],[-40.59,-65.262],[-5.691,-79.119],[-28.402,38.265]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"gf","o":{"a":0,"k":100},"r":1,"bm":0,"g":{"p":5,"k":{"a":0,"k":[0.27,0.541,0.282,0.745,0.633,0.771,0.476,0.869,0.88,1,0.669,0.992,0.918,0.691,0.393,0.736,0.99,0.381,0.117,0.481]}},"s":{"a":0,"k":[-29.592,-49.443]},"e":{"a":0,"k":[-65.714,125.633]},"t":1,"nm":"Gradient Fill 2","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Line 3","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[{"i":[[1.623,29.602],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0]],"v":[[29.707,125.517],[-227.265,64.982],[-217.788,73.424],[51.973,169.461]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[-0.737,29.602],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0]],"v":[[71.516,125.517],[-227.265,64.982],[-217.788,73.424],[61.404,169.461]],"c":true}]},{"t":68,"s":[{"i":[[1.623,29.602],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0]],"v":[[29.707,125.517],[-227.265,64.982],[-217.788,73.424],[51.973,169.461]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"gf","o":{"a":0,"k":100},"r":1,"bm":0,"g":{"p":3,"k":{"a":0,"k":[0.294,0.603,0.425,0.029,0.9,0.802,0.713,0.211,1,1,1,0.392,0,1,0.714,0.5,1,0]}},"s":{"a":0,"k":[-84.178,116.979]},"e":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[-80.272,101.164],"to":[0,0],"ti":[0,0]},{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[-80.109,98.498],"to":[0,0],"ti":[0,0]},{"t":68,"s":[-80.272,101.164]}]},"t":1,"nm":"Gradient Fill 77","hd":false},{"ty":"tr","p":{"a":0,"k":[-0.012,-1.852]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shadow Bottom","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[{"i":[[-18.016,-13.169],[0,0],[-34.744,13.8],[0,0],[-1.31,-66.003]],"o":[[0,0],[-35.268,-12.393],[0,0],[-17.495,13.858],[1.309,65.982]],"v":[[51.973,169.461],[-208.808,77.803],[-210.316,2.099],[46.625,-99.876],[18.248,35.42]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[8.181,-13.169],[0,0],[-0.625,7.36],[0,0],[0.595,-66.003]],"o":[[0,0],[-4.68,-2.017],[0,0],[7.945,13.858],[-0.595,65.982]],"v":[[61.405,169.461],[-222.547,74.202],[-222.438,6.736],[63.833,-99.876],[88.725,35.41]],"c":true}]},{"t":68,"s":[{"i":[[-18.016,-13.169],[0,0],[-34.744,13.8],[0,0],[-1.31,-66.003]],"o":[[0,0],[-35.268,-12.393],[0,0],[-17.495,13.858],[1.309,65.982]],"v":[[51.973,169.461],[-208.808,77.803],[-210.316,2.099],[46.625,-99.876],[18.248,35.42]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"gf","o":{"a":0,"k":100},"r":1,"bm":0,"g":{"p":5,"k":{"a":0,"k":[0.106,0.754,0.585,0.107,0.424,0.857,0.678,0.165,0.716,0.961,0.771,0.224,0.82,0.98,0.885,0.308,1,1,1,0.392]}},"s":{"a":0,"k":[-41.806,-28]},"e":{"a":0,"k":[-41.935,88.46]},"t":1,"nm":"Gradient Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"FILL MAIN","bm":0,"hd":false},{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[-5.961,-5.348],[0,0],[0,0],[0,0],[-12.811,-12.002]],"o":[[0,0],[0,0],[0,0],[10.358,10.651],[6.231,5.837]],"v":[[41.004,53.997],[0.364,-9.21],[-43.229,-43.427],[-12.35,2.891],[22.681,37.189]],"c":true}]},{"i":{"x":0.833,"y":1},"o":{"x":0.167,"y":0},"t":68,"s":[{"i":[[-6.09,-6.17],[0,0],[0,0],[0,0],[-14.17,-12.64]],"o":[[0,0],[0,0],[0,0],[13.46,9.08],[6.89,6.15]],"v":[[66.19,30.38],[-32.03,20.66],[-66.19,-30.38],[4.82,-20.93],[46.69,11.86]],"c":true}]},{"t":70,"s":[{"i":[[-6.09,-6.17],[0,0],[0,0],[0,0],[-14.17,-12.64]],"o":[[0,0],[0,0],[0,0],[13.46,9.08],[6.89,6.15]],"v":[[70.056,34.764],[-25.589,28.686],[-66.19,-30.38],[4.82,-20.93],[46.69,11.86]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"gf","o":{"a":0,"k":100},"r":1,"bm":0,"g":{"p":5,"k":{"a":0,"k":[0.004,1,1,1,0.091,1,1,1,0.5,1,1,1,0.898,1,1,1,1,1,1,1,0,0,0.084,0.5,0.5,1,0.892,0.5,1,0]}},"s":{"a":0,"k":[6.7,-18.814]},"e":{"a":0,"k":[1.45,29.107]},"t":1,"nm":"Gradient Fill 147","hd":false},{"ty":"tr","p":{"a":0,"k":[232.268,185.143]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":20},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 1","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[-6.005,-4.739]],"o":[[0,0],[0,0],[0,0],[6.991,5.955],[0,0]],"v":[[33.117,29.167],[-2.118,-26.309],[-24.364,-46.726],[13.512,13.04],[33.085,29.145]],"c":true}]},{"i":{"x":0.833,"y":1},"o":{"x":0.167,"y":0},"t":68,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[-4.76,-7]],"o":[[0,0],[0,0],[0,0],[6.47,7.62],[0,0]],"v":[[55.145,15.04],[-31.445,7.43],[-55.145,-15.04],[38.215,-7.02],[55.125,15]],"c":true}]},{"t":70,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[-4.76,-7]],"o":[[0,0],[0,0],[0,0],[6.47,7.62],[0,0]],"v":[[58.573,22.058],[-24.576,16.712],[-55.145,-15.04],[38.215,-7.02],[55.125,15]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"gf","o":{"a":0,"k":100},"r":1,"bm":0,"g":{"p":5,"k":{"a":0,"k":[0.004,1,1,1,0.091,1,1,1,0.5,1,1,1,0.898,1,1,1,1,1,1,1,0,0,0.084,0.5,0.5,1,0.892,0.5,1,0]}},"s":{"a":0,"k":[-0.314,-9.257]},"e":{"a":0,"k":[-1.281,17.248]},"t":1,"nm":"Gradient Fill 14777778","hd":false},{"ty":"tr","p":{"a":0,"k":[278.802,242.763]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":20},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 2","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[-4.448,4.325],[0,0],[0.003,0.006],[40.068,37.545],[11.822,4.167],[0,0],[-46.227,-43.315]],"o":[[0,0],[-0.003,-0.006],[-4.594,-11.209],[-40.081,-37.558],[0,0],[-4.448,4.325],[46.227,43.315]],"v":[[85.771,64.558],[85.578,61.086],[85.571,61.075],[15.793,-16.865],[-72.324,-87.35],[-75.971,-87.476],[-0.322,-1.216]],"c":true}]},{"t":68,"s":[{"i":[[13.086,-14.661],[0,0],[0.009,0],[44.316,39.546],[-1.575,20.025],[0,0],[-51.129,-45.621]],"o":[[0,0],[-0.009,0],[-19.719,3.816],[-44.334,-39.555],[0,0],[13.086,-14.661],[51.129,45.621]],"v":[[87.33,76.433],[80.139,80.222],[80.12,80.222],[-23.68,26.545],[-95.527,-75.029],[-92.584,-82.607],[23.687,-26.546]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.839215695858,0.505882382393,0.070588238537,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[255.268,223.545]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 3","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[-4.396,4.372],[0,0],[0.003,0.006],[39.928,37.704],[11.751,4.162],[0,0],[-46.065,-43.498]],"o":[[0,0],[-0.003,-0.006],[-4.608,-11.279],[-39.941,-37.716],[0,0],[-4.396,4.372],[46.065,43.498]],"v":[[72.425,79.252],[72.218,75.754],[72.212,75.743],[2.624,-2.568],[-85.132,-73.309],[-88.752,-73.425],[-13.302,13.25]],"c":true}]},{"t":68,"s":[{"i":[[13.086,-14.661],[0,0],[0.009,0],[44.316,39.546],[-1.575,20.025],[0,0],[-51.129,-45.621]],"o":[[0,0],[-0.009,0],[-19.719,3.816],[-44.334,-39.555],[0,0],[13.086,-14.661],[51.129,45.621]],"v":[[87.33,76.433],[80.139,80.222],[80.12,80.222],[-23.68,26.545],[-95.527,-75.029],[-92.584,-82.607],[23.687,-26.546]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.694117665291,0.411992400885,0,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[264.318,211.249]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[107.208,106.592]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Group 4","bm":0,"hd":false},{"ty":"tr","p":{"a":0,"k":[40.158,41.218]},"a":{"a":0,"k":[255.268,223.545]},"s":{"a":0,"k":[107.425,107.425]},"r":{"a":0,"k":47.218},"o":{"a":1,"k":[{"t":0,"s":[0],"h":1},{"t":47,"s":[100],"h":1},{"t":77,"s":[0],"h":1}]},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"cover","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[{"i":[[-2.193,-11.693],[0,0],[-5.92,23.132],[-12.41,-13.217],[0,0],[-1.519,-2.533]],"o":[[0,0],[0,0],[9.429,-36.842],[0,0],[1.735,1.883],[6.871,11.455]],"v":[[87.163,-48.611],[23.98,-30.769],[28.122,-63.164],[68.414,-95.532],[68.564,-95.371],[73.446,-88.688]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[0.996,-11.693],[0,0],[2.689,23.132],[5.636,-13.217],[0,0],[0.69,-2.533]],"o":[[0,0],[0,0],[-4.282,-36.842],[0,0],[-0.788,1.883],[-3.12,11.455]],"v":[[45.424,-48.611],[81.73,-30.287],[72.236,-63.164],[53.938,-95.532],[53.87,-95.371],[51.653,-88.688]],"c":true}]},{"t":68,"s":[{"i":[[-2.193,-11.693],[0,0],[-5.92,23.132],[-12.41,-13.217],[0,0],[-1.519,-2.533]],"o":[[0,0],[0,0],[9.429,-36.842],[0,0],[1.735,1.883],[6.871,11.455]],"v":[[87.163,-48.611],[23.98,-30.769],[28.122,-63.164],[68.414,-95.532],[68.564,-95.371],[73.446,-88.688]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"gf","o":{"a":0,"k":100},"r":1,"bm":0,"g":{"p":3,"k":{"a":0,"k":[0.294,0.603,0.425,0.029,0.9,0.713,0.514,0.077,1,0.824,0.604,0.125,0,1,0.714,0.5,1,0]}},"s":{"a":0,"k":[51.382,-91.352]},"e":{"a":0,"k":[58.598,-42.726]},"t":1,"nm":"Gradient Fill 77","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Sh Top","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[{"i":[[4.41,-32.122],[0,0],[-1.878,21.686],[0,0]],"o":[[0,0],[-1.386,-15.13],[0,0],[0,0]],"v":[[94.172,104.303],[22.637,87.669],[19.264,5.356],[93.119,-9.953]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[-2.003,-32.122],[0,0],[0.108,7.739],[0,0]],"o":[[0,0],[0.63,-15.13],[0,0],[0,0]],"v":[[41.604,88.949],[81.457,97.711],[87.434,1.843],[41.884,-2.163]],"c":true}]},{"t":68,"s":[{"i":[[4.41,-32.122],[0,0],[-1.878,21.686],[0,0]],"o":[[0,0],[-1.386,-15.13],[0,0],[0,0]],"v":[[94.172,104.303],[22.637,87.669],[19.264,5.356],[93.119,-9.953]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"gf","o":{"a":0,"k":100},"r":1,"bm":0,"g":{"p":5,"k":{"a":0,"k":[0,0.988,0.839,0.259,0.136,0.988,0.839,0.259,0.497,0.988,0.839,0.259,0.839,0.988,0.839,0.259,0.99,0.988,0.839,0.259,0,0,0.11,0.5,0.493,1,0.856,0.505,1,0.01]}},"s":{"a":0,"k":[54.658,1.171]},"e":{"a":0,"k":[53.727,87.153]},"t":1,"nm":"Gradient Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shine Back","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[{"i":[[12.468,37.763],[0,0],[9.012,-3.755]],"o":[[0,0],[0,0],[-9.012,3.755]],"v":[[33.263,135.969],[80.615,152.085],[65.835,171.098]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[-5.662,37.763],[0,0],[-4.092,-3.755]],"o":[[0,0],[0,0],[4.092,3.755]],"v":[[69.901,135.969],[48.398,152.085],[56.094,167.092]],"c":true}]},{"t":68,"s":[{"i":[[12.468,37.763],[0,0],[9.012,-3.755]],"o":[[0,0],[0,0],[-9.012,3.755]],"v":[[33.263,135.969],[80.615,152.085],[65.835,171.098]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"gf","o":{"a":0,"k":100},"r":1,"bm":0,"g":{"p":3,"k":{"a":0,"k":[0.192,0.988,0.839,0.259,0.885,0.906,0.722,0.192,1,0.824,0.604,0.125,0,1,0.714,0.5,1,0]}},"s":{"a":0,"k":[61.011,166.644]},"e":{"a":0,"k":[64.839,147.715]},"t":1,"nm":"Gradient Fill 77","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Sh Bot","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[{"i":[[21.831,-0.431],[0,0],[0.007,0.008],[1.315,65.982],[-17.495,13.858],[0,0],[-1.514,-76.122]],"o":[[0,0],[-0.007,-0.008],[-18.016,-13.169],[-1.316,-66.003],[0,0],[21.831,-0.431],[1.514,76.122]],"v":[[60.509,172.465],[51.987,169.476],[51.973,169.461],[18.248,35.42],[46.625,-99.876],[55.02,-103.21],[97.288,33.844]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.31,"y":0},"t":42,"s":[{"i":[[-9.914,-0.431],[0,0],[-0.003,0.008],[-0.597,65.982],[7.945,13.858],[0,0],[0.688,-76.122]],"o":[[0,0],[0.003,-0.008],[8.181,-13.169],[0.597,-66.003],[0,0],[-9.914,-0.431],[-0.688,76.122]],"v":[[58.512,168.459],[61.398,169.476],[61.404,169.461],[88.725,35.41],[63.833,-99.876],[61.149,-98.971],[40.826,33.844]],"c":true}]},{"t":68,"s":[{"i":[[21.831,-0.431],[0,0],[0.007,0.008],[1.315,65.982],[-17.495,13.858],[0,0],[-1.514,-76.122]],"o":[[0,0],[-0.007,-0.008],[-18.016,-13.169],[-1.316,-66.003],[0,0],[21.831,-0.431],[1.514,76.122]],"v":[[60.509,172.465],[51.987,169.476],[51.973,169.461],[18.248,35.42],[46.625,-99.876],[55.02,-103.21],[97.288,33.844]],"c":true}]}]},"nm":"Path 1","hd":false},{"ty":"fl","c":{"a":0,"k":[0.822745050169,0.604556214576,0.124540919884,1]},"o":{"a":0,"k":100},"r":1,"bm":0,"nm":"Fill 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Fill Back","bm":0,"hd":false}],"ip":0,"op":180,"st":45,"bm":0},{"ddd":0,"ind":40,"ty":4,"nm":"yellow2","parent":2,"sr":1,"ks":{"o":{"a":1,"k":[{"t":0,"s":[100],"h":1},{"t":37,"s":[0],"h":1},{"t":128,"s":[100],"h":1}]},"p":{"a":0,"k":[104,2,0]},"s":{"a":0,"k":[-57.168,99.39,100]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[0,0],[146.924,4.823],[50.863,-19.845],[-24.402,-18.879],[62.018,-68.887],[-15.209,-21.02]],"o":[[0,0],[-52.269,-1.716],[-74.021,28.88],[24.216,18.735],[-57.31,63.657],[10.153,14.033]],"v":[[178,-232],[150.138,-113.293],[167.021,-55.88],[165.868,-2.012],[162.31,115.343],[172.209,230.02]],"c":false}},"nm":"Path 1","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":0,"s":[13.731]},{"i":{"x":[0.69],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":45,"s":[100]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":141,"s":[5]},{"t":179,"s":[13.731]}]},"e":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":0,"s":[44.372]},{"i":{"x":[0.69],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":28.834,"s":[100]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":111,"s":[0]},{"t":179,"s":[44.372]}]},"o":{"a":0,"k":0},"m":1,"nm":"Trim Paths 1","hd":false},{"ty":"st","c":{"a":0,"k":[0.972549019608,0.96862745098,0.290196078431,1]},"o":{"a":0,"k":100},"w":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":0,"s":[15]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":28.834,"s":[15]},{"i":{"x":[0.69],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":44.717,"s":[0]},{"i":{"x":[0.69],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":123,"s":[0]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":141,"s":[15]},{"t":179,"s":[15]}]},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 2","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[0,0],[146.924,4.823],[50.863,-19.845],[-24.402,-18.879],[62.018,-68.887],[-15.209,-21.02]],"o":[[0,0],[-52.269,-1.716],[-74.021,28.88],[24.216,18.735],[-57.31,63.657],[10.153,14.033]],"v":[[178,-232],[160.138,-109.293],[174.021,-52.88],[170.868,-6.012],[162.31,117.343],[172.209,230.02]],"c":false}},"nm":"Path 1","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":0,"s":[13.731]},{"i":{"x":[0.69],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":44.717,"s":[100]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":141,"s":[5]},{"t":179,"s":[13.731]}]},"e":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":0,"s":[44.372]},{"i":{"x":[0.69],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":28.834,"s":[100]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":111,"s":[0]},{"t":179,"s":[44.372]}]},"o":{"a":0,"k":0},"m":1,"nm":"Trim Paths 1","hd":false},{"ty":"st","c":{"a":0,"k":[0.737254901961,0.737254901961,0.258823529412,1]},"o":{"a":0,"k":100},"w":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":0,"s":[15]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":28.834,"s":[15]},{"i":{"x":[0.69],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":44.717,"s":[0]},{"i":{"x":[0.69],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":123,"s":[0]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":141,"s":[15]},{"t":179,"s":[15]}]},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 3","bm":0,"hd":false}],"ip":0,"op":180,"st":58,"bm":0},{"ddd":0,"ind":41,"ty":4,"nm":"blue 2","parent":2,"sr":1,"ks":{"o":{"a":1,"k":[{"t":0,"s":[100],"h":1},{"t":27,"s":[0],"h":1},{"t":98,"s":[100],"h":1}]},"s":{"a":0,"k":[-100,100,100]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[0,0],[128.524,2.108],[50.863,-19.845],[-24.402,-18.879],[62.018,-68.887],[-15.209,-21.02]],"o":[[0,0],[-52.29,-0.858],[-74.021,28.88],[24.216,18.735],[-57.31,63.657],[10.153,14.033]],"v":[[178.042,-202.025],[150.138,-113.293],[167.021,-55.88],[165.868,-2.012],[162.31,115.343],[172.209,230.02]],"c":false}},"nm":"Path 1","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":0,"s":[56.383]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":31.717,"s":[100]},{"i":{"x":[0.3],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":117.283,"s":[0]},{"t":179,"s":[56.383]}]},"e":{"a":1,"k":[{"i":{"x":[0.651],"y":[1]},"o":{"x":[0.318],"y":[0.095]},"t":0,"s":[90.64]},{"i":{"x":[0.651],"y":[4.401]},"o":{"x":[0.316],"y":[0]},"t":9.434,"s":[100]},{"i":{"x":[0.834],"y":[0.887]},"o":{"x":[0.342],"y":[0.09]},"t":32,"s":[100]},{"i":{"x":[0.42],"y":[0.847]},"o":{"x":[0.177],"y":[0.218]},"t":93,"s":[0]},{"t":179,"s":[96]}]},"o":{"a":0,"k":0},"m":1,"nm":"Trim Paths 1","hd":false},{"ty":"st","c":{"a":0,"k":[0.137254901961,0.364705882353,0.992156862745,1]},"o":{"a":0,"k":100},"w":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":0,"s":[15]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":9.434,"s":[15]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":31.717,"s":[0]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":93,"s":[0]},{"i":{"x":[0.3],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":117.283,"s":[15]},{"t":179,"s":[15]}]},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 2","bm":0,"hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ks":{"a":0,"k":{"i":[[0,0],[121.781,1.829],[50.863,-19.845],[-24.402,-18.879],[62.018,-68.887],[-15.209,-21.02]],"o":[[0,0],[-52.291,-0.786],[-74.021,28.88],[24.216,18.735],[-57.31,63.657],[10.153,14.033]],"v":[[178.979,-203.9],[160.138,-109.293],[174.021,-52.88],[170.868,-6.012],[162.31,117.343],[172.209,230.02]],"c":false}},"nm":"Path 1","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":0,"s":[56.383]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":31.717,"s":[100]},{"i":{"x":[0.3],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":117.283,"s":[0]},{"t":179,"s":[56.383]}]},"e":{"a":1,"k":[{"i":{"x":[0.651],"y":[1]},"o":{"x":[0.318],"y":[0.111]},"t":0,"s":[92]},{"i":{"x":[0.651],"y":[4.401]},"o":{"x":[0.316],"y":[0]},"t":9.434,"s":[100]},{"i":{"x":[0.834],"y":[0.887]},"o":{"x":[0.342],"y":[0.09]},"t":32,"s":[100]},{"i":{"x":[0.42],"y":[0.847]},"o":{"x":[0.177],"y":[0.218]},"t":93,"s":[0]},{"t":179,"s":[96]}]},"o":{"a":0,"k":0},"m":1,"nm":"Trim Paths 1","hd":false},{"ty":"st","c":{"a":0,"k":[0,0.294117647059,0.717647058824,1]},"o":{"a":0,"k":100},"w":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":0,"s":[15]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":9.434,"s":[15]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":31.717,"s":[0]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":93,"s":[0]},{"i":{"x":[0.3],"y":[1]},"o":{"x":[0.31],"y":[0]},"t":117.283,"s":[15]},{"t":179,"s":[15]}]},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0]},"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"r":{"a":0,"k":0},"o":{"a":0,"k":100},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0},"nm":"Transform"}],"nm":"Shape 3","bm":0,"hd":false}],"ip":0,"op":180,"st":45,"bm":0}]} \ No newline at end of file diff --git a/TMessagesProj/src/main/res/raw/ic_boosts_replace.json b/TMessagesProj/src/main/res/raw/ic_boosts_replace.json new file mode 100644 index 00000000000..7acf76a8b39 --- /dev/null +++ b/TMessagesProj/src/main/res/raw/ic_boosts_replace.json @@ -0,0 +1 @@ +{"v":"5.12.1","fr":60,"ip":0,"op":90,"w":512,"h":512,"nm":"Comp 4","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":4,"nm":"ыArtboard Outlines 3","parent":3,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":11,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":12,"s":[0]},{"i":{"x":[0.833],"y":[1.14]},"o":{"x":[0.167],"y":[0]},"t":13,"s":[0]},{"i":{"x":[0.833],"y":[0.346]},"o":{"x":[0.167],"y":[0.083]},"t":14,"s":[0]},{"i":{"x":[0.833],"y":[0.854]},"o":{"x":[0.167],"y":[0.096]},"t":15,"s":[-1.683]},{"i":{"x":[0.833],"y":[0.717]},"o":{"x":[0.167],"y":[0.194]},"t":16,"s":[-13.202]},{"i":{"x":[0.833],"y":[0.825]},"o":{"x":[0.167],"y":[0.118]},"t":17,"s":[-21.881]},{"i":{"x":[0.833],"y":[0.875]},"o":{"x":[0.167],"y":[0.159]},"t":18,"s":[-42.689]},{"i":{"x":[0.833],"y":[0.756]},"o":{"x":[0.167],"y":[0.249]},"t":19,"s":[-65.632]},{"i":{"x":[0.833],"y":[0.841]},"o":{"x":[0.167],"y":[0.127]},"t":20,"s":[-77.183]},{"i":{"x":[0.833],"y":[0.879]},"o":{"x":[0.167],"y":[0.175]},"t":21,"s":[-99.424]},{"i":{"x":[0.833],"y":[0.77]},"o":{"x":[0.167],"y":[0.268]},"t":22,"s":[-119.61]},{"i":{"x":[0.833],"y":[0.849]},"o":{"x":[0.167],"y":[0.131]},"t":23,"s":[-128.714]},{"i":{"x":[0.833],"y":[0.882]},"o":{"x":[0.167],"y":[0.186]},"t":24,"s":[-144.735]},{"i":{"x":[0.833],"y":[0.781]},"o":{"x":[0.167],"y":[0.283]},"t":25,"s":[-157.784]},{"i":{"x":[0.833],"y":[0.856]},"o":{"x":[0.167],"y":[0.134]},"t":26,"s":[-163.227]},{"i":{"x":[0.833],"y":[0.885]},"o":{"x":[0.167],"y":[0.198]},"t":27,"s":[-172.101]},{"i":{"x":[0.833],"y":[0.796]},"o":{"x":[0.167],"y":[0.304]},"t":28,"s":[-178.567]},{"i":{"x":[0.833],"y":[0.868]},"o":{"x":[0.167],"y":[0.141]},"t":29,"s":[-181.011]},{"i":{"x":[0.833],"y":[0.893]},"o":{"x":[0.167],"y":[0.225]},"t":30,"s":[-184.564]},{"i":{"x":[0.833],"y":[0.845]},"o":{"x":[0.167],"y":[0.372]},"t":31,"s":[-186.647]},{"i":{"x":[0.833],"y":[0.943]},"o":{"x":[0.167],"y":[0.18]},"t":32,"s":[-187.247]},{"i":{"x":[0.833],"y":[0.782]},"o":{"x":[0.167],"y":[-0.184]},"t":33,"s":[-187.765]},{"i":{"x":[0.833],"y":[0.675]},"o":{"x":[0.167],"y":[0.135]},"t":34,"s":[-187.603]},{"i":{"x":[0.833],"y":[0.813]},"o":{"x":[0.167],"y":[0.112]},"t":35,"s":[-187.343]},{"i":{"x":[0.833],"y":[0.872]},"o":{"x":[0.167],"y":[0.15]},"t":36,"s":[-186.587]},{"i":{"x":[0.833],"y":[0.75]},"o":{"x":[0.167],"y":[0.24]},"t":37,"s":[-185.649]},{"i":{"x":[0.833],"y":[0.838]},"o":{"x":[0.167],"y":[0.125]},"t":38,"s":[-185.151]},{"i":{"x":[0.833],"y":[0.878]},"o":{"x":[0.167],"y":[0.172]},"t":39,"s":[-184.156]},{"i":{"x":[0.833],"y":[0.767]},"o":{"x":[0.167],"y":[0.264]},"t":40,"s":[-183.216]},{"i":{"x":[0.833],"y":[0.847]},"o":{"x":[0.167],"y":[0.13]},"t":41,"s":[-182.782]},{"i":{"x":[0.833],"y":[0.881]},"o":{"x":[0.167],"y":[0.183]},"t":42,"s":[-182.001]},{"i":{"x":[0.833],"y":[0.778]},"o":{"x":[0.167],"y":[0.279]},"t":43,"s":[-181.346]},{"i":{"x":[0.833],"y":[0.854]},"o":{"x":[0.167],"y":[0.133]},"t":44,"s":[-181.067]},{"i":{"x":[0.833],"y":[0.884]},"o":{"x":[0.167],"y":[0.194]},"t":45,"s":[-180.601]},{"i":{"x":[0.833],"y":[0.79]},"o":{"x":[0.167],"y":[0.296]},"t":46,"s":[-180.25]},{"i":{"x":[0.833],"y":[0.863]},"o":{"x":[0.167],"y":[0.138]},"t":47,"s":[-180.112]},{"i":{"x":[0.833],"y":[0.889]},"o":{"x":[0.167],"y":[0.214]},"t":48,"s":[-179.904]},{"i":{"x":[0.833],"y":[0.82]},"o":{"x":[0.167],"y":[0.338]},"t":49,"s":[-179.771]},{"i":{"x":[0.833],"y":[0.896]},"o":{"x":[0.167],"y":[0.155]},"t":50,"s":[-179.727]},{"i":{"x":[0.833],"y":[0.944]},"o":{"x":[0.167],"y":[0.416]},"t":51,"s":[-179.677]},{"i":{"x":[0.833],"y":[0.456]},"o":{"x":[0.167],"y":[-0.17]},"t":52,"s":[-179.664]},{"i":{"x":[0.833],"y":[0.787]},"o":{"x":[0.167],"y":[0.098]},"t":53,"s":[-179.668]},{"i":{"x":[0.833],"y":[0.869]},"o":{"x":[0.167],"y":[0.137]},"t":54,"s":[-179.691]},{"i":{"x":[0.833],"y":[0.742]},"o":{"x":[0.167],"y":[0.228]},"t":55,"s":[-179.727]},{"i":{"x":[0.833],"y":[0.834]},"o":{"x":[0.167],"y":[0.123]},"t":56,"s":[-179.747]},{"i":{"x":[0.833],"y":[0.877]},"o":{"x":[0.167],"y":[0.167]},"t":57,"s":[-179.79]},{"i":{"x":[0.833],"y":[0.763]},"o":{"x":[0.167],"y":[0.258]},"t":58,"s":[-179.833]},{"i":{"x":[0.833],"y":[0.845]},"o":{"x":[0.167],"y":[0.129]},"t":59,"s":[-179.853]},{"i":{"x":[0.833],"y":[0.88]},"o":{"x":[0.167],"y":[0.18]},"t":60,"s":[-179.89]},{"i":{"x":[0.833],"y":[0.775]},"o":{"x":[0.167],"y":[0.275]},"t":61,"s":[-179.922]},{"i":{"x":[0.833],"y":[0.852]},"o":{"x":[0.167],"y":[0.132]},"t":62,"s":[-179.936]},{"i":{"x":[0.833],"y":[0.883]},"o":{"x":[0.167],"y":[0.19]},"t":63,"s":[-179.96]},{"i":{"x":[0.833],"y":[0.786]},"o":{"x":[0.167],"y":[0.291]},"t":64,"s":[-179.979]},{"i":{"x":[0.833],"y":[0.86]},"o":{"x":[0.167],"y":[0.137]},"t":65,"s":[-179.986]},{"i":{"x":[0.833],"y":[0.887]},"o":{"x":[0.167],"y":[0.206]},"t":66,"s":[-179.998]},{"i":{"x":[0.833],"y":[0.807]},"o":{"x":[0.167],"y":[0.32]},"t":67,"s":[-180.006]},{"i":{"x":[0.833],"y":[0.88]},"o":{"x":[0.167],"y":[0.147]},"t":68,"s":[-180.009]},{"i":{"x":[0.833],"y":[1.011]},"o":{"x":[0.167],"y":[0.27]},"t":69,"s":[-180.013]},{"i":{"x":[0.833],"y":[0.835]},"o":{"x":[0.167],"y":[0.146]},"t":70,"s":[-180.014]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":183,"s":[-180]},{"t":184,"s":[-180]}],"ix":10},"p":{"a":0,"k":[20.937,22.342,0],"ix":2,"l":2},"a":{"a":0,"k":[20.937,22.342,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[-0.048,0.305],[0,0],[0.302,-0.441],[0,0],[-0.404,0],[0,0],[0.048,-0.305],[0,0],[-0.302,0.442],[0,0],[0.403,0]],"o":[[-0.308,0],[0,0],[0.084,-0.528],[0,0],[-0.228,0.333],[0,0],[0.308,0],[0,0],[-0.083,0.528],[0,0],[0.228,-0.333],[0,0]],"v":[[7.422,6.088],[6.927,5.508],[7.597,1.262],[6.687,0.9],[1.958,7.812],[2.373,8.597],[4.452,8.597],[4.947,9.177],[4.277,13.423],[5.187,13.784],[9.916,6.873],[9.501,6.088]],"c":true},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"mm","mm":1,"nm":"Merge Paths 1","mn":"ADBE Vector Filter - Merge","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[15,15],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 5","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":90,"st":5,"ct":1,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"ыArtboard Outlines 2","parent":3,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":11,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":12,"s":[0]},{"i":{"x":[0.833],"y":[1.14]},"o":{"x":[0.167],"y":[0]},"t":13,"s":[0]},{"i":{"x":[0.833],"y":[0.346]},"o":{"x":[0.167],"y":[0.083]},"t":14,"s":[0]},{"i":{"x":[0.833],"y":[0.854]},"o":{"x":[0.167],"y":[0.096]},"t":15,"s":[-1.683]},{"i":{"x":[0.833],"y":[0.717]},"o":{"x":[0.167],"y":[0.194]},"t":16,"s":[-13.202]},{"i":{"x":[0.833],"y":[0.825]},"o":{"x":[0.167],"y":[0.118]},"t":17,"s":[-21.881]},{"i":{"x":[0.833],"y":[0.875]},"o":{"x":[0.167],"y":[0.159]},"t":18,"s":[-42.689]},{"i":{"x":[0.833],"y":[0.756]},"o":{"x":[0.167],"y":[0.249]},"t":19,"s":[-65.632]},{"i":{"x":[0.833],"y":[0.841]},"o":{"x":[0.167],"y":[0.127]},"t":20,"s":[-77.183]},{"i":{"x":[0.833],"y":[0.879]},"o":{"x":[0.167],"y":[0.175]},"t":21,"s":[-99.424]},{"i":{"x":[0.833],"y":[0.77]},"o":{"x":[0.167],"y":[0.268]},"t":22,"s":[-119.61]},{"i":{"x":[0.833],"y":[0.849]},"o":{"x":[0.167],"y":[0.131]},"t":23,"s":[-128.714]},{"i":{"x":[0.833],"y":[0.882]},"o":{"x":[0.167],"y":[0.186]},"t":24,"s":[-144.735]},{"i":{"x":[0.833],"y":[0.781]},"o":{"x":[0.167],"y":[0.283]},"t":25,"s":[-157.784]},{"i":{"x":[0.833],"y":[0.856]},"o":{"x":[0.167],"y":[0.134]},"t":26,"s":[-163.227]},{"i":{"x":[0.833],"y":[0.885]},"o":{"x":[0.167],"y":[0.198]},"t":27,"s":[-172.101]},{"i":{"x":[0.833],"y":[0.796]},"o":{"x":[0.167],"y":[0.304]},"t":28,"s":[-178.567]},{"i":{"x":[0.833],"y":[0.868]},"o":{"x":[0.167],"y":[0.141]},"t":29,"s":[-181.011]},{"i":{"x":[0.833],"y":[0.893]},"o":{"x":[0.167],"y":[0.225]},"t":30,"s":[-184.564]},{"i":{"x":[0.833],"y":[0.845]},"o":{"x":[0.167],"y":[0.372]},"t":31,"s":[-186.647]},{"i":{"x":[0.833],"y":[0.943]},"o":{"x":[0.167],"y":[0.18]},"t":32,"s":[-187.247]},{"i":{"x":[0.833],"y":[0.782]},"o":{"x":[0.167],"y":[-0.184]},"t":33,"s":[-187.765]},{"i":{"x":[0.833],"y":[0.675]},"o":{"x":[0.167],"y":[0.135]},"t":34,"s":[-187.603]},{"i":{"x":[0.833],"y":[0.813]},"o":{"x":[0.167],"y":[0.112]},"t":35,"s":[-187.343]},{"i":{"x":[0.833],"y":[0.872]},"o":{"x":[0.167],"y":[0.15]},"t":36,"s":[-186.587]},{"i":{"x":[0.833],"y":[0.75]},"o":{"x":[0.167],"y":[0.24]},"t":37,"s":[-185.649]},{"i":{"x":[0.833],"y":[0.838]},"o":{"x":[0.167],"y":[0.125]},"t":38,"s":[-185.151]},{"i":{"x":[0.833],"y":[0.878]},"o":{"x":[0.167],"y":[0.172]},"t":39,"s":[-184.156]},{"i":{"x":[0.833],"y":[0.767]},"o":{"x":[0.167],"y":[0.264]},"t":40,"s":[-183.216]},{"i":{"x":[0.833],"y":[0.847]},"o":{"x":[0.167],"y":[0.13]},"t":41,"s":[-182.782]},{"i":{"x":[0.833],"y":[0.881]},"o":{"x":[0.167],"y":[0.183]},"t":42,"s":[-182.001]},{"i":{"x":[0.833],"y":[0.778]},"o":{"x":[0.167],"y":[0.279]},"t":43,"s":[-181.346]},{"i":{"x":[0.833],"y":[0.854]},"o":{"x":[0.167],"y":[0.133]},"t":44,"s":[-181.067]},{"i":{"x":[0.833],"y":[0.884]},"o":{"x":[0.167],"y":[0.194]},"t":45,"s":[-180.601]},{"i":{"x":[0.833],"y":[0.79]},"o":{"x":[0.167],"y":[0.296]},"t":46,"s":[-180.25]},{"i":{"x":[0.833],"y":[0.863]},"o":{"x":[0.167],"y":[0.138]},"t":47,"s":[-180.112]},{"i":{"x":[0.833],"y":[0.889]},"o":{"x":[0.167],"y":[0.214]},"t":48,"s":[-179.904]},{"i":{"x":[0.833],"y":[0.82]},"o":{"x":[0.167],"y":[0.338]},"t":49,"s":[-179.771]},{"i":{"x":[0.833],"y":[0.896]},"o":{"x":[0.167],"y":[0.155]},"t":50,"s":[-179.727]},{"i":{"x":[0.833],"y":[0.944]},"o":{"x":[0.167],"y":[0.416]},"t":51,"s":[-179.677]},{"i":{"x":[0.833],"y":[0.456]},"o":{"x":[0.167],"y":[-0.17]},"t":52,"s":[-179.664]},{"i":{"x":[0.833],"y":[0.787]},"o":{"x":[0.167],"y":[0.098]},"t":53,"s":[-179.668]},{"i":{"x":[0.833],"y":[0.869]},"o":{"x":[0.167],"y":[0.137]},"t":54,"s":[-179.691]},{"i":{"x":[0.833],"y":[0.742]},"o":{"x":[0.167],"y":[0.228]},"t":55,"s":[-179.727]},{"i":{"x":[0.833],"y":[0.834]},"o":{"x":[0.167],"y":[0.123]},"t":56,"s":[-179.747]},{"i":{"x":[0.833],"y":[0.877]},"o":{"x":[0.167],"y":[0.167]},"t":57,"s":[-179.79]},{"i":{"x":[0.833],"y":[0.763]},"o":{"x":[0.167],"y":[0.258]},"t":58,"s":[-179.833]},{"i":{"x":[0.833],"y":[0.845]},"o":{"x":[0.167],"y":[0.129]},"t":59,"s":[-179.853]},{"i":{"x":[0.833],"y":[0.88]},"o":{"x":[0.167],"y":[0.18]},"t":60,"s":[-179.89]},{"i":{"x":[0.833],"y":[0.775]},"o":{"x":[0.167],"y":[0.275]},"t":61,"s":[-179.922]},{"i":{"x":[0.833],"y":[0.852]},"o":{"x":[0.167],"y":[0.132]},"t":62,"s":[-179.936]},{"i":{"x":[0.833],"y":[0.883]},"o":{"x":[0.167],"y":[0.19]},"t":63,"s":[-179.96]},{"i":{"x":[0.833],"y":[0.786]},"o":{"x":[0.167],"y":[0.291]},"t":64,"s":[-179.979]},{"i":{"x":[0.833],"y":[0.86]},"o":{"x":[0.167],"y":[0.137]},"t":65,"s":[-179.986]},{"i":{"x":[0.833],"y":[0.887]},"o":{"x":[0.167],"y":[0.206]},"t":66,"s":[-179.998]},{"i":{"x":[0.833],"y":[0.807]},"o":{"x":[0.167],"y":[0.32]},"t":67,"s":[-180.006]},{"i":{"x":[0.833],"y":[0.88]},"o":{"x":[0.167],"y":[0.147]},"t":68,"s":[-180.009]},{"i":{"x":[0.833],"y":[1.011]},"o":{"x":[0.167],"y":[0.27]},"t":69,"s":[-180.013]},{"i":{"x":[0.833],"y":[0.835]},"o":{"x":[0.167],"y":[0.146]},"t":70,"s":[-180.014]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":183,"s":[-180]},{"t":184,"s":[-180]}],"ix":10},"p":{"a":0,"k":[9.067,7.662,0],"ix":2,"l":2},"a":{"a":0,"k":[9.067,7.662,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[-0.048,0.305],[0,0],[0.302,-0.442],[0,0],[-0.403,0],[0,0],[0.048,-0.305],[0,0],[-0.302,0.441],[0,0],[0.403,0]],"o":[[-0.308,0],[0,0],[0.083,-0.528],[0,0],[-0.228,0.333],[0,0],[0.308,0],[0,0],[-0.083,0.528],[0,0],[0.228,-0.333],[0,0]],"v":[[-4.448,-8.592],[-4.943,-9.172],[-4.273,-13.418],[-5.183,-13.78],[-9.911,-6.868],[-9.497,-6.083],[-7.418,-6.083],[-6.923,-5.503],[-7.593,-1.257],[-6.683,-0.896],[-1.954,-7.807],[-2.368,-8.592]],"c":true},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"mm","mm":1,"nm":"Merge Paths 1","mn":"ADBE Vector Filter - Merge","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[15,15],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 6","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":90,"st":5,"ct":1,"bm":0},{"ddd":0,"ind":3,"ty":3,"nm":"ыArtboard Outlines","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":11,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":12,"s":[0]},{"i":{"x":[0.833],"y":[0.86]},"o":{"x":[0.167],"y":[0]},"t":13,"s":[0]},{"i":{"x":[0.833],"y":[0.346]},"o":{"x":[0.167],"y":[0.083]},"t":14,"s":[0]},{"i":{"x":[0.833],"y":[0.854]},"o":{"x":[0.167],"y":[0.096]},"t":15,"s":[1.683]},{"i":{"x":[0.833],"y":[0.717]},"o":{"x":[0.167],"y":[0.194]},"t":16,"s":[13.202]},{"i":{"x":[0.833],"y":[0.825]},"o":{"x":[0.167],"y":[0.118]},"t":17,"s":[21.881]},{"i":{"x":[0.833],"y":[0.875]},"o":{"x":[0.167],"y":[0.159]},"t":18,"s":[42.689]},{"i":{"x":[0.833],"y":[0.756]},"o":{"x":[0.167],"y":[0.249]},"t":19,"s":[65.632]},{"i":{"x":[0.833],"y":[0.841]},"o":{"x":[0.167],"y":[0.127]},"t":20,"s":[77.183]},{"i":{"x":[0.833],"y":[0.879]},"o":{"x":[0.167],"y":[0.175]},"t":21,"s":[99.424]},{"i":{"x":[0.833],"y":[0.77]},"o":{"x":[0.167],"y":[0.268]},"t":22,"s":[119.61]},{"i":{"x":[0.833],"y":[0.849]},"o":{"x":[0.167],"y":[0.131]},"t":23,"s":[128.714]},{"i":{"x":[0.833],"y":[0.882]},"o":{"x":[0.167],"y":[0.186]},"t":24,"s":[144.735]},{"i":{"x":[0.833],"y":[0.781]},"o":{"x":[0.167],"y":[0.283]},"t":25,"s":[157.784]},{"i":{"x":[0.833],"y":[0.856]},"o":{"x":[0.167],"y":[0.134]},"t":26,"s":[163.227]},{"i":{"x":[0.833],"y":[0.885]},"o":{"x":[0.167],"y":[0.198]},"t":27,"s":[172.101]},{"i":{"x":[0.833],"y":[0.796]},"o":{"x":[0.167],"y":[0.304]},"t":28,"s":[178.567]},{"i":{"x":[0.833],"y":[0.868]},"o":{"x":[0.167],"y":[0.141]},"t":29,"s":[181.011]},{"i":{"x":[0.833],"y":[0.893]},"o":{"x":[0.167],"y":[0.225]},"t":30,"s":[184.564]},{"i":{"x":[0.833],"y":[0.845]},"o":{"x":[0.167],"y":[0.372]},"t":31,"s":[186.647]},{"i":{"x":[0.833],"y":[0.943]},"o":{"x":[0.167],"y":[0.18]},"t":32,"s":[187.247]},{"i":{"x":[0.833],"y":[0.782]},"o":{"x":[0.167],"y":[-0.184]},"t":33,"s":[187.765]},{"i":{"x":[0.833],"y":[0.675]},"o":{"x":[0.167],"y":[0.135]},"t":34,"s":[187.603]},{"i":{"x":[0.833],"y":[0.813]},"o":{"x":[0.167],"y":[0.112]},"t":35,"s":[187.343]},{"i":{"x":[0.833],"y":[0.872]},"o":{"x":[0.167],"y":[0.15]},"t":36,"s":[186.587]},{"i":{"x":[0.833],"y":[0.75]},"o":{"x":[0.167],"y":[0.24]},"t":37,"s":[185.649]},{"i":{"x":[0.833],"y":[0.838]},"o":{"x":[0.167],"y":[0.125]},"t":38,"s":[185.151]},{"i":{"x":[0.833],"y":[0.878]},"o":{"x":[0.167],"y":[0.172]},"t":39,"s":[184.156]},{"i":{"x":[0.833],"y":[0.767]},"o":{"x":[0.167],"y":[0.264]},"t":40,"s":[183.216]},{"i":{"x":[0.833],"y":[0.847]},"o":{"x":[0.167],"y":[0.13]},"t":41,"s":[182.782]},{"i":{"x":[0.833],"y":[0.881]},"o":{"x":[0.167],"y":[0.183]},"t":42,"s":[182.001]},{"i":{"x":[0.833],"y":[0.778]},"o":{"x":[0.167],"y":[0.279]},"t":43,"s":[181.346]},{"i":{"x":[0.833],"y":[0.854]},"o":{"x":[0.167],"y":[0.133]},"t":44,"s":[181.067]},{"i":{"x":[0.833],"y":[0.884]},"o":{"x":[0.167],"y":[0.194]},"t":45,"s":[180.601]},{"i":{"x":[0.833],"y":[0.79]},"o":{"x":[0.167],"y":[0.296]},"t":46,"s":[180.25]},{"i":{"x":[0.833],"y":[0.863]},"o":{"x":[0.167],"y":[0.138]},"t":47,"s":[180.112]},{"i":{"x":[0.833],"y":[0.889]},"o":{"x":[0.167],"y":[0.214]},"t":48,"s":[179.904]},{"i":{"x":[0.833],"y":[0.82]},"o":{"x":[0.167],"y":[0.338]},"t":49,"s":[179.771]},{"i":{"x":[0.833],"y":[0.896]},"o":{"x":[0.167],"y":[0.155]},"t":50,"s":[179.727]},{"i":{"x":[0.833],"y":[0.944]},"o":{"x":[0.167],"y":[0.416]},"t":51,"s":[179.677]},{"i":{"x":[0.833],"y":[0.456]},"o":{"x":[0.167],"y":[-0.17]},"t":52,"s":[179.664]},{"i":{"x":[0.833],"y":[0.787]},"o":{"x":[0.167],"y":[0.098]},"t":53,"s":[179.668]},{"i":{"x":[0.833],"y":[0.869]},"o":{"x":[0.167],"y":[0.137]},"t":54,"s":[179.691]},{"i":{"x":[0.833],"y":[0.742]},"o":{"x":[0.167],"y":[0.228]},"t":55,"s":[179.727]},{"i":{"x":[0.833],"y":[0.834]},"o":{"x":[0.167],"y":[0.123]},"t":56,"s":[179.747]},{"i":{"x":[0.833],"y":[0.877]},"o":{"x":[0.167],"y":[0.167]},"t":57,"s":[179.79]},{"i":{"x":[0.833],"y":[0.763]},"o":{"x":[0.167],"y":[0.258]},"t":58,"s":[179.833]},{"i":{"x":[0.833],"y":[0.845]},"o":{"x":[0.167],"y":[0.129]},"t":59,"s":[179.853]},{"i":{"x":[0.833],"y":[0.88]},"o":{"x":[0.167],"y":[0.18]},"t":60,"s":[179.89]},{"i":{"x":[0.833],"y":[0.775]},"o":{"x":[0.167],"y":[0.275]},"t":61,"s":[179.922]},{"i":{"x":[0.833],"y":[0.852]},"o":{"x":[0.167],"y":[0.132]},"t":62,"s":[179.936]},{"i":{"x":[0.833],"y":[0.883]},"o":{"x":[0.167],"y":[0.19]},"t":63,"s":[179.96]},{"i":{"x":[0.833],"y":[0.786]},"o":{"x":[0.167],"y":[0.291]},"t":64,"s":[179.979]},{"i":{"x":[0.833],"y":[0.86]},"o":{"x":[0.167],"y":[0.137]},"t":65,"s":[179.986]},{"i":{"x":[0.833],"y":[0.887]},"o":{"x":[0.167],"y":[0.206]},"t":66,"s":[179.998]},{"i":{"x":[0.833],"y":[0.807]},"o":{"x":[0.167],"y":[0.32]},"t":67,"s":[180.006]},{"i":{"x":[0.833],"y":[0.88]},"o":{"x":[0.167],"y":[0.147]},"t":68,"s":[180.009]},{"i":{"x":[0.833],"y":[1.011]},"o":{"x":[0.167],"y":[0.27]},"t":69,"s":[180.013]},{"i":{"x":[0.833],"y":[0.835]},"o":{"x":[0.167],"y":[0.146]},"t":70,"s":[180.014]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":183,"s":[180]},{"t":184,"s":[180]}],"ix":10},"p":{"a":0,"k":[256,256,0],"ix":2,"l":2},"a":{"a":0,"k":[15,15,0],"ix":1,"l":2},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833,0.833],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":11,"s":[1706.667,1706.667,100]},{"i":{"x":[0.833,0.833,0.833],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":12,"s":[1706.667,1706.667,100]},{"i":{"x":[0.833,0.833,0.833],"y":[1.492,1.492,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":13,"s":[1706.667,1706.667,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.346,0.346,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.083,0.083,0]},"t":14,"s":[1706.667,1706.667,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.854,0.854,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.096,0.096,0]},"t":15,"s":[1700.763,1700.763,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.733,0.733,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.194,0.194,0]},"t":16,"s":[1660.353,1660.353,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.867,0.867,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.121,0.121,0]},"t":17,"s":[1629.906,1629.906,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.896,0.896,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.223,0.223,0]},"t":18,"s":[1562.81,1562.81,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.875,0.875,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.415,0.415,0]},"t":19,"s":[1522.734,1522.734,100]},{"i":{"x":[0.833,0.833,0.833],"y":[1.077,1.077,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.251,0.251,0]},"t":20,"s":[1512.658,1512.658,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.843,0.843,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.04,0.04,0]},"t":21,"s":[1507.635,1507.635,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.705,0.705,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.177,0.177,0]},"t":22,"s":[1517.304,1517.304,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.821,0.821,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.116,0.116,0]},"t":23,"s":[1525.889,1525.889,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.874,0.874,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.156,0.156,0]},"t":24,"s":[1547.707,1547.707,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.754,0.754,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.246,0.246,0]},"t":25,"s":[1572.744,1572.744,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.84,0.84,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.126,0.126,0]},"t":26,"s":[1585.59,1585.59,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.879,0.879,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.174,0.174,0]},"t":27,"s":[1610.664,1610.664,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.769,0.769,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.266,0.266,0]},"t":28,"s":[1633.76,1633.76,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.848,0.848,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.13,0.13,0]},"t":29,"s":[1644.277,1644.277,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.882,0.882,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.184,0.184,0]},"t":30,"s":[1662.944,1662.944,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.78,0.78,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.281,0.281,0]},"t":31,"s":[1678.321,1678.321,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.855,0.855,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.134,0.134,0]},"t":32,"s":[1684.791,1684.791,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.885,0.885,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.196,0.196,0]},"t":33,"s":[1695.439,1695.439,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.793,0.793,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.301,0.301,0]},"t":34,"s":[1703.311,1703.311,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.866,0.866,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.14,0.14,0]},"t":35,"s":[1706.33,1706.33,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.891,0.891,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.22,0.22,0]},"t":36,"s":[1710.798,1710.798,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.356,0.356,0]},"t":37,"s":[1713.523,1713.523,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.916,0.916,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.166,0.166,0]},"t":38,"s":[1714.356,1714.356,100]},{"i":{"x":[0.833,0.833,0.833],"y":[5.591,5.591,1]},"o":{"x":[0.167,0.167,0.167],"y":[17.664,17.664,0]},"t":39,"s":[1715.196,1715.196,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.637,0.637,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.082,0.082,0]},"t":40,"s":[1715.2,1715.2,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.806,0.806,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.108,0.108,0]},"t":41,"s":[1714.977,1714.977,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.871,0.871,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.146,0.146,0]},"t":42,"s":[1714.226,1714.226,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.747,0.747,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.236,0.236,0]},"t":43,"s":[1713.228,1713.228,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.837,0.837,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.124,0.124,0]},"t":44,"s":[1712.684,1712.684,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.878,0.878,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.17,0.17,0]},"t":45,"s":[1711.577,1711.577,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.765,0.765,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.262,0.262,0]},"t":46,"s":[1710.513,1710.513,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.846,0.846,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.129,0.129,0]},"t":47,"s":[1710.016,1710.016,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.881,0.881,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.182,0.182,0]},"t":48,"s":[1709.113,1709.113,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.776,0.776,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.277,0.277,0]},"t":49,"s":[1708.347,1708.347,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.853,0.853,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.133,0.133,0]},"t":50,"s":[1708.018,1708.018,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.884,0.884,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.192,0.192,0]},"t":51,"s":[1707.464,1707.464,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.789,0.789,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.294,0.294,0]},"t":52,"s":[1707.041,1707.041,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.862,0.862,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.138,0.138,0]},"t":53,"s":[1706.873,1706.873,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.888,0.888,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.21,0.21,0]},"t":54,"s":[1706.616,1706.616,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.814,0.814,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.33,0.33,0]},"t":55,"s":[1706.447,1706.447,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.888,0.888,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.151,0.151,0]},"t":56,"s":[1706.39,1706.39,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.919,0.919,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.324,0.324,0]},"t":57,"s":[1706.319,1706.319,100]},{"i":{"x":[0.833,0.833,0.833],"y":[-1.387,-1.387,1]},"o":{"x":[0.167,0.167,0.167],"y":[-2.742,-2.742,0]},"t":58,"s":[1706.295,1706.295,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.764,0.764,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.086,0.086,0]},"t":59,"s":[1706.296,1706.296,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.867,0.867,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.129,0.129,0]},"t":60,"s":[1706.316,1706.316,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.737,0.737,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.222,0.222,0]},"t":61,"s":[1706.352,1706.352,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.832,0.832,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.122,0.122,0]},"t":62,"s":[1706.374,1706.374,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.877,0.877,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.166,0.166,0]},"t":63,"s":[1706.421,1706.421,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.762,0.762,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.256,0.256,0]},"t":64,"s":[1706.468,1706.468,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.844,0.844,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.128,0.128,0]},"t":65,"s":[1706.491,1706.491,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.88,0.88,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.179,0.179,0]},"t":66,"s":[1706.534,1706.534,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.773,0.773,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.273,0.273,0]},"t":67,"s":[1706.571,1706.571,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.851,0.851,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.132,0.132,0]},"t":68,"s":[1706.588,1706.588,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.997,0.997,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.189,0.189,0]},"t":69,"s":[1706.616,1706.616,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.835,0.835,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.294,0.294,0]},"t":70,"s":[1706.638,1706.638,100]},{"i":{"x":[0.833,0.833,0.833],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":183,"s":[1706.667,1706.667,100]},{"t":184,"s":[1706.667,1706.667,100]}],"ix":6,"l":2}},"ao":0,"ip":0,"op":90,"st":5,"ct":1,"bm":0},{"ddd":0,"ind":4,"ty":4,"nm":"ыArtboard Outlines 4","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[256,256,0],"ix":2,"l":2},"a":{"a":0,"k":[15,15,0],"ix":1,"l":2},"s":{"a":0,"k":[1706.667,1706.667,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0.535,0.272],[0.24,0.47],[0,1.4],[0,0]],"o":[[0,0],[-1.4,0],[-0.47,-0.24],[-0.272,-0.535],[0,0],[0,0]],"v":[[3.5,3.5],[0.5,3.5],[-2.135,3.228],[-3.228,2.135],[-3.5,-0.5],[-3.5,-3.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":22,"s":[0]},{"t":30,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1.66,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[10.5,21.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0,"y":1},"o":{"x":0.167,"y":0.167},"t":25,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-3,6.5],[0,3.5],[3,6.5]],"c":false}]},{"t":35,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-3,1.5],[0,-1.5],[3,1.5]],"c":false}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":25,"s":[50]},{"t":35,"s":[0]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":25,"s":[50]},{"t":35,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1.66,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[7,18.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 2","np":3,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[-0.535,-0.272],[-0.24,-0.47],[0,-1.4],[0,0]],"o":[[0,0],[1.4,0],[0.47,0.24],[0.272,0.535],[0,0],[0,0]],"v":[[-3.5,-4],[-0.5,-4],[2.135,-3.728],[3.228,-2.635],[3.5,0],[3.5,4]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1.66,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":22,"s":[0]},{"t":30,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":3,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"tr","p":{"a":0,"k":[19.5,9],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 3","np":3,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0,"y":1},"o":{"x":0.167,"y":0.167},"t":25,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[3,-4.5],[0,-1.5],[-3,-4.5]],"c":false}]},{"t":32,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[3,-1.5],[0,1.5],[-3,-1.5]],"c":false}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1.66,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":25,"s":[50]},{"t":35,"s":[0]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":25,"s":[50]},{"t":35,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":3,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"tr","p":{"a":0,"k":[23,11.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 4","np":3,"cix":2,"bm":0,"ix":4,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":90,"st":5,"ct":1,"bm":0},{"ddd":0,"ind":5,"ty":4,"nm":"ыArtboard Outlines","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[256,256,0],"ix":2,"l":2},"a":{"a":0,"k":[15,15,0],"ix":1,"l":2},"s":{"a":0,"k":[1706.667,1706.667,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0.535,0.272],[0.24,0.47],[0,1.4],[0,0]],"o":[[0,0],[-1.4,0],[-0.47,-0.24],[-0.272,-0.535],[0,0],[0,0]],"v":[[3.5,3.5],[0.5,3.5],[-2.135,3.228],[-3.228,2.135],[-3.5,-0.5],[-3.5,-3.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1.66,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":15,"s":[0]},{"t":20,"s":[100]}],"ix":1},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":3,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"tr","p":{"a":0,"k":[10.5,21.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-3,1.5],[0,-1.5],[3,1.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":15,"s":[100]},{"t":22,"s":[0]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1.66,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[7,18.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 2","np":3,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[-0.535,-0.272],[-0.24,-0.47],[0,-1.4],[0,0]],"o":[[0,0],[1.4,0],[0.47,0.24],[0.272,0.535],[0,0],[0,0]],"v":[[-3.5,-4],[-0.5,-4],[2.135,-3.728],[3.228,-2.635],[3.5,0],[3.5,4]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":15,"s":[0]},{"t":20,"s":[100]}],"ix":1},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1.66,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[19.5,9],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 3","np":3,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[3,-1.5],[0,1.5],[-3,-1.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":1.66,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":15,"s":[100]},{"t":22,"s":[0]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":3,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"tr","p":{"a":0,"k":[23,11.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 4","np":3,"cix":2,"bm":0,"ix":4,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":90,"st":5,"ct":1,"bm":0}],"markers":[],"props":{}} \ No newline at end of file diff --git a/TMessagesProj/src/main/res/raw/metadatascript.js b/TMessagesProj/src/main/res/raw/metadatascript.js new file mode 100644 index 00000000000..43ec9207d41 --- /dev/null +++ b/TMessagesProj/src/main/res/raw/metadatascript.js @@ -0,0 +1,11 @@ +Object.fromEntries( + [...document.querySelectorAll('meta')] + .map(a => [ + a.getAttribute('property') || a.getAttribute('name') || a.getAttribute('http-equiv'), + a.content + ]) + .concat([ + ['image', (document.querySelector('article[data-testid=tweet]:first-child div[data-testid=tweetPhoto] img') || {src:null}).src] + ]) + .filter(([k, v]) => k && v) +) \ No newline at end of file diff --git a/TMessagesProj/src/main/res/raw/rate.json b/TMessagesProj/src/main/res/raw/rate.json new file mode 100644 index 00000000000..eee84206d64 --- /dev/null +++ b/TMessagesProj/src/main/res/raw/rate.json @@ -0,0 +1 @@ +{"v":"5.10.1","fr":60,"ip":0,"op":90,"w":512,"h":512,"nm":"EFFECT_1","ddd":0,"assets":[{"id":"comp_0","nm":"placeholder_","fr":60,"layers":[{"ddd":0,"ind":1,"ty":4,"nm":"Star Copy 6","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[256.002,255.999,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[370,370,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0.85,-0.59],[0,0],[1.56,2.27],[-0.43,1.45],[0,0],[0.83,0.63],[0,0],[-1.68,2.19],[-1.52,0.04],[0,0],[-0.34,0.98],[0,0],[-2.6,-0.92],[-0.5,-1.43],[0,0],[-1.04,-0.02],[0,0],[0.07,-2.76],[1.21,-0.92],[0,0],[-0.3,-0.99],[0,0],[2.65,-0.79],[1.24,0.86],[0,0]],"o":[[0,0],[-2.27,1.57],[-0.86,-1.25],[0,0],[0.3,-0.99],[0,0],[-2.19,-1.68],[0.92,-1.2],[0,0],[1.03,-0.02],[0,0],[0.92,-2.6],[1.43,0.5],[0,0],[0.35,0.98],[0,0],[2.76,0.07],[-0.03,1.52],[0,0],[-0.82,0.63],[0,0],[0.78,2.65],[-1.45,0.43],[0,0],[-0.86,-0.59]],"v":[[-1.418,22.006],[-14.768,31.206],[-21.718,29.936],[-22.398,25.666],[-17.768,10.126],[-18.648,7.426],[-31.528,-2.424],[-32.458,-9.434],[-28.608,-11.394],[-12.398,-11.794],[-10.108,-13.464],[-4.718,-28.754],[1.662,-31.804],[4.712,-28.754],[10.102,-13.464],[12.402,-11.794],[28.612,-11.394],[33.482,-6.274],[31.522,-2.424],[18.642,7.426],[17.772,10.126],[22.402,25.666],[19.032,31.886],[14.772,31.206],[1.422,22.006]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Star Copy 6","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":1440,"st":0,"ct":1,"bm":0}]},{"id":"comp_1","nm":"star_4","fr":60,"layers":[{"ddd":0,"ind":1,"ty":4,"nm":"Shape Layer 12","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":270,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":6,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":22,"s":[79.087,96.5,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13.273,"s":[0]},{"t":22,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":6,"s":[0]},{"t":14.7265625,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":6,"op":24,"st":6,"ct":1,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"Shape Layer 11","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":90,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":6,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":22,"s":[116.913,96.5,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13.273,"s":[0]},{"t":22,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":6,"s":[0]},{"t":14.7265625,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":6,"op":24,"st":6,"ct":1,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"Shape Layer 10","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":180,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":6,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":22,"s":[98,115.413,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13.273,"s":[0]},{"t":22,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":6,"s":[0]},{"t":14.7265625,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":6,"op":24,"st":6,"ct":1,"bm":0},{"ddd":0,"ind":4,"ty":4,"nm":"Shape Layer 9","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":6,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":22,"s":[98,77.587,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13.273,"s":[0]},{"t":20,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":6,"s":[0]},{"t":17,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":6,"op":24,"st":6,"ct":1,"bm":0},{"ddd":0,"ind":5,"ty":4,"nm":"Shape Layer 8","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":270,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":3,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":19,"s":[79.087,96.5,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":10.273,"s":[0]},{"t":19,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":3,"s":[0]},{"t":11.7265625,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":3,"op":21,"st":3,"ct":1,"bm":0},{"ddd":0,"ind":6,"ty":4,"nm":"Shape Layer 7","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":90,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":3,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":19,"s":[116.913,96.5,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":10.273,"s":[0]},{"t":19,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":3,"s":[0]},{"t":11.7265625,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":3,"op":21,"st":3,"ct":1,"bm":0},{"ddd":0,"ind":7,"ty":4,"nm":"Shape Layer 6","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":180,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":3,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":19,"s":[98,115.413,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":10.273,"s":[0]},{"t":19,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":3,"s":[0]},{"t":11.7265625,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":3,"op":21,"st":3,"ct":1,"bm":0},{"ddd":0,"ind":8,"ty":4,"nm":"Shape Layer 5","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":3,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":19,"s":[98,77.587,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":10.273,"s":[0]},{"t":17,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":3,"s":[0]},{"t":14,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":3,"op":21,"st":3,"ct":1,"bm":0},{"ddd":0,"ind":9,"ty":4,"nm":"Shape Layer 4","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":270,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":0,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":16,"s":[79.087,96.5,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":7.273,"s":[0]},{"t":16,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"t":8.7265625,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":0,"op":18,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":10,"ty":4,"nm":"Shape Layer 2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":90,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":0,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":16,"s":[116.913,96.5,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":7.273,"s":[0]},{"t":16,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"t":8.7265625,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":0,"op":18,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":11,"ty":4,"nm":"Shape Layer 3","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":180,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":0,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":16,"s":[98,115.413,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":7.273,"s":[0]},{"t":16,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"t":8.7265625,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":0,"op":18,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":12,"ty":4,"nm":"Shape Layer 1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":0,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":16,"s":[98,77.587,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":7.273,"s":[0]},{"t":14,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"t":11,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":0,"op":18,"st":0,"ct":1,"bm":0}]},{"id":"comp_2","nm":"star_1","fr":60,"layers":[{"ddd":0,"ind":1,"ty":4,"nm":"Shape Layer 12","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":270,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":6,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":22,"s":[79.087,96.5,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13.273,"s":[0]},{"t":22,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":6,"s":[0]},{"t":14.7265625,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":6,"op":24,"st":6,"ct":1,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"Shape Layer 11","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":90,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":6,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":22,"s":[116.913,96.5,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13.273,"s":[0]},{"t":22,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":6,"s":[0]},{"t":14.7265625,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":6,"op":24,"st":6,"ct":1,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"Shape Layer 10","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":180,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":6,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":22,"s":[98,115.413,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13.273,"s":[0]},{"t":22,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":6,"s":[0]},{"t":14.7265625,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":6,"op":24,"st":6,"ct":1,"bm":0},{"ddd":0,"ind":4,"ty":4,"nm":"Shape Layer 9","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":6,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":22,"s":[98,77.587,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13.273,"s":[0]},{"t":20,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":6,"s":[0]},{"t":17,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":6,"op":24,"st":6,"ct":1,"bm":0},{"ddd":0,"ind":5,"ty":4,"nm":"Shape Layer 8","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":270,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":3,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":19,"s":[79.087,96.5,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":10.273,"s":[0]},{"t":19,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":3,"s":[0]},{"t":11.7265625,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":3,"op":21,"st":3,"ct":1,"bm":0},{"ddd":0,"ind":6,"ty":4,"nm":"Shape Layer 7","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":90,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":3,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":19,"s":[116.913,96.5,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":10.273,"s":[0]},{"t":19,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":3,"s":[0]},{"t":11.7265625,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":3,"op":21,"st":3,"ct":1,"bm":0},{"ddd":0,"ind":7,"ty":4,"nm":"Shape Layer 6","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":180,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":3,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":19,"s":[98,115.413,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":10.273,"s":[0]},{"t":19,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":3,"s":[0]},{"t":11.7265625,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":3,"op":21,"st":3,"ct":1,"bm":0},{"ddd":0,"ind":8,"ty":4,"nm":"Shape Layer 5","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":3,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":19,"s":[98,77.587,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":10.273,"s":[0]},{"t":17,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":3,"s":[0]},{"t":14,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":3,"op":21,"st":3,"ct":1,"bm":0},{"ddd":0,"ind":9,"ty":4,"nm":"Shape Layer 4","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":270,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":0,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":16,"s":[79.087,96.5,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":7.273,"s":[0]},{"t":16,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"t":8.7265625,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":0,"op":18,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":10,"ty":4,"nm":"Shape Layer 2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":90,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":0,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":16,"s":[116.913,96.5,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":7.273,"s":[0]},{"t":16,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"t":8.7265625,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":0,"op":18,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":11,"ty":4,"nm":"Shape Layer 3","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":180,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":0,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":16,"s":[98,115.413,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":7.273,"s":[0]},{"t":16,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"t":8.7265625,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":0,"op":18,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":12,"ty":4,"nm":"Shape Layer 1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":0,"s":[98,96.5,0],"to":[0,0,0],"ti":[0,0,0]},{"t":16,"s":[98,77.587,0]}],"ix":2,"l":2},"a":{"a":0,"k":[209,-183.5,0],"ix":1,"l":2},"s":{"a":0,"k":[75.652,75.652,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[209,-183.5],[209,-219.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":7.273,"s":[0]},{"t":14,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"t":11,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":0,"op":18,"st":0,"ct":1,"bm":0}]}],"layers":[{"ddd":0,"ind":1,"ty":0,"nm":"placeholder_1","refId":"comp_0","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.3],"y":[0]},"t":39,"s":[100]},{"t":46.884765625,"s":[0]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.504,"y":0.922},"o":{"x":0.195,"y":0.205},"t":1,"s":[252.091,237.615,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.46,"y":1},"o":{"x":0.279,"y":0.04},"t":11,"s":[233.908,167.9,0],"to":[0,0,0],"ti":[0,0,0]},{"t":57,"s":[224.408,531.598,0]}],"ix":2,"l":2},"a":{"a":0,"k":[256,256,0],"ix":1,"l":2},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":2,"s":[12,12,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":11,"s":[20,20,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":34,"s":[28,28,100]},{"t":45,"s":[4,4,100]}],"ix":6,"l":2}},"ao":0,"w":512,"h":512,"ip":1,"op":44,"st":-35,"ct":1,"bm":0},{"ddd":0,"ind":2,"ty":0,"nm":"placeholder_2","refId":"comp_0","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.3],"y":[0]},"t":41,"s":[100]},{"t":51,"s":[0]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.109,"y":0.548},"o":{"x":0.02,"y":0},"t":1,"s":[232.612,242.312,0],"to":[10.707,-41.715,0],"ti":[-38.374,0.791,0]},{"i":{"x":0.708,"y":1},"o":{"x":0.474,"y":0.378},"t":11,"s":[301.744,107.351,0],"to":[43.367,-0.894,0],"ti":[-12.625,-246.426,0]},{"t":59,"s":[428.813,563.068,0]}],"ix":2,"l":2},"a":{"a":0,"k":[256,256,0],"ix":1,"l":2},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":1,"s":[-8,8,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":11,"s":[-14,14,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":41,"s":[-21,21,100]},{"t":50,"s":[-4,4,100]}],"ix":6,"l":2}},"ao":0,"w":512,"h":512,"ip":1,"op":47,"st":-119,"ct":1,"bm":0},{"ddd":0,"ind":3,"ty":0,"nm":"placeholder_3","refId":"comp_0","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.3],"y":[0]},"t":45,"s":[100]},{"t":53.8671875,"s":[0]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.417,"y":0.808},"o":{"x":0.137,"y":0.309},"t":2,"s":[230.521,219.86,0],"to":[-14.958,-20.45,0],"ti":[37.448,-4.406,0]},{"i":{"x":0.434,"y":1},"o":{"x":0.385,"y":0.212},"t":15,"s":[111.774,115.421,0],"to":[-42.843,5.041,0],"ti":[0,0,0]},{"t":71.47265625,"s":[33.914,525.091,0]}],"ix":2,"l":2},"a":{"a":0,"k":[256,256,0],"ix":1,"l":2},"s":{"a":1,"k":[{"i":{"x":[0.8,0.8,0.8],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,-5]},"t":2,"s":[15,15,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,6.583]},"o":{"x":[0.167,0.167,0.167],"y":[0.175,0.175,-3.333]},"t":40,"s":[24,24,100]},{"t":53,"s":[5,5,100]}],"ix":6,"l":2}},"ao":0,"w":512,"h":512,"ip":2,"op":51,"st":2,"ct":1,"bm":0},{"ddd":0,"ind":4,"ty":0,"nm":"placeholder_4","refId":"comp_0","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.3],"y":[0]},"t":53,"s":[100]},{"t":63,"s":[0]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.266,"y":0.874},"o":{"x":0.02,"y":0},"t":5,"s":[256.959,251.56,0],"to":[-18.201,-59.818,0],"ti":[35.75,1.133,0]},{"i":{"x":0.708,"y":1},"o":{"x":0.556,"y":0.158},"t":17,"s":[193.976,78.891,0],"to":[-53.216,3.549,0],"ti":[0,0,0]},{"t":72,"s":[143.866,595.231,0]}],"ix":2,"l":2},"a":{"a":0,"k":[256,256,0],"ix":1,"l":2},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":6,"s":[-12,12,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":51,"s":[-24,24,100]},{"t":61,"s":[-4,4,100]}],"ix":6,"l":2}},"ao":0,"w":512,"h":512,"ip":6,"op":59,"st":3,"ct":1,"bm":0},{"ddd":0,"ind":5,"ty":0,"nm":"placeholder_5","refId":"comp_0","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.3],"y":[0]},"t":58,"s":[100]},{"t":68.109375,"s":[0]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.069,"y":0.731},"o":{"x":0.02,"y":0},"t":3,"s":[281.869,229.212,0],"to":[14.597,-73.616,0],"ti":[-41.626,1.096,0]},{"i":{"x":0.705,"y":1},"o":{"x":0.466,"y":0.207},"t":18,"s":[354.524,42.391,0],"to":[50.945,-1.349,0],"ti":[-11.393,-177.824,0]},{"t":74,"s":[450.127,531.605,0]}],"ix":2,"l":2},"a":{"a":0,"k":[256,256,0],"ix":1,"l":2},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":3,"s":[8,8,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":11,"s":[15,15,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":55.842,"s":[20,20,100]},{"t":67,"s":[7,7,100]}],"ix":6,"l":2}},"ao":0,"w":512,"h":512,"ip":3,"op":65,"st":-1,"ct":1,"bm":0},{"ddd":0,"ind":6,"ty":0,"nm":"placeholder_6","refId":"comp_0","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.3],"y":[0]},"t":64,"s":[100]},{"t":75,"s":[0]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.1,"y":1},"o":{"x":0.02,"y":0},"t":3,"s":[242.551,249.491,0],"to":[0,0,0],"ti":[-14.652,1.25,0]},{"i":{"x":0.767,"y":0.698},"o":{"x":0.43,"y":0},"t":19,"s":[252.146,35.091,0],"to":[15.073,-1.286,0],"ti":[0.606,-106.051,0]},{"t":77,"s":[280.793,601.5,0]}],"ix":2,"l":2},"a":{"a":0,"k":[256,256,0],"ix":1,"l":2},"s":{"a":1,"k":[{"i":{"x":[0.7,0.7,0.7],"y":[1,1,1]},"o":{"x":[0.2,0.2,0.2],"y":[0,0,0]},"t":5,"s":[-10,10,100]},{"i":{"x":[0.45,0.45,0.45],"y":[1,1,1]},"o":{"x":[0.27,0.27,0.27],"y":[0,0,0]},"t":60,"s":[-24,24,100]},{"t":73,"s":[-8,5,100]}],"ix":6,"l":2}},"ao":0,"w":512,"h":512,"ip":3,"op":70,"st":-88,"ct":1,"bm":0},{"ddd":0,"ind":7,"ty":0,"nm":"placeholder_7","refId":"comp_0","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.3],"y":[0]},"t":48,"s":[100]},{"t":58,"s":[0]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.109,"y":0.78},"o":{"x":0.02,"y":0},"t":6,"s":[251.612,222.312,0],"to":[5.707,-35.715,0],"ti":[-29.498,-0.693,0]},{"i":{"x":0.708,"y":1},"o":{"x":0.474,"y":0.154},"t":17,"s":[287.744,95.351,0],"to":[38.182,0.896,0],"ti":[-12.625,-187.426,0]},{"t":65,"s":[354.813,543.068,0]}],"ix":2,"l":2},"a":{"a":0,"k":[256,256,0],"ix":1,"l":2},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":6,"s":[8,8,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":17,"s":[14,14,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":48,"s":[17,17,100]},{"t":58,"s":[5,5,100]}],"ix":6,"l":2}},"ao":0,"w":512,"h":512,"ip":6,"op":54,"st":-42,"ct":1,"bm":0},{"ddd":0,"ind":8,"ty":4,"nm":"Points","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":23,"s":[100]},{"t":35,"s":[0]}],"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[-5]},{"t":35,"s":[0]}],"ix":10},"p":{"a":0,"k":[256.959,254.547,0],"ix":2,"l":2},"a":{"a":0,"k":[0.5,3.5,0],"ix":1,"l":2},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,1]},"o":{"x":[0.107,0.107,0.347],"y":[0,0,0]},"t":-2,"s":[9.8,9.8,100]},{"i":{"x":[0.708,0.708,0.479],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0]},"t":9,"s":[25.2,25.2,100]},{"t":35,"s":[29.96,29.96,100]}],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[7.226,-9.469],[9.469,7.226],[-7.226,9.469],[-9.469,-7.226]],"o":[[-7.226,9.469],[-9.469,-7.226],[7.226,-9.469],[9.469,7.226]],"v":[[385.072,18.248],[354.843,22.31],[350.781,-7.919],[381.011,-11.981]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[367.391,4.348],"ix":2},"a":{"a":0,"k":[367.391,4.348],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.317,0.317],"y":[0,0]},"t":-2,"s":[150,150]},{"i":{"x":[0.496,0.496],"y":[1,1]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":9,"s":[100,100]},{"t":35,"s":[40,40]}],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[11.805,-1.586],[1.586,11.805],[-11.805,1.586],[-1.586,-11.805]],"o":[[-11.805,1.586],[-1.586,-11.805],[11.805,-1.586],[1.586,11.805]],"v":[[261.231,286.312],[236.984,267.809],[255.487,243.561],[279.735,262.064]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[256.522,264.13],"ix":2},"a":{"a":0,"k":[256.522,264.13],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.317,0.317],"y":[0,0]},"t":-2,"s":[150,150]},{"i":{"x":[0.496,0.496],"y":[1,1]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":9,"s":[100,100]},{"t":35,"s":[40,40]}],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 2","np":2,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[9.469,7.226],[-7.226,9.469],[-9.469,-7.226],[7.226,-9.469]],"o":[[-9.469,-7.226],[7.226,-9.469],[9.469,7.226],[-7.226,9.469]],"v":[[-14.927,388.164],[-18.989,357.934],[11.24,353.873],[15.302,384.102]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[-4.348,370.652],"ix":2},"a":{"a":0,"k":[-4.348,370.652],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.2,0.2],"y":[0,0]},"t":-2,"s":[150,150]},{"i":{"x":[0.3,0.3],"y":[1,1]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":9,"s":[100,100]},{"t":35,"s":[40,40]}],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 3","np":2,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[1.586,11.805],[-11.805,1.586],[-1.586,-11.805],[11.805,-1.586]],"o":[[-1.586,-11.805],[11.805,-1.586],[1.586,11.805],[-11.805,1.586]],"v":[[-282.221,264.91],[-263.717,240.663],[-239.47,259.166],[-257.973,283.413]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[-263.043,260.87],"ix":2},"a":{"a":0,"k":[-263.043,260.87],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.2,0.2],"y":[0,0]},"t":-2,"s":[150,150]},{"i":{"x":[0.3,0.3],"y":[1,1]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":9,"s":[100,100]},{"t":35,"s":[40,40]}],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 4","np":2,"cix":2,"bm":0,"ix":4,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-7.226,9.469],[-9.469,-7.226],[7.226,-9.469],[9.469,7.226]],"o":[[7.226,-9.469],[9.469,7.226],[-7.226,9.469],[-9.469,-7.226]],"v":[[-384.072,-11.248],[-353.843,-15.31],[-349.781,14.919],[-380.011,18.981]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[-369.565,0],"ix":2},"a":{"a":0,"k":[-369.565,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.2,0.2],"y":[0,0]},"t":-2,"s":[150,150]},{"i":{"x":[0.3,0.3],"y":[1,1]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":9,"s":[100,100]},{"t":35,"s":[40,40]}],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 5","np":2,"cix":2,"bm":0,"ix":5,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-11.805,1.586],[-1.586,-11.805],[11.805,-1.586],[1.586,11.805]],"o":[[11.805,-1.586],[1.586,11.805],[-11.805,1.586],[-1.586,-11.805]],"v":[[-260.819,-278.542],[-236.571,-260.039],[-255.075,-235.791],[-279.322,-254.295]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[-259.783,-258.696],"ix":2},"a":{"a":0,"k":[-259.783,-258.696],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.2,0.2],"y":[0,0]},"t":-2,"s":[150,150]},{"i":{"x":[0.3,0.3],"y":[1,1]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":9,"s":[100,100]},{"t":35,"s":[40,40]}],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 6","np":2,"cix":2,"bm":0,"ix":6,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-9.469,-7.226],[7.226,-9.469],[9.469,7.226],[-7.226,9.469]],"o":[[9.469,7.226],[-7.226,9.469],[-9.469,-7.226],[7.226,-9.469]],"v":[[15.927,-381.164],[19.989,-350.934],[-10.24,-346.873],[-14.302,-377.102]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,-365.217],"ix":2},"a":{"a":0,"k":[0,-365.217],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.2,0.2],"y":[0,0]},"t":-2,"s":[150,150]},{"i":{"x":[0.3,0.3],"y":[1,1]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":9,"s":[100,100]},{"t":35,"s":[40,40]}],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 7","np":2,"cix":2,"bm":0,"ix":7,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-1.586,-11.805],[11.805,-1.586],[1.586,11.805],[-11.805,1.586]],"o":[[1.586,11.805],[-11.805,1.586],[-1.586,-11.805],[11.805,-1.586]],"v":[[283.221,-257.91],[264.717,-233.663],[240.47,-252.166],[258.973,-276.413]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[257.609,-254.348],"ix":2},"a":{"a":0,"k":[257.609,-254.348],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.2,0.2],"y":[0,0]},"t":-2,"s":[150,150]},{"i":{"x":[0.3,0.3],"y":[1,1]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":9,"s":[100,100]},{"t":35,"s":[40,40]}],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 8","np":2,"cix":2,"bm":0,"ix":8,"mn":"ADBE Vector Group","hd":false}],"ip":5,"op":34,"st":-2,"ct":1,"bm":0},{"ddd":0,"ind":9,"ty":4,"nm":"Circle 3","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":5,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":23,"s":[100]},{"t":35,"s":[0]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[257,254.5,0],"ix":2,"l":2},"a":{"a":0,"k":[-2.049,6.248,0],"ix":1,"l":2},"s":{"a":0,"k":[28,28,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0,"y":1},"o":{"x":0.05,"y":0},"t":0,"s":[{"i":[[0,-17.536],[17.536,0],[0,17.536],[-17.536,0]],"o":[[0,17.536],[-17.536,0],[0,-17.536],[17.536,0]],"v":[[29.703,6.248],[-2.049,38],[-33.801,6.248],[-2.049,-25.504]],"c":true}]},{"t":35,"s":[{"i":[[0,-196.476],[196.476,0],[0,196.476],[-196.476,0]],"o":[[0,196.476],[-196.476,0],[0,-196.476],[196.476,0]],"v":[[353.703,6.248],[-2.049,362],[-357.801,6.248],[-2.049,-349.504]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.05],"y":[0]},"t":0,"s":[120]},{"i":{"x":[0.5],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":11,"s":[18]},{"t":35,"s":[6]}],"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":1,"op":34,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":10,"ty":0,"nm":"star_4","refId":"comp_1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[380,318,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[34,34,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":72,"op":89,"st":72,"bm":0},{"ddd":0,"ind":11,"ty":0,"nm":"star_1","refId":"comp_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[196,436,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[47,47,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":69,"op":87,"st":69,"bm":0},{"ddd":0,"ind":12,"ty":0,"nm":"star_1","refId":"comp_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[65,445,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[43,43,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":64,"op":82,"st":64,"bm":0},{"ddd":0,"ind":13,"ty":0,"nm":"star_4","refId":"comp_1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[310,133,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[23,23,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":61,"op":79,"st":61,"bm":0},{"ddd":0,"ind":14,"ty":0,"nm":"star_4","refId":"comp_1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[327,77,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[42,42,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":63,"op":81,"st":63,"bm":0},{"ddd":0,"ind":15,"ty":0,"nm":"star_1","refId":"comp_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[435,215,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[23,23,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":59,"op":77,"st":59,"bm":0},{"ddd":0,"ind":16,"ty":0,"nm":"star_1","refId":"comp_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[436,56,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[32,32,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":55,"op":73,"st":55,"bm":0},{"ddd":0,"ind":17,"ty":0,"nm":"star_1","refId":"comp_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[236,196,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[23,23,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":47,"op":65,"st":47,"bm":0},{"ddd":0,"ind":18,"ty":0,"nm":"star_1","refId":"comp_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[148,229,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[47,47,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":52,"op":70,"st":52,"bm":0},{"ddd":0,"ind":19,"ty":0,"nm":"star_4","refId":"comp_1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[63,368,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[29,29,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":50,"op":68,"st":50,"bm":0},{"ddd":0,"ind":20,"ty":0,"nm":"star_4","refId":"comp_1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[315,429,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[29,29,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":48,"op":66,"st":48,"bm":0},{"ddd":0,"ind":21,"ty":0,"nm":"star_4","refId":"comp_1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[116,53,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[29,29,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":46,"op":64,"st":46,"bm":0},{"ddd":0,"ind":22,"ty":0,"nm":"star_4","refId":"comp_1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[446,392,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[34,34,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":42,"op":60,"st":42,"bm":0},{"ddd":0,"ind":23,"ty":0,"nm":"star_1","refId":"comp_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[456,368,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[47,47,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":39,"op":57,"st":39,"bm":0},{"ddd":0,"ind":24,"ty":0,"nm":"star_1","refId":"comp_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[190,299,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[47,47,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":34,"op":52,"st":34,"bm":0},{"ddd":0,"ind":25,"ty":0,"nm":"star_4","refId":"comp_1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[433,48,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[34,34,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":31,"op":49,"st":31,"bm":0},{"ddd":0,"ind":26,"ty":0,"nm":"star_4","refId":"comp_1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[333,452,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[42,42,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":33,"op":51,"st":33,"bm":0},{"ddd":0,"ind":27,"ty":0,"nm":"star_1","refId":"comp_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[71,146,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[47,47,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":29,"op":47,"st":29,"bm":0},{"ddd":0,"ind":28,"ty":0,"nm":"star_1","refId":"comp_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[102,64,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[33,33,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":26,"op":44,"st":26,"bm":0},{"ddd":0,"ind":29,"ty":0,"nm":"star_4","refId":"comp_1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[123,360,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[30,30,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":15,"op":33,"st":15,"bm":0},{"ddd":0,"ind":30,"ty":0,"nm":"star_4","refId":"comp_1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[231,153,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[30,30,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":18,"op":36,"st":18,"bm":0},{"ddd":0,"ind":31,"ty":0,"nm":"star_4","refId":"comp_1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[385,254,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[48,48,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":13,"op":31,"st":13,"bm":0},{"ddd":0,"ind":32,"ty":0,"nm":"star_1","refId":"comp_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[426,439,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[43,43,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":10,"op":28,"st":10,"bm":0},{"ddd":0,"ind":33,"ty":0,"nm":"star_1","refId":"comp_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[456,131,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[33,33,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":15,"op":33,"st":15,"bm":0},{"ddd":0,"ind":34,"ty":0,"nm":"star_1","refId":"comp_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[91,322,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[33,33,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":8,"op":26,"st":8,"bm":0},{"ddd":0,"ind":35,"ty":0,"nm":"star_1","refId":"comp_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[372,115,0],"ix":2,"l":2},"a":{"a":0,"k":[100,100,0],"ix":1,"l":2},"s":{"a":0,"k":[42,42,100],"ix":6,"l":2}},"ao":0,"w":200,"h":200,"ip":0,"op":18,"st":0,"bm":0},{"ddd":0,"ind":36,"ty":3,"nm":"NULL CONTROL 42","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":32,"ix":10},"p":{"a":0,"k":[403.775,95.984,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[79.632,79.632,100],"ix":6,"l":2}},"ao":0,"ip":36,"op":53,"st":-33,"bm":0},{"ddd":0,"ind":37,"ty":4,"nm":"Shape Layer 145","parent":36,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-98,"ix":10},"p":{"a":0,"k":[141.175,96.966,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[94.767,94.767,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":42,"s":[0,0]},{"t":52,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":47,"s":[8]},{"t":52,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":42,"op":53,"st":25,"ct":1,"bm":0},{"ddd":0,"ind":38,"ty":4,"nm":"Shape Layer 144","parent":36,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-98,"ix":10},"p":{"a":0,"k":[120.235,111.393,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[94.767,94.767,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":39,"s":[0,0]},{"t":49,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":44,"s":[8]},{"t":49,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":39,"op":50,"st":22,"ct":1,"bm":0},{"ddd":0,"ind":39,"ty":4,"nm":"Shape Layer 143","parent":36,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-98,"ix":10},"p":{"a":0,"k":[88.825,133.033,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[94.767,94.767,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":36,"s":[0,0]},{"t":46,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":41,"s":[8]},{"t":46,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":36,"op":47,"st":19,"ct":1,"bm":0},{"ddd":0,"ind":40,"ty":3,"nm":"NULL CONTROL 41","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":149,"ix":10},"p":{"a":0,"k":[170.775,405.984,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"ip":69,"op":86,"st":0,"bm":0},{"ddd":0,"ind":41,"ty":4,"nm":"Shape Layer 142","parent":40,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-98,"ix":10},"p":{"a":0,"k":[141.175,96.966,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[94.767,94.767,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":75,"s":[0,0]},{"t":85,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":80,"s":[8]},{"t":85,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":75,"op":86,"st":58,"ct":1,"bm":0},{"ddd":0,"ind":42,"ty":4,"nm":"Shape Layer 141","parent":40,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-98,"ix":10},"p":{"a":0,"k":[120.235,111.393,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[94.767,94.767,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":72,"s":[0,0]},{"t":82,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":77,"s":[8]},{"t":82,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":72,"op":83,"st":55,"ct":1,"bm":0},{"ddd":0,"ind":43,"ty":4,"nm":"Shape Layer 140","parent":40,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-98,"ix":10},"p":{"a":0,"k":[88.825,133.033,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[94.767,94.767,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":69,"s":[0,0]},{"t":79,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":74,"s":[8]},{"t":79,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":69,"op":80,"st":52,"ct":1,"bm":0},{"ddd":0,"ind":44,"ty":3,"nm":"NULL CONTROL 40","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-38,"ix":10},"p":{"a":0,"k":[361.667,296.006,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[64.144,64.144,100],"ix":6,"l":2}},"ao":0,"ip":56,"op":79,"st":3,"bm":0},{"ddd":0,"ind":45,"ty":4,"nm":"Shape Layer 139","parent":44,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[116.5,159.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":64,"s":[0,0]},{"t":78,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":69,"s":[20]},{"t":78,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":64,"op":79,"st":47,"ct":1,"bm":0},{"ddd":0,"ind":46,"ty":4,"nm":"Shape Layer 138","parent":44,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[104.5,102.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":59,"s":[0,0]},{"t":73,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":64,"s":[20]},{"t":73,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":59,"op":74,"st":42,"ct":1,"bm":0},{"ddd":0,"ind":47,"ty":4,"nm":"Shape Layer 137","parent":44,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[125.5,70.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":56,"s":[0,0]},{"t":70,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":61,"s":[20]},{"t":70,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":56,"op":71,"st":39,"ct":1,"bm":0},{"ddd":0,"ind":48,"ty":3,"nm":"NULL CONTROL 39","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-194,"ix":10},"p":{"a":0,"k":[104.849,283.658,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[79.053,79.053,100],"ix":6,"l":2}},"ao":0,"ip":49,"op":68,"st":7,"bm":0},{"ddd":0,"ind":49,"ty":4,"nm":"circles_water 60","parent":48,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[144.5,86.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":57,"s":[0,0]},{"t":67,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":62,"s":[8]},{"t":67,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":57,"op":68,"st":40,"ct":1,"bm":0},{"ddd":0,"ind":50,"ty":4,"nm":"circles_water 59","parent":48,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[127.5,98.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":54,"s":[0,0]},{"t":64,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":59,"s":[8]},{"t":64,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":54,"op":65,"st":37,"ct":1,"bm":0},{"ddd":0,"ind":51,"ty":4,"nm":"circles_water 58","parent":48,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[95.5,118.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":51,"s":[0,0]},{"t":65,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":56,"s":[8]},{"t":65,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":51,"op":66,"st":34,"ct":1,"bm":0},{"ddd":0,"ind":52,"ty":4,"nm":"circles_water 57","parent":48,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[85.5,143.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":49,"s":[0,0]},{"t":59,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":54,"s":[8]},{"t":59,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":49,"op":60,"st":32,"ct":1,"bm":0},{"ddd":0,"ind":53,"ty":3,"nm":"NULL CONTROL 38","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-201,"ix":10},"p":{"a":0,"k":[161.993,160.558,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[109.232,109.232,100],"ix":6,"l":2}},"ao":0,"ip":39,"op":56,"st":-33,"bm":0},{"ddd":0,"ind":54,"ty":4,"nm":"Shape Layer 136","parent":53,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[130,145,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":45,"s":[0,0]},{"t":55,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":50,"s":[8]},{"t":55,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":45,"op":56,"st":28,"ct":1,"bm":0},{"ddd":0,"ind":55,"ty":4,"nm":"Shape Layer 135","parent":53,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[118,121,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":42,"s":[0,0]},{"t":52,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":47,"s":[8]},{"t":52,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":42,"op":53,"st":25,"ct":1,"bm":0},{"ddd":0,"ind":56,"ty":4,"nm":"Shape Layer 134","parent":53,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[100,85,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":39,"s":[0,0]},{"t":49,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":44,"s":[8]},{"t":49,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":39,"op":50,"st":22,"ct":1,"bm":0},{"ddd":0,"ind":57,"ty":3,"nm":"NULL CONTROL 37","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":28,"ix":10},"p":{"a":0,"k":[135.919,409.814,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[45.483,45.483,100],"ix":6,"l":2}},"ao":0,"ip":28,"op":51,"st":-25,"bm":0},{"ddd":0,"ind":58,"ty":4,"nm":"Shape Layer 133","parent":57,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[116.5,159.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":36,"s":[0,0]},{"t":50,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":41,"s":[20]},{"t":50,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":36,"op":51,"st":19,"ct":1,"bm":0},{"ddd":0,"ind":59,"ty":4,"nm":"Shape Layer 132","parent":57,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[104.5,102.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":31,"s":[0,0]},{"t":45,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":36,"s":[20]},{"t":45,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":31,"op":46,"st":14,"ct":1,"bm":0},{"ddd":0,"ind":60,"ty":4,"nm":"Shape Layer 131","parent":57,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[125.5,70.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":28,"s":[0,0]},{"t":42,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":33,"s":[20]},{"t":42,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":28,"op":43,"st":11,"ct":1,"bm":0},{"ddd":0,"ind":61,"ty":3,"nm":"NULL CONTROL 36","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":104,"ix":10},"p":{"a":0,"k":[429.936,406.449,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[63.273,63.273,100],"ix":6,"l":2}},"ao":0,"ip":41,"op":62,"st":-1,"bm":0},{"ddd":0,"ind":62,"ty":4,"nm":"circles_water 56","parent":61,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[144.5,86.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":50.334,"s":[0,0]},{"t":62,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":56.166,"s":[8]},{"t":62,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":49,"op":62,"st":32,"ct":1,"bm":0},{"ddd":0,"ind":63,"ty":4,"nm":"circles_water 55","parent":61,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[127.5,98.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":46.834,"s":[0,0]},{"t":58.5,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":52.666,"s":[8]},{"t":58.5,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":46,"op":59,"st":29,"ct":1,"bm":0},{"ddd":0,"ind":64,"ty":4,"nm":"circles_water 54","parent":61,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[95.5,118.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":43.334,"s":[0,0]},{"t":59.666015625,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":49.166,"s":[8]},{"t":59.666015625,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":43,"op":60,"st":26,"ct":1,"bm":0},{"ddd":0,"ind":65,"ty":4,"nm":"circles_water 53","parent":61,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[85.5,143.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":41,"s":[0,0]},{"t":52.666015625,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":46.834,"s":[8]},{"t":52.666015625,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":41,"op":54,"st":24,"ct":1,"bm":0},{"ddd":0,"ind":66,"ty":3,"nm":"NULL CONTROL 35","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-6,"ix":10},"p":{"a":0,"k":[354.201,369.74,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[91.776,91.776,100],"ix":6,"l":2}},"ao":0,"ip":28,"op":45,"st":-44,"bm":0},{"ddd":0,"ind":67,"ty":4,"nm":"Shape Layer 130","parent":66,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[130,145,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":34,"s":[0,0]},{"t":44,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":39,"s":[8]},{"t":44,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":34,"op":45,"st":17,"ct":1,"bm":0},{"ddd":0,"ind":68,"ty":4,"nm":"Shape Layer 129","parent":66,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[118,121,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":31,"s":[0,0]},{"t":41,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":36,"s":[8]},{"t":41,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":31,"op":42,"st":14,"ct":1,"bm":0},{"ddd":0,"ind":69,"ty":4,"nm":"Shape Layer 128","parent":66,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[100,85,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":28,"s":[0,0]},{"t":38,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":33,"s":[8]},{"t":38,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":28,"op":39,"st":11,"ct":1,"bm":0},{"ddd":0,"ind":70,"ty":3,"nm":"NULL CONTROL 34","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":115,"ix":10},"p":{"a":0,"k":[116.24,118.527,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[60.561,60.561,100],"ix":6,"l":2}},"ao":0,"ip":13,"op":36,"st":-40,"bm":0},{"ddd":0,"ind":71,"ty":4,"nm":"Shape Layer 127","parent":70,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[116.5,159.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":21,"s":[0,0]},{"t":35,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":26,"s":[20]},{"t":35,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":21,"op":36,"st":4,"ct":1,"bm":0},{"ddd":0,"ind":72,"ty":4,"nm":"Shape Layer 126","parent":70,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[104.5,102.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":16,"s":[0,0]},{"t":30,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":21,"s":[20]},{"t":30,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":16,"op":31,"st":-1,"ct":1,"bm":0},{"ddd":0,"ind":73,"ty":4,"nm":"Shape Layer 125","parent":70,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[125.5,70.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":13,"s":[0,0]},{"t":27,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":18,"s":[20]},{"t":27,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":13,"op":28,"st":-4,"ct":1,"bm":0},{"ddd":0,"ind":74,"ty":3,"nm":"NULL CONTROL 33","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":118,"ix":10},"p":{"a":0,"k":[234.893,318.753,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[73.409,73.409,100],"ix":6,"l":2}},"ao":0,"ip":38,"op":59,"st":-4,"bm":0},{"ddd":0,"ind":75,"ty":4,"nm":"circles_water 52","parent":74,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[144.5,86.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":47.334,"s":[0,0]},{"t":63,"s":[100,100]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":53.166,"s":[20]},{"t":63,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":46,"op":64,"st":29,"ct":1,"bm":0},{"ddd":0,"ind":76,"ty":4,"nm":"circles_water 51","parent":74,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[127.5,98.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":43.834,"s":[0,0]},{"t":59,"s":[100,100]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":49.666,"s":[20]},{"t":59,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":43,"op":61,"st":26,"ct":1,"bm":0},{"ddd":0,"ind":77,"ty":4,"nm":"circles_water 50","parent":74,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[95.5,118.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":40.334,"s":[0,0]},{"t":61,"s":[100,100]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":46.166,"s":[20]},{"t":61,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":40,"op":62,"st":23,"ct":1,"bm":0},{"ddd":0,"ind":78,"ty":4,"nm":"circles_water 49","parent":74,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[85.5,143.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":38,"s":[0,0]},{"t":54,"s":[100,100]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":43.834,"s":[20]},{"t":54,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":38,"op":56,"st":21,"ct":1,"bm":0},{"ddd":0,"ind":79,"ty":3,"nm":"NULL CONTROL 32","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-19,"ix":10},"p":{"a":0,"k":[310.614,163.154,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[83.285,83.285,100],"ix":6,"l":2}},"ao":0,"ip":32,"op":51,"st":-10,"bm":0},{"ddd":0,"ind":80,"ty":4,"nm":"circles_water 48","parent":79,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[144.5,86.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":40,"s":[0,0]},{"t":54,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":45,"s":[20]},{"t":54,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":40,"op":56,"st":23,"ct":1,"bm":0},{"ddd":0,"ind":81,"ty":4,"nm":"circles_water 47","parent":79,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[127.5,98.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":37,"s":[0,0]},{"t":51,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":42,"s":[20]},{"t":51,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":37,"op":53,"st":20,"ct":1,"bm":0},{"ddd":0,"ind":82,"ty":4,"nm":"circles_water 46","parent":79,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[95.5,118.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":34,"s":[0,0]},{"t":52,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":39,"s":[20]},{"t":52,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":34,"op":54,"st":17,"ct":1,"bm":0},{"ddd":0,"ind":83,"ty":4,"nm":"circles_water 45","parent":79,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[85.5,143.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":32,"s":[0,0]},{"t":46,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":37,"s":[20]},{"t":46,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":32,"op":48,"st":15,"ct":1,"bm":0},{"ddd":0,"ind":84,"ty":3,"nm":"NULL CONTROL 31","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":112,"ix":10},"p":{"a":0,"k":[201.906,230.732,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[68.667,68.667,100],"ix":6,"l":2}},"ao":0,"ip":25,"op":42,"st":-47,"bm":0},{"ddd":0,"ind":85,"ty":4,"nm":"Shape Layer 124","parent":84,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[130,145,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":31,"s":[0,0]},{"t":41,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":36,"s":[20]},{"t":41,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":31,"op":42,"st":14,"ct":1,"bm":0},{"ddd":0,"ind":86,"ty":4,"nm":"Shape Layer 123","parent":84,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[118,121,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":28,"s":[0,0]},{"t":38,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":33,"s":[20]},{"t":38,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":28,"op":39,"st":11,"ct":1,"bm":0},{"ddd":0,"ind":87,"ty":4,"nm":"Shape Layer 122","parent":84,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[100,85,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":25,"s":[0,0]},{"t":35,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":30,"s":[20]},{"t":35,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":25,"op":36,"st":8,"ct":1,"bm":0},{"ddd":0,"ind":88,"ty":3,"nm":"NULL CONTROL 30","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-38,"ix":10},"p":{"a":0,"k":[255.736,330.597,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[62.687,62.687,100],"ix":6,"l":2}},"ao":0,"ip":9,"op":32,"st":-44,"bm":0},{"ddd":0,"ind":89,"ty":4,"nm":"Shape Layer 121","parent":88,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[116.5,159.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":17,"s":[0,0]},{"t":31,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":22,"s":[18]},{"t":31,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":17,"op":32,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":90,"ty":4,"nm":"Shape Layer 120","parent":88,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[104.5,102.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":12,"s":[0,0]},{"t":26,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":17,"s":[18]},{"t":26,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":12,"op":27,"st":-5,"ct":1,"bm":0},{"ddd":0,"ind":91,"ty":4,"nm":"Shape Layer 119","parent":88,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[125.5,70.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":9,"s":[0,0]},{"t":23,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[18]},{"t":23,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":9,"op":24,"st":-8,"ct":1,"bm":0},{"ddd":0,"ind":92,"ty":3,"nm":"NULL CONTROL 29","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[300.927,201.64,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[55.198,55.198,100],"ix":6,"l":2}},"ao":0,"ip":5,"op":24,"st":-37,"bm":0},{"ddd":0,"ind":93,"ty":4,"nm":"circles_water 44","parent":92,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[144.5,86.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":13,"s":[0,0]},{"t":23,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":18,"s":[20]},{"t":23,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":13,"op":24,"st":-4,"ct":1,"bm":0},{"ddd":0,"ind":94,"ty":4,"nm":"circles_water 43","parent":92,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[127.5,98.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":10,"s":[0,0]},{"t":20,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":15,"s":[20]},{"t":20,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":10,"op":21,"st":-7,"ct":1,"bm":0},{"ddd":0,"ind":95,"ty":4,"nm":"circles_water 42","parent":92,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[95.5,118.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":7,"s":[0,0]},{"t":21,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":10,"s":[20]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[20]},{"t":21,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":7,"op":22,"st":-10,"ct":1,"bm":0},{"ddd":0,"ind":96,"ty":4,"nm":"circles_water 41","parent":92,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[85.5,143.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":5,"s":[0,0]},{"t":15,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":10,"s":[20]},{"t":15,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":5,"op":16,"st":-12,"ct":1,"bm":0},{"ddd":0,"ind":97,"ty":3,"nm":"NULL CONTROL 28","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":32,"ix":10},"p":{"a":0,"k":[403.775,95.984,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[79.632,79.632,100],"ix":6,"l":2}},"ao":0,"ip":34,"op":51,"st":-35,"bm":0},{"ddd":0,"ind":98,"ty":4,"nm":"Shape Layer 118","parent":97,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-98,"ix":10},"p":{"a":0,"k":[141.175,96.966,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[94.767,94.767,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":40,"s":[0,0]},{"t":50,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":45,"s":[8]},{"t":50,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":40,"op":51,"st":23,"ct":1,"bm":0},{"ddd":0,"ind":99,"ty":4,"nm":"Shape Layer 117","parent":97,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-98,"ix":10},"p":{"a":0,"k":[120.235,111.393,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[94.767,94.767,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":37,"s":[0,0]},{"t":47,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":42,"s":[8]},{"t":47,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":37,"op":48,"st":20,"ct":1,"bm":0},{"ddd":0,"ind":100,"ty":4,"nm":"Shape Layer 116","parent":97,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-98,"ix":10},"p":{"a":0,"k":[88.825,133.033,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[94.767,94.767,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":34,"s":[0,0]},{"t":44,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":39,"s":[8]},{"t":44,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":34,"op":45,"st":17,"ct":1,"bm":0},{"ddd":0,"ind":101,"ty":3,"nm":"NULL CONTROL 27","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":149,"ix":10},"p":{"a":0,"k":[170.775,405.984,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"ip":67,"op":84,"st":-2,"bm":0},{"ddd":0,"ind":102,"ty":4,"nm":"Shape Layer 115","parent":101,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-98,"ix":10},"p":{"a":0,"k":[141.175,96.966,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[94.767,94.767,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":73,"s":[0,0]},{"t":83,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":78,"s":[8]},{"t":83,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":73,"op":84,"st":56,"ct":1,"bm":0},{"ddd":0,"ind":103,"ty":4,"nm":"Shape Layer 114","parent":101,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-98,"ix":10},"p":{"a":0,"k":[120.235,111.393,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[94.767,94.767,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":70,"s":[0,0]},{"t":80,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":75,"s":[8]},{"t":80,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":70,"op":81,"st":53,"ct":1,"bm":0},{"ddd":0,"ind":104,"ty":4,"nm":"Shape Layer 113","parent":101,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-98,"ix":10},"p":{"a":0,"k":[88.825,133.033,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[94.767,94.767,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":67,"s":[0,0]},{"t":77,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":72,"s":[8]},{"t":77,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":67,"op":78,"st":50,"ct":1,"bm":0},{"ddd":0,"ind":105,"ty":3,"nm":"NULL CONTROL 26","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-38,"ix":10},"p":{"a":0,"k":[361.667,296.006,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[64.144,64.144,100],"ix":6,"l":2}},"ao":0,"ip":54,"op":77,"st":1,"bm":0},{"ddd":0,"ind":106,"ty":4,"nm":"Shape Layer 112","parent":105,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[116.5,159.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":62,"s":[0,0]},{"t":76,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":67,"s":[20]},{"t":76,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":62,"op":77,"st":45,"ct":1,"bm":0},{"ddd":0,"ind":107,"ty":4,"nm":"Shape Layer 111","parent":105,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[104.5,102.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":57,"s":[0,0]},{"t":71,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":62,"s":[20]},{"t":71,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":57,"op":72,"st":40,"ct":1,"bm":0},{"ddd":0,"ind":108,"ty":4,"nm":"Shape Layer 110","parent":105,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[125.5,70.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":54,"s":[0,0]},{"t":68,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":59,"s":[20]},{"t":68,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":54,"op":69,"st":37,"ct":1,"bm":0},{"ddd":0,"ind":109,"ty":3,"nm":"NULL CONTROL 25","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-194,"ix":10},"p":{"a":0,"k":[104.849,283.658,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[79.053,79.053,100],"ix":6,"l":2}},"ao":0,"ip":47,"op":66,"st":5,"bm":0},{"ddd":0,"ind":110,"ty":4,"nm":"circles_water 40","parent":109,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[144.5,86.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":55,"s":[0,0]},{"t":65,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":60,"s":[8]},{"t":65,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":55,"op":66,"st":38,"ct":1,"bm":0},{"ddd":0,"ind":111,"ty":4,"nm":"circles_water 39","parent":109,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[127.5,98.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":52,"s":[0,0]},{"t":62,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":57,"s":[8]},{"t":62,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":52,"op":63,"st":35,"ct":1,"bm":0},{"ddd":0,"ind":112,"ty":4,"nm":"circles_water 38","parent":109,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[95.5,118.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":49,"s":[0,0]},{"t":63,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":54,"s":[8]},{"t":63,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":49,"op":64,"st":32,"ct":1,"bm":0},{"ddd":0,"ind":113,"ty":4,"nm":"circles_water 37","parent":109,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[85.5,143.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":47,"s":[0,0]},{"t":57,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":52,"s":[8]},{"t":57,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":47,"op":58,"st":30,"ct":1,"bm":0},{"ddd":0,"ind":114,"ty":3,"nm":"NULL CONTROL 24","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-201,"ix":10},"p":{"a":0,"k":[161.993,160.558,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[109.232,109.232,100],"ix":6,"l":2}},"ao":0,"ip":37,"op":54,"st":-35,"bm":0},{"ddd":0,"ind":115,"ty":4,"nm":"Shape Layer 109","parent":114,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[130,145,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":43,"s":[0,0]},{"t":53,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":48,"s":[8]},{"t":53,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":43,"op":54,"st":26,"ct":1,"bm":0},{"ddd":0,"ind":116,"ty":4,"nm":"Shape Layer 108","parent":114,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[118,121,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":40,"s":[0,0]},{"t":50,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":45,"s":[8]},{"t":50,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":40,"op":51,"st":23,"ct":1,"bm":0},{"ddd":0,"ind":117,"ty":4,"nm":"Shape Layer 107","parent":114,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[100,85,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":37,"s":[0,0]},{"t":47,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":42,"s":[8]},{"t":47,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":37,"op":48,"st":20,"ct":1,"bm":0},{"ddd":0,"ind":118,"ty":3,"nm":"NULL CONTROL 23","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":28,"ix":10},"p":{"a":0,"k":[135.919,409.814,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[45.483,45.483,100],"ix":6,"l":2}},"ao":0,"ip":26,"op":49,"st":-27,"bm":0},{"ddd":0,"ind":119,"ty":4,"nm":"Shape Layer 106","parent":118,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[116.5,159.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":34,"s":[0,0]},{"t":48,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":39,"s":[20]},{"t":48,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":34,"op":49,"st":17,"ct":1,"bm":0},{"ddd":0,"ind":120,"ty":4,"nm":"Shape Layer 105","parent":118,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[104.5,102.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":29,"s":[0,0]},{"t":43,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":34,"s":[20]},{"t":43,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":29,"op":44,"st":12,"ct":1,"bm":0},{"ddd":0,"ind":121,"ty":4,"nm":"Shape Layer 104","parent":118,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[125.5,70.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":26,"s":[0,0]},{"t":40,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":31,"s":[20]},{"t":40,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":26,"op":41,"st":9,"ct":1,"bm":0},{"ddd":0,"ind":122,"ty":3,"nm":"NULL CONTROL 22","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":104,"ix":10},"p":{"a":0,"k":[429.936,406.449,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[63.273,63.273,100],"ix":6,"l":2}},"ao":0,"ip":39,"op":60,"st":-3,"bm":0},{"ddd":0,"ind":123,"ty":4,"nm":"circles_water 36","parent":122,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[144.5,86.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":48.334,"s":[0,0]},{"t":60,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":54.166,"s":[8]},{"t":60,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":47,"op":60,"st":30,"ct":1,"bm":0},{"ddd":0,"ind":124,"ty":4,"nm":"circles_water 35","parent":122,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[127.5,98.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":44.834,"s":[0,0]},{"t":56.5,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":50.666,"s":[8]},{"t":56.5,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":44,"op":57,"st":27,"ct":1,"bm":0},{"ddd":0,"ind":125,"ty":4,"nm":"circles_water 34","parent":122,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[95.5,118.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":41.334,"s":[0,0]},{"t":57.666015625,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":47.166,"s":[8]},{"t":57.666015625,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":41,"op":58,"st":24,"ct":1,"bm":0},{"ddd":0,"ind":126,"ty":4,"nm":"circles_water 33","parent":122,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[85.5,143.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":39,"s":[0,0]},{"t":50.666015625,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":44.834,"s":[8]},{"t":50.666015625,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":39,"op":52,"st":22,"ct":1,"bm":0},{"ddd":0,"ind":127,"ty":3,"nm":"NULL CONTROL 21","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-6,"ix":10},"p":{"a":0,"k":[354.201,369.74,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[91.776,91.776,100],"ix":6,"l":2}},"ao":0,"ip":26,"op":43,"st":-46,"bm":0},{"ddd":0,"ind":128,"ty":4,"nm":"Shape Layer 103","parent":127,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[130,145,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":32,"s":[0,0]},{"t":42,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":37,"s":[8]},{"t":42,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":32,"op":43,"st":15,"ct":1,"bm":0},{"ddd":0,"ind":129,"ty":4,"nm":"Shape Layer 102","parent":127,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[118,121,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":29,"s":[0,0]},{"t":39,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":34,"s":[8]},{"t":39,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":29,"op":40,"st":12,"ct":1,"bm":0},{"ddd":0,"ind":130,"ty":4,"nm":"Shape Layer 101","parent":127,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[100,85,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":26,"s":[0,0]},{"t":36,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":31,"s":[8]},{"t":36,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":26,"op":37,"st":9,"ct":1,"bm":0},{"ddd":0,"ind":131,"ty":3,"nm":"NULL CONTROL 20","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":115,"ix":10},"p":{"a":0,"k":[116.24,118.527,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[60.561,60.561,100],"ix":6,"l":2}},"ao":0,"ip":11,"op":34,"st":-42,"bm":0},{"ddd":0,"ind":132,"ty":4,"nm":"Shape Layer 100","parent":131,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[116.5,159.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":19,"s":[0,0]},{"t":33,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":24,"s":[20]},{"t":33,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":19,"op":34,"st":2,"ct":1,"bm":0},{"ddd":0,"ind":133,"ty":4,"nm":"Shape Layer 99","parent":131,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[104.5,102.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":14,"s":[0,0]},{"t":28,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":19,"s":[20]},{"t":28,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":14,"op":29,"st":-3,"ct":1,"bm":0},{"ddd":0,"ind":134,"ty":4,"nm":"Shape Layer 98","parent":131,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[125.5,70.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":11,"s":[0,0]},{"t":25,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[20]},{"t":25,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":11,"op":26,"st":-6,"ct":1,"bm":0},{"ddd":0,"ind":135,"ty":3,"nm":"NULL CONTROL 19","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":118,"ix":10},"p":{"a":0,"k":[234.893,318.753,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[73.409,73.409,100],"ix":6,"l":2}},"ao":0,"ip":36,"op":57,"st":-6,"bm":0},{"ddd":0,"ind":136,"ty":4,"nm":"circles_water 32","parent":135,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[144.5,86.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":45.334,"s":[0,0]},{"t":61,"s":[100,100]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":51.166,"s":[20]},{"t":61,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":44,"op":62,"st":27,"ct":1,"bm":0},{"ddd":0,"ind":137,"ty":4,"nm":"circles_water 31","parent":135,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[127.5,98.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":41.834,"s":[0,0]},{"t":57,"s":[100,100]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":47.666,"s":[20]},{"t":57,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":41,"op":59,"st":24,"ct":1,"bm":0},{"ddd":0,"ind":138,"ty":4,"nm":"circles_water 30","parent":135,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[95.5,118.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":38.334,"s":[0,0]},{"t":59,"s":[100,100]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":44.166,"s":[20]},{"t":59,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":38,"op":60,"st":21,"ct":1,"bm":0},{"ddd":0,"ind":139,"ty":4,"nm":"circles_water 29","parent":135,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[85.5,143.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":36,"s":[0,0]},{"t":52,"s":[100,100]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":41.834,"s":[20]},{"t":52,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":36,"op":54,"st":19,"ct":1,"bm":0},{"ddd":0,"ind":140,"ty":3,"nm":"NULL CONTROL 18","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-19,"ix":10},"p":{"a":0,"k":[310.614,163.154,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[83.285,83.285,100],"ix":6,"l":2}},"ao":0,"ip":30,"op":49,"st":-12,"bm":0},{"ddd":0,"ind":141,"ty":4,"nm":"circles_water 28","parent":140,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[144.5,86.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":38,"s":[0,0]},{"t":52,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":43,"s":[20]},{"t":52,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":38,"op":54,"st":21,"ct":1,"bm":0},{"ddd":0,"ind":142,"ty":4,"nm":"circles_water 27","parent":140,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[127.5,98.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":35,"s":[0,0]},{"t":49,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":40,"s":[20]},{"t":49,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":35,"op":51,"st":18,"ct":1,"bm":0},{"ddd":0,"ind":143,"ty":4,"nm":"circles_water 26","parent":140,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[95.5,118.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":32,"s":[0,0]},{"t":50,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":37,"s":[20]},{"t":50,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":32,"op":52,"st":15,"ct":1,"bm":0},{"ddd":0,"ind":144,"ty":4,"nm":"circles_water 25","parent":140,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[85.5,143.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":30,"s":[0,0]},{"t":44,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":35,"s":[20]},{"t":44,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":30,"op":46,"st":13,"ct":1,"bm":0},{"ddd":0,"ind":145,"ty":3,"nm":"NULL CONTROL 17","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":112,"ix":10},"p":{"a":0,"k":[201.906,230.732,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[68.667,68.667,100],"ix":6,"l":2}},"ao":0,"ip":23,"op":40,"st":-49,"bm":0},{"ddd":0,"ind":146,"ty":4,"nm":"Shape Layer 97","parent":145,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[130,145,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":29,"s":[0,0]},{"t":39,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":34,"s":[20]},{"t":39,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":29,"op":40,"st":12,"ct":1,"bm":0},{"ddd":0,"ind":147,"ty":4,"nm":"Shape Layer 96","parent":145,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[118,121,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":26,"s":[0,0]},{"t":36,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":31,"s":[20]},{"t":36,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":26,"op":37,"st":9,"ct":1,"bm":0},{"ddd":0,"ind":148,"ty":4,"nm":"Shape Layer 95","parent":145,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[100,85,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":23,"s":[0,0]},{"t":33,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":28,"s":[20]},{"t":33,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":23,"op":34,"st":6,"ct":1,"bm":0},{"ddd":0,"ind":149,"ty":3,"nm":"NULL CONTROL 16","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":-38,"ix":10},"p":{"a":0,"k":[255.736,330.597,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[62.687,62.687,100],"ix":6,"l":2}},"ao":0,"ip":7,"op":30,"st":-46,"bm":0},{"ddd":0,"ind":150,"ty":4,"nm":"Shape Layer 94","parent":149,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[116.5,159.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":15,"s":[0,0]},{"t":29,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":20,"s":[18]},{"t":29,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":15,"op":30,"st":-2,"ct":1,"bm":0},{"ddd":0,"ind":151,"ty":4,"nm":"Shape Layer 93","parent":149,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[104.5,102.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":10,"s":[0,0]},{"t":24,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":15,"s":[18]},{"t":24,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":10,"op":25,"st":-7,"ct":1,"bm":0},{"ddd":0,"ind":152,"ty":4,"nm":"Shape Layer 92","parent":149,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[125.5,70.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":7,"s":[0,0]},{"t":21,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[18]},{"t":21,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":7,"op":22,"st":-10,"ct":1,"bm":0},{"ddd":0,"ind":153,"ty":3,"nm":"NULL CONTROL 15","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[300.927,201.64,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[55.198,55.198,100],"ix":6,"l":2}},"ao":0,"ip":3,"op":22,"st":-39,"bm":0},{"ddd":0,"ind":154,"ty":4,"nm":"circles_water 24","parent":153,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[144.5,86.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":11,"s":[0,0]},{"t":21,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[20]},{"t":21,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":11,"op":22,"st":-6,"ct":1,"bm":0},{"ddd":0,"ind":155,"ty":4,"nm":"circles_water 23","parent":153,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[127.5,98.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":8,"s":[0,0]},{"t":18,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[20]},{"t":18,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":8,"op":19,"st":-9,"ct":1,"bm":0},{"ddd":0,"ind":156,"ty":4,"nm":"circles_water 22","parent":153,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[95.5,118.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":5,"s":[0,0]},{"t":19,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":8,"s":[20]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":10,"s":[20]},{"t":19,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":5,"op":20,"st":-12,"ct":1,"bm":0},{"ddd":0,"ind":157,"ty":4,"nm":"circles_water 21","parent":153,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[85.5,143.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":3,"s":[0,0]},{"t":13,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":8,"s":[20]},{"t":13,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":3,"op":14,"st":-14,"ct":1,"bm":0},{"ddd":0,"ind":158,"ty":3,"nm":"NULL CONTROL 14","sr":1,"ks":{"o":{"a":0,"k":0,"ix":11},"r":{"a":0,"k":32,"ix":10},"p":{"a":0,"k":[403.775,95.984,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[79.632,79.632,100],"ix":6,"l":2}},"ao":0,"ip":32,"op":49,"st":-37,"bm":0},{"ddd":0,"ind":159,"ty":4,"nm":"Shape Layer 91","parent":158,"sr":1,"ks":{"o":{"a":0,"k":80,"ix":11},"r":{"a":0,"k":-98,"ix":10},"p":{"a":0,"k":[141.175,96.966,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[94.767,94.767,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":38,"s":[0,0]},{"t":48,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":43,"s":[8]},{"t":48,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":38,"op":49,"st":21,"ct":1,"bm":0},{"ddd":0,"ind":160,"ty":4,"nm":"Shape Layer 90","parent":158,"sr":1,"ks":{"o":{"a":0,"k":80,"ix":11},"r":{"a":0,"k":-98,"ix":10},"p":{"a":0,"k":[120.235,111.393,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[94.767,94.767,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":35,"s":[0,0]},{"t":45,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":40,"s":[8]},{"t":45,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":35,"op":46,"st":18,"ct":1,"bm":0},{"ddd":0,"ind":161,"ty":4,"nm":"Shape Layer 89","parent":158,"sr":1,"ks":{"o":{"a":0,"k":80,"ix":11},"r":{"a":0,"k":-98,"ix":10},"p":{"a":0,"k":[88.825,133.033,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[94.767,94.767,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":32,"s":[0,0]},{"t":42,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":37,"s":[8]},{"t":42,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":32,"op":43,"st":15,"ct":1,"bm":0},{"ddd":0,"ind":162,"ty":3,"nm":"NULL CONTROL","sr":1,"ks":{"o":{"a":0,"k":80,"ix":11},"r":{"a":0,"k":149,"ix":10},"p":{"a":0,"k":[170.775,405.984,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"ip":65,"op":82,"st":-4,"bm":0},{"ddd":0,"ind":163,"ty":4,"nm":"Shape Layer 88","parent":162,"sr":1,"ks":{"o":{"a":0,"k":80,"ix":11},"r":{"a":0,"k":-98,"ix":10},"p":{"a":0,"k":[141.175,96.966,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[94.767,94.767,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":71,"s":[0,0]},{"t":81,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":76,"s":[8]},{"t":81,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":71,"op":82,"st":54,"ct":1,"bm":0},{"ddd":0,"ind":164,"ty":4,"nm":"Shape Layer 87","parent":162,"sr":1,"ks":{"o":{"a":0,"k":80,"ix":11},"r":{"a":0,"k":-98,"ix":10},"p":{"a":0,"k":[120.235,111.393,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[94.767,94.767,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":68,"s":[0,0]},{"t":78,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":73,"s":[8]},{"t":78,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":68,"op":79,"st":51,"ct":1,"bm":0},{"ddd":0,"ind":165,"ty":4,"nm":"Shape Layer 86","parent":162,"sr":1,"ks":{"o":{"a":0,"k":80,"ix":11},"r":{"a":0,"k":-98,"ix":10},"p":{"a":0,"k":[88.825,133.033,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[94.767,94.767,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":65,"s":[0,0]},{"t":75,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":70,"s":[8]},{"t":75,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":65,"op":76,"st":48,"ct":1,"bm":0},{"ddd":0,"ind":166,"ty":3,"nm":"NULL CONTROL 13","sr":1,"ks":{"o":{"a":0,"k":80,"ix":11},"r":{"a":0,"k":-38,"ix":10},"p":{"a":0,"k":[361.667,296.006,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[64.144,64.144,100],"ix":6,"l":2}},"ao":0,"ip":52,"op":75,"st":-1,"bm":0},{"ddd":0,"ind":167,"ty":4,"nm":"Shape Layer 85","parent":166,"sr":1,"ks":{"o":{"a":0,"k":80,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[116.5,159.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":60,"s":[0,0]},{"t":74,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":65,"s":[20]},{"t":74,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":60,"op":75,"st":43,"ct":1,"bm":0},{"ddd":0,"ind":168,"ty":4,"nm":"Shape Layer 84","parent":166,"sr":1,"ks":{"o":{"a":0,"k":80,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[104.5,102.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":55,"s":[0,0]},{"t":69,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":60,"s":[20]},{"t":69,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":55,"op":70,"st":38,"ct":1,"bm":0},{"ddd":0,"ind":169,"ty":4,"nm":"Shape Layer 83","parent":166,"sr":1,"ks":{"o":{"a":0,"k":80,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[125.5,70.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":52,"s":[0,0]},{"t":66,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":57,"s":[20]},{"t":66,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":52,"op":67,"st":35,"ct":1,"bm":0},{"ddd":0,"ind":170,"ty":3,"nm":"NULL CONTROL 12","sr":1,"ks":{"o":{"a":0,"k":0,"ix":11},"r":{"a":0,"k":-194,"ix":10},"p":{"a":0,"k":[104.849,283.658,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[79.053,79.053,100],"ix":6,"l":2}},"ao":0,"ip":45,"op":64,"st":3,"bm":0},{"ddd":0,"ind":171,"ty":4,"nm":"circles_water 20","parent":170,"sr":1,"ks":{"o":{"a":0,"k":70,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[144.5,86.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":53,"s":[0,0]},{"t":63,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":58,"s":[8]},{"t":63,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":53,"op":64,"st":36,"ct":1,"bm":0},{"ddd":0,"ind":172,"ty":4,"nm":"circles_water 19","parent":170,"sr":1,"ks":{"o":{"a":0,"k":70,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[127.5,98.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":50,"s":[0,0]},{"t":60,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":55,"s":[8]},{"t":60,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":50,"op":61,"st":33,"ct":1,"bm":0},{"ddd":0,"ind":173,"ty":4,"nm":"circles_water 18","parent":170,"sr":1,"ks":{"o":{"a":0,"k":70,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[95.5,118.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":47,"s":[0,0]},{"t":61,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":52,"s":[8]},{"t":61,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":47,"op":62,"st":30,"ct":1,"bm":0},{"ddd":0,"ind":174,"ty":4,"nm":"circles_water 17","parent":170,"sr":1,"ks":{"o":{"a":0,"k":70,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[85.5,143.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":45,"s":[0,0]},{"t":55,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":50,"s":[8]},{"t":55,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":45,"op":56,"st":28,"ct":1,"bm":0},{"ddd":0,"ind":175,"ty":3,"nm":"NULL CONTROL 11","sr":1,"ks":{"o":{"a":0,"k":70,"ix":11},"r":{"a":0,"k":-201,"ix":10},"p":{"a":0,"k":[161.993,160.558,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[109.232,109.232,100],"ix":6,"l":2}},"ao":0,"ip":35,"op":52,"st":-37,"bm":0},{"ddd":0,"ind":176,"ty":4,"nm":"Shape Layer 82","parent":175,"sr":1,"ks":{"o":{"a":0,"k":70,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[130,145,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":41,"s":[0,0]},{"t":51,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":46,"s":[8]},{"t":51,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":41,"op":52,"st":24,"ct":1,"bm":0},{"ddd":0,"ind":177,"ty":4,"nm":"Shape Layer 81","parent":175,"sr":1,"ks":{"o":{"a":0,"k":70,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[118,121,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":38,"s":[0,0]},{"t":48,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":43,"s":[8]},{"t":48,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":38,"op":49,"st":21,"ct":1,"bm":0},{"ddd":0,"ind":178,"ty":4,"nm":"Shape Layer 80","parent":175,"sr":1,"ks":{"o":{"a":0,"k":70,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[100,85,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":35,"s":[0,0]},{"t":45,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":40,"s":[8]},{"t":45,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":35,"op":46,"st":18,"ct":1,"bm":0},{"ddd":0,"ind":179,"ty":3,"nm":"NULL CONTROL 10","sr":1,"ks":{"o":{"a":0,"k":70,"ix":11},"r":{"a":0,"k":28,"ix":10},"p":{"a":0,"k":[135.919,409.814,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[45.483,45.483,100],"ix":6,"l":2}},"ao":0,"ip":24,"op":47,"st":-29,"bm":0},{"ddd":0,"ind":180,"ty":4,"nm":"Shape Layer 79","parent":179,"sr":1,"ks":{"o":{"a":0,"k":70,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[116.5,159.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":32,"s":[0,0]},{"t":46,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":37,"s":[20]},{"t":46,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":32,"op":47,"st":15,"ct":1,"bm":0},{"ddd":0,"ind":181,"ty":4,"nm":"Shape Layer 78","parent":179,"sr":1,"ks":{"o":{"a":0,"k":70,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[104.5,102.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":27,"s":[0,0]},{"t":41,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":32,"s":[20]},{"t":41,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":27,"op":42,"st":10,"ct":1,"bm":0},{"ddd":0,"ind":182,"ty":4,"nm":"Shape Layer 77","parent":179,"sr":1,"ks":{"o":{"a":0,"k":70,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[125.5,70.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":24,"s":[0,0]},{"t":38,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":29,"s":[20]},{"t":38,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":24,"op":39,"st":7,"ct":1,"bm":0},{"ddd":0,"ind":183,"ty":3,"nm":"NULL CONTROL 9","sr":1,"ks":{"o":{"a":0,"k":0,"ix":11},"r":{"a":0,"k":104,"ix":10},"p":{"a":0,"k":[429.936,406.449,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[63.273,63.273,100],"ix":6,"l":2}},"ao":0,"ip":37,"op":58,"st":-5,"bm":0},{"ddd":0,"ind":184,"ty":4,"nm":"circles_water 16","parent":183,"sr":1,"ks":{"o":{"a":0,"k":60,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[144.5,86.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":46.334,"s":[0,0]},{"t":58,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":52.166,"s":[8]},{"t":58,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":45,"op":58,"st":28,"ct":1,"bm":0},{"ddd":0,"ind":185,"ty":4,"nm":"circles_water 15","parent":183,"sr":1,"ks":{"o":{"a":0,"k":60,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[127.5,98.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":42.834,"s":[0,0]},{"t":54.5,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":48.666,"s":[8]},{"t":54.5,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":42,"op":55,"st":25,"ct":1,"bm":0},{"ddd":0,"ind":186,"ty":4,"nm":"circles_water 14","parent":183,"sr":1,"ks":{"o":{"a":0,"k":60,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[95.5,118.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":39.334,"s":[0,0]},{"t":55.666015625,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":45.166,"s":[8]},{"t":55.666015625,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":39,"op":56,"st":22,"ct":1,"bm":0},{"ddd":0,"ind":187,"ty":4,"nm":"circles_water 13","parent":183,"sr":1,"ks":{"o":{"a":0,"k":60,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[85.5,143.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":37,"s":[0,0]},{"t":48.666015625,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":42.834,"s":[8]},{"t":48.666015625,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":37,"op":50,"st":20,"ct":1,"bm":0},{"ddd":0,"ind":188,"ty":3,"nm":"NULL CONTROL 8","sr":1,"ks":{"o":{"a":0,"k":60,"ix":11},"r":{"a":0,"k":-6,"ix":10},"p":{"a":0,"k":[354.201,369.74,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[91.776,91.776,100],"ix":6,"l":2}},"ao":0,"ip":24,"op":41,"st":-48,"bm":0},{"ddd":0,"ind":189,"ty":4,"nm":"Shape Layer 76","parent":188,"sr":1,"ks":{"o":{"a":0,"k":60,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[130,145,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":30,"s":[0,0]},{"t":40,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":35,"s":[8]},{"t":40,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":30,"op":41,"st":13,"ct":1,"bm":0},{"ddd":0,"ind":190,"ty":4,"nm":"Shape Layer 75","parent":188,"sr":1,"ks":{"o":{"a":0,"k":60,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[118,121,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":27,"s":[0,0]},{"t":37,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":32,"s":[8]},{"t":37,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":27,"op":38,"st":10,"ct":1,"bm":0},{"ddd":0,"ind":191,"ty":4,"nm":"Shape Layer 74","parent":188,"sr":1,"ks":{"o":{"a":0,"k":60,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[100,85,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":24,"s":[0,0]},{"t":34,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":29,"s":[8]},{"t":34,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":24,"op":35,"st":7,"ct":1,"bm":0},{"ddd":0,"ind":192,"ty":3,"nm":"NULL CONTROL 7","sr":1,"ks":{"o":{"a":0,"k":0,"ix":11},"r":{"a":0,"k":115,"ix":10},"p":{"a":0,"k":[116.24,118.527,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[60.561,60.561,100],"ix":6,"l":2}},"ao":0,"ip":9,"op":32,"st":-44,"bm":0},{"ddd":0,"ind":193,"ty":4,"nm":"Shape Layer 73","parent":192,"sr":1,"ks":{"o":{"a":0,"k":90,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[116.5,159.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":17,"s":[0,0]},{"t":31,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":22,"s":[20]},{"t":31,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":17,"op":32,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":194,"ty":4,"nm":"Shape Layer 72","parent":192,"sr":1,"ks":{"o":{"a":0,"k":90,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[104.5,102.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":12,"s":[0,0]},{"t":26,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":17,"s":[20]},{"t":26,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":12,"op":27,"st":-5,"ct":1,"bm":0},{"ddd":0,"ind":195,"ty":4,"nm":"Shape Layer 71","parent":192,"sr":1,"ks":{"o":{"a":0,"k":90,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[125.5,70.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":9,"s":[0,0]},{"t":23,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[20]},{"t":23,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":9,"op":24,"st":-8,"ct":1,"bm":0},{"ddd":0,"ind":196,"ty":3,"nm":"NULL CONTROL 6","sr":1,"ks":{"o":{"a":0,"k":0,"ix":11},"r":{"a":0,"k":118,"ix":10},"p":{"a":0,"k":[234.893,318.753,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[73.409,73.409,100],"ix":6,"l":2}},"ao":0,"ip":34,"op":55,"st":-8,"bm":0},{"ddd":0,"ind":197,"ty":4,"nm":"circles_water 12","parent":196,"sr":1,"ks":{"o":{"a":0,"k":80,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[144.5,86.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":43.334,"s":[0,0]},{"t":59,"s":[100,100]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":49.166,"s":[20]},{"t":59,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":42,"op":60,"st":25,"ct":1,"bm":0},{"ddd":0,"ind":198,"ty":4,"nm":"circles_water 11","parent":196,"sr":1,"ks":{"o":{"a":0,"k":80,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[127.5,98.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":39.834,"s":[0,0]},{"t":55,"s":[100,100]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":45.666,"s":[20]},{"t":55,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":39,"op":57,"st":22,"ct":1,"bm":0},{"ddd":0,"ind":199,"ty":4,"nm":"circles_water 10","parent":196,"sr":1,"ks":{"o":{"a":0,"k":80,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[95.5,118.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":36.334,"s":[0,0]},{"t":57,"s":[100,100]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":42.166,"s":[20]},{"t":57,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":36,"op":58,"st":19,"ct":1,"bm":0},{"ddd":0,"ind":200,"ty":4,"nm":"circles_water 9","parent":196,"sr":1,"ks":{"o":{"a":0,"k":80,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[85.5,143.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":34,"s":[0,0]},{"t":50,"s":[100,100]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":39.834,"s":[20]},{"t":50,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":34,"op":52,"st":17,"ct":1,"bm":0},{"ddd":0,"ind":201,"ty":3,"nm":"NULL CONTROL 5","sr":1,"ks":{"o":{"a":0,"k":0,"ix":11},"r":{"a":0,"k":-19,"ix":10},"p":{"a":0,"k":[310.614,163.154,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[83.285,83.285,100],"ix":6,"l":2}},"ao":0,"ip":28,"op":47,"st":-14,"bm":0},{"ddd":0,"ind":202,"ty":4,"nm":"circles_water 8","parent":201,"sr":1,"ks":{"o":{"a":0,"k":70,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[144.5,86.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":36,"s":[0,0]},{"t":50,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":41,"s":[20]},{"t":50,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":36,"op":52,"st":19,"ct":1,"bm":0},{"ddd":0,"ind":203,"ty":4,"nm":"circles_water 7","parent":201,"sr":1,"ks":{"o":{"a":0,"k":70,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[127.5,98.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":33,"s":[0,0]},{"t":47,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":38,"s":[20]},{"t":47,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":33,"op":49,"st":16,"ct":1,"bm":0},{"ddd":0,"ind":204,"ty":4,"nm":"circles_water 6","parent":201,"sr":1,"ks":{"o":{"a":0,"k":70,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[95.5,118.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":30,"s":[0,0]},{"t":48,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":35,"s":[20]},{"t":48,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":30,"op":50,"st":13,"ct":1,"bm":0},{"ddd":0,"ind":205,"ty":4,"nm":"circles_water 5","parent":201,"sr":1,"ks":{"o":{"a":0,"k":70,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[85.5,143.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":28,"s":[0,0]},{"t":42,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":33,"s":[20]},{"t":42,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":28,"op":44,"st":11,"ct":1,"bm":0},{"ddd":0,"ind":206,"ty":3,"nm":"NULL CONTROL 3","sr":1,"ks":{"o":{"a":0,"k":0,"ix":11},"r":{"a":0,"k":112,"ix":10},"p":{"a":0,"k":[201.906,230.732,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[68.667,68.667,100],"ix":6,"l":2}},"ao":0,"ip":21,"op":38,"st":-51,"bm":0},{"ddd":0,"ind":207,"ty":4,"nm":"Shape Layer 70","parent":206,"sr":1,"ks":{"o":{"a":0,"k":90,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[130,145,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":27,"s":[0,0]},{"t":37,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":32,"s":[20]},{"t":37,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":27,"op":38,"st":10,"ct":1,"bm":0},{"ddd":0,"ind":208,"ty":4,"nm":"Shape Layer 69","parent":206,"sr":1,"ks":{"o":{"a":0,"k":90,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[118,121,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":24,"s":[0,0]},{"t":34,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":29,"s":[20]},{"t":34,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":24,"op":35,"st":7,"ct":1,"bm":0},{"ddd":0,"ind":209,"ty":4,"nm":"Shape Layer 68","parent":206,"sr":1,"ks":{"o":{"a":0,"k":90,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[100,85,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":21,"s":[0,0]},{"t":31,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":26,"s":[20]},{"t":31,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":21,"op":32,"st":4,"ct":1,"bm":0},{"ddd":0,"ind":210,"ty":3,"nm":"NULL CONTROL 2","sr":1,"ks":{"o":{"a":0,"k":0,"ix":11},"r":{"a":0,"k":-38,"ix":10},"p":{"a":0,"k":[255.736,330.597,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[62.687,62.687,100],"ix":6,"l":2}},"ao":0,"ip":5,"op":28,"st":-48,"bm":0},{"ddd":0,"ind":211,"ty":4,"nm":"Shape Layer 67","parent":210,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[116.5,159.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":13,"s":[0,0]},{"t":27,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":18,"s":[18]},{"t":27,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":13,"op":28,"st":-4,"ct":1,"bm":0},{"ddd":0,"ind":212,"ty":4,"nm":"Shape Layer 66","parent":210,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[104.5,102.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":8,"s":[0,0]},{"t":22,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[18]},{"t":22,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":8,"op":23,"st":-9,"ct":1,"bm":0},{"ddd":0,"ind":213,"ty":4,"nm":"Shape Layer 65","parent":210,"sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[125.5,70.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":5,"s":[0,0]},{"t":19,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":10,"s":[18]},{"t":19,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":5,"op":20,"st":-12,"ct":1,"bm":0},{"ddd":0,"ind":214,"ty":3,"nm":"NULL CONTROL","sr":1,"ks":{"o":{"a":0,"k":0,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[300.927,201.64,0],"ix":2,"l":2},"a":{"a":0,"k":[115,115,0],"ix":1,"l":2},"s":{"a":0,"k":[55.198,55.198,100],"ix":6,"l":2}},"ao":0,"ip":1,"op":20,"st":-41,"bm":0},{"ddd":0,"ind":215,"ty":4,"nm":"circles_water 4","parent":214,"sr":1,"ks":{"o":{"a":0,"k":70,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[144.5,86.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":9,"s":[0,0]},{"t":19,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[20]},{"t":19,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":9,"op":20,"st":-8,"ct":1,"bm":0},{"ddd":0,"ind":216,"ty":4,"nm":"circles_water 3","parent":214,"sr":1,"ks":{"o":{"a":0,"k":70,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[127.5,98.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":6,"s":[0,0]},{"t":16,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":11,"s":[20]},{"t":16,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":6,"op":17,"st":-11,"ct":1,"bm":0},{"ddd":0,"ind":217,"ty":4,"nm":"circles_water 2","parent":214,"sr":1,"ks":{"o":{"a":0,"k":70,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[95.5,118.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":3,"s":[0,0]},{"t":17,"s":[82,82]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":6,"s":[20]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":8,"s":[20]},{"t":17,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":3,"op":18,"st":-14,"ct":1,"bm":0},{"ddd":0,"ind":218,"ty":4,"nm":"circles_water","parent":214,"sr":1,"ks":{"o":{"a":0,"k":70,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[85.5,143.5,0],"ix":2,"l":2},"a":{"a":0,"k":[142.291,-161.709,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"t":1,"s":[0,0]},{"t":11,"s":[51,51]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":6,"s":[20]},{"t":11,"s":[0]}],"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[142.291,-161.709],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":1,"op":12,"st":-16,"ct":1,"bm":0}],"markers":[{"tm":-2,"cm":"1","dr":0},{"tm":88,"cm":"2","dr":0}]} \ No newline at end of file diff --git a/TMessagesProj/src/main/res/raw/speaker_to_bt.json b/TMessagesProj/src/main/res/raw/speaker_to_bt.json new file mode 100644 index 00000000000..90a6ba022b4 --- /dev/null +++ b/TMessagesProj/src/main/res/raw/speaker_to_bt.json @@ -0,0 +1 @@ +{"v":"5.10.1","fr":60,"ip":0,"op":60,"w":156,"h":156,"nm":"speaker_to_bt","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":3,"nm":"Top 2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[1.004]},"o":{"x":[0.167],"y":[0]},"t":0,"s":[0]},{"i":{"x":[0.833],"y":[0.689]},"o":{"x":[0.167],"y":[0.083]},"t":1,"s":[0]},{"i":{"x":[0.833],"y":[0.638]},"o":{"x":[0.167],"y":[0.114]},"t":2,"s":[-0.042]},{"i":{"x":[0.833],"y":[0.808]},"o":{"x":[0.167],"y":[0.108]},"t":3,"s":[-0.157]},{"i":{"x":[0.833],"y":[0.829]},"o":{"x":[0.167],"y":[0.148]},"t":4,"s":[-0.54]},{"i":{"x":[0.833],"y":[0.876]},"o":{"x":[0.167],"y":[0.162]},"t":5,"s":[-1.038]},{"i":{"x":[0.833],"y":[0.802]},"o":{"x":[0.167],"y":[0.256]},"t":6,"s":[-1.564]},{"i":{"x":[0.833],"y":[0.942]},"o":{"x":[0.167],"y":[0.144]},"t":7,"s":[-1.817]},{"i":{"x":[0.833],"y":[0.368]},"o":{"x":[0.167],"y":[-0.195]},"t":8,"s":[-2.167]},{"i":{"x":[0.833],"y":[0.785]},"o":{"x":[0.167],"y":[0.096]},"t":9,"s":[-2.062]},{"i":{"x":[0.833],"y":[0.869]},"o":{"x":[0.167],"y":[0.136]},"t":10,"s":[-1.374]},{"i":{"x":[0.833],"y":[0.746]},"o":{"x":[0.167],"y":[0.23]},"t":11,"s":[-0.289]},{"i":{"x":[0.833],"y":[0.839]},"o":{"x":[0.167],"y":[0.124]},"t":12,"s":[0.327]},{"i":{"x":[0.833],"y":[0.879]},"o":{"x":[0.167],"y":[0.172]},"t":13,"s":[1.583]},{"i":{"x":[0.833],"y":[0.774]},"o":{"x":[0.167],"y":[0.269]},"t":14,"s":[2.759]},{"i":{"x":[0.833],"y":[0.854]},"o":{"x":[0.167],"y":[0.132]},"t":15,"s":[3.288]},{"i":{"x":[0.833],"y":[0.885]},"o":{"x":[0.167],"y":[0.193]},"t":16,"s":[4.193]},{"i":{"x":[0.833],"y":[0.834]},"o":{"x":[0.167],"y":[0.302]},"t":17,"s":[4.879]},{"i":{"x":[0.833],"y":[1.096]},"o":{"x":[0.167],"y":[0.167]},"t":18,"s":[5.14]},{"i":{"x":[0.833],"y":[0.84]},"o":{"x":[0.167],"y":[0.045]},"t":19,"s":[5.401]},{"i":{"x":[0.833],"y":[0.706]},"o":{"x":[0.167],"y":[0.173]},"t":20,"s":[4.839]},{"i":{"x":[0.833],"y":[0.824]},"o":{"x":[0.167],"y":[0.116]},"t":21,"s":[4.319]},{"i":{"x":[0.833],"y":[0.875]},"o":{"x":[0.167],"y":[0.158]},"t":22,"s":[3.002]},{"i":{"x":[0.833],"y":[0.762]},"o":{"x":[0.167],"y":[0.251]},"t":23,"s":[1.531]},{"i":{"x":[0.833],"y":[0.846]},"o":{"x":[0.167],"y":[0.128]},"t":24,"s":[0.802]},{"i":{"x":[0.833],"y":[0.882]},"o":{"x":[0.167],"y":[0.182]},"t":25,"s":[-0.555]},{"i":{"x":[0.833],"y":[0.791]},"o":{"x":[0.167],"y":[0.282]},"t":26,"s":[-1.705]},{"i":{"x":[0.833],"y":[0.89]},"o":{"x":[0.167],"y":[0.139]},"t":27,"s":[-2.186]},{"i":{"x":[0.833],"y":[0.93]},"o":{"x":[0.167],"y":[0.342]},"t":28,"s":[-2.912]},{"i":{"x":[0.833],"y":[0.264]},"o":{"x":[0.167],"y":[-0.443]},"t":29,"s":[-3.145]},{"i":{"x":[0.833],"y":[0.781]},"o":{"x":[0.167],"y":[0.094]},"t":30,"s":[-3.108]},{"i":{"x":[0.833],"y":[0.869]},"o":{"x":[0.167],"y":[0.135]},"t":31,"s":[-2.818]},{"i":{"x":[0.833],"y":[0.746]},"o":{"x":[0.167],"y":[0.229]},"t":32,"s":[-2.348]},{"i":{"x":[0.833],"y":[0.838]},"o":{"x":[0.167],"y":[0.124]},"t":33,"s":[-2.08]},{"i":{"x":[0.833],"y":[0.879]},"o":{"x":[0.167],"y":[0.172]},"t":34,"s":[-1.528]},{"i":{"x":[0.833],"y":[0.774]},"o":{"x":[0.167],"y":[0.268]},"t":35,"s":[-1.009]},{"i":{"x":[0.833],"y":[0.853]},"o":{"x":[0.167],"y":[0.132]},"t":36,"s":[-0.775]},{"i":{"x":[0.833],"y":[0.885]},"o":{"x":[0.167],"y":[0.193]},"t":37,"s":[-0.372]},{"i":{"x":[0.833],"y":[0.797]},"o":{"x":[0.167],"y":[0.301]},"t":38,"s":[-0.066]},{"i":{"x":[0.833],"y":[0.871]},"o":{"x":[0.167],"y":[0.141]},"t":39,"s":[0.051]},{"i":{"x":[0.833],"y":[0.897]},"o":{"x":[0.167],"y":[0.237]},"t":40,"s":[0.219]},{"i":{"x":[0.833],"y":[0.89]},"o":{"x":[0.167],"y":[0.431]},"t":41,"s":[0.31]},{"i":{"x":[0.833],"y":[1.258]},"o":{"x":[0.167],"y":[0.345]},"t":42,"s":[0.332]},{"i":{"x":[0.833],"y":[0.85]},"o":{"x":[0.167],"y":[0.063]},"t":43,"s":[0.339]},{"i":{"x":[0.833],"y":[0.715]},"o":{"x":[0.167],"y":[0.187]},"t":44,"s":[0.31]},{"i":{"x":[0.833],"y":[0.827]},"o":{"x":[0.167],"y":[0.118]},"t":45,"s":[0.287]},{"i":{"x":[0.833],"y":[0.876]},"o":{"x":[0.167],"y":[0.16]},"t":46,"s":[0.232]},{"i":{"x":[0.833],"y":[0.764]},"o":{"x":[0.167],"y":[0.254]},"t":47,"s":[0.172]},{"i":{"x":[0.833],"y":[0.847]},"o":{"x":[0.167],"y":[0.129]},"t":48,"s":[0.143]},{"i":{"x":[0.833],"y":[0.882]},"o":{"x":[0.167],"y":[0.183]},"t":49,"s":[0.089]},{"i":{"x":[0.833],"y":[0.786]},"o":{"x":[0.167],"y":[0.285]},"t":50,"s":[0.045]},{"i":{"x":[0.833],"y":[0.862]},"o":{"x":[0.167],"y":[0.136]},"t":51,"s":[0.026]},{"i":{"x":[0.833],"y":[0.889]},"o":{"x":[0.167],"y":[0.21]},"t":52,"s":[-0.003]},{"i":{"x":[0.833],"y":[0.823]},"o":{"x":[0.167],"y":[0.337]},"t":53,"s":[-0.022]},{"i":{"x":[0.833],"y":[0.902]},"o":{"x":[0.167],"y":[0.157]},"t":54,"s":[-0.029]},{"i":{"x":[0.833],"y":[0.981]},"o":{"x":[0.167],"y":[0.549]},"t":55,"s":[-0.036]},{"i":{"x":[0.833],"y":[0.564]},"o":{"x":[0.167],"y":[-0.024]},"t":56,"s":[-0.037]},{"i":{"x":[0.833],"y":[0.799]},"o":{"x":[0.167],"y":[0.103]},"t":57,"s":[-0.036]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.142]},"t":58,"s":[-0.032]},{"t":59,"s":[-0.026]}],"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.713},"o":{"x":0.167,"y":0.167},"t":0,"s":[77.722,79.526,0],"to":[0,-0.053,0],"ti":[0,0.183,0]},{"i":{"x":0.833,"y":0.864},"o":{"x":0.167,"y":0.117},"t":1,"s":[77.722,79.208,0],"to":[0,-0.183,0],"ti":[0,0.211,0]},{"i":{"x":0.833,"y":0.736},"o":{"x":0.167,"y":0.216},"t":2,"s":[77.722,78.429,0],"to":[0,-0.211,0],"ti":[0,0.258,0]},{"i":{"x":0.833,"y":0.834},"o":{"x":0.167,"y":0.122},"t":3,"s":[77.722,77.94,0],"to":[0,-0.258,0],"ti":[0,0.351,0]},{"i":{"x":0.833,"y":0.856},"o":{"x":0.167,"y":0.168},"t":4,"s":[77.722,76.881,0],"to":[0,-0.351,0],"ti":[0,0.302,0]},{"i":{"x":0.833,"y":0.894},"o":{"x":0.167,"y":0.197},"t":5,"s":[77.722,75.834,0],"to":[0,-0.302,0],"ti":[0,0.118,0]},{"i":{"x":0.833,"y":0.567},"o":{"x":0.167,"y":0.388},"t":6,"s":[77.722,75.068,0],"to":[0,-0.118,0],"ti":[0,-0.156,0]},{"i":{"x":0.833,"y":0.849},"o":{"x":0.167,"y":0.103},"t":7,"s":[77.722,75.125,0],"to":[0,0.156,0],"ti":[0,-0.265,0]},{"i":{"x":0.833,"y":0.715},"o":{"x":0.167,"y":0.187},"t":8,"s":[77.722,76.004,0],"to":[0,0.265,0],"ti":[0,-0.404,0]},{"i":{"x":0.833,"y":0.827},"o":{"x":0.167,"y":0.118},"t":9,"s":[77.722,76.712,0],"to":[0,0.404,0],"ti":[0,-0.595,0]},{"i":{"x":0.833,"y":0.876},"o":{"x":0.167,"y":0.16},"t":10,"s":[77.722,78.426,0],"to":[0,0.595,0],"ti":[0,-0.46,0]},{"i":{"x":0.833,"y":0.763},"o":{"x":0.167,"y":0.254},"t":11,"s":[77.722,80.28,0],"to":[0,0.46,0],"ti":[0,-0.428,0]},{"i":{"x":0.833,"y":0.847},"o":{"x":0.167,"y":0.129},"t":12,"s":[77.722,81.184,0],"to":[0,0.428,0],"ti":[0,-0.508,0]},{"i":{"x":0.833,"y":0.882},"o":{"x":0.167,"y":0.183},"t":13,"s":[77.722,82.846,0],"to":[0,0.508,0],"ti":[0,-0.327,0]},{"i":{"x":0.833,"y":0.785},"o":{"x":0.167,"y":0.285},"t":14,"s":[77.722,84.232,0],"to":[0,0.327,0],"ti":[0,-0.246,0]},{"i":{"x":0.833,"y":0.875},"o":{"x":0.167,"y":0.136},"t":15,"s":[77.722,84.806,0],"to":[0,0.246,0],"ti":[0,-0.226,0]},{"i":{"x":0.833,"y":0.873},"o":{"x":0.167,"y":0.25},"t":16,"s":[77.722,85.708,0],"to":[0,0.226,0],"ti":[0,-0.042,0]},{"i":{"x":0.833,"y":0.524},"o":{"x":0.167,"y":0.244},"t":17,"s":[77.722,86.16,0],"to":[0,0.042,0],"ti":[0,0.217,0]},{"i":{"x":0.833,"y":0.79},"o":{"x":0.167,"y":0.101},"t":18,"s":[77.722,85.96,0],"to":[0,-0.217,0],"ti":[0,0.464,0]},{"i":{"x":0.833,"y":0.87},"o":{"x":0.167,"y":0.138},"t":19,"s":[77.722,84.857,0],"to":[0,-0.464,0],"ti":[0,0.437,0]},{"i":{"x":0.833,"y":0.748},"o":{"x":0.167,"y":0.232},"t":20,"s":[77.722,83.177,0],"to":[0,-0.437,0],"ti":[0,0.475,0]},{"i":{"x":0.833,"y":0.839},"o":{"x":0.167,"y":0.124},"t":21,"s":[77.722,82.236,0],"to":[0,-0.475,0],"ti":[0,0.614,0]},{"i":{"x":0.833,"y":0.879},"o":{"x":0.167,"y":0.173},"t":22,"s":[77.722,80.328,0],"to":[0,-0.614,0],"ti":[0,0.428,0]},{"i":{"x":0.833,"y":0.775},"o":{"x":0.167,"y":0.27},"t":23,"s":[77.722,78.552,0],"to":[0,-0.428,0],"ti":[0,0.358,0]},{"i":{"x":0.833,"y":0.858},"o":{"x":0.167,"y":0.132},"t":24,"s":[77.722,77.757,0],"to":[0,-0.358,0],"ti":[0,0.386,0]},{"i":{"x":0.833,"y":0.897},"o":{"x":0.167,"y":0.201},"t":25,"s":[77.722,76.403,0],"to":[0,-0.386,0],"ti":[0,0.197,0]},{"i":{"x":0.833,"y":0.886},"o":{"x":0.167,"y":0.441},"t":26,"s":[77.722,75.443,0],"to":[0,-0.197,0],"ti":[0,0.038,0]},{"i":{"x":0.833,"y":0.495},"o":{"x":0.167,"y":0.307},"t":27,"s":[77.722,75.22,0],"to":[0,-0.038,0],"ti":[0,-0.07,0]},{"i":{"x":0.833,"y":0.855},"o":{"x":0.167,"y":0.1},"t":28,"s":[77.722,75.218,0],"to":[0,0.07,0],"ti":[0,-0.122,0]},{"i":{"x":0.833,"y":0.722},"o":{"x":0.167,"y":0.196},"t":29,"s":[77.722,75.639,0],"to":[0,0.122,0],"ti":[0,-0.173,0]},{"i":{"x":0.833,"y":0.829},"o":{"x":0.167,"y":0.119},"t":30,"s":[77.722,75.95,0],"to":[0,0.173,0],"ti":[0,-0.249,0]},{"i":{"x":0.833,"y":0.877},"o":{"x":0.167,"y":0.162},"t":31,"s":[77.722,76.677,0],"to":[0,0.249,0],"ti":[0,-0.189,0]},{"i":{"x":0.833,"y":0.765},"o":{"x":0.167,"y":0.256},"t":32,"s":[77.722,77.443,0],"to":[0,0.189,0],"ti":[0,-0.173,0]},{"i":{"x":0.833,"y":0.848},"o":{"x":0.167,"y":0.129},"t":33,"s":[77.722,77.812,0],"to":[0,0.173,0],"ti":[0,-0.204,0]},{"i":{"x":0.833,"y":0.883},"o":{"x":0.167,"y":0.185},"t":34,"s":[77.722,78.483,0],"to":[0,0.204,0],"ti":[0,-0.13,0]},{"i":{"x":0.833,"y":0.787},"o":{"x":0.167,"y":0.287},"t":35,"s":[77.722,79.035,0],"to":[0,0.13,0],"ti":[0,-0.096,0]},{"i":{"x":0.833,"y":0.863},"o":{"x":0.167,"y":0.137},"t":36,"s":[77.722,79.261,0],"to":[0,0.096,0],"ti":[0,-0.096,0]},{"i":{"x":0.833,"y":0.89},"o":{"x":0.167,"y":0.212},"t":37,"s":[77.722,79.612,0],"to":[0,0.096,0],"ti":[0,-0.05,0]},{"i":{"x":0.833,"y":0.828},"o":{"x":0.167,"y":0.344},"t":38,"s":[77.722,79.839,0],"to":[0,0.05,0],"ti":[0,-0.025,0]},{"i":{"x":0.833,"y":0.904},"o":{"x":0.167,"y":0.162},"t":39,"s":[77.722,79.912,0],"to":[0,0.025,0],"ti":[0,-0.014,0]},{"i":{"x":0.833,"y":0.807},"o":{"x":0.167,"y":0.617},"t":40,"s":[77.722,79.989,0],"to":[0,0.014,0],"ti":[0,0.002,0]},{"i":{"x":0.833,"y":0.617},"o":{"x":0.167,"y":0.143},"t":41,"s":[77.722,79.996,0],"to":[0,-0.002,0],"ti":[0,0.012,0]},{"i":{"x":0.833,"y":0.805},"o":{"x":0.167,"y":0.107},"t":42,"s":[77.722,79.979,0],"to":[0,-0.012,0],"ti":[0,0.023,0]},{"i":{"x":0.833,"y":0.872},"o":{"x":0.167,"y":0.146},"t":43,"s":[77.722,79.921,0],"to":[0,-0.023,0],"ti":[0,0.02,0]},{"i":{"x":0.833,"y":0.753},"o":{"x":0.167,"y":0.239},"t":44,"s":[77.722,79.844,0],"to":[0,-0.02,0],"ti":[0,0.021,0]},{"i":{"x":0.833,"y":0.841},"o":{"x":0.167,"y":0.126},"t":45,"s":[77.722,79.802,0],"to":[0,-0.021,0],"ti":[0,0.026,0]},{"i":{"x":0.833,"y":0.88},"o":{"x":0.167,"y":0.176},"t":46,"s":[77.722,79.72,0],"to":[0,-0.026,0],"ti":[0,0.018,0]},{"i":{"x":0.833,"y":0.777},"o":{"x":0.167,"y":0.273},"t":47,"s":[77.722,79.646,0],"to":[0,-0.018,0],"ti":[0,0.014,0]},{"i":{"x":0.833,"y":0.856},"o":{"x":0.167,"y":0.133},"t":48,"s":[77.722,79.614,0],"to":[0,-0.014,0],"ti":[0,0.016,0]},{"i":{"x":0.833,"y":0.886},"o":{"x":0.167,"y":0.197},"t":49,"s":[77.722,79.56,0],"to":[0,-0.016,0],"ti":[0,0.009,0]},{"i":{"x":0.833,"y":0.803},"o":{"x":0.167,"y":0.31},"t":50,"s":[77.722,79.52,0],"to":[0,-0.009,0],"ti":[0,0.006,0]},{"i":{"x":0.833,"y":0.877},"o":{"x":0.167,"y":0.145},"t":51,"s":[77.722,79.505,0],"to":[0,-0.006,0],"ti":[0,0.005,0]},{"i":{"x":0.833,"y":0.903},"o":{"x":0.167,"y":0.26},"t":52,"s":[77.722,79.485,0],"to":[0,-0.005,0],"ti":[0,0.002,0]},{"i":{"x":0.833,"y":0.835},"o":{"x":0.167,"y":0.56},"t":53,"s":[77.722,79.476,0],"to":[0,-0.002,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.613},"o":{"x":0.167,"y":0.164},"t":54,"s":[77.722,79.475,0],"to":[0,0,0],"ti":[0,-0.001,0]},{"i":{"x":0.833,"y":0.861},"o":{"x":0.167,"y":0.106},"t":55,"s":[77.722,79.476,0],"to":[0,0.001,0],"ti":[0,-0.002,0]},{"i":{"x":0.833,"y":0.731},"o":{"x":0.167,"y":0.209},"t":56,"s":[77.722,79.482,0],"to":[0,0.002,0],"ti":[0,-0.002,0]},{"i":{"x":0.833,"y":0.832},"o":{"x":0.167,"y":0.121},"t":57,"s":[77.722,79.486,0],"to":[0,0.002,0],"ti":[0,-0.003,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.166},"t":58,"s":[77.722,79.495,0],"to":[0,0.003,0],"ti":[0,-0.001,0]},{"t":59,"s":[77.722,79.504,0]}],"ix":2,"l":2},"a":{"a":0,"k":[-0.918,5,0],"ix":1,"l":2},"s":{"a":0,"k":[30.469,30.469,100],"ix":6,"l":2}},"ao":0,"ip":0,"op":60,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"Bluetooth Outlines","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[-0.006,-0.008,0],"ix":2,"l":2},"a":{"a":0,"k":[78,78,0],"ix":1,"l":2},"s":{"a":1,"k":[{"i":{"x":[0.2,0.2,0.2],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0]},"t":0,"s":[0,0,100]},{"t":20,"s":[328.205,328.205,100]}],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-17.75,20.5],[17.75,-15.5],[2.25,-31.5],[2.25,31.5],[17.75,15.5],[-17.75,-20.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.2],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":10,"s":[50]},{"t":30,"s":[0]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.2],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":10,"s":[50]},{"t":30,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":7,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[69.25,78],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"wave2","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[-0.006,-0.008,0],"ix":2,"l":2},"a":{"a":0,"k":[78,78,0],"ix":1,"l":2},"s":{"a":0,"k":[328.205,328.205,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.2,"y":1},"o":{"x":0.167,"y":0.167},"t":5,"s":[{"i":[[0,0],[0,-12.307],[12.759,-6.613]],"o":[[12.759,6.613],[0,12.307],[0,0]],"v":[[-19.177,-28.707],[-0.039,-0.329],[-19.177,28.05]],"c":false}]},{"t":25,"s":[{"i":[[0,0],[0,-6.48],[4.153,-6.175]],"o":[[4.153,5.794],[0,6.48],[0,0]],"v":[[-3.115,-18.698],[3.115,-0.286],[-3.115,18.698]],"c":false}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":7.5,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[108.267,78.286],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":4,"ty":4,"nm":"wave1","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[-0.006,-0.008,0],"ix":2,"l":2},"a":{"a":0,"k":[78,78,0],"ix":1,"l":2},"s":{"a":0,"k":[328.205,328.205,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.2,"y":1},"o":{"x":0.167,"y":0.167},"t":5,"s":[{"i":[[0.5,0],[0,0],[0,0],[0,0],[-2.401,2.725],[0,5.188],[2.401,2.725]],"o":[[0,0],[0,0],[0,0],[0.5,0],[2.401,-2.725],[0,-5.188],[-2.401,-2.725]],"v":[[7,-16.704],[7,-8.362],[7,3.554],[7,15.471],[12.427,11.318],[17.103,-0.617],[12.427,-12.552]],"c":true}]},{"t":25,"s":[{"i":[[1.297,-1.308],[0,0],[-1.076,-1.085],[0,0],[-0.856,1.669],[-0.026,1.81],[0.896,1.763]],"o":[[0,0],[-1.076,1.085],[0,0],[1.297,1.308],[0.843,-1.646],[0.027,-1.924],[-0.856,-1.586]],"v":[[16.736,-6.202],[12.679,-2.112],[12.679,1.811],[16.736,5.901],[21.124,5.261],[22.426,0.002],[21.124,-5.617]],"c":true}]}],"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"mm","mm":1,"nm":"Merge Paths 1","mn":"ADBE Vector Filter - Merge","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[78,78],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":5,"ty":4,"nm":"Speaker Base","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[-0.006,-0.008,0],"ix":2,"l":2},"a":{"a":0,"k":[78,78,0],"ix":1,"l":2},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,1]},"o":{"x":[0.6,0.6,0.6],"y":[0,0,0]},"t":0,"s":[328.205,328.205,100]},{"t":15,"s":[0,0,100]}],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,-2.198],[0,0],[-2.213,0],[0,0],[0,0],[0,3.557],[0,0],[2.534,-2.518],[0,0],[0,0]],"o":[[0,0],[0,2.198],[0,0],[0,0],[2.534,2.518],[0,0],[0,-3.557],[0,0],[0,0],[-2.213,0]],"v":[[-34,-8.59],[-34,7.397],[-29.977,11.394],[-19.908,11.394],[-6.672,24.544],[-1,22.914],[-1,-24.147],[-6.672,-25.777],[-19.908,-12.587],[-29.977,-12.587]],"c":true},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"mm","mm":1,"nm":"Merge Paths 1","mn":"ADBE Vector Filter - Merge","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[78,78],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 3","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60,"st":0,"ct":1,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/TMessagesProj/src/main/res/raw/spoiler_compute.glsl b/TMessagesProj/src/main/res/raw/spoiler_compute.glsl deleted file mode 100644 index 12649abb0f4..00000000000 --- a/TMessagesProj/src/main/res/raw/spoiler_compute.glsl +++ /dev/null @@ -1,39 +0,0 @@ -#version 310 es - -layout(local_size_x = 256) in; // This can be adjusted - -struct Particle { - vec2 position; - vec2 velocity; - float time; - float duration; -}; - -layout(std430, binding = 0) buffer ParticleBuffer { - Particle particles[]; -}; - -uniform float deltaTime; -uniform vec2 size; - -void main() { - uint id = gl_GlobalInvocationID.x; - - if (id >= uint(particles.length())) - return; - - Particle p = particles[id]; - - p.time += deltaTime * p.duration; - if (p.time >= 1.0) { - p.time = 0.0; - } - - p.velocity += vec2(.5, .5); - p.velocity *= .99; - - p.position += p.velocity; - p.position = fract(p.position / size) * size; - - particles[id] = p; -} diff --git a/TMessagesProj/src/main/res/raw/star_fill.json b/TMessagesProj/src/main/res/raw/star_fill.json new file mode 100644 index 00000000000..cbaa4c8bf9c --- /dev/null +++ b/TMessagesProj/src/main/res/raw/star_fill.json @@ -0,0 +1 @@ +{"v":"5.10.0","fr":60,"ip":0,"op":180,"w":96,"h":96,"nm":"star","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":4,"nm":"ыефкк Outlines 2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[48,48,0],"ix":2,"l":2},"a":{"a":0,"k":[43,43,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"hasMask":true,"masksProperties":[{"inv":false,"mode":"a","pt":{"a":0,"k":{"i":[[0.855,-0.588],[0,0],[1.568,2.267],[-0.432,1.447],[0,0],[0.825,0.629],[0,0],[-1.678,2.187],[-1.514,0.037],[0,0],[-0.345,0.976],[0,0],[-2.605,-0.915],[-0.503,-1.424],[0,0],[-1.038,-0.025],[0,0],[0.068,-2.752],[1.203,-0.917],[0,0],[-0.296,-0.992],[0,0],[2.647,-0.786],[1.247,0.857],[0,0]],"o":[[0,0],[-2.274,1.563],[-0.86,-1.243],[0,0],[0.296,-0.992],[0,0],[-2.194,-1.673],[0.92,-1.199],[0,0],[1.038,-0.025],[0,0],[0.918,-2.597],[1.429,0.502],[0,0],[0.345,0.976],[0,0],[2.762,0.068],[-0.037,1.509],[0,0],[-0.825,0.629],[0,0],[0.789,2.639],[-1.452,0.431],[0,0],[-0.855,-0.588]],"v":[[41.58,65.44],[28.227,74.618],[21.27,73.344],[20.594,69.091],[25.225,53.597],[24.348,50.905],[11.463,41.083],[10.529,34.095],[14.379,32.14],[30.595,31.742],[32.892,30.078],[38.283,14.83],[44.662,11.785],[47.718,14.83],[53.108,30.078],[55.405,31.742],[71.621,32.14],[76.498,37.246],[74.537,41.083],[61.652,50.905],[60.775,53.597],[65.406,69.091],[62.041,75.292],[57.773,74.618],[44.42,65.44]],"c":true},"ix":1},"o":{"a":0,"k":100,"ix":3},"x":{"a":0,"k":0,"ix":4},"nm":"Mask 1"}],"shapes":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0.855,-0.588],[0,0],[1.568,2.267],[-0.432,1.447],[0,0],[0.825,0.629],[0,0],[-1.678,2.187],[-1.514,0.037],[0,0],[-0.345,0.976],[0,0],[-2.605,-0.915],[-0.503,-1.424],[0,0],[-1.038,-0.025],[0,0],[0.068,-2.752],[1.203,-0.917],[0,0],[-0.296,-0.992],[0,0],[2.647,-0.786],[1.247,0.857],[0,0]],"o":[[0,0],[-2.274,1.563],[-0.86,-1.243],[0,0],[0.296,-0.992],[0,0],[-2.194,-1.673],[0.92,-1.199],[0,0],[1.038,-0.025],[0,0],[0.918,-2.597],[1.429,0.502],[0,0],[0.345,0.976],[0,0],[2.762,0.068],[-0.037,1.509],[0,0],[-0.825,0.629],[0,0],[0.789,2.639],[-1.452,0.431],[0,0],[-0.855,-0.588]],"v":[[41.08,64.94],[27.727,74.118],[20.77,72.844],[20.094,68.591],[24.725,53.097],[23.848,50.405],[10.963,40.583],[10.029,33.595],[13.879,31.64],[30.095,31.242],[32.392,29.578],[37.783,14.33],[44.162,11.285],[47.218,14.33],[52.608,29.578],[54.905,31.242],[71.121,31.64],[75.998,36.746],[74.037,40.583],[61.152,50.405],[60.275,53.097],[64.906,68.591],[61.541,74.792],[57.273,74.118],[43.92,64.94]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":37,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false}],"ip":0,"op":180,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"ыефкк Outlines","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[43,43,0],"ix":2,"l":2},"a":{"a":0,"k":[43,43,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0.855,-0.588],[0,0],[1.568,2.267],[-0.432,1.447],[0,0],[0.825,0.629],[0,0],[-1.678,2.187],[-1.514,0.037],[0,0],[-0.345,0.976],[0,0],[-2.605,-0.915],[-0.503,-1.424],[0,0],[-1.038,-0.025],[0,0],[0.068,-2.752],[1.203,-0.917],[0,0],[-0.296,-0.992],[0,0],[2.647,-0.786],[1.247,0.857],[0,0]],"o":[[0,0],[-2.274,1.563],[-0.86,-1.243],[0,0],[0.296,-0.992],[0,0],[-2.194,-1.673],[0.92,-1.199],[0,0],[1.038,-0.025],[0,0],[0.918,-2.597],[1.429,0.502],[0,0],[0.345,0.976],[0,0],[2.762,0.068],[-0.037,1.509],[0,0],[-0.825,0.629],[0,0],[0.789,2.639],[-1.452,0.431],[0,0],[-0.855,-0.588]],"v":[[-1.92,21.94],[-15.273,31.118],[-22.23,29.844],[-22.906,25.591],[-18.275,10.097],[-19.152,7.405],[-32.037,-2.417],[-32.971,-9.405],[-29.121,-11.36],[-12.905,-11.758],[-10.608,-13.422],[-5.217,-28.67],[1.162,-31.715],[4.218,-28.67],[9.608,-13.422],[11.905,-11.758],[28.121,-11.36],[32.998,-6.254],[31.037,-2.417],[18.152,7.405],[17.275,10.097],[21.906,25.591],[18.541,31.792],[14.273,31.118],[0.92,21.94]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":0,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[43,43],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":180,"st":0,"ct":1,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/TMessagesProj/src/main/res/raw/star_stroke.json b/TMessagesProj/src/main/res/raw/star_stroke.json new file mode 100644 index 00000000000..5a24482b409 --- /dev/null +++ b/TMessagesProj/src/main/res/raw/star_stroke.json @@ -0,0 +1 @@ +{"v":"5.10.0","fr":60,"ip":0,"op":180,"w":96,"h":96,"nm":"star","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":4,"nm":"ыефкк Outlines 2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[48,48,0],"ix":2,"l":2},"a":{"a":0,"k":[43,43,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"hasMask":true,"masksProperties":[{"inv":false,"mode":"a","pt":{"a":0,"k":{"i":[[0.855,-0.588],[0,0],[1.568,2.267],[-0.432,1.447],[0,0],[0.825,0.629],[0,0],[-1.678,2.187],[-1.514,0.037],[0,0],[-0.345,0.976],[0,0],[-2.605,-0.915],[-0.503,-1.424],[0,0],[-1.038,-0.025],[0,0],[0.068,-2.752],[1.203,-0.917],[0,0],[-0.296,-0.992],[0,0],[2.647,-0.786],[1.247,0.857],[0,0]],"o":[[0,0],[-2.274,1.563],[-0.86,-1.243],[0,0],[0.296,-0.992],[0,0],[-2.194,-1.673],[0.92,-1.199],[0,0],[1.038,-0.025],[0,0],[0.918,-2.597],[1.429,0.502],[0,0],[0.345,0.976],[0,0],[2.762,0.068],[-0.037,1.509],[0,0],[-0.825,0.629],[0,0],[0.789,2.639],[-1.452,0.431],[0,0],[-0.855,-0.588]],"v":[[41.58,65.44],[28.227,74.618],[21.27,73.344],[20.594,69.091],[25.225,53.597],[24.348,50.905],[11.463,41.083],[10.529,34.095],[14.379,32.14],[30.595,31.742],[32.892,30.078],[38.283,14.83],[44.662,11.785],[47.718,14.83],[53.108,30.078],[55.405,31.742],[71.621,32.14],[76.498,37.246],[74.537,41.083],[61.652,50.905],[60.775,53.597],[65.406,69.091],[62.041,75.292],[57.773,74.618],[44.42,65.44]],"c":true},"ix":1},"o":{"a":0,"k":100,"ix":3},"x":{"a":0,"k":0,"ix":4},"nm":"Mask 1"}],"shapes":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0.855,-0.588],[0,0],[1.568,2.267],[-0.432,1.447],[0,0],[0.825,0.629],[0,0],[-1.678,2.187],[-1.514,0.037],[0,0],[-0.345,0.976],[0,0],[-2.605,-0.915],[-0.503,-1.424],[0,0],[-1.038,-0.025],[0,0],[0.068,-2.752],[1.203,-0.917],[0,0],[-0.296,-0.992],[0,0],[2.647,-0.786],[1.247,0.857],[0,0]],"o":[[0,0],[-2.274,1.563],[-0.86,-1.243],[0,0],[0.296,-0.992],[0,0],[-2.194,-1.673],[0.92,-1.199],[0,0],[1.038,-0.025],[0,0],[0.918,-2.597],[1.429,0.502],[0,0],[0.345,0.976],[0,0],[2.762,0.068],[-0.037,1.509],[0,0],[-0.825,0.629],[0,0],[0.789,2.639],[-1.452,0.431],[0,0],[-0.855,-0.588]],"v":[[41.08,64.94],[27.727,74.118],[20.77,72.844],[20.094,68.591],[24.725,53.097],[23.848,50.405],[10.963,40.583],[10.029,33.595],[13.879,31.64],[30.095,31.242],[32.392,29.578],[37.783,14.33],[44.162,11.285],[47.218,14.33],[52.608,29.578],[54.905,31.242],[71.121,31.64],[75.998,36.746],[74.037,40.583],[61.152,50.405],[60.275,53.097],[64.906,68.591],[61.541,74.792],[57.273,74.118],[43.92,64.94]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":6,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false}],"ip":0,"op":180,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"ыефкк Outlines","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[43,43,0],"ix":2,"l":2},"a":{"a":0,"k":[43,43,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0.855,-0.588],[0,0],[1.568,2.267],[-0.432,1.447],[0,0],[0.825,0.629],[0,0],[-1.678,2.187],[-1.514,0.037],[0,0],[-0.345,0.976],[0,0],[-2.605,-0.915],[-0.503,-1.424],[0,0],[-1.038,-0.025],[0,0],[0.068,-2.752],[1.203,-0.917],[0,0],[-0.296,-0.992],[0,0],[2.647,-0.786],[1.247,0.857],[0,0]],"o":[[0,0],[-2.274,1.563],[-0.86,-1.243],[0,0],[0.296,-0.992],[0,0],[-2.194,-1.673],[0.92,-1.199],[0,0],[1.038,-0.025],[0,0],[0.918,-2.597],[1.429,0.502],[0,0],[0.345,0.976],[0,0],[2.762,0.068],[-0.037,1.509],[0,0],[-0.825,0.629],[0,0],[0.789,2.639],[-1.452,0.431],[0,0],[-0.855,-0.588]],"v":[[-1.92,21.94],[-15.273,31.118],[-22.23,29.844],[-22.906,25.591],[-18.275,10.097],[-19.152,7.405],[-32.037,-2.417],[-32.971,-9.405],[-29.121,-11.36],[-12.905,-11.758],[-10.608,-13.422],[-5.217,-28.67],[1.162,-31.715],[4.218,-28.67],[9.608,-13.422],[11.905,-11.758],[28.121,-11.36],[32.998,-6.254],[31.037,-2.417],[18.152,7.405],[17.275,10.097],[21.906,25.591],[18.541,31.792],[14.273,31.118],[0.92,21.94]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":6,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[43,43],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":180,"st":0,"ct":1,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/TMessagesProj/src/main/res/raw/thanos_fragment.glsl b/TMessagesProj/src/main/res/raw/thanos_fragment.glsl new file mode 100644 index 00000000000..0e9afc1d977 --- /dev/null +++ b/TMessagesProj/src/main/res/raw/thanos_fragment.glsl @@ -0,0 +1,18 @@ +#version 300 es + +precision highp float; + +in vec2 uvcenter; +in vec2 uvsize; +in float alpha; + +out vec4 fragColor; + +uniform sampler2D tex; + +void main() { + if (alpha <= 0.0) { + discard; + } + fragColor = texture(tex, uvcenter + gl_PointCoord * uvsize).rgba * alpha; +} \ No newline at end of file diff --git a/TMessagesProj/src/main/res/raw/thanos_vertex.glsl b/TMessagesProj/src/main/res/raw/thanos_vertex.glsl new file mode 100644 index 00000000000..b7822055a6a --- /dev/null +++ b/TMessagesProj/src/main/res/raw/thanos_vertex.glsl @@ -0,0 +1,145 @@ +#version 300 es + +precision highp float; + +layout(location = 0) in vec2 inUV; +layout(location = 1) in vec2 inPosition; +layout(location = 2) in vec2 inVelocity; +layout(location = 3) in float inTime; + +out vec2 outUV; +out vec2 outPosition; +out vec2 outVelocity; +out float outTime; + +out vec2 uvcenter; +out vec2 uvsize; +out float alpha; + +uniform mat3 matrix; +uniform vec2 rectSize; +// uniform vec2 rectPos; + +uniform float reset; +uniform float time; +uniform float deltaTime; +uniform float particlesCount; +uniform vec2 size; +uniform vec3 gridSize; +uniform float seed; +uniform float longevity; +uniform float dp; +uniform vec2 offset; + +#define noiseScale 12.0 +#define noiseSpeed 0.6 +#define noiseMovement 3.0 +#define snapDuration 0.6 +#define velocityMult 0.99 +#define forceMult 18.31 +#define dampingMult 0.95 + +float rand(vec2 n) { + return fract(sin(dot(n,vec2(12.9898,4.1414-seed*.42)))*43758.5453); +} +float mod289(float x){return x-floor(x*(1./(289.+seed)))*(289.+seed);} +vec4 mod289(vec4 x){return x-floor(x*(1./(289.+seed)))*(289.0+seed);} +vec4 perm(vec4 x){return mod289(((x*34.)+1.)*x);} +float noise(vec3 p){ + + vec3 a = floor(p); + vec3 d = p - a; + d = d * d * (3. - 2. * d); + + vec4 b = a.xxyy + vec4(0., 1., 0., 1.); + vec4 k1 = perm(b.xyxy); + vec4 k2 = perm(k1.xyxy + b.zzww); + + vec4 c = k2 + a.zzzz; + vec4 k3 = perm(c); + vec4 k4 = perm(c + 1.0); + + vec4 o3 = fract(k4 / 41.0) * d.z + fract(k3 / 41.0) * (1.0 - d.z); + vec2 o4 = o3.yw * d.x + o3.xz * (1.0 - d.x); + + return o4.y * d.y + o4.x * (1.0 - d.y); +} +vec3 grad(vec3 p) { + const vec2 e = vec2(.1, .0); + return vec3( + noise(p + e.xyy) - noise(p - e.xyy), + noise(p + e.yxy) - noise(p - e.yxy), + noise(p + e.yyx) - noise(p - e.yyx) + ) / (2.0 * e.x); +} +vec3 curlNoise(vec3 p) { + p.xy /= size; + p.x *= (size.x / size.y); + p.xy = fract(p.xy); + p.xy *= noiseScale; + + const vec2 e = vec2(.01, .0); + return grad(p).yzx - vec3( + grad(p + e.yxy).z, + grad(p + e.yyx).x, + grad(p + e.xyy).y + ); +} +float modI(float a,float b) { + return floor(a-floor((a+0.5)/b)*b+0.5); +} + +float particleEaseInWindowFunction(float t) { + return t; +} + +float particleEaseInValueAt(float fraction, float t) { + float windowSize = 0.8; + + float effectiveT = t; + float windowStartOffset = -windowSize; + float windowEndOffset = 1.0; + + float windowPosition = (1.0 - fraction) * windowStartOffset + fraction * windowEndOffset; + float windowT = max(0.0, min(windowSize, effectiveT - windowPosition)) / windowSize; + float localT = 1.0 - particleEaseInWindowFunction(windowT); + + return localT; +} + +void main() { + vec2 uv = inUV; + vec2 position = inPosition; + vec2 velocity = inVelocity; + float particleTime = inTime; + + float id = float(gl_VertexID); + if (reset > 0.) { + uv = vec2( + mod(id, gridSize.x), + floor(id / gridSize.x) + ) / gridSize.xy; + position = (matrix * vec3(uv + .5 / gridSize.xy, 1.0)).xy; + float direction = rand(3. * uv) * (3.14159265 * 2.0); + velocity = vec2(cos(direction), sin(direction)) * (0.1 + rand(5. * uv) * (0.2 - 0.1)) * 210.0 * dp; + particleTime = (0.7 + rand(uv) * (1.5 - 0.7)) / 1.15; + } + + float effectFraction = max(0.0, min(0.35, time)) / 0.35; + float particleFraction = max(0.0, min(0.2, .1 + time - uv.x * 0.6)) / 0.2; + position += velocity * deltaTime * particleFraction; + velocity += vec2(12.0, -60.0) * deltaTime * dp * particleFraction; + particleTime = max(0.0, particleTime - 1.2 * deltaTime * effectFraction); + + outUV = uv; + outPosition = position; + outVelocity = velocity; + outTime = particleTime; + + alpha = max(0.0, min(0.55, particleTime) / 0.55); + gl_PointSize = (gridSize.z + 1.); + position.y = size.y - position.y; + gl_Position = vec4((position + offset) / size * 2.0 - vec2(1.0), 0.0, 1.0); + uvcenter = uv; + uvsize = (vec2(gridSize.z + 1.) / rectSize.xy); +} \ No newline at end of file diff --git a/TMessagesProj/src/main/res/raw/video_start.json b/TMessagesProj/src/main/res/raw/video_start.json new file mode 100644 index 00000000000..f59fd293a63 --- /dev/null +++ b/TMessagesProj/src/main/res/raw/video_start.json @@ -0,0 +1 @@ +{"v":"5.10.1","fr":60,"ip":0,"op":60,"w":156,"h":156,"nm":"video_start","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":3,"nm":"Top 3","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.35]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"i":{"x":[0.833],"y":[0.855]},"o":{"x":[0.167],"y":[0.096]},"t":1,"s":[0.042]},{"i":{"x":[0.833],"y":[0.721]},"o":{"x":[0.167],"y":[0.195]},"t":2,"s":[0.328]},{"i":{"x":[0.833],"y":[0.829]},"o":{"x":[0.167],"y":[0.119]},"t":3,"s":[0.54]},{"i":{"x":[0.833],"y":[0.839]},"o":{"x":[0.167],"y":[0.162]},"t":4,"s":[1.038]},{"i":{"x":[0.833],"y":[0.898]},"o":{"x":[0.167],"y":[0.172]},"t":5,"s":[1.564]},{"i":{"x":[0.833],"y":[1.207]},"o":{"x":[0.167],"y":[0.458]},"t":6,"s":[2.057]},{"i":{"x":[0.833],"y":[0.827]},"o":{"x":[0.167],"y":[0.059]},"t":7,"s":[2.167]},{"i":{"x":[0.833],"y":[0.697]},"o":{"x":[0.167],"y":[0.161]},"t":8,"s":[1.785]},{"i":{"x":[0.833],"y":[0.821]},"o":{"x":[0.167],"y":[0.115]},"t":9,"s":[1.374]},{"i":{"x":[0.833],"y":[0.875]},"o":{"x":[0.167],"y":[0.156]},"t":10,"s":[0.289]},{"i":{"x":[0.833],"y":[0.76]},"o":{"x":[0.167],"y":[0.249]},"t":11,"s":[-0.957]},{"i":{"x":[0.833],"y":[0.845]},"o":{"x":[0.167],"y":[0.128]},"t":12,"s":[-1.583]},{"i":{"x":[0.833],"y":[0.881]},"o":{"x":[0.167],"y":[0.181]},"t":13,"s":[-2.759]},{"i":{"x":[0.833],"y":[0.782]},"o":{"x":[0.167],"y":[0.281]},"t":14,"s":[-3.767]},{"i":{"x":[0.833],"y":[0.859]},"o":{"x":[0.167],"y":[0.135]},"t":15,"s":[-4.193]},{"i":{"x":[0.833],"y":[0.888]},"o":{"x":[0.167],"y":[0.205]},"t":16,"s":[-4.879]},{"i":{"x":[0.833],"y":[0.814]},"o":{"x":[0.167],"y":[0.325]},"t":17,"s":[-5.351]},{"i":{"x":[0.833],"y":[0.937]},"o":{"x":[0.167],"y":[0.151]},"t":18,"s":[-5.513]},{"i":{"x":[0.833],"y":[0.375]},"o":{"x":[0.167],"y":[-0.262]},"t":19,"s":[-5.713]},{"i":{"x":[0.833],"y":[0.626]},"o":{"x":[0.167],"y":[0.096]},"t":20,"s":[-5.665]},{"i":{"x":[0.833],"y":[0.806]},"o":{"x":[0.167],"y":[0.107]},"t":21,"s":[-5.352]},{"i":{"x":[0.833],"y":[0.872]},"o":{"x":[0.167],"y":[0.146]},"t":22,"s":[-4.261]},{"i":{"x":[0.833],"y":[0.753]},"o":{"x":[0.167],"y":[0.24]},"t":23,"s":[-2.819]},{"i":{"x":[0.833],"y":[0.842]},"o":{"x":[0.167],"y":[0.126]},"t":24,"s":[-2.05]},{"i":{"x":[0.833],"y":[0.88]},"o":{"x":[0.167],"y":[0.176]},"t":25,"s":[-0.539]},{"i":{"x":[0.833],"y":[0.778]},"o":{"x":[0.167],"y":[0.274]},"t":26,"s":[0.821]},{"i":{"x":[0.833],"y":[0.859]},"o":{"x":[0.167],"y":[0.133]},"t":27,"s":[1.416]},{"i":{"x":[0.833],"y":[0.898]},"o":{"x":[0.167],"y":[0.205]},"t":28,"s":[2.408]},{"i":{"x":[0.833],"y":[0.93]},"o":{"x":[0.167],"y":[0.46]},"t":29,"s":[3.089]},{"i":{"x":[0.833],"y":[-0.224]},"o":{"x":[0.167],"y":[-0.436]},"t":30,"s":[3.24]},{"i":{"x":[0.833],"y":[0.857]},"o":{"x":[0.167],"y":[0.089]},"t":31,"s":[3.216]},{"i":{"x":[0.833],"y":[0.724]},"o":{"x":[0.167],"y":[0.199]},"t":32,"s":[2.885]},{"i":{"x":[0.833],"y":[0.83]},"o":{"x":[0.167],"y":[0.119]},"t":33,"s":[2.647]},{"i":{"x":[0.833],"y":[0.877]},"o":{"x":[0.167],"y":[0.163]},"t":34,"s":[2.097]},{"i":{"x":[0.833],"y":[0.766]},"o":{"x":[0.167],"y":[0.257]},"t":35,"s":[1.524]},{"i":{"x":[0.833],"y":[0.848]},"o":{"x":[0.167],"y":[0.129]},"t":36,"s":[1.249]},{"i":{"x":[0.833],"y":[0.883]},"o":{"x":[0.167],"y":[0.185]},"t":37,"s":[0.751]},{"i":{"x":[0.833],"y":[0.788]},"o":{"x":[0.167],"y":[0.288]},"t":38,"s":[0.344]},{"i":{"x":[0.833],"y":[0.863]},"o":{"x":[0.167],"y":[0.137]},"t":39,"s":[0.178]},{"i":{"x":[0.833],"y":[0.89]},"o":{"x":[0.167],"y":[0.213]},"t":40,"s":[-0.079]},{"i":{"x":[0.833],"y":[0.83]},"o":{"x":[0.167],"y":[0.347]},"t":41,"s":[-0.243]},{"i":{"x":[0.833],"y":[0.914]},"o":{"x":[0.167],"y":[0.164]},"t":42,"s":[-0.295]},{"i":{"x":[0.833],"y":[1.491]},"o":{"x":[0.167],"y":[2.434]},"t":43,"s":[-0.349]},{"i":{"x":[0.833],"y":[0.632]},"o":{"x":[0.167],"y":[0.071]},"t":44,"s":[-0.351]},{"i":{"x":[0.833],"y":[0.808]},"o":{"x":[0.167],"y":[0.108]},"t":45,"s":[-0.338]},{"i":{"x":[0.833],"y":[0.872]},"o":{"x":[0.167],"y":[0.147]},"t":46,"s":[-0.293]},{"i":{"x":[0.833],"y":[0.753]},"o":{"x":[0.167],"y":[0.24]},"t":47,"s":[-0.234]},{"i":{"x":[0.833],"y":[0.842]},"o":{"x":[0.167],"y":[0.126]},"t":48,"s":[-0.203]},{"i":{"x":[0.833],"y":[0.88]},"o":{"x":[0.167],"y":[0.176]},"t":49,"s":[-0.142]},{"i":{"x":[0.833],"y":[0.778]},"o":{"x":[0.167],"y":[0.274]},"t":50,"s":[-0.087]},{"i":{"x":[0.833],"y":[0.856]},"o":{"x":[0.167],"y":[0.133]},"t":51,"s":[-0.063]},{"i":{"x":[0.833],"y":[0.886]},"o":{"x":[0.167],"y":[0.198]},"t":52,"s":[-0.023]},{"i":{"x":[0.833],"y":[0.804]},"o":{"x":[0.167],"y":[0.311]},"t":53,"s":[0.006]},{"i":{"x":[0.833],"y":[0.878]},"o":{"x":[0.167],"y":[0.145]},"t":54,"s":[0.017]},{"i":{"x":[0.833],"y":[0.904]},"o":{"x":[0.167],"y":[0.265]},"t":55,"s":[0.031]},{"i":{"x":[0.833],"y":[1.045]},"o":{"x":[0.167],"y":[0.647]},"t":56,"s":[0.038]},{"i":{"x":[0.833],"y":[0.658]},"o":{"x":[0.167],"y":[0.029]},"t":57,"s":[0.038]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.11]},"t":58,"s":[0.037]},{"t":59,"s":[0.032]}],"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.713},"o":{"x":0.167,"y":0.167},"t":0,"s":[77.722,79.526,0],"to":[0,0.053,0],"ti":[0,-0.183,0]},{"i":{"x":0.833,"y":0.864},"o":{"x":0.167,"y":0.117},"t":1,"s":[77.722,79.844,0],"to":[0,0.183,0],"ti":[0,-0.211,0]},{"i":{"x":0.833,"y":0.736},"o":{"x":0.167,"y":0.216},"t":2,"s":[77.722,80.623,0],"to":[0,0.211,0],"ti":[0,-0.258,0]},{"i":{"x":0.833,"y":0.849},"o":{"x":0.167,"y":0.122},"t":3,"s":[77.722,81.112,0],"to":[0,0.258,0],"ti":[0,-0.32,0]},{"i":{"x":0.833,"y":0.881},"o":{"x":0.167,"y":0.186},"t":4,"s":[77.722,82.17,0],"to":[0,0.32,0],"ti":[0,-0.087,0]},{"i":{"x":0.833,"y":0.796},"o":{"x":0.167,"y":0.276},"t":5,"s":[77.722,83.03,0],"to":[0,0.087,0],"ti":[0,0.147,0]},{"i":{"x":0.833,"y":0.679},"o":{"x":0.167,"y":0.141},"t":6,"s":[77.722,82.69,0],"to":[0,-0.147,0],"ti":[0,0.348,0]},{"i":{"x":0.833,"y":0.868},"o":{"x":0.167,"y":0.112},"t":7,"s":[77.722,82.15,0],"to":[0,-0.348,0],"ti":[0,0.409,0]},{"i":{"x":0.833,"y":0.743},"o":{"x":0.167,"y":0.225},"t":8,"s":[77.722,80.604,0],"to":[0,-0.409,0],"ti":[0,0.468,0]},{"i":{"x":0.833,"y":0.837},"o":{"x":0.167,"y":0.123},"t":9,"s":[77.722,79.694,0],"to":[0,-0.468,0],"ti":[0,0.62,0]},{"i":{"x":0.833,"y":0.879},"o":{"x":0.167,"y":0.17},"t":10,"s":[77.722,77.793,0],"to":[0,-0.62,0],"ti":[0,0.442,0]},{"i":{"x":0.833,"y":0.772},"o":{"x":0.167,"y":0.266},"t":11,"s":[77.722,75.972,0],"to":[0,-0.442,0],"ti":[0,0.378,0]},{"i":{"x":0.833,"y":0.852},"o":{"x":0.167,"y":0.131},"t":12,"s":[77.722,75.142,0],"to":[0,-0.378,0],"ti":[0,0.425,0]},{"i":{"x":0.833,"y":0.884},"o":{"x":0.167,"y":0.191},"t":13,"s":[77.722,73.703,0],"to":[0,-0.425,0],"ti":[0,0.257,0]},{"i":{"x":0.833,"y":0.795},"o":{"x":0.167,"y":0.298},"t":14,"s":[77.722,72.593,0],"to":[0,-0.257,0],"ti":[0,0.176,0]},{"i":{"x":0.833,"y":0.87},"o":{"x":0.167,"y":0.14},"t":15,"s":[77.722,72.163,0],"to":[0,-0.176,0],"ti":[0,0.164,0]},{"i":{"x":0.833,"y":0.868},"o":{"x":0.167,"y":0.231},"t":16,"s":[77.722,71.535,0],"to":[0,-0.164,0],"ti":[0,0.042,0]},{"i":{"x":0.833,"y":0.396},"o":{"x":0.167,"y":0.227},"t":17,"s":[77.722,71.181,0],"to":[0,-0.042,0],"ti":[0,-0.232,0]},{"i":{"x":0.833,"y":0.774},"o":{"x":0.167,"y":0.097},"t":18,"s":[77.722,71.285,0],"to":[0,0.232,0],"ti":[0,-0.581,0]},{"i":{"x":0.833,"y":0.868},"o":{"x":0.167,"y":0.132},"t":19,"s":[77.722,72.57,0],"to":[0,0.581,0],"ti":[0,-0.579,0]},{"i":{"x":0.833,"y":0.744},"o":{"x":0.167,"y":0.227},"t":20,"s":[77.722,74.769,0],"to":[0,0.579,0],"ti":[0,-0.653,0]},{"i":{"x":0.833,"y":0.838},"o":{"x":0.167,"y":0.124},"t":21,"s":[77.722,76.045,0],"to":[0,0.653,0],"ti":[0,-0.859,0]},{"i":{"x":0.833,"y":0.879},"o":{"x":0.167,"y":0.171},"t":22,"s":[77.722,78.687,0],"to":[0,0.859,0],"ti":[0,-0.608,0]},{"i":{"x":0.833,"y":0.773},"o":{"x":0.167,"y":0.267},"t":23,"s":[77.722,81.197,0],"to":[0,0.608,0],"ti":[0,-0.517,0]},{"i":{"x":0.833,"y":0.853},"o":{"x":0.167,"y":0.132},"t":24,"s":[77.722,82.334,0],"to":[0,0.517,0],"ti":[0,-0.578,0]},{"i":{"x":0.833,"y":0.89},"o":{"x":0.167,"y":0.192},"t":25,"s":[77.722,84.298,0],"to":[0,0.578,0],"ti":[0,-0.332,0]},{"i":{"x":0.833,"y":0.878},"o":{"x":0.167,"y":0.341},"t":26,"s":[77.722,85.804,0],"to":[0,0.332,0],"ti":[0,-0.115,0]},{"i":{"x":0.833,"y":0.716},"o":{"x":0.167,"y":0.261},"t":27,"s":[77.722,86.29,0],"to":[0,0.115,0],"ti":[0,0.058,0]},{"i":{"x":0.833,"y":0.847},"o":{"x":0.167,"y":0.118},"t":28,"s":[77.722,86.491,0],"to":[0,-0.058,0],"ti":[0,0.168,0]},{"i":{"x":0.833,"y":0.712},"o":{"x":0.167,"y":0.183},"t":29,"s":[77.722,85.944,0],"to":[0,-0.168,0],"ti":[0,0.264,0]},{"i":{"x":0.833,"y":0.826},"o":{"x":0.167,"y":0.117},"t":30,"s":[77.722,85.485,0],"to":[0,-0.264,0],"ti":[0,0.392,0]},{"i":{"x":0.833,"y":0.876},"o":{"x":0.167,"y":0.16},"t":31,"s":[77.722,84.362,0],"to":[0,-0.392,0],"ti":[0,0.305,0]},{"i":{"x":0.833,"y":0.763},"o":{"x":0.167,"y":0.253},"t":32,"s":[77.722,83.135,0],"to":[0,-0.305,0],"ti":[0,0.285,0]},{"i":{"x":0.833,"y":0.847},"o":{"x":0.167,"y":0.128},"t":33,"s":[77.722,82.533,0],"to":[0,-0.285,0],"ti":[0,0.34,0]},{"i":{"x":0.833,"y":0.882},"o":{"x":0.167,"y":0.183},"t":34,"s":[77.722,81.423,0],"to":[0,-0.34,0],"ti":[0,0.219,0]},{"i":{"x":0.833,"y":0.785},"o":{"x":0.167,"y":0.284},"t":35,"s":[77.722,80.493,0],"to":[0,-0.219,0],"ti":[0,0.166,0]},{"i":{"x":0.833,"y":0.861},"o":{"x":0.167,"y":0.136},"t":36,"s":[77.722,80.107,0],"to":[0,-0.166,0],"ti":[0,0.169,0]},{"i":{"x":0.833,"y":0.889},"o":{"x":0.167,"y":0.209},"t":37,"s":[77.722,79.497,0],"to":[0,-0.169,0],"ti":[0,0.09,0]},{"i":{"x":0.833,"y":0.821},"o":{"x":0.167,"y":0.334},"t":38,"s":[77.722,79.09,0],"to":[0,-0.09,0],"ti":[0,0.048,0]},{"i":{"x":0.833,"y":0.899},"o":{"x":0.167,"y":0.156},"t":39,"s":[77.722,78.956,0],"to":[0,-0.048,0],"ti":[0,0.031,0]},{"i":{"x":0.833,"y":0.868},"o":{"x":0.167,"y":0.478},"t":40,"s":[77.722,78.801,0],"to":[0,-0.031,0],"ti":[0,0.002,0]},{"i":{"x":0.833,"y":0.553},"o":{"x":0.167,"y":0.229},"t":41,"s":[77.722,78.768,0],"to":[0,-0.002,0],"ti":[0,-0.017,0]},{"i":{"x":0.833,"y":0.795},"o":{"x":0.167,"y":0.103},"t":42,"s":[77.722,78.786,0],"to":[0,0.017,0],"ti":[0,-0.034,0]},{"i":{"x":0.833,"y":0.871},"o":{"x":0.167,"y":0.141},"t":43,"s":[77.722,78.868,0],"to":[0,0.034,0],"ti":[0,-0.031,0]},{"i":{"x":0.833,"y":0.749},"o":{"x":0.167,"y":0.234},"t":44,"s":[77.722,78.987,0],"to":[0,0.031,0],"ti":[0,-0.033,0]},{"i":{"x":0.833,"y":0.84},"o":{"x":0.167,"y":0.125},"t":45,"s":[77.722,79.053,0],"to":[0,0.033,0],"ti":[0,-0.042,0]},{"i":{"x":0.833,"y":0.88},"o":{"x":0.167,"y":0.174},"t":46,"s":[77.722,79.186,0],"to":[0,0.042,0],"ti":[0,-0.029,0]},{"i":{"x":0.833,"y":0.775},"o":{"x":0.167,"y":0.271},"t":47,"s":[77.722,79.308,0],"to":[0,0.029,0],"ti":[0,-0.024,0]},{"i":{"x":0.833,"y":0.854},"o":{"x":0.167,"y":0.133},"t":48,"s":[77.722,79.362,0],"to":[0,0.024,0],"ti":[0,-0.027,0]},{"i":{"x":0.833,"y":0.885},"o":{"x":0.167,"y":0.195},"t":49,"s":[77.722,79.454,0],"to":[0,0.027,0],"ti":[0,-0.016,0]},{"i":{"x":0.833,"y":0.8},"o":{"x":0.167,"y":0.305},"t":50,"s":[77.722,79.523,0],"to":[0,0.016,0],"ti":[0,-0.01,0]},{"i":{"x":0.833,"y":0.874},"o":{"x":0.167,"y":0.143},"t":51,"s":[77.722,79.549,0],"to":[0,0.01,0],"ti":[0,-0.009,0]},{"i":{"x":0.833,"y":0.899},"o":{"x":0.167,"y":0.247},"t":52,"s":[77.722,79.585,0],"to":[0,0.009,0],"ti":[0,-0.004,0]},{"i":{"x":0.833,"y":0.884},"o":{"x":0.167,"y":0.484},"t":53,"s":[77.722,79.603,0],"to":[0,0.004,0],"ti":[0,-0.001,0]},{"i":{"x":0.833,"y":0.492},"o":{"x":0.167,"y":0.278},"t":54,"s":[77.722,79.607,0],"to":[0,0.001,0],"ti":[0,0.001,0]},{"i":{"x":0.833,"y":0.857},"o":{"x":0.167,"y":0.099},"t":55,"s":[77.722,79.607,0],"to":[0,-0.001,0],"ti":[0,0.002,0]},{"i":{"x":0.833,"y":0.724},"o":{"x":0.167,"y":0.199},"t":56,"s":[77.722,79.598,0],"to":[0,-0.002,0],"ti":[0,0.003,0]},{"i":{"x":0.833,"y":0.83},"o":{"x":0.167,"y":0.119},"t":57,"s":[77.722,79.592,0],"to":[0,-0.003,0],"ti":[0,0.005,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.163},"t":58,"s":[77.722,79.579,0],"to":[0,-0.005,0],"ti":[0,0.002,0]},{"t":59,"s":[77.722,79.564,0]}],"ix":2,"l":2},"a":{"a":0,"k":[-0.918,5,0],"ix":1,"l":2},"s":{"a":0,"k":[30.469,30.469,100],"ix":6,"l":2}},"ao":0,"ip":0,"op":60,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"slash","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[-0.006,-0.008,0],"ix":2,"l":2},"a":{"a":0,"k":[78,78,0],"ix":1,"l":2},"s":{"a":0,"k":[328.205,328.205,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-29,-29],[29,29]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":3,"s":[100]},{"t":17,"s":[0]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":7,"ix":5},"lc":2,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[69,80],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"base","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[-0.006,-0.008,0],"ix":2,"l":2},"a":{"a":0,"k":[78,78,0],"ix":1,"l":2},"s":{"a":0,"k":[328.205,328.205,100],"ix":6,"l":2}},"ao":0,"hasMask":true,"masksProperties":[{"inv":false,"mode":"a","pt":{"a":1,"k":[{"i":{"x":0.833,"y":1},"o":{"x":0.167,"y":0},"t":4,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[-2.745,2.55],[0,0],[2.734,2.734],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[2.672,2.672],[0,0],[2.734,-2.734],[0,0],[0,0],[0,0]],"v":[[156.5,0.5],[156.5,156.5],[0.5,156.5],[0.5,21.398],[93.55,114.45],[103.261,114.632],[103.45,114.45],[103.45,104.55],[0.5,1.601],[0.5,0.5]],"c":true}]},{"t":17,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[-2.745,2.55],[0,0],[2.734,2.734],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[2.672,2.672],[0,0],[2.734,-2.734],[0,0],[0,0],[0,0]],"v":[[156.5,0.5],[156.5,156.5],[0.5,156.5],[0.5,21.398],[33.55,54.45],[43.261,54.632],[43.45,54.45],[43.45,44.55],[0.5,1.601],[0.5,0.5]],"c":true}]}],"ix":1},"o":{"a":0,"k":100,"ix":3},"x":{"a":0,"k":0,"ix":4},"nm":"Mask 1"}],"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-6.075,0],[0,0],[0,-6.075],[0,0],[6.075,0],[0,0],[0,6.075],[0,0]],"o":[[0,0],[6.075,0],[0,0],[0,6.075],[0,0],[-6.075,0],[0,0],[0,-6.075]],"v":[[-24,-23],[7,-23],[18,-12],[18,12],[7,23],[-24,23],[-35,12],[-35,-12]],"c":true},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[0,1.5],[0,0],[-1.205,0.992],[0,0],[-1.149,-1.173],[0,-0.704],[0,0],[1.719,0],[0.569,0.461],[0,0]],"o":[[0,0],[0,-1.485],[0,0],[1.279,-1.054],[0.513,0.524],[0,0],[0,1.577],[-0.759,0],[0,0],[-1.225,-0.993]],"v":[[23.747,6.301],[23.747,-5.25],[25.64,-9.143],[36.808,-18.343],[41.203,-18.126],[42,-16.219],[42,17.124],[38.887,19.979],[36.826,19.263],[25.674,10.222]],"c":true},"ix":2},"nm":"Path 3","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[78,78],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60,"st":0,"ct":1,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/TMessagesProj/src/main/res/raw/video_stop.json b/TMessagesProj/src/main/res/raw/video_stop.json new file mode 100644 index 00000000000..31b10a27132 --- /dev/null +++ b/TMessagesProj/src/main/res/raw/video_stop.json @@ -0,0 +1 @@ +{"v":"5.10.1","fr":60,"ip":0,"op":60,"w":156,"h":156,"nm":"video_stop","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":3,"nm":"Top 2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[1.004]},"o":{"x":[0.167],"y":[0]},"t":0,"s":[0]},{"i":{"x":[0.833],"y":[0.689]},"o":{"x":[0.167],"y":[0.083]},"t":1,"s":[0]},{"i":{"x":[0.833],"y":[0.638]},"o":{"x":[0.167],"y":[0.114]},"t":2,"s":[-0.042]},{"i":{"x":[0.833],"y":[0.808]},"o":{"x":[0.167],"y":[0.108]},"t":3,"s":[-0.157]},{"i":{"x":[0.833],"y":[0.829]},"o":{"x":[0.167],"y":[0.148]},"t":4,"s":[-0.54]},{"i":{"x":[0.833],"y":[0.876]},"o":{"x":[0.167],"y":[0.162]},"t":5,"s":[-1.038]},{"i":{"x":[0.833],"y":[0.802]},"o":{"x":[0.167],"y":[0.256]},"t":6,"s":[-1.564]},{"i":{"x":[0.833],"y":[0.942]},"o":{"x":[0.167],"y":[0.144]},"t":7,"s":[-1.817]},{"i":{"x":[0.833],"y":[0.368]},"o":{"x":[0.167],"y":[-0.195]},"t":8,"s":[-2.167]},{"i":{"x":[0.833],"y":[0.785]},"o":{"x":[0.167],"y":[0.096]},"t":9,"s":[-2.062]},{"i":{"x":[0.833],"y":[0.869]},"o":{"x":[0.167],"y":[0.136]},"t":10,"s":[-1.374]},{"i":{"x":[0.833],"y":[0.746]},"o":{"x":[0.167],"y":[0.23]},"t":11,"s":[-0.289]},{"i":{"x":[0.833],"y":[0.839]},"o":{"x":[0.167],"y":[0.124]},"t":12,"s":[0.327]},{"i":{"x":[0.833],"y":[0.879]},"o":{"x":[0.167],"y":[0.172]},"t":13,"s":[1.583]},{"i":{"x":[0.833],"y":[0.774]},"o":{"x":[0.167],"y":[0.269]},"t":14,"s":[2.759]},{"i":{"x":[0.833],"y":[0.854]},"o":{"x":[0.167],"y":[0.132]},"t":15,"s":[3.288]},{"i":{"x":[0.833],"y":[0.885]},"o":{"x":[0.167],"y":[0.193]},"t":16,"s":[4.193]},{"i":{"x":[0.833],"y":[0.834]},"o":{"x":[0.167],"y":[0.302]},"t":17,"s":[4.879]},{"i":{"x":[0.833],"y":[1.096]},"o":{"x":[0.167],"y":[0.167]},"t":18,"s":[5.14]},{"i":{"x":[0.833],"y":[0.84]},"o":{"x":[0.167],"y":[0.045]},"t":19,"s":[5.401]},{"i":{"x":[0.833],"y":[0.706]},"o":{"x":[0.167],"y":[0.173]},"t":20,"s":[4.839]},{"i":{"x":[0.833],"y":[0.824]},"o":{"x":[0.167],"y":[0.116]},"t":21,"s":[4.319]},{"i":{"x":[0.833],"y":[0.875]},"o":{"x":[0.167],"y":[0.158]},"t":22,"s":[3.002]},{"i":{"x":[0.833],"y":[0.762]},"o":{"x":[0.167],"y":[0.251]},"t":23,"s":[1.531]},{"i":{"x":[0.833],"y":[0.846]},"o":{"x":[0.167],"y":[0.128]},"t":24,"s":[0.802]},{"i":{"x":[0.833],"y":[0.882]},"o":{"x":[0.167],"y":[0.182]},"t":25,"s":[-0.555]},{"i":{"x":[0.833],"y":[0.791]},"o":{"x":[0.167],"y":[0.282]},"t":26,"s":[-1.705]},{"i":{"x":[0.833],"y":[0.89]},"o":{"x":[0.167],"y":[0.139]},"t":27,"s":[-2.186]},{"i":{"x":[0.833],"y":[0.93]},"o":{"x":[0.167],"y":[0.342]},"t":28,"s":[-2.912]},{"i":{"x":[0.833],"y":[0.264]},"o":{"x":[0.167],"y":[-0.443]},"t":29,"s":[-3.145]},{"i":{"x":[0.833],"y":[0.781]},"o":{"x":[0.167],"y":[0.094]},"t":30,"s":[-3.108]},{"i":{"x":[0.833],"y":[0.869]},"o":{"x":[0.167],"y":[0.135]},"t":31,"s":[-2.818]},{"i":{"x":[0.833],"y":[0.746]},"o":{"x":[0.167],"y":[0.229]},"t":32,"s":[-2.348]},{"i":{"x":[0.833],"y":[0.838]},"o":{"x":[0.167],"y":[0.124]},"t":33,"s":[-2.08]},{"i":{"x":[0.833],"y":[0.879]},"o":{"x":[0.167],"y":[0.172]},"t":34,"s":[-1.528]},{"i":{"x":[0.833],"y":[0.774]},"o":{"x":[0.167],"y":[0.268]},"t":35,"s":[-1.009]},{"i":{"x":[0.833],"y":[0.853]},"o":{"x":[0.167],"y":[0.132]},"t":36,"s":[-0.775]},{"i":{"x":[0.833],"y":[0.885]},"o":{"x":[0.167],"y":[0.193]},"t":37,"s":[-0.372]},{"i":{"x":[0.833],"y":[0.797]},"o":{"x":[0.167],"y":[0.301]},"t":38,"s":[-0.066]},{"i":{"x":[0.833],"y":[0.871]},"o":{"x":[0.167],"y":[0.141]},"t":39,"s":[0.051]},{"i":{"x":[0.833],"y":[0.897]},"o":{"x":[0.167],"y":[0.237]},"t":40,"s":[0.219]},{"i":{"x":[0.833],"y":[0.89]},"o":{"x":[0.167],"y":[0.431]},"t":41,"s":[0.31]},{"i":{"x":[0.833],"y":[1.258]},"o":{"x":[0.167],"y":[0.345]},"t":42,"s":[0.332]},{"i":{"x":[0.833],"y":[0.85]},"o":{"x":[0.167],"y":[0.063]},"t":43,"s":[0.339]},{"i":{"x":[0.833],"y":[0.715]},"o":{"x":[0.167],"y":[0.187]},"t":44,"s":[0.31]},{"i":{"x":[0.833],"y":[0.827]},"o":{"x":[0.167],"y":[0.118]},"t":45,"s":[0.287]},{"i":{"x":[0.833],"y":[0.876]},"o":{"x":[0.167],"y":[0.16]},"t":46,"s":[0.232]},{"i":{"x":[0.833],"y":[0.764]},"o":{"x":[0.167],"y":[0.254]},"t":47,"s":[0.172]},{"i":{"x":[0.833],"y":[0.847]},"o":{"x":[0.167],"y":[0.129]},"t":48,"s":[0.143]},{"i":{"x":[0.833],"y":[0.882]},"o":{"x":[0.167],"y":[0.183]},"t":49,"s":[0.089]},{"i":{"x":[0.833],"y":[0.786]},"o":{"x":[0.167],"y":[0.285]},"t":50,"s":[0.045]},{"i":{"x":[0.833],"y":[0.862]},"o":{"x":[0.167],"y":[0.136]},"t":51,"s":[0.026]},{"i":{"x":[0.833],"y":[0.889]},"o":{"x":[0.167],"y":[0.21]},"t":52,"s":[-0.003]},{"i":{"x":[0.833],"y":[0.823]},"o":{"x":[0.167],"y":[0.337]},"t":53,"s":[-0.022]},{"i":{"x":[0.833],"y":[0.902]},"o":{"x":[0.167],"y":[0.157]},"t":54,"s":[-0.029]},{"i":{"x":[0.833],"y":[0.981]},"o":{"x":[0.167],"y":[0.549]},"t":55,"s":[-0.036]},{"i":{"x":[0.833],"y":[0.564]},"o":{"x":[0.167],"y":[-0.024]},"t":56,"s":[-0.037]},{"i":{"x":[0.833],"y":[0.799]},"o":{"x":[0.167],"y":[0.103]},"t":57,"s":[-0.036]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.142]},"t":58,"s":[-0.032]},{"t":59,"s":[-0.026]}],"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.713},"o":{"x":0.167,"y":0.167},"t":0,"s":[77.722,79.526,0],"to":[0,-0.053,0],"ti":[0,0.183,0]},{"i":{"x":0.833,"y":0.864},"o":{"x":0.167,"y":0.117},"t":1,"s":[77.722,79.208,0],"to":[0,-0.183,0],"ti":[0,0.211,0]},{"i":{"x":0.833,"y":0.736},"o":{"x":0.167,"y":0.216},"t":2,"s":[77.722,78.429,0],"to":[0,-0.211,0],"ti":[0,0.258,0]},{"i":{"x":0.833,"y":0.834},"o":{"x":0.167,"y":0.122},"t":3,"s":[77.722,77.94,0],"to":[0,-0.258,0],"ti":[0,0.351,0]},{"i":{"x":0.833,"y":0.856},"o":{"x":0.167,"y":0.168},"t":4,"s":[77.722,76.881,0],"to":[0,-0.351,0],"ti":[0,0.302,0]},{"i":{"x":0.833,"y":0.894},"o":{"x":0.167,"y":0.197},"t":5,"s":[77.722,75.834,0],"to":[0,-0.302,0],"ti":[0,0.118,0]},{"i":{"x":0.833,"y":0.567},"o":{"x":0.167,"y":0.388},"t":6,"s":[77.722,75.068,0],"to":[0,-0.118,0],"ti":[0,-0.156,0]},{"i":{"x":0.833,"y":0.849},"o":{"x":0.167,"y":0.103},"t":7,"s":[77.722,75.125,0],"to":[0,0.156,0],"ti":[0,-0.265,0]},{"i":{"x":0.833,"y":0.715},"o":{"x":0.167,"y":0.187},"t":8,"s":[77.722,76.004,0],"to":[0,0.265,0],"ti":[0,-0.404,0]},{"i":{"x":0.833,"y":0.827},"o":{"x":0.167,"y":0.118},"t":9,"s":[77.722,76.712,0],"to":[0,0.404,0],"ti":[0,-0.595,0]},{"i":{"x":0.833,"y":0.876},"o":{"x":0.167,"y":0.16},"t":10,"s":[77.722,78.426,0],"to":[0,0.595,0],"ti":[0,-0.46,0]},{"i":{"x":0.833,"y":0.763},"o":{"x":0.167,"y":0.254},"t":11,"s":[77.722,80.28,0],"to":[0,0.46,0],"ti":[0,-0.428,0]},{"i":{"x":0.833,"y":0.847},"o":{"x":0.167,"y":0.129},"t":12,"s":[77.722,81.184,0],"to":[0,0.428,0],"ti":[0,-0.508,0]},{"i":{"x":0.833,"y":0.882},"o":{"x":0.167,"y":0.183},"t":13,"s":[77.722,82.846,0],"to":[0,0.508,0],"ti":[0,-0.327,0]},{"i":{"x":0.833,"y":0.785},"o":{"x":0.167,"y":0.285},"t":14,"s":[77.722,84.232,0],"to":[0,0.327,0],"ti":[0,-0.246,0]},{"i":{"x":0.833,"y":0.875},"o":{"x":0.167,"y":0.136},"t":15,"s":[77.722,84.806,0],"to":[0,0.246,0],"ti":[0,-0.226,0]},{"i":{"x":0.833,"y":0.873},"o":{"x":0.167,"y":0.25},"t":16,"s":[77.722,85.708,0],"to":[0,0.226,0],"ti":[0,-0.042,0]},{"i":{"x":0.833,"y":0.524},"o":{"x":0.167,"y":0.244},"t":17,"s":[77.722,86.16,0],"to":[0,0.042,0],"ti":[0,0.217,0]},{"i":{"x":0.833,"y":0.79},"o":{"x":0.167,"y":0.101},"t":18,"s":[77.722,85.96,0],"to":[0,-0.217,0],"ti":[0,0.464,0]},{"i":{"x":0.833,"y":0.87},"o":{"x":0.167,"y":0.138},"t":19,"s":[77.722,84.857,0],"to":[0,-0.464,0],"ti":[0,0.437,0]},{"i":{"x":0.833,"y":0.748},"o":{"x":0.167,"y":0.232},"t":20,"s":[77.722,83.177,0],"to":[0,-0.437,0],"ti":[0,0.475,0]},{"i":{"x":0.833,"y":0.839},"o":{"x":0.167,"y":0.124},"t":21,"s":[77.722,82.236,0],"to":[0,-0.475,0],"ti":[0,0.614,0]},{"i":{"x":0.833,"y":0.879},"o":{"x":0.167,"y":0.173},"t":22,"s":[77.722,80.328,0],"to":[0,-0.614,0],"ti":[0,0.428,0]},{"i":{"x":0.833,"y":0.775},"o":{"x":0.167,"y":0.27},"t":23,"s":[77.722,78.552,0],"to":[0,-0.428,0],"ti":[0,0.358,0]},{"i":{"x":0.833,"y":0.858},"o":{"x":0.167,"y":0.132},"t":24,"s":[77.722,77.757,0],"to":[0,-0.358,0],"ti":[0,0.386,0]},{"i":{"x":0.833,"y":0.897},"o":{"x":0.167,"y":0.201},"t":25,"s":[77.722,76.403,0],"to":[0,-0.386,0],"ti":[0,0.197,0]},{"i":{"x":0.833,"y":0.886},"o":{"x":0.167,"y":0.441},"t":26,"s":[77.722,75.443,0],"to":[0,-0.197,0],"ti":[0,0.038,0]},{"i":{"x":0.833,"y":0.495},"o":{"x":0.167,"y":0.307},"t":27,"s":[77.722,75.22,0],"to":[0,-0.038,0],"ti":[0,-0.07,0]},{"i":{"x":0.833,"y":0.855},"o":{"x":0.167,"y":0.1},"t":28,"s":[77.722,75.218,0],"to":[0,0.07,0],"ti":[0,-0.122,0]},{"i":{"x":0.833,"y":0.722},"o":{"x":0.167,"y":0.196},"t":29,"s":[77.722,75.639,0],"to":[0,0.122,0],"ti":[0,-0.173,0]},{"i":{"x":0.833,"y":0.829},"o":{"x":0.167,"y":0.119},"t":30,"s":[77.722,75.95,0],"to":[0,0.173,0],"ti":[0,-0.249,0]},{"i":{"x":0.833,"y":0.877},"o":{"x":0.167,"y":0.162},"t":31,"s":[77.722,76.677,0],"to":[0,0.249,0],"ti":[0,-0.189,0]},{"i":{"x":0.833,"y":0.765},"o":{"x":0.167,"y":0.256},"t":32,"s":[77.722,77.443,0],"to":[0,0.189,0],"ti":[0,-0.173,0]},{"i":{"x":0.833,"y":0.848},"o":{"x":0.167,"y":0.129},"t":33,"s":[77.722,77.812,0],"to":[0,0.173,0],"ti":[0,-0.204,0]},{"i":{"x":0.833,"y":0.883},"o":{"x":0.167,"y":0.185},"t":34,"s":[77.722,78.483,0],"to":[0,0.204,0],"ti":[0,-0.13,0]},{"i":{"x":0.833,"y":0.787},"o":{"x":0.167,"y":0.287},"t":35,"s":[77.722,79.035,0],"to":[0,0.13,0],"ti":[0,-0.096,0]},{"i":{"x":0.833,"y":0.863},"o":{"x":0.167,"y":0.137},"t":36,"s":[77.722,79.261,0],"to":[0,0.096,0],"ti":[0,-0.096,0]},{"i":{"x":0.833,"y":0.89},"o":{"x":0.167,"y":0.212},"t":37,"s":[77.722,79.612,0],"to":[0,0.096,0],"ti":[0,-0.05,0]},{"i":{"x":0.833,"y":0.828},"o":{"x":0.167,"y":0.344},"t":38,"s":[77.722,79.839,0],"to":[0,0.05,0],"ti":[0,-0.025,0]},{"i":{"x":0.833,"y":0.904},"o":{"x":0.167,"y":0.162},"t":39,"s":[77.722,79.912,0],"to":[0,0.025,0],"ti":[0,-0.014,0]},{"i":{"x":0.833,"y":0.807},"o":{"x":0.167,"y":0.617},"t":40,"s":[77.722,79.989,0],"to":[0,0.014,0],"ti":[0,0.002,0]},{"i":{"x":0.833,"y":0.617},"o":{"x":0.167,"y":0.143},"t":41,"s":[77.722,79.996,0],"to":[0,-0.002,0],"ti":[0,0.012,0]},{"i":{"x":0.833,"y":0.805},"o":{"x":0.167,"y":0.107},"t":42,"s":[77.722,79.979,0],"to":[0,-0.012,0],"ti":[0,0.023,0]},{"i":{"x":0.833,"y":0.872},"o":{"x":0.167,"y":0.146},"t":43,"s":[77.722,79.921,0],"to":[0,-0.023,0],"ti":[0,0.02,0]},{"i":{"x":0.833,"y":0.753},"o":{"x":0.167,"y":0.239},"t":44,"s":[77.722,79.844,0],"to":[0,-0.02,0],"ti":[0,0.021,0]},{"i":{"x":0.833,"y":0.841},"o":{"x":0.167,"y":0.126},"t":45,"s":[77.722,79.802,0],"to":[0,-0.021,0],"ti":[0,0.026,0]},{"i":{"x":0.833,"y":0.88},"o":{"x":0.167,"y":0.176},"t":46,"s":[77.722,79.72,0],"to":[0,-0.026,0],"ti":[0,0.018,0]},{"i":{"x":0.833,"y":0.777},"o":{"x":0.167,"y":0.273},"t":47,"s":[77.722,79.646,0],"to":[0,-0.018,0],"ti":[0,0.014,0]},{"i":{"x":0.833,"y":0.856},"o":{"x":0.167,"y":0.133},"t":48,"s":[77.722,79.614,0],"to":[0,-0.014,0],"ti":[0,0.016,0]},{"i":{"x":0.833,"y":0.886},"o":{"x":0.167,"y":0.197},"t":49,"s":[77.722,79.56,0],"to":[0,-0.016,0],"ti":[0,0.009,0]},{"i":{"x":0.833,"y":0.803},"o":{"x":0.167,"y":0.31},"t":50,"s":[77.722,79.52,0],"to":[0,-0.009,0],"ti":[0,0.006,0]},{"i":{"x":0.833,"y":0.877},"o":{"x":0.167,"y":0.145},"t":51,"s":[77.722,79.505,0],"to":[0,-0.006,0],"ti":[0,0.005,0]},{"i":{"x":0.833,"y":0.903},"o":{"x":0.167,"y":0.26},"t":52,"s":[77.722,79.485,0],"to":[0,-0.005,0],"ti":[0,0.002,0]},{"i":{"x":0.833,"y":0.835},"o":{"x":0.167,"y":0.56},"t":53,"s":[77.722,79.476,0],"to":[0,-0.002,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.613},"o":{"x":0.167,"y":0.164},"t":54,"s":[77.722,79.475,0],"to":[0,0,0],"ti":[0,-0.001,0]},{"i":{"x":0.833,"y":0.861},"o":{"x":0.167,"y":0.106},"t":55,"s":[77.722,79.476,0],"to":[0,0.001,0],"ti":[0,-0.002,0]},{"i":{"x":0.833,"y":0.731},"o":{"x":0.167,"y":0.209},"t":56,"s":[77.722,79.482,0],"to":[0,0.002,0],"ti":[0,-0.002,0]},{"i":{"x":0.833,"y":0.832},"o":{"x":0.167,"y":0.121},"t":57,"s":[77.722,79.486,0],"to":[0,0.002,0],"ti":[0,-0.003,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.166},"t":58,"s":[77.722,79.495,0],"to":[0,0.003,0],"ti":[0,-0.001,0]},{"t":59,"s":[77.722,79.504,0]}],"ix":2,"l":2},"a":{"a":0,"k":[-0.918,5,0],"ix":1,"l":2},"s":{"a":0,"k":[30.469,30.469,100],"ix":6,"l":2}},"ao":0,"ip":0,"op":60,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"slash","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[-0.006,-0.008,0],"ix":2,"l":2},"a":{"a":0,"k":[78,78,0],"ix":1,"l":2},"s":{"a":0,"k":[328.205,328.205,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-29,-29],[29,29]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":3,"s":[0]},{"t":17,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":7,"ix":5},"lc":2,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[69,80],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60,"st":0,"ct":1,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"base","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[-0.006,-0.008,0],"ix":2,"l":2},"a":{"a":0,"k":[78,78,0],"ix":1,"l":2},"s":{"a":0,"k":[328.205,328.205,100],"ix":6,"l":2}},"ao":0,"hasMask":true,"masksProperties":[{"inv":false,"mode":"a","pt":{"a":1,"k":[{"i":{"x":0.5,"y":1},"o":{"x":0.6,"y":0},"t":3,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[-2.745,2.55],[0,0],[2.734,2.734],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[2.672,2.672],[0,0],[2.734,-2.734],[0,0],[0,0],[0,0]],"v":[[156.5,0.5],[156.5,156.5],[0.5,156.5],[0.5,21.398],[33.55,54.45],[43.261,54.632],[43.45,54.45],[43.45,44.55],[0.5,1.601],[0.5,0.5]],"c":true}]},{"t":16,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[-2.745,2.55],[0,0],[2.734,2.734],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[2.672,2.672],[0,0],[2.734,-2.734],[0,0],[0,0],[0,0]],"v":[[156.5,0.5],[156.5,156.5],[0.5,156.5],[0.5,21.398],[93.55,114.45],[103.261,114.632],[103.45,114.45],[103.45,104.55],[0.5,1.601],[0.5,0.5]],"c":true}]}],"ix":1},"o":{"a":0,"k":100,"ix":3},"x":{"a":0,"k":0,"ix":4},"nm":"Mask 1"}],"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-6.075,0],[0,0],[0,-6.075],[0,0],[6.075,0],[0,0],[0,6.075],[0,0]],"o":[[0,0],[6.075,0],[0,0],[0,6.075],[0,0],[-6.075,0],[0,0],[0,-6.075]],"v":[[-24,-23],[7,-23],[18,-12],[18,12],[7,23],[-24,23],[-35,12],[-35,-12]],"c":true},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[0,1.5],[0,0],[-1.205,0.992],[0,0],[-1.149,-1.173],[0,-0.704],[0,0],[1.719,0],[0.569,0.461],[0,0]],"o":[[0,0],[0,-1.485],[0,0],[1.279,-1.054],[0.513,0.524],[0,0],[0,1.577],[-0.759,0],[0,0],[-1.225,-0.993]],"v":[[23.747,6.301],[23.747,-5.25],[25.64,-9.143],[36.808,-18.343],[41.203,-18.126],[42,-16.219],[42,17.124],[38.887,19.979],[36.826,19.263],[25.674,10.222]],"c":true},"ix":2},"nm":"Path 3","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[78,78],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60,"st":0,"ct":1,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/TMessagesProj/src/main/res/values/strings.xml b/TMessagesProj/src/main/res/values/strings.xml index f2c3d2f8a9b..486ef9e392f 100644 --- a/TMessagesProj/src/main/res/values/strings.xml +++ b/TMessagesProj/src/main/res/values/strings.xml @@ -241,6 +241,7 @@ Select Chat Choose Bot Choose User + Choose Users Choose Group Choose Channel Select Chats @@ -988,6 +989,14 @@ un1 disabled anti-spam un1 changed channel color from %1$s to %2$s un1 changed channel emoji from %1$s to %2$s + un1 changed channel color and icon from %1$s to %2$s + un1 changed channel profile color and icon from %1$s to %2$s + un1 changed channel emoji status from %1$s to %2$s + un1 changed channel emoji status to %2$s + un1 changed channel emoji status from %1$s to %2$s (for %3$s) + un1 changed channel emoji status to %2$s (for %3$s) + un1 changed channel wallpaper + un1 removed channel wallpaper none New Broadcast List @@ -1013,6 +1022,7 @@ Search Recent Files Free %1$s of %2$s Unknown error + Unknown error: %s Access error File size shouldn\'t be greater than %1$s Storage not mounted @@ -2183,7 +2193,7 @@ Motion Change Chat Background Change Your Color - Change Name Color + Appearance Chat Background Reset Chat Backgrounds Remove all uploaded chat backgrounds and restore the pre-installed ones. @@ -2272,6 +2282,7 @@ Terms of Service By signing up,\nyou agree to the *Terms of Service*. https://telegram.org/privacy + https://telegram.org/tos https://telegram.org/tos/mini-apps Delete language pack Are you sure you want to delete **%1$s** language pack? @@ -2448,6 +2459,8 @@ Other Paste Flip + Cut Out + Undo Cut Paste from clipboard Proxy Settings Proxy Details @@ -3604,6 +3617,8 @@ Close anyway Allow messaging Allow this bot to send you messages? + You can select maximum %1$d user. + You can select maximum %1$d users. Done Open @@ -3730,6 +3745,8 @@ Self-Destructing Video Photo has expired Video has expired + Round video has expired + Voice expired GIF Location Live Location @@ -4063,6 +4080,7 @@ Ongoing Voice Chat Ongoing Live Stream End call + End Call Another call in progress You currently have an ongoing call with **%1$s**. Would you like to hang up and start a new one with **%2$s**? End call with **%1$s** and start voice chat in **%2$s**? @@ -4399,6 +4417,13 @@ Record live stream Choose how to record this chat Start Recording + Rate this call + Please rate the quality of this call. + Encryption key of this call + This call is end to end encrypted + Weak network signal + Hide Emoji + Your microphone is turned off **Oops!** Telegram doesn\'t see any stream coming from your streaming app.\n\nPlease make sure you entered the right Server URL and Stream Key in your app. %1$s is currently not broadcasting live stream data to Telegram. @@ -5507,9 +5532,12 @@ Apply Theme Reset Theme No Theme + No Wallpaper You changed the chat theme to %1$s %1$s changed the chat theme to %2$s + %1$s changed the channel theme to %2$s %1$s disabled the chat theme + %1$s disabled the channel theme You disabled the chat theme Save changes? Do you want to apply the new theme for this chat? @@ -5916,12 +5944,43 @@ You need an official Telegram app to subscribe to **Telegram Premium**.\n\nOnce subscribed, you will be able to use the benefits of Telegram Premium in any apps that support it, including unofficial ones. Install official app Gift Subscription for %1$s + Gift %1$d Subscription for %2$s + Gift %1$d Subscriptions for %2$s -%1$d%% %1$s / month %1$s/month %1$s/year You can review the list of features and terms of use for Telegram Premium *here*. Gift Premium + Premium Gifting + Activate For Free + You already have Telegram Premium + You can activate this gift code after %1$s or **send the link** to a friend. + Use Gift + Gift Sent! + Gifts Sent! + **%1$s** + **%1$s**, **%2$s** + **%1$s**, **%2$s**, **%3$s** + %1$s have been notified about the gifts you purchased. + %2$s and **%1$d** other have been notified about the gifts you purchased. + %2$s and **%1$d** others have been notified about the gifts you purchased. + Give %1$s access to exclusive features. + Give %2$s and **%1$d** more friend access to exclusive features. + Give %2$s and **%1$d** more friends access to exclusive features. + They now have access to additional features. + **%1$s** now has access to additional features. + Frequent Contacts + Search people to gift Premium to... + You will receive ⚡ **%1$d** boost. + You will receive ⚡ **%1$d** boosts. + By gifting Telegram Premium you agree to the **Telegram Terms of Service** and %1$s + **Privacy Policy**. + Choose Recipients + Proceed + This link allows you or **anyone you choose** to activate a %1$s. + **Telegram Premium** subscription + What\'s included Gift Telegram Premium Let **%1$s** enjoy exclusive features of Telegram with **Telegram Premium**. Telegram Premium @@ -6325,6 +6384,12 @@ You shared a user with un2 You shared a chat with un2 You shared a channel with un2 + You shared a user with un2 + You shared a users with un2 + You shared a chat with un2 + You shared a chats with un2 + You shared a channel with un2 + You shared a channels with un2 Hide with spoiler Remove spoiler Update Public Photo @@ -6496,6 +6561,7 @@ Animated spoiler effect Night theme blur Zoom animations + Dust-effect deletion Animations in Calls Autoplay Videos Autoplay GIFs @@ -6631,6 +6697,7 @@ Set Background From Gallery Setting new wallpaper... You set a new wallpaper for this chat + Channel set a new wallpaper You set the same wallpaper for the chat %s set a new wallpaper for this chat %s set a new wallpaper for this chat @@ -6748,7 +6815,11 @@ Theme will also be applied for %s Background dimming Apply the wallpaper in this chat + Du Rove\'s Channel + Breaking News + Details to follow shortly.\nStay tuned! %s will be able to apply this wallpaper + All subscribers will see this wallpaper APPLY FOR THIS CHAT Remove Wallpaper Set a Color as Wallpaper @@ -7057,6 +7128,7 @@ Automatically enabled Disabled No one has viewed this story so far. + No one has reacted to this story so far. You are seeing this story because **%s** added you to their list of Close Friends Only some contacts **%s** selected can view this story. Only **%s’s** contacts can view this story. @@ -7065,6 +7137,7 @@ Saved Stories New Story Edit + Repost Allow Screenshots **Select people** who won’t see your stories. **%d people** won’t see your stories. @@ -7172,7 +7245,7 @@ Hide Stories Unhide Stories Subscribe to **Telegram Premium** to add links and text formatting to your stories. - Subscribe to **Telegram Premium** to add up to 5 reactions and tags to a story. + Subscribe to **Telegram Premium** to add up to %d reactions and tags to a story. Maximum Length Reach Increase this limit **%1$d** times to **%2$s** symbols by subscribing to __Telegram Premium.__ To unlock viewers’ lists for expired and saved stories, subscribe to **Telegram Premium.** @@ -7215,6 +7288,7 @@ Stealth Mode Is Active Please wait until the **Stealth Mode** is ready to use again. View Location > + View Message > Like Long tap for more reactions Add Reactions... @@ -7233,8 +7307,10 @@ You can select maximum %1$d reactions. All viewers Reactions First + Reposts First Newest First Choose the order for the 
list of viewers. + Choose the order for the 
list of reactions. None of your contacts viewed this story. Viewers You are in Stealth Mode now @@ -7248,10 +7324,11 @@ No views To unlock viewers’ lists for expired and saved stories, subscribe to **Telegram Premium.** %d like - %d likes - %d likes - %d likes %d likes + %d repost + %d reposts + %d reaction + %d reactions You can post **%1$d** stories in **24** hours.\nSubscribe to **Telegram Premium** to increase this limit to **%2$d**. Sorry, you can’t post more than **%1$d** stories in **24** hours. You can post **%1$d** stories in a week.\nSubscribe to **Telegram Premium** to increase this limit to **%2$d**. @@ -7365,6 +7442,7 @@ To be distributed %1$dm Giveaway + Giveaway results Channel started a giveaway The recipient will be selected when the giveaway ends. Incomplete Giveaway @@ -7437,11 +7515,22 @@ Gift link forwarded to **%1$s**. Gift link forwarded to **Saved Messages**. Premium Subscriptions Gifted + Enter Your Prize + Additional Prizes + Show Winners + Choose whether to make the list of winners public when the giveaway ends. + Turn this on if you want to give the winners your own prizes in addition to Premium subscriptions. + All prizes: **%1$d** %2$s with Telegram Premium subscriptions for %3$s. + All prizes: **%1$d** %2$s with Telegram Premium subscriptions for %3$s. + All prizes: **%1$d** Telegram Premium subscriptions for %2$s. + All prizes: **%1$d** Telegram Premium subscriptions for %2$s. This giveaway is sponsored by the admins of **%2$s**, who acquired **%3$s Telegram Premium** subscription for %4$s for its followers. This giveaway is sponsored by the admins of **%2$s**, who acquired **%3$s Telegram Premium** subscriptions for %4$s for its followers. This giveaway is sponsored by the admins of **%2$s**, who acquired **%3$s Telegram Premium** subscriptions for %4$s for its followers. This giveaway is sponsored by the admins of **%2$s**, who acquired **%3$s Telegram Premium** subscriptions for %4$s for its followers. This giveaway is sponsored by the admins of **%2$s**, who acquired **%3$s Telegram Premium** subscriptions for %4$s for its followers. + **%2$s** also included **%1$d** **%3$s** in the prizes. Admins of the channel are responsible for delivering these prizes. + **%2$s** also included **%1$d** **%3$s** in the prizes. Admins of the channel are responsible for delivering these prizes. On **%2$s**, Telegram will automatically select **%3$s** random user that joined **%4$s**. On **%2$s**, Telegram will automatically select **%3$s** random users that joined **%4$s**. On **%2$s**, Telegram will automatically select **%3$s** random users that joined **%4$s**. @@ -7583,12 +7672,15 @@ %d other channels %d other channels To boost **%1$s**, get more boosts by gifting **Telegram Premium** to a friend. + To boost **%2$s**, get **%1$d** more boost by gifting **Telegram Premium** to a friend. + To boost **%2$s**, get **%1$d** more boosts by gifting **Telegram Premium** to a friend. Reassign Boosts - To boost **%2$s**, reassign a previous boost or gift **Telegram Premium** to a friend to get **%1$d** additional boost. - To boost **%2$s**, reassign a previous boost or gift **Telegram Premium** to a friend to get **%1$d** additional boosts. - To boost **%2$s**, reassign a previous boost or gift **Telegram Premium** to a friend to get **%1$d** additional boosts. - To boost **%2$s**, reassign a previous boost or gift **Telegram Premium** to a friend to get **%1$d** additional boosts. - To boost **%2$s**, reassign a previous boost or gift **Telegram Premium** to a friend to get **%1$d** additional boosts. + To boost **%2$s**, reassign a previous boost or %3$s to a friend to get **%1$d** additional boost. + To boost **%2$s**, reassign a previous boost or %3$s to a friend to get **%1$d** additional boosts. + **gift Telegram Premium** + Send gifts **to your friends!** + Give Telegram Premium for Christmas. + You can gift Telegram Premium later in Settings Remove your boost from available in %1$s Duration of Premium subscriptions @@ -7597,6 +7689,14 @@ from %1$s **Winners Selection Date** **Giveaway Prizes** + **Winners Selected!** + **Winner** + **Winners** + All winners received gift links in private messages. + **And %1$d more!** + **And %1$d more!** + %1$d winner of the **Giveaway** was randomly selected by Telegram. + %1$d winners of the **Giveaway** were randomly selected by Telegram. **%1$d** Telegram Premium **%1$d** Telegram Premium **%1$d** Telegram Premium @@ -7612,7 +7712,10 @@ All subscribers of the channels: New subscribers of the channel: New subscribers of the channels: - You can’t add up more than 5 reactions tags to a story. + with + **%1$d** %2$s + **%1$d** %2$s + You can’t add up more than %d reactions tags to a story. Someone just got access to your messages! Yes, it’s me No, it’s not me! @@ -7635,6 +7738,7 @@ Boosts to level up Your channel is currently boosted by these users. Boost Channel + Unlock Features BOOST Enable stories for the channel Enable stories for channel @@ -7655,12 +7759,19 @@ Terms of Use boost expires on %s boosts expire on %s - Your channel needs to reach **Level %d** to change channel color.\n\nAsk your **Premium** subscribers to boost your channel with this link: + Your channel needs to reach **Level %d** to change channel color to selected.\n\nAsk your **Premium** subscribers to boost your channel with this link: + Your channel needs to reach **Level %d** to change channel profile color to selected.\n\nAsk your **Premium** subscribers to boost your channel with this link: + Your channel needs to reach **Level %d** to change channel link style icon.\n\nAsk your **Premium** subscribers to boost your channel with this link: + Your channel needs to reach **Level %d** to change channel profile icon.\n\nAsk your **Premium** subscribers to boost your channel with this link: + Your channel needs to reach **Level %d** to set channel emoji status.\n\nAsk your **Premium** subscribers to boost your channel with this link: + Your channel needs to reach **Level %d** to change channel wallpaper.\n\nAsk your **Premium** subscribers to boost your channel with this link: + Your channel needs to reach **Level %d** to change channel wallpaper to custom photo.\n\nAsk your **Premium** subscribers to boost your channel with this link: Your channel needs %s to enable posting stories.\n\nAsk your **Premium** subscribers to boost your channel with this link: Your channel needs %1$s to be able post **%2$s** per day.\n\nAsk your Premium subscribers to boost your channel with this link: This channel need %s to enable stories. Help make it possible! This channel need %1$s to be able post **%2$s** per day. Help make it possible! This channel need %s more boosts to enable stories. + **%1$s** needs %2$s to unlock new features. Help upgrade channel This channel reached **Level 1** and can now post stories. This channel reached **Level %1$d** and can now post **%2$s** per day. @@ -7681,7 +7792,12 @@ **%d** stories **%d** stories Increase Story Limit - Enable Colors + Unlock Colors + Unlock Profile Colors + Unlock Emoji Status + Unlock Wallpaper + Unlock Link Icons + Unlock Profile Icons Slide left or right to seek Quote to... Reply to... @@ -7739,7 +7855,7 @@ Choose a color and an icon for your profile Off Off - Your Channel Color + Appearance You can choose an individual color to tint your channel’s name, the links it sends, and replies to its messages. The name of the channel and replies to its message will be shown in the selected color Reply to your channel message @@ -7773,9 +7889,10 @@ Unsaved Changes You have changed your color or icon settings. Apply changes? Unsaved Changes - You have changed your color or icon settings. Apply changes? + You have changed appearance settings. Apply changes? OPEN Your name color has been updated! + Channel appearance has been updated Your channel color has been updated! Your profile color has been updated! Your channel profile color has been updated! @@ -7787,10 +7904,12 @@ Similar Channels Apply for Me Apply for Me and %s + Apply for %s Set Background Remove wallpaper Are you sure you want to reset wallpaper back? Repost\nStory + Repost\nto Story Remove Video Are you sure you want to remove your video message? You have **%1$d** free voice transcription left. @@ -7809,4 +7928,47 @@ Unlock more channels Show More Channels Subscribe to **Telegram Premium**\nto unlock up to %s similar channels. + commented + Level %d + Level %d+ + %d Story Per Day + %d Stories Per Day + %d Custom Reaction + %d Custom Reactions + %d Channel Name Colors + %d Styles for Links and Quotes + %d Styles for Links and Quotes + Custom Logo for Links and Quotes + %d Colors for Channel Cover + Custom Logo for Channel Cover + %s Emoji Statuses + %d Channel Background + %d Channel Backgrounds + Custom Channel Background + Level %d Unlocks: + Level %1$d Required + Reply Logo + Choose a color for the name of your channel, the links it sends, and replies to its messages. + Profile Logo + Choose a color and a logo for the channel\'s profile. + Channel Emoji Status + Choose a status that will be shown next to the channel\'s name. + Channel Wallpaper + Choose from Gallery + Remove Wallpaper + Upload your own background image for the channel. + Set a wallpaper that will be visible for everyone reading your channel. + Read More + Message reposted to your profile. + Message reposted to **%s**. + Select Wallpaper + This voice message can only be played once. + This message will disappear once **%s** plays it once. + Delete and Close + Close + Please, turn on the sound first. + Tap to set this message to **Play Once**. + The recipient will be able to listen to it only once. + Close Voice Message + Are you sure you want to stop listening and delete voice message? diff --git a/TMessagesProj/src/main/res/xml/automotive_app_desc.xml b/TMessagesProj/src/main/res/xml/automotive_app_desc.xml deleted file mode 100644 index 79ec6f5e25c..00000000000 --- a/TMessagesProj/src/main/res/xml/automotive_app_desc.xml +++ /dev/null @@ -1,5 +0,0 @@ - - - - - \ No newline at end of file diff --git a/TMessagesProj_AppStandalone/src/main/res/mipmap-anydpi-v26/icon_2_launcher_sa.xml b/TMessagesProj_AppStandalone/src/main/res/mipmap-anydpi-v26/icon_2_launcher_sa.xml index d8044115b1d..e7996ea533e 100644 --- a/TMessagesProj_AppStandalone/src/main/res/mipmap-anydpi-v26/icon_2_launcher_sa.xml +++ b/TMessagesProj_AppStandalone/src/main/res/mipmap-anydpi-v26/icon_2_launcher_sa.xml @@ -1,5 +1,5 @@ - + \ No newline at end of file diff --git a/gradle.properties b/gradle.properties index ff409bf10f7..460cd677d7f 100644 --- a/gradle.properties +++ b/gradle.properties @@ -13,8 +13,8 @@ # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects # org.gradle.parallel=true #Sat Mar 12 05:53:50 MSK 2016 -APP_VERSION_CODE=4145 -APP_VERSION_NAME=10.3.2 +APP_VERSION_CODE=4228 +APP_VERSION_NAME=10.5.0 APP_PACKAGE=org.telegram.messenger RELEASE_KEY_PASSWORD=android RELEASE_KEY_ALIAS=androidkey