From 97a9089ebfcd076cc051ee1c4683b5c1cf3fb1ae Mon Sep 17 00:00:00 2001 From: John Preston Date: Sat, 13 Jan 2018 15:45:11 +0300 Subject: [PATCH] Move draw / getState code to HistoryView::Message. Item dimensions broken for now. Also remove history.h from pch. --- Telegram/SourceFiles/apiwrap.cpp | 5 +- Telegram/SourceFiles/app.cpp | 3 +- Telegram/SourceFiles/app.h | 9 +- Telegram/SourceFiles/boxes/confirm_box.cpp | 1 + .../SourceFiles/boxes/edit_caption_box.cpp | 2 + Telegram/SourceFiles/boxes/edit_caption_box.h | 2 + .../SourceFiles/boxes/edit_privacy_box.cpp | 1 + .../boxes/peer_list_controllers.cpp | 6 + .../SourceFiles/boxes/peer_list_controllers.h | 4 +- Telegram/SourceFiles/boxes/share_box.cpp | 1 + .../calls/calls_box_controller.cpp | 2 + .../SourceFiles/chat_helpers/bot_keyboard.cpp | 1 + .../SourceFiles/data/data_channel_admins.cpp | 2 + Telegram/SourceFiles/data/data_document.cpp | 2 + Telegram/SourceFiles/data/data_drafts.cpp | 1 + Telegram/SourceFiles/data/data_feed.cpp | 1 + Telegram/SourceFiles/data/data_peer.cpp | 1 + .../data/data_search_controller.cpp | 1 + .../SourceFiles/data/data_search_controller.h | 1 + Telegram/SourceFiles/data/data_session.cpp | 18 +- Telegram/SourceFiles/data/data_session.h | 6 + .../SourceFiles/data/data_shared_media.cpp | 2 + Telegram/SourceFiles/data/data_types.h | 3 + .../SourceFiles/dialogs/dialogs_entry.cpp | 1 + .../dialogs/dialogs_indexed_list.cpp | 1 + .../dialogs/dialogs_inner_widget.cpp | 9 +- .../dialogs/dialogs_inner_widget.h | 4 +- Telegram/SourceFiles/dialogs/dialogs_key.cpp | 1 + .../SourceFiles/dialogs/dialogs_layout.cpp | 13 +- Telegram/SourceFiles/dialogs/dialogs_list.cpp | 1 + Telegram/SourceFiles/dialogs/dialogs_list.h | 2 + Telegram/SourceFiles/dialogs/dialogs_row.cpp | 5 + Telegram/SourceFiles/dialogs/dialogs_row.h | 4 +- .../SourceFiles/dialogs/dialogs_widget.cpp | 2 + Telegram/SourceFiles/dialogs/dialogs_widget.h | 1 + Telegram/SourceFiles/facades.cpp | 47 +- Telegram/SourceFiles/facades.h | 14 +- .../admin_log/history_admin_log_inner.cpp | 75 +- .../admin_log/history_admin_log_item.cpp | 1 + Telegram/SourceFiles/history/history.cpp | 49 +- Telegram/SourceFiles/history/history.h | 52 +- .../history/history_inner_widget.cpp | 103 +- .../history/history_inner_widget.h | 4 +- Telegram/SourceFiles/history/history_item.cpp | 218 +-- Telegram/SourceFiles/history/history_item.h | 170 +-- .../history/history_item_components.cpp | 10 +- .../SourceFiles/history/history_media.cpp | 24 + Telegram/SourceFiles/history/history_media.h | 66 +- .../history/history_media_grouped.cpp | 65 +- .../history/history_media_grouped.h | 5 +- .../history/history_media_types.cpp | 1301 +++++++++-------- .../SourceFiles/history/history_media_types.h | 152 +- .../SourceFiles/history/history_message.cpp | 1105 +------------- .../SourceFiles/history/history_message.h | 64 +- .../SourceFiles/history/history_service.cpp | 150 +- .../SourceFiles/history/history_service.h | 15 +- .../SourceFiles/history/history_widget.cpp | 66 +- Telegram/SourceFiles/history/history_widget.h | 8 +- .../history/view/history_view_element.cpp | 196 ++- .../history/view/history_view_element.h | 116 +- .../history/view/history_view_list_widget.cpp | 81 +- .../history/view/history_view_message.cpp | 1169 +++++++++++++++ .../history/view/history_view_message.h | 58 + .../history/view/history_view_object.h | 62 + .../view/history_view_service_message.cpp | 289 +++- .../view/history_view_service_message.h | 27 +- .../view/history_view_top_bar_widget.cpp | 9 +- .../info/media/info_media_list_widget.cpp | 1 + .../inline_bots/inline_bot_send_data.cpp | 1 + Telegram/SourceFiles/mainwidget.cpp | 7 +- Telegram/SourceFiles/mainwidget.h | 2 + Telegram/SourceFiles/mainwindow.cpp | 1 + .../media/player/media_player_float.cpp | 1 + .../media/player/media_player_instance.cpp | 6 +- .../media/player/media_player_panel.cpp | 2 + .../media/view/media_view_group_thumbs.cpp | 1 + Telegram/SourceFiles/mediaview.cpp | 3 +- Telegram/SourceFiles/messenger.cpp | 2 + Telegram/SourceFiles/mtproto/type_utils.h | 20 +- .../SourceFiles/overview/overview_layout.cpp | 1 + .../SourceFiles/overview/overview_layout.h | 1 + .../platform/win/main_window_win.cpp | 1 + .../win/notifications_manager_win.cpp | 1 + .../profile/profile_channel_controllers.cpp | 1 + .../settings/settings_privacy_controllers.cpp | 1 + Telegram/SourceFiles/storage/localstorage.cpp | 1 + Telegram/SourceFiles/ui/images.cpp | 1 + Telegram/SourceFiles/ui/text/text_entity.h | 25 + Telegram/SourceFiles/ui/text_options.cpp | 1 + Telegram/SourceFiles/window/main_window.cpp | 1 + .../window/notifications_manager.cpp | 5 +- .../window/notifications_manager_default.cpp | 3 +- .../SourceFiles/window/window_controller.cpp | 1 + .../SourceFiles/window/window_peer_menu.cpp | 1 + Telegram/gyp/telegram_sources.txt | 1 + 95 files changed, 3215 insertions(+), 2768 deletions(-) create mode 100644 Telegram/SourceFiles/history/view/history_view_object.h diff --git a/Telegram/SourceFiles/apiwrap.cpp b/Telegram/SourceFiles/apiwrap.cpp index 431e34122..195d1a905 100644 --- a/Telegram/SourceFiles/apiwrap.cpp +++ b/Telegram/SourceFiles/apiwrap.cpp @@ -20,6 +20,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "messenger.h" #include "mainwidget.h" #include "boxes/add_contact_box.h" +#include "history/history.h" #include "history/history_message.h" #include "history/history_media_types.h" #include "history/history_item_components.h" @@ -1726,7 +1727,7 @@ void ApiWrap::gotWebPages(ChannelData *channel, const MTPmessages_Messages &msgs v->at(index), NewMessageExisting); if (item) { - item->setPendingInitDimensions(); + Auth().data().requestItemViewResize(item); } } @@ -1738,7 +1739,7 @@ void ApiWrap::gotWebPages(ChannelData *channel, const MTPmessages_Messages &msgs auto j = items.constFind(i.key()); if (j != items.cend()) { for_const (auto item, j.value()) { - item->setPendingInitDimensions(); + Auth().data().requestItemViewResize(item); } } } diff --git a/Telegram/SourceFiles/app.cpp b/Telegram/SourceFiles/app.cpp index 87cb65941..9de0c9571 100644 --- a/Telegram/SourceFiles/app.cpp +++ b/Telegram/SourceFiles/app.cpp @@ -19,6 +19,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "lang/lang_keys.h" #include "data/data_abstract_structure.h" #include "data/data_session.h" +#include "history/history.h" #include "history/history_location_manager.h" #include "history/history_media_types.h" #include "history/history_item_components.h" @@ -1607,7 +1608,7 @@ namespace { auto i = items.constFind(result); if (i != items.cend()) { for_const (auto item, i.value()) { - item->setPendingInitDimensions(); + Auth().data().requestItemViewResize(item); } } } diff --git a/Telegram/SourceFiles/app.h b/Telegram/SourceFiles/app.h index b0ad996f2..fe1d48fa1 100644 --- a/Telegram/SourceFiles/app.h +++ b/Telegram/SourceFiles/app.h @@ -8,17 +8,20 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #pragma once #include "core/basic_types.h" -#include "history/history.h" +#include "data/data_types.h" +#include "data/data_peer.h" +enum NewMessageType : char; class Messenger; class MainWindow; class MainWidget; class LocationCoords; struct LocationData; class HistoryItem; - +class History; +class Histories; namespace HistoryView { -class Message; +class Element; } // namespace HistoryView using HistoryItemsMap = base::flat_set>; diff --git a/Telegram/SourceFiles/boxes/confirm_box.cpp b/Telegram/SourceFiles/boxes/confirm_box.cpp index 20d58e425..fe28fad39 100644 --- a/Telegram/SourceFiles/boxes/confirm_box.cpp +++ b/Telegram/SourceFiles/boxes/confirm_box.cpp @@ -13,6 +13,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "mainwindow.h" #include "apiwrap.h" #include "application.h" +#include "history/history.h" #include "history/history_item.h" #include "ui/widgets/checkbox.h" #include "ui/widgets/buttons.h" diff --git a/Telegram/SourceFiles/boxes/edit_caption_box.cpp b/Telegram/SourceFiles/boxes/edit_caption_box.cpp index 8371a3616..6fd79fa0a 100644 --- a/Telegram/SourceFiles/boxes/edit_caption_box.cpp +++ b/Telegram/SourceFiles/boxes/edit_caption_box.cpp @@ -10,6 +10,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "ui/widgets/input_fields.h" #include "ui/text_options.h" #include "media/media_clip_reader.h" +#include "history/history.h" +#include "history/history_item.h" #include "history/history_media_types.h" #include "lang/lang_keys.h" #include "window/window_controller.h" diff --git a/Telegram/SourceFiles/boxes/edit_caption_box.h b/Telegram/SourceFiles/boxes/edit_caption_box.h index 829fd6529..2b2a63cfb 100644 --- a/Telegram/SourceFiles/boxes/edit_caption_box.h +++ b/Telegram/SourceFiles/boxes/edit_caption_box.h @@ -9,6 +9,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "boxes/abstract_box.h" +class HistoryMedia; + namespace Ui { class InputArea; } // namespace Ui diff --git a/Telegram/SourceFiles/boxes/edit_privacy_box.cpp b/Telegram/SourceFiles/boxes/edit_privacy_box.cpp index ca076ef1e..536257bfc 100644 --- a/Telegram/SourceFiles/boxes/edit_privacy_box.cpp +++ b/Telegram/SourceFiles/boxes/edit_privacy_box.cpp @@ -12,6 +12,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "ui/widgets/labels.h" #include "ui/widgets/buttons.h" #include "ui/wrap/slide_wrap.h" +#include "history/history.h" #include "boxes/peer_list_controllers.h" #include "apiwrap.h" #include "auth_session.h" diff --git a/Telegram/SourceFiles/boxes/peer_list_controllers.cpp b/Telegram/SourceFiles/boxes/peer_list_controllers.cpp index 4bad86635..4cdb9126b 100644 --- a/Telegram/SourceFiles/boxes/peer_list_controllers.cpp +++ b/Telegram/SourceFiles/boxes/peer_list_controllers.cpp @@ -17,6 +17,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "apiwrap.h" #include "mainwidget.h" #include "lang/lang_keys.h" +#include "history/history.h" #include "dialogs/dialogs_indexed_list.h" namespace { @@ -208,6 +209,11 @@ bool PeerListGlobalSearchController::isLoading() { return _timer.isActive() || _requestId; } +ChatsListBoxController::Row::Row(not_null history) +: PeerListRow(history->peer) +, _history(history) { +} + ChatsListBoxController::ChatsListBoxController( std::unique_ptr searchController) : PeerListController(std::move(searchController)) { diff --git a/Telegram/SourceFiles/boxes/peer_list_controllers.h b/Telegram/SourceFiles/boxes/peer_list_controllers.h index d9db02b77..de241040d 100644 --- a/Telegram/SourceFiles/boxes/peer_list_controllers.h +++ b/Telegram/SourceFiles/boxes/peer_list_controllers.h @@ -91,8 +91,8 @@ public: protected: class Row : public PeerListRow { public: - Row(not_null history) : PeerListRow(history->peer), _history(history) { - } + Row(not_null history); + not_null history() const { return _history; } diff --git a/Telegram/SourceFiles/boxes/share_box.cpp b/Telegram/SourceFiles/boxes/share_box.cpp index 9550204f4..ae65c87ce 100644 --- a/Telegram/SourceFiles/boxes/share_box.cpp +++ b/Telegram/SourceFiles/boxes/share_box.cpp @@ -23,6 +23,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "ui/widgets/buttons.h" #include "ui/widgets/scroll_area.h" #include "ui/text_options.h" +#include "history/history.h" #include "history/history_media_types.h" #include "history/history_message.h" #include "window/themes/window_theme.h" diff --git a/Telegram/SourceFiles/calls/calls_box_controller.cpp b/Telegram/SourceFiles/calls/calls_box_controller.cpp index 2ade7bd33..313067412 100644 --- a/Telegram/SourceFiles/calls/calls_box_controller.cpp +++ b/Telegram/SourceFiles/calls/calls_box_controller.cpp @@ -13,6 +13,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "observer_peer.h" #include "ui/effects/ripple_animation.h" #include "calls/calls_instance.h" +#include "history/history.h" +#include "history/history_item.h" #include "history/history_media_types.h" #include "mainwidget.h" #include "auth_session.h" diff --git a/Telegram/SourceFiles/chat_helpers/bot_keyboard.cpp b/Telegram/SourceFiles/chat_helpers/bot_keyboard.cpp index b48155d11..5978a6411 100644 --- a/Telegram/SourceFiles/chat_helpers/bot_keyboard.cpp +++ b/Telegram/SourceFiles/chat_helpers/bot_keyboard.cpp @@ -7,6 +7,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL */ #include "chat_helpers/bot_keyboard.h" +#include "history/history.h" #include "history/history_item_components.h" #include "styles/style_widgets.h" #include "styles/style_history.h" diff --git a/Telegram/SourceFiles/data/data_channel_admins.cpp b/Telegram/SourceFiles/data/data_channel_admins.cpp index 1dc5d1e74..cc3ab3a7d 100644 --- a/Telegram/SourceFiles/data/data_channel_admins.cpp +++ b/Telegram/SourceFiles/data/data_channel_admins.cpp @@ -7,6 +7,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL */ #include "data/data_channel_admins.h" +#include "history/history.h" + namespace Data { ChannelAdminChanges::ChannelAdminChanges(not_null channel) diff --git a/Telegram/SourceFiles/data/data_document.cpp b/Telegram/SourceFiles/data/data_document.cpp index ffe275015..38f1507b7 100644 --- a/Telegram/SourceFiles/data/data_document.cpp +++ b/Telegram/SourceFiles/data/data_document.cpp @@ -15,6 +15,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "media/media_audio.h" #include "storage/localstorage.h" #include "platform/platform_specific.h" +#include "history/history.h" +#include "history/history_item.h" #include "history/history_media_types.h" #include "auth_session.h" #include "messenger.h" diff --git a/Telegram/SourceFiles/data/data_drafts.cpp b/Telegram/SourceFiles/data/data_drafts.cpp index d66e754c3..8c7dd19cd 100644 --- a/Telegram/SourceFiles/data/data_drafts.cpp +++ b/Telegram/SourceFiles/data/data_drafts.cpp @@ -9,6 +9,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "ui/widgets/input_fields.h" #include "chat_helpers/message_field.h" +#include "history/history.h" #include "history/history_widget.h" #include "mainwidget.h" #include "storage/localstorage.h" diff --git a/Telegram/SourceFiles/data/data_feed.cpp b/Telegram/SourceFiles/data/data_feed.cpp index f65e7f75b..a676c7b4d 100644 --- a/Telegram/SourceFiles/data/data_feed.cpp +++ b/Telegram/SourceFiles/data/data_feed.cpp @@ -8,6 +8,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "data/data_feed.h" #include "dialogs/dialogs_key.h" +#include "history/history.h" #include "history/history_item.h" namespace Data { diff --git a/Telegram/SourceFiles/data/data_peer.cpp b/Telegram/SourceFiles/data/data_peer.cpp index fbcc40d2c..1df8f58eb 100644 --- a/Telegram/SourceFiles/data/data_peer.cpp +++ b/Telegram/SourceFiles/data/data_peer.cpp @@ -14,6 +14,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "data/data_photo.h" #include "data/data_feed.h" #include "data/data_session.h" +#include "history/history.h" #include "lang/lang_keys.h" #include "observer_peer.h" #include "mainwidget.h" diff --git a/Telegram/SourceFiles/data/data_search_controller.cpp b/Telegram/SourceFiles/data/data_search_controller.cpp index 894a6cd03..8cfbda2d0 100644 --- a/Telegram/SourceFiles/data/data_search_controller.cpp +++ b/Telegram/SourceFiles/data/data_search_controller.cpp @@ -10,6 +10,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "auth_session.h" #include "data/data_session.h" #include "data/data_messages.h" +#include "history/history.h" #include "history/history_item.h" namespace Api { diff --git a/Telegram/SourceFiles/data/data_search_controller.h b/Telegram/SourceFiles/data/data_search_controller.h index 222a984d0..ec31dc66b 100644 --- a/Telegram/SourceFiles/data/data_search_controller.h +++ b/Telegram/SourceFiles/data/data_search_controller.h @@ -12,6 +12,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "storage/storage_sparse_ids_list.h" #include "storage/storage_shared_media.h" #include "base/value_ordering.h" +#include "base/timer.h" namespace Data { enum class LoadDirection : char; diff --git a/Telegram/SourceFiles/data/data_session.cpp b/Telegram/SourceFiles/data/data_session.cpp index ca47a347f..ef2fb650a 100644 --- a/Telegram/SourceFiles/data/data_session.cpp +++ b/Telegram/SourceFiles/data/data_session.cpp @@ -43,6 +43,22 @@ rpl::producer> Session::itemRepaintRequest() const return _itemRepaintRequest.events(); } +void Session::requestItemViewResize(not_null item) { + _itemViewResizeRequest.fire_copy(item); +} + +rpl::producer> Session::itemViewResizeRequest() const { + return _itemViewResizeRequest.events(); +} + +void Session::requestItemViewRefresh(not_null item) { + _itemViewRefreshRequest.fire_copy(item); +} + +rpl::producer> Session::itemViewRefreshRequest() const { + return _itemViewRefreshRequest.events(); +} + void Session::markItemRemoved(not_null item) { _itemRemoved.fire_copy(item); } @@ -130,7 +146,7 @@ void Session::userIsContactUpdated(not_null user) { const auto i = items.constFind(peerToUser(user->id)); if (i != items.cend()) { for (const auto item : std::as_const(i.value())) { - item->setPendingInitDimensions(); + requestItemViewResize(item); } } } diff --git a/Telegram/SourceFiles/data/data_session.h b/Telegram/SourceFiles/data/data_session.h index 4cb9d4b89..b776a290f 100644 --- a/Telegram/SourceFiles/data/data_session.h +++ b/Telegram/SourceFiles/data/data_session.h @@ -44,6 +44,10 @@ public: rpl::producer> itemLayoutChanged() const; void requestItemRepaint(not_null item); rpl::producer> itemRepaintRequest() const; + void requestItemViewResize(not_null item); + rpl::producer> itemViewResizeRequest() const; + void requestItemViewRefresh(not_null item); + rpl::producer> itemViewRefreshRequest() const; void markItemRemoved(not_null item); rpl::producer> itemRemoved() const; void markHistoryUnloaded(not_null history); @@ -179,6 +183,8 @@ private: base::Observable _queryItemVisibility; rpl::event_stream> _itemLayoutChanged; rpl::event_stream> _itemRepaintRequest; + rpl::event_stream> _itemViewResizeRequest; + rpl::event_stream> _itemViewRefreshRequest; rpl::event_stream> _itemRemoved; rpl::event_stream> _historyUnloaded; rpl::event_stream> _historyCleared; diff --git a/Telegram/SourceFiles/data/data_shared_media.cpp b/Telegram/SourceFiles/data/data_shared_media.cpp index a45013125..4493d13b4 100644 --- a/Telegram/SourceFiles/data/data_shared_media.cpp +++ b/Telegram/SourceFiles/data/data_shared_media.cpp @@ -12,6 +12,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "apiwrap.h" #include "storage/storage_facade.h" #include "storage/storage_shared_media.h" +#include "history/history.h" +#include "history/history_item.h" #include "history/history_media_types.h" #include "data/data_sparse_ids.h" #include "info/info_memento.h" diff --git a/Telegram/SourceFiles/data/data_types.h b/Telegram/SourceFiles/data/data_types.h index 880cbb647..1410a6707 100644 --- a/Telegram/SourceFiles/data/data_types.h +++ b/Telegram/SourceFiles/data/data_types.h @@ -7,6 +7,9 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL */ #pragma once +class HistoryItem; +using HistoryItemsList = std::vector>; + namespace Data { struct UploadState { diff --git a/Telegram/SourceFiles/dialogs/dialogs_entry.cpp b/Telegram/SourceFiles/dialogs/dialogs_entry.cpp index 28e4f0b60..ab2d15937 100644 --- a/Telegram/SourceFiles/dialogs/dialogs_entry.cpp +++ b/Telegram/SourceFiles/dialogs/dialogs_entry.cpp @@ -12,6 +12,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "mainwidget.h" #include "styles/style_dialogs.h" #include "history/history_item.h" +#include "history/history.h" namespace Dialogs { namespace { diff --git a/Telegram/SourceFiles/dialogs/dialogs_indexed_list.cpp b/Telegram/SourceFiles/dialogs/dialogs_indexed_list.cpp index 0a5461d7c..a7bc336f1 100644 --- a/Telegram/SourceFiles/dialogs/dialogs_indexed_list.cpp +++ b/Telegram/SourceFiles/dialogs/dialogs_indexed_list.cpp @@ -9,6 +9,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "auth_session.h" #include "data/data_session.h" +#include "history/history.h" namespace Dialogs { diff --git a/Telegram/SourceFiles/dialogs/dialogs_inner_widget.cpp b/Telegram/SourceFiles/dialogs/dialogs_inner_widget.cpp index 49db8f558..e257d8ea1 100644 --- a/Telegram/SourceFiles/dialogs/dialogs_inner_widget.cpp +++ b/Telegram/SourceFiles/dialogs/dialogs_inner_widget.cpp @@ -11,6 +11,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "dialogs/dialogs_layout.h" #include "dialogs/dialogs_search_from_controllers.h" #include "history/feed/history_feed_section.h" +#include "history/history.h" #include "history/history_item.h" #include "styles/style_dialogs.h" #include "styles/style_chat_helpers.h" @@ -323,7 +324,7 @@ void DialogsInner::paintRegion(Painter &p, const QRegion ®ion, bool paintingO const auto row = _filterResults[from]; const auto key = row->key(); const auto history = key.history(); - const auto peer = history ? history->peer : nullptr; + const auto peer = history ? history->peer.get() : nullptr; const auto active = !activeMsgId && peer && activePeer @@ -1356,6 +1357,12 @@ void DialogsInner::updateSelectedRow(Dialogs::Key key) { } } +Dialogs::IndexedList *DialogsInner::shownDialogs() const { + return (Global::DialogsMode() == Dialogs::Mode::Important) + ? _dialogsImportant.get() + : _dialogs.get(); +} + void DialogsInner::leaveEventHook(QEvent *e) { setMouseTracking(false); clearSelection(); diff --git a/Telegram/SourceFiles/dialogs/dialogs_inner_widget.h b/Telegram/SourceFiles/dialogs/dialogs_inner_widget.h index f7f7f031b..b770dcf27 100644 --- a/Telegram/SourceFiles/dialogs/dialogs_inner_widget.h +++ b/Telegram/SourceFiles/dialogs/dialogs_inner_widget.h @@ -244,9 +244,7 @@ private: void clearSearchResults(bool clearPeerSearchResults = true); void updateSelectedRow(Dialogs::Key key = Dialogs::Key()); - Dialogs::IndexedList *shownDialogs() const { - return (Global::DialogsMode() == Dialogs::Mode::Important) ? _dialogsImportant.get() : _dialogs.get(); - } + Dialogs::IndexedList *shownDialogs() const; void checkReorderPinnedStart(QPoint localPosition); int shownPinnedCount() const; diff --git a/Telegram/SourceFiles/dialogs/dialogs_key.cpp b/Telegram/SourceFiles/dialogs/dialogs_key.cpp index 93375427f..bd7a2127c 100644 --- a/Telegram/SourceFiles/dialogs/dialogs_key.cpp +++ b/Telegram/SourceFiles/dialogs/dialogs_key.cpp @@ -8,6 +8,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "dialogs/dialogs_key.h" #include "data/data_feed.h" +#include "history/history.h" namespace Dialogs { diff --git a/Telegram/SourceFiles/dialogs/dialogs_layout.cpp b/Telegram/SourceFiles/dialogs/dialogs_layout.cpp index 2f129986f..298c14971 100644 --- a/Telegram/SourceFiles/dialogs/dialogs_layout.cpp +++ b/Telegram/SourceFiles/dialogs/dialogs_layout.cpp @@ -16,6 +16,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "ui/text_options.h" #include "lang/lang_keys.h" #include "history/history_item.h" +#include "history/history.h" namespace Dialogs { namespace Layout { @@ -355,7 +356,7 @@ void RowPainter::paint( TimeMs ms) { const auto entry = row->entry(); const auto history = row->history(); - const auto peer = history ? history->peer : nullptr; + const auto peer = history ? history->peer.get() : nullptr; const auto unreadCount = entry->chatListUnreadCount(); const auto unreadMuted = entry->chatListMutedBadge(); const auto item = entry->chatsListItem(); @@ -385,7 +386,7 @@ void RowPainter::paint( const auto from = history ? (history->peer->migrateTo() ? history->peer->migrateTo() - : history->peer) + : history->peer.get()) : nullptr; const auto flags = (active ? Flag::Active : Flag(0)) | (selected ? Flag::Selected : Flag(0)) @@ -520,12 +521,14 @@ void RowPainter::paint( const auto from = [&] { if (auto searchPeer = row->searchInPeer()) { if (searchPeer->isSelf()) { - return item->senderOriginal(); + return item->senderOriginal().get(); } else if (!searchPeer->isChannel() || searchPeer->isMegagroup()) { - return item->from(); + return item->from().get(); } } - return (history->peer->migrateTo() ? history->peer->migrateTo() : history->peer); + return history->peer->migrateTo() + ? history->peer->migrateTo() + : history->peer.get(); }(); const auto drawInDialogWay = [&] { if (auto searchPeer = row->searchInPeer()) { diff --git a/Telegram/SourceFiles/dialogs/dialogs_list.cpp b/Telegram/SourceFiles/dialogs/dialogs_list.cpp index de685718b..e72e3b4bd 100644 --- a/Telegram/SourceFiles/dialogs/dialogs_list.cpp +++ b/Telegram/SourceFiles/dialogs/dialogs_list.cpp @@ -7,6 +7,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL */ #include "dialogs/dialogs_list.h" +#include "dialogs/dialogs_entry.h" #include "dialogs/dialogs_layout.h" #include "styles/style_dialogs.h" #include "mainwidget.h" diff --git a/Telegram/SourceFiles/dialogs/dialogs_list.h b/Telegram/SourceFiles/dialogs/dialogs_list.h index d32dcde85..bd5596215 100644 --- a/Telegram/SourceFiles/dialogs/dialogs_list.h +++ b/Telegram/SourceFiles/dialogs/dialogs_list.h @@ -12,6 +12,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL class PeerData; namespace Dialogs { +enum class SortMode; + class List { public: List(SortMode sortMode); diff --git a/Telegram/SourceFiles/dialogs/dialogs_row.cpp b/Telegram/SourceFiles/dialogs/dialogs_row.cpp index ee4f46bd8..fd1b08edc 100644 --- a/Telegram/SourceFiles/dialogs/dialogs_row.cpp +++ b/Telegram/SourceFiles/dialogs/dialogs_row.cpp @@ -9,6 +9,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "styles/style_dialogs.h" #include "ui/effects/ripple_animation.h" +#include "dialogs/dialogs_entry.h" #include "mainwidget.h" namespace Dialogs { @@ -39,6 +40,10 @@ void RippleRow::paintRipple(Painter &p, int x, int y, int outerWidth, TimeMs ms, } } +uint64 Row::sortKey() const { + return _id.entry()->sortKeyInChatList(); +} + FakeRow::FakeRow(PeerData *searchInPeer, not_null item) : _searchInPeer(searchInPeer) , _item(item) diff --git a/Telegram/SourceFiles/dialogs/dialogs_row.h b/Telegram/SourceFiles/dialogs/dialogs_row.h index b7bf6a6f5..d87b7a884 100644 --- a/Telegram/SourceFiles/dialogs/dialogs_row.h +++ b/Telegram/SourceFiles/dialogs/dialogs_row.h @@ -67,9 +67,7 @@ public: int pos() const { return _pos; } - uint64 sortKey() const { - return _id.entry()->sortKeyInChatList(); - } + uint64 sortKey() const; // for any attached data, for example View in contacts list void *attached = nullptr; diff --git a/Telegram/SourceFiles/dialogs/dialogs_widget.cpp b/Telegram/SourceFiles/dialogs/dialogs_widget.cpp index b8b577a7e..7f99543e7 100644 --- a/Telegram/SourceFiles/dialogs/dialogs_widget.cpp +++ b/Telegram/SourceFiles/dialogs/dialogs_widget.cpp @@ -10,6 +10,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "dialogs/dialogs_inner_widget.h" #include "dialogs/dialogs_search_from_controllers.h" #include "dialogs/dialogs_key.h" +#include "dialogs/dialogs_entry.h" +#include "history/history.h" #include "ui/widgets/buttons.h" #include "ui/widgets/input_fields.h" #include "ui/wrap/fade_wrap.h" diff --git a/Telegram/SourceFiles/dialogs/dialogs_widget.h b/Telegram/SourceFiles/dialogs/dialogs_widget.h index 8ded93941..b1c8774d3 100644 --- a/Telegram/SourceFiles/dialogs/dialogs_widget.h +++ b/Telegram/SourceFiles/dialogs/dialogs_widget.h @@ -17,6 +17,7 @@ struct RowDescriptor; class Row; class FakeRow; class IndexedList; +class Key; } // namespace Dialogs namespace Ui { diff --git a/Telegram/SourceFiles/facades.cpp b/Telegram/SourceFiles/facades.cpp index 332da2957..d03281d19 100644 --- a/Telegram/SourceFiles/facades.cpp +++ b/Telegram/SourceFiles/facades.cpp @@ -22,6 +22,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "window/layer_widget.h" #include "lang/lang_keys.h" #include "base/observer.h" +#include "history/history.h" #include "history/history_item.h" #include "history/history_media.h" #include "styles/style_history.h" @@ -258,6 +259,10 @@ void showPeerProfile(const PeerId &peer) { } } +void showPeerProfile(not_null history) { + showPeerProfile(history->peer->id); +} + void showPeerHistory( const PeerId &peer, MsgId msgId) { @@ -274,6 +279,10 @@ void showPeerHistoryAtItem(not_null item) { showPeerHistory(item->history()->peer->id, item->id); } +void showPeerHistory(not_null history, MsgId msgId) { + showPeerHistory(history->peer->id, msgId); +} + PeerData *getPeerForMouseAction() { return Messenger::Instance().ui_getPeerForMouseAction(); } @@ -343,25 +352,25 @@ void handlePendingHistoryUpdate() { Auth().data().requestItemRepaint(item); // Start the video if it is waiting for that. - if (item->pendingInitDimensions()) { - if (const auto media = item->getMedia()) { - if (const auto reader = media->getClipReader()) { - const auto startRequired = [&] { - if (reader->started()) { - return false; - } - using Mode = Media::Clip::Reader::Mode; - return (reader->mode() == Mode::Video); - }; - if (startRequired()) { - const auto width = std::max( - item->width(), - st::historyMinimalWidth); - item->resizeGetHeight(width); - } - } - } - } + //if (item->pendingInitDimensions()) { // #TODO floating player video + // if (const auto media = item->getMedia()) { + // if (const auto reader = media->getClipReader()) { + // const auto startRequired = [&] { + // if (reader->started()) { + // return false; + // } + // using Mode = Media::Clip::Reader::Mode; + // return (reader->mode() == Mode::Video); + // }; + // if (startRequired()) { + // const auto width = std::max( + // item->width(), + // st::historyMinimalWidth); + // item->resizeGetHeight(width); + // } + // } + // } + //} } } diff --git a/Telegram/SourceFiles/facades.h b/Telegram/SourceFiles/facades.h index a762c52f5..1ac8d1472 100644 --- a/Telegram/SourceFiles/facades.h +++ b/Telegram/SourceFiles/facades.h @@ -12,6 +12,10 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL class BoxContent; +namespace Dialogs { +enum class Mode; +} // namespace Dialogs + namespace InlineBots { namespace Layout { class ItemBase; @@ -184,9 +188,7 @@ void showPeerProfile(const PeerId &peer); inline void showPeerProfile(const PeerData *peer) { showPeerProfile(peer->id); } -inline void showPeerProfile(const History *history) { - showPeerProfile(history->peer->id); -} +void showPeerProfile(not_null history); void showPeerHistory(const PeerId &peer, MsgId msgId); void showPeerHistoryAtItem(not_null item); @@ -194,11 +196,7 @@ void showPeerHistoryAtItem(not_null item); inline void showPeerHistory(const PeerData *peer, MsgId msgId) { showPeerHistory(peer->id, msgId); } -inline void showPeerHistory( - const History *history, - MsgId msgId) { - showPeerHistory(history->peer->id, msgId); -} +void showPeerHistory(not_null history, MsgId msgId); inline void showChatsList() { showPeerHistory(PeerId(0), 0); } diff --git a/Telegram/SourceFiles/history/admin_log/history_admin_log_inner.cpp b/Telegram/SourceFiles/history/admin_log/history_admin_log_inner.cpp index 592cd4c42..036da17c2 100644 --- a/Telegram/SourceFiles/history/admin_log/history_admin_log_inner.cpp +++ b/Telegram/SourceFiles/history/admin_log/history_admin_log_inner.cpp @@ -8,6 +8,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "history/admin_log/history_admin_log_inner.h" #include "styles/style_history.h" +#include "history/history.h" #include "history/history_media_types.h" #include "history/history_message.h" #include "history/history_item_components.h" @@ -56,16 +57,16 @@ void InnerWidget::enumerateItems(Method method) { auto begin = std::rbegin(_items), end = std::rend(_items); auto from = TopToBottom ? std::lower_bound(begin, end, _visibleTop, [this](auto &elem, int top) { - return this->itemTop(elem) + elem->data()->height() <= top; + return this->itemTop(elem) + elem->height() <= top; }) : std::upper_bound(begin, end, _visibleBottom, [this](int bottom, auto &elem) { - return this->itemTop(elem) + elem->data()->height() >= bottom; + return this->itemTop(elem) + elem->height() >= bottom; }); auto wasEnd = (from == end); if (wasEnd) { --from; } if (TopToBottom) { - Assert(itemTop(from->get()) + from->get()->data()->height() > _visibleTop); + Assert(itemTop(from->get()) + from->get()->height() > _visibleTop); } else { Assert(itemTop(from->get()) < _visibleBottom); } @@ -73,7 +74,7 @@ void InnerWidget::enumerateItems(Method method) { while (true) { auto item = from->get(); auto itemtop = itemTop(item); - auto itembottom = itemtop + item->data()->height(); + auto itembottom = itemtop + item->height(); // Binary search should've skipped all the items that are above / below the visible area. if (TopToBottom) { @@ -121,19 +122,19 @@ void InnerWidget::enumerateUserpics(Method method) { const auto message = view->data()->toHistoryMessage(); if (!message) return true; - if (lowestAttachedItemTop < 0 && message->isAttachedToNext()) { - lowestAttachedItemTop = itemtop + message->marginTop(); + if (lowestAttachedItemTop < 0 && view->isAttachedToNext()) { + lowestAttachedItemTop = itemtop + view->marginTop(); } // Call method on a userpic for all messages that have it and for those who are not showing it // because of their attachment to the next message if they are bottom-most visible. - if (message->displayFromPhoto() || (message->hasFromPhoto() && itembottom >= _visibleBottom)) { + if (view->displayFromPhoto() || (view->hasFromPhoto() && itembottom >= _visibleBottom)) { if (lowestAttachedItemTop < 0) { - lowestAttachedItemTop = itemtop + message->marginTop(); + lowestAttachedItemTop = itemtop + view->marginTop(); } // Attach userpic to the bottom of the visible area with the same margin as the last message. auto userpicMinBottomSkip = st::historyPaddingBottom + st::msgMargin.bottom(); - auto userpicBottom = qMin(itembottom - message->marginBottom(), _visibleBottom - userpicMinBottomSkip); + auto userpicBottom = qMin(itembottom - view->marginBottom(), _visibleBottom - userpicMinBottomSkip); // Do not let the userpic go above the attached messages pack top line. userpicBottom = qMax(userpicBottom, lowestAttachedItemTop + st::msgPhotoSize); @@ -146,7 +147,7 @@ void InnerWidget::enumerateUserpics(Method method) { } // Forget the found top of the pack, search for the next one from scratch. - if (!message->isAttachedToNext()) { + if (!view->isAttachedToNext()) { lowestAttachedItemTop = -1; } @@ -165,14 +166,14 @@ void InnerWidget::enumerateDates(Method method) { auto dateCallback = [&](not_null view, int itemtop, int itembottom) { const auto item = view->data(); if (lowestInOneDayItemBottom < 0 && item->isInOneDayWithPrevious()) { - lowestInOneDayItemBottom = itembottom - item->marginBottom(); + lowestInOneDayItemBottom = itembottom - view->marginBottom(); } // Call method on a date for all messages that have it and for those who are not showing it // because they are in a one day together with the previous message if they are top-most visible. if (item->displayDate() || (!item->isEmpty() && itemtop <= _visibleTop)) { if (lowestInOneDayItemBottom < 0) { - lowestInOneDayItemBottom = itembottom - item->marginBottom(); + lowestInOneDayItemBottom = itembottom - view->marginBottom(); } // Attach date to the top of the visible area with the same margin as it has in service message. auto dateTop = qMax(itemtop, _visibleTop) + st::msgServiceMargin.top(); @@ -224,7 +225,7 @@ InnerWidget::InnerWidget( } if (const auto view = viewForItem(query.item)) { auto top = itemTop(view); - if (top >= 0 && top + query.item->height() > _visibleTop && top < _visibleBottom) { + if (top >= 0 && top + view->height() > _visibleTop && top < _visibleBottom) { *query.isVisible = true; } } @@ -257,7 +258,7 @@ void InnerWidget::updateVisibleTopItem() { } else { auto begin = std::rbegin(_items), end = std::rend(_items); auto from = std::lower_bound(begin, end, _visibleTop, [this](auto &&elem, int top) { - return this->itemTop(elem) + elem->data()->height() <= top; + return this->itemTop(elem) + elem->height() <= top; }); if (from != end) { _visibleTopItem = *from; @@ -589,15 +590,15 @@ void InnerWidget::itemsAdded(Direction direction, int addedCount) { auto checkTo = (direction == Direction::Up) ? (_items.size() + 1) : (addedCount + 1); for (auto i = checkFrom; i != checkTo; ++i) { if (i > 0) { - const auto item = _items[i - 1]->data(); + const auto view = _items[i - 1].get(); if (i < _items.size()) { - const auto previous = _items[i]->data(); - item->setLogEntryDisplayDate(item->date.date() != previous->date.date()); - auto attachToPrevious = item->computeIsAttachToPrevious(previous); - item->setLogEntryAttachToPrevious(attachToPrevious); - previous->setLogEntryAttachToNext(attachToPrevious); + const auto previous = _items[i].get(); + view->setDisplayDate(view->data()->date.date() != previous->data()->date.date()); + const auto attach = view->computeIsAttachToPrevious(previous); + view->setAttachToPrevious(attach); + previous->setAttachToNext(attach); } else { - item->setLogEntryDisplayDate(true); + view->setDisplayDate(true); } } } @@ -617,7 +618,7 @@ int InnerWidget::resizeGetHeight(int newWidth) { auto newHeight = 0; for (auto &item : base::reversed(_items)) { item->setY(newHeight); - newHeight += item->data()->resizeGetHeight(newWidth); + newHeight += item->resizeGetHeight(newWidth); } _itemsHeight = newHeight; _itemsTop = (_minHeight > _itemsHeight + st::historyPaddingBottom) ? (_minHeight - _itemsHeight - st::historyPaddingBottom) : 0; @@ -646,7 +647,7 @@ void InnerWidget::paintEvent(QPaintEvent *e) { } else { auto begin = std::rbegin(_items), end = std::rend(_items); auto from = std::lower_bound(begin, end, clip.top(), [this](auto &elem, int top) { - return this->itemTop(elem) + elem->data()->height() <= top; + return this->itemTop(elem) + elem->height() <= top; }); auto to = std::lower_bound(begin, end, clip.top() + clip.height(), [this](auto &elem, int bottom) { return this->itemTop(elem) < bottom; @@ -659,9 +660,9 @@ void InnerWidget::paintEvent(QPaintEvent *e) { const auto selection = (view == _selectedItem) ? _selectedText : TextSelection(); + view->draw(p, clip.translated(0, -top), selection, ms); const auto item = view->data(); - item->draw(p, clip.translated(0, -top), selection, ms); - auto height = item->height(); + auto height = view->height(); top += height; p.translate(0, height); } @@ -678,7 +679,7 @@ void InnerWidget::paintEvent(QPaintEvent *e) { const auto message = view->data()->toHistoryMessage(); Assert(message != nullptr); - message->from()->paintUserpicLeft(p, st::historyPhotoLeft, userpicTop, message->width(), st::msgPhotoSize); + message->from()->paintUserpicLeft(p, st::historyPhotoLeft, userpicTop, view->width(), st::msgPhotoSize); } return true; }); @@ -711,7 +712,7 @@ void InnerWidget::paintEvent(QPaintEvent *e) { if (opacity > 0.) { p.setOpacity(opacity); int dateY = /*noFloatingDate ? itemtop :*/ (dateTop - st::msgServiceMargin.top()); - int width = item->width(); + int width = view->width(); if (auto date = item->Get()) { date->paint(p, dateY, width); } else { @@ -797,7 +798,7 @@ void InnerWidget::mouseDoubleClickEvent(QMouseEvent *e) { if (((_mouseAction == MouseAction::Selecting && _selectedItem != nullptr) || (_mouseAction == MouseAction::None)) && _mouseSelectType == TextSelectType::Letters && _mouseActionItem) { HistoryStateRequest request; request.flags |= Text::StateRequest::Flag::LookupSymbol; - auto dragState = _mouseActionItem->data()->getState(_dragStartPosition, request); + auto dragState = _mouseActionItem->getState(_dragStartPosition, request); if (dragState.cursor == HistoryInTextCursorState) { _mouseTextSymbol = dragState.symbol; _mouseSelectType = TextSelectType::Words; @@ -839,7 +840,7 @@ void InnerWidget::showContextMenu(QContextMenuEvent *e, bool showFromTouch) { App::mousedItem()); HistoryStateRequest request; request.flags |= Text::StateRequest::Flag::LookupSymbol; - auto dragState = App::mousedItem()->data()->getState(mousePos, request); + auto dragState = App::mousedItem()->getState(mousePos, request); if (dragState.cursor == HistoryInTextCursorState && dragState.symbol >= selFrom && dragState.symbol < selTo) { isUponSelected = 1; } @@ -1214,7 +1215,7 @@ void InnerWidget::mouseActionStart(const QPoint &screenPos, Qt::MouseButton butt if (_trippleClickTimer.isActive() && (screenPos - _trippleClickPoint).manhattanLength() < QApplication::startDragDistance()) { HistoryStateRequest request; request.flags = Text::StateRequest::Flag::LookupSymbol; - dragState = _mouseActionItem->data()->getState(_dragStartPosition, request); + dragState = _mouseActionItem->getState(_dragStartPosition, request); if (dragState.cursor == HistoryInTextCursorState) { auto selection = TextSelection { dragState.symbol, dragState.symbol }; repaintItem(std::exchange(_selectedItem, _mouseActionItem)); @@ -1228,7 +1229,7 @@ void InnerWidget::mouseActionStart(const QPoint &screenPos, Qt::MouseButton butt } else if (App::pressedItem()) { HistoryStateRequest request; request.flags = Text::StateRequest::Flag::LookupSymbol; - dragState = _mouseActionItem->data()->getState(_dragStartPosition, request); + dragState = _mouseActionItem->getState(_dragStartPosition, request); } if (_mouseSelectType != TextSelectType::Paragraphs) { if (App::pressedItem()) { @@ -1324,7 +1325,7 @@ void InnerWidget::updateSelected() { auto begin = std::rbegin(_items), end = std::rend(_items); auto from = (point.y() >= _itemsTop && point.y() < _itemsTop + _itemsHeight) ? std::lower_bound(begin, end, point.y(), [this](auto &elem, int top) { - return this->itemTop(elem) + elem->data()->height() <= top; + return this->itemTop(elem) + elem->height() <= top; }) : end; const auto view = (from != end) ? from->get() : nullptr; @@ -1332,7 +1333,7 @@ void InnerWidget::updateSelected() { if (item) { App::mousedItem(view); itemPoint = mapPointToItem(point, view); - if (item->hasPoint(itemPoint)) { + if (view->hasPoint(itemPoint)) { if (App::hoveredItem() != view) { repaintItem(App::hoveredItem()); App::hoveredItem(view); @@ -1362,11 +1363,11 @@ void InnerWidget::updateSelected() { } else { selectingText = false; } - dragState = item->getState(itemPoint, request); + dragState = view->getState(itemPoint, request); lnkhost = view; if (!dragState.link && itemPoint.x() >= st::historyPhotoLeft && itemPoint.x() < st::historyPhotoLeft + st::msgPhotoSize) { if (auto message = item->toHistoryMessage()) { - if (message->hasFromPhoto()) { + if (view->hasFromPhoto()) { enumerateUserpics([&](not_null view, int userpicTop) { // stop enumeration if the userpic is below our point if (userpicTop > point.y()) { @@ -1441,7 +1442,7 @@ void InnerWidget::updateSelected() { // Voice message seek support. if (const auto pressedView = App::pressedLinkItem()) { const auto adjustedPoint = mapPointToItem(point, pressedView); - pressedView->data()->updatePressed(adjustedPoint); + pressedView->updatePressed(adjustedPoint); } //if (_mouseAction == MouseAction::Selecting) { @@ -1557,7 +1558,7 @@ void InnerWidget::repaintItem(const Element *view) { if (!view) { return; } - update(0, itemTop(view), width(), view->data()->height()); + update(0, itemTop(view), width(), view->height()); } QPoint InnerWidget::mapPointToItem(QPoint point, const Element *view) const { diff --git a/Telegram/SourceFiles/history/admin_log/history_admin_log_item.cpp b/Telegram/SourceFiles/history/admin_log/history_admin_log_item.cpp index c481ab3b3..8b36b43c6 100644 --- a/Telegram/SourceFiles/history/admin_log/history_admin_log_item.cpp +++ b/Telegram/SourceFiles/history/admin_log/history_admin_log_item.cpp @@ -11,6 +11,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "history/view/history_view_element.h" #include "history/history_service.h" #include "history/history_message.h" +#include "history/history.h" #include "lang/lang_keys.h" #include "boxes/sticker_set_box.h" #include "core/tl_help.h" diff --git a/Telegram/SourceFiles/history/history.cpp b/Telegram/SourceFiles/history/history.cpp index e58327acf..efe6facd4 100644 --- a/Telegram/SourceFiles/history/history.cpp +++ b/Telegram/SourceFiles/history/history.cpp @@ -65,8 +65,10 @@ History::History(const PeerId &peerId) , cloudDraftTextCache(st::dialogsTextWidthMin) , _mute(peer->isMuted()) , _sendActionText(st::dialogsTextWidthMin) { - if (peer->isUser() && peer->asUser()->botInfo) { - outboxReadBefore = INT_MAX; + if (const auto user = peer->asUser()) { + if (user->botInfo) { + outboxReadBefore = INT_MAX; + } } } @@ -1391,8 +1393,9 @@ void History::addItemToBlock(not_null item) { block->messages.push_back(item->createView( App::wnd()->controller(), HistoryView::Context::History)); - block->messages.back()->attachToBlock(block, block->messages.size() - 1); - item->previousItemChanged(); + const auto view = block->messages.back().get(); + view->attachToBlock(block, block->messages.size() - 1); + view->previousInBlocksChanged(); if (isBuildingFrontBlock() && _buildingFrontBlock->expectedItemsCount > 0) { --_buildingFrontBlock->expectedItemsCount; @@ -1969,16 +1972,16 @@ not_null History::addNewInTheMiddle( App::wnd()->controller(), HistoryView::Context::History)); (*it)->attachToBlock(block.get(), itemIndex); - newItem->previousItemChanged(); + (*it)->previousInBlocksChanged(); if (itemIndex + 1 < block->messages.size()) { for (auto i = itemIndex + 1, l = int(block->messages.size()); i != l; ++i) { block->messages[i]->setIndexInBlock(i); } - block->messages[itemIndex + 1]->data()->previousItemChanged(); + block->messages[itemIndex + 1]->previousInBlocksChanged(); } else if (blockIndex + 1 < blocks.size() && !blocks[blockIndex + 1]->messages.empty()) { - blocks[blockIndex + 1]->messages.front()->data()->previousItemChanged(); + blocks[blockIndex + 1]->messages.front()->previousInBlocksChanged(); } else { - newItem->nextItemChanged(); + (*it)->nextInBlocksChanged(); } const auto [groupFrom, groupTill] = recountGroupingFromTill(newItem); @@ -2155,14 +2158,15 @@ HistoryBlock *History::finishBuildingFrontBlock() { auto block = _buildingFrontBlock->block; if (block) { if (blocks.size() > 1) { - const auto last = block->messages.back()->data(); // ... item, item, item, last ], [ first, item, item ... - const auto first = blocks[1]->messages.front()->data(); + // ... item, item, item, last ], [ first, item, item ... + const auto last = block->messages.back().get(); + const auto first = blocks[1]->messages.front().get(); // we've added a new front block, so previous item for // the old first item of a first block was changed - first->previousItemChanged(); + first->previousInBlocksChanged(); } else { - block->messages.back()->data()->nextItemChanged(); + block->messages.back()->nextInBlocksChanged(); } } @@ -2509,9 +2513,6 @@ void History::changedChatListPinHook() { Notify::PeerUpdate::Flag::PinnedChanged); } -void History::changeMsgId(MsgId oldId, MsgId newId) { -} - void History::removeBlock(not_null block) { Expects(block->messages.empty()); @@ -2525,9 +2526,9 @@ void History::removeBlock(not_null block) { for (int i = index, l = blocks.size(); i < l; ++i) { blocks[i]->setIndexInHistory(i); } - blocks[index]->messages.front()->data()->previousItemChanged(); + blocks[index]->messages.front()->previousInBlocksChanged(); } else if (!blocks.empty() && !blocks.back()->messages.empty()) { - blocks.back()->messages.back()->data()->nextItemChanged(); + blocks.back()->messages.back()->nextInBlocksChanged(); } } @@ -2543,12 +2544,10 @@ int HistoryBlock::resizeGetHeight(int newWidth, bool resizeAllItems) { auto y = 0; for (const auto &message : messages) { message->setY(y); - - const auto item = message->data(); - if (resizeAllItems || item->pendingResize()) { - y += item->resizeGetHeight(newWidth); + if (resizeAllItems || message->pendingResize()) { + y += message->resizeGetHeight(newWidth); } else { - y += item->height(); + y += message->height(); } } _height = y; @@ -2606,11 +2605,11 @@ void HistoryBlock::remove(not_null view) { // Deletes this. _history->removeBlock(this); } else if (itemIndex < messages.size()) { - messages[itemIndex]->data()->previousItemChanged(); + messages[itemIndex]->previousInBlocksChanged(); } else if (blockIndex + 1 < _history->blocks.size()) { - _history->blocks[blockIndex + 1]->messages.front()->data()->previousItemChanged(); + _history->blocks[blockIndex + 1]->messages.front()->previousInBlocksChanged(); } else if (!_history->blocks.empty() && !_history->blocks.back()->messages.empty()) { - _history->blocks.back()->messages.back()->data()->nextItemChanged(); + _history->blocks.back()->messages.back()->nextInBlocksChanged(); } if (needGroupRecount) { diff --git a/Telegram/SourceFiles/history/history.h b/Telegram/SourceFiles/history/history.h index f563fc430..690a500af 100644 --- a/Telegram/SourceFiles/history/history.h +++ b/Telegram/SourceFiles/history/history.h @@ -18,10 +18,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "base/flags.h" class History; -class HistoryItem; -using HistoryItemsList = std::vector>; -enum NewMessageType { +enum NewMessageType : char { NewMessageUnread, NewMessageLast, NewMessageExisting, @@ -104,50 +102,6 @@ private: class HistoryBlock; -enum HistoryMediaType { - MediaTypePhoto, - MediaTypeVideo, - MediaTypeContact, - MediaTypeCall, - MediaTypeFile, - MediaTypeGif, - MediaTypeSticker, - MediaTypeLocation, - MediaTypeWebPage, - MediaTypeMusicFile, - MediaTypeVoiceFile, - MediaTypeGame, - MediaTypeInvoice, - MediaTypeGrouped, - - MediaTypeCount -}; - -struct TextWithTags { - struct Tag { - int offset, length; - QString id; - }; - using Tags = QVector; - - QString text; - Tags tags; -}; - -inline bool operator==(const TextWithTags::Tag &a, const TextWithTags::Tag &b) { - return (a.offset == b.offset) && (a.length == b.length) && (a.id == b.id); -} -inline bool operator!=(const TextWithTags::Tag &a, const TextWithTags::Tag &b) { - return !(a == b); -} - -inline bool operator==(const TextWithTags &a, const TextWithTags &b) { - return (a.text == b.text) && (a.tags == b.tags); -} -inline bool operator!=(const TextWithTags &a, const TextWithTags &b) { - return !(a == b); -} - namespace Data { struct Draft; } // namespace Data @@ -334,7 +288,7 @@ public: HistoryItem *showFrom = nullptr; HistoryItem *unreadBar = nullptr; - PeerData *peer; + not_null peer; bool oldLoaded = false; bool newLoaded = true; HistoryItem *lastMsg = nullptr; @@ -426,8 +380,6 @@ public: mtpRequestId sendRequestId = 0; - void changeMsgId(MsgId oldId, MsgId newId); - Text cloudDraftTextCache; protected: diff --git a/Telegram/SourceFiles/history/history_inner_widget.cpp b/Telegram/SourceFiles/history/history_inner_widget.cpp index 4d85dfeec..feaf634ab 100644 --- a/Telegram/SourceFiles/history/history_inner_widget.cpp +++ b/Telegram/SourceFiles/history/history_inner_widget.cpp @@ -10,6 +10,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include #include "styles/style_history.h" #include "core/file_utilities.h" +#include "history/history.h" #include "history/history_message.h" #include "history/history_media_types.h" #include "history/history_item_components.h" @@ -198,7 +199,7 @@ void HistoryInner::repaintItem(const Element *view) { if (view) { const auto top = itemTop(view); if (top >= 0) { - update(0, top, width(), view->data()->height()); + update(0, top, width(), view->height()); } } } @@ -228,7 +229,7 @@ void HistoryInner::enumerateItemsInHistory(History *history, int historytop, Met while (true) { auto view = block->messages[itemIndex].get(); auto itemtop = blocktop + view->y(); - auto itembottom = itemtop + view->data()->height(); + auto itembottom = itemtop + view->height(); // Binary search should've skipped all the items that are above / below the visible area. if (TopToBottom) { @@ -310,19 +311,19 @@ void HistoryInner::enumerateUserpics(Method method) { const auto message = item->toHistoryMessage(); if (!message) return true; - if (lowestAttachedItemTop < 0 && message->isAttachedToNext()) { - lowestAttachedItemTop = itemtop + message->marginTop(); + if (lowestAttachedItemTop < 0 && view->isAttachedToNext()) { + lowestAttachedItemTop = itemtop + view->marginTop(); } // Call method on a userpic for all messages that have it and for those who are not showing it // because of their attachment to the next message if they are bottom-most visible. - if (message->displayFromPhoto() || (message->hasFromPhoto() && itembottom >= _visibleAreaBottom)) { + if (view->displayFromPhoto() || (view->hasFromPhoto() && itembottom >= _visibleAreaBottom)) { if (lowestAttachedItemTop < 0) { - lowestAttachedItemTop = itemtop + message->marginTop(); + lowestAttachedItemTop = itemtop + view->marginTop(); } // Attach userpic to the bottom of the visible area with the same margin as the last message. auto userpicMinBottomSkip = st::historyPaddingBottom + st::msgMargin.bottom(); - auto userpicBottom = qMin(itembottom - message->marginBottom(), _visibleAreaBottom - userpicMinBottomSkip); + auto userpicBottom = qMin(itembottom - view->marginBottom(), _visibleAreaBottom - userpicMinBottomSkip); // Do not let the userpic go above the attached messages pack top line. userpicBottom = qMax(userpicBottom, lowestAttachedItemTop + st::msgPhotoSize); @@ -335,7 +336,7 @@ void HistoryInner::enumerateUserpics(Method method) { } // Forget the found top of the pack, search for the next one from scratch. - if (!message->isAttachedToNext()) { + if (!view->isAttachedToNext()) { lowestAttachedItemTop = -1; } @@ -356,7 +357,7 @@ void HistoryInner::enumerateDates(Method method) { auto dateCallback = [&](not_null view, int itemtop, int itembottom) { const auto item = view->data(); if (lowestInOneDayItemBottom < 0 && item->isInOneDayWithPrevious()) { - lowestInOneDayItemBottom = itembottom - item->marginBottom(); + lowestInOneDayItemBottom = itembottom - view->marginBottom(); } // Call method on a date for all messages that have it and for those who are not showing it @@ -375,7 +376,7 @@ void HistoryInner::enumerateDates(Method method) { } if (lowestInOneDayItemBottom < 0) { - lowestInOneDayItemBottom = itembottom - item->marginBottom(); + lowestInOneDayItemBottom = itembottom - view->marginBottom(); } // Attach date to the top of the visible area with the same margin as it has in service message. int dateTop = qMax(itemtop, _visibleAreaTop) + st::msgServiceMargin.top(); @@ -499,7 +500,7 @@ void HistoryInner::paintEvent(QPaintEvent *e) { if (selfromy < 0 || seltoy < 0) { selfromy = seltoy = -1; } else { - seltoy += _dragSelTo->data()->height(); + seltoy += _dragSelTo->height(); } auto mtop = migratedTop(); @@ -515,12 +516,12 @@ void HistoryInner::paintEvent(QPaintEvent *e) { auto y = mtop + block->y() + view->y(); p.save(); p.translate(0, y); - if (clip.y() < y + item->height()) while (y < drawToY) { + if (clip.y() < y + view->height()) while (y < drawToY) { const auto selection = itemRenderSelection( view, selfromy - mtop, seltoy - mtop); - item->draw(p, clip.translated(0, -y), selection, ms); + view->draw(p, clip.translated(0, -y), selection, ms); if (item->hasViews()) { App::main()->scheduleViewIncrement(item); @@ -530,7 +531,7 @@ void HistoryInner::paintEvent(QPaintEvent *e) { _widget->enqueueMessageHighlight(item); } - int32 h = item->height(); + int32 h = view->height(); p.translate(0, h); y += h; @@ -560,13 +561,13 @@ void HistoryInner::paintEvent(QPaintEvent *e) { p.save(); p.translate(0, y); while (y < drawToY) { - auto h = item->height(); + auto h = view->height(); if (hclip.y() < y + h && hdrawtop < y + h) { const auto selection = itemRenderSelection( view, selfromy - htop, seltoy - htop); - item->draw(p, hclip.translated(0, -y), selection, ms); + view->draw(p, hclip.translated(0, -y), selection, ms); if (item->hasViews()) { App::main()->scheduleViewIncrement(item); @@ -916,11 +917,12 @@ void HistoryInner::mouseActionStart(const QPoint &screenPos, Qt::MouseButton but repaintItem(App::pressedItem()); } + const auto mouseActionView = App::mousedItem(); _mouseAction = MouseAction::None; - _mouseActionItem = App::mousedItem() - ? App::mousedItem()->data().get() + _mouseActionItem = mouseActionView + ? mouseActionView->data().get() : nullptr; - _dragStartPosition = mapPointToItem(mapFromGlobal(screenPos), _mouseActionItem); + _dragStartPosition = mapPointToItem(mapFromGlobal(screenPos), mouseActionView); _pressWasInactive = _controller->window()->wasInactivePress(); if (_pressWasInactive) _controller->window()->setInactivePress(false); @@ -937,12 +939,12 @@ void HistoryInner::mouseActionStart(const QPoint &screenPos, Qt::MouseButton but } } } - if (_mouseAction == MouseAction::None && _mouseActionItem) { + if (_mouseAction == MouseAction::None && mouseActionView) { HistoryTextState dragState; if (_trippleClickTimer.isActive() && (screenPos - _trippleClickPoint).manhattanLength() < QApplication::startDragDistance()) { HistoryStateRequest request; request.flags = Text::StateRequest::Flag::LookupSymbol; - dragState = _mouseActionItem->getState(_dragStartPosition, request); + dragState = mouseActionView->getState(_dragStartPosition, request); if (dragState.cursor == HistoryInTextCursorState) { TextSelection selStatus = { dragState.symbol, dragState.symbol }; if (selStatus != FullSelection && (_selected.empty() || _selected.cbegin()->second != FullSelection)) { @@ -961,7 +963,7 @@ void HistoryInner::mouseActionStart(const QPoint &screenPos, Qt::MouseButton but } else if (App::pressedItem()) { HistoryStateRequest request; request.flags = Text::StateRequest::Flag::LookupSymbol; - dragState = _mouseActionItem->getState(_dragStartPosition, request); + dragState = mouseActionView->getState(_dragStartPosition, request); } if (_mouseSelectType != TextSelectType::Paragraphs) { if (App::pressedItem()) { @@ -1026,15 +1028,18 @@ void HistoryInner::mouseActionCancel() { void HistoryInner::performDrag() { if (_mouseAction != MouseAction::Dragging) return; + const auto mouseActionView = _mouseActionItem + ? _mouseActionItem->mainView() + : nullptr; bool uponSelected = false; - if (_mouseActionItem) { + if (mouseActionView) { if (!_selected.empty() && _selected.cbegin()->second == FullSelection) { uponSelected = _dragStateItem && (_selected.find(_dragStateItem) != _selected.cend()); } else { HistoryStateRequest request; request.flags |= Text::StateRequest::Flag::LookupSymbol; - auto dragState = _mouseActionItem->getState(_dragStartPosition, request); + auto dragState = mouseActionView->getState(_dragStartPosition, request); uponSelected = (dragState.cursor == HistoryInTextCursorState); if (uponSelected) { if (_selected.empty() @@ -1243,10 +1248,21 @@ void HistoryInner::mouseDoubleClickEvent(QMouseEvent *e) { if (!_history) return; mouseActionStart(e->globalPos(), e->button()); - if (((_mouseAction == MouseAction::Selecting && !_selected.empty() && _selected.cbegin()->second != FullSelection) || (_mouseAction == MouseAction::None && (_selected.empty() || _selected.cbegin()->second != FullSelection))) && _mouseSelectType == TextSelectType::Letters && _mouseActionItem) { + + const auto mouseActionView = _mouseActionItem + ? _mouseActionItem->mainView() + : nullptr; + if (_mouseSelectType == TextSelectType::Letters + && mouseActionView + && ((_mouseAction == MouseAction::Selecting + && !_selected.empty() + && _selected.cbegin()->second != FullSelection) + || (_mouseAction == MouseAction::None + && (_selected.empty() + || _selected.cbegin()->second != FullSelection)))) { HistoryStateRequest request; request.flags |= Text::StateRequest::Flag::LookupSymbol; - auto dragState = _mouseActionItem->getState(_dragStartPosition, request); + auto dragState = mouseActionView->getState(_dragStartPosition, request); if (dragState.cursor == HistoryInTextCursorState) { _mouseTextSymbol = dragState.symbol; _mouseSelectType = TextSelectType::Words; @@ -1298,7 +1314,7 @@ void HistoryInner::showContextMenu(QContextMenuEvent *e, bool showFromTouch) { auto mousePos = mapPointToItem(mapFromGlobal(_mousePosition), App::mousedItem()); HistoryStateRequest request; request.flags |= Text::StateRequest::Flag::LookupSymbol; - auto dragState = App::mousedItem()->data()->getState(mousePos, request); + auto dragState = App::mousedItem()->getState(mousePos, request); if (dragState.cursor == HistoryInTextCursorState && dragState.symbol >= selFrom && dragState.symbol < selTo) { isUponSelected = 1; } @@ -1809,7 +1825,7 @@ void HistoryInner::recountHistoryGeometry() { if (!_migrated->isEmpty() && !_history->isEmpty() && _migrated->loadedAtBottom() && _history->loadedAtTop()) { if (_migrated->blocks.back()->messages.back()->data()->date.date() == _history->blocks.front()->messages.front()->data()->date.date()) { if (_migrated->blocks.back()->messages.back()->data()->isGroupMigrate() && _history->blocks.front()->messages.front()->data()->isGroupMigrate()) { - _historySkipHeight += _history->blocks.front()->messages.front()->data()->height(); + _historySkipHeight += _history->blocks.front()->messages.front()->height(); } else { _historySkipHeight += _history->blocks.front()->messages.front()->data()->displayedDateHeight(); } @@ -2105,7 +2121,7 @@ void HistoryInner::adjustCurrent(int32 y, History *history) const { while (block->messages[_curItem]->y() + by > y && _curItem > 0) { --_curItem; } - while (block->messages[_curItem]->y() + block->messages[_curItem]->data()->height() + by <= y && _curItem + 1 < block->messages.size()) { + while (block->messages[_curItem]->y() + block->messages[_curItem]->height() + by <= y && _curItem + 1 < block->messages.size()) { ++_curItem; } } @@ -2237,7 +2253,7 @@ void HistoryInner::onUpdateSelected() { App::mousedItem(view); m = mapPointToItem(point, view); - if (item->hasPoint(m)) { + if (view->hasPoint(m)) { if (App::hoveredItem() != view) { repaintItem(App::hoveredItem()); App::hoveredItem(view); @@ -2336,12 +2352,12 @@ void HistoryInner::onUpdateSelected() { } else { selectingText = false; } - dragState = item->getState(m, request); + dragState = view->getState(m, request); _dragStateItem = App::histItemById(dragState.itemId); lnkhost = view; if (!dragState.link && m.x() >= st::historyPhotoLeft && m.x() < st::historyPhotoLeft + st::msgPhotoSize) { if (auto msg = item->toHistoryMessage()) { - if (msg->hasFromPhoto()) { + if (view->hasFromPhoto()) { enumerateUserpics([&](not_null view, int userpicTop) -> bool { // stop enumeration if the userpic is below our point if (userpicTop > point.y()) { @@ -2410,15 +2426,15 @@ void HistoryInner::onUpdateSelected() { auto selectingDown = (itemTop(_mouseActionItem) < itemTop(item)) || (_mouseActionItem == item && _dragStartPosition.y() < m.y()); auto dragSelFrom = _mouseActionItem->mainView(); auto dragSelTo = view; - if (!dragSelFrom->data()->hasPoint(_dragStartPosition)) { // maybe exclude dragSelFrom + if (!dragSelFrom->hasPoint(_dragStartPosition)) { // maybe exclude dragSelFrom if (selectingDown) { - if (_dragStartPosition.y() >= dragSelFrom->data()->height() - dragSelFrom->data()->marginBottom() || ((view == dragSelFrom) && (m.y() < _dragStartPosition.y() + QApplication::startDragDistance() || m.y() < dragSelFrom->data()->marginTop()))) { + if (_dragStartPosition.y() >= dragSelFrom->height() - dragSelFrom->marginBottom() || ((view == dragSelFrom) && (m.y() < _dragStartPosition.y() + QApplication::startDragDistance() || m.y() < dragSelFrom->marginTop()))) { dragSelFrom = (dragSelFrom != dragSelTo) ? nextItem(dragSelFrom) : nullptr; } } else { - if (_dragStartPosition.y() < dragSelFrom->data()->marginTop() || ((view == dragSelFrom) && (m.y() >= _dragStartPosition.y() - QApplication::startDragDistance() || m.y() >= dragSelFrom->data()->height() - dragSelFrom->data()->marginBottom()))) { + if (_dragStartPosition.y() < dragSelFrom->marginTop() || ((view == dragSelFrom) && (m.y() >= _dragStartPosition.y() - QApplication::startDragDistance() || m.y() >= dragSelFrom->height() - dragSelFrom->marginBottom()))) { dragSelFrom = (dragSelFrom != dragSelTo) ? prevItem(dragSelFrom) : nullptr; @@ -2427,13 +2443,13 @@ void HistoryInner::onUpdateSelected() { } if (_mouseActionItem != item) { // maybe exclude dragSelTo if (selectingDown) { - if (m.y() < dragSelTo->data()->marginTop()) { + if (m.y() < dragSelTo->marginTop()) { dragSelTo = (dragSelFrom != dragSelTo) ? prevItem(dragSelFrom) : nullptr; } } else { - if (m.y() >= dragSelTo->data()->height() - dragSelTo->data()->marginBottom()) { + if (m.y() >= dragSelTo->height() - dragSelTo->marginBottom()) { dragSelTo = (dragSelFrom != dragSelTo) ? nextItem(dragSelFrom) : nullptr; @@ -2471,10 +2487,10 @@ void HistoryInner::onUpdateSelected() { // Voice message seek support. if (const auto pressedItem = _dragStateItem) { - if (pressedItem->mainView()) { + if (const auto pressedView = pressedItem->mainView()) { if (pressedItem->history() == _history || pressedItem->history() == _migrated) { - auto adjustedPoint = mapPointToItem(point, pressedItem); - pressedItem->updatePressed(adjustedPoint); + auto adjustedPoint = mapPointToItem(point, pressedView); + pressedView->updatePressed(adjustedPoint); } } } @@ -2756,6 +2772,11 @@ void HistoryInner::deleteItem(not_null item) { Ui::show(Box(item, suggestModerateActions)); } +bool HistoryInner::hasPendingResizedItems() const { + return (_history && _history->hasPendingResizedItems()) + || (_migrated && _migrated->hasPendingResizedItems()); +} + void HistoryInner::deleteAsGroup(FullMsgId itemId) { if (const auto item = App::histItemById(itemId)) { const auto group = item->getFullGroup(); @@ -2793,7 +2814,7 @@ void HistoryInner::applyDragSelection( const auto selfromy = itemTop(_dragSelFrom); const auto seltoy = [&] { auto result = itemTop(_dragSelTo); - return (result < 0) ? result : (result + _dragSelTo->data()->height()); + return (result < 0) ? result : (result + _dragSelTo->height()); }(); if (selfromy < 0 || seltoy < 0) { return; diff --git a/Telegram/SourceFiles/history/history_inner_widget.h b/Telegram/SourceFiles/history/history_inner_widget.h index e9d2ec435..3409cf2fe 100644 --- a/Telegram/SourceFiles/history/history_inner_widget.h +++ b/Telegram/SourceFiles/history/history_inner_widget.h @@ -262,9 +262,7 @@ private: void deleteAsGroup(FullMsgId itemId); // Does any of the shown histories has this flag set. - bool hasPendingResizedItems() const { - return (_history && _history->hasPendingResizedItems()) || (_migrated && _migrated->hasPendingResizedItems()); - } + bool hasPendingResizedItems() const; not_null _controller; diff --git a/Telegram/SourceFiles/history/history_item.cpp b/Telegram/SourceFiles/history/history_item.cpp index 4479ccd0b..9ec5cf4b6 100644 --- a/Telegram/SourceFiles/history/history_item.cpp +++ b/Telegram/SourceFiles/history/history_item.cpp @@ -16,6 +16,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "history/history_media_types.h" #include "history/history_media_grouped.h" #include "history/history_message.h" +#include "history/history.h" #include "media/media_clip_reader.h" #include "styles/style_dialogs.h" #include "styles/style_history.h" @@ -35,13 +36,6 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "data/data_messages.h" #include "data/data_feed.h" -namespace { - -// a new message from the same sender is attached to previous within 15 minutes -constexpr int kAttachMessageToPreviousSecondsDelta = 900; - -} // namespace - namespace internal { TextSelection unshiftSelection(TextSelection selection, uint16 byLength) { @@ -65,12 +59,12 @@ HistoryItem::HistoryItem( MsgId id, MTPDmessage::Flags flags, QDateTime date, - UserId from) : HistoryElement() -, id(id) + UserId from) +: id(id) , date(date) , _history(history) , _from(from ? App::user(from) : history->peer) -, _flags(flags | MTPDmessage_ClientFlag::f_pending_init_dimensions | MTPDmessage_ClientFlag::f_pending_resize) { +, _flags(flags) { } void HistoryItem::finishCreate() { @@ -78,23 +72,23 @@ void HistoryItem::finishCreate() { } void HistoryItem::finishEdition(int oldKeyboardTop) { - setPendingInitDimensions(); + Auth().data().requestItemViewRefresh(this); invalidateChatsListEntry(); //if (groupId()) { // history()->fixGroupAfterEdition(this); //} - if (isHiddenByGroup()) { - // Perhaps caption was changed, we should refresh the group. - const auto group = Get(); - group->leader->setPendingInitDimensions(); - group->leader->invalidateChatsListEntry(); - } + //if (isHiddenByGroup()) { // #TODO group views + // // Perhaps caption was changed, we should refresh the group. + // const auto group = Get(); + // group->leader->setPendingInitDimensions(); + // group->leader->invalidateChatsListEntry(); + //} - if (oldKeyboardTop >= 0) { - if (auto keyboard = Get()) { - keyboard->oldTop = oldKeyboardTop; - } - } + //if (oldKeyboardTop >= 0) { // #TODO edit bot message + // if (auto keyboard = Get()) { + // keyboard->oldTop = oldKeyboardTop; + // } + //} App::historyUpdateDependent(this); } @@ -134,7 +128,6 @@ void HistoryItem::invalidateChatsListEntry() { } void HistoryItem::finishEditionToEmpty() { - recountDisplayDate(); finishEdition(-1); _history->removeNotification(this); @@ -149,13 +142,6 @@ void HistoryItem::finishEditionToEmpty() { if ((!out() || isPost()) && unread() && history()->unreadCount() > 0) { history()->setUnreadCount(history()->unreadCount() - 1); } - - if (auto next = nextItem()) { - next->previousItemChanged(); - } - if (auto previous = previousItem()) { - previous->nextItemChanged(); - } } bool HistoryItem::isMediaUnread() const { @@ -218,6 +204,17 @@ UserData *HistoryItem::viaBot() const { return nullptr; } +UserData *HistoryItem::getMessageBot() const { + if (const auto bot = viaBot()) { + return bot; + } + auto bot = from()->asUser(); + if (!bot) { + bot = history()->peer->asUser(); + } + return (bot && bot->botInfo) ? bot : nullptr; +}; + void HistoryItem::destroy() { const auto history = this->history(); if (isLogEntry()) { @@ -286,6 +283,9 @@ void HistoryItem::clearMainView() { } } +void HistoryItem::addToUnreadMentions(UnreadMentionType type) { +} + Storage::SharedMediaTypesMask HistoryItem::sharedMediaTypes() const { return {}; } @@ -304,71 +304,9 @@ void HistoryItem::indexAsNewItem() { } } -void HistoryItem::previousItemChanged() { - Expects(!isLogEntry()); +void HistoryItem::setRealId(MsgId newId) { + Expects(!IsServerMsgId(id)); - recountDisplayDate(); - recountAttachToPrevious(); -} - -// Called only if there is no more next item! Not always when it changes! -void HistoryItem::nextItemChanged() { - Expects(!isLogEntry()); - - setAttachToNext(false); -} - -bool HistoryItem::computeIsAttachToPrevious(not_null previous) { - if (!Has() && !Has()) { - const auto possible = !isPost() && !previous->isPost() - && !serviceMsg() && !previous->serviceMsg() - && !isEmpty() && !previous->isEmpty() - && (qAbs(previous->date.secsTo(date)) < kAttachMessageToPreviousSecondsDelta); - if (possible) { - if (history()->peer->isSelf()) { - return previous->senderOriginal() == senderOriginal() - && (previous->Has() == Has()); - } else { - return previous->from() == from(); - } - } - } - return false; -} - -void HistoryItem::recountAttachToPrevious() { - Expects(!isLogEntry()); - - auto attachToPrevious = false; - if (auto previous = previousItem()) { - attachToPrevious = computeIsAttachToPrevious(previous); - previous->setAttachToNext(attachToPrevious); - } - setAttachToPrevious(attachToPrevious); -} - -void HistoryItem::setAttachToNext(bool attachToNext) { - if (attachToNext && !(_flags & MTPDmessage_ClientFlag::f_attach_to_next)) { - _flags |= MTPDmessage_ClientFlag::f_attach_to_next; - Global::RefPendingRepaintItems().insert(this); - } else if (!attachToNext && (_flags & MTPDmessage_ClientFlag::f_attach_to_next)) { - _flags &= ~MTPDmessage_ClientFlag::f_attach_to_next; - Global::RefPendingRepaintItems().insert(this); - } -} - -void HistoryItem::setAttachToPrevious(bool attachToPrevious) { - if (attachToPrevious && !(_flags & MTPDmessage_ClientFlag::f_attach_to_previous)) { - _flags |= MTPDmessage_ClientFlag::f_attach_to_previous; - setPendingInitDimensions(); - } else if (!attachToPrevious && (_flags & MTPDmessage_ClientFlag::f_attach_to_previous)) { - _flags &= ~MTPDmessage_ClientFlag::f_attach_to_previous; - setPendingInitDimensions(); - } -} - -void HistoryItem::setId(MsgId newId) { - history()->changeMsgId(id, newId); id = newId; // We don't need to call Notify::replyMarkupUpdated(this) and update keyboard @@ -568,6 +506,10 @@ QString HistoryItem::directLink() const { return QString(); } +ChannelId HistoryItem::channelId() const { + return _history->channelId(); +} + Data::MessagePosition HistoryItem::position() const { return Data::MessagePosition(toServerTime(date.toTime_t()).v, fullId()); } @@ -579,6 +521,10 @@ MsgId HistoryItem::replyToId() const { return 0; } +not_null HistoryItem::author() const { + return isPost() ? history()->peer : from(); +} + QDateTime HistoryItem::dateOriginal() const { if (const auto forwarded = Get()) { return forwarded->originalDate; @@ -586,7 +532,7 @@ QDateTime HistoryItem::dateOriginal() const { return date; } -PeerData *HistoryItem::senderOriginal() const { +not_null HistoryItem::senderOriginal() const { if (const auto forwarded = Get()) { return forwarded->originalSender; } @@ -594,7 +540,7 @@ PeerData *HistoryItem::senderOriginal() const { return (peer->isChannel() && !peer->isMegagroup()) ? peer : from(); } -PeerData *HistoryItem::fromOriginal() const { +not_null HistoryItem::fromOriginal() const { if (const auto forwarded = Get()) { if (const auto user = forwarded->originalSender->asUser()) { return user; @@ -626,6 +572,10 @@ bool HistoryItem::hasOutLayout() const { return out() && !isPost(); } +bool HistoryItem::needCheck() const { + return out() || (id < 0 && history()->peer->isSelf()); +} + bool HistoryItem::unread() const { // Messages from myself are always read. if (history()->peer->isSelf()) return false; @@ -667,24 +617,23 @@ void HistoryItem::destroyUnreadBar() { Assert(!isLogEntry()); RemoveComponents(HistoryMessageUnreadBar::Bit()); - setPendingInitDimensions(); + Auth().data().requestItemViewResize(this); if (_history->unreadBar == this) { _history->unreadBar = nullptr; } - - recountAttachToPrevious(); + // #TODO recount attach to previous } } void HistoryItem::setUnreadBarCount(int count) { Expects(!isLogEntry()); + if (count > 0) { HistoryMessageUnreadBar *bar; if (!Has()) { AddComponents(HistoryMessageUnreadBar::Bit()); - setPendingInitDimensions(); - - recountAttachToPrevious(); + Auth().data().requestItemViewResize(this); + // #TODO recount attach to previous bar = Get(); } else { @@ -721,7 +670,7 @@ bool HistoryItem::groupIdValidityChanged() { return false; } RemoveComponents(HistoryMessageGroup::Bit()); - setPendingInitDimensions(); + Auth().data().requestItemViewResize(this); return true; } return false; @@ -737,7 +686,7 @@ void HistoryItem::makeGroupMember(not_null leader) { _media = std::move(single); } _flags |= MTPDmessage_ClientFlag::f_hidden_by_group; - setPendingInitDimensions(); + Auth().data().requestItemViewResize(this); group->leader = leader; base::take(group->others); @@ -758,7 +707,7 @@ void HistoryItem::makeGroupLeader( if (leaderChanged) { group->leader = this; _flags &= ~MTPDmessage_ClientFlag::f_hidden_by_group; - setPendingInitDimensions(); + Auth().data().requestItemViewResize(this); } group->others = std::move(others); if (!_media || !_media->applyGroup(group->others)) { @@ -786,7 +735,7 @@ void HistoryItem::resetGroupMedia( } else if (_media) { _media = _media->takeLastFromGroup(); } - setPendingInitDimensions(); + Auth().data().requestItemViewResize(this); } int HistoryItem::displayedDateHeight() const { @@ -796,22 +745,6 @@ int HistoryItem::displayedDateHeight() const { return 0; } -int HistoryItem::marginTop() const { - auto result = 0; - if (!isHiddenByGroup()) { - if (isAttachedToPrevious()) { - result += st::msgMarginTopAttached; - } else { - result += st::msgMargin.top(); - } - } - result += displayedDateHeight(); - if (const auto unreadbar = Get()) { - result += unreadbar->height(); - } - return result; -} - bool HistoryItem::displayDate() const { return Has(); } @@ -822,10 +755,6 @@ bool HistoryItem::isEmpty() const { && !Has(); } -int HistoryItem::marginBottom() const { - return isHiddenByGroup() ? 0 : st::msgMargin.bottom(); -} - void HistoryItem::clipCallback(Media::Clip::Notification notification) { using namespace Media::Clip; @@ -857,7 +786,7 @@ void HistoryItem::clipCallback(Media::Clip::Notification notification) { media->stopInline(); } if (!stopped) { - setPendingInitDimensions(); + Auth().data().requestItemViewResize(this); Auth().data().markItemLayoutChanged(this); Global::RefPendingRepaintItems().insert(this); } @@ -922,31 +851,6 @@ HistoryItem *HistoryItem::nextItem() const { return nullptr; } -void HistoryItem::recountDisplayDate() { - Expects(!isLogEntry()); - setDisplayDate([&] { - if (isEmpty()) { - return false; - } - - if (auto previous = previousItem()) { - return previous->isEmpty() || (previous->date.date() != date.date()); - } - return true; - }()); -} - -void HistoryItem::setDisplayDate(bool displayDate) { - if (displayDate && !Has()) { - AddComponents(HistoryMessageDate::Bit()); - Get()->init(date); - setPendingInitDimensions(); - } else if (!displayDate && Has()) { - RemoveComponents(HistoryMessageDate::Bit()); - setPendingInitDimensions(); - } -} - QString HistoryItem::notificationText() const { auto getText = [this]() { if (emptyText()) { @@ -1016,8 +920,10 @@ HistoryItem::~HistoryItem() { } } -ClickHandlerPtr goToMessageClickHandler(PeerData *peer, MsgId msgId) { - return std::make_shared([peer, msgId] { +ClickHandlerPtr goToMessageClickHandler( + not_null peer, + MsgId msgId) { + return std::make_shared([=] { if (App::main()) { auto view = App::mousedItem(); if (view && view->data()->history()->peer == peer) { @@ -1030,3 +936,7 @@ ClickHandlerPtr goToMessageClickHandler(PeerData *peer, MsgId msgId) { } }); } + +ClickHandlerPtr goToMessageClickHandler(not_null item) { + return goToMessageClickHandler(item->history()->peer, item->id); +} diff --git a/Telegram/SourceFiles/history/history_item.h b/Telegram/SourceFiles/history/history_item.h index 43adf25dd..1ed5c1583 100644 --- a/Telegram/SourceFiles/history/history_item.h +++ b/Telegram/SourceFiles/history/history_item.h @@ -13,6 +13,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "history/history_media_pointer.h" #include "history/view/history_view_cursor_state.h" +enum class UnreadMentionType; struct MessageGroupId; struct HistoryMessageGroup; struct HistoryMessageReplyMarkup; @@ -51,35 +52,6 @@ namespace HistoryView { enum class Context : char; } // namespace HistoryView -class HistoryElement { -public: - HistoryElement() = default; - HistoryElement(const HistoryElement &other) = delete; - HistoryElement &operator=(const HistoryElement &other) = delete; - - int maxWidth() const { - return _maxw; - } - int minHeight() const { - return _minh; - } - int width() const { - return _width; - } - int height() const { - return _height; - } - - virtual ~HistoryElement() = default; - -protected: - mutable int _maxw = 0; - mutable int _minh = 0; - mutable int _width = 0; - mutable int _height = 0; - -}; - namespace internal { TextSelection unshiftSelection(TextSelection selection, uint16 byLength); @@ -93,23 +65,8 @@ inline TextSelection shiftSelection(TextSelection selection, const Text &byText) } // namespace internal -class HistoryItem - : public HistoryElement - , public RuntimeComposer { +class HistoryItem : public RuntimeComposer { public: - int resizeGetHeight(int newWidth) { - if (_flags & MTPDmessage_ClientFlag::f_pending_init_dimensions) { - _flags &= ~MTPDmessage_ClientFlag::f_pending_init_dimensions; - initDimensions(); - } - if (_flags & MTPDmessage_ClientFlag::f_pending_resize) { - _flags &= ~MTPDmessage_ClientFlag::f_pending_resize; - } - _width = newWidth; - return resizeContentGetHeight(); - } - virtual void draw(Painter &p, QRect clip, TextSelection selection, TimeMs ms) const = 0; - virtual void dependencyItemRemoved(HistoryItem *dependency) { } virtual bool updateDependencyItem() { @@ -126,16 +83,7 @@ public: } UserData *viaBot() const; - UserData *getMessageBot() const { - if (auto bot = viaBot()) { - return bot; - } - auto bot = from()->asUser(); - if (!bot) { - bot = history()->peer->asUser(); - } - return (bot && bot->botInfo) ? bot : nullptr; - }; + UserData *getMessageBot() const; bool isLogEntry() const { return (id > ServerMaxMsgId); @@ -148,7 +96,7 @@ public: not_null history() const { return _history; } - PeerData *from() const { + not_null from() const { return _from; } HistoryView::Element *mainView() const { @@ -202,18 +150,7 @@ public: return hasViews() ? 1 : -1; } - virtual bool needCheck() const { - return out() || (id < 0 && history()->peer->isSelf()); - } - virtual bool hasPoint(QPoint point) const { - return false; - } - - [[nodiscard]] virtual HistoryTextState getState( - QPoint point, - HistoryStateRequest request) const = 0; - virtual void updatePressed(QPoint point) { - } + virtual bool needCheck() const; [[nodiscard]] virtual TextSelection adjustSelection( TextSelection selection, @@ -233,8 +170,7 @@ public: virtual void updateReplyMarkup(const MTPReplyMarkup *markup) { } - virtual void addToUnreadMentions(UnreadMentionType type) { - } + virtual void addToUnreadMentions(UnreadMentionType type); virtual void eraseFromUnreadMentions() { } virtual Storage::SharedMediaTypesMask sharedMediaTypes() const; @@ -244,9 +180,6 @@ public: return false; } - void previousItemChanged(); - void nextItemChanged(); - virtual TextWithEntities selectedText(TextSelection selection) const { return { qsl("[-]"), EntitiesInText() }; } @@ -283,7 +216,7 @@ public: } virtual void setViewsCount(int32 count) { } - virtual void setId(MsgId newId); + virtual void setRealId(MsgId newId); virtual bool displayEditedBadge() const { return false; @@ -322,9 +255,7 @@ public: MsgId id; QDateTime date; - ChannelId channelId() const { - return _history->channelId(); - } + ChannelId channelId() const; FullMsgId fullId() const { return FullMsgId(channelId(), id); } @@ -370,13 +301,11 @@ public: } MsgId replyToId() const; - PeerData *author() const { - return isPost() ? history()->peer : from(); - } + not_null author() const; QDateTime dateOriginal() const; - PeerData *senderOriginal() const; - PeerData *fromOriginal() const; + not_null senderOriginal() const; + not_null fromOriginal() const; QString authorOriginal() const; MsgId idOriginal() const; @@ -391,30 +320,7 @@ public: // when the new messages arrive in this chat history void setUnreadBarFreezed(); - bool pendingResize() const { - return _flags & MTPDmessage_ClientFlag::f_pending_resize; - } - void setPendingResize() { - _flags |= MTPDmessage_ClientFlag::f_pending_resize; - _history->setHasPendingResizedItems(); - } - bool pendingInitDimensions() const { - return _flags & MTPDmessage_ClientFlag::f_pending_init_dimensions; - } - void setPendingInitDimensions() { - _flags |= MTPDmessage_ClientFlag::f_pending_init_dimensions; - setPendingResize(); - } - int displayedDateHeight() const; - int marginTop() const; - int marginBottom() const; - bool isAttachedToPrevious() const { - return _flags & MTPDmessage_ClientFlag::f_attach_to_previous; - } - bool isAttachedToNext() const { - return _flags & MTPDmessage_ClientFlag::f_attach_to_next; - } bool displayDate() const; bool isInOneDayWithPrevious() const { @@ -439,20 +345,6 @@ public: void clipCallback(Media::Clip::Notification notification); void audioTrackUpdated(); - bool computeIsAttachToPrevious(not_null previous); - void setLogEntryDisplayDate(bool displayDate) { - Expects(isLogEntry()); - setDisplayDate(displayDate); - } - void setLogEntryAttachToPrevious(bool attachToPrevious) { - Expects(isLogEntry()); - setAttachToPrevious(attachToPrevious); - } - void setLogEntryAttachToNext(bool attachToNext) { - Expects(isLogEntry()); - setAttachToNext(attachToNext); - } - bool isUnderCursor() const; HistoryItem *previousItem() const; @@ -462,7 +354,7 @@ public: not_null controller, HistoryView::Context context) = 0; - ~HistoryItem(); + virtual ~HistoryItem(); protected: HistoryItem( @@ -476,14 +368,9 @@ protected: // a virtual method, it can not be done from constructor. virtual void finishCreate(); - // Called from resizeGetHeight() when MTPDmessage_ClientFlag::f_pending_init_dimensions is set. - virtual void initDimensions() = 0; - virtual void markMediaAsReadHook() { } - virtual int resizeContentGetHeight() = 0; - void finishEdition(int oldKeyboardTop); void finishEditionToEmpty(); @@ -491,29 +378,6 @@ protected: not_null _from; MTPDmessage::Flags _flags = 0; - // This should be called only from previousItemChanged() - // to add required bits to the Composer mask - // after that always use Has(). - void recountDisplayDate(); - - // This should be called only from previousItemChanged() or when - // HistoryMessageDate or HistoryMessageUnreadBar bit is changed in the Composer mask - // then the result should be cached in a client side flag MTPDmessage_ClientFlag::f_attach_to_previous. - void recountAttachToPrevious(); - - // This should be called only from recountDisplayDate(). - // Also this is called from setLogEntryDisplayDate() for channel log entries. - void setDisplayDate(bool displayDate); - - // This should be called only from recountAttachToPrevious(). - // Also this is called from setLogEntryAttachToPrevious() for channel log entries. - void setAttachToPrevious(bool attachToNext); - - // This should be called only from recountAttachToPrevious() of the next item - // or when the next item is removed through nextItemChanged() call. - // Also this is called from setLogEntryAttachToNext() for channel log entries. - void setAttachToNext(bool attachToNext); - const HistoryMessageReplyMarkup *inlineReplyMarkup() const { return const_cast(this)->inlineReplyMarkup(); } @@ -543,6 +407,7 @@ private: void resetGroupMedia(const std::vector> &others); HistoryView::Element *_mainView = nullptr; + friend class HistoryView::Element; }; @@ -562,8 +427,7 @@ public: }; -ClickHandlerPtr goToMessageClickHandler(PeerData *peer, MsgId msgId); - -inline ClickHandlerPtr goToMessageClickHandler(HistoryItem *item) { - return goToMessageClickHandler(item->history()->peer, item->id); -} +ClickHandlerPtr goToMessageClickHandler( + not_null peer, + MsgId msgId); +ClickHandlerPtr goToMessageClickHandler(not_null item); diff --git a/Telegram/SourceFiles/history/history_item_components.cpp b/Telegram/SourceFiles/history/history_item_components.cpp index 7f70cd312..fbfc6decf 100644 --- a/Telegram/SourceFiles/history/history_item_components.cpp +++ b/Telegram/SourceFiles/history/history_item_components.cpp @@ -16,6 +16,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "history/view/history_view_service_message.h" #include "media/media_audio.h" #include "media/player/media_player_instance.h" +#include "auth_session.h" +#include "data/data_session.h" #include "styles/style_widgets.h" #include "styles/style_history.h" @@ -160,7 +162,7 @@ bool HistoryMessageReply::updateData(HistoryMessage *holder, bool force) { replyToMsgId = 0; } if (force) { - holder->setPendingInitDimensions(); + Auth().data().requestItemViewResize(holder); } return (replyToMsg || !replyToMsgId); } @@ -211,10 +213,12 @@ void HistoryMessageReply::resize(int width) const { } } -void HistoryMessageReply::itemRemoved(HistoryMessage *holder, HistoryItem *removed) { +void HistoryMessageReply::itemRemoved( + HistoryMessage *holder, + HistoryItem *removed) { if (replyToMsg == removed) { clearData(holder); - holder->setPendingInitDimensions(); + Auth().data().requestItemViewResize(holder); } } diff --git a/Telegram/SourceFiles/history/history_media.cpp b/Telegram/SourceFiles/history/history_media.cpp index de34cea8b..0feae83f5 100644 --- a/Telegram/SourceFiles/history/history_media.cpp +++ b/Telegram/SourceFiles/history/history_media.cpp @@ -7,8 +7,32 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL */ #include "history/history_media.h" +#include "history/history_item.h" #include "storage/storage_shared_media.h" Storage::SharedMediaTypesMask HistoryMedia::sharedMediaTypes() const { return {}; } + +bool HistoryMedia::isDisplayed() const { + return !_parent->isHiddenByGroup(); +} + +QSize HistoryMedia::countCurrentSize(int newWidth) { + return QSize(qMin(newWidth, maxWidth()), minHeight()); +} + +TextSelection HistoryMedia::skipSelection(TextSelection selection) const { + return internal::unshiftSelection(selection, fullSelectionLength()); +} + +TextSelection HistoryMedia::unskipSelection(TextSelection selection) const { + return internal::shiftSelection(selection, fullSelectionLength()); +} + +HistoryTextState HistoryMedia::getStateGrouped( + const QRect &geometry, + QPoint point, + HistoryStateRequest request) const { + Unexpected("Grouping method call."); +} diff --git a/Telegram/SourceFiles/history/history_media.h b/Telegram/SourceFiles/history/history_media.h index 870dd3ad4..3bdc4d7c3 100644 --- a/Telegram/SourceFiles/history/history_media.h +++ b/Telegram/SourceFiles/history/history_media.h @@ -7,9 +7,12 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL */ #pragma once -#include "history/history_item.h" +#include "history/view/history_view_object.h" struct HistoryMessageEdited; +struct HistoryTextState; +struct HistoryStateRequest; +struct TextSelection; namespace base { template @@ -28,7 +31,26 @@ enum class MediaInBubbleState { Bottom, }; -class HistoryMedia : public HistoryElement { +enum HistoryMediaType : char { + MediaTypePhoto, + MediaTypeVideo, + MediaTypeContact, + MediaTypeCall, + MediaTypeFile, + MediaTypeGif, + MediaTypeSticker, + MediaTypeLocation, + MediaTypeWebPage, + MediaTypeMusicFile, + MediaTypeVoiceFile, + MediaTypeGame, + MediaTypeInvoice, + MediaTypeGrouped, + + MediaTypeCount +}; + +class HistoryMedia : public HistoryView::Object { public: HistoryMedia(not_null parent) : _parent(parent) { } @@ -48,12 +70,10 @@ public: virtual TextWithEntities selectedText(TextSelection selection) const = 0; bool hasPoint(QPoint point) const { - return QRect(0, 0, _width, _height).contains(point); + return QRect(0, 0, width(), height()).contains(point); } - virtual bool isDisplayed() const { - return !_parent->isHiddenByGroup(); - } + virtual bool isDisplayed() const; virtual void updateNeedBubbleState() { } virtual bool isAboveMessage() const { @@ -65,13 +85,8 @@ public: virtual bool allowsFastShare() const { return false; } - virtual void initDimensions() = 0; virtual void refreshParentId(not_null realParent) { } - virtual int resizeGetHeight(int width) { - _width = qMin(width, _maxw); - return _height; - } virtual void draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const = 0; virtual HistoryTextState getState(QPoint point, HistoryStateRequest request) const = 0; virtual void updatePressed(QPoint point) { @@ -102,17 +117,9 @@ public: return 0; } [[nodiscard]] TextSelection skipSelection( - TextSelection selection) const { - return internal::unshiftSelection( - selection, - fullSelectionLength()); - } + TextSelection selection) const; [[nodiscard]] TextSelection unskipSelection( - TextSelection selection) const { - return internal::shiftSelection( - selection, - fullSelectionLength()); - } + TextSelection selection) const; // if we press and drag this link should we drag the item virtual bool dragItemByHandler(const ClickHandlerPtr &p) const = 0; @@ -174,11 +181,9 @@ public: Unexpected("Grouping method call."); } virtual HistoryTextState getStateGrouped( - const QRect &geometry, - QPoint point, - HistoryStateRequest request) const { - Unexpected("Grouping method call."); - } + const QRect &geometry, + QPoint point, + HistoryStateRequest request) const; virtual std::unique_ptr takeLastFromGroup() { return nullptr; } @@ -252,18 +257,21 @@ public: return false; } - // Sometimes click on media in message is overloaded by the messsage: + // Sometimes click on media in message is overloaded by the message: // (for example it can open a link or a game instead of opening media) // But the overloading click handler should be used only when media - // is already loaded (not a photo or gif waiting for load with auto + // is already loaded (not a photo or GIF waiting for load with auto // load being disabled - in such case media should handle the click). virtual bool isReadyForOpen() const { return true; } + virtual ~HistoryMedia() = default; + protected: + QSize countCurrentSize(int newWidth) override; + not_null _parent; MediaInBubbleState _inBubbleState = MediaInBubbleState::None; - }; diff --git a/Telegram/SourceFiles/history/history_media_grouped.cpp b/Telegram/SourceFiles/history/history_media_grouped.cpp index 1bc53880b..476b7d3a5 100644 --- a/Telegram/SourceFiles/history/history_media_grouped.cpp +++ b/Telegram/SourceFiles/history/history_media_grouped.cpp @@ -38,7 +38,7 @@ std::unique_ptr HistoryGroupedMedia::clone( return main()->clone(newParent, realParent); } -void HistoryGroupedMedia::initDimensions() { +QSize HistoryGroupedMedia::countOptimalSize() { if (_caption.hasSkipBlock()) { _caption.setSkipBlock( _parent->skipBlockWidth(), @@ -60,33 +60,35 @@ void HistoryGroupedMedia::initDimensions() { st::historyGroupSkip); Assert(layout.size() == _elements.size()); - _maxw = _minh = 0; + auto maxWidth = 0; + auto minHeight = 0; for (auto i = 0, count = int(layout.size()); i != count; ++i) { const auto &item = layout[i]; - accumulate_max(_maxw, item.geometry.x() + item.geometry.width()); - accumulate_max(_minh, item.geometry.y() + item.geometry.height()); + accumulate_max(maxWidth, item.geometry.x() + item.geometry.width()); + accumulate_max(minHeight, item.geometry.y() + item.geometry.height()); _elements[i].initialGeometry = item.geometry; _elements[i].sides = item.sides; } if (!_caption.isEmpty()) { - auto captionw = _maxw - st::msgPadding.left() - st::msgPadding.right(); - _minh += st::mediaCaptionSkip + _caption.countHeight(captionw); + auto captionw = maxWidth - st::msgPadding.left() - st::msgPadding.right(); + minHeight += st::mediaCaptionSkip + _caption.countHeight(captionw); if (isBubbleBottom()) { - _minh += st::msgPadding.bottom(); + minHeight += st::msgPadding.bottom(); } } + return { maxWidth, minHeight }; } -int HistoryGroupedMedia::resizeGetHeight(int width) { - _width = std::min(width, _maxw); - _height = 0; - if (_width < st::historyGroupWidthMin) { - return _height; +QSize HistoryGroupedMedia::countCurrentSize(int newWidth) { + accumulate_min(newWidth, maxWidth()); + auto newHeight = 0; + if (newWidth < st::historyGroupWidthMin) { + return { newWidth, newHeight }; } const auto initialSpacing = st::historyGroupSkip; - const auto factor = _width / float64(_maxw); + const auto factor = newWidth / float64(maxWidth()); const auto scale = [&](int value) { return int(std::round(value * factor)); }; @@ -114,18 +116,18 @@ int HistoryGroupedMedia::resizeGetHeight(int width) { - (needBottomSkip ? spacing : 0); element.geometry = QRect(left, top, width, height); - accumulate_max(_height, top + height); + accumulate_max(newHeight, top + height); } if (!_caption.isEmpty()) { - const auto captionw = _width - st::msgPadding.left() - st::msgPadding.right(); - _height += st::mediaCaptionSkip + _caption.countHeight(captionw); + const auto captionw = newWidth - st::msgPadding.left() - st::msgPadding.right(); + newHeight += st::mediaCaptionSkip + _caption.countHeight(captionw); if (isBubbleBottom()) { - _height += st::msgPadding.bottom(); + newHeight += st::msgPadding.bottom(); } } - return _height; + return { newWidth, newHeight }; } void HistoryGroupedMedia::refreshParentId( @@ -168,23 +170,23 @@ void HistoryGroupedMedia::draw( // date const auto selected = (selection == FullSelection); if (!_caption.isEmpty()) { - const auto captionw = _width - st::msgPadding.left() - st::msgPadding.right(); + const auto captionw = width() - st::msgPadding.left() - st::msgPadding.right(); const auto outbg = _parent->hasOutLayout(); - const auto captiony = _height + const auto captiony = height() - (isBubbleBottom() ? st::msgPadding.bottom() : 0) - _caption.countHeight(captionw); p.setPen(outbg ? (selected ? st::historyTextOutFgSelected : st::historyTextOutFg) : (selected ? st::historyTextInFgSelected : st::historyTextInFg)); _caption.draw(p, st::msgPadding.left(), captiony, captionw, style::al_left, 0, -1, selection); } else if (_parent->getMedia() == this) { - auto fullRight = _width; - auto fullBottom = _height; + auto fullRight = width(); + auto fullBottom = height(); if (needInfoDisplay()) { - _parent->drawInfo(p, fullRight, fullBottom, _width, selected, InfoDisplayOverImage); + _parent->drawInfo(p, fullRight, fullBottom, width(), selected, InfoDisplayOverImage); } if (!_parent->hasBubble() && _parent->displayRightAction()) { auto fastShareLeft = (fullRight + st::historyFastShareLeft); auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize); - _parent->drawRightAction(p, fastShareLeft, fastShareTop, _width); + _parent->drawRightAction(p, fastShareLeft, fastShareTop, width()); } } } @@ -210,19 +212,19 @@ HistoryTextState HistoryGroupedMedia::getState( HistoryStateRequest request) const { auto result = getElementState(point, request); if (!result.link && !_caption.isEmpty()) { - const auto captionw = _width - st::msgPadding.left() - st::msgPadding.right(); - const auto captiony = _height + const auto captionw = width() - st::msgPadding.left() - st::msgPadding.right(); + const auto captiony = height() - (isBubbleBottom() ? st::msgPadding.bottom() : 0) - _caption.countHeight(captionw); - if (QRect(st::msgPadding.left(), captiony, captionw, _height - captiony).contains(point)) { + if (QRect(st::msgPadding.left(), captiony, captionw, height() - captiony).contains(point)) { return HistoryTextState(_parent, _caption.getState( point - QPoint(st::msgPadding.left(), captiony), captionw, request.forText())); } } else if (_parent->getMedia() == this) { - auto fullRight = _width; - auto fullBottom = _height; + auto fullRight = width(); + auto fullBottom = height(); if (_parent->pointInTime(fullRight, fullBottom, point, InfoDisplayOverImage)) { result.cursor = HistoryInDateCursorState; } @@ -347,7 +349,7 @@ bool HistoryGroupedMedia::applyGroup( pushElement(item); } _elements.push_back(std::move(mainElement)); - _parent->setPendingInitDimensions(); + //_parent->setPendingInitDimensions(); // #TODO group view return true; } @@ -452,7 +454,8 @@ bool HistoryGroupedMedia::computeNeedBubble() const { if (message->viaBot() || message->Has() || message->displayForwardedFrom() - || message->displayFromName()) { +// || message->displayFromName() // #TODO media views + ) { return true; } } diff --git a/Telegram/SourceFiles/history/history_media_grouped.h b/Telegram/SourceFiles/history/history_media_grouped.h index d5d8db1af..d6ee6b814 100644 --- a/Telegram/SourceFiles/history/history_media_grouped.h +++ b/Telegram/SourceFiles/history/history_media_grouped.h @@ -24,8 +24,6 @@ public: not_null newParent, not_null realParent) const override; - void initDimensions() override; - int resizeGetHeight(int width) override; void refreshParentId(not_null realParent) override; void updateSentMedia(const MTPMessageMedia &media) override; bool needReSetInlineResultMedia(const MTPMessageMedia &media) override; @@ -115,6 +113,9 @@ private: }; + QSize countOptimalSize() override; + QSize countCurrentSize(int newWidth) override; + bool needInfoDisplay() const; bool computeNeedBubble() const; not_null main() const; diff --git a/Telegram/SourceFiles/history/history_media_types.cpp b/Telegram/SourceFiles/history/history_media_types.cpp index 9f0d03027..a81b65d63 100644 --- a/Telegram/SourceFiles/history/history_media_types.cpp +++ b/Telegram/SourceFiles/history/history_media_types.cpp @@ -20,6 +20,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "boxes/confirm_box.h" #include "boxes/add_contact_box.h" #include "core/click_handler_types.h" +#include "history/history.h" #include "history/history_item_components.h" #include "history/history_location_manager.h" #include "history/history_message.h" @@ -55,25 +56,25 @@ bool needReSetInlineResultDocument(const MTPMessageMedia &media, DocumentData *e return true; } -int32 documentMaxStatusWidth(DocumentData *document) { - int32 result = st::normalFont->width(formatDownloadText(document->size, document->size)); +int documentMaxStatusWidth(DocumentData *document) { + auto result = st::normalFont->width(formatDownloadText(document->size, document->size)); if (const auto song = document->song()) { - result = qMax(result, st::normalFont->width(formatPlayedText(song->duration, song->duration))); - result = qMax(result, st::normalFont->width(formatDurationAndSizeText(song->duration, document->size))); + accumulate_max(result, st::normalFont->width(formatPlayedText(song->duration, song->duration))); + accumulate_max(result, st::normalFont->width(formatDurationAndSizeText(song->duration, document->size))); } else if (const auto voice = document->voice()) { - result = qMax(result, st::normalFont->width(formatPlayedText(voice->duration, voice->duration))); - result = qMax(result, st::normalFont->width(formatDurationAndSizeText(voice->duration, document->size))); + accumulate_max(result, st::normalFont->width(formatPlayedText(voice->duration, voice->duration))); + accumulate_max(result, st::normalFont->width(formatDurationAndSizeText(voice->duration, document->size))); } else if (document->isVideoFile()) { - result = qMax(result, st::normalFont->width(formatDurationAndSizeText(document->duration(), document->size))); + accumulate_max(result, st::normalFont->width(formatDurationAndSizeText(document->duration(), document->size))); } else { - result = qMax(result, st::normalFont->width(formatSizeText(document->size))); + accumulate_max(result, st::normalFont->width(formatSizeText(document->size))); } return result; } -int32 gifMaxStatusWidth(DocumentData *document) { - int32 result = st::normalFont->width(formatDownloadText(document->size, document->size)); - result = qMax(result, st::normalFont->width(formatGifAndSizeText(document->size))); +int gifMaxStatusWidth(DocumentData *document) { + auto result = st::normalFont->width(formatDownloadText(document->size, document->size)); + accumulate_max(result, st::normalFont->width(formatGifAndSizeText(document->size))); return result; } @@ -161,7 +162,7 @@ void HistoryFileMedia::refreshParentId(not_null realParent) { _cancell->setMessageId(contextId); } -void HistoryFileMedia::setStatusSize(int32 newSize, int32 fullSize, int32 duration, qint64 realDuration) const { +void HistoryFileMedia::setStatusSize(int newSize, int fullSize, int duration, qint64 realDuration) const { _statusSize = newSize; if (_statusSize == FileStatusSizeReady) { _statusText = (duration >= 0) ? formatDurationAndSizeText(duration, fullSize) : (duration < -1 ? formatGifAndSizeText(fullSize) : formatSizeText(fullSize)); @@ -200,6 +201,29 @@ void HistoryFileMedia::checkAnimationFinished() const { } } } +void HistoryFileMedia::setDocumentLinks( + not_null document, + not_null realParent, + bool inlinegif) { + FileClickHandlerPtr open, save; + const auto context = realParent->fullId(); + if (inlinegif) { + open = std::make_shared(document, context); + } else { + open = std::make_shared(document, context); + } + if (inlinegif) { + save = std::make_shared(document, context); + } else if (document->isVoiceMessage()) { + save = std::make_shared(document, context); + } else { + save = std::make_shared(document, context); + } + setLinks( + std::move(open), + std::move(save), + std::make_shared(document, context)); +} HistoryFileMedia::~HistoryFileMedia() = default; @@ -230,14 +254,13 @@ HistoryPhoto::HistoryPhoto( not_null photo, int width) : HistoryFileMedia(parent) -, _data(photo) { +, _data(photo) +, _serviceWidth(width) { const auto fullId = parent->fullId(); setLinks( std::make_shared(_data, fullId, chat), std::make_shared(_data, fullId, chat), std::make_shared(_data, fullId, chat)); - - _width = width; init(); } @@ -271,12 +294,16 @@ void HistoryPhoto::init() { _data->thumb->load(); } -void HistoryPhoto::initDimensions() { +QSize HistoryPhoto::countOptimalSize() { if (_caption.hasSkipBlock()) { _caption.setSkipBlock(_parent->skipBlockWidth(), _parent->skipBlockHeight()); } - int32 tw = convertScale(_data->full->width()), th = convertScale(_data->full->height()); + auto maxWidth = 0; + auto minHeight = 0; + + auto tw = convertScale(_data->full->width()); + auto th = convertScale(_data->full->height()); if (!tw || !th) { tw = th = 1; } @@ -289,24 +316,24 @@ void HistoryPhoto::initDimensions() { th = st::maxMediaSize; } - if (_parent->toHistoryMessage()) { - int32 minWidth = qMax(st::minPhotoSize, _parent->infoWidth() + 2 * (st::msgDateImgDelta + st::msgDateImgPadding.x())); - int32 maxActualWidth = qMax(tw, minWidth); - _maxw = qMax(maxActualWidth, th); - _minh = qMax(th, int32(st::minPhotoSize)); - if (_parent->hasBubble() && !_caption.isEmpty()) { - auto captionw = maxActualWidth - st::msgPadding.left() - st::msgPadding.right(); - _minh += st::mediaCaptionSkip + _caption.countHeight(captionw); - if (isBubbleBottom()) { - _minh += st::msgPadding.bottom(); - } - } - } else { - _maxw = _minh = _width; + if (!_parent->toHistoryMessage()) { + return { _serviceWidth, _serviceWidth }; } + const auto minWidth = qMax(st::minPhotoSize, _parent->infoWidth() + 2 * (st::msgDateImgDelta + st::msgDateImgPadding.x())); + const auto maxActualWidth = qMax(tw, minWidth); + maxWidth = qMax(maxActualWidth, th); + minHeight = qMax(th, st::minPhotoSize); + if (_parent->hasBubble() && !_caption.isEmpty()) { + auto captionw = maxActualWidth - st::msgPadding.left() - st::msgPadding.right(); + minHeight += st::mediaCaptionSkip + _caption.countHeight(captionw); + if (isBubbleBottom()) { + minHeight += st::msgPadding.bottom(); + } + } + return { maxWidth, minHeight }; } -int HistoryPhoto::resizeGetHeight(int width) { +QSize HistoryPhoto::countCurrentSize(int newWidth) { int tw = convertScale(_data->full->width()), th = convertScale(_data->full->height()); if (tw > st::maxMediaSize) { th = (st::maxMediaSize * th) / tw; @@ -317,47 +344,48 @@ int HistoryPhoto::resizeGetHeight(int width) { th = st::maxMediaSize; } - _pixw = qMin(width, _maxw); + _pixw = qMin(newWidth, maxWidth()); _pixh = th; if (tw > _pixw) { _pixh = (_pixw * _pixh / tw); } else { _pixw = tw; } - if (_pixh > width) { - _pixw = (_pixw * width) / _pixh; - _pixh = width; + if (_pixh > newWidth) { + _pixw = (_pixw * newWidth) / _pixh; + _pixh = newWidth; } if (_pixw < 1) _pixw = 1; if (_pixh < 1) _pixh = 1; - int minWidth = qMax(st::minPhotoSize, _parent->infoWidth() + 2 * (st::msgDateImgDelta + st::msgDateImgPadding.x())); - _width = qMax(_pixw, int16(minWidth)); - _height = qMax(_pixh, int16(st::minPhotoSize)); + auto minWidth = qMax(st::minPhotoSize, _parent->infoWidth() + 2 * (st::msgDateImgDelta + st::msgDateImgPadding.x())); + newWidth = qMax(_pixw, minWidth); + auto newHeight = qMax(_pixh, st::minPhotoSize); if (_parent->hasBubble() && !_caption.isEmpty()) { - const auto captionw = _width + const auto captionw = newWidth - st::msgPadding.left() - st::msgPadding.right(); - _height += st::mediaCaptionSkip + _caption.countHeight(captionw); + newHeight += st::mediaCaptionSkip + _caption.countHeight(captionw); if (isBubbleBottom()) { - _height += st::msgPadding.bottom(); + newHeight += st::msgPadding.bottom(); } } - return _height; + return { newWidth, newHeight }; } void HistoryPhoto::draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const { - if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return; + if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) return; _data->automaticLoad(_parent); - bool selected = (selection == FullSelection); - bool loaded = _data->loaded(), displayLoading = _data->displayLoading(); + auto selected = (selection == FullSelection); + auto loaded = _data->loaded(); + auto displayLoading = _data->displayLoading(); - bool notChild = (_parent->getMedia() == this); - auto skipx = 0, skipy = 0, width = _width, height = _height; - bool bubble = _parent->hasBubble(); + auto notChild = (_parent->getMedia() == this); + auto paintx = 0, painty = 0, paintw = width(), painth = height(); + auto bubble = _parent->hasBubble(); - int captionw = width - st::msgPadding.left() - st::msgPadding.right(); + auto captionw = paintw - st::msgPadding.left() - st::msgPadding.right(); if (displayLoading) { ensureAnimation(); @@ -367,26 +395,26 @@ void HistoryPhoto::draw(Painter &p, const QRect &r, TextSelection selection, Tim } bool radial = isRadialAnimation(ms); - auto rthumb = rtlrect(skipx, skipy, width, height, _width); + auto rthumb = rtlrect(paintx, painty, paintw, painth, width()); if (_parent->toHistoryMessage()) { if (bubble) { if (!_caption.isEmpty()) { - height -= st::mediaCaptionSkip + _caption.countHeight(captionw); + painth -= st::mediaCaptionSkip + _caption.countHeight(captionw); if (isBubbleBottom()) { - height -= st::msgPadding.bottom(); + painth -= st::msgPadding.bottom(); } - rthumb = rtlrect(skipx, skipy, width, height, _width); + rthumb = rtlrect(paintx, painty, paintw, painth, width()); } } else { - App::roundShadow(p, 0, 0, width, height, selected ? st::msgInShadowSelected : st::msgInShadow, selected ? InSelectedShadowCorners : InShadowCorners); + App::roundShadow(p, 0, 0, paintw, painth, selected ? st::msgInShadowSelected : st::msgInShadow, selected ? InSelectedShadowCorners : InShadowCorners); } auto inWebPage = (_parent->getMedia() != this); auto roundRadius = inWebPage ? ImageRoundRadius::Small : ImageRoundRadius::Large; auto roundCorners = inWebPage ? RectPart::AllCorners : ((isBubbleTop() ? (RectPart::TopLeft | RectPart::TopRight) : RectPart::None) | ((isBubbleBottom() && _caption.isEmpty()) ? (RectPart::BottomLeft | RectPart::BottomRight) : RectPart::None)); const auto pix = loaded - ? _data->full->pixSingle(_pixw, _pixh, width, height, roundRadius, roundCorners) - : _data->thumb->pixBlurredSingle(_pixw, _pixh, width, height, roundRadius, roundCorners); + ? _data->full->pixSingle(_pixw, _pixh, paintw, painth, roundRadius, roundCorners) + : _data->thumb->pixBlurredSingle(_pixw, _pixh, paintw, painth, roundRadius, roundCorners); p.drawPixmap(rthumb.topLeft(), pix); if (selected) { App::complexOverlayRect(p, rthumb, roundRadius, roundCorners); @@ -445,17 +473,17 @@ void HistoryPhoto::draw(Painter &p, const QRect &r, TextSelection selection, Tim if (!_caption.isEmpty()) { auto outbg = _parent->hasOutLayout(); p.setPen(outbg ? (selected ? st::historyTextOutFgSelected : st::historyTextOutFg) : (selected ? st::historyTextInFgSelected : st::historyTextInFg)); - _caption.draw(p, st::msgPadding.left(), skipy + height + st::mediaCaptionSkip, captionw, style::al_left, 0, -1, selection); + _caption.draw(p, st::msgPadding.left(), painty + painth + st::mediaCaptionSkip, captionw, style::al_left, 0, -1, selection); } else if (notChild) { - auto fullRight = skipx + width; - auto fullBottom = skipy + height; + auto fullRight = paintx + paintw; + auto fullBottom = painty + painth; if (needInfoDisplay()) { - _parent->drawInfo(p, fullRight, fullBottom, 2 * skipx + width, selected, InfoDisplayOverImage); + _parent->drawInfo(p, fullRight, fullBottom, 2 * paintx + paintw, selected, InfoDisplayOverImage); } if (!bubble && _parent->displayRightAction()) { auto fastShareLeft = (fullRight + st::historyFastShareLeft); auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize); - _parent->drawRightAction(p, fastShareLeft, fastShareTop, 2 * skipx + width); + _parent->drawRightAction(p, fastShareLeft, fastShareTop, 2 * paintx + paintw); } } } @@ -463,30 +491,30 @@ void HistoryPhoto::draw(Painter &p, const QRect &r, TextSelection selection, Tim HistoryTextState HistoryPhoto::getState(QPoint point, HistoryStateRequest request) const { auto result = HistoryTextState(_parent); - if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) { + if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) { return result; } - auto skipx = 0, skipy = 0, width = _width, height = _height; + auto paintx = 0, painty = 0, paintw = width(), painth = height(); auto bubble = _parent->hasBubble(); if (bubble && !_caption.isEmpty()) { - const auto captionw = width + const auto captionw = paintw - st::msgPadding.left() - st::msgPadding.right(); - height -= _caption.countHeight(captionw); + painth -= _caption.countHeight(captionw); if (isBubbleBottom()) { - height -= st::msgPadding.bottom(); + painth -= st::msgPadding.bottom(); } - if (QRect(st::msgPadding.left(), height, captionw, _height - height).contains(point)) { + if (QRect(st::msgPadding.left(), painth, captionw, height() - painth).contains(point)) { result = HistoryTextState(_parent, _caption.getState( - point - QPoint(st::msgPadding.left(), height), + point - QPoint(st::msgPadding.left(), painth), captionw, request.forText())); return result; } - height -= st::mediaCaptionSkip; + painth -= st::mediaCaptionSkip; } - if (QRect(skipx, skipy, width, height).contains(point)) { + if (QRect(paintx, painty, paintw, painth).contains(point)) { if (_data->uploading()) { result.link = _cancell; } else if (_data->loaded()) { @@ -501,8 +529,8 @@ HistoryTextState HistoryPhoto::getState(QPoint point, HistoryStateRequest reques } } if (_caption.isEmpty() && _parent->getMedia() == this) { - auto fullRight = skipx + width; - auto fullBottom = skipy + height; + auto fullRight = paintx + paintw; + auto fullBottom = painty + painth; if (_parent->pointInTime(fullRight, fullBottom, point, InfoDisplayOverImage)) { result.cursor = HistoryInDateCursorState; } @@ -550,7 +578,7 @@ void HistoryPhoto::drawGrouped( const auto radial = isRadialAnimation(ms); if (!bubble) { -// App::roundShadow(p, 0, 0, width, height, selected ? st::msgInShadowSelected : st::msgInShadow, selected ? InSelectedShadowCorners : InShadowCorners); +// App::roundShadow(p, 0, 0, paintw, painth, selected ? st::msgInShadowSelected : st::msgInShadow, selected ? InSelectedShadowCorners : InShadowCorners); } p.drawPixmap(geometry.topLeft(), *cache); if (selected) { @@ -703,9 +731,9 @@ void HistoryPhoto::updateSentMedia(const MTPMessageMedia &media) { if (photo.type() == mtpc_photo) { auto &sizes = photo.c_photo().vsizes.v; - int32 max = 0; + auto max = 0; const MTPDfileLocation *maxLocation = 0; - for (int32 i = 0, l = sizes.size(); i < l; ++i) { + for (auto i = 0, l = int(sizes.size()); i != l; ++i) { char size = 0; const MTPFileLocation *loc = 0; switch (sizes.at(i).type()) { @@ -793,7 +821,8 @@ bool HistoryPhoto::needsBubble() const { return message->viaBot() || message->Has() || message->displayForwardedFrom() - || message->displayFromName(); +// || message->displayFromName() // #TODO media views +; } return false; } @@ -847,12 +876,13 @@ HistoryVideo::HistoryVideo( setStatusSize(other._statusSize); } -void HistoryVideo::initDimensions() { +QSize HistoryVideo::countOptimalSize() { if (_caption.hasSkipBlock()) { _caption.setSkipBlock(_parent->skipBlockWidth(), _parent->skipBlockHeight()); } - int32 tw = convertScale(_data->thumb->width()), th = convertScale(_data->thumb->height()); + auto tw = convertScale(_data->thumb->width()); + auto th = convertScale(_data->thumb->height()); if (!tw || !th) { tw = th = 1; } @@ -865,22 +895,23 @@ void HistoryVideo::initDimensions() { } _thumbw = qMax(tw, 1); - int32 minWidth = qMax(st::minPhotoSize, _parent->infoWidth() + 2 * (st::msgDateImgDelta + st::msgDateImgPadding.x())); - minWidth = qMax(minWidth, documentMaxStatusWidth(_data) + 2 * int32(st::msgDateImgDelta + st::msgDateImgPadding.x())); - _maxw = qMax(_thumbw, int32(minWidth)); - _minh = qMax(th, int32(st::minPhotoSize)); + auto minWidth = qMax(st::minPhotoSize, _parent->infoWidth() + 2 * (st::msgDateImgDelta + st::msgDateImgPadding.x())); + minWidth = qMax(minWidth, documentMaxStatusWidth(_data) + 2 * (st::msgDateImgDelta + st::msgDateImgPadding.x())); + auto maxWidth = qMax(_thumbw, minWidth); + auto minHeight = qMax(th, st::minPhotoSize); if (_parent->hasBubble() && !_caption.isEmpty()) { - const auto captionw = _maxw + const auto captionw = maxWidth - st::msgPadding.left() - st::msgPadding.right(); - _minh += st::mediaCaptionSkip + _caption.countHeight(captionw); + minHeight += st::mediaCaptionSkip + _caption.countHeight(captionw); if (isBubbleBottom()) { - _minh += st::msgPadding.bottom(); + minHeight += st::msgPadding.bottom(); } } + return { maxWidth, minHeight }; } -int HistoryVideo::resizeGetHeight(int width) { +QSize HistoryVideo::countCurrentSize(int newWidth) { int tw = convertScale(_data->thumb->width()), th = convertScale(_data->thumb->height()); if (!tw || !th) { tw = th = 1; @@ -893,39 +924,39 @@ int HistoryVideo::resizeGetHeight(int width) { th = st::msgVideoSize.height(); } - if (width < tw) { - th = qRound((width / float64(tw)) * th); - tw = width; + if (newWidth < tw) { + th = qRound((newWidth / float64(tw)) * th); + tw = newWidth; } _thumbw = qMax(tw, 1); - int minWidth = qMax(st::minPhotoSize, _parent->infoWidth() + 2 * (st::msgDateImgDelta + st::msgDateImgPadding.x())); - minWidth = qMax(minWidth, documentMaxStatusWidth(_data) + 2 * int(st::msgDateImgDelta + st::msgDateImgPadding.x())); - _width = qMax(_thumbw, int(minWidth)); - _height = qMax(th, int(st::minPhotoSize)); + auto minWidth = qMax(st::minPhotoSize, _parent->infoWidth() + 2 * (st::msgDateImgDelta + st::msgDateImgPadding.x())); + minWidth = qMax(minWidth, documentMaxStatusWidth(_data) + 2 * (st::msgDateImgDelta + st::msgDateImgPadding.x())); + newWidth = qMax(_thumbw, minWidth); + auto newHeight = qMax(th, st::minPhotoSize); if (_parent->hasBubble() && !_caption.isEmpty()) { - const auto captionw = _width + const auto captionw = newWidth - st::msgPadding.left() - st::msgPadding.right(); - _height += st::mediaCaptionSkip + _caption.countHeight(captionw); + newHeight += st::mediaCaptionSkip + _caption.countHeight(captionw); if (isBubbleBottom()) { - _height += st::msgPadding.bottom(); + newHeight += st::msgPadding.bottom(); } } - return _height; + return { newWidth, newHeight }; } void HistoryVideo::draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const { - if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return; + if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) return; _data->automaticLoad(_parent); bool loaded = _data->loaded(), displayLoading = _data->displayLoading(); bool selected = (selection == FullSelection); - auto skipx = 0, skipy = 0, width = _width, height = _height; + auto paintx = 0, painty = 0, paintw = width(), painth = height(); bool bubble = _parent->hasBubble(); - int captionw = width - st::msgPadding.left() - st::msgPadding.right(); + int captionw = paintw - st::msgPadding.left() - st::msgPadding.right(); if (displayLoading) { ensureAnimation(); @@ -938,21 +969,21 @@ void HistoryVideo::draw(Painter &p, const QRect &r, TextSelection selection, Tim if (bubble) { if (!_caption.isEmpty()) { - height -= st::mediaCaptionSkip + _caption.countHeight(captionw); + painth -= st::mediaCaptionSkip + _caption.countHeight(captionw); if (isBubbleBottom()) { - height -= st::msgPadding.bottom(); + painth -= st::msgPadding.bottom(); } } } else { - App::roundShadow(p, 0, 0, width, height, selected ? st::msgInShadowSelected : st::msgInShadow, selected ? InSelectedShadowCorners : InShadowCorners); + App::roundShadow(p, 0, 0, paintw, painth, selected ? st::msgInShadowSelected : st::msgInShadow, selected ? InSelectedShadowCorners : InShadowCorners); } auto inWebPage = (_parent->getMedia() != this); auto roundRadius = inWebPage ? ImageRoundRadius::Small : ImageRoundRadius::Large; auto roundCorners = inWebPage ? RectPart::AllCorners : ((isBubbleTop() ? (RectPart::TopLeft | RectPart::TopRight) : RectPart::None) | ((isBubbleBottom() && _caption.isEmpty()) ? (RectPart::BottomLeft | RectPart::BottomRight) : RectPart::None)); - QRect rthumb(rtlrect(skipx, skipy, width, height, _width)); - p.drawPixmap(rthumb.topLeft(), _data->thumb->pixBlurredSingle(_thumbw, 0, width, height, roundRadius, roundCorners)); + QRect rthumb(rtlrect(paintx, painty, paintw, painth, width())); + p.drawPixmap(rthumb.topLeft(), _data->thumb->pixBlurredSingle(_thumbw, 0, paintw, painth, roundRadius, roundCorners)); if (selected) { App::complexOverlayRect(p, rthumb, roundRadius, roundCorners); } @@ -997,58 +1028,58 @@ void HistoryVideo::draw(Painter &p, const QRect &r, TextSelection selection, Tim _animation->radial.draw(p, rinner, st::msgFileRadialLine, selected ? st::historyFileThumbRadialFgSelected : st::historyFileThumbRadialFg); } - auto statusX = skipx + st::msgDateImgDelta + st::msgDateImgPadding.x(), statusY = skipy + st::msgDateImgDelta + st::msgDateImgPadding.y(); + auto statusX = paintx + st::msgDateImgDelta + st::msgDateImgPadding.x(), statusY = painty + st::msgDateImgDelta + st::msgDateImgPadding.y(); auto statusW = st::normalFont->width(_statusText) + 2 * st::msgDateImgPadding.x(); auto statusH = st::normalFont->height + 2 * st::msgDateImgPadding.y(); - App::roundRect(p, rtlrect(statusX - st::msgDateImgPadding.x(), statusY - st::msgDateImgPadding.y(), statusW, statusH, _width), selected ? st::msgDateImgBgSelected : st::msgDateImgBg, selected ? DateSelectedCorners : DateCorners); + App::roundRect(p, rtlrect(statusX - st::msgDateImgPadding.x(), statusY - st::msgDateImgPadding.y(), statusW, statusH, width()), selected ? st::msgDateImgBgSelected : st::msgDateImgBg, selected ? DateSelectedCorners : DateCorners); p.setFont(st::normalFont); p.setPen(st::msgDateImgFg); - p.drawTextLeft(statusX, statusY, _width, _statusText, statusW - 2 * st::msgDateImgPadding.x()); + p.drawTextLeft(statusX, statusY, width(), _statusText, statusW - 2 * st::msgDateImgPadding.x()); // date if (!_caption.isEmpty()) { auto outbg = _parent->hasOutLayout(); p.setPen(outbg ? (selected ? st::historyTextOutFgSelected : st::historyTextOutFg) : (selected ? st::historyTextInFgSelected : st::historyTextInFg)); - _caption.draw(p, st::msgPadding.left(), skipy + height + st::mediaCaptionSkip, captionw, style::al_left, 0, -1, selection); + _caption.draw(p, st::msgPadding.left(), painty + painth + st::mediaCaptionSkip, captionw, style::al_left, 0, -1, selection); } else if (_parent->getMedia() == this) { - auto fullRight = skipx + width, fullBottom = skipy + height; - _parent->drawInfo(p, fullRight, fullBottom, 2 * skipx + width, selected, InfoDisplayOverImage); + auto fullRight = paintx + paintw, fullBottom = painty + painth; + _parent->drawInfo(p, fullRight, fullBottom, 2 * paintx + paintw, selected, InfoDisplayOverImage); if (!bubble && _parent->displayRightAction()) { auto fastShareLeft = (fullRight + st::historyFastShareLeft); auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize); - _parent->drawRightAction(p, fastShareLeft, fastShareTop, 2 * skipx + width); + _parent->drawRightAction(p, fastShareLeft, fastShareTop, 2 * paintx + paintw); } } } HistoryTextState HistoryVideo::getState(QPoint point, HistoryStateRequest request) const { - if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) { + if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) { return {}; } auto result = HistoryTextState(_parent); bool loaded = _data->loaded(); - auto skipx = 0, skipy = 0, width = _width, height = _height; + auto paintx = 0, painty = 0, paintw = width(), painth = height(); bool bubble = _parent->hasBubble(); if (bubble && !_caption.isEmpty()) { - const auto captionw = width + const auto captionw = paintw - st::msgPadding.left() - st::msgPadding.right(); - height -= _caption.countHeight(captionw); + painth -= _caption.countHeight(captionw); if (isBubbleBottom()) { - height -= st::msgPadding.bottom(); + painth -= st::msgPadding.bottom(); } - if (QRect(st::msgPadding.left(), height, captionw, _height - height).contains(point)) { + if (QRect(st::msgPadding.left(), painth, captionw, height() - painth).contains(point)) { result = HistoryTextState(_parent, _caption.getState( - point - QPoint(st::msgPadding.left(), height), + point - QPoint(st::msgPadding.left(), painth), captionw, request.forText())); } - height -= st::mediaCaptionSkip; + painth -= st::mediaCaptionSkip; } - if (QRect(skipx, skipy, width, height).contains(point)) { + if (QRect(paintx, painty, paintw, painth).contains(point)) { if (_data->uploading()) { result.link = _cancell; } else { @@ -1056,8 +1087,8 @@ HistoryTextState HistoryVideo::getState(QPoint point, HistoryStateRequest reques } } if (_caption.isEmpty() && _parent->getMedia() == this) { - auto fullRight = skipx + width; - auto fullBottom = skipy + height; + auto fullRight = paintx + paintw; + auto fullBottom = painty + painth; if (_parent->pointInTime(fullRight, fullBottom, point, InfoDisplayOverImage)) { result.cursor = HistoryInDateCursorState; } @@ -1105,7 +1136,7 @@ void HistoryVideo::drawGrouped( const auto radial = isRadialAnimation(ms); if (!bubble) { -// App::roundShadow(p, 0, 0, width, height, selected ? st::msgInShadowSelected : st::msgInShadow, selected ? InSelectedShadowCorners : InShadowCorners); +// App::roundShadow(p, 0, 0, paintw, painth, selected ? st::msgInShadowSelected : st::msgInShadow, selected ? InSelectedShadowCorners : InShadowCorners); } p.drawPixmap(geometry.topLeft(), *cache); if (selected) { @@ -1235,7 +1266,7 @@ void HistoryVideo::validateGroupedCache( *cache = image->pixNoCache(pixWidth, pixHeight, options, width, height); } -void HistoryVideo::setStatusSize(int32 newSize) const { +void HistoryVideo::setStatusSize(int newSize) const { HistoryFileMedia::setStatusSize(newSize, _data->size, _data->duration(), 0); } @@ -1262,7 +1293,8 @@ bool HistoryVideo::needsBubble() const { return message->viaBot() || message->Has() || message->displayForwardedFrom() - || message->displayFromName(); +// || message->displayFromName() // #TODO media views +; } return false; } @@ -1275,8 +1307,9 @@ Storage::SharedMediaTypesMask HistoryVideo::sharedMediaTypes() const { } void HistoryVideo::updateStatusText() const { - bool showPause = false; - int32 statusSize = 0, realDuration = 0; + auto showPause = false; + auto statusSize = 0; + auto realDuration = 0; if (_data->status == FileDownloadFailed || _data->status == FileUploadFailed) { statusSize = FileStatusSizeFailed; } else if (_data->uploading()) { @@ -1319,7 +1352,8 @@ bool HistoryVideo::needReSetInlineResultMedia(const MTPMessageMedia &media) { ImagePtr HistoryVideo::replyPreview() { if (_data->replyPreview->isNull() && !_data->thumb->isNull()) { if (_data->thumb->loaded()) { - int w = convertScale(_data->thumb->width()), h = convertScale(_data->thumb->height()); + auto w = convertScale(_data->thumb->width()); + auto h = convertScale(_data->thumb->height()); if (w <= 0) w = 1; if (h <= 0) h = 1; _data->replyPreview = ImagePtr(w > h ? _data->thumb->pix(w * st::msgReplyBarSize.height() / h, st::msgReplyBarSize.height()) : _data->thumb->pix(st::msgReplyBarSize.height()), "PNG"); @@ -1428,7 +1462,7 @@ void HistoryDocument::fillNamedFromData(HistoryDocumentNamed *named) { named->_namew = st::semiboldFont->width(nameString); } -void HistoryDocument::initDimensions() { +QSize HistoryDocument::countOptimalSize() { auto captioned = Get(); if (captioned && captioned->_caption.hasSkipBlock()) { captioned->_caption.setSkipBlock(_parent->skipBlockWidth(), _parent->skipBlockHeight()); @@ -1437,7 +1471,8 @@ void HistoryDocument::initDimensions() { auto thumbed = Get(); if (thumbed) { _data->thumb->load(); - int32 tw = convertScale(_data->thumb->width()), th = convertScale(_data->thumb->height()); + auto tw = convertScale(_data->thumb->width()); + auto th = convertScale(_data->thumb->height()); if (tw > th) { thumbed->_thumbw = (tw * st::msgFileThumbSize) / th; } else { @@ -1445,82 +1480,84 @@ void HistoryDocument::initDimensions() { } } - _maxw = st::msgFileMinWidth; + auto maxWidth = st::msgFileMinWidth; - int32 tleft = 0, tright = 0; + auto tleft = 0; + auto tright = 0; if (thumbed) { tleft = st::msgFileThumbPadding.left() + st::msgFileThumbSize + st::msgFileThumbPadding.right(); tright = st::msgFileThumbPadding.left(); - _maxw = qMax(_maxw, tleft + documentMaxStatusWidth(_data) + tright); + accumulate_max(maxWidth, tleft + documentMaxStatusWidth(_data) + tright); } else { tleft = st::msgFilePadding.left() + st::msgFileSize + st::msgFilePadding.right(); tright = st::msgFileThumbPadding.left(); auto unread = _data->isVoiceMessage() ? (st::mediaUnreadSkip + st::mediaUnreadSize) : 0; - _maxw = qMax(_maxw, tleft + documentMaxStatusWidth(_data) + unread + _parent->skipBlockWidth() + st::msgPadding.right()); + accumulate_max(maxWidth, tleft + documentMaxStatusWidth(_data) + unread + _parent->skipBlockWidth() + st::msgPadding.right()); } if (auto named = Get()) { - _maxw = qMax(tleft + named->_namew + tright, _maxw); - _maxw = qMin(_maxw, int(st::msgMaxWidth)); + accumulate_max(maxWidth, tleft + named->_namew + tright); + accumulate_max(maxWidth, st::msgMaxWidth); } + auto minHeight = 0; if (thumbed) { - _minh = st::msgFileThumbPadding.top() + st::msgFileThumbSize + st::msgFileThumbPadding.bottom(); + minHeight = st::msgFileThumbPadding.top() + st::msgFileThumbSize + st::msgFileThumbPadding.bottom(); } else { - _minh = st::msgFilePadding.top() + st::msgFileSize + st::msgFilePadding.bottom(); + minHeight = st::msgFilePadding.top() + st::msgFileSize + st::msgFilePadding.bottom(); } if (!captioned && (_parent->Has() || _parent->displayEditedBadge())) { - _minh += st::msgDateFont->height - st::msgDateDelta.y(); + minHeight += st::msgDateFont->height - st::msgDateDelta.y(); } if (!isBubbleTop()) { - _minh -= st::msgFileTopMinus; + minHeight -= st::msgFileTopMinus; } if (captioned) { - auto captionw = _maxw + auto captionw = maxWidth - st::msgPadding.left() - st::msgPadding.right(); - _minh += captioned->_caption.countHeight(captionw); + minHeight += captioned->_caption.countHeight(captionw); if (isBubbleBottom()) { - _minh += st::msgPadding.bottom(); + minHeight += st::msgPadding.bottom(); } - } else { - _height = _minh; } + return { maxWidth, minHeight }; } -int HistoryDocument::resizeGetHeight(int width) { +QSize HistoryDocument::countCurrentSize(int newWidth) { auto captioned = Get(); if (!captioned) { - return HistoryFileMedia::resizeGetHeight(width); + return HistoryFileMedia::countCurrentSize(newWidth); } - _width = qMin(width, _maxw); + accumulate_min(newWidth, maxWidth()); + auto newHeight = 0; if (Get()) { - _height = st::msgFileThumbPadding.top() + st::msgFileThumbSize + st::msgFileThumbPadding.bottom(); + newHeight = st::msgFileThumbPadding.top() + st::msgFileThumbSize + st::msgFileThumbPadding.bottom(); } else { - _height = st::msgFilePadding.top() + st::msgFileSize + st::msgFilePadding.bottom(); + newHeight = st::msgFilePadding.top() + st::msgFileSize + st::msgFilePadding.bottom(); } if (!isBubbleTop()) { - _height -= st::msgFileTopMinus; + newHeight -= st::msgFileTopMinus; } - auto captionw = _width - st::msgPadding.left() - st::msgPadding.right(); - _height += captioned->_caption.countHeight(captionw); + auto captionw = newWidth - st::msgPadding.left() - st::msgPadding.right(); + newHeight += captioned->_caption.countHeight(captionw); if (isBubbleBottom()) { - _height += st::msgPadding.bottom(); + newHeight += st::msgPadding.bottom(); } - return _height; + return { newWidth, newHeight }; } void HistoryDocument::draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const { - if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return; + if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) return; _data->automaticLoad(_parent); bool loaded = _data->loaded(), displayLoading = _data->displayLoading(); bool selected = (selection == FullSelection); - int captionw = _width - st::msgPadding.left() - st::msgPadding.right(); + int captionw = width() - st::msgPadding.left() - st::msgPadding.right(); auto outbg = _parent->hasOutLayout(); if (displayLoading) { @@ -1544,7 +1581,7 @@ void HistoryDocument::draw(Painter &p, const QRect &r, TextSelection selection, auto inWebPage = (_parent->getMedia() != this); auto roundRadius = inWebPage ? ImageRoundRadius::Small : ImageRoundRadius::Large; - QRect rthumb(rtlrect(st::msgFileThumbPadding.left(), st::msgFileThumbPadding.top() - topMinus, st::msgFileThumbSize, st::msgFileThumbSize, _width)); + QRect rthumb(rtlrect(st::msgFileThumbPadding.left(), st::msgFileThumbPadding.top() - topMinus, st::msgFileThumbSize, st::msgFileThumbSize, width())); QPixmap thumb; if (loaded) { thumb = _data->thumb->pixSingle(thumbed->_thumbw, 0, st::msgFileThumbSize, st::msgFileThumbSize, roundRadius); @@ -1601,7 +1638,7 @@ void HistoryDocument::draw(Painter &p, const QRect &r, TextSelection selection, bool over = ClickHandler::showAsActive(lnk); p.setFont(over ? st::semiboldFont->underline() : st::semiboldFont); p.setPen(outbg ? (selected ? st::msgFileThumbLinkOutFgSelected : st::msgFileThumbLinkOutFg) : (selected ? st::msgFileThumbLinkInFgSelected : st::msgFileThumbLinkInFg)); - p.drawTextLeft(nameleft, linktop, _width, thumbed->_link, thumbed->_linkw); + p.drawTextLeft(nameleft, linktop, width(), thumbed->_link, thumbed->_linkw); } } else { nameleft = st::msgFilePadding.left() + st::msgFileSize + st::msgFilePadding.right(); @@ -1610,7 +1647,7 @@ void HistoryDocument::draw(Painter &p, const QRect &r, TextSelection selection, statustop = st::msgFileStatusTop - topMinus; bottom = st::msgFilePadding.top() + st::msgFileSize + st::msgFilePadding.bottom() - topMinus; - QRect inner(rtlrect(st::msgFilePadding.left(), st::msgFilePadding.top() - topMinus, st::msgFileSize, st::msgFileSize, _width)); + QRect inner(rtlrect(st::msgFilePadding.left(), st::msgFilePadding.top() - topMinus, st::msgFileSize, st::msgFileSize, width())); p.setPen(Qt::NoPen); if (selected) { p.setBrush(outbg ? st::msgFileOutBgSelected : st::msgFileInBgSelected); @@ -1650,7 +1687,7 @@ void HistoryDocument::draw(Painter &p, const QRect &r, TextSelection selection, })(); icon->paintInCenter(p, inner); } - auto namewidth = _width - nameleft - nameright; + auto namewidth = width() - nameleft - nameright; auto statuswidth = namewidth; auto voiceStatusOverride = QString(); @@ -1730,9 +1767,9 @@ void HistoryDocument::draw(Painter &p, const QRect &r, TextSelection selection, p.setFont(st::semiboldFont); p.setPen(outbg ? (selected ? st::historyFileNameOutFgSelected : st::historyFileNameOutFg) : (selected ? st::historyFileNameInFgSelected : st::historyFileNameInFg)); if (namewidth < named->_namew) { - p.drawTextLeft(nameleft, nametop, _width, st::semiboldFont->elided(named->_name, namewidth, Qt::ElideMiddle)); + p.drawTextLeft(nameleft, nametop, width(), st::semiboldFont->elided(named->_name, namewidth, Qt::ElideMiddle)); } else { - p.drawTextLeft(nameleft, nametop, _width, named->_name, named->_namew); + p.drawTextLeft(nameleft, nametop, width(), named->_name, named->_namew); } } @@ -1740,7 +1777,7 @@ void HistoryDocument::draw(Painter &p, const QRect &r, TextSelection selection, auto status = outbg ? (selected ? st::mediaOutFgSelected : st::mediaOutFg) : (selected ? st::mediaInFgSelected : st::mediaInFg); p.setFont(st::normalFont); p.setPen(status); - p.drawTextLeft(nameleft, statustop, _width, statusText); + p.drawTextLeft(nameleft, statustop, width(), statusText); if (_parent->isMediaUnread()) { auto w = st::normalFont->width(statusText); @@ -1750,7 +1787,7 @@ void HistoryDocument::draw(Painter &p, const QRect &r, TextSelection selection, { PainterHighQualityEnabler hq(p); - p.drawEllipse(rtlrect(nameleft + w + st::mediaUnreadSkip, statustop + st::mediaUnreadTop, st::mediaUnreadSize, st::mediaUnreadSize, _width)); + p.drawEllipse(rtlrect(nameleft + w + st::mediaUnreadSkip, statustop + st::mediaUnreadTop, st::mediaUnreadSize, st::mediaUnreadSize, width())); } } } @@ -1764,7 +1801,7 @@ void HistoryDocument::draw(Painter &p, const QRect &r, TextSelection selection, HistoryTextState HistoryDocument::getState(QPoint point, HistoryStateRequest request) const { auto result = HistoryTextState(_parent); - if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) { + if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) { return result; } @@ -1772,7 +1809,7 @@ HistoryTextState HistoryDocument::getState(QPoint point, HistoryStateRequest req bool showPause = updateStatusText(); - int32 nameleft = 0, nametop = 0, nameright = 0, statustop = 0, linktop = 0, bottom = 0; + auto nameleft = 0, nametop = 0, nameright = 0, statustop = 0, linktop = 0, bottom = 0; auto topMinus = isBubbleTop() ? 0 : st::msgFileTopMinus; if (auto thumbed = Get()) { nameleft = st::msgFileThumbPadding.left() + st::msgFileThumbSize + st::msgFileThumbPadding.right(); @@ -1781,7 +1818,7 @@ HistoryTextState HistoryDocument::getState(QPoint point, HistoryStateRequest req linktop = st::msgFileThumbLinkTop - topMinus; bottom = st::msgFileThumbPadding.top() + st::msgFileThumbSize + st::msgFileThumbPadding.bottom() - topMinus; - QRect rthumb(rtlrect(st::msgFileThumbPadding.left(), st::msgFileThumbPadding.top() - topMinus, st::msgFileThumbSize, st::msgFileThumbSize, _width)); + QRect rthumb(rtlrect(st::msgFileThumbPadding.left(), st::msgFileThumbPadding.top() - topMinus, st::msgFileThumbSize, st::msgFileThumbSize, width())); if ((_data->loading() || _data->uploading() || !loaded) && rthumb.contains(point)) { result.link = (_data->loading() || _data->uploading()) ? _cancell : _savel; @@ -1789,7 +1826,7 @@ HistoryTextState HistoryDocument::getState(QPoint point, HistoryStateRequest req } if (_data->status != FileUploadFailed) { - if (rtlrect(nameleft, linktop, thumbed->_linkw, st::semiboldFont->height, _width).contains(point)) { + if (rtlrect(nameleft, linktop, thumbed->_linkw, st::semiboldFont->height, width()).contains(point)) { result.link = (_data->loading() || _data->uploading()) ? thumbed->_linkcancell : thumbed->_linksavel; @@ -1802,7 +1839,7 @@ HistoryTextState HistoryDocument::getState(QPoint point, HistoryStateRequest req nametop = st::msgFileNameTop - topMinus; bottom = st::msgFilePadding.top() + st::msgFileSize + st::msgFilePadding.bottom() - topMinus; - QRect inner(rtlrect(st::msgFilePadding.left(), st::msgFilePadding.top() - topMinus, st::msgFileSize, st::msgFileSize, _width)); + QRect inner(rtlrect(st::msgFilePadding.left(), st::msgFilePadding.top() - topMinus, st::msgFileSize, st::msgFileSize, width())); if ((_data->loading() || _data->uploading() || !loaded) && inner.contains(point)) { result.link = (_data->loading() || _data->uploading()) ? _cancell : _savel; return result; @@ -1810,7 +1847,7 @@ HistoryTextState HistoryDocument::getState(QPoint point, HistoryStateRequest req } if (auto voice = Get()) { - auto namewidth = _width - nameleft - nameright; + auto namewidth = width() - nameleft - nameright; auto waveformbottom = st::msgFilePadding.top() - topMinus + st::msgWaveformMax + st::msgWaveformMin; if (QRect(nameleft, nametop, namewidth, waveformbottom - nametop).contains(point)) { auto state = Media::Player::mixer()->currentState(AudioMsgId::Type::Voice); @@ -1824,22 +1861,22 @@ HistoryTextState HistoryDocument::getState(QPoint point, HistoryStateRequest req } } - auto height = _height; + auto painth = height(); if (auto captioned = Get()) { if (point.y() >= bottom) { result = HistoryTextState(_parent, captioned->_caption.getState( point - QPoint(st::msgPadding.left(), bottom), - _width - st::msgPadding.left() - st::msgPadding.right(), + width() - st::msgPadding.left() - st::msgPadding.right(), request.forText())); return result; } - auto captionw = _width - st::msgPadding.left() - st::msgPadding.right(); - height -= captioned->_caption.countHeight(captionw); + auto captionw = width() - st::msgPadding.left() - st::msgPadding.right(); + painth -= captioned->_caption.countHeight(captionw); if (isBubbleBottom()) { - height -= st::msgPadding.bottom(); + painth -= st::msgPadding.bottom(); } } - if (QRect(0, 0, _width, height).contains(point) && !_data->loading() && !_data->uploading() && _data->isValid()) { + if (QRect(0, 0, width(), painth).contains(point) && !_data->loading() && !_data->uploading() && _data->isValid()) { result.link = _openl; return result; } @@ -1857,7 +1894,7 @@ void HistoryDocument::updatePressed(QPoint point) { nameleft = st::msgFilePadding.left() + st::msgFileSize + st::msgFilePadding.right(); nameright = st::msgFilePadding.left(); } - voice->setSeekingCurrent(snap((point.x() - nameleft) / float64(_width - nameleft - nameright), 0., 1.)); + voice->setSeekingCurrent(snap((point.x() - nameleft) / float64(width() - nameleft - nameright), 0., 1.)); Auth().data().requestItemRepaint(_parent); } } @@ -1950,8 +1987,8 @@ void HistoryDocument::buildStringRepresentation(Callback callback) const { return callback(attachType, attachFileName, *caption); } -void HistoryDocument::setStatusSize(int32 newSize, qint64 realDuration) const { - int32 duration = _data->isSong() +void HistoryDocument::setStatusSize(int newSize, qint64 realDuration) const { + auto duration = _data->isSong() ? _data->song()->duration : (_data->isVoiceMessage() ? _data->voice()->duration @@ -1974,8 +2011,9 @@ void HistoryDocument::setStatusSize(int32 newSize, qint64 realDuration) const { } bool HistoryDocument::updateStatusText() const { - bool showPause = false; - int32 statusSize = 0, realDuration = 0; + auto showPause = false; + auto statusSize = 0; + auto realDuration = 0; if (_data->status == FileDownloadFailed || _data->status == FileUploadFailed) { statusSize = FileStatusSizeFailed; } else if (_data->uploading()) { @@ -2182,7 +2220,7 @@ HistoryGif::HistoryGif( setStatusSize(other._statusSize); } -void HistoryGif::initDimensions() { +QSize HistoryGif::countOptimalSize() { if (_caption.hasSkipBlock()) { _caption.setSkipBlock(_parent->skipBlockWidth(), _parent->skipBlockHeight()); } @@ -2192,7 +2230,8 @@ void HistoryGif::initDimensions() { _parent->fullId()); } - int32 tw = 0, th = 0; + auto tw = 0; + auto th = 0; if (_gif && _gif->state() == Media::Clip::State::Error) { if (!_gif->autoplay()) { Ui::show(Box(lang(lng_gif_error))); @@ -2223,18 +2262,18 @@ void HistoryGif::initDimensions() { } _thumbw = tw; _thumbh = th; - _maxw = qMax(tw, int32(st::minPhotoSize)); - _minh = qMax(th, int32(st::minPhotoSize)); - _maxw = qMax(_maxw, _parent->infoWidth() + 2 * int32(st::msgDateImgDelta + st::msgDateImgPadding.x())); + auto maxWidth = qMax(tw, st::minPhotoSize); + auto minHeight = qMax(th, st::minPhotoSize); + accumulate_max(maxWidth, _parent->infoWidth() + 2 * (st::msgDateImgDelta + st::msgDateImgPadding.x())); if (!_gif || !_gif->ready()) { - _maxw = qMax(_maxw, gifMaxStatusWidth(_data) + 2 * int32(st::msgDateImgDelta + st::msgDateImgPadding.x())); + accumulate_max(maxWidth, gifMaxStatusWidth(_data) + 2 * (st::msgDateImgDelta + st::msgDateImgPadding.x())); } if (_parent->hasBubble()) { if (!_caption.isEmpty()) { - auto captionw = _maxw - st::msgPadding.left() - st::msgPadding.right(); - _minh += st::mediaCaptionSkip + _caption.countHeight(captionw); + auto captionw = maxWidth - st::msgPadding.left() - st::msgPadding.right(); + minHeight += st::mediaCaptionSkip + _caption.countHeight(captionw); if (isBubbleBottom()) { - _minh += st::msgPadding.bottom(); + minHeight += st::msgPadding.bottom(); } } } else if (isSeparateRoundVideo()) { @@ -2244,11 +2283,14 @@ void HistoryGif::initDimensions() { if (forwarded) { forwarded->create(via); } - _maxw += additionalWidth(via, reply, forwarded); + maxWidth += additionalWidth(via, reply, forwarded); } + return { maxWidth, minHeight }; } -int HistoryGif::resizeGetHeight(int width) { +QSize HistoryGif::countCurrentSize(int newWidth) { + auto availableWidth = newWidth; + int tw = 0, th = 0; if (_gif && _gif->ready()) { tw = convertScale(_gif->width()); @@ -2272,16 +2314,16 @@ int HistoryGif::resizeGetHeight(int width) { tw = th = 1; } - if (width < tw) { - th = qRound((width / float64(tw)) * th); - tw = width; + if (newWidth < tw) { + th = qRound((newWidth / float64(tw)) * th); + tw = newWidth; } _thumbw = tw; _thumbh = th; - _width = qMax(tw, int32(st::minPhotoSize)); - _height = qMax(th, int32(st::minPhotoSize)); - _width = qMax(_width, _parent->infoWidth() + 2 * int32(st::msgDateImgDelta + st::msgDateImgPadding.x())); + newWidth = qMax(tw, st::minPhotoSize); + auto newHeight = qMax(th, st::minPhotoSize); + accumulate_max(newWidth, _parent->infoWidth() + 2 * st::msgDateImgDelta + st::msgDateImgPadding.x()); if (_gif && _gif->ready()) { if (!_gif->started()) { auto isRound = _data->isVideoMessage(); @@ -2289,17 +2331,17 @@ int HistoryGif::resizeGetHeight(int width) { auto roundRadius = isRound ? ImageRoundRadius::Ellipse : inWebPage ? ImageRoundRadius::Small : ImageRoundRadius::Large; auto roundCorners = (isRound || inWebPage) ? RectPart::AllCorners : ((isBubbleTop() ? (RectPart::TopLeft | RectPart::TopRight) : RectPart::None) | ((isBubbleBottom() && _caption.isEmpty()) ? (RectPart::BottomLeft | RectPart::BottomRight) : RectPart::None)); - _gif->start(_thumbw, _thumbh, _width, _height, roundRadius, roundCorners); + _gif->start(_thumbw, _thumbh, newWidth, newHeight, roundRadius, roundCorners); } } else { - _width = qMax(_width, gifMaxStatusWidth(_data) + 2 * int32(st::msgDateImgDelta + st::msgDateImgPadding.x())); + accumulate_max(newWidth, gifMaxStatusWidth(_data) + 2 * (st::msgDateImgDelta + st::msgDateImgPadding.x())); } if (_parent->hasBubble()) { if (!_caption.isEmpty()) { - auto captionw = _width - st::msgPadding.left() - st::msgPadding.right(); - _height += st::mediaCaptionSkip + _caption.countHeight(captionw); + auto captionw = newWidth - st::msgPadding.left() - st::msgPadding.right(); + newHeight += st::mediaCaptionSkip + _caption.countHeight(captionw); if (isBubbleBottom()) { - _height += st::msgPadding.bottom(); + newHeight += st::msgPadding.bottom(); } } } else if (isSeparateRoundVideo()) { @@ -2308,10 +2350,10 @@ int HistoryGif::resizeGetHeight(int width) { auto forwarded = _parent->Get(); if (via || reply || forwarded) { auto additional = additionalWidth(via, reply, forwarded); - _width += additional; - accumulate_min(_width, width); - auto usew = _maxw - additional; - auto availw = _width - usew - st::msgReplyPadding.left() - st::msgReplyPadding.left() - st::msgReplyPadding.left(); + newWidth += additional; + accumulate_min(newWidth, availableWidth); + auto usew = maxWidth() - additional; + auto availw = newWidth - usew - st::msgReplyPadding.left() - st::msgReplyPadding.left() - st::msgReplyPadding.left(); if (!forwarded && via) { via->resize(availw); } @@ -2321,11 +2363,11 @@ int HistoryGif::resizeGetHeight(int width) { } } - return _height; + return { newWidth, newHeight }; } void HistoryGif::draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const { - if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return; + if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) return; _data->automaticLoad(_parent); auto loaded = _data->loaded(); @@ -2337,12 +2379,12 @@ void HistoryGif::draw(Painter &p, const QRect &r, TextSelection selection, TimeM Ui::autoplayMediaInlineAsync(_parent->fullId()); } - auto skipx = 0, skipy = 0, width = _width, height = _height; + auto paintx = 0, painty = 0, paintw = width(), painth = height(); bool bubble = _parent->hasBubble(); auto outbg = _parent->hasOutLayout(); auto isChildMedia = (_parent->getMedia() != this); - auto captionw = width - st::msgPadding.left() - st::msgPadding.right(); + auto captionw = paintw - st::msgPadding.left() - st::msgPadding.right(); auto isRound = _data->isVideoMessage(); auto displayMute = false; @@ -2363,29 +2405,29 @@ void HistoryGif::draw(Painter &p, const QRect &r, TextSelection selection, TimeM if (bubble) { if (!_caption.isEmpty()) { - height -= st::mediaCaptionSkip + _caption.countHeight(captionw); + painth -= st::mediaCaptionSkip + _caption.countHeight(captionw); if (isBubbleBottom()) { - height -= st::msgPadding.bottom(); + painth -= st::msgPadding.bottom(); } } } else if (!isRound) { - App::roundShadow(p, 0, 0, width, _height, selected ? st::msgInShadowSelected : st::msgInShadow, selected ? InSelectedShadowCorners : InShadowCorners); + App::roundShadow(p, 0, 0, paintw, height(), selected ? st::msgInShadowSelected : st::msgInShadow, selected ? InSelectedShadowCorners : InShadowCorners); } - auto usex = 0, usew = width; + auto usex = 0, usew = paintw; auto separateRoundVideo = isSeparateRoundVideo(); auto via = separateRoundVideo ? _parent->Get() : nullptr; auto reply = separateRoundVideo ? _parent->Get() : nullptr; auto forwarded = separateRoundVideo ? _parent->Get() : nullptr; if (via || reply || forwarded) { - usew = _maxw - additionalWidth(via, reply, forwarded); + usew = maxWidth() - additionalWidth(via, reply, forwarded); if (outbg) { - usex = _width - usew; + usex = width() - usew; } } - if (rtl()) usex = _width - usex - usew; + if (rtl()) usex = width() - usex - usew; - QRect rthumb(rtlrect(usex + skipx, skipy, usew, height, _width)); + QRect rthumb(rtlrect(usex + paintx, painty, usew, painth, width())); auto roundRadius = isRound ? ImageRoundRadius::Ellipse : isChildMedia ? ImageRoundRadius::Small : ImageRoundRadius::Large; auto roundCorners = (isRound || isChildMedia) ? RectPart::AllCorners : ((isBubbleTop() ? (RectPart::TopLeft | RectPart::TopRight) : RectPart::None) @@ -2399,7 +2441,7 @@ void HistoryGif::draw(Painter &p, const QRect &r, TextSelection selection, TimeM displayMute = true; } } - p.drawPixmap(rthumb.topLeft(), _gif->current(_thumbw, _thumbh, usew, height, roundRadius, roundCorners, paused ? 0 : ms)); + p.drawPixmap(rthumb.topLeft(), _gif->current(_thumbw, _thumbh, usew, painth, roundRadius, roundCorners, paused ? 0 : ms)); if (displayMute) { _roundPlayback.reset(); @@ -2426,7 +2468,7 @@ void HistoryGif::draw(Painter &p, const QRect &r, TextSelection selection, TimeM } } } else { - p.drawPixmap(rthumb.topLeft(), _data->thumb->pixBlurredSingle(_thumbw, _thumbh, usew, height, roundRadius, roundCorners)); + p.drawPixmap(rthumb.topLeft(), _data->thumb->pixBlurredSingle(_thumbw, _thumbh, usew, painth, roundRadius, roundCorners)); } if (selected) { @@ -2475,18 +2517,18 @@ void HistoryGif::draw(Painter &p, const QRect &r, TextSelection selection, TimeM } if (!isRound && (!animating || _parent->id < 0)) { - auto statusX = skipx + st::msgDateImgDelta + st::msgDateImgPadding.x(); - auto statusY = skipy + st::msgDateImgDelta + st::msgDateImgPadding.y(); + auto statusX = paintx + st::msgDateImgDelta + st::msgDateImgPadding.x(); + auto statusY = painty + st::msgDateImgDelta + st::msgDateImgPadding.y(); auto statusW = st::normalFont->width(_statusText) + 2 * st::msgDateImgPadding.x(); auto statusH = st::normalFont->height + 2 * st::msgDateImgPadding.y(); - App::roundRect(p, rtlrect(statusX - st::msgDateImgPadding.x(), statusY - st::msgDateImgPadding.y(), statusW, statusH, _width), selected ? st::msgDateImgBgSelected : st::msgDateImgBg, selected ? DateSelectedCorners : DateCorners); + App::roundRect(p, rtlrect(statusX - st::msgDateImgPadding.x(), statusY - st::msgDateImgPadding.y(), statusW, statusH, width()), selected ? st::msgDateImgBgSelected : st::msgDateImgBg, selected ? DateSelectedCorners : DateCorners); p.setFont(st::normalFont); p.setPen(st::msgDateImgFg); - p.drawTextLeft(statusX, statusY, _width, _statusText, statusW - 2 * st::msgDateImgPadding.x()); + p.drawTextLeft(statusX, statusY, width(), _statusText, statusW - 2 * st::msgDateImgPadding.x()); } } if (displayMute) { - auto muteRect = rtlrect(rthumb.x() + (rthumb.width() - st::historyVideoMessageMuteSize) / 2, rthumb.y() + st::msgDateImgDelta, st::historyVideoMessageMuteSize, st::historyVideoMessageMuteSize, _width); + auto muteRect = rtlrect(rthumb.x() + (rthumb.width() - st::historyVideoMessageMuteSize) / 2, rthumb.y() + st::msgDateImgDelta, st::historyVideoMessageMuteSize, st::historyVideoMessageMuteSize, width()); p.setPen(Qt::NoPen); p.setBrush(selected ? st::msgDateImgBgSelected : st::msgDateImgBg); PainterHighQualityEnabler hq(p); @@ -2498,26 +2540,26 @@ void HistoryGif::draw(Painter &p, const QRect &r, TextSelection selection, TimeM auto mediaUnread = _parent->isMediaUnread(); auto statusW = st::normalFont->width(_statusText) + 2 * st::msgDateImgPadding.x(); auto statusH = st::normalFont->height + 2 * st::msgDateImgPadding.y(); - auto statusX = usex + skipx + st::msgDateImgDelta + st::msgDateImgPadding.x(); - auto statusY = skipy + height - st::msgDateImgDelta - statusH + st::msgDateImgPadding.y(); + auto statusX = usex + paintx + st::msgDateImgDelta + st::msgDateImgPadding.x(); + auto statusY = painty + painth - st::msgDateImgDelta - statusH + st::msgDateImgPadding.y(); if (_parent->isMediaUnread()) { statusW += st::mediaUnreadSkip + st::mediaUnreadSize; } - App::roundRect(p, rtlrect(statusX - st::msgDateImgPadding.x(), statusY - st::msgDateImgPadding.y(), statusW, statusH, _width), selected ? st::msgServiceBgSelected : st::msgServiceBg, selected ? StickerSelectedCorners : StickerCorners); + App::roundRect(p, rtlrect(statusX - st::msgDateImgPadding.x(), statusY - st::msgDateImgPadding.y(), statusW, statusH, width()), selected ? st::msgServiceBgSelected : st::msgServiceBg, selected ? StickerSelectedCorners : StickerCorners); p.setFont(st::normalFont); p.setPen(st::msgServiceFg); - p.drawTextLeft(statusX, statusY, _width, _statusText, statusW - 2 * st::msgDateImgPadding.x()); + p.drawTextLeft(statusX, statusY, width(), _statusText, statusW - 2 * st::msgDateImgPadding.x()); if (mediaUnread) { p.setPen(Qt::NoPen); p.setBrush(st::msgServiceFg); { PainterHighQualityEnabler hq(p); - p.drawEllipse(rtlrect(statusX - st::msgDateImgPadding.x() + statusW - st::msgDateImgPadding.x() - st::mediaUnreadSize, statusY + st::mediaUnreadTop, st::mediaUnreadSize, st::mediaUnreadSize, _width)); + p.drawEllipse(rtlrect(statusX - st::msgDateImgPadding.x() + statusW - st::msgDateImgPadding.x() - st::mediaUnreadSize, statusY + st::mediaUnreadTop, st::mediaUnreadSize, st::mediaUnreadSize, width())); } } if (via || reply || forwarded) { - auto rectw = _width - usew - st::msgReplyPadding.left(); + auto rectw = width() - usew - st::msgReplyPadding.left(); auto innerw = rectw - (st::msgReplyPadding.left() + st::msgReplyPadding.right()); auto recth = st::msgReplyPadding.top() + st::msgReplyPadding.bottom(); auto forwardedHeightReal = forwarded ? forwarded->text.countHeight(innerw) : 0; @@ -2531,8 +2573,8 @@ void HistoryGif::draw(Painter &p, const QRect &r, TextSelection selection, TimeM recth += st::msgReplyBarSize.height(); } int rectx = outbg ? 0 : (usew + st::msgReplyPadding.left()); - int recty = skipy; - if (rtl()) rectx = _width - rectx - rectw; + int recty = painty; + if (rtl()) rectx = width() - rectx - rectw; App::roundRect(p, rectx, recty, rectw, recth, selected ? st::msgServiceBgSelected : st::msgServiceBg, selected ? StickerSelectedCorners : StickerCorners); p.setPen(st::msgServiceFg); @@ -2560,10 +2602,11 @@ void HistoryGif::draw(Painter &p, const QRect &r, TextSelection selection, TimeM } if (!isRound && !_caption.isEmpty()) { p.setPen(outbg ? (selected ? st::historyTextOutFgSelected : st::historyTextOutFg) : (selected ? st::historyTextInFgSelected : st::historyTextInFg)); - _caption.draw(p, st::msgPadding.left(), skipy + height + st::mediaCaptionSkip, captionw, style::al_left, 0, -1, selection); + _caption.draw(p, st::msgPadding.left(), painty + painth + st::mediaCaptionSkip, captionw, style::al_left, 0, -1, selection); } else if (!isChildMedia) { - auto fullRight = skipx + usex + usew; - auto fullBottom = skipy + height; + auto fullRight = paintx + usex + usew; + auto fullBottom = painty + painth; + // #TODO view media auto maxRight = _parent->history()->width - st::msgMargin.left(); if (_parent->history()->canHaveFromPhotos()) { maxRight -= st::msgMargin.right(); @@ -2581,7 +2624,7 @@ void HistoryGif::draw(Painter &p, const QRect &r, TextSelection selection, TimeM } } if (isRound || needInfoDisplay()) { - _parent->drawInfo(p, fullRight, fullBottom, 2 * skipx + width, selected, isRound ? InfoDisplayOverBackground : InfoDisplayOverImage); + _parent->drawInfo(p, fullRight, fullBottom, 2 * paintx + paintw, selected, isRound ? InfoDisplayOverBackground : InfoDisplayOverImage); } if (!bubble && _parent->displayRightAction()) { auto fastShareLeft = (fullRight + st::historyFastShareLeft); @@ -2590,7 +2633,7 @@ void HistoryGif::draw(Painter &p, const QRect &r, TextSelection selection, TimeM fastShareLeft = (fullRight - st::historyFastShareSize - st::msgDateImgDelta); fastShareTop -= (st::msgDateImgDelta + st::msgDateImgPadding.y() + st::msgDateFont->height + st::msgDateImgPadding.y()); } - _parent->drawRightAction(p, fastShareLeft, fastShareTop, 2 * skipx + width); + _parent->drawRightAction(p, fastShareLeft, fastShareTop, 2 * paintx + paintw); } } } @@ -2598,45 +2641,45 @@ void HistoryGif::draw(Painter &p, const QRect &r, TextSelection selection, TimeM HistoryTextState HistoryGif::getState(QPoint point, HistoryStateRequest request) const { auto result = HistoryTextState(_parent); - if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) { + if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) { return result; } - int32 skipx = 0, skipy = 0, width = _width, height = _height; - bool bubble = _parent->hasBubble(); + auto paintx = 0, painty = 0, paintw = width(), painth = height(); + auto bubble = _parent->hasBubble(); if (bubble && !_caption.isEmpty()) { - auto captionw = width - st::msgPadding.left() - st::msgPadding.right(); - height -= _caption.countHeight(captionw); + auto captionw = paintw - st::msgPadding.left() - st::msgPadding.right(); + painth -= _caption.countHeight(captionw); if (isBubbleBottom()) { - height -= st::msgPadding.bottom(); + painth -= st::msgPadding.bottom(); } - if (QRect(st::msgPadding.left(), height, captionw, _height - height).contains(point)) { + if (QRect(st::msgPadding.left(), painth, captionw, height() - painth).contains(point)) { result = HistoryTextState(_parent, _caption.getState( - point - QPoint(st::msgPadding.left(), height), + point - QPoint(st::msgPadding.left(), painth), captionw, request.forText())); return result; } - height -= st::mediaCaptionSkip; + painth -= st::mediaCaptionSkip; } auto outbg = _parent->hasOutLayout(); auto isChildMedia = (_parent->getMedia() != this); auto isRound = _data->isVideoMessage(); - auto usew = width, usex = 0; + auto usew = paintw, usex = 0; auto separateRoundVideo = isSeparateRoundVideo(); auto via = separateRoundVideo ? _parent->Get() : nullptr; auto reply = separateRoundVideo ? _parent->Get() : nullptr; auto forwarded = separateRoundVideo ? _parent->Get() : nullptr; if (via || reply || forwarded) { - usew = _maxw - additionalWidth(via, reply, forwarded); + usew = maxWidth() - additionalWidth(via, reply, forwarded); if (outbg) { - usex = _width - usew; + usex = width() - usew; } } - if (rtl()) usex = _width - usex - usew; + if (rtl()) usex = width() - usex - usew; if (via || reply || forwarded) { - auto rectw = width - usew - st::msgReplyPadding.left(); + auto rectw = paintw - usew - st::msgReplyPadding.left(); auto innerw = rectw - (st::msgReplyPadding.left() + st::msgReplyPadding.right()); auto recth = st::msgReplyPadding.top() + st::msgReplyPadding.bottom(); auto forwardedHeightReal = forwarded ? forwarded->text.countHeight(innerw) : 0; @@ -2650,8 +2693,8 @@ HistoryTextState HistoryGif::getState(QPoint point, HistoryStateRequest request) recth += st::msgReplyBarSize.height(); } auto rectx = outbg ? 0 : (usew + st::msgReplyPadding.left()); - auto recty = skipy; - if (rtl()) rectx = _width - rectx - rectw; + auto recty = painty; + if (rtl()) rectx = width() - rectx - rectw; if (forwarded) { if (QRect(rectx, recty, rectw, st::msgReplyPadding.top() + forwardedHeight).contains(point)) { @@ -2692,7 +2735,7 @@ HistoryTextState HistoryGif::getState(QPoint point, HistoryStateRequest request) } } } - if (QRect(usex + skipx, skipy, usew, height).contains(point)) { + if (QRect(usex + paintx, painty, usew, painth).contains(point)) { if (_data->uploading()) { result.link = _cancell; } else if (!_gif || !cAutoPlayGif() || _data->isVideoMessage()) { @@ -2702,8 +2745,9 @@ HistoryTextState HistoryGif::getState(QPoint point, HistoryStateRequest request) } } if (isRound || _caption.isEmpty()) { - auto fullRight = usex + skipx + usew; - auto fullBottom = skipy + height; + auto fullRight = usex + paintx + usew; + auto fullBottom = painty + painth; + // #TODO view media auto maxRight = _parent->history()->width - st::msgMargin.left(); if (_parent->history()->canHaveFromPhotos()) { maxRight -= st::msgMargin.right(); @@ -2763,7 +2807,8 @@ bool HistoryGif::needsBubble() const { return message->viaBot() || message->Has() || message->displayForwardedFrom() - || message->displayFromName(); +// || message->displayFromName() // #TODO media views + ; } return false; } @@ -2799,7 +2844,7 @@ bool HistoryGif::isSeparateRoundVideo() const { && !_parent->hasBubble(); } -void HistoryGif::setStatusSize(int32 newSize) const { +void HistoryGif::setStatusSize(int newSize) const { if (_data->isVideoMessage()) { _statusSize = newSize; if (newSize < 0) { @@ -2813,8 +2858,9 @@ void HistoryGif::setStatusSize(int32 newSize) const { } void HistoryGif::updateStatusText() const { - bool showPause = false; - int32 statusSize = 0, realDuration = 0; + auto showPause = false; + auto statusSize = 0; + auto realDuration = 0; if (_data->status == FileDownloadFailed || _data->status == FileUploadFailed) { statusSize = FileStatusSizeFailed; } else if (_data->uploading()) { @@ -2953,7 +2999,7 @@ void HistoryGif::stopInline() { } clearClipReader(); - _parent->setPendingInitDimensions(); + Auth().data().requestItemViewResize(_parent); Auth().data().markItemLayoutChanged(_parent); } @@ -3003,7 +3049,7 @@ HistorySticker::HistorySticker( } } -void HistorySticker::initDimensions() { +QSize HistorySticker::countOptimalSize() { auto sticker = _data->sticker(); if (!_packLink && sticker && sticker->set.type() != mtpc_inputStickerSetEmpty) { @@ -3027,23 +3073,22 @@ void HistorySticker::initDimensions() { } if (_pixw < 1) _pixw = 1; if (_pixh < 1) _pixh = 1; - _maxw = qMax(_pixw, int16(st::minPhotoSize)); - _minh = qMax(_pixh, int16(st::minPhotoSize)); + auto maxWidth = qMax(_pixw, st::minPhotoSize); + auto minHeight = qMax(_pixh, st::minPhotoSize); if (_parent->getMedia() == this) { - _maxw += additionalWidth(); + maxWidth += additionalWidth(); } - - _height = _minh; + return { maxWidth, minHeight }; } -int HistorySticker::resizeGetHeight(int width) { // return new height - _width = qMin(width, _maxw); +QSize HistorySticker::countCurrentSize(int newWidth) { + accumulate_min(newWidth, maxWidth()); if (_parent->getMedia() == this) { auto via = _parent->Get(); auto reply = _parent->Get(); if (via || reply) { - int usew = _maxw - additionalWidth(via, reply); - int availw = _width - usew - st::msgReplyPadding.left() - st::msgReplyPadding.left() - st::msgReplyPadding.left(); + int usew = maxWidth() - additionalWidth(via, reply); + int availw = newWidth - usew - st::msgReplyPadding.left() - st::msgReplyPadding.left() - st::msgReplyPadding.left(); if (via) { via->resize(availw); } @@ -3052,14 +3097,14 @@ int HistorySticker::resizeGetHeight(int width) { // return new height } } } - return _height; + return { newWidth, minHeight() }; } void HistorySticker::draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const { auto sticker = _data->sticker(); if (!sticker) return; - if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return; + if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) return; _data->checkSticker(); bool loaded = _data->loaded(); @@ -3068,37 +3113,37 @@ void HistorySticker::draw(Painter &p, const QRect &r, TextSelection selection, T auto outbg = _parent->hasOutLayout(); auto childmedia = (_parent->getMedia() != this); - int usew = _maxw, usex = 0; + int usew = maxWidth(), usex = 0; auto via = childmedia ? nullptr : _parent->Get(); auto reply = childmedia ? nullptr : _parent->Get(); if (via || reply) { usew -= additionalWidth(via, reply); if (outbg) { - usex = _width - usew; + usex = width() - usew; } } - if (rtl()) usex = _width - usex - usew; + if (rtl()) usex = width() - usex - usew; if (selected) { if (sticker->img->isNull()) { - p.drawPixmap(QPoint(usex + (usew - _pixw) / 2, (_minh - _pixh) / 2), _data->thumb->pixBlurredColored(st::msgStickerOverlay, _pixw, _pixh)); + p.drawPixmap(QPoint(usex + (usew - _pixw) / 2, (minHeight() - _pixh) / 2), _data->thumb->pixBlurredColored(st::msgStickerOverlay, _pixw, _pixh)); } else { - p.drawPixmap(QPoint(usex + (usew - _pixw) / 2, (_minh - _pixh) / 2), sticker->img->pixColored(st::msgStickerOverlay, _pixw, _pixh)); + p.drawPixmap(QPoint(usex + (usew - _pixw) / 2, (minHeight() - _pixh) / 2), sticker->img->pixColored(st::msgStickerOverlay, _pixw, _pixh)); } } else { if (sticker->img->isNull()) { - p.drawPixmap(QPoint(usex + (usew - _pixw) / 2, (_minh - _pixh) / 2), _data->thumb->pixBlurred(_pixw, _pixh)); + p.drawPixmap(QPoint(usex + (usew - _pixw) / 2, (minHeight() - _pixh) / 2), _data->thumb->pixBlurred(_pixw, _pixh)); } else { - p.drawPixmap(QPoint(usex + (usew - _pixw) / 2, (_minh - _pixh) / 2), sticker->img->pix(_pixw, _pixh)); + p.drawPixmap(QPoint(usex + (usew - _pixw) / 2, (minHeight() - _pixh) / 2), sticker->img->pix(_pixw, _pixh)); } } if (!childmedia) { auto fullRight = usex + usew; - auto fullBottom = _height; + auto fullBottom = height(); _parent->drawInfo(p, fullRight, fullBottom, usex * 2 + usew, selected, InfoDisplayOverBackground); if (via || reply) { - int rectw = _width - usew - st::msgReplyPadding.left(); + int rectw = width() - usew - st::msgReplyPadding.left(); int recth = st::msgReplyPadding.top() + st::msgReplyPadding.bottom(); if (via) { recth += st::msgServiceNameFont->height + (reply ? st::msgReplyPadding.top() : 0); @@ -3108,7 +3153,7 @@ void HistorySticker::draw(Painter &p, const QRect &r, TextSelection selection, T } int rectx = outbg ? 0 : (usew + st::msgReplyPadding.left()); int recty = st::msgDateImgDelta; - if (rtl()) rectx = _width - rectx - rectw; + if (rtl()) rectx = width() - rectx - rectw; App::roundRect(p, rectx, recty, rectw, recth, selected ? st::msgServiceBgSelected : st::msgServiceBg, selected ? StickerSelectedCorners : StickerCorners); p.setPen(st::msgServiceFg); @@ -3138,26 +3183,26 @@ void HistorySticker::draw(Painter &p, const QRect &r, TextSelection selection, T HistoryTextState HistorySticker::getState(QPoint point, HistoryStateRequest request) const { auto result = HistoryTextState(_parent); - if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) { + if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) { return result; } auto outbg = _parent->hasOutLayout(); auto childmedia = (_parent->getMedia() != this); - int usew = _maxw, usex = 0; + int usew = maxWidth(), usex = 0; auto via = childmedia ? nullptr : _parent->Get(); auto reply = childmedia ? nullptr : _parent->Get(); if (via || reply) { usew -= additionalWidth(via, reply); if (outbg) { - usex = _width - usew; + usex = width() - usew; } } - if (rtl()) usex = _width - usex - usew; + if (rtl()) usex = width() - usex - usew; if (via || reply) { - int rectw = _width - usew - st::msgReplyPadding.left(); + int rectw = width() - usew - st::msgReplyPadding.left(); int recth = st::msgReplyPadding.top() + st::msgReplyPadding.bottom(); if (via) { recth += st::msgServiceNameFont->height + (reply ? st::msgReplyPadding.top() : 0); @@ -3167,7 +3212,7 @@ HistoryTextState HistorySticker::getState(QPoint point, HistoryStateRequest requ } int rectx = outbg ? 0 : (usew + st::msgReplyPadding.left()); int recty = st::msgDateImgDelta; - if (rtl()) rectx = _width - rectx - rectw; + if (rtl()) rectx = width() - rectx - rectw; if (via) { int viah = st::msgReplyPadding.top() + st::msgServiceNameFont->height + (reply ? 0 : st::msgReplyPadding.bottom()); @@ -3188,7 +3233,7 @@ HistoryTextState HistorySticker::getState(QPoint point, HistoryStateRequest requ } if (_parent->getMedia() == this) { auto fullRight = usex + usew; - auto fullBottom = _height; + auto fullBottom = height(); if (_parent->pointInTime(fullRight, fullBottom, point, InfoDisplayOverImage)) { result.cursor = HistoryInDateCursorState; } @@ -3202,7 +3247,7 @@ HistoryTextState HistorySticker::getState(QPoint point, HistoryStateRequest requ } auto pixLeft = usex + (usew - _pixw) / 2; - auto pixTop = (_minh - _pixh) / 2; + auto pixTop = (minHeight() - _pixh) / 2; if (QRect(pixLeft, pixTop, _pixw, _pixh).contains(point)) { result.link = _packLink; return result; @@ -3312,8 +3357,8 @@ HistoryContact::HistoryContact(not_null parent, int32 userId, cons _phonew = st::normalFont->width(_phone); } -void HistoryContact::initDimensions() { - _maxw = st::msgFileMinWidth; +QSize HistoryContact::countOptimalSize() { + auto maxWidth = st::msgFileMinWidth; _contact = _userId ? App::userLoaded(_userId) : nullptr; if (_contact) { @@ -3333,46 +3378,45 @@ void HistoryContact::initDimensions() { } _linkw = _link.isEmpty() ? 0 : st::semiboldFont->width(_link); - int32 tleft = 0, tright = 0; + auto tleft = 0; + auto tright = 0; if (_userId) { tleft = st::msgFileThumbPadding.left() + st::msgFileThumbSize + st::msgFileThumbPadding.right(); tright = st::msgFileThumbPadding.left(); - _maxw = qMax(_maxw, tleft + _phonew + tright); + accumulate_max(maxWidth, tleft + _phonew + tright); } else { tleft = st::msgFilePadding.left() + st::msgFileSize + st::msgFilePadding.right(); tright = st::msgFileThumbPadding.left(); - _maxw = qMax(_maxw, tleft + _phonew + _parent->skipBlockWidth() + st::msgPadding.right()); + accumulate_max(maxWidth, tleft + _phonew + _parent->skipBlockWidth() + st::msgPadding.right()); } - _maxw = qMax(tleft + _name.maxWidth() + tright, _maxw); - _maxw = qMin(_maxw, int(st::msgMaxWidth)); - + accumulate_max(maxWidth, tleft + _name.maxWidth() + tright); + accumulate_max(maxWidth, st::msgMaxWidth); + auto minHeight = 0; if (_userId) { - _minh = st::msgFileThumbPadding.top() + st::msgFileThumbSize + st::msgFileThumbPadding.bottom(); + minHeight = st::msgFileThumbPadding.top() + st::msgFileThumbSize + st::msgFileThumbPadding.bottom(); if (_parent->Has()) { - _minh += st::msgDateFont->height - st::msgDateDelta.y(); + minHeight += st::msgDateFont->height - st::msgDateDelta.y(); } } else { - _minh = st::msgFilePadding.top() + st::msgFileSize + st::msgFilePadding.bottom(); + minHeight = st::msgFilePadding.top() + st::msgFileSize + st::msgFilePadding.bottom(); } if (!isBubbleTop()) { - _minh -= st::msgFileTopMinus; + minHeight -= st::msgFileTopMinus; } - _height = _minh; + return { maxWidth, minHeight }; } void HistoryContact::draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const { - if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return; - int32 skipx = 0, skipy = 0, width = _width, height = _height; + if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) return; + auto paintx = 0, painty = 0, paintw = width(), painth = height(); auto outbg = _parent->hasOutLayout(); bool selected = (selection == FullSelection); - if (width >= _maxw) { - width = _maxw; - } + accumulate_min(paintw, maxWidth()); - int32 nameleft = 0, nametop = 0, nameright = 0, statustop = 0, linktop = 0; + auto nameleft = 0, nametop = 0, nameright = 0, statustop = 0, linktop = 0; auto topMinus = isBubbleTop() ? 0 : st::msgFileTopMinus; if (_userId) { nameleft = st::msgFileThumbPadding.left() + st::msgFileThumbSize + st::msgFileThumbPadding.right(); @@ -3381,11 +3425,11 @@ void HistoryContact::draw(Painter &p, const QRect &r, TextSelection selection, T statustop = st::msgFileThumbStatusTop - topMinus; linktop = st::msgFileThumbLinkTop - topMinus; - QRect rthumb(rtlrect(st::msgFileThumbPadding.left(), st::msgFileThumbPadding.top() - topMinus, st::msgFileThumbSize, st::msgFileThumbSize, width)); + QRect rthumb(rtlrect(st::msgFileThumbPadding.left(), st::msgFileThumbPadding.top() - topMinus, st::msgFileThumbSize, st::msgFileThumbSize, paintw)); if (_contact) { _contact->paintUserpic(p, rthumb.x(), rthumb.y(), st::msgFileThumbSize); } else { - _photoEmpty->paint(p, st::msgFileThumbPadding.left(), st::msgFileThumbPadding.top() - topMinus, width, st::msgFileThumbSize); + _photoEmpty->paint(p, st::msgFileThumbPadding.left(), st::msgFileThumbPadding.top() - topMinus, paintw, st::msgFileThumbSize); } if (selected) { PainterHighQualityEnabler hq(p); @@ -3397,41 +3441,41 @@ void HistoryContact::draw(Painter &p, const QRect &r, TextSelection selection, T bool over = ClickHandler::showAsActive(_linkl); p.setFont(over ? st::semiboldFont->underline() : st::semiboldFont); p.setPen(outbg ? (selected ? st::msgFileThumbLinkOutFgSelected : st::msgFileThumbLinkOutFg) : (selected ? st::msgFileThumbLinkInFgSelected : st::msgFileThumbLinkInFg)); - p.drawTextLeft(nameleft, linktop, width, _link, _linkw); + p.drawTextLeft(nameleft, linktop, paintw, _link, _linkw); } else { nameleft = st::msgFilePadding.left() + st::msgFileSize + st::msgFilePadding.right(); nametop = st::msgFileNameTop - topMinus; nameright = st::msgFilePadding.left(); statustop = st::msgFileStatusTop - topMinus; - _photoEmpty->paint(p, st::msgFilePadding.left(), st::msgFilePadding.top() - topMinus, width, st::msgFileSize); + _photoEmpty->paint(p, st::msgFilePadding.left(), st::msgFilePadding.top() - topMinus, paintw, st::msgFileSize); } - int32 namewidth = width - nameleft - nameright; + auto namewidth = paintw - nameleft - nameright; p.setFont(st::semiboldFont); p.setPen(outbg ? (selected ? st::historyFileNameOutFgSelected : st::historyFileNameOutFg) : (selected ? st::historyFileNameInFgSelected : st::historyFileNameInFg)); - _name.drawLeftElided(p, nameleft, nametop, namewidth, width); + _name.drawLeftElided(p, nameleft, nametop, namewidth, paintw); auto &status = outbg ? (selected ? st::mediaOutFgSelected : st::mediaOutFg) : (selected ? st::mediaInFgSelected : st::mediaInFg); p.setFont(st::normalFont); p.setPen(status); - p.drawTextLeft(nameleft, statustop, width, _phone); + p.drawTextLeft(nameleft, statustop, paintw, _phone); } HistoryTextState HistoryContact::getState(QPoint point, HistoryStateRequest request) const { auto result = HistoryTextState(_parent); - int32 nameleft = 0, nametop = 0, nameright = 0, statustop = 0, linktop = 0; + auto nameleft = 0, nametop = 0, nameright = 0, statustop = 0, linktop = 0; auto topMinus = isBubbleTop() ? 0 : st::msgFileTopMinus; if (_userId) { nameleft = st::msgFileThumbPadding.left() + st::msgFileThumbSize + st::msgFileThumbPadding.right(); linktop = st::msgFileThumbLinkTop - topMinus; - if (rtlrect(nameleft, linktop, _linkw, st::semiboldFont->height, _width).contains(point)) { + if (rtlrect(nameleft, linktop, _linkw, st::semiboldFont->height, width()).contains(point)) { result.link = _linkl; return result; } } - if (QRect(0, 0, _width, _height).contains(point) && _contact) { + if (QRect(0, 0, width(), height()).contains(point) && _contact) { result.link = _contact->openLink(); return result; } @@ -3509,33 +3553,29 @@ HistoryCall::FinishReason HistoryCall::GetReason(const MTPDmessageActionPhoneCal return FinishReason::Hangup; } -void HistoryCall::initDimensions() { - _maxw = st::msgFileMinWidth; - +QSize HistoryCall::countOptimalSize() { _link = std::make_shared([peer = _parent->history()->peer] { if (auto user = peer->asUser()) { Calls::Current().startOutgoingCall(user); } }); - _maxw = st::historyCallWidth; - _minh = st::historyCallHeight; + auto maxWidth = st::historyCallWidth; + auto minHeight = st::historyCallHeight; if (!isBubbleTop()) { - _minh -= st::msgFileTopMinus; + minHeight -= st::msgFileTopMinus; } - _height = _minh; + return { maxWidth, minHeight }; } void HistoryCall::draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const { - if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return; - auto skipx = 0, skipy = 0, width = _width, height = _height; + if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) return; + auto paintx = 0, painty = 0, paintw = width(), painth = height(); auto outbg = _parent->hasOutLayout(); auto selected = (selection == FullSelection); - if (width >= _maxw) { - width = _maxw; - } + accumulate_min(paintw, maxWidth()); auto nameleft = 0, nametop = 0, nameright = 0, statustop = 0; auto topMinus = isBubbleTop() ? 0 : st::msgFileTopMinus; @@ -3545,30 +3585,30 @@ void HistoryCall::draw(Painter &p, const QRect &r, TextSelection selection, Time nameright = st::msgFilePadding.left(); statustop = st::historyCallStatusTop - topMinus; - auto namewidth = width - nameleft - nameright; + auto namewidth = paintw - nameleft - nameright; p.setFont(st::semiboldFont); p.setPen(outbg ? (selected ? st::historyFileNameOutFgSelected : st::historyFileNameOutFg) : (selected ? st::historyFileNameInFgSelected : st::historyFileNameInFg)); - p.drawTextLeft(nameleft, nametop, width, _text); + p.drawTextLeft(nameleft, nametop, paintw, _text); auto statusleft = nameleft; auto missed = (_reason == FinishReason::Missed || _reason == FinishReason::Busy); auto &arrow = outbg ? (selected ? st::historyCallArrowOutSelected : st::historyCallArrowOut) : missed ? (selected ? st::historyCallArrowMissedInSelected : st::historyCallArrowMissedIn) : (selected ? st::historyCallArrowInSelected : st::historyCallArrowIn); - arrow.paint(p, statusleft + st::historyCallArrowPosition.x(), statustop + st::historyCallArrowPosition.y(), width); + arrow.paint(p, statusleft + st::historyCallArrowPosition.x(), statustop + st::historyCallArrowPosition.y(), paintw); statusleft += arrow.width() + st::historyCallStatusSkip; auto &statusFg = outbg ? (selected ? st::mediaOutFgSelected : st::mediaOutFg) : (selected ? st::mediaInFgSelected : st::mediaInFg); p.setFont(st::normalFont); p.setPen(statusFg); - p.drawTextLeft(statusleft, statustop, width, _status); + p.drawTextLeft(statusleft, statustop, paintw, _status); auto &icon = outbg ? (selected ? st::historyCallOutIconSelected : st::historyCallOutIcon) : (selected ? st::historyCallInIconSelected : st::historyCallInIcon); - icon.paint(p, width - st::historyCallIconPosition.x() - icon.width(), st::historyCallIconPosition.y() - topMinus, width); + icon.paint(p, paintw - st::historyCallIconPosition.x() - icon.width(), st::historyCallIconPosition.y() - topMinus, paintw); } HistoryTextState HistoryCall::getState(QPoint point, HistoryStateRequest request) const { auto result = HistoryTextState(_parent); - if (QRect(0, 0, _width, _height).contains(point)) { + if (QRect(0, 0, width(), height()).contains(point)) { result.link = _link; return result; } @@ -3592,12 +3632,13 @@ TextWithEntities HistoryCall::selectedText(TextSelection selection) const { namespace { -int32 articleThumbWidth(PhotoData *thumb, int32 height) { - int32 w = thumb->medium->width(), h = thumb->medium->height(); +int articleThumbWidth(PhotoData *thumb, int height) { + auto w = thumb->medium->width(); + auto h = thumb->medium->height(); return qMax(qMin(height * w / h, height), 1); } -int32 articleThumbHeight(PhotoData *thumb, int32 width) { +int articleThumbHeight(PhotoData *thumb, int width) { return qMax(thumb->medium->height() * width / thumb->medium->width(), 1); } @@ -3628,10 +3669,9 @@ HistoryWebPage::HistoryWebPage( , _pixh(other._pixh) { } -void HistoryWebPage::initDimensions() { +QSize HistoryWebPage::countOptimalSize() { if (_data->pendingTill) { - _maxw = _minh = _height = 0; - return; + return { 0, 0 }; } const auto versionChanged = (_dataVersion != _data->version); if (versionChanged) { @@ -3725,8 +3765,8 @@ void HistoryWebPage::initDimensions() { // init dimensions auto l = st::msgPadding.left() + st::webPageLeft, r = st::msgPadding.right(); auto skipBlockWidth = _parent->skipBlockWidth(); - _maxw = skipBlockWidth; - _minh = 0; + auto maxWidth = skipBlockWidth; + auto minHeight = 0; auto siteNameHeight = _data->siteName.isEmpty() ? 0 : lineHeight; auto titleMinHeight = _title.isEmpty() ? 0 : lineHeight; @@ -3740,23 +3780,23 @@ void HistoryWebPage::initDimensions() { if (_siteNameWidth) { if (_title.isEmpty() && _description.isEmpty()) { - accumulate_max(_maxw, _siteNameWidth + _parent->skipBlockWidth()); + accumulate_max(maxWidth, _siteNameWidth + _parent->skipBlockWidth()); } else { - accumulate_max(_maxw, _siteNameWidth + articlePhotoMaxWidth); + accumulate_max(maxWidth, _siteNameWidth + articlePhotoMaxWidth); } - _minh += lineHeight; + minHeight += lineHeight; } if (!_title.isEmpty()) { - accumulate_max(_maxw, _title.maxWidth() + articlePhotoMaxWidth); - _minh += titleMinHeight; + accumulate_max(maxWidth, _title.maxWidth() + articlePhotoMaxWidth); + minHeight += titleMinHeight; } if (!_description.isEmpty()) { - accumulate_max(_maxw, _description.maxWidth() + articlePhotoMaxWidth); - _minh += descriptionMinHeight; + accumulate_max(maxWidth, _description.maxWidth() + articlePhotoMaxWidth); + minHeight += descriptionMinHeight; } if (_attach) { auto attachAtTop = !_siteNameWidth && _title.isEmpty() && _description.isEmpty(); - if (!attachAtTop) _minh += st::mediaInBubbleSkip; + if (!attachAtTop) minHeight += st::mediaInBubbleSkip; _attach->initDimensions(); auto bubble = _attach->bubbleMargins(); @@ -3764,34 +3804,33 @@ void HistoryWebPage::initDimensions() { if (isBubbleBottom() && _attach->customInfoLayout()) { maxMediaWidth += skipBlockWidth; } - accumulate_max(_maxw, maxMediaWidth); - _minh += _attach->minHeight() - bubble.top() - bubble.bottom(); + accumulate_max(maxWidth, maxMediaWidth); + minHeight += _attach->minHeight() - bubble.top() - bubble.bottom(); if (!_attach->additionalInfoString().isEmpty()) { - _minh += bottomInfoPadding(); + minHeight += bottomInfoPadding(); } } if (_data->type == WebPageVideo && _data->duration) { _duration = formatDurationText(_data->duration); _durationWidth = st::msgDateFont->width(_duration); } - _maxw += st::msgPadding.left() + st::webPageLeft + st::msgPadding.right(); + maxWidth += st::msgPadding.left() + st::webPageLeft + st::msgPadding.right(); auto padding = inBubblePadding(); - _minh += padding.top() + padding.bottom(); + minHeight += padding.top() + padding.bottom(); if (_asArticle) { - _minh = resizeGetHeight(_maxw); + minHeight = resizeGetHeight(maxWidth); } + return { maxWidth, minHeight }; } -int HistoryWebPage::resizeGetHeight(int width) { +QSize HistoryWebPage::countCurrentSize(int newWidth) { if (_data->pendingTill) { - _width = width; - _height = _minh; - return _height; + return { newWidth, minHeight() }; } - _width = width/* = qMin(width, _maxw)*/; - width -= st::msgPadding.left() + st::webPageLeft + st::msgPadding.right(); + auto innerWidth = newWidth - st::msgPadding.left() - st::webPageLeft - st::msgPadding.right(); + auto newHeight = 0; auto lineHeight = unitedLineHeight(); auto linesMax = isLogEntryOriginal() ? kMaxOriginalEntryLines : 5; @@ -3801,9 +3840,9 @@ int HistoryWebPage::resizeGetHeight(int width) { _pixh = linesMax * lineHeight; do { _pixw = articleThumbWidth(_data->photo, _pixh); - int32 wleft = width - st::webPagePhotoDelta - qMax(_pixw, int16(lineHeight)); + auto wleft = innerWidth - st::webPagePhotoDelta - qMax(_pixw, lineHeight); - _height = siteNameHeight; + newHeight = siteNameHeight; if (_title.isEmpty()) { _titleLines = 0; @@ -3813,73 +3852,81 @@ int HistoryWebPage::resizeGetHeight(int width) { } else { _titleLines = 2; } - _height += _titleLines * lineHeight; + newHeight += _titleLines * lineHeight; } auto descriptionHeight = _description.countHeight(wleft); if (descriptionHeight < (linesMax - siteNameLines - _titleLines) * st::webPageDescriptionFont->height) { // We have height for all the lines. _descriptionLines = -1; - _height += descriptionHeight; + newHeight += descriptionHeight; } else { _descriptionLines = (linesMax - siteNameLines - _titleLines); - _height += _descriptionLines * lineHeight; + newHeight += _descriptionLines * lineHeight; } - if (_height >= _pixh) { + if (newHeight >= _pixh) { break; } _pixh -= lineHeight; } while (_pixh > lineHeight); - _height += bottomInfoPadding(); + newHeight += bottomInfoPadding(); } else { - _height = siteNameHeight; + newHeight = siteNameHeight; if (_title.isEmpty()) { _titleLines = 0; } else { - if (_title.countHeight(width) < 2 * st::webPageTitleFont->height) { + if (_title.countHeight(innerWidth) < 2 * st::webPageTitleFont->height) { _titleLines = 1; } else { _titleLines = 2; } - _height += _titleLines * lineHeight; + newHeight += _titleLines * lineHeight; } if (_description.isEmpty()) { _descriptionLines = 0; } else { - auto descriptionHeight = _description.countHeight(width); + auto descriptionHeight = _description.countHeight(innerWidth); if (descriptionHeight < (linesMax - siteNameLines - _titleLines) * st::webPageDescriptionFont->height) { // We have height for all the lines. _descriptionLines = -1; - _height += descriptionHeight; + newHeight += descriptionHeight; } else { _descriptionLines = (linesMax - siteNameLines - _titleLines); - _height += _descriptionLines * lineHeight; + newHeight += _descriptionLines * lineHeight; } } if (_attach) { auto attachAtTop = !_siteNameWidth && !_titleLines && !_descriptionLines; - if (!attachAtTop) _height += st::mediaInBubbleSkip; + if (!attachAtTop) newHeight += st::mediaInBubbleSkip; auto bubble = _attach->bubbleMargins(); - _attach->resizeGetHeight(width + bubble.left() + bubble.right()); - _height += _attach->height() - bubble.top() - bubble.bottom(); + _attach->resizeGetHeight(innerWidth + bubble.left() + bubble.right()); + newHeight += _attach->height() - bubble.top() - bubble.bottom(); if (!_attach->additionalInfoString().isEmpty()) { - _height += bottomInfoPadding(); - } else if (isBubbleBottom() && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) { - _height += bottomInfoPadding(); + newHeight += bottomInfoPadding(); + } else if (isBubbleBottom() && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > innerWidth + bubble.left() + bubble.right()) { + newHeight += bottomInfoPadding(); } } } auto padding = inBubblePadding(); - _height += padding.top() + padding.bottom(); + newHeight += padding.top() + padding.bottom(); - return _height; + return { newWidth, newHeight }; +} + +TextSelection HistoryWebPage::toDescriptionSelection(TextSelection selection) const { + return internal::unshiftSelection(selection, _title); +} + +TextSelection HistoryWebPage::fromDescriptionSelection(TextSelection selection) const { + return internal::shiftSelection(selection, _title); } void HistoryWebPage::refreshParentId(not_null realParent) { @@ -3889,8 +3936,8 @@ void HistoryWebPage::refreshParentId(not_null realParent) { } void HistoryWebPage::draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const { - if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return; - int32 skipx = 0, skipy = 0, width = _width, height = _height; + if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) return; + auto paintx = 0, painty = 0, paintw = width(), painth = height(); auto outbg = _parent->hasOutLayout(); bool selected = (selection == FullSelection); @@ -3903,17 +3950,17 @@ void HistoryWebPage::draw(Painter &p, const QRect &r, TextSelection selection, T auto padding = inBubblePadding(); auto tshift = padding.top(); auto bshift = padding.bottom(); - width -= padding.left() + padding.right(); + paintw -= padding.left() + padding.right(); auto attachAdditionalInfoText = _attach ? _attach->additionalInfoString() : QString(); if (_asArticle) { bshift += bottomInfoPadding(); } else if (!attachAdditionalInfoText.isEmpty()) { bshift += bottomInfoPadding(); - } else if (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) { + } else if (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > paintw + bubble.left() + bubble.right()) { bshift += bottomInfoPadding(); } - QRect bar(rtlrect(st::msgPadding.left(), tshift, st::webPageBar, _height - tshift - bshift, _width)); + QRect bar(rtlrect(st::msgPadding.left(), tshift, st::webPageBar, height() - tshift - bshift, width())); p.fillRect(bar, barfg); auto lineHeight = unitedLineHeight(); @@ -3921,9 +3968,10 @@ void HistoryWebPage::draw(Painter &p, const QRect &r, TextSelection selection, T _data->photo->medium->load(false, false); bool full = _data->photo->medium->loaded(); QPixmap pix; - int32 pw = qMax(_pixw, int16(lineHeight)), ph = _pixh; - int32 pixw = _pixw, pixh = articleThumbHeight(_data->photo, _pixw); - int32 maxw = convertScale(_data->photo->medium->width()), maxh = convertScale(_data->photo->medium->height()); + auto pw = qMax(_pixw, lineHeight); + auto ph = _pixh; + auto pixw = _pixw, pixh = articleThumbHeight(_data->photo, _pixw); + auto maxw = convertScale(_data->photo->medium->width()), maxh = convertScale(_data->photo->medium->height()); if (pixw * ph != pixh * pw) { float64 coef = (pixw * ph > pixh * pw) ? qMin(ph / float64(pixh), maxh / float64(pixh)) : qMin(pw / float64(pixw), maxw / float64(pixw)); pixh = qRound(pixh * coef); @@ -3934,39 +3982,39 @@ void HistoryWebPage::draw(Painter &p, const QRect &r, TextSelection selection, T } else { pix = _data->photo->thumb->pixBlurredSingle(pixw, pixh, pw, ph, ImageRoundRadius::Small); } - p.drawPixmapLeft(padding.left() + width - pw, tshift, _width, pix); + p.drawPixmapLeft(padding.left() + paintw - pw, tshift, width(), pix); if (selected) { - App::roundRect(p, rtlrect(padding.left() + width - pw, tshift, pw, _pixh, _width), p.textPalette().selectOverlay, SelectedOverlaySmallCorners); + App::roundRect(p, rtlrect(padding.left() + paintw - pw, tshift, pw, _pixh, width()), p.textPalette().selectOverlay, SelectedOverlaySmallCorners); } - width -= pw + st::webPagePhotoDelta; + paintw -= pw + st::webPagePhotoDelta; } if (_siteNameWidth) { p.setFont(st::webPageTitleFont); p.setPen(semibold); - p.drawTextLeft(padding.left(), tshift, _width, (width >= _siteNameWidth) ? _data->siteName : st::webPageTitleFont->elided(_data->siteName, width)); + p.drawTextLeft(padding.left(), tshift, width(), (paintw >= _siteNameWidth) ? _data->siteName : st::webPageTitleFont->elided(_data->siteName, paintw)); tshift += lineHeight; } if (_titleLines) { p.setPen(outbg ? st::webPageTitleOutFg : st::webPageTitleInFg); - int32 endskip = 0; + auto endskip = 0; if (_title.hasSkipBlock()) { endskip = _parent->skipBlockWidth(); } - _title.drawLeftElided(p, padding.left(), tshift, width, _width, _titleLines, style::al_left, 0, -1, endskip, false, selection); + _title.drawLeftElided(p, padding.left(), tshift, paintw, width(), _titleLines, style::al_left, 0, -1, endskip, false, selection); tshift += _titleLines * lineHeight; } if (_descriptionLines) { p.setPen(outbg ? st::webPageDescriptionOutFg : st::webPageDescriptionInFg); - int32 endskip = 0; + auto endskip = 0; if (_description.hasSkipBlock()) { endskip = _parent->skipBlockWidth(); } if (_descriptionLines > 0) { - _description.drawLeftElided(p, padding.left(), tshift, width, _width, _descriptionLines, style::al_left, 0, -1, endskip, false, toDescriptionSelection(selection)); + _description.drawLeftElided(p, padding.left(), tshift, paintw, width(), _descriptionLines, style::al_left, 0, -1, endskip, false, toDescriptionSelection(selection)); tshift += _descriptionLines * lineHeight; } else { - _description.drawLeft(p, padding.left(), tshift, width, _width, style::al_left, 0, -1, toDescriptionSelection(selection)); - tshift += _description.countHeight(width); + _description.drawLeft(p, padding.left(), tshift, paintw, width(), style::al_left, 0, -1, toDescriptionSelection(selection)); + tshift += _description.countHeight(paintw); } } if (_attach) { @@ -3975,27 +4023,28 @@ void HistoryWebPage::draw(Painter &p, const QRect &r, TextSelection selection, T auto attachLeft = padding.left() - bubble.left(); auto attachTop = tshift - bubble.top(); - if (rtl()) attachLeft = _width - attachLeft - _attach->width(); + if (rtl()) attachLeft = width() - attachLeft - _attach->width(); p.translate(attachLeft, attachTop); auto attachSelection = selected ? FullSelection : TextSelection { 0, 0 }; _attach->draw(p, r.translated(-attachLeft, -attachTop), attachSelection, ms); - int32 pixwidth = _attach->width(), pixheight = _attach->height(); + auto pixwidth = _attach->width(); + auto pixheight = _attach->height(); if (_data->type == WebPageVideo && _attach->type() == MediaTypePhoto) { if (_attach->isReadyForOpen()) { if (_data->siteName == qstr("YouTube")) { - st::youtubeIcon.paint(p, (pixwidth - st::youtubeIcon.width()) / 2, (pixheight - st::youtubeIcon.height()) / 2, _width); + st::youtubeIcon.paint(p, (pixwidth - st::youtubeIcon.width()) / 2, (pixheight - st::youtubeIcon.height()) / 2, width()); } else { - st::videoIcon.paint(p, (pixwidth - st::videoIcon.width()) / 2, (pixheight - st::videoIcon.height()) / 2, _width); + st::videoIcon.paint(p, (pixwidth - st::videoIcon.width()) / 2, (pixheight - st::videoIcon.height()) / 2, width()); } } if (_durationWidth) { - int32 dateX = pixwidth - _durationWidth - st::msgDateImgDelta - 2 * st::msgDateImgPadding.x(); - int32 dateY = pixheight - st::msgDateFont->height - 2 * st::msgDateImgPadding.y() - st::msgDateImgDelta; - int32 dateW = pixwidth - dateX - st::msgDateImgDelta; - int32 dateH = pixheight - dateY - st::msgDateImgDelta; + auto dateX = pixwidth - _durationWidth - st::msgDateImgDelta - 2 * st::msgDateImgPadding.x(); + auto dateY = pixheight - st::msgDateFont->height - 2 * st::msgDateImgPadding.y() - st::msgDateImgDelta; + auto dateW = pixwidth - dateX - st::msgDateImgDelta; + auto dateH = pixheight - dateY - st::msgDateImgDelta; App::roundRect(p, dateX, dateY, dateW, dateH, selected ? st::msgDateImgBgSelected : st::msgDateImgBg, selected ? DateSelectedCorners : DateCorners); @@ -4010,7 +4059,7 @@ void HistoryWebPage::draw(Painter &p, const QRect &r, TextSelection selection, T if (!attachAdditionalInfoText.isEmpty()) { p.setFont(st::msgDateFont); p.setPen(selected ? (outbg ? st::msgOutDateFgSelected : st::msgInDateFgSelected) : (outbg ? st::msgOutDateFg : st::msgInDateFg)); - p.drawTextLeft(st::msgPadding.left(), bar.y() + bar.height() + st::mediaInBubbleSkip, _width, attachAdditionalInfoText); + p.drawTextLeft(st::msgPadding.left(), bar.y() + bar.height() + st::mediaInBubbleSkip, width(), attachAdditionalInfoText); } } } @@ -4018,28 +4067,28 @@ void HistoryWebPage::draw(Painter &p, const QRect &r, TextSelection selection, T HistoryTextState HistoryWebPage::getState(QPoint point, HistoryStateRequest request) const { auto result = HistoryTextState(_parent); - if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) { + if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) { return result; } - int32 skipx = 0, skipy = 0, width = _width, height = _height; + auto paintx = 0, painty = 0, paintw = width(), painth = height(); QMargins bubble(_attach ? _attach->bubbleMargins() : QMargins()); auto padding = inBubblePadding(); auto tshift = padding.top(); auto bshift = padding.bottom(); - if (_asArticle || (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right())) { + if (_asArticle || (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > paintw + bubble.left() + bubble.right())) { bshift += bottomInfoPadding(); } - width -= padding.left() + padding.right(); + paintw -= padding.left() + padding.right(); auto lineHeight = unitedLineHeight(); auto inThumb = false; if (_asArticle) { - int32 pw = qMax(_pixw, int16(lineHeight)); - if (rtlrect(padding.left() + width - pw, 0, pw, _pixh, _width).contains(point)) { + auto pw = qMax(_pixw, lineHeight); + if (rtlrect(padding.left() + paintw - pw, 0, pw, _pixh, width()).contains(point)) { inThumb = true; } - width -= pw + st::webPagePhotoDelta; + paintw -= pw + st::webPagePhotoDelta; } int symbolAdd = 0; if (_siteNameWidth) { @@ -4051,8 +4100,8 @@ HistoryTextState HistoryWebPage::getState(QPoint point, HistoryStateRequest requ titleRequest.lines = _titleLines; result = HistoryTextState(_parent, _title.getStateElidedLeft( point - QPoint(padding.left(), tshift), - width, - _width, + paintw, + width(), titleRequest)); } else if (point.y() >= tshift + _titleLines * lineHeight) { symbolAdd += _title.length(); @@ -4060,21 +4109,21 @@ HistoryTextState HistoryWebPage::getState(QPoint point, HistoryStateRequest requ tshift += _titleLines * lineHeight; } if (_descriptionLines) { - auto descriptionHeight = (_descriptionLines > 0) ? _descriptionLines * lineHeight : _description.countHeight(width); + auto descriptionHeight = (_descriptionLines > 0) ? _descriptionLines * lineHeight : _description.countHeight(paintw); if (point.y() >= tshift && point.y() < tshift + descriptionHeight) { if (_descriptionLines > 0) { Text::StateRequestElided descriptionRequest = request.forText(); descriptionRequest.lines = _descriptionLines; result = HistoryTextState(_parent, _description.getStateElidedLeft( point - QPoint(padding.left(), tshift), - width, - _width, + paintw, + width(), descriptionRequest)); } else { result = HistoryTextState(_parent, _description.getStateLeft( point - QPoint(padding.left(), tshift), - width, - _width, + paintw, + width(), request.forText())); } } else if (point.y() >= tshift + descriptionHeight) { @@ -4088,10 +4137,10 @@ HistoryTextState HistoryWebPage::getState(QPoint point, HistoryStateRequest requ auto attachAtTop = !_siteNameWidth && !_titleLines && !_descriptionLines; if (!attachAtTop) tshift += st::mediaInBubbleSkip; - if (QRect(padding.left(), tshift, width, _height - tshift - bshift).contains(point)) { + if (QRect(padding.left(), tshift, paintw, height() - tshift - bshift).contains(point)) { auto attachLeft = padding.left() - bubble.left(); auto attachTop = tshift - bubble.top(); - if (rtl()) attachLeft = _width - attachLeft - _attach->width(); + if (rtl()) attachLeft = width() - attachLeft - _attach->width(); result = _attach->getState(point - QPoint(attachLeft, attachTop), request); if (result.link && !_data->document && _data->photo && _attach->isReadyForOpen()) { @@ -4218,7 +4267,7 @@ HistoryGame::HistoryGame( , _description(other._description) { } -void HistoryGame::initDimensions() { +QSize HistoryGame::countOptimalSize() { auto lineHeight = unitedLineHeight(); if (!_openl && IsServerMsgId(_parent->id)) { @@ -4273,27 +4322,27 @@ void HistoryGame::initDimensions() { } // init dimensions - int32 l = st::msgPadding.left() + st::webPageLeft, r = st::msgPadding.right(); - int32 skipBlockWidth = _parent->skipBlockWidth(); - _maxw = skipBlockWidth; - _minh = 0; + auto l = st::msgPadding.left() + st::webPageLeft, r = st::msgPadding.right(); + auto skipBlockWidth = _parent->skipBlockWidth(); + auto maxWidth = skipBlockWidth; + auto minHeight = 0; - int32 titleMinHeight = _title.isEmpty() ? 0 : lineHeight; + auto titleMinHeight = _title.isEmpty() ? 0 : lineHeight; // enable any count of lines in game description / message - int descMaxLines = 4096; - int32 descriptionMinHeight = _description.isEmpty() ? 0 : qMin(_description.minHeight(), descMaxLines * lineHeight); + auto descMaxLines = 4096; + auto descriptionMinHeight = _description.isEmpty() ? 0 : qMin(_description.minHeight(), descMaxLines * lineHeight); if (!_title.isEmpty()) { - accumulate_max(_maxw, _title.maxWidth()); - _minh += titleMinHeight; + accumulate_max(maxWidth, _title.maxWidth()); + minHeight += titleMinHeight; } if (!_description.isEmpty()) { - accumulate_max(_maxw, _description.maxWidth()); - _minh += descriptionMinHeight; + accumulate_max(maxWidth, _description.maxWidth()); + minHeight += descriptionMinHeight; } if (_attach) { auto attachAtTop = !_titleLines && !_descriptionLines; - if (!attachAtTop) _minh += st::mediaInBubbleSkip; + if (!attachAtTop) minHeight += st::mediaInBubbleSkip; _attach->initDimensions(); QMargins bubble(_attach->bubbleMargins()); @@ -4301,16 +4350,17 @@ void HistoryGame::initDimensions() { if (isBubbleBottom() && _attach->customInfoLayout()) { maxMediaWidth += skipBlockWidth; } - accumulate_max(_maxw, maxMediaWidth); - _minh += _attach->minHeight() - bubble.top() - bubble.bottom(); + accumulate_max(maxWidth, maxMediaWidth); + minHeight += _attach->minHeight() - bubble.top() - bubble.bottom(); } - _maxw += st::msgPadding.left() + st::webPageLeft + st::msgPadding.right(); + maxWidth += st::msgPadding.left() + st::webPageLeft + st::msgPadding.right(); auto padding = inBubblePadding(); - _minh += padding.top() + padding.bottom(); + minHeight += padding.top() + padding.bottom(); if (!_gameTagWidth) { _gameTagWidth = st::msgDateFont->width(lang(lng_game_tag).toUpper()); } + return { maxWidth, minHeight }; } void HistoryGame::refreshParentId(not_null realParent) { @@ -4322,58 +4372,66 @@ void HistoryGame::refreshParentId(not_null realParent) { } } -int HistoryGame::resizeGetHeight(int width) { - _width = width = qMin(width, _maxw); - width -= st::msgPadding.left() + st::webPageLeft + st::msgPadding.right(); +QSize HistoryGame::countCurrentSize(int newWidth) { + accumulate_min(newWidth, maxWidth()); + auto innerWidth = newWidth - st::msgPadding.left() - st::webPageLeft - st::msgPadding.right(); // enable any count of lines in game description / message auto linesMax = 4096; auto lineHeight = unitedLineHeight(); - _height = 0; + auto newHeight = 0; if (_title.isEmpty()) { _titleLines = 0; } else { - if (_title.countHeight(width) < 2 * st::webPageTitleFont->height) { + if (_title.countHeight(innerWidth) < 2 * st::webPageTitleFont->height) { _titleLines = 1; } else { _titleLines = 2; } - _height += _titleLines * lineHeight; + newHeight += _titleLines * lineHeight; } if (_description.isEmpty()) { _descriptionLines = 0; } else { - int32 descriptionHeight = _description.countHeight(width); + auto descriptionHeight = _description.countHeight(innerWidth); if (descriptionHeight < (linesMax - _titleLines) * st::webPageDescriptionFont->height) { _descriptionLines = (descriptionHeight / st::webPageDescriptionFont->height); } else { _descriptionLines = (linesMax - _titleLines); } - _height += _descriptionLines * lineHeight; + newHeight += _descriptionLines * lineHeight; } if (_attach) { auto attachAtTop = !_titleLines && !_descriptionLines; - if (!attachAtTop) _height += st::mediaInBubbleSkip; + if (!attachAtTop) newHeight += st::mediaInBubbleSkip; QMargins bubble(_attach->bubbleMargins()); - _attach->resizeGetHeight(width + bubble.left() + bubble.right()); - _height += _attach->height() - bubble.top() - bubble.bottom(); - if (isBubbleBottom() && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) { - _height += bottomInfoPadding(); + _attach->resizeGetHeight(innerWidth + bubble.left() + bubble.right()); + newHeight += _attach->height() - bubble.top() - bubble.bottom(); + if (isBubbleBottom() && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > innerWidth + bubble.left() + bubble.right()) { + newHeight += bottomInfoPadding(); } } auto padding = inBubblePadding(); - _height += padding.top() + padding.bottom(); + newHeight += padding.top() + padding.bottom(); - return _height; + return { newWidth, newHeight }; +} + +TextSelection HistoryGame::toDescriptionSelection(TextSelection selection) const { + return internal::unshiftSelection(selection, _title); +} + +TextSelection HistoryGame::fromDescriptionSelection(TextSelection selection) const { + return internal::shiftSelection(selection, _title); } void HistoryGame::draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const { - if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return; - int32 width = _width, height = _height; + if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) return; + auto paintw = width(), painth = height(); auto outbg = _parent->hasOutLayout(); bool selected = (selection == FullSelection); @@ -4386,31 +4444,31 @@ void HistoryGame::draw(Painter &p, const QRect &r, TextSelection selection, Time auto padding = inBubblePadding(); auto tshift = padding.top(); auto bshift = padding.bottom(); - width -= padding.left() + padding.right(); - if (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) { + paintw -= padding.left() + padding.right(); + if (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > paintw + bubble.left() + bubble.right()) { bshift += bottomInfoPadding(); } - QRect bar(rtlrect(st::msgPadding.left(), tshift, st::webPageBar, _height - tshift - bshift, _width)); + QRect bar(rtlrect(st::msgPadding.left(), tshift, st::webPageBar, height() - tshift - bshift, width())); p.fillRect(bar, barfg); auto lineHeight = unitedLineHeight(); if (_titleLines) { p.setPen(semibold); - int32 endskip = 0; + auto endskip = 0; if (_title.hasSkipBlock()) { endskip = _parent->skipBlockWidth(); } - _title.drawLeftElided(p, padding.left(), tshift, width, _width, _titleLines, style::al_left, 0, -1, endskip, false, selection); + _title.drawLeftElided(p, padding.left(), tshift, paintw, width(), _titleLines, style::al_left, 0, -1, endskip, false, selection); tshift += _titleLines * lineHeight; } if (_descriptionLines) { p.setPen(outbg ? st::webPageDescriptionOutFg : st::webPageDescriptionInFg); - int32 endskip = 0; + auto endskip = 0; if (_description.hasSkipBlock()) { endskip = _parent->skipBlockWidth(); } - _description.drawLeftElided(p, padding.left(), tshift, width, _width, _descriptionLines, style::al_left, 0, -1, endskip, false, toDescriptionSelection(selection)); + _description.drawLeftElided(p, padding.left(), tshift, paintw, width(), _descriptionLines, style::al_left, 0, -1, endskip, false, toDescriptionSelection(selection)); tshift += _descriptionLines * lineHeight; } if (_attach) { @@ -4419,7 +4477,7 @@ void HistoryGame::draw(Painter &p, const QRect &r, TextSelection selection, Time auto attachLeft = padding.left() - bubble.left(); auto attachTop = tshift - bubble.top(); - if (rtl()) attachLeft = _width - attachLeft - _attach->width(); + if (rtl()) attachLeft = width() - attachLeft - _attach->width(); auto attachSelection = selected ? FullSelection : TextSelection { 0, 0 }; @@ -4446,19 +4504,19 @@ void HistoryGame::draw(Painter &p, const QRect &r, TextSelection selection, Time HistoryTextState HistoryGame::getState(QPoint point, HistoryStateRequest request) const { auto result = HistoryTextState(_parent); - if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) { + if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) { return result; } - int32 width = _width, height = _height; + auto paintw = width(), painth = height(); QMargins bubble(_attach ? _attach->bubbleMargins() : QMargins()); auto padding = inBubblePadding(); auto tshift = padding.top(); auto bshift = padding.bottom(); - if (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) { + if (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > paintw + bubble.left() + bubble.right()) { bshift += bottomInfoPadding(); } - width -= padding.left() + padding.right(); + paintw -= padding.left() + padding.right(); auto inThumb = false; auto symbolAdd = 0; @@ -4469,8 +4527,8 @@ HistoryTextState HistoryGame::getState(QPoint point, HistoryStateRequest request titleRequest.lines = _titleLines; result = HistoryTextState(_parent, _title.getStateElidedLeft( point - QPoint(padding.left(), tshift), - width, - _width, + paintw, + width(), titleRequest)); } else if (point.y() >= tshift + _titleLines * lineHeight) { symbolAdd += _title.length(); @@ -4483,8 +4541,8 @@ HistoryTextState HistoryGame::getState(QPoint point, HistoryStateRequest request descriptionRequest.lines = _descriptionLines; result = HistoryTextState(_parent, _description.getStateElidedLeft( point - QPoint(padding.left(), tshift), - width, - _width, + paintw, + width(), descriptionRequest)); } else if (point.y() >= tshift + _descriptionLines * lineHeight) { symbolAdd += _description.length(); @@ -4501,9 +4559,9 @@ HistoryTextState HistoryGame::getState(QPoint point, HistoryStateRequest request auto attachLeft = padding.left() - bubble.left(); auto attachTop = tshift - bubble.top(); - if (rtl()) attachLeft = _width - attachLeft - _attach->width(); + if (rtl()) attachLeft = width() - attachLeft - _attach->width(); - if (QRect(attachLeft, tshift, _attach->width(), _height - tshift - bshift).contains(point)) { + if (QRect(attachLeft, tshift, _attach->width(), height() - tshift - bshift).contains(point)) { if (_attach->isReadyForOpen()) { if (!_parent->isLogEntry()) { result.link = _openl; @@ -4734,7 +4792,7 @@ void HistoryInvoice::fillFromData(const MTPDmessageMediaInvoice &data) { } } -void HistoryInvoice::initDimensions() { +QSize HistoryInvoice::countOptimalSize() { auto lineHeight = unitedLineHeight(); if (_attach) { @@ -4746,27 +4804,27 @@ void HistoryInvoice::initDimensions() { } // init dimensions - int32 l = st::msgPadding.left(), r = st::msgPadding.right(); - int32 skipBlockWidth = _parent->skipBlockWidth(); - _maxw = skipBlockWidth; - _minh = 0; + auto l = st::msgPadding.left(), r = st::msgPadding.right(); + auto skipBlockWidth = _parent->skipBlockWidth(); + auto maxWidth = skipBlockWidth; + auto minHeight = 0; - int32 titleMinHeight = _title.isEmpty() ? 0 : lineHeight; + auto titleMinHeight = _title.isEmpty() ? 0 : lineHeight; // enable any count of lines in game description / message - int descMaxLines = 4096; - int32 descriptionMinHeight = _description.isEmpty() ? 0 : qMin(_description.minHeight(), descMaxLines * lineHeight); + auto descMaxLines = 4096; + auto descriptionMinHeight = _description.isEmpty() ? 0 : qMin(_description.minHeight(), descMaxLines * lineHeight); if (!_title.isEmpty()) { - accumulate_max(_maxw, _title.maxWidth()); - _minh += titleMinHeight; + accumulate_max(maxWidth, _title.maxWidth()); + minHeight += titleMinHeight; } if (!_description.isEmpty()) { - accumulate_max(_maxw, _description.maxWidth()); - _minh += descriptionMinHeight; + accumulate_max(maxWidth, _description.maxWidth()); + minHeight += descriptionMinHeight; } if (_attach) { auto attachAtTop = _title.isEmpty() && _description.isEmpty(); - if (!attachAtTop) _minh += st::mediaInBubbleSkip; + if (!attachAtTop) minHeight += st::mediaInBubbleSkip; _attach->initDimensions(); auto bubble = _attach->bubbleMargins(); @@ -4774,60 +4832,69 @@ void HistoryInvoice::initDimensions() { if (isBubbleBottom() && _attach->customInfoLayout()) { maxMediaWidth += skipBlockWidth; } - accumulate_max(_maxw, maxMediaWidth); - _minh += _attach->minHeight() - bubble.top() - bubble.bottom(); + accumulate_max(maxWidth, maxMediaWidth); + minHeight += _attach->minHeight() - bubble.top() - bubble.bottom(); } else { - accumulate_max(_maxw, _status.maxWidth()); - _minh += st::mediaInBubbleSkip + _status.minHeight(); + accumulate_max(maxWidth, _status.maxWidth()); + minHeight += st::mediaInBubbleSkip + _status.minHeight(); } auto padding = inBubblePadding(); - _maxw += padding.left() + padding.right(); - _minh += padding.top() + padding.bottom(); + maxWidth += padding.left() + padding.right(); + minHeight += padding.top() + padding.bottom(); + return { maxWidth, minHeight }; } -int HistoryInvoice::resizeGetHeight(int width) { - _width = width = qMin(width, _maxw); - width -= st::msgPadding.left() + st::msgPadding.right(); +QSize HistoryInvoice::countCurrentSize(int newWidth) { + accumulate_min(newWidth, maxWidth()); + auto innerWidth = newWidth - st::msgPadding.left() - st::msgPadding.right(); auto lineHeight = unitedLineHeight(); - _height = 0; + auto newHeight = 0; if (_title.isEmpty()) { _titleHeight = 0; } else { - if (_title.countHeight(width) < 2 * st::webPageTitleFont->height) { + if (_title.countHeight(innerWidth) < 2 * st::webPageTitleFont->height) { _titleHeight = lineHeight; } else { _titleHeight = 2 * lineHeight; } - _height += _titleHeight; + newHeight += _titleHeight; } if (_description.isEmpty()) { _descriptionHeight = 0; } else { - _descriptionHeight = _description.countHeight(width); - _height += _descriptionHeight; + _descriptionHeight = _description.countHeight(innerWidth); + newHeight += _descriptionHeight; } if (_attach) { auto attachAtTop = !_title.isEmpty() && _description.isEmpty(); - if (!attachAtTop) _height += st::mediaInBubbleSkip; + if (!attachAtTop) newHeight += st::mediaInBubbleSkip; QMargins bubble(_attach->bubbleMargins()); - _attach->resizeGetHeight(width + bubble.left() + bubble.right()); - _height += _attach->height() - bubble.top() - bubble.bottom(); - if (isBubbleBottom() && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) { - _height += bottomInfoPadding(); + _attach->resizeGetHeight(innerWidth + bubble.left() + bubble.right()); + newHeight += _attach->height() - bubble.top() - bubble.bottom(); + if (isBubbleBottom() && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > innerWidth + bubble.left() + bubble.right()) { + newHeight += bottomInfoPadding(); } } else { - _height += st::mediaInBubbleSkip + _status.countHeight(width); + newHeight += st::mediaInBubbleSkip + _status.countHeight(innerWidth); } auto padding = inBubblePadding(); - _height += padding.top() + padding.bottom(); + newHeight += padding.top() + padding.bottom(); - return _height; + return { newWidth, newHeight }; +} + +TextSelection HistoryInvoice::toDescriptionSelection(TextSelection selection) const { + return internal::unshiftSelection(selection, _title); +} + +TextSelection HistoryInvoice::fromDescriptionSelection(TextSelection selection) const { + return internal::shiftSelection(selection, _title); } void HistoryInvoice::refreshParentId(not_null realParent) { @@ -4837,8 +4904,8 @@ void HistoryInvoice::refreshParentId(not_null realParent) { } void HistoryInvoice::draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const { - if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return; - int32 width = _width, height = _height; + if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) return; + auto paintw = width(), painth = height(); auto outbg = _parent->hasOutLayout(); bool selected = (selection == FullSelection); @@ -4851,8 +4918,8 @@ void HistoryInvoice::draw(Painter &p, const QRect &r, TextSelection selection, T auto padding = inBubblePadding(); auto tshift = padding.top(); auto bshift = padding.bottom(); - width -= padding.left() + padding.right(); - if (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) { + paintw -= padding.left() + padding.right(); + if (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > paintw + bubble.left() + bubble.right()) { bshift += bottomInfoPadding(); } @@ -4861,18 +4928,18 @@ void HistoryInvoice::draw(Painter &p, const QRect &r, TextSelection selection, T p.setPen(semibold); p.setTextPalette(selected ? (outbg ? st::outTextPaletteSelected : st::inTextPaletteSelected) : (outbg ? st::outSemiboldPalette : st::inSemiboldPalette)); - int32 endskip = 0; + auto endskip = 0; if (_title.hasSkipBlock()) { endskip = _parent->skipBlockWidth(); } - _title.drawLeftElided(p, padding.left(), tshift, width, _width, _titleHeight / lineHeight, style::al_left, 0, -1, endskip, false, selection); + _title.drawLeftElided(p, padding.left(), tshift, paintw, width(), _titleHeight / lineHeight, style::al_left, 0, -1, endskip, false, selection); tshift += _titleHeight; p.setTextPalette(selected ? (outbg ? st::outTextPaletteSelected : st::inTextPaletteSelected) : (outbg ? st::outTextPalette : st::inTextPalette)); } if (_descriptionHeight) { p.setPen(outbg ? st::webPageDescriptionOutFg : st::webPageDescriptionInFg); - _description.drawLeft(p, padding.left(), tshift, width, _width, style::al_left, 0, -1, toDescriptionSelection(selection)); + _description.drawLeft(p, padding.left(), tshift, paintw, width(), style::al_left, 0, -1, toDescriptionSelection(selection)); tshift += _descriptionHeight; } if (_attach) { @@ -4881,7 +4948,7 @@ void HistoryInvoice::draw(Painter &p, const QRect &r, TextSelection selection, T auto attachLeft = padding.left() - bubble.left(); auto attachTop = tshift - bubble.top(); - if (rtl()) attachLeft = _width - attachLeft - _attach->width(); + if (rtl()) attachLeft = width() - attachLeft - _attach->width(); auto attachSelection = selected ? FullSelection : TextSelection { 0, 0 }; @@ -4905,26 +4972,26 @@ void HistoryInvoice::draw(Painter &p, const QRect &r, TextSelection selection, T p.translate(-attachLeft, -attachTop); } else { p.setPen(outbg ? st::webPageDescriptionOutFg : st::webPageDescriptionInFg); - _status.drawLeft(p, padding.left(), tshift + st::mediaInBubbleSkip, width, _width); + _status.drawLeft(p, padding.left(), tshift + st::mediaInBubbleSkip, paintw, width()); } } HistoryTextState HistoryInvoice::getState(QPoint point, HistoryStateRequest request) const { auto result = HistoryTextState(_parent); - if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) { + if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) { return result; } - int32 width = _width, height = _height; + auto paintw = width(), painth = height(); QMargins bubble(_attach ? _attach->bubbleMargins() : QMargins()); auto padding = inBubblePadding(); auto tshift = padding.top(); auto bshift = padding.bottom(); - if (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) { + if (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > paintw + bubble.left() + bubble.right()) { bshift += bottomInfoPadding(); } - width -= padding.left() + padding.right(); + paintw -= padding.left() + padding.right(); auto lineHeight = unitedLineHeight(); auto symbolAdd = 0; @@ -4934,8 +5001,8 @@ HistoryTextState HistoryInvoice::getState(QPoint point, HistoryStateRequest requ titleRequest.lines = _titleHeight / lineHeight; result = HistoryTextState(_parent, _title.getStateElidedLeft( point - QPoint(padding.left(), tshift), - width, - _width, + paintw, + width(), titleRequest)); } else if (point.y() >= tshift + _titleHeight) { symbolAdd += _title.length(); @@ -4946,8 +5013,8 @@ HistoryTextState HistoryInvoice::getState(QPoint point, HistoryStateRequest requ if (point.y() >= tshift && point.y() < tshift + _descriptionHeight) { result = HistoryTextState(_parent, _description.getStateLeft( point - QPoint(padding.left(), tshift), - width, - _width, + paintw, + width(), request.forText())); } else if (point.y() >= tshift + _descriptionHeight) { symbolAdd += _description.length(); @@ -4960,9 +5027,9 @@ HistoryTextState HistoryInvoice::getState(QPoint point, HistoryStateRequest requ auto attachLeft = padding.left() - bubble.left(); auto attachTop = tshift - bubble.top(); - if (rtl()) attachLeft = _width - attachLeft - _attach->width(); + if (rtl()) attachLeft = width() - attachLeft - _attach->width(); - if (QRect(attachLeft, tshift, _attach->width(), _height - tshift - bshift).contains(point)) { + if (QRect(attachLeft, tshift, _attach->width(), height() - tshift - bshift).contains(point)) { result = _attach->getState(point - QPoint(attachLeft, attachTop), request); } } @@ -5068,69 +5135,80 @@ HistoryLocation::HistoryLocation(not_null parent, const HistoryLoc , _link(std::make_shared(_data->coords)) { } -void HistoryLocation::initDimensions() { - int32 tw = fullWidth(), th = fullHeight(); +QSize HistoryLocation::countOptimalSize() { + auto tw = fullWidth(); + auto th = fullHeight(); if (tw > st::maxMediaSize) { th = (st::maxMediaSize * th) / tw; tw = st::maxMediaSize; } - int32 minWidth = qMax(st::minPhotoSize, _parent->infoWidth() + 2 * (st::msgDateImgDelta + st::msgDateImgPadding.x())); - _maxw = qMax(tw, int32(minWidth)); - _minh = qMax(th, int32(st::minPhotoSize)); + auto minWidth = qMax(st::minPhotoSize, _parent->infoWidth() + 2 * (st::msgDateImgDelta + st::msgDateImgPadding.x())); + auto maxWidth = qMax(tw, minWidth); + auto minHeight = qMax(th, st::minPhotoSize); if (_parent->hasBubble()) { if (!_title.isEmpty()) { - _minh += qMin(_title.countHeight(_maxw - st::msgPadding.left() - st::msgPadding.right()), 2 * st::webPageTitleFont->height); + minHeight += qMin(_title.countHeight(maxWidth - st::msgPadding.left() - st::msgPadding.right()), 2 * st::webPageTitleFont->height); } if (!_description.isEmpty()) { - _minh += qMin(_description.countHeight(_maxw - st::msgPadding.left() - st::msgPadding.right()), 3 * st::webPageDescriptionFont->height); + minHeight += qMin(_description.countHeight(maxWidth - st::msgPadding.left() - st::msgPadding.right()), 3 * st::webPageDescriptionFont->height); } if (!_title.isEmpty() || !_description.isEmpty()) { - _minh += st::mediaInBubbleSkip; + minHeight += st::mediaInBubbleSkip; if (isBubbleTop()) { - _minh += st::msgPadding.top(); + minHeight += st::msgPadding.top(); } } } + return { maxWidth, minHeight }; } -int HistoryLocation::resizeGetHeight(int width) { - _width = qMin(width, _maxw); +QSize HistoryLocation::countCurrentSize(int newWidth) { + accumulate_min(newWidth, maxWidth()); - int32 tw = fullWidth(), th = fullHeight(); + auto tw = fullWidth(); + auto th = fullHeight(); if (tw > st::maxMediaSize) { th = (st::maxMediaSize * th) / tw; tw = st::maxMediaSize; } - _height = th; - if (tw > _width) { - _height = (_width * _height / tw); + auto newHeight = th; + if (tw > newWidth) { + newHeight = (newWidth * newHeight / tw); } else { - _width = tw; + newWidth = tw; } - int32 minWidth = qMax(st::minPhotoSize, _parent->infoWidth() + 2 * (st::msgDateImgDelta + st::msgDateImgPadding.x())); - _width = qMax(_width, int32(minWidth)); - _height = qMax(_height, int32(st::minPhotoSize)); + auto minWidth = qMax(st::minPhotoSize, _parent->infoWidth() + 2 * (st::msgDateImgDelta + st::msgDateImgPadding.x())); + accumulate_max(newWidth, minWidth); + accumulate_max(newHeight, st::minPhotoSize); if (_parent->hasBubble()) { if (!_title.isEmpty()) { - _height += qMin(_title.countHeight(_width - st::msgPadding.left() - st::msgPadding.right()), st::webPageTitleFont->height * 2); + newHeight += qMin(_title.countHeight(newWidth - st::msgPadding.left() - st::msgPadding.right()), st::webPageTitleFont->height * 2); } if (!_description.isEmpty()) { - _height += qMin(_description.countHeight(_width - st::msgPadding.left() - st::msgPadding.right()), st::webPageDescriptionFont->height * 3); + newHeight += qMin(_description.countHeight(newWidth - st::msgPadding.left() - st::msgPadding.right()), st::webPageDescriptionFont->height * 3); } if (!_title.isEmpty() || !_description.isEmpty()) { - _height += st::mediaInBubbleSkip; + newHeight += st::mediaInBubbleSkip; if (isBubbleTop()) { - _height += st::msgPadding.top(); + newHeight += st::msgPadding.top(); } } } - return _height; + return { newWidth, newHeight }; +} + +TextSelection HistoryLocation::toDescriptionSelection(TextSelection selection) const { + return internal::unshiftSelection(selection, _title); +} + +TextSelection HistoryLocation::fromDescriptionSelection(TextSelection selection) const { + return internal::shiftSelection(selection, _title); } void HistoryLocation::draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const { - if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return; - int32 skipx = 0, skipy = 0, width = _width, height = _height; + if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) return; + auto paintx = 0, painty = 0, paintw = width(), painth = height(); bool bubble = _parent->hasBubble(); auto outbg = _parent->hasOutLayout(); bool selected = (selection == FullSelection); @@ -5138,46 +5216,46 @@ void HistoryLocation::draw(Painter &p, const QRect &r, TextSelection selection, if (bubble) { if (!_title.isEmpty() || !_description.isEmpty()) { if (isBubbleTop()) { - skipy += st::msgPadding.top(); + painty += st::msgPadding.top(); } } - int32 textw = _width - st::msgPadding.left() - st::msgPadding.right(); + auto textw = width() - st::msgPadding.left() - st::msgPadding.right(); if (!_title.isEmpty()) { p.setPen(outbg ? st::webPageTitleOutFg : st::webPageTitleInFg); - _title.drawLeftElided(p, skipx + st::msgPadding.left(), skipy, textw, _width, 2, style::al_left, 0, -1, 0, false, selection); - skipy += qMin(_title.countHeight(textw), 2 * st::webPageTitleFont->height); + _title.drawLeftElided(p, paintx + st::msgPadding.left(), painty, textw, width(), 2, style::al_left, 0, -1, 0, false, selection); + painty += qMin(_title.countHeight(textw), 2 * st::webPageTitleFont->height); } if (!_description.isEmpty()) { p.setPen(outbg ? st::webPageDescriptionOutFg : st::webPageDescriptionInFg); - _description.drawLeftElided(p, skipx + st::msgPadding.left(), skipy, textw, _width, 3, style::al_left, 0, -1, 0, false, toDescriptionSelection(selection)); - skipy += qMin(_description.countHeight(textw), 3 * st::webPageDescriptionFont->height); + _description.drawLeftElided(p, paintx + st::msgPadding.left(), painty, textw, width(), 3, style::al_left, 0, -1, 0, false, toDescriptionSelection(selection)); + painty += qMin(_description.countHeight(textw), 3 * st::webPageDescriptionFont->height); } if (!_title.isEmpty() || !_description.isEmpty()) { - skipy += st::mediaInBubbleSkip; + painty += st::mediaInBubbleSkip; } - height -= skipy; + painth -= painty; } else { - App::roundShadow(p, 0, 0, width, height, selected ? st::msgInShadowSelected : st::msgInShadow, selected ? InSelectedShadowCorners : InShadowCorners); + App::roundShadow(p, 0, 0, paintw, painth, selected ? st::msgInShadowSelected : st::msgInShadow, selected ? InSelectedShadowCorners : InShadowCorners); } _data->load(); auto roundRadius = ImageRoundRadius::Large; auto roundCorners = ((isBubbleTop() && _title.isEmpty() && _description.isEmpty()) ? (RectPart::TopLeft | RectPart::TopRight) : RectPart::None) | (isBubbleBottom() ? (RectPart::BottomLeft | RectPart::BottomRight) : RectPart::None); - auto rthumb = QRect(skipx, skipy, width, height); + auto rthumb = QRect(paintx, painty, paintw, painth); if (_data && !_data->thumb->isNull()) { - int32 w = _data->thumb->width(), h = _data->thumb->height(); + auto w = _data->thumb->width(), h = _data->thumb->height(); QPixmap pix; - if (width * h == height * w || (w == fullWidth() && h == fullHeight())) { - pix = _data->thumb->pixSingle(width, height, width, height, roundRadius, roundCorners); - } else if (width * h > height * w) { - int32 nw = height * w / h; - pix = _data->thumb->pixSingle(nw, height, width, height, roundRadius, roundCorners); + if (paintw * h == painth * w || (w == fullWidth() && h == fullHeight())) { + pix = _data->thumb->pixSingle(paintw, painth, paintw, painth, roundRadius, roundCorners); + } else if (paintw * h > painth * w) { + auto nw = painth * w / h; + pix = _data->thumb->pixSingle(nw, painth, paintw, painth, roundRadius, roundCorners); } else { - int32 nh = width * h / w; - pix = _data->thumb->pixSingle(width, nh, width, height, roundRadius, roundCorners); + auto nh = paintw * h / w; + pix = _data->thumb->pixSingle(paintw, nh, paintw, painth, roundRadius, roundCorners); } p.drawPixmap(rthumb.topLeft(), pix); } else { @@ -5188,13 +5266,13 @@ void HistoryLocation::draw(Painter &p, const QRect &r, TextSelection selection, } if (_parent->getMedia() == this) { - auto fullRight = skipx + width; - auto fullBottom = _height; - _parent->drawInfo(p, fullRight, fullBottom, skipx * 2 + width, selected, InfoDisplayOverImage); + auto fullRight = paintx + paintw; + auto fullBottom = height(); + _parent->drawInfo(p, fullRight, fullBottom, paintx * 2 + paintw, selected, InfoDisplayOverImage); if (!bubble && _parent->displayRightAction()) { auto fastShareLeft = (fullRight + st::historyFastShareLeft); auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize); - _parent->drawRightAction(p, fastShareLeft, fastShareTop, 2 * skipx + width); + _parent->drawRightAction(p, fastShareLeft, fastShareTop, 2 * paintx + paintw); } } } @@ -5203,59 +5281,59 @@ HistoryTextState HistoryLocation::getState(QPoint point, HistoryStateRequest req auto result = HistoryTextState(_parent); auto symbolAdd = 0; - if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) { + if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) { return result; } - int32 skipx = 0, skipy = 0, width = _width, height = _height; + auto paintx = 0, painty = 0, paintw = width(), painth = height(); bool bubble = _parent->hasBubble(); if (bubble) { if (!_title.isEmpty() || !_description.isEmpty()) { if (isBubbleTop()) { - skipy += st::msgPadding.top(); + painty += st::msgPadding.top(); } } - auto textw = _width - st::msgPadding.left() - st::msgPadding.right(); + auto textw = width() - st::msgPadding.left() - st::msgPadding.right(); if (!_title.isEmpty()) { auto titleh = qMin(_title.countHeight(textw), 2 * st::webPageTitleFont->height); - if (point.y() >= skipy && point.y() < skipy + titleh) { + if (point.y() >= painty && point.y() < painty + titleh) { result = HistoryTextState(_parent, _title.getStateLeft( - point - QPoint(skipx + st::msgPadding.left(), skipy), + point - QPoint(paintx + st::msgPadding.left(), painty), textw, - _width, + width(), request.forText())); return result; - } else if (point.y() >= skipy + titleh) { + } else if (point.y() >= painty + titleh) { symbolAdd += _title.length(); } - skipy += titleh; + painty += titleh; } if (!_description.isEmpty()) { auto descriptionh = qMin(_description.countHeight(textw), 3 * st::webPageDescriptionFont->height); - if (point.y() >= skipy && point.y() < skipy + descriptionh) { + if (point.y() >= painty && point.y() < painty + descriptionh) { result = HistoryTextState(_parent, _description.getStateLeft( - point - QPoint(skipx + st::msgPadding.left(), skipy), + point - QPoint(paintx + st::msgPadding.left(), painty), textw, - _width, + width(), request.forText())); - } else if (point.y() >= skipy + descriptionh) { + } else if (point.y() >= painty + descriptionh) { symbolAdd += _description.length(); } - skipy += descriptionh; + painty += descriptionh; } if (!_title.isEmpty() || !_description.isEmpty()) { - skipy += st::mediaInBubbleSkip; + painty += st::mediaInBubbleSkip; } - height -= skipy; + painth -= painty; } - if (QRect(skipx, skipy, width, height).contains(point) && _data) { + if (QRect(paintx, painty, paintw, painth).contains(point) && _data) { result.link = _link; } if (_parent->getMedia() == this) { - auto fullRight = skipx + width; - auto fullBottom = _height; + auto fullRight = paintx + paintw; + auto fullBottom = height(); if (_parent->pointInTime(fullRight, fullBottom, point, InfoDisplayOverImage)) { result.cursor = HistoryInDateCursorState; } @@ -5323,15 +5401,16 @@ bool HistoryLocation::needsBubble() const { return message->viaBot() || message->Has() || message->displayForwardedFrom() - || message->displayFromName(); +// || message->displayFromName() // #TODO media views +; } return false; } -int32 HistoryLocation::fullWidth() const { +int HistoryLocation::fullWidth() const { return st::locationSize.width(); } -int32 HistoryLocation::fullHeight() const { +int HistoryLocation::fullHeight() const { return st::locationSize.height(); } diff --git a/Telegram/SourceFiles/history/history_media_types.h b/Telegram/SourceFiles/history/history_media_types.h index dca037b15..c6bd39701 100644 --- a/Telegram/SourceFiles/history/history_media_types.h +++ b/Telegram/SourceFiles/history/history_media_types.h @@ -7,6 +7,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL */ #pragma once +#include "base/runtime_composer.h" #include "history/history_media.h" #include "ui/effects/radial_animation.h" #include "data/data_document.h" @@ -72,39 +73,20 @@ protected: FileClickHandlerPtr &&savel, FileClickHandlerPtr &&cancell); void setDocumentLinks( - not_null document, - not_null realParent, - bool inlinegif = false) { - FileClickHandlerPtr open, save; - const auto context = realParent->fullId(); - if (inlinegif) { - open = std::make_shared(document, context); - } else { - open = std::make_shared(document, context); - } - if (inlinegif) { - save = std::make_shared(document, context); - } else if (document->isVoiceMessage()) { - save = std::make_shared(document, context); - } else { - save = std::make_shared(document, context); - } - setLinks( - std::move(open), - std::move(save), - std::make_shared(document, context)); - } + not_null document, + not_null realParent, + bool inlinegif = false); // >= 0 will contain download / upload string, _statusSize = loaded bytes // < 0 will contain played string, _statusSize = -(seconds + 1) played // 0x7FFFFFF0 will contain status for not yet downloaded file // 0x7FFFFFF1 will contain status for already downloaded file // 0x7FFFFFF2 will contain status for failed to download / upload file - mutable int32 _statusSize; + mutable int _statusSize; mutable QString _statusText; // duration = -1 - no duration, duration = -2 - "GIF" duration - void setStatusSize(int32 newSize, int32 fullSize, int32 duration, qint64 realDuration) const; + void setStatusSize(int newSize, int fullSize, int duration, qint64 realDuration) const; void step_radial(TimeMs ms, bool timer); void thumbAnimationCallback(); @@ -174,9 +156,6 @@ public: return std::make_unique(newParent, realParent, *this); } - void initDimensions() override; - int resizeGetHeight(int width) override; - void draw(Painter &p, const QRect &clip, TextSelection selection, TimeMs ms) const override; HistoryTextState getState(QPoint point, HistoryStateRequest request) const override; @@ -254,6 +233,9 @@ protected: bool dataLoaded() const override; private: + QSize countOptimalSize() override; + QSize countCurrentSize(int newWidth) override; + bool needInfoDisplay() const; void validateGroupedCache( const QRect &geometry, @@ -262,8 +244,9 @@ private: not_null cache) const; not_null _data; - int16 _pixw = 1; - int16 _pixh = 1; + int _serviceWidth = 0; + int _pixw = 1; + int _pixh = 1; Text _caption; }; @@ -288,9 +271,6 @@ public: return std::make_unique(newParent, realParent, *this); } - void initDimensions() override; - int resizeGetHeight(int width) override; - void draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const override; HistoryTextState getState(QPoint point, HistoryStateRequest request) const override; @@ -369,16 +349,19 @@ protected: bool dataLoaded() const override; private: + QSize countOptimalSize() override; + QSize countCurrentSize(int newWidth) override; + void validateGroupedCache( const QRect &geometry, RectParts corners, not_null cacheKey, not_null cache) const; - void setStatusSize(int32 newSize) const; + void setStatusSize(int newSize) const; void updateStatusText() const; not_null _data; - int32 _thumbw; + int _thumbw; Text _caption; }; @@ -408,9 +391,6 @@ public: return std::make_unique(newParent, *this); } - void initDimensions() override; - int resizeGetHeight(int width) override; - void draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const override; HistoryTextState getState(QPoint point, HistoryStateRequest request) const override; void updatePressed(QPoint point) override; @@ -475,10 +455,13 @@ protected: bool dataLoaded() const override; private: + QSize countOptimalSize() override; + QSize countCurrentSize(int newWidth) override; + void createComponents(bool caption); void fillNamedFromData(HistoryDocumentNamed *named); - void setStatusSize(int32 newSize, qint64 realDuration = 0) const; + void setStatusSize(int newSize, qint64 realDuration = 0) const; bool updateStatusText() const; // returns showPause // Callback is a void(const QString &, const QString &, const Text &) functor. @@ -509,9 +492,6 @@ public: return std::make_unique(newParent, *this); } - void initDimensions() override; - int resizeGetHeight(int width) override; - void draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const override; HistoryTextState getState(QPoint point, HistoryStateRequest request) const override; @@ -591,6 +571,9 @@ protected: } private: + QSize countOptimalSize() override; + QSize countCurrentSize(int newWidth) override; + bool needInfoDisplay() const; int additionalWidth( const HistoryMessageVia *via, @@ -602,14 +585,14 @@ private: not_null _data; ClickHandlerPtr _openInMediaviewLink; - int32 _thumbw = 1; - int32 _thumbh = 1; + int _thumbw = 1; + int _thumbh = 1; Text _caption; mutable std::unique_ptr _roundPlayback; Media::Clip::ReaderPointer _gif; - void setStatusSize(int32 newSize) const; + void setStatusSize(int newSize) const; void updateStatusText() const; }; @@ -631,9 +614,6 @@ public: return std::make_unique(newParent, _data); } - void initDimensions() override; - int resizeGetHeight(int width) override; - void draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const override; HistoryTextState getState(QPoint point, HistoryStateRequest request) const override; @@ -676,12 +656,15 @@ public: } private: + QSize countOptimalSize() override; + QSize countCurrentSize(int newWidth) override; + int additionalWidth(const HistoryMessageVia *via, const HistoryMessageReply *reply) const; int additionalWidth() const; QString toString() const; - int16 _pixw = 1; - int16 _pixh = 1; + int _pixw = 1; + int _pixh = 1; ClickHandlerPtr _packLink; not_null _data; QString _emoji; @@ -713,8 +696,6 @@ public: _phone); } - void initDimensions() override; - void draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const override; HistoryTextState getState(QPoint point, HistoryStateRequest request) const override; @@ -753,6 +734,8 @@ public: ~HistoryContact(); private: + QSize countOptimalSize() override; + int32 _userId = 0; UserData *_contact = nullptr; @@ -782,8 +765,6 @@ public: Unexpected("Clone HistoryCall."); } - void initDimensions() override; - void draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const override; HistoryTextState getState(QPoint point, HistoryStateRequest request) const override; @@ -815,6 +796,8 @@ public: } private: + QSize countOptimalSize() override; + static FinishReason GetReason(const MTPDmessageActionPhoneCall &call); FinishReason _reason = FinishReason::Missed; @@ -847,8 +830,6 @@ public: return std::make_unique(newParent, *this); } - void initDimensions() override; - int resizeGetHeight(int width) override; void refreshParentId(not_null realParent) override; void draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const override; @@ -918,12 +899,11 @@ public: } private: - TextSelection toDescriptionSelection(TextSelection selection) const { - return internal::unshiftSelection(selection, _title); - } - TextSelection fromDescriptionSelection(TextSelection selection) const { - return internal::shiftSelection(selection, _title); - } + QSize countOptimalSize() override; + QSize countCurrentSize(int newWidth) override; + + TextSelection toDescriptionSelection(TextSelection selection) const; + TextSelection fromDescriptionSelection(TextSelection selection) const; QMargins inBubblePadding() const; int bottomInfoPadding() const; bool isLogEntryOriginal() const; @@ -943,8 +923,8 @@ private: QString _duration; int _durationWidth = 0; - int16 _pixw = 0; - int16 _pixh = 0; + int _pixw = 0; + int _pixh = 0; }; @@ -964,8 +944,6 @@ public: return std::make_unique(newParent, *this); } - void initDimensions() override; - int resizeGetHeight(int width) override; void refreshParentId(not_null realParent) override; void draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const override; @@ -1044,12 +1022,11 @@ public: } private: - TextSelection toDescriptionSelection(TextSelection selection) const { - return internal::unshiftSelection(selection, _title); - } - TextSelection fromDescriptionSelection(TextSelection selection) const { - return internal::shiftSelection(selection, _title); - } + QSize countOptimalSize() override; + QSize countCurrentSize(int newWidth) override; + + TextSelection toDescriptionSelection(TextSelection selection) const; + TextSelection fromDescriptionSelection(TextSelection selection) const; QMargins inBubblePadding() const; int bottomInfoPadding() const; @@ -1057,7 +1034,7 @@ private: std::shared_ptr _openl; std::unique_ptr _attach; - int32 _titleLines, _descriptionLines; + int _titleLines, _descriptionLines; Text _title, _description; @@ -1085,8 +1062,6 @@ public: return std::make_unique(newParent, *this); } - void initDimensions() override; - int resizeGetHeight(int width) override; void refreshParentId(not_null realParent) override; MsgId getReceiptMsgId() const { @@ -1145,14 +1120,13 @@ public: } private: + QSize countOptimalSize() override; + QSize countCurrentSize(int newWidth) override; + void fillFromData(const MTPDmessageMediaInvoice &data); - TextSelection toDescriptionSelection(TextSelection selection) const { - return internal::unshiftSelection(selection, _title); - } - TextSelection fromDescriptionSelection(TextSelection selection) const { - return internal::shiftSelection(selection, _title); - } + TextSelection toDescriptionSelection(TextSelection selection) const; + TextSelection fromDescriptionSelection(TextSelection selection) const; QMargins inBubblePadding() const; int bottomInfoPadding() const; @@ -1193,9 +1167,6 @@ public: return std::make_unique(newParent, *this); } - void initDimensions() override; - int resizeGetHeight(int32 width) override; - void draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const override; HistoryTextState getState(QPoint point, HistoryStateRequest request) const override; @@ -1230,18 +1201,17 @@ public: } private: - TextSelection toDescriptionSelection(TextSelection selection) const { - return internal::unshiftSelection(selection, _title); - } - TextSelection fromDescriptionSelection(TextSelection selection) const { - return internal::shiftSelection(selection, _title); - } + QSize countOptimalSize() override; + QSize countCurrentSize(int newWidth) override; + + TextSelection toDescriptionSelection(TextSelection selection) const; + TextSelection fromDescriptionSelection(TextSelection selection) const; LocationData *_data; Text _title, _description; ClickHandlerPtr _link; - int32 fullWidth() const; - int32 fullHeight() const; + int fullWidth() const; + int fullHeight() const; }; diff --git a/Telegram/SourceFiles/history/history_message.cpp b/Telegram/SourceFiles/history/history_message.cpp index 223d786c0..d741e8f34 100644 --- a/Telegram/SourceFiles/history/history_message.cpp +++ b/Telegram/SourceFiles/history/history_message.cpp @@ -11,6 +11,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "mainwidget.h" #include "mainwindow.h" #include "apiwrap.h" +#include "history/history.h" #include "history/history_item_components.h" #include "history/history_location_manager.h" #include "history/history_media_types.h" @@ -37,133 +38,6 @@ namespace { constexpr auto kPinnedMessageTextLimit = 16; -class KeyboardStyle : public ReplyKeyboard::Style { -public: - using ReplyKeyboard::Style::Style; - - int buttonRadius() const override; - - void startPaint(Painter &p) const override; - const style::TextStyle &textStyle() const override; - void repaint(not_null item) const override; - -protected: - void paintButtonBg( - Painter &p, - const QRect &rect, - float64 howMuchOver) const override; - void paintButtonIcon(Painter &p, const QRect &rect, int outerWidth, HistoryMessageMarkupButton::Type type) const override; - void paintButtonLoading(Painter &p, const QRect &rect) const override; - int minButtonWidth(HistoryMessageMarkupButton::Type type) const override; - -}; - -void KeyboardStyle::startPaint(Painter &p) const { - p.setPen(st::msgServiceFg); -} - -const style::TextStyle &KeyboardStyle::textStyle() const { - return st::serviceTextStyle; -} - -void KeyboardStyle::repaint(not_null item) const { - Auth().data().requestItemRepaint(item); -} - -int KeyboardStyle::buttonRadius() const { - return st::dateRadius; -} - -void KeyboardStyle::paintButtonBg( - Painter &p, - const QRect &rect, - float64 howMuchOver) const { - App::roundRect(p, rect, st::msgServiceBg, StickerCorners); - if (howMuchOver > 0) { - auto o = p.opacity(); - p.setOpacity(o * howMuchOver); - App::roundRect(p, rect, st::msgBotKbOverBgAdd, BotKbOverCorners); - p.setOpacity(o); - } -} - -void KeyboardStyle::paintButtonIcon( - Painter &p, - const QRect &rect, - int outerWidth, - HistoryMessageMarkupButton::Type type) const { - using Button = HistoryMessageMarkupButton; - auto getIcon = [](Button::Type type) -> const style::icon* { - switch (type) { - case Button::Type::Url: return &st::msgBotKbUrlIcon; - case Button::Type::SwitchInlineSame: - case Button::Type::SwitchInline: return &st::msgBotKbSwitchPmIcon; - } - return nullptr; - }; - if (auto icon = getIcon(type)) { - icon->paint(p, rect.x() + rect.width() - icon->width() - st::msgBotKbIconPadding, rect.y() + st::msgBotKbIconPadding, outerWidth); - } -} - -void KeyboardStyle::paintButtonLoading(Painter &p, const QRect &rect) const { - auto icon = &st::historySendingInvertedIcon; - icon->paint(p, rect.x() + rect.width() - icon->width() - st::msgBotKbIconPadding, rect.y() + rect.height() - icon->height() - st::msgBotKbIconPadding, rect.x() * 2 + rect.width()); -} - -int KeyboardStyle::minButtonWidth( - HistoryMessageMarkupButton::Type type) const { - using Button = HistoryMessageMarkupButton; - int result = 2 * buttonPadding(), iconWidth = 0; - switch (type) { - case Button::Type::Url: iconWidth = st::msgBotKbUrlIcon.width(); break; - case Button::Type::SwitchInlineSame: - case Button::Type::SwitchInline: iconWidth = st::msgBotKbSwitchPmIcon.width(); break; - case Button::Type::Callback: - case Button::Type::Game: iconWidth = st::historySendingInvertedIcon.width(); break; - } - if (iconWidth > 0) { - result = std::max(result, 2 * iconWidth + 4 * int(st::msgBotKbIconPadding)); - } - return result; -} - -QString AdminBadgeText() { - return lang(lng_admin_badge); -} - -QString FastReplyText() { - return lang(lng_fast_reply); -} - -style::color FromNameFg(not_null peer, bool selected) { - if (selected) { - const style::color colors[] = { - st::historyPeer1NameFgSelected, - st::historyPeer2NameFgSelected, - st::historyPeer3NameFgSelected, - st::historyPeer4NameFgSelected, - st::historyPeer5NameFgSelected, - st::historyPeer6NameFgSelected, - st::historyPeer7NameFgSelected, - st::historyPeer8NameFgSelected, - }; - return colors[Data::PeerColorIndex(peer->id)]; - } else { - const style::color colors[] = { - st::historyPeer1NameFg, - st::historyPeer2NameFg, - st::historyPeer3NameFg, - st::historyPeer4NameFg, - st::historyPeer5NameFg, - st::historyPeer6NameFg, - st::historyPeer7NameFg, - st::historyPeer8NameFg, - }; - return colors[Data::PeerColorIndex(peer->id)]; - } -} - MTPDmessage::Flags NewForwardedFlags( not_null peer, UserId from, @@ -682,56 +556,6 @@ void HistoryMessage::createComponentsHelper( createComponents(config); } -void HistoryMessage::updateMediaInBubbleState() { - auto mediaHasSomethingBelow = false; - auto mediaHasSomethingAbove = false; - auto getMediaHasSomethingAbove = [this] { - return displayFromName() - || displayForwardedFrom() - || Has() - || Has(); - }; - auto entry = Get(); - if (entry) { - mediaHasSomethingBelow = true; - mediaHasSomethingAbove = getMediaHasSomethingAbove(); - auto entryState = (mediaHasSomethingAbove || !emptyText() || (_media && _media->isDisplayed())) ? MediaInBubbleState::Bottom : MediaInBubbleState::None; - entry->_page->setInBubbleState(entryState); - } - if (!_media) { - return; - } - - _media->updateNeedBubbleState(); - if (!drawBubble()) { - _media->setInBubbleState(MediaInBubbleState::None); - return; - } - - if (!entry) { - mediaHasSomethingAbove = getMediaHasSomethingAbove(); - } - if (!emptyText()) { - if (_media->isAboveMessage()) { - mediaHasSomethingBelow = true; - } else { - mediaHasSomethingAbove = true; - } - } - auto computeState = [mediaHasSomethingAbove, mediaHasSomethingBelow] { - if (mediaHasSomethingAbove) { - if (mediaHasSomethingBelow) { - return MediaInBubbleState::Middle; - } - return MediaInBubbleState::Bottom; - } else if (mediaHasSomethingBelow) { - return MediaInBubbleState::Top; - } - return MediaInBubbleState::None; - }; - _media->setInBubbleState(computeState()); -} - int HistoryMessage::viewsCount() const { if (const auto views = Get()) { return views->_views; @@ -777,7 +601,7 @@ void HistoryMessage::applyGroupAdminChanges( } else { _flags &= ~MTPDmessage_ClientFlag::f_has_admin_badge; } - setPendingInitDimensions(); + Auth().data().requestItemViewResize(this); } } @@ -1075,131 +899,6 @@ int32 HistoryMessage::plainMaxWidth() const { return st::msgPadding.left() + _text.maxWidth() + st::msgPadding.right(); } -void HistoryMessage::initDimensions() { - updateMediaInBubbleState(); - refreshEditedBadge(); - if (drawBubble()) { - auto forwarded = Get(); - auto reply = Get(); - auto via = Get(); - auto entry = Get(); - if (forwarded) { - forwarded->create(via); - } - if (reply) { - reply->updateName(); - } - if (displayFromName()) { - updateAdminBadgeState(); - } - - auto mediaDisplayed = false; - if (_media) { - mediaDisplayed = _media->isDisplayed(); - _media->initDimensions(); - } - if (entry) { - entry->_page->initDimensions(); - } - - // Entry page is always a bubble bottom. - auto mediaOnBottom = (mediaDisplayed && _media->isBubbleBottom()) || (entry/* && entry->_page->isBubbleBottom()*/); - auto mediaOnTop = (mediaDisplayed && _media->isBubbleTop()) || (entry && entry->_page->isBubbleTop()); - - if (mediaOnBottom) { - if (_text.hasSkipBlock()) { - _text.removeSkipBlock(); - _textWidth = -1; - _textHeight = 0; - } - } else if (!_text.hasSkipBlock()) { - _text.setSkipBlock(skipBlockWidth(), skipBlockHeight()); - _textWidth = -1; - _textHeight = 0; - } - - _maxw = plainMaxWidth(); - _minh = emptyText() ? 0 : _text.minHeight(); - if (!mediaOnBottom) { - _minh += st::msgPadding.bottom(); - if (mediaDisplayed) _minh += st::mediaInBubbleSkip; - } - if (!mediaOnTop) { - _minh += st::msgPadding.top(); - if (mediaDisplayed) _minh += st::mediaInBubbleSkip; - if (entry) _minh += st::mediaInBubbleSkip; - } - if (mediaDisplayed) { - // Parts don't participate in maxWidth() in case of media message. - accumulate_max(_maxw, _media->maxWidth()); - _minh += _media->minHeight(); - } else { - // Count parts in maxWidth(), don't count them in minHeight(). - // They will be added in resizeGetHeight() anyway. - if (displayFromName()) { - auto namew = st::msgPadding.left() - + displayFrom()->nameText.maxWidth() - + st::msgPadding.right(); - if (via && !forwarded) { - namew += st::msgServiceFont->spacew + via->maxWidth; - } - const auto replyWidth = hasFastReply() - ? st::msgFont->width(FastReplyText()) - : 0; - if (_flags & MTPDmessage_ClientFlag::f_has_admin_badge) { - const auto badgeWidth = st::msgFont->width( - AdminBadgeText()); - namew += st::msgPadding.right() - + std::max(badgeWidth, replyWidth); - } else if (replyWidth) { - namew += st::msgPadding.right() + replyWidth; - } - accumulate_max(_maxw, namew); - } else if (via && !forwarded) { - accumulate_max(_maxw, st::msgPadding.left() + via->maxWidth + st::msgPadding.right()); - } - if (forwarded) { - auto namew = st::msgPadding.left() + forwarded->text.maxWidth() + st::msgPadding.right(); - if (via) { - namew += st::msgServiceFont->spacew + via->maxWidth; - } - accumulate_max(_maxw, namew); - } - if (reply) { - auto replyw = st::msgPadding.left() + reply->maxReplyWidth - st::msgReplyPadding.left() - st::msgReplyPadding.right() + st::msgPadding.right(); - if (reply->replyToVia) { - replyw += st::msgServiceFont->spacew + reply->replyToVia->maxWidth; - } - accumulate_max(_maxw, replyw); - } - if (entry) { - accumulate_max(_maxw, entry->_page->maxWidth()); - _minh += entry->_page->minHeight(); - } - } - } else if (_media) { - _media->initDimensions(); - _maxw = _media->maxWidth(); - _minh = _media->isDisplayed() ? _media->minHeight() : 0; - } else { - _maxw = st::msgMinWidth; - _minh = 0; - } - if (const auto markup = inlineReplyMarkup()) { - if (!markup->inlineKeyboard) { - markup->inlineKeyboard = std::make_unique( - this, - std::make_unique(st::msgBotKbButton)); - } - - // if we have a text bubble we can resize it to fit the keyboard - // but if we have only media we don't do that - if (!emptyText()) { - accumulate_max(_maxw, markup->inlineKeyboard->naturalWidth()); - } - } -} - bool HistoryMessage::drawBubble() const { if (isHiddenByGroup()) { return false; @@ -1209,11 +908,6 @@ bool HistoryMessage::drawBubble() const { return _media ? (!emptyText() || _media->needsBubble()) : !isEmpty(); } -bool HistoryMessage::hasFromName() const { - return !hasOutLayout() - && (!history()->peer->isUser() || history()->peer->isSelf()); -} - bool HistoryMessage::hasFastReply() const { return !hasOutLayout() && (history()->peer->isChat() || history()->peer->isMegagroup()); @@ -1223,71 +917,14 @@ bool HistoryMessage::displayFastReply() const { return hasFastReply() && history()->peer->canWrite(); } -QRect HistoryMessage::countGeometry() const { - auto maxwidth = qMin(st::msgMaxWidth, _maxw); - if (_media && _media->width() < maxwidth) { - maxwidth = qMax(_media->width(), qMin(maxwidth, plainMaxWidth())); - } - - const auto outLayout = hasOutLayout(); - auto contentLeft = (outLayout && !Adaptive::ChatWide()) - ? st::msgMargin.right() - : st::msgMargin.left(); - if (hasFromPhoto()) { - contentLeft += st::msgPhotoSkip; -// } else if (!Adaptive::Wide() && !out() && !fromChannel() && st::msgPhotoSkip - (hmaxwidth - hwidth) > 0) { -// contentLeft += st::msgPhotoSkip - (hmaxwidth - hwidth); - } - - auto contentWidth = width() - st::msgMargin.left() - st::msgMargin.right(); - if (history()->peer->isSelf() && !outLayout) { - contentWidth -= st::msgPhotoSkip; - } - if (contentWidth > maxwidth) { - if (outLayout && !Adaptive::ChatWide()) { - contentLeft += contentWidth - maxwidth; - } - contentWidth = maxwidth; - } - - const auto contentTop = marginTop(); - return QRect( - contentLeft, - contentTop, - contentWidth, - _height - contentTop - marginBottom()); -} - -void HistoryMessage::fromNameUpdated(int32 width) const { - const auto replyWidth = hasFastReply() - ? st::msgFont->width(FastReplyText()) - : 0; - if (_flags & MTPDmessage_ClientFlag::f_has_admin_badge) { - const auto badgeWidth = st::msgFont->width(AdminBadgeText()); - width -= st::msgPadding.right() + std::max(badgeWidth, replyWidth); - } else if (replyWidth) { - width -= st::msgPadding.right() + replyWidth; - } - _fromNameVersion = displayFrom()->nameVersion; - if (!Has()) { - if (auto via = Get()) { - via->resize(width - - st::msgPadding.left() - - st::msgPadding.right() - - author()->nameText.maxWidth() - - st::msgServiceFont->spacew); - } - } -} - void HistoryMessage::applyEdition(const MTPDmessage &message) { int keyboardTop = -1; - if (!pendingResize()) { - if (auto keyboard = inlineReplyKeyboard()) { - int h = st::msgBotKbButton.margin + keyboard->naturalHeight(); - keyboardTop = _height - h + st::msgBotKbButton.margin - marginBottom(); - } - } + //if (!pendingResize()) {// #TODO edit bot message + // if (auto keyboard = inlineReplyKeyboard()) { + // int h = st::msgBotKbButton.margin + keyboard->naturalHeight(); + // keyboardTop = _height - h + st::msgBotKbButton.margin - marginBottom(); + // } + //} if (message.has_edit_date()) { _flags |= MTPDmessage::Flag::f_edit_date; @@ -1375,7 +1012,7 @@ void HistoryMessage::updateMedia(const MTPMessageMedia *media) { } else { setMedia(media); } - setPendingInitDimensions(); + Auth().data().requestItemViewResize(this); } void HistoryMessage::addToUnreadMentions(UnreadMentionType type) { @@ -1539,7 +1176,7 @@ void HistoryMessage::setReplyMarkup(const MTPReplyMarkup *markup) { if (Has()) { RemoveComponents(HistoryMessageReplyMarkup::Bit()); } - setPendingInitDimensions(); + Auth().data().requestItemViewResize(this); Notify::replyMarkupUpdated(this); } return; @@ -1558,8 +1195,7 @@ void HistoryMessage::setReplyMarkup(const MTPReplyMarkup *markup) { changed = true; } if (changed) { - setPendingInitDimensions(); - + Auth().data().requestItemViewResize(this); Notify::replyMarkupUpdated(this); } } else { @@ -1570,8 +1206,7 @@ void HistoryMessage::setReplyMarkup(const MTPReplyMarkup *markup) { AddComponents(HistoryMessageReplyMarkup::Bit()); } Get()->create(*markup); - setPendingInitDimensions(); - + Auth().data().requestItemViewResize(this); Notify::replyMarkupUpdated(this); } } @@ -1711,168 +1346,19 @@ void HistoryMessage::setViewsCount(int32 count) { _textWidth = -1; _textHeight = 0; } - setPendingInitDimensions(); + Auth().data().requestItemViewResize(this); } } -void HistoryMessage::setId(MsgId newId) { - bool wasPositive = (id > 0), positive = (newId > 0); - HistoryItem::setId(newId); - if (wasPositive == positive) { - Auth().data().requestItemRepaint(this); - } else { - if (_text.hasSkipBlock()) { - _text.setSkipBlock(HistoryMessage::skipBlockWidth(), HistoryMessage::skipBlockHeight()); - _textWidth = -1; - _textHeight = 0; - } - setPendingInitDimensions(); - } -} - -void HistoryMessage::draw(Painter &p, QRect clip, TextSelection selection, TimeMs ms) const { - auto outbg = hasOutLayout(); - auto bubble = drawBubble(); - auto selected = (selection == FullSelection); - - auto g = countGeometry(); - if (g.width() < 1) { - return; - } - - auto dateh = 0; - if (auto date = Get()) { - dateh = date->height(); - } - if (auto unreadbar = Get()) { - auto unreadbarh = unreadbar->height(); - if (clip.intersects(QRect(0, dateh, width(), unreadbarh))) { - p.translate(0, dateh); - unreadbar->paint(p, 0, width()); - p.translate(0, -dateh); - } - } - - auto fullAnimMs = App::main() ? App::main()->highlightStartTime(this) : 0LL; - if (fullAnimMs > 0 && fullAnimMs <= ms) { - auto animms = ms - fullAnimMs; - if (animms < st::activeFadeInDuration + st::activeFadeOutDuration) { - auto top = marginTop(); - auto bottom = marginBottom(); - auto fill = qMin(top, bottom); - auto skiptop = top - fill; - auto fillheight = fill + g.height() + fill; - - auto dt = (animms > st::activeFadeInDuration) ? (1. - (animms - st::activeFadeInDuration) / float64(st::activeFadeOutDuration)) : (animms / float64(st::activeFadeInDuration)); - auto o = p.opacity(); - p.setOpacity(o * dt); - p.fillRect(0, skiptop, width(), fillheight, st::defaultTextPalette.selectOverlay); - p.setOpacity(o); - } - } - - p.setTextPalette(selected ? (outbg ? st::outTextPaletteSelected : st::inTextPaletteSelected) : (outbg ? st::outTextPalette : st::inTextPalette)); - - auto keyboard = inlineReplyKeyboard(); - if (keyboard) { - auto keyboardHeight = st::msgBotKbButton.margin + keyboard->naturalHeight(); - g.setHeight(g.height() - keyboardHeight); - auto keyboardPosition = QPoint(g.left(), g.top() + g.height() + st::msgBotKbButton.margin); - p.translate(keyboardPosition); - keyboard->paint(p, g.width(), clip.translated(-keyboardPosition), ms); - p.translate(-keyboardPosition); - } - - if (bubble) { - if (displayFromName() && displayFrom()->nameVersion > _fromNameVersion) { - fromNameUpdated(g.width()); - } - - auto entry = Get(); - auto mediaDisplayed = _media && _media->isDisplayed(); - - auto skipTail = isAttachedToNext() - || (_media && _media->skipBubbleTail()) - || (keyboard != nullptr); - auto displayTail = skipTail ? RectPart::None : (outbg && !Adaptive::ChatWide()) ? RectPart::Right : RectPart::Left; - HistoryView::paintBubble(p, g, width(), selected, outbg, displayTail); - - // Entry page is always a bubble bottom. - auto mediaOnBottom = (mediaDisplayed && _media->isBubbleBottom()) || (entry/* && entry->_page->isBubbleBottom()*/); - auto mediaOnTop = (mediaDisplayed && _media->isBubbleTop()) || (entry && entry->_page->isBubbleTop()); - - auto trect = g.marginsRemoved(st::msgPadding); - if (mediaOnBottom) { - trect.setHeight(trect.height() + st::msgPadding.bottom()); - } - if (mediaOnTop) { - trect.setY(trect.y() - st::msgPadding.top()); - } else { - paintFromName(p, trect, selected); - paintForwardedInfo(p, trect, selected); - paintReplyInfo(p, trect, selected); - paintViaBotIdInfo(p, trect, selected); - } - if (entry) { - trect.setHeight(trect.height() - entry->_page->height()); - } - auto needDrawInfo = mediaOnBottom ? !(entry ? entry->_page->customInfoLayout() : _media->customInfoLayout()) : true; - if (mediaDisplayed) { - auto mediaAboveText = _media->isAboveMessage(); - auto mediaHeight = _media->height(); - auto mediaLeft = g.left(); - auto mediaTop = mediaAboveText ? trect.y() : (trect.y() + trect.height() - mediaHeight); - if (!mediaAboveText) { - paintText(p, trect, selection); - } - p.translate(mediaLeft, mediaTop); - _media->draw(p, clip.translated(-mediaLeft, -mediaTop), skipTextSelection(selection), ms); - p.translate(-mediaLeft, -mediaTop); - - if (mediaAboveText) { - trect.setY(trect.y() + mediaHeight); - paintText(p, trect, selection); - } else { - needDrawInfo = !_media->customInfoLayout(); - } - } else { - paintText(p, trect, selection); - } - if (entry) { - auto entryLeft = g.left(); - auto entryTop = trect.y() + trect.height(); - p.translate(entryLeft, entryTop); - auto entrySelection = skipTextSelection(selection); - if (mediaDisplayed) { - entrySelection = _media->skipSelection(entrySelection); - } - entry->_page->draw(p, clip.translated(-entryLeft, -entryTop), entrySelection, ms); - p.translate(-entryLeft, -entryTop); - } - if (needDrawInfo) { - HistoryMessage::drawInfo(p, g.left() + g.width(), g.top() + g.height(), 2 * g.left() + g.width(), selected, InfoDisplayDefault); - } - if (displayRightAction()) { - const auto fastShareSkip = snap( - (g.height() - st::historyFastShareSize) / 2, - 0, - st::historyFastShareBottom); - const auto fastShareLeft = g.left() + g.width() + st::historyFastShareLeft; - const auto fastShareTop = g.top() + g.height() - fastShareSkip - st::historyFastShareSize; - drawRightAction(p, fastShareLeft, fastShareTop, width()); - } - } else if (_media && _media->isDisplayed()) { - p.translate(g.topLeft()); - _media->draw(p, clip.translated(-g.topLeft()), skipTextSelection(selection), ms); - p.translate(-g.topLeft()); - } - - p.restoreTextPalette(); - - const auto reply = Get(); - if (reply && reply->isNameUpdated()) { - const_cast(this)->setPendingInitDimensions(); +void HistoryMessage::setRealId(MsgId newId) { + HistoryItem::setRealId(newId); + + if (_text.hasSkipBlock()) { + _text.setSkipBlock(HistoryMessage::skipBlockWidth(), HistoryMessage::skipBlockHeight()); + _textWidth = -1; + _textHeight = 0; } + Auth().data().requestItemViewResize(this); } void HistoryMessage::drawRightAction(Painter &p, int left, int top, int outerWidth) const { @@ -1890,261 +1376,12 @@ void HistoryMessage::drawRightAction(Painter &p, int left, int top, int outerWid } } -void HistoryMessage::paintFromName( - Painter &p, - QRect &trect, - bool selected) const { - if (displayFromName()) { - const auto badgeWidth = [&] { - if (_flags & MTPDmessage_ClientFlag::f_has_admin_badge) { - return st::msgFont->width(AdminBadgeText()); - } - return 0; - }(); - const auto replyWidth = [&] { - if (isUnderCursor() && displayFastReply()) { - return st::msgFont->width(FastReplyText()); - } - return 0; - }(); - const auto rightWidth = replyWidth ? replyWidth : badgeWidth; - auto availableLeft = trect.left(); - auto availableWidth = trect.width(); - if (rightWidth) { - availableWidth -= st::msgPadding.right() + rightWidth; - } - - p.setFont(st::msgNameFont); - if (isPost()) { - p.setPen(selected ? st::msgInServiceFgSelected : st::msgInServiceFg); - } else { - p.setPen(FromNameFg(author(), selected)); - } - displayFrom()->nameText.drawElided(p, availableLeft, trect.top(), availableWidth); - auto skipWidth = author()->nameText.maxWidth() + st::msgServiceFont->spacew; - availableLeft += skipWidth; - availableWidth -= skipWidth; - - auto forwarded = Get(); - auto via = Get(); - if (via && !forwarded && availableWidth > 0) { - auto outbg = hasOutLayout(); - p.setPen(selected ? (outbg ? st::msgOutServiceFgSelected : st::msgInServiceFgSelected) : (outbg ? st::msgOutServiceFg : st::msgInServiceFg)); - p.drawText(availableLeft, trect.top() + st::msgServiceFont->ascent, via->text); - auto skipWidth = via->width + st::msgServiceFont->spacew; - availableLeft += skipWidth; - availableWidth -= skipWidth; - } - if (rightWidth) { - p.setPen(selected ? st::msgInDateFgSelected : st::msgInDateFg); - p.setFont(ClickHandler::showAsActive(_fastReplyLink) - ? st::msgFont->underline() - : st::msgFont); - p.drawText( - trect.left() + trect.width() - rightWidth, - trect.top() + st::msgFont->ascent, - replyWidth ? FastReplyText() : AdminBadgeText()); - } - trect.setY(trect.y() + st::msgNameFont->height); - } -} - -void HistoryMessage::paintForwardedInfo(Painter &p, QRect &trect, bool selected) const { - if (displayForwardedFrom()) { - style::font serviceFont(st::msgServiceFont), serviceName(st::msgServiceNameFont); - - auto outbg = hasOutLayout(); - p.setPen(selected ? (outbg ? st::msgOutServiceFgSelected : st::msgInServiceFgSelected) : (outbg ? st::msgOutServiceFg : st::msgInServiceFg)); - p.setFont(serviceFont); - - auto forwarded = Get(); - auto breakEverywhere = (forwarded->text.countHeight(trect.width()) > 2 * serviceFont->height); - p.setTextPalette(selected ? (outbg ? st::outFwdTextPaletteSelected : st::inFwdTextPaletteSelected) : (outbg ? st::outFwdTextPalette : st::inFwdTextPalette)); - forwarded->text.drawElided(p, trect.x(), trect.y(), trect.width(), 2, style::al_left, 0, -1, 0, breakEverywhere); - p.setTextPalette(selected ? (outbg ? st::outTextPaletteSelected : st::inTextPaletteSelected) : (outbg ? st::outTextPalette : st::inTextPalette)); - - trect.setY(trect.y() + (((forwarded->text.maxWidth() > trect.width()) ? 2 : 1) * serviceFont->height)); - } -} - -void HistoryMessage::paintReplyInfo(Painter &p, QRect &trect, bool selected) const { - if (auto reply = Get()) { - int32 h = st::msgReplyPadding.top() + st::msgReplyBarSize.height() + st::msgReplyPadding.bottom(); - - auto flags = HistoryMessageReply::PaintFlag::InBubble | 0; - if (selected) { - flags |= HistoryMessageReply::PaintFlag::Selected; - } - reply->paint(p, this, trect.x(), trect.y(), trect.width(), flags); - - trect.setY(trect.y() + h); - } -} - -void HistoryMessage::paintViaBotIdInfo(Painter &p, QRect &trect, bool selected) const { - if (!displayFromName() && !Has()) { - if (auto via = Get()) { - p.setFont(st::msgServiceNameFont); - p.setPen(selected ? (hasOutLayout() ? st::msgOutServiceFgSelected : st::msgInServiceFgSelected) : (hasOutLayout() ? st::msgOutServiceFg : st::msgInServiceFg)); - p.drawTextLeft(trect.left(), trect.top(), width(), via->text); - trect.setY(trect.y() + st::msgServiceNameFont->height); - } - } -} - -void HistoryMessage::paintText(Painter &p, QRect &trect, TextSelection selection) const { - auto outbg = hasOutLayout(); - auto selected = (selection == FullSelection); - p.setPen(outbg ? (selected ? st::historyTextOutFgSelected : st::historyTextOutFg) : (selected ? st::historyTextInFgSelected : st::historyTextInFg)); - p.setFont(st::msgFont); - _text.draw(p, trect.x(), trect.y(), trect.width(), style::al_left, 0, -1, selection); -} - void HistoryMessage::dependencyItemRemoved(HistoryItem *dependency) { if (auto reply = Get()) { reply->itemRemoved(this, dependency); } } -int HistoryMessage::resizeContentGetHeight() { - const auto result = performResizeGetHeight(); - - const auto keyboard = inlineReplyKeyboard(); - if (const auto markup = Get()) { - const auto oldTop = markup->oldTop; - if (oldTop >= 0) { - markup->oldTop = -1; - if (keyboard) { - const auto height = st::msgBotKbButton.margin + keyboard->naturalHeight(); - const auto keyboardTop = _height - height + st::msgBotKbButton.margin - marginBottom(); - if (keyboardTop != oldTop) { - Notify::inlineKeyboardMoved(this, oldTop, keyboardTop); - } - } - } - } - - return result; -} - -int HistoryMessage::performResizeGetHeight() { - if (width() < st::msgMinWidth) { - return _height; - } - - auto contentWidth = width() - (st::msgMargin.left() + st::msgMargin.right()); - if (history()->peer->isSelf() && !hasOutLayout()) { - contentWidth -= st::msgPhotoSkip; - } - if (contentWidth < st::msgPadding.left() + st::msgPadding.right() + 1) { - contentWidth = st::msgPadding.left() + st::msgPadding.right() + 1; - } else if (contentWidth > st::msgMaxWidth) { - contentWidth = st::msgMaxWidth; - } - if (drawBubble()) { - auto forwarded = Get(); - auto reply = Get(); - auto via = Get(); - auto entry = Get(); - - auto mediaDisplayed = false; - if (_media) { - mediaDisplayed = _media->isDisplayed(); - } - - // Entry page is always a bubble bottom. - auto mediaOnBottom = (mediaDisplayed && _media->isBubbleBottom()) || (entry/* && entry->_page->isBubbleBottom()*/); - auto mediaOnTop = (mediaDisplayed && _media->isBubbleTop()) || (entry && entry->_page->isBubbleTop()); - - if (contentWidth >= _maxw) { - _height = _minh; - if (mediaDisplayed) { - _media->resizeGetHeight(_maxw); - if (entry) { - _height += entry->_page->resizeGetHeight(countGeometry().width()); - } - } else if (entry) { - // In case of text-only message it is counted in _minh already. - entry->_page->resizeGetHeight(countGeometry().width()); - } - } else { - if (emptyText()) { - _height = 0; - } else { - auto textWidth = qMax(contentWidth - st::msgPadding.left() - st::msgPadding.right(), 1); - if (textWidth != _textWidth) { - _textWidth = textWidth; - _textHeight = _text.countHeight(textWidth); - } - _height = _textHeight; - } - if (!mediaOnBottom) { - _height += st::msgPadding.bottom(); - if (mediaDisplayed) _height += st::mediaInBubbleSkip; - } - if (!mediaOnTop) { - _height += st::msgPadding.top(); - if (mediaDisplayed) _height += st::mediaInBubbleSkip; - if (entry) _height += st::mediaInBubbleSkip; - } - if (mediaDisplayed) { - _height += _media->resizeGetHeight(contentWidth); - if (entry) { - _height += entry->_page->resizeGetHeight(countGeometry().width()); - } - } else if (entry) { - _height += entry->_page->resizeGetHeight(contentWidth); - } - } - - if (displayFromName()) { - fromNameUpdated(countGeometry().width()); - _height += st::msgNameFont->height; - } else if (via && !forwarded) { - via->resize(countGeometry().width() - st::msgPadding.left() - st::msgPadding.right()); - _height += st::msgNameFont->height; - } - - if (displayForwardedFrom()) { - auto fwdheight = ((forwarded->text.maxWidth() > (countGeometry().width() - st::msgPadding.left() - st::msgPadding.right())) ? 2 : 1) * st::semiboldFont->height; - _height += fwdheight; - } - - if (reply) { - reply->resize(countGeometry().width() - st::msgPadding.left() - st::msgPadding.right()); - _height += st::msgReplyPadding.top() + st::msgReplyBarSize.height() + st::msgReplyPadding.bottom(); - } - } else if (_media && _media->isDisplayed()) { - _height = _media->resizeGetHeight(contentWidth); - } else { - _height = 0; - } - if (const auto keyboard = inlineReplyKeyboard()) { - const auto g = countGeometry(); - const auto keyboardHeight = st::msgBotKbButton.margin + keyboard->naturalHeight(); - _height += keyboardHeight; - keyboard->resize(g.width(), keyboardHeight - st::msgBotKbButton.margin); - } - - _height += marginTop() + marginBottom(); - return _height; -} - -bool HistoryMessage::hasPoint(QPoint point) const { - const auto g = countGeometry(); - if (g.width() < 1) { - return false; - } - - if (drawBubble()) { - return g.contains(point); - } else if (_media) { - return _media->hasPoint(point - g.topLeft()); - } else { - return false; - } -} - bool HistoryMessage::pointInTime(int right, int bottom, QPoint point, InfoDisplayType type) const { auto infoRight = right; auto infoBottom = bottom; @@ -2163,112 +1400,6 @@ bool HistoryMessage::pointInTime(int right, int bottom, QPoint point, InfoDispla return QRect(dateX, dateY, HistoryMessage::timeWidth(), st::msgDateFont->height).contains(point); } -HistoryTextState HistoryMessage::getState(QPoint point, HistoryStateRequest request) const { - auto result = HistoryTextState(this); - - auto g = countGeometry(); - if (g.width() < 1) { - return result; - } - - auto keyboard = inlineReplyKeyboard(); - auto keyboardHeight = 0; - if (keyboard) { - keyboardHeight = keyboard->naturalHeight(); - g.setHeight(g.height() - st::msgBotKbButton.margin - keyboardHeight); - } - - if (drawBubble()) { - auto entry = Get(); - auto mediaDisplayed = _media && _media->isDisplayed(); - - // Entry page is always a bubble bottom. - auto mediaOnBottom = (mediaDisplayed && _media->isBubbleBottom()) || (entry/* && entry->_page->isBubbleBottom()*/); - auto mediaOnTop = (mediaDisplayed && _media->isBubbleTop()) || (entry && entry->_page->isBubbleTop()); - - auto trect = g.marginsRemoved(st::msgPadding); - if (mediaOnBottom) { - trect.setHeight(trect.height() + st::msgPadding.bottom()); - } - if (mediaOnTop) { - trect.setY(trect.y() - st::msgPadding.top()); - } else { - if (getStateFromName(point, trect, &result)) return result; - if (getStateForwardedInfo(point, trect, &result, request)) return result; - if (getStateReplyInfo(point, trect, &result)) return result; - if (getStateViaBotIdInfo(point, trect, &result)) return result; - } - if (entry) { - auto entryHeight = entry->_page->height(); - trect.setHeight(trect.height() - entryHeight); - auto entryLeft = g.left(); - auto entryTop = trect.y() + trect.height(); - if (point.y() >= entryTop && point.y() < entryTop + entryHeight) { - result = entry->_page->getState( - point - QPoint(entryLeft, entryTop), - request); - result.symbol += _text.length() + (mediaDisplayed ? _media->fullSelectionLength() : 0); - } - } - - auto needDateCheck = mediaOnBottom ? !(entry ? entry->_page->customInfoLayout() : _media->customInfoLayout()) : true; - if (mediaDisplayed) { - auto mediaAboveText = _media->isAboveMessage(); - auto mediaHeight = _media->height(); - auto mediaLeft = trect.x() - st::msgPadding.left(); - auto mediaTop = mediaAboveText ? trect.y() : (trect.y() + trect.height() - mediaHeight); - - if (point.y() >= mediaTop && point.y() < mediaTop + mediaHeight) { - result = _media->getState(point - QPoint(mediaLeft, mediaTop), request); - result.symbol += _text.length(); - } else { - if (mediaAboveText) { - trect.setY(trect.y() + mediaHeight); - } - if (trect.contains(point)) { - getStateText(point, trect, &result, request); - } - } - } else if (trect.contains(point)) { - getStateText(point, trect, &result, request); - } - if (needDateCheck) { - if (HistoryMessage::pointInTime(g.left() + g.width(), g.top() + g.height(), point, InfoDisplayDefault)) { - result.cursor = HistoryInDateCursorState; - } - } - if (displayRightAction()) { - const auto fastShareSkip = snap( - (g.height() - st::historyFastShareSize) / 2, - 0, - st::historyFastShareBottom); - const auto fastShareLeft = g.left() + g.width() + st::historyFastShareLeft; - const auto fastShareTop = g.top() + g.height() - fastShareSkip - st::historyFastShareSize; - if (QRect( - fastShareLeft, - fastShareTop, - st::historyFastShareSize, - st::historyFastShareSize - ).contains(point)) { - result.link = rightActionLink(); - } - } - } else if (_media && _media->isDisplayed()) { - result = _media->getState(point - g.topLeft(), request); - result.symbol += _text.length(); - } - - if (keyboard && !isLogEntry()) { - auto keyboardTop = g.top() + g.height() + st::msgBotKbButton.margin; - if (QRect(g.left(), keyboardTop, g.width(), keyboardHeight).contains(point)) { - result.link = keyboard->getState(point - QPoint(g.left(), keyboardTop)); - return result; - } - } - - return result; -} - ClickHandlerPtr HistoryMessage::rightActionLink() const { if (!_rightActionLink) { const auto itemId = fullId(); @@ -2305,185 +1436,6 @@ ClickHandlerPtr HistoryMessage::fastReplyLink() const { return _fastReplyLink; } -// Forward to _media. -void HistoryMessage::updatePressed(QPoint point) { - if (!_media) return; - - auto g = countGeometry(); - auto keyboard = inlineReplyKeyboard(); - if (keyboard) { - auto keyboardHeight = st::msgBotKbButton.margin + keyboard->naturalHeight(); - g.setHeight(g.height() - keyboardHeight); - } - - if (drawBubble()) { - auto mediaDisplayed = _media && _media->isDisplayed(); - auto top = marginTop(); - auto trect = g.marginsAdded(-st::msgPadding); - if (mediaDisplayed && _media->isBubbleTop()) { - trect.setY(trect.y() - st::msgPadding.top()); - } else { - if (displayFromName()) trect.setTop(trect.top() + st::msgNameFont->height); - if (displayForwardedFrom()) { - auto forwarded = Get(); - auto fwdheight = ((forwarded->text.maxWidth() > trect.width()) ? 2 : 1) * st::semiboldFont->height; - trect.setTop(trect.top() + fwdheight); - } - if (Get()) { - auto h = st::msgReplyPadding.top() + st::msgReplyBarSize.height() + st::msgReplyPadding.bottom(); - trect.setTop(trect.top() + h); - } - if (!displayFromName() && !Has()) { - if (auto via = Get()) { - trect.setTop(trect.top() + st::msgNameFont->height); - } - } - } - if (mediaDisplayed && _media->isBubbleBottom()) { - trect.setHeight(trect.height() + st::msgPadding.bottom()); - } - - auto needDateCheck = true; - if (mediaDisplayed) { - auto mediaAboveText = _media->isAboveMessage(); - auto mediaHeight = _media->height(); - auto mediaLeft = trect.x() - st::msgPadding.left(); - auto mediaTop = mediaAboveText ? trect.y() : (trect.y() + trect.height() - mediaHeight); - _media->updatePressed(point - QPoint(mediaLeft, mediaTop)); - } - } else { - _media->updatePressed(point - g.topLeft()); - } -} - -bool HistoryMessage::getStateFromName( - QPoint point, - QRect &trect, - not_null outResult) const { - if (displayFromName()) { - const auto replyWidth = [&] { - if (isUnderCursor() && displayFastReply()) { - return st::msgFont->width(FastReplyText()); - } - return 0; - }(); - if (replyWidth - && point.x() >= trect.left() + trect.width() - replyWidth - && point.x() < trect.left() + trect.width() + st::msgPadding.right() - && point.y() >= trect.top() - st::msgPadding.top() - && point.y() < trect.top() + st::msgServiceFont->height) { - outResult->link = fastReplyLink(); - return true; - } - if (point.y() >= trect.top() && point.y() < trect.top() + st::msgNameFont->height) { - auto availableLeft = trect.left(); - auto availableWidth = trect.width(); - if (replyWidth) { - availableWidth -= st::msgPadding.right() + replyWidth; - } - auto user = displayFrom(); - if (point.x() >= availableLeft - && point.x() < availableLeft + availableWidth - && point.x() < availableLeft + user->nameText.maxWidth()) { - outResult->link = user->openLink(); - return true; - } - auto forwarded = Get(); - auto via = Get(); - if (via - && !forwarded - && point.x() >= availableLeft + author()->nameText.maxWidth() + st::msgServiceFont->spacew - && point.x() < availableLeft + availableWidth - && point.x() < availableLeft + user->nameText.maxWidth() + st::msgServiceFont->spacew + via->width) { - outResult->link = via->link; - return true; - } - } - trect.setTop(trect.top() + st::msgNameFont->height); - } - return false; -} - -bool HistoryMessage::getStateForwardedInfo( - QPoint point, - QRect &trect, - not_null outResult, - HistoryStateRequest request) const { - if (displayForwardedFrom()) { - auto forwarded = Get(); - auto fwdheight = ((forwarded->text.maxWidth() > trect.width()) ? 2 : 1) * st::semiboldFont->height; - if (point.y() >= trect.top() && point.y() < trect.top() + fwdheight) { - auto breakEverywhere = (forwarded->text.countHeight(trect.width()) > 2 * st::semiboldFont->height); - auto textRequest = request.forText(); - if (breakEverywhere) { - textRequest.flags |= Text::StateRequest::Flag::BreakEverywhere; - } - *outResult = HistoryTextState(this, forwarded->text.getState( - point - trect.topLeft(), - trect.width(), - textRequest)); - outResult->symbol = 0; - outResult->afterSymbol = false; - if (breakEverywhere) { - outResult->cursor = HistoryInForwardedCursorState; - } else { - outResult->cursor = HistoryDefaultCursorState; - } - return true; - } - trect.setTop(trect.top() + fwdheight); - } - return false; -} - -bool HistoryMessage::getStateReplyInfo( - QPoint point, - QRect &trect, - not_null outResult) const { - if (auto reply = Get()) { - int32 h = st::msgReplyPadding.top() + st::msgReplyBarSize.height() + st::msgReplyPadding.bottom(); - if (point.y() >= trect.top() && point.y() < trect.top() + h) { - if (reply->replyToMsg && QRect(trect.x(), trect.y() + st::msgReplyPadding.top(), trect.width(), st::msgReplyBarSize.height()).contains(point)) { - outResult->link = reply->replyToLink(); - } - return true; - } - trect.setTop(trect.top() + h); - } - return false; -} - -bool HistoryMessage::getStateViaBotIdInfo( - QPoint point, - QRect &trect, - not_null outResult) const { - if (!displayFromName() && !Has()) { - if (auto via = Get()) { - if (QRect(trect.x(), trect.y(), via->width, st::msgNameFont->height).contains(point)) { - outResult->link = via->link; - return true; - } - trect.setTop(trect.top() + st::msgNameFont->height); - } - } - return false; -} - -bool HistoryMessage::getStateText( - QPoint point, - QRect &trect, - not_null outResult, - HistoryStateRequest request) const { - if (trect.contains(point)) { - *outResult = HistoryTextState(this, _text.getState( - point - trect.topLeft(), - trect.width(), - request.forText())); - return true; - } - return false; -} - TextSelection HistoryMessage::adjustSelection(TextSelection selection, TextSelectType type) const { auto result = _text.adjustSelection(selection, type); auto beforeMediaLength = _text.length(); @@ -2523,21 +1475,6 @@ QString HistoryMessage::notificationHeader() const { return (!_history->peer->isUser() && !isPost()) ? from()->name : QString(); } -bool HistoryMessage::displayFromPhoto() const { - return hasFromPhoto() && !isAttachedToNext(); -} - -bool HistoryMessage::hasFromPhoto() const { - if (isPost() || isEmpty()) { - return false; - } else if (Adaptive::ChatWide()) { - return true; - } else if (history()->peer->isSelf()) { - return Has(); - } - return !out() && !history()->peer->isUser(); -} - std::unique_ptr HistoryMessage::createView( not_null controller, HistoryView::Context context) { diff --git a/Telegram/SourceFiles/history/history_message.h b/Telegram/SourceFiles/history/history_message.h index 3728cff7a..f79c895b2 100644 --- a/Telegram/SourceFiles/history/history_message.h +++ b/Telegram/SourceFiles/history/history_message.h @@ -9,6 +9,10 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "history/history_item.h" +namespace HistoryView { +class Message; +} // namespace HistoryView + struct HistoryMessageEdited; base::lambda HistoryDependentItemCallback( @@ -141,21 +145,13 @@ public: void initTime(); void initMedia(const MTPMessageMedia *media); void initMediaFromDocument(DocumentData *doc, const QString &caption); - void fromNameUpdated(int32 width) const; int32 plainMaxWidth() const; - QRect countGeometry() const; bool drawBubble() const; bool hasBubble() const override { return drawBubble(); } - bool hasFromName() const; - bool displayFromName() const { - if (!hasFromName()) return false; - if (isAttachedToPrevious()) return false; - return true; - } bool hasFastReply() const; bool displayFastReply() const; bool displayForwardedFrom() const; @@ -168,18 +164,13 @@ public: void drawInfo(Painter &p, int32 right, int32 bottom, int32 width, bool selected, InfoDisplayType type) const override; void drawRightAction(Painter &p, int left, int top, int outerWidth) const override; void setViewsCount(int32 count) override; - void setId(MsgId newId) override; - void draw(Painter &p, QRect clip, TextSelection selection, TimeMs ms) const override; + void setRealId(MsgId newId) override; ClickHandlerPtr rightActionLink() const override; void dependencyItemRemoved(HistoryItem *dependency) override; - bool hasPoint(QPoint point) const override; bool pointInTime(int right, int bottom, QPoint point, InfoDisplayType type) const override; - HistoryTextState getState(QPoint point, HistoryStateRequest request) const override; - void updatePressed(QPoint point) override; - TextSelection adjustSelection(TextSelection selection, TextSelectType type) const override; QString notificationHeader() const override; @@ -223,11 +214,6 @@ public: return this; } - // hasFromPhoto() returns true even if we don't display the photo - // but we need to skip a place at the left side for this photo - bool displayFromPhoto() const; - bool hasFromPhoto() const; - std::unique_ptr createView( not_null controller, HistoryView::Context context) override; @@ -300,49 +286,19 @@ private: friend class HistoryItemInstantiated; void setEmptyText(); + bool hasAdminBadge() const { + return _flags & MTPDmessage_ClientFlag::f_has_admin_badge; + } // For an invoice button we replace the button text with a "Receipt" key. // It should show the receipt for the payed invoice. Still let mobile apps do that. void replaceBuyWithReceiptInMarkup(); - void initDimensions() override; - int resizeContentGetHeight() override; - int performResizeGetHeight(); void applyEditionToEmpty(); QDateTime displayedEditDate(bool hasViaBotOrInlineMarkup) const; const HistoryMessageEdited *displayedEditBadge() const; HistoryMessageEdited *displayedEditBadge(); - void paintFromName(Painter &p, QRect &trect, bool selected) const; - void paintForwardedInfo(Painter &p, QRect &trect, bool selected) const; - void paintReplyInfo(Painter &p, QRect &trect, bool selected) const; - // this method draws "via @bot" if it is not painted in forwarded info or in from name - void paintViaBotIdInfo(Painter &p, QRect &trect, bool selected) const; - void paintText(Painter &p, QRect &trect, TextSelection selection) const; - - bool getStateFromName( - QPoint point, - QRect &trect, - not_null outResult) const; - bool getStateForwardedInfo( - QPoint point, - QRect &trect, - not_null outResult, - HistoryStateRequest request) const; - bool getStateReplyInfo( - QPoint point, - QRect &trect, - not_null outResult) const; - bool getStateViaBotIdInfo( - QPoint point, - QRect &trect, - not_null outResult) const; - bool getStateText( - QPoint point, - QRect &trect, - not_null outResult, - HistoryStateRequest request) const; - void setMedia(const MTPMessageMedia *media); void setReplyMarkup(const MTPReplyMarkup *markup); @@ -353,7 +309,6 @@ private: void createComponentsHelper(MTPDmessage::Flags flags, MsgId replyTo, UserId viaBotId, const QString &postAuthor, const MTPReplyMarkup &markup); void createComponents(const CreateConfig &config); - void updateMediaInBubbleState(); void updateAdminBadgeState(); ClickHandlerPtr fastReplyLink() const; @@ -364,4 +319,7 @@ private: mutable ClickHandlerPtr _fastReplyLink; mutable int32 _fromNameVersion = 0; + friend class HistoryView::Element; + friend class HistoryView::Message; + }; diff --git a/Telegram/SourceFiles/history/history_service.cpp b/Telegram/SourceFiles/history/history_service.cpp index 8907a6ba7..a02e89a09 100644 --- a/Telegram/SourceFiles/history/history_service.cpp +++ b/Telegram/SourceFiles/history/history_service.cpp @@ -9,14 +9,16 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "lang/lang_keys.h" #include "mainwidget.h" +#include "auth_session.h" #include "apiwrap.h" #include "layout.h" +#include "history/history.h" #include "history/history_media_types.h" #include "history/history_message.h" #include "history/history_item_components.h" #include "history/view/history_view_service_message.h" #include "data/data_feed.h" -#include "auth_session.h" +#include "data/data_session.h" #include "window/notifications_manager.h" #include "window/window_controller.h" #include "storage/storage_shared_media.h" @@ -442,14 +444,6 @@ HistoryService::HistoryService(not_null history, MsgId msgId, QDateTim } } -void HistoryService::initDimensions() { - _maxw = _text.maxWidth() + st::msgServicePadding.left() + st::msgServicePadding.right(); - _minh = _text.minHeight(); - if (_media) { - _media->initDimensions(); - } -} - bool HistoryService::updateDependencyItem() { if (GetDependentData()) { return updateDependent(true); @@ -457,14 +451,6 @@ bool HistoryService::updateDependencyItem() { return HistoryItem::updateDependencyItem(); } -QRect HistoryService::countGeometry() const { - auto result = QRect(0, 0, width(), _height); - if (Adaptive::ChatWide()) { - result.setWidth(qMin(result.width(), st::msgMaxWidth + 2 * st::msgPhotoSkip + 2 * st::msgMargin.left())); - } - return result.marginsRemoved(st::msgServiceMargin); -} - TextWithEntities HistoryService::selectedText(TextSelection selection) const { return _text.originalTextWithEntities((selection == FullSelection) ? AllTextSelection : selection); } @@ -494,77 +480,10 @@ void HistoryService::setServiceText(const PreparedText &prepared) { // Link indices start with 1. _text.setLink(++linkIndex, link); } - - setPendingInitDimensions(); _textWidth = -1; _textHeight = 0; } -void HistoryService::draw(Painter &p, QRect clip, TextSelection selection, TimeMs ms) const { - auto height = _height - st::msgServiceMargin.top() - st::msgServiceMargin.bottom(); - auto dateh = 0; - auto unreadbarh = 0; - if (auto date = Get()) { - dateh = date->height(); - p.translate(0, dateh); - clip.translate(0, -dateh); - height -= dateh; - } - if (auto unreadbar = Get()) { - unreadbarh = unreadbar->height(); - if (clip.intersects(QRect(0, 0, width(), unreadbarh))) { - unreadbar->paint(p, 0, width()); - } - p.translate(0, unreadbarh); - clip.translate(0, -unreadbarh); - height -= unreadbarh; - } - - HistoryView::PaintContext context(ms, clip, selection); - HistoryView::ServiceMessagePainter::paint(p, this, context, height); - - if (auto skiph = dateh + unreadbarh) { - p.translate(0, -skiph); - } -} - -int HistoryService::resizeContentGetHeight() { - _height = displayedDateHeight(); - if (auto unreadbar = Get()) { - _height += unreadbar->height(); - } - - if (_text.isEmpty()) { - _textHeight = 0; - } else { - auto contentWidth = width(); - if (Adaptive::ChatWide()) { - accumulate_min(contentWidth, st::msgMaxWidth + 2 * st::msgPhotoSkip + 2 * st::msgMargin.left()); - } - contentWidth -= st::msgServiceMargin.left() + st::msgServiceMargin.left(); // two small margins - if (contentWidth < st::msgServicePadding.left() + st::msgServicePadding.right() + 1) { - contentWidth = st::msgServicePadding.left() + st::msgServicePadding.right() + 1; - } - - auto nwidth = qMax(contentWidth - st::msgServicePadding.left() - st::msgServicePadding.right(), 0); - if (nwidth != _textWidth) { - _textWidth = nwidth; - _textHeight = _text.countHeight(nwidth); - } - if (contentWidth >= _maxw) { - _height += _minh; - } else { - _height += _textHeight; - } - _height += st::msgServicePadding.top() + st::msgServicePadding.bottom() + st::msgServiceMargin.top() + st::msgServiceMargin.bottom(); - if (_media) { - _height += st::msgServiceMargin.top() + _media->resizeGetHeight(_media->width()); - } - } - - return _height; -} - void HistoryService::markMediaAsReadHook() { if (auto selfdestruct = Get()) { if (!selfdestruct->destructAt) { @@ -594,68 +513,6 @@ TimeMs HistoryService::getSelfDestructIn(TimeMs now) { return 0; } -bool HistoryService::hasPoint(QPoint point) const { - auto g = countGeometry(); - if (g.width() < 1) { - return false; - } - - if (auto dateh = displayedDateHeight()) { - g.setTop(g.top() + dateh); - } - if (auto unreadbar = Get()) { - g.setTop(g.top() + unreadbar->height()); - } - if (_media) { - g.setHeight(g.height() - (st::msgServiceMargin.top() + _media->height())); - } - return g.contains(point); -} - -HistoryTextState HistoryService::getState(QPoint point, HistoryStateRequest request) const { - auto result = HistoryTextState(this); - - auto g = countGeometry(); - if (g.width() < 1) { - return result; - } - - if (auto dateh = displayedDateHeight()) { - point.setY(point.y() - dateh); - g.setHeight(g.height() - dateh); - } - if (auto unreadbar = Get()) { - auto unreadbarh = unreadbar->height(); - point.setY(point.y() - unreadbarh); - g.setHeight(g.height() - unreadbarh); - } - - if (_media) { - g.setHeight(g.height() - (st::msgServiceMargin.top() + _media->height())); - } - auto trect = g.marginsAdded(-st::msgServicePadding); - if (trect.contains(point)) { - auto textRequest = request.forText(); - textRequest.align = style::al_center; - result = HistoryTextState(this, _text.getState( - point - trect.topLeft(), - trect.width(), - textRequest)); - if (auto gamescore = Get()) { - if (!result.link && result.cursor == HistoryInTextCursorState && g.contains(point)) { - result.link = gamescore->lnk; - } - } else if (auto payment = Get()) { - if (!result.link && result.cursor == HistoryInTextCursorState && g.contains(point)) { - result.link = payment->lnk; - } - } - } else if (_media) { - result = _media->getState(point - QPoint(st::msgServiceMargin.left() + (g.width() - _media->maxWidth()) / 2, st::msgServiceMargin.top() + g.height() + st::msgServiceMargin.top()), request); - } - return result; -} - void HistoryService::createFromMtp(const MTPDmessage &message) { auto mediaType = message.vmedia.type(); switch (mediaType) { @@ -770,6 +627,7 @@ void HistoryService::updateDependentText() { } setServiceText(text); + Auth().data().requestItemViewResize(this); if (history()->textCachedFor == this) { history()->textCachedFor = nullptr; } diff --git a/Telegram/SourceFiles/history/history_service.h b/Telegram/SourceFiles/history/history_service.h index e20ea6faa..892398511 100644 --- a/Telegram/SourceFiles/history/history_service.h +++ b/Telegram/SourceFiles/history/history_service.h @@ -9,6 +9,10 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "history/history_item.h" +namespace HistoryView { +class Service; +} // namespace HistoryView + struct HistoryServiceDependentData { MsgId msgId = 0; HistoryItem *msg = nullptr; @@ -78,12 +82,6 @@ public: return true; } - QRect countGeometry() const; - - void draw(Painter &p, QRect clip, TextSelection selection, TimeMs ms) const override; - bool hasPoint(QPoint point) const override; - HistoryTextState getState(QPoint point, HistoryStateRequest request) const override; - [[nodiscard]] TextSelection adjustSelection( TextSelection selection, TextSelectType type) const override { @@ -119,9 +117,6 @@ protected: HistoryService(not_null history, MsgId msgId, QDateTime date, const PreparedText &message, MTPDmessage::Flags flags = 0, UserId from = 0, PhotoData *photo = 0); friend class HistoryItemInstantiated; - void initDimensions() override; - int resizeContentGetHeight() override; - void markMediaAsReadHook() override; void setServiceText(const PreparedText &prepared); @@ -162,6 +157,8 @@ private: PreparedText prepareGameScoreText(); PreparedText preparePaymentSentText(); + friend class HistoryView::Service; + }; class HistoryJoined : public HistoryService, private HistoryItemInstantiated { diff --git a/Telegram/SourceFiles/history/history_widget.cpp b/Telegram/SourceFiles/history/history_widget.cpp index 6bbc4f48a..7853fa96c 100644 --- a/Telegram/SourceFiles/history/history_widget.cpp +++ b/Telegram/SourceFiles/history/history_widget.cpp @@ -23,6 +23,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "inline_bots/inline_bot_result.h" #include "data/data_drafts.h" #include "data/data_session.h" +#include "history/history.h" +#include "history/history_item.h" #include "history/history_message.h" #include "history/history_media_types.h" #include "history/history_drag_area.h" @@ -628,14 +630,18 @@ HistoryWidget::HistoryWidget(QWidget *parent, not_null cont })); subscribe(Auth().data().pendingHistoryResize(), [this] { handlePendingHistoryUpdate(); }); subscribe(Auth().data().queryItemVisibility(), [this](const Data::Session::ItemVisibilityQuery &query) { - if (_a_show.animating() || _history != query.item->history() || !query.item->mainView() || !isVisible()) { + if (_a_show.animating() + || _history != query.item->history() + || !query.item->mainView() || !isVisible()) { return; } - auto top = _list->itemTop(query.item); - if (top >= 0) { - auto scrollTop = _scroll->scrollTop(); - if (top + query.item->height() > scrollTop && top < scrollTop + _scroll->height()) { - *query.isVisible = true; + if (const auto view = query.item->mainView()) { + auto top = _list->itemTop(view); + if (top >= 0) { + auto scrollTop = _scroll->scrollTop(); + if (top + view->height() > scrollTop && top < scrollTop + _scroll->height()) { + *query.isVisible = true; + } } } }); @@ -684,13 +690,15 @@ void HistoryWidget::scrollToCurrentVoiceMessage(FullMsgId fromId, FullMsgId toId // And the scrollTop will be reset back to scrollTopItem + scrollTopOffset. handlePendingHistoryUpdate(); - auto toTop = _list->itemTop(to); - if (toTop >= 0 && !isItemCompletelyHidden(from)) { - auto scrollTop = _scroll->scrollTop(); - auto scrollBottom = scrollTop + _scroll->height(); - auto toBottom = toTop + to->height(); - if ((toTop < scrollTop && toBottom < scrollBottom) || (toTop > scrollTop && toBottom > scrollBottom)) { - animatedScrollToItem(to->id); + if (const auto toView = to->mainView()) { + auto toTop = _list->itemTop(toView); + if (toTop >= 0 && !isItemCompletelyHidden(from)) { + auto scrollTop = _scroll->scrollTop(); + auto scrollBottom = scrollTop + _scroll->height(); + auto toBottom = toTop + toView->height(); + if ((toTop < scrollTop && toBottom < scrollBottom) || (toTop > scrollTop && toBottom > scrollBottom)) { + animatedScrollToItem(to->id); + } } } } @@ -867,13 +875,16 @@ void HistoryWidget::clearHighlightMessages() { } int HistoryWidget::itemTopForHighlight(not_null item) const { + const auto view = item->mainView(); + Assert(view != nullptr); + if (const auto group = item->getFullGroup()) { item = group->leader; } - auto itemTop = _list->itemTop(item); + auto itemTop = _list->itemTop(view); Assert(itemTop >= 0); - auto heightLeft = (_scroll->height() - item->height()); + auto heightLeft = (_scroll->height() - view->height()); if (heightLeft <= 0) { return itemTop; } @@ -1367,7 +1378,7 @@ void HistoryWidget::notify_migrateUpdated(PeerData *peer) { if (_peer == peer) { if (peer->migrateTo()) { showHistory(peer->migrateTo()->id, (_showAtMsgId > 0) ? (-_showAtMsgId) : _showAtMsgId, true); - } else if ((_migrated ? _migrated->peer : 0) != peer->migrateFrom()) { + } else if ((_migrated ? _migrated->peer.get() : nullptr) != peer->migrateFrom()) { auto migrated = _history->migrateFrom(); if (_migrated || (migrated && migrated->unreadCount() > 0)) { showHistory(peer->id, peer->migrateFrom() ? _showAtMsgId : ((_showAtMsgId < 0 && -_showAtMsgId < ServerMaxMsgId) ? ShowAtUnreadMsgId : _showAtMsgId), true); @@ -2537,7 +2548,7 @@ void HistoryWidget::loadMessages() { MTP_int(maxId), MTP_int(minId), MTP_int(historyHash)), - rpcDone(&HistoryWidget::messagesReceived, from->peer), + rpcDone(&HistoryWidget::messagesReceived, from->peer.get()), rpcFail(&HistoryWidget::messagesFailed)); } @@ -2581,7 +2592,7 @@ void HistoryWidget::loadMessagesDown() { MTP_int(maxId), MTP_int(minId), MTP_int(historyHash)), - rpcDone(&HistoryWidget::messagesReceived, from->peer), + rpcDone(&HistoryWidget::messagesReceived, from->peer.get()), rpcFail(&HistoryWidget::messagesFailed)); } @@ -2647,12 +2658,16 @@ void HistoryWidget::onScroll() { } bool HistoryWidget::isItemCompletelyHidden(HistoryItem *item) const { + const auto view = item ? item->mainView() : nullptr; + if (!view) { + return true; + } auto top = _list ? _list->itemTop(item) : -2; if (top < 0) { return true; } - auto bottom = top + item->height(); + auto bottom = top + view->height(); auto scrollTop = _scroll->scrollTop(); auto scrollBottom = scrollTop + _scroll->height(); return (top >= scrollBottom || bottom <= scrollTop); @@ -3414,7 +3429,11 @@ bool HistoryWidget::insertBotCommand(const QString &cmd) { auto insertingInlineBot = !cmd.isEmpty() && (cmd.at(0) == '@'); auto toInsert = cmd; if (!toInsert.isEmpty() && !insertingInlineBot) { - auto bot = _peer->isUser() ? _peer : (App::hoveredLinkItem() ? App::hoveredLinkItem()->data()->fromOriginal() : nullptr); + auto bot = _peer->isUser() + ? _peer + : (App::hoveredLinkItem() + ? App::hoveredLinkItem()->data()->fromOriginal().get() + : nullptr); if (bot && (!bot->isUser() || !bot->asUser()->botInfo)) { bot = nullptr; } @@ -3940,7 +3959,7 @@ void HistoryWidget::updateFieldPlaceholder() { auto text = _inlineBot->botInfo->inlinePlaceholder.mid(1); _field->setPlaceholder([text] { return text; }, _inlineBot->username.size() + 2); } else { - const auto peer = _history ? _history->peer : nullptr; + const auto peer = _history ? _history->peer.get() : nullptr; _field->setPlaceholder(langFactory( (peer && peer->isChannel() && !peer->isMegagroup()) ? (peer->notifySilentPosts() @@ -4790,6 +4809,11 @@ void HistoryWidget::updateListSize() { _updateHistoryGeometryRequired = true; } +bool HistoryWidget::hasPendingResizedItems() const { + return (_history && _history->hasPendingResizedItems()) + || (_migrated && _migrated->hasPendingResizedItems()); +} + int HistoryWidget::unreadBarTop() const { auto getUnreadBar = [this]() -> HistoryItem* { if (_migrated && _migrated->unreadBar) { diff --git a/Telegram/SourceFiles/history/history_widget.h b/Telegram/SourceFiles/history/history_widget.h index eb74e7862..77dba35ae 100644 --- a/Telegram/SourceFiles/history/history_widget.h +++ b/Telegram/SourceFiles/history/history_widget.h @@ -31,6 +31,10 @@ class Widget; class Result; } // namespace InlineBots +namespace Data { +struct Draft; +} // namespace Data + namespace Ui { class AbstractButton; class InnerDropdown; @@ -664,9 +668,7 @@ private: void updateListSize(); // Does any of the shown histories has this flag set. - bool hasPendingResizedItems() const { - return (_history && _history->hasPendingResizedItems()) || (_migrated && _migrated->hasPendingResizedItems()); - } + bool hasPendingResizedItems() const; // Counts scrollTop for placing the scroll right at the unread // messages bar, choosing from _history and _migrated unreadBar. diff --git a/Telegram/SourceFiles/history/view/history_view_element.cpp b/Telegram/SourceFiles/history/view/history_view_element.cpp index 36ae4f6a6..f22d3b2b9 100644 --- a/Telegram/SourceFiles/history/view/history_view_element.cpp +++ b/Telegram/SourceFiles/history/view/history_view_element.cpp @@ -9,11 +9,17 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "history/history_item_components.h" #include "history/history_media.h" +#include "history/history.h" #include "data/data_session.h" #include "auth_session.h" namespace HistoryView { +namespace { +// A new message from the same sender is attached to previous within 15 minutes. +constexpr int kAttachMessageToPreviousSecondsDelta = 900; + +} // namespace Element::Element(not_null data, Context context) : _data(data) , _context(context) { @@ -23,6 +29,181 @@ not_null Element::data() const { return _data; } +Context Element::context() const { + return _context; +} + +int Element::y() const { + return _y; +} + +void Element::setY(int y) { + _y = y; +} + +int Element::marginTop() const { + const auto item = data(); + auto result = 0; + if (!item->isHiddenByGroup()) { + if (isAttachedToPrevious()) { + result += st::msgMarginTopAttached; + } else { + result += st::msgMargin.top(); + } + } + result += item->displayedDateHeight(); + if (const auto unreadbar = item->Get()) { + result += unreadbar->height(); + } + return result; +} + +int Element::marginBottom() const { + const auto item = data(); + return item->isHiddenByGroup() ? 0 : st::msgMargin.bottom(); +} + +void Element::setPendingResize() { + _flags |= Flag::NeedsResize; + if (_context == Context::History) { + data()->_history->setHasPendingResizedItems(); + } +} + +bool Element::pendingResize() const { + return _flags & Flag::NeedsResize; +} + +bool Element::isAttachedToPrevious() const { + return _flags & Flag::AttachedToPrevious; +} + +bool Element::isAttachedToNext() const { + return _flags & Flag::AttachedToNext; +} + +void Element::previousInBlocksChanged() { + recountDisplayDateInBlocks(); + recountAttachToPreviousInBlocks(); +} + +// Called only if there is no more next item! Not always when it changes! +void Element::nextInBlocksChanged() { + setAttachToNext(false); +} + +bool Element::computeIsAttachToPrevious(not_null previous) { + const auto item = data(); + if (!item->Has() && !item->Has()) { + const auto prev = previous->data(); + const auto possible = !item->isPost() && !prev->isPost() + && !item->serviceMsg() && !prev->serviceMsg() + && !item->isEmpty() && !prev->isEmpty() + && (qAbs(prev->date.secsTo(item->date)) < kAttachMessageToPreviousSecondsDelta); + if (possible) { + if (item->history()->peer->isSelf()) { + return prev->senderOriginal() == item->senderOriginal() + && (prev->Has() == item->Has()); + } else { + return prev->from() == item->from(); + } + } + } + return false; +} + +void Element::recountAttachToPreviousInBlocks() { + auto attachToPrevious = false; + if (const auto previous = previousInBlocks()) { + attachToPrevious = computeIsAttachToPrevious(previous); + previous->setAttachToNext(attachToPrevious); + } + setAttachToPrevious(attachToPrevious); +} + +void Element::recountDisplayDateInBlocks() { + setDisplayDate([&] { + const auto item = data(); + if (item->isEmpty()) { + return false; + } + + if (auto previous = previousInBlocks()) { + const auto prev = previous->data(); + return prev->isEmpty() || (prev->date.date() != item->date.date()); + } + return true; + }()); +} + +QSize Element::countOptimalSize() { + return performCountOptimalSize(); +} + +QSize Element::countCurrentSize(int newWidth) { + if (_flags & Flag::NeedsResize) { + _flags &= ~Flag::NeedsResize; + initDimensions(); + } + return performCountCurrentSize(newWidth); +} + +void Element::setDisplayDate(bool displayDate) { + const auto item = data(); + if (displayDate && !item->Has()) { + item->AddComponents(HistoryMessageDate::Bit()); + item->Get()->init(item->date); + setPendingResize(); + } else if (!displayDate && item->Has()) { + item->RemoveComponents(HistoryMessageDate::Bit()); + setPendingResize(); + } +} + +void Element::setAttachToNext(bool attachToNext) { + if (attachToNext && !(_flags & Flag::AttachedToNext)) { + _flags |= Flag::AttachedToNext; + Auth().data().requestItemRepaint(data()); + } else if (!attachToNext && (_flags & Flag::AttachedToNext)) { + _flags &= ~Flag::AttachedToNext; + Auth().data().requestItemRepaint(data()); + } +} + +void Element::setAttachToPrevious(bool attachToPrevious) { + if (attachToPrevious && !(_flags & Flag::AttachedToPrevious)) { + _flags |= Flag::AttachedToPrevious; + setPendingResize(); + } else if (!attachToPrevious && (_flags & Flag::AttachedToPrevious)) { + _flags &= ~Flag::AttachedToPrevious; + setPendingResize(); + } +} + +bool Element::displayFromPhoto() const { + return false; +} + +bool Element::hasFromPhoto() const { + return false; +} + +bool Element::hasFromName() const { + return false; +} + +bool Element::displayFromName() const { + return false; +} + +HistoryBlock *Element::block() { + return _block; +} + +const HistoryBlock *Element::block() const { + return _block; +} + void Element::attachToBlock(not_null block, int index) { Expects(!_data->isLogEntry()); Expects(_block == nullptr); @@ -32,7 +213,6 @@ void Element::attachToBlock(not_null block, int index) { _block = block; _indexInBlock = index; _data->setMainView(this); - _data->setPendingResize(); } void Element::removeFromBlock() { @@ -41,6 +221,20 @@ void Element::removeFromBlock() { _block->remove(this); } +void Element::setIndexInBlock(int index) { + Expects(_block != nullptr); + Expects(index >= 0); + + _indexInBlock = index; +} + +int Element::indexInBlock() const { + Expects((_indexInBlock >= 0) == (_block != nullptr)); + Expects((_block == nullptr) || (_block->messages[_indexInBlock].get() == this)); + + return _indexInBlock; +} + Element *Element::previousInBlocks() const { if (_block && _indexInBlock >= 0) { if (_indexInBlock > 0) { diff --git a/Telegram/SourceFiles/history/view/history_view_element.h b/Telegram/SourceFiles/history/view/history_view_element.h index 927818b9e..7bdaca116 100644 --- a/Telegram/SourceFiles/history/view/history_view_element.h +++ b/Telegram/SourceFiles/history/view/history_view_element.h @@ -7,9 +7,14 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL */ #pragma once +#include "history/view/history_view_object.h" #include "base/runtime_composer.h" +#include "base/flags.h" +class HistoryBlock; class HistoryItem; +struct HistoryTextState; +struct HistoryStateRequest; namespace HistoryView { @@ -20,44 +25,58 @@ enum class Context : char { }; class Element - : public RuntimeComposer + : public Object +// , public RuntimeComposer , public ClickHandlerHost { public: Element(not_null data, Context context); + enum class Flag : uchar { + NeedsResize = 0x01, + AttachedToPrevious = 0x02, + AttachedToNext = 0x04, + }; + using Flags = base::flags; + friend inline constexpr auto is_flag_type(Flag) { return true; } + not_null data() const; + Context context() const; - int y() const { - return _y; - } - void setY(int y) { - _y = y; - } + int y() const; + void setY(int y); - HistoryBlock *block() { - return _block; - } - const HistoryBlock *block() const { - return _block; - } - void attachToBlock(not_null block, int index); - void removeFromBlock(); - void setIndexInBlock(int index) { - Expects(_block != nullptr); - Expects(index >= 0); + int marginTop() const; + int marginBottom() const; + void setPendingResize(); + bool pendingResize() const; - _indexInBlock = index; - } - int indexInBlock() const { - Expects((_indexInBlock >= 0) == (_block != nullptr)); - Expects((_block == nullptr) || (_block->messages[_indexInBlock].get() == this)); + bool isAttachedToPrevious() const; + bool isAttachedToNext() const; - return _indexInBlock; - } - Element *previousInBlocks() const; - Element *nextInBlocks() const; + // For blocks context this should be called only from recountAttachToPreviousInBlocks(). + void setAttachToPrevious(bool attachToNext); - // ClickHandlerHost interface + // For blocks context this should be called only from recountAttachToPreviousInBlocks() + // of the next item or when the next item is removed through nextInBlocksChanged() call. + void setAttachToNext(bool attachToNext); + + // For blocks context this should be called only from recountDisplayDate(). + void setDisplayDate(bool displayDate); + + bool computeIsAttachToPrevious(not_null previous); + + virtual void draw( + Painter &p, + QRect clip, + TextSelection selection, + TimeMs ms) const = 0; + [[nodiscard]] virtual bool hasPoint(QPoint point) const = 0; + [[nodiscard]] virtual HistoryTextState getState( + QPoint point, + HistoryStateRequest request) const = 0; + virtual void updatePressed(QPoint point) = 0; + + // ClickHandlerHost interface. void clickHandlerActiveChanged( const ClickHandlerPtr &handler, bool active) override; @@ -65,13 +84,54 @@ public: const ClickHandlerPtr &handler, bool pressed) override; + // hasFromPhoto() returns true even if we don't display the photo + // but we need to skip a place at the left side for this photo + virtual bool displayFromPhoto() const; + virtual bool hasFromPhoto() const; + virtual bool hasFromName() const; + virtual bool displayFromName() const; + + // Legacy blocks structure. + HistoryBlock *block(); + const HistoryBlock *block() const; + void attachToBlock(not_null block, int index); + void removeFromBlock(); + void setIndexInBlock(int index); + int indexInBlock() const; + Element *previousInBlocks() const; + Element *nextInBlocks() const; + void previousInBlocksChanged(); + void nextInBlocksChanged(); + virtual ~Element(); +protected: + void setInitialSize(int maxWidth, int minHeight); + void setCurrentSize(int width, int height); + private: + // This should be called only from previousInBlocksChanged() + // to add required bits to the Composer mask + // after that always use Has(). + void recountDisplayDateInBlocks(); + + // This should be called only from previousInBlocksChanged() or when + // HistoryMessageDate or HistoryMessageUnreadBar bit is changed in the Composer mask + // then the result should be cached in a client side flag MTPDmessage_ClientFlag::f_attach_to_previous. + void recountAttachToPreviousInBlocks(); + + QSize countOptimalSize() final override; + QSize countCurrentSize(int newWidth) final override; + + virtual QSize performCountOptimalSize() = 0; + virtual QSize performCountCurrentSize(int newWidth) = 0; + const not_null _data; int _y = 0; Context _context; + Flags _flags = Flag::NeedsResize; + HistoryBlock *_block = nullptr; int _indexInBlock = -1; diff --git a/Telegram/SourceFiles/history/view/history_view_list_widget.cpp b/Telegram/SourceFiles/history/view/history_view_list_widget.cpp index 7c71d9910..d8cc07b0d 100644 --- a/Telegram/SourceFiles/history/view/history_view_list_widget.cpp +++ b/Telegram/SourceFiles/history/view/history_view_list_widget.cpp @@ -61,21 +61,21 @@ void ListWidget::enumerateItems(Method method) { ending, _visibleTop, [this](auto &elem, int top) { - return this->itemTop(elem) + elem->data()->height() <= top; + return this->itemTop(elem) + elem->height() <= top; }) : std::upper_bound( beginning, ending, _visibleBottom, [this](int bottom, auto &elem) { - return this->itemTop(elem) + elem->data()->height() >= bottom; + return this->itemTop(elem) + elem->height() >= bottom; }); auto wasEnd = (from == ending); if (wasEnd) { --from; } if (TopToBottom) { - Assert(itemTop(from->get()) + from->get()->data()->height() > _visibleTop); + Assert(itemTop(from->get()) + from->get()->height() > _visibleTop); } else { Assert(itemTop(from->get()) < _visibleBottom); } @@ -83,7 +83,7 @@ void ListWidget::enumerateItems(Method method) { while (true) { auto view = from->get(); auto itemtop = itemTop(view); - auto itembottom = itemtop + view->data()->height(); + auto itembottom = itemtop + view->height(); // Binary search should've skipped all the items that are above / below the visible area. if (TopToBottom) { @@ -131,19 +131,19 @@ void ListWidget::enumerateUserpics(Method method) { auto message = view->data()->toHistoryMessage(); if (!message) return true; - if (lowestAttachedItemTop < 0 && message->isAttachedToNext()) { - lowestAttachedItemTop = itemtop + message->marginTop(); + if (lowestAttachedItemTop < 0 && view->isAttachedToNext()) { + lowestAttachedItemTop = itemtop + view->marginTop(); } // Call method on a userpic for all messages that have it and for those who are not showing it // because of their attachment to the next message if they are bottom-most visible. - if (message->displayFromPhoto() || (message->hasFromPhoto() && itembottom >= _visibleBottom)) { + if (view->displayFromPhoto() || (view->hasFromPhoto() && itembottom >= _visibleBottom)) { if (lowestAttachedItemTop < 0) { - lowestAttachedItemTop = itemtop + message->marginTop(); + lowestAttachedItemTop = itemtop + view->marginTop(); } // Attach userpic to the bottom of the visible area with the same margin as the last message. auto userpicMinBottomSkip = st::historyPaddingBottom + st::msgMargin.bottom(); - auto userpicBottom = qMin(itembottom - message->marginBottom(), _visibleBottom - userpicMinBottomSkip); + auto userpicBottom = qMin(itembottom - view->marginBottom(), _visibleBottom - userpicMinBottomSkip); // Do not let the userpic go above the attached messages pack top line. userpicBottom = qMax(userpicBottom, lowestAttachedItemTop + st::msgPhotoSize); @@ -156,7 +156,7 @@ void ListWidget::enumerateUserpics(Method method) { } // Forget the found top of the pack, search for the next one from scratch. - if (!message->isAttachedToNext()) { + if (!view->isAttachedToNext()) { lowestAttachedItemTop = -1; } @@ -175,14 +175,14 @@ void ListWidget::enumerateDates(Method method) { auto dateCallback = [&](not_null view, int itemtop, int itembottom) { const auto item = view->data(); if (lowestInOneDayItemBottom < 0 && item->isInOneDayWithPrevious()) { - lowestInOneDayItemBottom = itembottom - item->marginBottom(); + lowestInOneDayItemBottom = itembottom - view->marginBottom(); } // Call method on a date for all messages that have it and for those who are not showing it // because they are in a one day together with the previous message if they are top-most visible. if (item->displayDate() || (!item->isEmpty() && itemtop <= _visibleTop)) { if (lowestInOneDayItemBottom < 0) { - lowestInOneDayItemBottom = itembottom - item->marginBottom(); + lowestInOneDayItemBottom = itembottom - view->marginBottom(); } // Attach date to the top of the visible area with the same margin as it has in service message. auto dateTop = qMax(itemtop, _visibleTop) + st::msgServiceMargin.top(); @@ -231,7 +231,7 @@ ListWidget::ListWidget( if (const auto view = viewForItem(query.item)) { const auto top = itemTop(view); if (top >= 0 - && top + query.item->height() > _visibleTop + && top + view->height() > _visibleTop && top < _visibleBottom) { *query.isVisible = true; } @@ -517,16 +517,16 @@ void ListWidget::itemsAdded(Direction direction, int addedCount) { : (addedCount + 1); for (auto i = checkFrom; i != checkTo; ++i) { if (i > 0) { - const auto item = _items[i - 1]->data(); + const auto view = _items[i - 1].get(); if (i < _items.size()) { // #TODO feeds show - auto previous = _items[i]->data(); - item->setLogEntryDisplayDate(item->date.date() != previous->date.date()); - auto attachToPrevious = item->computeIsAttachToPrevious(previous); - item->setLogEntryAttachToPrevious(attachToPrevious); - previous->setLogEntryAttachToNext(attachToPrevious); + auto previous = _items[i].get(); + view->setDisplayDate(view->data()->date.date() != previous->data()->date.date()); + auto attachToPrevious = view->computeIsAttachToPrevious(previous); + view->setAttachToPrevious(attachToPrevious); + previous->setAttachToNext(attachToPrevious); } else { - item->setLogEntryDisplayDate(true); + view->setDisplayDate(true); } } } @@ -543,9 +543,9 @@ int ListWidget::resizeGetHeight(int newWidth) { update(); auto newHeight = 0; - for (auto &item : _items) { - item->setY(newHeight); - newHeight += item->data()->resizeGetHeight(newWidth); + for (auto &view : _items) { + view->setY(newHeight); + newHeight += view->resizeGetHeight(newWidth); } _itemsHeight = newHeight; _itemsTop = (_minHeight > _itemsHeight + st::historyPaddingBottom) ? (_minHeight - _itemsHeight - st::historyPaddingBottom) : 0; @@ -570,7 +570,7 @@ void ListWidget::paintEvent(QPaintEvent *e) { auto clip = e->rect(); auto from = std::lower_bound(begin(_items), end(_items), clip.top(), [this](auto &elem, int top) { - return this->itemTop(elem) + elem->data()->height() <= top; + return this->itemTop(elem) + elem->height() <= top; }); auto to = std::lower_bound(begin(_items), end(_items), clip.top() + clip.height(), [this](auto &elem, int bottom) { return this->itemTop(elem) < bottom; @@ -579,9 +579,12 @@ void ListWidget::paintEvent(QPaintEvent *e) { auto top = itemTop(from->get()); p.translate(0, top); for (auto i = from; i != to; ++i) { - auto selection = (*i == _selectedItem) ? _selectedText : TextSelection(); - (*i)->data()->draw(p, clip.translated(0, -top), selection, ms); - auto height = (*i)->data()->height(); + const auto view = *i; + const auto selection = (view == _selectedItem) + ? _selectedText + : TextSelection(); + view->draw(p, clip.translated(0, -top), selection, ms); + const auto height = view->height(); top += height; p.translate(0, height); } @@ -602,7 +605,7 @@ void ListWidget::paintEvent(QPaintEvent *e) { p, st::historyPhotoLeft, userpicTop, - message->width(), + view->width(), st::msgPhotoSize); } return true; @@ -636,7 +639,7 @@ void ListWidget::paintEvent(QPaintEvent *e) { if (opacity > 0.) { p.setOpacity(opacity); int dateY = /*noFloatingDate ? itemtop :*/ (dateTop - st::msgServiceMargin.top()); - int width = item->width(); + int width = view->width(); if (const auto date = item->Get()) { date->paint(p, dateY, width); } else { @@ -667,7 +670,7 @@ not_null ListWidget::findItemByY(int y) const { end(_items), y, [this](auto &elem, int top) { - return this->itemTop(elem) + elem->data()->height() <= top; + return this->itemTop(elem) + elem->height() <= top; }); return (i != end(_items)) ? i->get() : _items.back().get(); } @@ -711,7 +714,7 @@ void ListWidget::mouseDoubleClickEvent(QMouseEvent *e) { if (((_mouseAction == MouseAction::Selecting && _selectedItem != nullptr) || (_mouseAction == MouseAction::None)) && _mouseSelectType == TextSelectType::Letters && _mouseActionItem) { HistoryStateRequest request; request.flags |= Text::StateRequest::Flag::LookupSymbol; - auto dragState = _mouseActionItem->data()->getState(_dragStartPosition, request); + auto dragState = _mouseActionItem->getState(_dragStartPosition, request); if (dragState.cursor == HistoryInTextCursorState) { _mouseTextSymbol = dragState.symbol; _mouseSelectType = TextSelectType::Words; @@ -753,7 +756,7 @@ void ListWidget::showContextMenu(QContextMenuEvent *e, bool showFromTouch) { App::mousedItem()); HistoryStateRequest request; request.flags |= Text::StateRequest::Flag::LookupSymbol; - auto dragState = App::mousedItem()->data()->getState(mousePos, request); + auto dragState = App::mousedItem()->getState(mousePos, request); if (dragState.cursor == HistoryInTextCursorState && dragState.symbol >= selFrom && dragState.symbol < selTo) { isUponSelected = 1; } @@ -1045,7 +1048,7 @@ void ListWidget::mouseActionStart(const QPoint &screenPos, Qt::MouseButton butto if (_trippleClickTimer.isActive() && (screenPos - _trippleClickPoint).manhattanLength() < QApplication::startDragDistance()) { HistoryStateRequest request; request.flags = Text::StateRequest::Flag::LookupSymbol; - dragState = _mouseActionItem->data()->getState(_dragStartPosition, request); + dragState = _mouseActionItem->getState(_dragStartPosition, request); if (dragState.cursor == HistoryInTextCursorState) { auto selection = TextSelection { dragState.symbol, dragState.symbol }; repaintItem(std::exchange(_selectedItem, _mouseActionItem)); @@ -1059,7 +1062,7 @@ void ListWidget::mouseActionStart(const QPoint &screenPos, Qt::MouseButton butto } else if (App::pressedItem()) { HistoryStateRequest request; request.flags = Text::StateRequest::Flag::LookupSymbol; - dragState = _mouseActionItem->data()->getState(_dragStartPosition, request); + dragState = _mouseActionItem->getState(_dragStartPosition, request); } if (_mouseSelectType != TextSelectType::Paragraphs) { if (App::pressedItem()) { @@ -1157,7 +1160,7 @@ void ListWidget::updateSelected() { if (view) { App::mousedItem(view); itemPoint = mapPointToItem(point, view); - if (item->hasPoint(itemPoint)) { + if (view->hasPoint(itemPoint)) { if (App::hoveredItem() != view) { repaintItem(App::hoveredItem()); App::hoveredItem(view); @@ -1187,11 +1190,11 @@ void ListWidget::updateSelected() { } else { selectingText = false; } - dragState = item->getState(itemPoint, request); + dragState = view->getState(itemPoint, request); lnkhost = view; if (!dragState.link && itemPoint.x() >= st::historyPhotoLeft && itemPoint.x() < st::historyPhotoLeft + st::msgPhotoSize) { if (auto message = item->toHistoryMessage()) { - if (message->hasFromPhoto()) { + if (view->hasFromPhoto()) { enumerateUserpics([&](not_null view, int userpicTop) -> bool { // stop enumeration if the userpic is below our point if (userpicTop > point.y()) { @@ -1264,7 +1267,7 @@ void ListWidget::updateSelected() { // Voice message seek support. if (const auto pressedView = App::pressedLinkItem()) { auto adjustedPoint = mapPointToItem(point, pressedView); - pressedView->data()->updatePressed(adjustedPoint); + pressedView->updatePressed(adjustedPoint); } //if (_mouseAction == MouseAction::Selecting) { @@ -1380,7 +1383,7 @@ void ListWidget::repaintItem(const Element *view) { if (!view) { return; } - update(0, itemTop(view), width(), view->data()->height()); + update(0, itemTop(view), width(), view->height()); } QPoint ListWidget::mapPointToItem( diff --git a/Telegram/SourceFiles/history/view/history_view_message.cpp b/Telegram/SourceFiles/history/view/history_view_message.cpp index f7297c43b..e2c933c28 100644 --- a/Telegram/SourceFiles/history/view/history_view_message.cpp +++ b/Telegram/SourceFiles/history/view/history_view_message.cpp @@ -7,9 +7,175 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL */ #include "history/view/history_view_message.h" +#include "history/view/history_view_cursor_state.h" +#include "history/history_item_components.h" #include "history/history_message.h" +#include "history/history_media_types.h" +#include "history/history_media.h" +#include "history/history.h" +#include "data/data_session.h" +#include "lang/lang_keys.h" +#include "mainwidget.h" +#include "auth_session.h" +#include "layout.h" +#include "styles/style_widgets.h" +#include "styles/style_history.h" +#include "styles/style_dialogs.h" namespace HistoryView { +namespace { + +class KeyboardStyle : public ReplyKeyboard::Style { +public: + using ReplyKeyboard::Style::Style; + + int buttonRadius() const override; + + void startPaint(Painter &p) const override; + const style::TextStyle &textStyle() const override; + void repaint(not_null item) const override; + +protected: + void paintButtonBg( + Painter &p, + const QRect &rect, + float64 howMuchOver) const override; + void paintButtonIcon(Painter &p, const QRect &rect, int outerWidth, HistoryMessageMarkupButton::Type type) const override; + void paintButtonLoading(Painter &p, const QRect &rect) const override; + int minButtonWidth(HistoryMessageMarkupButton::Type type) const override; + +}; + +void KeyboardStyle::startPaint(Painter &p) const { + p.setPen(st::msgServiceFg); +} + +const style::TextStyle &KeyboardStyle::textStyle() const { + return st::serviceTextStyle; +} + +void KeyboardStyle::repaint(not_null item) const { + Auth().data().requestItemRepaint(item); +} + +int KeyboardStyle::buttonRadius() const { + return st::dateRadius; +} + +void KeyboardStyle::paintButtonBg( + Painter &p, + const QRect &rect, + float64 howMuchOver) const { + App::roundRect(p, rect, st::msgServiceBg, StickerCorners); + if (howMuchOver > 0) { + auto o = p.opacity(); + p.setOpacity(o * howMuchOver); + App::roundRect(p, rect, st::msgBotKbOverBgAdd, BotKbOverCorners); + p.setOpacity(o); + } +} + +void KeyboardStyle::paintButtonIcon( + Painter &p, + const QRect &rect, + int outerWidth, + HistoryMessageMarkupButton::Type type) const { + using Button = HistoryMessageMarkupButton; + auto getIcon = [](Button::Type type) -> const style::icon* { + switch (type) { + case Button::Type::Url: return &st::msgBotKbUrlIcon; + case Button::Type::SwitchInlineSame: + case Button::Type::SwitchInline: return &st::msgBotKbSwitchPmIcon; + } + return nullptr; + }; + if (auto icon = getIcon(type)) { + icon->paint(p, rect.x() + rect.width() - icon->width() - st::msgBotKbIconPadding, rect.y() + st::msgBotKbIconPadding, outerWidth); + } +} + +void KeyboardStyle::paintButtonLoading(Painter &p, const QRect &rect) const { + auto icon = &st::historySendingInvertedIcon; + icon->paint(p, rect.x() + rect.width() - icon->width() - st::msgBotKbIconPadding, rect.y() + rect.height() - icon->height() - st::msgBotKbIconPadding, rect.x() * 2 + rect.width()); +} + +int KeyboardStyle::minButtonWidth( + HistoryMessageMarkupButton::Type type) const { + using Button = HistoryMessageMarkupButton; + int result = 2 * buttonPadding(), iconWidth = 0; + switch (type) { + case Button::Type::Url: iconWidth = st::msgBotKbUrlIcon.width(); break; + case Button::Type::SwitchInlineSame: + case Button::Type::SwitchInline: iconWidth = st::msgBotKbSwitchPmIcon.width(); break; + case Button::Type::Callback: + case Button::Type::Game: iconWidth = st::historySendingInvertedIcon.width(); break; + } + if (iconWidth > 0) { + result = std::max(result, 2 * iconWidth + 4 * int(st::msgBotKbIconPadding)); + } + return result; +} + +QString AdminBadgeText() { + return lang(lng_admin_badge); +} + +QString FastReplyText() { + return lang(lng_fast_reply); +} + +void PaintBubble(Painter &p, QRect rect, int outerWidth, bool selected, bool outbg, RectPart tailSide) { + auto &bg = selected ? (outbg ? st::msgOutBgSelected : st::msgInBgSelected) : (outbg ? st::msgOutBg : st::msgInBg); + auto &sh = selected ? (outbg ? st::msgOutShadowSelected : st::msgInShadowSelected) : (outbg ? st::msgOutShadow : st::msgInShadow); + auto cors = selected ? (outbg ? MessageOutSelectedCorners : MessageInSelectedCorners) : (outbg ? MessageOutCorners : MessageInCorners); + auto parts = RectPart::FullTop | RectPart::NoTopBottom | RectPart::Bottom; + if (tailSide == RectPart::Right) { + parts |= RectPart::BottomLeft; + p.fillRect(rect.x() + rect.width() - st::historyMessageRadius, rect.y() + rect.height() - st::historyMessageRadius, st::historyMessageRadius, st::historyMessageRadius, bg); + auto &tail = selected ? st::historyBubbleTailOutRightSelected : st::historyBubbleTailOutRight; + tail.paint(p, rect.x() + rect.width(), rect.y() + rect.height() - tail.height(), outerWidth); + p.fillRect(rect.x() + rect.width() - st::historyMessageRadius, rect.y() + rect.height(), st::historyMessageRadius + tail.width(), st::msgShadow, sh); + } else if (tailSide == RectPart::Left) { + parts |= RectPart::BottomRight; + p.fillRect(rect.x(), rect.y() + rect.height() - st::historyMessageRadius, st::historyMessageRadius, st::historyMessageRadius, bg); + auto &tail = selected ? (outbg ? st::historyBubbleTailOutLeftSelected : st::historyBubbleTailInLeftSelected) : (outbg ? st::historyBubbleTailOutLeft : st::historyBubbleTailInLeft); + tail.paint(p, rect.x() - tail.width(), rect.y() + rect.height() - tail.height(), outerWidth); + p.fillRect(rect.x() - tail.width(), rect.y() + rect.height(), st::historyMessageRadius + tail.width(), st::msgShadow, sh); + } else { + parts |= RectPart::FullBottom; + } + App::roundRect(p, rect, bg, cors, &sh, parts); +} + +style::color FromNameFg(not_null peer, bool selected) { + if (selected) { + const style::color colors[] = { + st::historyPeer1NameFgSelected, + st::historyPeer2NameFgSelected, + st::historyPeer3NameFgSelected, + st::historyPeer4NameFgSelected, + st::historyPeer5NameFgSelected, + st::historyPeer6NameFgSelected, + st::historyPeer7NameFgSelected, + st::historyPeer8NameFgSelected, + }; + return colors[Data::PeerColorIndex(peer->id)]; + } else { + const style::color colors[] = { + st::historyPeer1NameFg, + st::historyPeer2NameFg, + st::historyPeer3NameFg, + st::historyPeer4NameFg, + st::historyPeer5NameFg, + st::historyPeer6NameFg, + st::historyPeer7NameFg, + st::historyPeer8NameFg, + }; + return colors[Data::PeerColorIndex(peer->id)]; + } +} + +} // namespace Message::Message(not_null data, Context context) : Element(data, context) { @@ -19,4 +185,1007 @@ not_null Message::message() const { return static_cast(data().get()); } +QSize Message::performCountOptimalSize() { + const auto item = message(); + const auto media = item->getMedia(); + + auto maxWidth = 0; + auto minHeight = 0; + + updateMediaInBubbleState(); + item->refreshEditedBadge(); + if (item->drawBubble()) { + auto forwarded = item->Get(); + auto reply = item->Get(); + auto via = item->Get(); + auto entry = item->Get(); + if (forwarded) { + forwarded->create(via); + } + if (reply) { + reply->updateName(); + } + if (displayFromName()) { + item->updateAdminBadgeState(); + } + + auto mediaDisplayed = false; + if (media) { + mediaDisplayed = media->isDisplayed(); + media->initDimensions(); + } + if (entry) { + entry->_page->initDimensions(); + } + + // Entry page is always a bubble bottom. + auto mediaOnBottom = (mediaDisplayed && media->isBubbleBottom()) || (entry/* && entry->_page->isBubbleBottom()*/); + auto mediaOnTop = (mediaDisplayed && media->isBubbleTop()) || (entry && entry->_page->isBubbleTop()); + + if (mediaOnBottom) { + if (item->_text.hasSkipBlock()) { + item->_text.removeSkipBlock(); + item->_textWidth = -1; + item->_textHeight = 0; + } + } else if (!item->_text.hasSkipBlock()) { + item->_text.setSkipBlock(item->skipBlockWidth(), item->skipBlockHeight()); + item->_textWidth = -1; + item->_textHeight = 0; + } + + maxWidth = item->plainMaxWidth(); + minHeight = item->emptyText() ? 0 : item->_text.minHeight(); + if (!mediaOnBottom) { + minHeight += st::msgPadding.bottom(); + if (mediaDisplayed) minHeight += st::mediaInBubbleSkip; + } + if (!mediaOnTop) { + minHeight += st::msgPadding.top(); + if (mediaDisplayed) minHeight += st::mediaInBubbleSkip; + if (entry) minHeight += st::mediaInBubbleSkip; + } + if (mediaDisplayed) { + // Parts don't participate in maxWidth() in case of media message. + accumulate_max(maxWidth, media->maxWidth()); + minHeight += media->minHeight(); + } else { + // Count parts in maxWidth(), don't count them in minHeight(). + // They will be added in resizeGetHeight() anyway. + if (displayFromName()) { + auto namew = st::msgPadding.left() + + item->displayFrom()->nameText.maxWidth() + + st::msgPadding.right(); + if (via && !forwarded) { + namew += st::msgServiceFont->spacew + via->maxWidth; + } + const auto replyWidth = item->hasFastReply() + ? st::msgFont->width(FastReplyText()) + : 0; + if (item->hasAdminBadge()) { + const auto badgeWidth = st::msgFont->width( + AdminBadgeText()); + namew += st::msgPadding.right() + + std::max(badgeWidth, replyWidth); + } else if (replyWidth) { + namew += st::msgPadding.right() + replyWidth; + } + accumulate_max(maxWidth, namew); + } else if (via && !forwarded) { + accumulate_max(maxWidth, st::msgPadding.left() + via->maxWidth + st::msgPadding.right()); + } + if (forwarded) { + auto namew = st::msgPadding.left() + forwarded->text.maxWidth() + st::msgPadding.right(); + if (via) { + namew += st::msgServiceFont->spacew + via->maxWidth; + } + accumulate_max(maxWidth, namew); + } + if (reply) { + auto replyw = st::msgPadding.left() + reply->maxReplyWidth - st::msgReplyPadding.left() - st::msgReplyPadding.right() + st::msgPadding.right(); + if (reply->replyToVia) { + replyw += st::msgServiceFont->spacew + reply->replyToVia->maxWidth; + } + accumulate_max(maxWidth, replyw); + } + if (entry) { + accumulate_max(maxWidth, entry->_page->maxWidth()); + minHeight += entry->_page->minHeight(); + } + } + } else if (media) { + media->initDimensions(); + maxWidth = media->maxWidth(); + minHeight = media->isDisplayed() ? media->minHeight() : 0; + } else { + maxWidth = st::msgMinWidth; + minHeight = 0; + } + if (const auto markup = item->inlineReplyMarkup()) { + if (!markup->inlineKeyboard) { + markup->inlineKeyboard = std::make_unique( + item, + std::make_unique(st::msgBotKbButton)); + } + + // if we have a text bubble we can resize it to fit the keyboard + // but if we have only media we don't do that + if (!item->emptyText()) { + accumulate_max(maxWidth, markup->inlineKeyboard->naturalWidth()); + } + } + return QSize(maxWidth, minHeight); +} + +void Message::draw( + Painter &p, + QRect clip, + TextSelection selection, + TimeMs ms) const { + const auto item = message(); + const auto media = item->getMedia(); + + auto outbg = item->hasOutLayout(); + auto bubble = item->drawBubble(); + auto selected = (selection == FullSelection); + + auto g = countGeometry(); + if (g.width() < 1) { + return; + } + + auto dateh = 0; + if (auto date = item->Get()) { + dateh = date->height(); + } + if (auto unreadbar = item->Get()) { + auto unreadbarh = unreadbar->height(); + if (clip.intersects(QRect(0, dateh, width(), unreadbarh))) { + p.translate(0, dateh); + unreadbar->paint(p, 0, width()); + p.translate(0, -dateh); + } + } + + auto fullAnimMs = App::main() ? App::main()->highlightStartTime(item) : 0LL; + if (fullAnimMs > 0 && fullAnimMs <= ms) { + auto animms = ms - fullAnimMs; + if (animms < st::activeFadeInDuration + st::activeFadeOutDuration) { + auto top = marginTop(); + auto bottom = marginBottom(); + auto fill = qMin(top, bottom); + auto skiptop = top - fill; + auto fillheight = fill + g.height() + fill; + + auto dt = (animms > st::activeFadeInDuration) ? (1. - (animms - st::activeFadeInDuration) / float64(st::activeFadeOutDuration)) : (animms / float64(st::activeFadeInDuration)); + auto o = p.opacity(); + p.setOpacity(o * dt); + p.fillRect(0, skiptop, width(), fillheight, st::defaultTextPalette.selectOverlay); + p.setOpacity(o); + } + } + + p.setTextPalette(selected ? (outbg ? st::outTextPaletteSelected : st::inTextPaletteSelected) : (outbg ? st::outTextPalette : st::inTextPalette)); + + auto keyboard = item->inlineReplyKeyboard(); + if (keyboard) { + auto keyboardHeight = st::msgBotKbButton.margin + keyboard->naturalHeight(); + g.setHeight(g.height() - keyboardHeight); + auto keyboardPosition = QPoint(g.left(), g.top() + g.height() + st::msgBotKbButton.margin); + p.translate(keyboardPosition); + keyboard->paint(p, g.width(), clip.translated(-keyboardPosition), ms); + p.translate(-keyboardPosition); + } + + if (bubble) { + if (displayFromName() && item->displayFrom()->nameVersion > item->_fromNameVersion) { + fromNameUpdated(g.width()); + } + + auto entry = item->Get(); + auto mediaDisplayed = media && media->isDisplayed(); + + auto skipTail = isAttachedToNext() + || (media && media->skipBubbleTail()) + || (keyboard != nullptr); + auto displayTail = skipTail ? RectPart::None : (outbg && !Adaptive::ChatWide()) ? RectPart::Right : RectPart::Left; + PaintBubble(p, g, width(), selected, outbg, displayTail); + + // Entry page is always a bubble bottom. + auto mediaOnBottom = (mediaDisplayed && media->isBubbleBottom()) || (entry/* && entry->_page->isBubbleBottom()*/); + auto mediaOnTop = (mediaDisplayed && media->isBubbleTop()) || (entry && entry->_page->isBubbleTop()); + + auto trect = g.marginsRemoved(st::msgPadding); + if (mediaOnBottom) { + trect.setHeight(trect.height() + st::msgPadding.bottom()); + } + if (mediaOnTop) { + trect.setY(trect.y() - st::msgPadding.top()); + } else { + paintFromName(p, trect, selected); + paintForwardedInfo(p, trect, selected); + paintReplyInfo(p, trect, selected); + paintViaBotIdInfo(p, trect, selected); + } + if (entry) { + trect.setHeight(trect.height() - entry->_page->height()); + } + auto needDrawInfo = mediaOnBottom ? !(entry ? entry->_page->customInfoLayout() : media->customInfoLayout()) : true; + if (mediaDisplayed) { + auto mediaAboveText = media->isAboveMessage(); + auto mediaHeight = media->height(); + auto mediaLeft = g.left(); + auto mediaTop = mediaAboveText ? trect.y() : (trect.y() + trect.height() - mediaHeight); + if (!mediaAboveText) { + paintText(p, trect, selection); + } + p.translate(mediaLeft, mediaTop); + media->draw(p, clip.translated(-mediaLeft, -mediaTop), item->skipTextSelection(selection), ms); + p.translate(-mediaLeft, -mediaTop); + + if (mediaAboveText) { + trect.setY(trect.y() + mediaHeight); + paintText(p, trect, selection); + } else { + needDrawInfo = !media->customInfoLayout(); + } + } else { + paintText(p, trect, selection); + } + if (entry) { + auto entryLeft = g.left(); + auto entryTop = trect.y() + trect.height(); + p.translate(entryLeft, entryTop); + auto entrySelection = item->skipTextSelection(selection); + if (mediaDisplayed) { + entrySelection = media->skipSelection(entrySelection); + } + entry->_page->draw(p, clip.translated(-entryLeft, -entryTop), entrySelection, ms); + p.translate(-entryLeft, -entryTop); + } + if (needDrawInfo) { + item->HistoryMessage::drawInfo(p, g.left() + g.width(), g.top() + g.height(), 2 * g.left() + g.width(), selected, InfoDisplayDefault); + } + if (item->displayRightAction()) { + const auto fastShareSkip = snap( + (g.height() - st::historyFastShareSize) / 2, + 0, + st::historyFastShareBottom); + const auto fastShareLeft = g.left() + g.width() + st::historyFastShareLeft; + const auto fastShareTop = g.top() + g.height() - fastShareSkip - st::historyFastShareSize; + item->drawRightAction(p, fastShareLeft, fastShareTop, width()); + } + } else if (media && media->isDisplayed()) { + p.translate(g.topLeft()); + media->draw(p, clip.translated(-g.topLeft()), item->skipTextSelection(selection), ms); + p.translate(-g.topLeft()); + } + + p.restoreTextPalette(); + + const auto reply = item->Get(); + if (reply && reply->isNameUpdated()) { + const_cast(this)->setPendingResize(); + } +} + +void Message::paintFromName( + Painter &p, + QRect &trect, + bool selected) const { + const auto item = message(); + if (displayFromName()) { + const auto badgeWidth = [&] { + if (item->hasAdminBadge()) { + return st::msgFont->width(AdminBadgeText()); + } + return 0; + }(); + const auto replyWidth = [&] { + if (item->isUnderCursor() && item->displayFastReply()) { + return st::msgFont->width(FastReplyText()); + } + return 0; + }(); + const auto rightWidth = replyWidth ? replyWidth : badgeWidth; + auto availableLeft = trect.left(); + auto availableWidth = trect.width(); + if (rightWidth) { + availableWidth -= st::msgPadding.right() + rightWidth; + } + + p.setFont(st::msgNameFont); + if (item->isPost()) { + p.setPen(selected ? st::msgInServiceFgSelected : st::msgInServiceFg); + } else { + p.setPen(FromNameFg(item->author(), selected)); + } + item->displayFrom()->nameText.drawElided(p, availableLeft, trect.top(), availableWidth); + auto skipWidth = item->author()->nameText.maxWidth() + st::msgServiceFont->spacew; + availableLeft += skipWidth; + availableWidth -= skipWidth; + + auto forwarded = item->Get(); + auto via = item->Get(); + if (via && !forwarded && availableWidth > 0) { + auto outbg = item->hasOutLayout(); + p.setPen(selected ? (outbg ? st::msgOutServiceFgSelected : st::msgInServiceFgSelected) : (outbg ? st::msgOutServiceFg : st::msgInServiceFg)); + p.drawText(availableLeft, trect.top() + st::msgServiceFont->ascent, via->text); + auto skipWidth = via->width + st::msgServiceFont->spacew; + availableLeft += skipWidth; + availableWidth -= skipWidth; + } + if (rightWidth) { + p.setPen(selected ? st::msgInDateFgSelected : st::msgInDateFg); + p.setFont(ClickHandler::showAsActive(item->_fastReplyLink) + ? st::msgFont->underline() + : st::msgFont); + p.drawText( + trect.left() + trect.width() - rightWidth, + trect.top() + st::msgFont->ascent, + replyWidth ? FastReplyText() : AdminBadgeText()); + } + trect.setY(trect.y() + st::msgNameFont->height); + } +} + +void Message::paintForwardedInfo(Painter &p, QRect &trect, bool selected) const { + const auto item = message(); + if (item->displayForwardedFrom()) { + style::font serviceFont(st::msgServiceFont), serviceName(st::msgServiceNameFont); + + auto outbg = item->hasOutLayout(); + p.setPen(selected ? (outbg ? st::msgOutServiceFgSelected : st::msgInServiceFgSelected) : (outbg ? st::msgOutServiceFg : st::msgInServiceFg)); + p.setFont(serviceFont); + + auto forwarded = item->Get(); + auto breakEverywhere = (forwarded->text.countHeight(trect.width()) > 2 * serviceFont->height); + p.setTextPalette(selected ? (outbg ? st::outFwdTextPaletteSelected : st::inFwdTextPaletteSelected) : (outbg ? st::outFwdTextPalette : st::inFwdTextPalette)); + forwarded->text.drawElided(p, trect.x(), trect.y(), trect.width(), 2, style::al_left, 0, -1, 0, breakEverywhere); + p.setTextPalette(selected ? (outbg ? st::outTextPaletteSelected : st::inTextPaletteSelected) : (outbg ? st::outTextPalette : st::inTextPalette)); + + trect.setY(trect.y() + (((forwarded->text.maxWidth() > trect.width()) ? 2 : 1) * serviceFont->height)); + } +} + +void Message::paintReplyInfo(Painter &p, QRect &trect, bool selected) const { + const auto item = message(); + if (auto reply = item->Get()) { + int32 h = st::msgReplyPadding.top() + st::msgReplyBarSize.height() + st::msgReplyPadding.bottom(); + + auto flags = HistoryMessageReply::PaintFlag::InBubble | 0; + if (selected) { + flags |= HistoryMessageReply::PaintFlag::Selected; + } + reply->paint(p, item, trect.x(), trect.y(), trect.width(), flags); + + trect.setY(trect.y() + h); + } +} + +void Message::paintViaBotIdInfo(Painter &p, QRect &trect, bool selected) const { + const auto item = message(); + if (!displayFromName() && !item->Has()) { + if (auto via = item->Get()) { + p.setFont(st::msgServiceNameFont); + p.setPen(selected ? (item->hasOutLayout() ? st::msgOutServiceFgSelected : st::msgInServiceFgSelected) : (item->hasOutLayout() ? st::msgOutServiceFg : st::msgInServiceFg)); + p.drawTextLeft(trect.left(), trect.top(), width(), via->text); + trect.setY(trect.y() + st::msgServiceNameFont->height); + } + } +} + +void Message::paintText(Painter &p, QRect &trect, TextSelection selection) const { + const auto item = message(); + + auto outbg = item->hasOutLayout(); + auto selected = (selection == FullSelection); + p.setPen(outbg ? (selected ? st::historyTextOutFgSelected : st::historyTextOutFg) : (selected ? st::historyTextInFgSelected : st::historyTextInFg)); + p.setFont(st::msgFont); + item->_text.draw(p, trect.x(), trect.y(), trect.width(), style::al_left, 0, -1, selection); +} + +bool Message::hasPoint(QPoint point) const { + const auto g = countGeometry(); + if (g.width() < 1) { + return false; + } + + const auto item = message(); + if (item->drawBubble()) { + return g.contains(point); + } else if (const auto media = item->getMedia()) { + return media->hasPoint(point - g.topLeft()); + } else { + return false; + } +} + +bool Message::displayFromPhoto() const { + return hasFromPhoto() && !isAttachedToNext(); +} + +bool Message::hasFromPhoto() const { + switch (context()) { + case Context::AdminLog: + case Context::Feed: + return true; + case Context::History: { + const auto item = message(); + if (item->isPost() || item->isEmpty()) { + return false; + } else if (Adaptive::ChatWide()) { + return true; + } else if (item->history()->peer->isSelf()) { + return item->Has(); + } + return !item->out() && !item->history()->peer->isUser(); + } break; + } + Unexpected("Context in Message::hasFromPhoto."); +} + +HistoryTextState Message::getState( + QPoint point, + HistoryStateRequest request) const { + const auto item = message(); + const auto media = item->getMedia(); + + auto result = HistoryTextState(item); + + auto g = countGeometry(); + if (g.width() < 1) { + return result; + } + + auto keyboard = item->inlineReplyKeyboard(); + auto keyboardHeight = 0; + if (keyboard) { + keyboardHeight = keyboard->naturalHeight(); + g.setHeight(g.height() - st::msgBotKbButton.margin - keyboardHeight); + } + + if (item->drawBubble()) { + auto entry = item->Get(); + auto mediaDisplayed = media && media->isDisplayed(); + + // Entry page is always a bubble bottom. + auto mediaOnBottom = (mediaDisplayed && media->isBubbleBottom()) || (entry/* && entry->_page->isBubbleBottom()*/); + auto mediaOnTop = (mediaDisplayed && media->isBubbleTop()) || (entry && entry->_page->isBubbleTop()); + + auto trect = g.marginsRemoved(st::msgPadding); + if (mediaOnBottom) { + trect.setHeight(trect.height() + st::msgPadding.bottom()); + } + if (mediaOnTop) { + trect.setY(trect.y() - st::msgPadding.top()); + } else { + if (getStateFromName(point, trect, &result)) return result; + if (getStateForwardedInfo(point, trect, &result, request)) return result; + if (getStateReplyInfo(point, trect, &result)) return result; + if (getStateViaBotIdInfo(point, trect, &result)) return result; + } + if (entry) { + auto entryHeight = entry->_page->height(); + trect.setHeight(trect.height() - entryHeight); + auto entryLeft = g.left(); + auto entryTop = trect.y() + trect.height(); + if (point.y() >= entryTop && point.y() < entryTop + entryHeight) { + result = entry->_page->getState( + point - QPoint(entryLeft, entryTop), + request); + result.symbol += item->_text.length() + (mediaDisplayed ? media->fullSelectionLength() : 0); + } + } + + auto needDateCheck = mediaOnBottom ? !(entry ? entry->_page->customInfoLayout() : media->customInfoLayout()) : true; + if (mediaDisplayed) { + auto mediaAboveText = media->isAboveMessage(); + auto mediaHeight = media->height(); + auto mediaLeft = trect.x() - st::msgPadding.left(); + auto mediaTop = mediaAboveText ? trect.y() : (trect.y() + trect.height() - mediaHeight); + + if (point.y() >= mediaTop && point.y() < mediaTop + mediaHeight) { + result = media->getState(point - QPoint(mediaLeft, mediaTop), request); + result.symbol += item->_text.length(); + } else { + if (mediaAboveText) { + trect.setY(trect.y() + mediaHeight); + } + if (trect.contains(point)) { + getStateText(point, trect, &result, request); + } + } + } else if (trect.contains(point)) { + getStateText(point, trect, &result, request); + } + if (needDateCheck) { + if (item->HistoryMessage::pointInTime(g.left() + g.width(), g.top() + g.height(), point, InfoDisplayDefault)) { + result.cursor = HistoryInDateCursorState; + } + } + if (item->displayRightAction()) { + const auto fastShareSkip = snap( + (g.height() - st::historyFastShareSize) / 2, + 0, + st::historyFastShareBottom); + const auto fastShareLeft = g.left() + g.width() + st::historyFastShareLeft; + const auto fastShareTop = g.top() + g.height() - fastShareSkip - st::historyFastShareSize; + if (QRect( + fastShareLeft, + fastShareTop, + st::historyFastShareSize, + st::historyFastShareSize + ).contains(point)) { + result.link = item->rightActionLink(); + } + } + } else if (media && media->isDisplayed()) { + result = media->getState(point - g.topLeft(), request); + result.symbol += item->_text.length(); + } + + if (keyboard && !item->isLogEntry()) { + auto keyboardTop = g.top() + g.height() + st::msgBotKbButton.margin; + if (QRect(g.left(), keyboardTop, g.width(), keyboardHeight).contains(point)) { + result.link = keyboard->getState(point - QPoint(g.left(), keyboardTop)); + return result; + } + } + + return result; +} + +bool Message::getStateFromName( + QPoint point, + QRect &trect, + not_null outResult) const { + const auto item = message(); + if (displayFromName()) { + const auto replyWidth = [&] { + if (item->isUnderCursor() && item->displayFastReply()) { + return st::msgFont->width(FastReplyText()); + } + return 0; + }(); + if (replyWidth + && point.x() >= trect.left() + trect.width() - replyWidth + && point.x() < trect.left() + trect.width() + st::msgPadding.right() + && point.y() >= trect.top() - st::msgPadding.top() + && point.y() < trect.top() + st::msgServiceFont->height) { + outResult->link = item->fastReplyLink(); + return true; + } + if (point.y() >= trect.top() && point.y() < trect.top() + st::msgNameFont->height) { + auto availableLeft = trect.left(); + auto availableWidth = trect.width(); + if (replyWidth) { + availableWidth -= st::msgPadding.right() + replyWidth; + } + auto user = item->displayFrom(); + if (point.x() >= availableLeft + && point.x() < availableLeft + availableWidth + && point.x() < availableLeft + user->nameText.maxWidth()) { + outResult->link = user->openLink(); + return true; + } + auto forwarded = item->Get(); + auto via = item->Get(); + if (via + && !forwarded + && point.x() >= availableLeft + item->author()->nameText.maxWidth() + st::msgServiceFont->spacew + && point.x() < availableLeft + availableWidth + && point.x() < availableLeft + user->nameText.maxWidth() + st::msgServiceFont->spacew + via->width) { + outResult->link = via->link; + return true; + } + } + trect.setTop(trect.top() + st::msgNameFont->height); + } + return false; +} + +bool Message::getStateForwardedInfo( + QPoint point, + QRect &trect, + not_null outResult, + HistoryStateRequest request) const { + const auto item = message(); + if (item->displayForwardedFrom()) { + auto forwarded = item->Get(); + auto fwdheight = ((forwarded->text.maxWidth() > trect.width()) ? 2 : 1) * st::semiboldFont->height; + if (point.y() >= trect.top() && point.y() < trect.top() + fwdheight) { + auto breakEverywhere = (forwarded->text.countHeight(trect.width()) > 2 * st::semiboldFont->height); + auto textRequest = request.forText(); + if (breakEverywhere) { + textRequest.flags |= Text::StateRequest::Flag::BreakEverywhere; + } + *outResult = HistoryTextState(item, forwarded->text.getState( + point - trect.topLeft(), + trect.width(), + textRequest)); + outResult->symbol = 0; + outResult->afterSymbol = false; + if (breakEverywhere) { + outResult->cursor = HistoryInForwardedCursorState; + } else { + outResult->cursor = HistoryDefaultCursorState; + } + return true; + } + trect.setTop(trect.top() + fwdheight); + } + return false; +} + +bool Message::getStateReplyInfo( + QPoint point, + QRect &trect, + not_null outResult) const { + const auto item = message(); + if (auto reply = item->Get()) { + int32 h = st::msgReplyPadding.top() + st::msgReplyBarSize.height() + st::msgReplyPadding.bottom(); + if (point.y() >= trect.top() && point.y() < trect.top() + h) { + if (reply->replyToMsg && QRect(trect.x(), trect.y() + st::msgReplyPadding.top(), trect.width(), st::msgReplyBarSize.height()).contains(point)) { + outResult->link = reply->replyToLink(); + } + return true; + } + trect.setTop(trect.top() + h); + } + return false; +} + +bool Message::getStateViaBotIdInfo( + QPoint point, + QRect &trect, + not_null outResult) const { + const auto item = message(); + if (!displayFromName() && !item->Has()) { + if (auto via = item->Get()) { + if (QRect(trect.x(), trect.y(), via->width, st::msgNameFont->height).contains(point)) { + outResult->link = via->link; + return true; + } + trect.setTop(trect.top() + st::msgNameFont->height); + } + } + return false; +} + +bool Message::getStateText( + QPoint point, + QRect &trect, + not_null outResult, + HistoryStateRequest request) const { + const auto item = message(); + if (trect.contains(point)) { + *outResult = HistoryTextState(item, item->_text.getState( + point - trect.topLeft(), + trect.width(), + request.forText())); + return true; + } + return false; +} + +// Forward to media. +void Message::updatePressed(QPoint point) { + const auto item = message(); + const auto media = item->getMedia(); + if (!media) return; + + auto g = countGeometry(); + auto keyboard = item->inlineReplyKeyboard(); + if (keyboard) { + auto keyboardHeight = st::msgBotKbButton.margin + keyboard->naturalHeight(); + g.setHeight(g.height() - keyboardHeight); + } + + if (item->drawBubble()) { + auto mediaDisplayed = media && media->isDisplayed(); + auto top = marginTop(); + auto trect = g.marginsAdded(-st::msgPadding); + if (mediaDisplayed && media->isBubbleTop()) { + trect.setY(trect.y() - st::msgPadding.top()); + } else { + if (displayFromName()) trect.setTop(trect.top() + st::msgNameFont->height); + if (item->displayForwardedFrom()) { + auto forwarded = item->Get(); + auto fwdheight = ((forwarded->text.maxWidth() > trect.width()) ? 2 : 1) * st::semiboldFont->height; + trect.setTop(trect.top() + fwdheight); + } + if (item->Get()) { + auto h = st::msgReplyPadding.top() + st::msgReplyBarSize.height() + st::msgReplyPadding.bottom(); + trect.setTop(trect.top() + h); + } + if (!displayFromName() && !item->Has()) { + if (auto via = item->Get()) { + trect.setTop(trect.top() + st::msgNameFont->height); + } + } + } + if (mediaDisplayed && media->isBubbleBottom()) { + trect.setHeight(trect.height() + st::msgPadding.bottom()); + } + + auto needDateCheck = true; + if (mediaDisplayed) { + auto mediaAboveText = media->isAboveMessage(); + auto mediaHeight = media->height(); + auto mediaLeft = trect.x() - st::msgPadding.left(); + auto mediaTop = mediaAboveText ? trect.y() : (trect.y() + trect.height() - mediaHeight); + media->updatePressed(point - QPoint(mediaLeft, mediaTop)); + } + } else { + media->updatePressed(point - g.topLeft()); + } +} + +bool Message::hasFromName() const { + switch (context()) { + case Context::AdminLog: + case Context::Feed: + return true; + case Context::History: { + const auto item = message(); + return !item->hasOutLayout() + && (!item->history()->peer->isUser() + || item->history()->peer->isSelf()); + } break; + } + Unexpected("Context in Message::hasFromPhoto."); +} + +bool Message::displayFromName() const { + if (!hasFromName()) return false; + if (isAttachedToPrevious()) return false; + return true; +} + +void Message::updateMediaInBubbleState() { + const auto item = message(); + const auto media = item->getMedia(); + + auto mediaHasSomethingBelow = false; + auto mediaHasSomethingAbove = false; + auto getMediaHasSomethingAbove = [&] { + return displayFromName() + || item->displayForwardedFrom() + || item->Has() + || item->Has(); + }; + auto entry = item->Get(); + if (entry) { + mediaHasSomethingBelow = true; + mediaHasSomethingAbove = getMediaHasSomethingAbove(); + auto entryState = (mediaHasSomethingAbove + || !item->emptyText() + || (media && media->isDisplayed())) + ? MediaInBubbleState::Bottom + : MediaInBubbleState::None; + entry->_page->setInBubbleState(entryState); + } + if (!media) { + return; + } + + media->updateNeedBubbleState(); + if (!item->drawBubble()) { + media->setInBubbleState(MediaInBubbleState::None); + return; + } + + if (!entry) { + mediaHasSomethingAbove = getMediaHasSomethingAbove(); + } + if (!item->emptyText()) { + if (media->isAboveMessage()) { + mediaHasSomethingBelow = true; + } else { + mediaHasSomethingAbove = true; + } + } + const auto state = [&] { + if (mediaHasSomethingAbove) { + if (mediaHasSomethingBelow) { + return MediaInBubbleState::Middle; + } + return MediaInBubbleState::Bottom; + } else if (mediaHasSomethingBelow) { + return MediaInBubbleState::Top; + } + return MediaInBubbleState::None; + }(); + media->setInBubbleState(state); +} + +void Message::fromNameUpdated(int width) const { + const auto item = message(); + const auto replyWidth = item->hasFastReply() + ? st::msgFont->width(FastReplyText()) + : 0; + if (item->hasAdminBadge()) { + const auto badgeWidth = st::msgFont->width(AdminBadgeText()); + width -= st::msgPadding.right() + std::max(badgeWidth, replyWidth); + } else if (replyWidth) { + width -= st::msgPadding.right() + replyWidth; + } + item->_fromNameVersion = item->displayFrom()->nameVersion; + if (!item->Has()) { + if (auto via = item->Get()) { + via->resize(width + - st::msgPadding.left() + - st::msgPadding.right() + - item->author()->nameText.maxWidth() + - st::msgServiceFont->spacew); + } + } +} + +QRect Message::countGeometry() const { + const auto item = message(); + const auto media = item->getMedia(); + + auto maxwidth = qMin(st::msgMaxWidth, maxWidth()); + if (media && media->width() < maxwidth) { + maxwidth = qMax(media->width(), qMin(maxwidth, item->plainMaxWidth())); + } + + const auto outLayout = item->hasOutLayout(); + auto contentLeft = (outLayout && !Adaptive::ChatWide()) + ? st::msgMargin.right() + : st::msgMargin.left(); + if (hasFromPhoto()) { + contentLeft += st::msgPhotoSkip; + //} else if (!Adaptive::Wide() && !out() && !fromChannel() && st::msgPhotoSkip - (hmaxwidth - hwidth) > 0) { + // contentLeft += st::msgPhotoSkip - (hmaxwidth - hwidth); + } + + auto contentWidth = width() - st::msgMargin.left() - st::msgMargin.right(); + if (item->history()->peer->isSelf() && !outLayout) { + contentWidth -= st::msgPhotoSkip; + } + if (contentWidth > maxwidth) { + if (outLayout && !Adaptive::ChatWide()) { + contentLeft += contentWidth - maxwidth; + } + contentWidth = maxwidth; + } + + const auto contentTop = marginTop(); + return QRect( + contentLeft, + contentTop, + contentWidth, + height() - contentTop - marginBottom()); +} + +int Message::resizeContentGetHeight(int newWidth) { + const auto item = message(); + const auto media = item->getMedia(); + + if (newWidth < st::msgMinWidth) { + return height(); + } + + auto newHeight = minHeight(); + auto contentWidth = newWidth - (st::msgMargin.left() + st::msgMargin.right()); + if (item->history()->peer->isSelf() && !item->hasOutLayout()) { + contentWidth -= st::msgPhotoSkip; + } + if (contentWidth < st::msgPadding.left() + st::msgPadding.right() + 1) { + contentWidth = st::msgPadding.left() + st::msgPadding.right() + 1; + } else if (contentWidth > st::msgMaxWidth) { + contentWidth = st::msgMaxWidth; + } + if (item->drawBubble()) { + auto forwarded = item->Get(); + auto reply = item->Get(); + auto via = item->Get(); + auto entry = item->Get(); + + auto mediaDisplayed = false; + if (media) { + mediaDisplayed = media->isDisplayed(); + } + + // Entry page is always a bubble bottom. + auto mediaOnBottom = (mediaDisplayed && media->isBubbleBottom()) || (entry/* && entry->_page->isBubbleBottom()*/); + auto mediaOnTop = (mediaDisplayed && media->isBubbleTop()) || (entry && entry->_page->isBubbleTop()); + + if (contentWidth >= maxWidth()) { + if (mediaDisplayed) { + media->resizeGetHeight(maxWidth()); + if (entry) { + newHeight += entry->_page->resizeGetHeight(countGeometry().width()); + } + } else if (entry) { + // In case of text-only message it is counted in minHeight already. + entry->_page->resizeGetHeight(countGeometry().width()); + } + } else { + if (item->emptyText()) { + newHeight = 0; + } else { + auto textWidth = qMax(contentWidth - st::msgPadding.left() - st::msgPadding.right(), 1); + if (textWidth != item->_textWidth) { + item->_textWidth = textWidth; + item->_textHeight = item->_text.countHeight(textWidth); + } + newHeight = item->_textHeight; + } + if (!mediaOnBottom) { + newHeight += st::msgPadding.bottom(); + if (mediaDisplayed) newHeight += st::mediaInBubbleSkip; + } + if (!mediaOnTop) { + newHeight += st::msgPadding.top(); + if (mediaDisplayed) newHeight += st::mediaInBubbleSkip; + if (entry) newHeight += st::mediaInBubbleSkip; + } + if (mediaDisplayed) { + newHeight += media->resizeGetHeight(contentWidth); + if (entry) { + newHeight += entry->_page->resizeGetHeight(countGeometry().width()); + } + } else if (entry) { + newHeight += entry->_page->resizeGetHeight(contentWidth); + } + } + + if (displayFromName()) { + fromNameUpdated(countGeometry().width()); + newHeight += st::msgNameFont->height; + } else if (via && !forwarded) { + via->resize(countGeometry().width() - st::msgPadding.left() - st::msgPadding.right()); + newHeight += st::msgNameFont->height; + } + + if (item->displayForwardedFrom()) { + auto fwdheight = ((forwarded->text.maxWidth() > (countGeometry().width() - st::msgPadding.left() - st::msgPadding.right())) ? 2 : 1) * st::semiboldFont->height; + newHeight += fwdheight; + } + + if (reply) { + reply->resize(countGeometry().width() - st::msgPadding.left() - st::msgPadding.right()); + newHeight += st::msgReplyPadding.top() + st::msgReplyBarSize.height() + st::msgReplyPadding.bottom(); + } + } else if (media && media->isDisplayed()) { + newHeight = media->resizeGetHeight(contentWidth); + } else { + newHeight = 0; + } + if (const auto keyboard = item->inlineReplyKeyboard()) { + const auto g = countGeometry(); + const auto keyboardHeight = st::msgBotKbButton.margin + keyboard->naturalHeight(); + newHeight += keyboardHeight; + keyboard->resize(g.width(), keyboardHeight - st::msgBotKbButton.margin); + } + + newHeight += marginTop() + marginBottom(); + return newHeight; +} + +QSize Message::performCountCurrentSize(int newWidth) { + const auto item = message(); + const auto newHeight = resizeContentGetHeight(newWidth); + + const auto keyboard = item->inlineReplyKeyboard(); + if (const auto markup = item->Get()) { + const auto oldTop = markup->oldTop; + if (oldTop >= 0) { + markup->oldTop = -1; + if (keyboard) { + const auto height = st::msgBotKbButton.margin + keyboard->naturalHeight(); + const auto keyboardTop = newHeight - height + st::msgBotKbButton.margin - marginBottom(); + if (keyboardTop != oldTop) { + Notify::inlineKeyboardMoved(item, oldTop, keyboardTop); + } + } + } + } + return { newWidth, newHeight }; +} + } // namespace HistoryView diff --git a/Telegram/SourceFiles/history/view/history_view_message.h b/Telegram/SourceFiles/history/view/history_view_message.h index 74663e9ef..311b11fe3 100644 --- a/Telegram/SourceFiles/history/view/history_view_message.h +++ b/Telegram/SourceFiles/history/view/history_view_message.h @@ -17,9 +17,67 @@ class Message : public Element { public: Message(not_null data, Context context); + void draw( + Painter &p, + QRect clip, + TextSelection selection, + TimeMs ms) const override; + bool hasPoint(QPoint point) const override; + HistoryTextState getState( + QPoint point, + HistoryStateRequest request) const override; + void updatePressed(QPoint point) override; + + // hasFromPhoto() returns true even if we don't display the photo + // but we need to skip a place at the left side for this photo + bool displayFromPhoto() const override; + bool hasFromPhoto() const override; + + bool hasFromName() const override; + bool displayFromName() const override; + private: not_null message() const; + void fromNameUpdated(int width) const; + + void paintFromName(Painter &p, QRect &trect, bool selected) const; + void paintForwardedInfo(Painter &p, QRect &trect, bool selected) const; + void paintReplyInfo(Painter &p, QRect &trect, bool selected) const; + // this method draws "via @bot" if it is not painted in forwarded info or in from name + void paintViaBotIdInfo(Painter &p, QRect &trect, bool selected) const; + void paintText(Painter &p, QRect &trect, TextSelection selection) const; + + bool getStateFromName( + QPoint point, + QRect &trect, + not_null outResult) const; + bool getStateForwardedInfo( + QPoint point, + QRect &trect, + not_null outResult, + HistoryStateRequest request) const; + bool getStateReplyInfo( + QPoint point, + QRect &trect, + not_null outResult) const; + bool getStateViaBotIdInfo( + QPoint point, + QRect &trect, + not_null outResult) const; + bool getStateText( + QPoint point, + QRect &trect, + not_null outResult, + HistoryStateRequest request) const; + + void updateMediaInBubbleState(); + QRect countGeometry() const; + + int resizeContentGetHeight(int newWidth); + QSize performCountOptimalSize() override; + QSize performCountCurrentSize(int newWidth) override; + }; } // namespace HistoryView diff --git a/Telegram/SourceFiles/history/view/history_view_object.h b/Telegram/SourceFiles/history/view/history_view_object.h new file mode 100644 index 000000000..2e0ac67fd --- /dev/null +++ b/Telegram/SourceFiles/history/view/history_view_object.h @@ -0,0 +1,62 @@ +/* +This file is part of Telegram Desktop, +the official desktop application for the Telegram messaging service. + +For license and copyright information please follow this link: +https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL +*/ +#pragma once + +namespace HistoryView { + +class Object { +public: + Object() = default; + Object(const Object &other) = delete; + Object &operator=(const Object &other) = delete; + + void initDimensions() { + setOptimalSize(countOptimalSize()); + } + int resizeGetHeight(int newWidth) { + setCurrentSize(countCurrentSize(newWidth)); + return _height; + } + + int maxWidth() const { + return _maxWidth; + } + int minHeight() const { + return _minHeight; + } + int width() const { + return _width; + } + int height() const { + return _height; + } + + virtual ~Object() = default; + +protected: + void setOptimalSize(QSize size) { + _maxWidth = size.width(); + _minHeight = size.height(); + } + void setCurrentSize(QSize size) { + _width = size.width(); + _height = size.height(); + } + +private: + virtual QSize countOptimalSize() = 0; + virtual QSize countCurrentSize(int newWidth) = 0; + + int _maxWidth = 0; + int _minHeight = 0; + int _width = 0; + int _height = 0; + +}; + +} // namespace HistoryView diff --git a/Telegram/SourceFiles/history/view/history_view_service_message.cpp b/Telegram/SourceFiles/history/view/history_view_service_message.cpp index 014e6aee5..d2dce1aa5 100644 --- a/Telegram/SourceFiles/history/view/history_view_service_message.cpp +++ b/Telegram/SourceFiles/history/view/history_view_service_message.cpp @@ -7,8 +7,11 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL */ #include "history/view/history_view_service_message.h" +#include "history/history.h" #include "history/history_service.h" #include "history/history_media.h" +#include "history/history_item_components.h" +#include "history/view/history_view_cursor_state.h" #include "data/data_abstract_structure.h" #include "styles/style_history.h" #include "mainwidget.h" @@ -171,66 +174,10 @@ void paintPreparedDate(Painter &p, const QString &dateText, int dateTextWidth, i } // namepsace -Service::Service(not_null data, Context context) -: Element(data, context) { -} - -not_null Service::message() const { - return static_cast(data().get()); -} - int WideChatWidth() { return st::msgMaxWidth + 2 * st::msgPhotoSkip + 2 * st::msgMargin.left(); } -void ServiceMessagePainter::paint( - Painter &p, - not_null message, - const PaintContext &context, - int height) { - auto g = message->countGeometry(); - if (g.width() < 1) return; - - auto fullAnimMs = App::main() ? App::main()->highlightStartTime(message) : 0LL; - if (fullAnimMs > 0 && fullAnimMs <= context.ms) { - auto animms = context.ms - fullAnimMs; - if (animms < st::activeFadeInDuration + st::activeFadeOutDuration) { - auto top = st::msgServiceMargin.top(); - auto bottom = st::msgServiceMargin.bottom(); - auto fill = qMin(top, bottom); - auto skiptop = top - fill; - auto fillheight = fill + height + fill; - - auto dt = (animms > st::activeFadeInDuration) ? (1. - (animms - st::activeFadeInDuration) / float64(st::activeFadeOutDuration)) : (animms / float64(st::activeFadeInDuration)); - auto o = p.opacity(); - p.setOpacity(o * dt); - p.fillRect(0, skiptop, message->history()->width, fillheight, st::defaultTextPalette.selectOverlay); - p.setOpacity(o); - } - } - - p.setTextPalette(st::serviceTextPalette); - - if (auto media = message->getMedia()) { - height -= st::msgServiceMargin.top() + media->height(); - auto left = st::msgServiceMargin.left() + (g.width() - media->maxWidth()) / 2, top = st::msgServiceMargin.top() + height + st::msgServiceMargin.top(); - p.translate(left, top); - media->draw(p, context.clip.translated(-left, -top), message->skipTextSelection(context.selection), context.ms); - p.translate(-left, -top); - } - - auto trect = QRect(g.left(), st::msgServiceMargin.top(), g.width(), height).marginsAdded(-st::msgServicePadding); - - paintComplexBubble(p, g.left(), g.width(), message->_text, trect); - - p.setBrush(Qt::NoBrush); - p.setPen(st::msgServiceFg); - p.setFont(st::msgServiceFont); - message->_text.draw(p, trect.x(), trect.y(), trect.width(), Qt::AlignCenter, 0, -1, context.selection, false); - - p.restoreTextPalette(); -} - void ServiceMessagePainter::paintDate(Painter &p, const QDateTime &date, int y, int w) { auto dateText = langDayOfMonthFull(date.date()); auto dateTextWidth = st::msgServiceFont->width(dateText); @@ -344,27 +291,217 @@ void serviceColorsUpdated() { } } -void paintBubble(Painter &p, QRect rect, int outerWidth, bool selected, bool outbg, RectPart tailSide) { - auto &bg = selected ? (outbg ? st::msgOutBgSelected : st::msgInBgSelected) : (outbg ? st::msgOutBg : st::msgInBg); - auto &sh = selected ? (outbg ? st::msgOutShadowSelected : st::msgInShadowSelected) : (outbg ? st::msgOutShadow : st::msgInShadow); - auto cors = selected ? (outbg ? MessageOutSelectedCorners : MessageInSelectedCorners) : (outbg ? MessageOutCorners : MessageInCorners); - auto parts = RectPart::FullTop | RectPart::NoTopBottom | RectPart::Bottom; - if (tailSide == RectPart::Right) { - parts |= RectPart::BottomLeft; - p.fillRect(rect.x() + rect.width() - st::historyMessageRadius, rect.y() + rect.height() - st::historyMessageRadius, st::historyMessageRadius, st::historyMessageRadius, bg); - auto &tail = selected ? st::historyBubbleTailOutRightSelected : st::historyBubbleTailOutRight; - tail.paint(p, rect.x() + rect.width(), rect.y() + rect.height() - tail.height(), outerWidth); - p.fillRect(rect.x() + rect.width() - st::historyMessageRadius, rect.y() + rect.height(), st::historyMessageRadius + tail.width(), st::msgShadow, sh); - } else if (tailSide == RectPart::Left) { - parts |= RectPart::BottomRight; - p.fillRect(rect.x(), rect.y() + rect.height() - st::historyMessageRadius, st::historyMessageRadius, st::historyMessageRadius, bg); - auto &tail = selected ? (outbg ? st::historyBubbleTailOutLeftSelected : st::historyBubbleTailInLeftSelected) : (outbg ? st::historyBubbleTailOutLeft : st::historyBubbleTailInLeft); - tail.paint(p, rect.x() - tail.width(), rect.y() + rect.height() - tail.height(), outerWidth); - p.fillRect(rect.x() - tail.width(), rect.y() + rect.height(), st::historyMessageRadius + tail.width(), st::msgShadow, sh); - } else { - parts |= RectPart::FullBottom; +Service::Service(not_null data, Context context) +: Element(data, context) { +} + +not_null Service::message() const { + return static_cast(data().get()); +} + +QRect Service::countGeometry() const { + auto result = QRect(0, 0, width(), height()); + if (Adaptive::ChatWide()) { + result.setWidth(qMin(result.width(), st::msgMaxWidth + 2 * st::msgPhotoSkip + 2 * st::msgMargin.left())); } - App::roundRect(p, rect, bg, cors, &sh, parts); + return result.marginsRemoved(st::msgServiceMargin); +} + +QSize Service::performCountCurrentSize(int newWidth) { + const auto item = message(); + const auto media = item->getMedia(); + + auto newHeight = item->displayedDateHeight(); + if (auto unreadbar = item->Get()) { + newHeight += unreadbar->height(); + } + + if (item->_text.isEmpty()) { + item->_textHeight = 0; + } else { + auto contentWidth = newWidth; + if (Adaptive::ChatWide()) { + accumulate_min(contentWidth, st::msgMaxWidth + 2 * st::msgPhotoSkip + 2 * st::msgMargin.left()); + } + contentWidth -= st::msgServiceMargin.left() + st::msgServiceMargin.left(); // two small margins + if (contentWidth < st::msgServicePadding.left() + st::msgServicePadding.right() + 1) { + contentWidth = st::msgServicePadding.left() + st::msgServicePadding.right() + 1; + } + + auto nwidth = qMax(contentWidth - st::msgServicePadding.left() - st::msgServicePadding.right(), 0); + if (nwidth != item->_textWidth) { + item->_textWidth = nwidth; + item->_textHeight = item->_text.countHeight(nwidth); + } + if (contentWidth >= maxWidth()) { + newHeight += minHeight(); + } else { + newHeight += item->_textHeight; + } + newHeight += st::msgServicePadding.top() + st::msgServicePadding.bottom() + st::msgServiceMargin.top() + st::msgServiceMargin.bottom(); + if (media) { + newHeight += st::msgServiceMargin.top() + media->resizeGetHeight(media->width()); + } + } + + return { newWidth, newHeight }; +} + +QSize Service::performCountOptimalSize() { + const auto item = message(); + const auto media = item->getMedia(); + + auto maxWidth = item->_text.maxWidth() + st::msgServicePadding.left() + st::msgServicePadding.right(); + auto minHeight = item->_text.minHeight(); + if (media) { + media->initDimensions(); + } + return { maxWidth, minHeight }; +} + +void Service::draw( + Painter &p, + QRect clip, + TextSelection selection, + TimeMs ms) const { + const auto item = message(); + auto g = countGeometry(); + if (g.width() < 1) { + return; + } + + auto height = this->height() - st::msgServiceMargin.top() - st::msgServiceMargin.bottom(); + auto dateh = 0; + auto unreadbarh = 0; + if (auto date = item->Get()) { + dateh = date->height(); + p.translate(0, dateh); + clip.translate(0, -dateh); + height -= dateh; + } + if (auto unreadbar = item->Get()) { + unreadbarh = unreadbar->height(); + if (clip.intersects(QRect(0, 0, width(), unreadbarh))) { + unreadbar->paint(p, 0, width()); + } + p.translate(0, unreadbarh); + clip.translate(0, -unreadbarh); + height -= unreadbarh; + } + + auto fullAnimMs = App::main() ? App::main()->highlightStartTime(item) : 0LL; + if (fullAnimMs > 0 && fullAnimMs <= ms) { + auto animms = ms - fullAnimMs; + if (animms < st::activeFadeInDuration + st::activeFadeOutDuration) { + auto top = st::msgServiceMargin.top(); + auto bottom = st::msgServiceMargin.bottom(); + auto fill = qMin(top, bottom); + auto skiptop = top - fill; + auto fillheight = fill + height + fill; + + auto dt = (animms > st::activeFadeInDuration) ? (1. - (animms - st::activeFadeInDuration) / float64(st::activeFadeOutDuration)) : (animms / float64(st::activeFadeInDuration)); + auto o = p.opacity(); + p.setOpacity(o * dt); + p.fillRect(0, skiptop, item->history()->width, fillheight, st::defaultTextPalette.selectOverlay); + p.setOpacity(o); + } + } + + p.setTextPalette(st::serviceTextPalette); + + if (auto media = item->getMedia()) { + height -= st::msgServiceMargin.top() + media->height(); + auto left = st::msgServiceMargin.left() + (g.width() - media->maxWidth()) / 2, top = st::msgServiceMargin.top() + height + st::msgServiceMargin.top(); + p.translate(left, top); + media->draw(p, clip.translated(-left, -top), item->skipTextSelection(selection), ms); + p.translate(-left, -top); + } + + auto trect = QRect(g.left(), st::msgServiceMargin.top(), g.width(), height).marginsAdded(-st::msgServicePadding); + + ServiceMessagePainter::paintComplexBubble(p, g.left(), g.width(), item->_text, trect); + + p.setBrush(Qt::NoBrush); + p.setPen(st::msgServiceFg); + p.setFont(st::msgServiceFont); + item->_text.draw(p, trect.x(), trect.y(), trect.width(), Qt::AlignCenter, 0, -1, selection, false); + + p.restoreTextPalette(); + + if (auto skiph = dateh + unreadbarh) { + p.translate(0, -skiph); + } +} + +bool Service::hasPoint(QPoint point) const { + const auto item = message(); + const auto media = item->getMedia(); + + auto g = countGeometry(); + if (g.width() < 1) { + return false; + } + + if (auto dateh = item->displayedDateHeight()) { + g.setTop(g.top() + dateh); + } + if (auto unreadbar = item->Get()) { + g.setTop(g.top() + unreadbar->height()); + } + if (media) { + g.setHeight(g.height() - (st::msgServiceMargin.top() + media->height())); + } + return g.contains(point); +} + +HistoryTextState Service::getState(QPoint point, HistoryStateRequest request) const { + const auto item = message(); + const auto media = item->getMedia(); + + auto result = HistoryTextState(item); + + auto g = countGeometry(); + if (g.width() < 1) { + return result; + } + + if (auto dateh = item->displayedDateHeight()) { + point.setY(point.y() - dateh); + g.setHeight(g.height() - dateh); + } + if (auto unreadbar = item->Get()) { + auto unreadbarh = unreadbar->height(); + point.setY(point.y() - unreadbarh); + g.setHeight(g.height() - unreadbarh); + } + + if (media) { + g.setHeight(g.height() - (st::msgServiceMargin.top() + media->height())); + } + auto trect = g.marginsAdded(-st::msgServicePadding); + if (trect.contains(point)) { + auto textRequest = request.forText(); + textRequest.align = style::al_center; + result = HistoryTextState(item, item->_text.getState( + point - trect.topLeft(), + trect.width(), + textRequest)); + if (auto gamescore = item->Get()) { + if (!result.link && result.cursor == HistoryInTextCursorState && g.contains(point)) { + result.link = gamescore->lnk; + } + } else if (auto payment = item->Get()) { + if (!result.link && result.cursor == HistoryInTextCursorState && g.contains(point)) { + result.link = payment->lnk; + } + } + } else if (media) { + result = media->getState(point - QPoint(st::msgServiceMargin.left() + (g.width() - media->maxWidth()) / 2, st::msgServiceMargin.top() + g.height() + st::msgServiceMargin.top()), request); + } + return result; +} + +void Service::updatePressed(QPoint point) { } } // namespace HistoryView diff --git a/Telegram/SourceFiles/history/view/history_view_service_message.h b/Telegram/SourceFiles/history/view/history_view_service_message.h index c2bff03d3..8ac773b39 100644 --- a/Telegram/SourceFiles/history/view/history_view_service_message.h +++ b/Telegram/SourceFiles/history/view/history_view_service_message.h @@ -17,9 +17,25 @@ class Service : public Element { public: Service(not_null data, Context context); + void draw( + Painter &p, + QRect clip, + TextSelection selection, + TimeMs ms) const override; + bool hasPoint(QPoint point) const override; + HistoryTextState getState( + QPoint point, + HistoryStateRequest request) const override; + void updatePressed(QPoint point) override; + private: not_null message() const; + QRect countGeometry() const; + + QSize performCountOptimalSize() override; + QSize performCountCurrentSize(int newWidth) override; + }; int WideChatWidth(); @@ -37,19 +53,14 @@ struct PaintContext { class ServiceMessagePainter { public: - static void paint( - Painter &p, - not_null message, - const PaintContext &context, - int height); - static void paintDate(Painter &p, const QDateTime &date, int y, int w); static void paintDate(Painter &p, const QString &dateText, int dateTextWidth, int y, int w); static void paintBubble(Painter &p, int x, int y, int w, int h); -private: static void paintComplexBubble(Painter &p, int left, int width, const Text &text, const QRect &textRect); + +private: static QVector countLineWidths(const Text &text, const QRect &textRect); }; @@ -58,6 +69,4 @@ void paintEmpty(Painter &p, int width, int height); void serviceColorsUpdated(); -void paintBubble(Painter &p, QRect rect, int outerWidth, bool selected, bool outbg, RectPart tailSide); - } // namespace HistoryView diff --git a/Telegram/SourceFiles/history/view/history_view_top_bar_widget.cpp b/Telegram/SourceFiles/history/view/history_view_top_bar_widget.cpp index 474a9a4d9..6e4d3957b 100644 --- a/Telegram/SourceFiles/history/view/history_view_top_bar_widget.cpp +++ b/Telegram/SourceFiles/history/view/history_view_top_bar_widget.cpp @@ -9,10 +9,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include #include -#include "styles/style_window.h" -#include "styles/style_dialogs.h" -#include "styles/style_history.h" -#include "styles/style_info.h" +#include "history/history.h" #include "boxes/add_contact_box.h" #include "boxes/confirm_box.h" #include "info/info_memento.h" @@ -33,6 +30,10 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "data/data_peer_values.h" #include "observer_peer.h" #include "apiwrap.h" +#include "styles/style_window.h" +#include "styles/style_dialogs.h" +#include "styles/style_history.h" +#include "styles/style_info.h" namespace HistoryView { diff --git a/Telegram/SourceFiles/info/media/info_media_list_widget.cpp b/Telegram/SourceFiles/info/media/info_media_list_widget.cpp index f099dd7b0..ed5bcd13a 100644 --- a/Telegram/SourceFiles/info/media/info_media_list_widget.cpp +++ b/Telegram/SourceFiles/info/media/info_media_list_widget.cpp @@ -11,6 +11,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "overview/overview_layout.h" #include "history/history_media_types.h" #include "history/history_item.h" +#include "history/history.h" #include "window/themes/window_theme.h" #include "window/window_controller.h" #include "window/window_peer_menu.h" diff --git a/Telegram/SourceFiles/inline_bots/inline_bot_send_data.cpp b/Telegram/SourceFiles/inline_bots/inline_bot_send_data.cpp index 654140fa7..074286e30 100644 --- a/Telegram/SourceFiles/inline_bots/inline_bot_send_data.cpp +++ b/Telegram/SourceFiles/inline_bots/inline_bot_send_data.cpp @@ -11,6 +11,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "inline_bots/inline_bot_result.h" #include "storage/localstorage.h" #include "lang/lang_keys.h" +#include "history/history.h" namespace InlineBots { namespace internal { diff --git a/Telegram/SourceFiles/mainwidget.cpp b/Telegram/SourceFiles/mainwidget.cpp index 55aab289f..a0049c4f0 100644 --- a/Telegram/SourceFiles/mainwidget.cpp +++ b/Telegram/SourceFiles/mainwidget.cpp @@ -35,6 +35,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "apiwrap.h" #include "dialogs/dialogs_widget.h" #include "dialogs/dialogs_key.h" +#include "history/history.h" #include "history/history_widget.h" #include "history/history_message.h" #include "history/history_media.h" @@ -705,7 +706,7 @@ void MainWidget::webPagesOrGamesUpdate() { auto j = items.constFind(App::webPage(webPageId)); if (j != items.cend()) { for_const (auto item, j.value()) { - item->setPendingInitDimensions(); + Auth().data().requestItemViewResize(item); } } } @@ -717,7 +718,7 @@ void MainWidget::webPagesOrGamesUpdate() { auto j = items.constFind(App::game(gameId)); if (j != items.cend()) { for_const (auto item, j.value()) { - item->setPendingInitDimensions(); + Auth().data().requestItemViewResize(item); } } } @@ -4909,7 +4910,7 @@ void MainWidget::feedUpdate(const MTPUpdate &update) { } App::historyUnregItem(local); Auth().messageIdChanging.notify({ local, newId }, true); - local->setId(d.vid.v); + local->setRealId(d.vid.v); App::historyRegItem(local); Auth().data().requestItemRepaint(local); } diff --git a/Telegram/SourceFiles/mainwidget.h b/Telegram/SourceFiles/mainwidget.h index 8d1738b81..ea11e2c11 100644 --- a/Telegram/SourceFiles/mainwidget.h +++ b/Telegram/SourceFiles/mainwidget.h @@ -27,6 +27,8 @@ struct PeerUpdate; namespace Dialogs { struct RowDescriptor; class Row; +class Key; +class IndexedList; } // namespace Dialogs namespace Media { diff --git a/Telegram/SourceFiles/mainwindow.cpp b/Telegram/SourceFiles/mainwindow.cpp index 27039b0b2..cf92c8b4d 100644 --- a/Telegram/SourceFiles/mainwindow.cpp +++ b/Telegram/SourceFiles/mainwindow.cpp @@ -12,6 +12,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "styles/style_dialogs.h" #include "styles/style_window.h" #include "styles/style_boxes.h" +#include "history/history.h" #include "ui/widgets/popup_menu.h" #include "ui/widgets/buttons.h" #include "ui/widgets/shadow.h" diff --git a/Telegram/SourceFiles/media/player/media_player_float.cpp b/Telegram/SourceFiles/media/player/media_player_float.cpp index 9184013e2..822d8236c 100644 --- a/Telegram/SourceFiles/media/player/media_player_float.cpp +++ b/Telegram/SourceFiles/media/player/media_player_float.cpp @@ -11,6 +11,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "data/data_document.h" #include "data/data_session.h" #include "history/history_media.h" +#include "history/history_item.h" #include "media/media_clip_reader.h" #include "media/view/media_clip_playback.h" #include "media/media_audio.h" diff --git a/Telegram/SourceFiles/media/player/media_player_instance.cpp b/Telegram/SourceFiles/media/player/media_player_instance.cpp index 74c65a0b3..2ea7f7daa 100644 --- a/Telegram/SourceFiles/media/player/media_player_instance.cpp +++ b/Telegram/SourceFiles/media/player/media_player_instance.cpp @@ -13,6 +13,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "messenger.h" #include "auth_session.h" #include "calls/calls_instance.h" +#include "history/history.h" +#include "history/history_item.h" #include "history/history_media.h" namespace Media { @@ -97,7 +99,9 @@ void Instance::setCurrent(const AudioMsgId &audioId) { auto history = data->history; auto migrated = data->migrated; - auto item = data->current ? App::histItemById(data->current.contextId()) : nullptr; + auto item = data->current + ? App::histItemById(data->current.contextId()) + : nullptr; if (item) { data->history = item->history()->migrateToOrMe(); data->migrated = data->history->migrateFrom(); diff --git a/Telegram/SourceFiles/media/player/media_player_panel.cpp b/Telegram/SourceFiles/media/player/media_player_panel.cpp index 01ae36851..97889e5ae 100644 --- a/Telegram/SourceFiles/media/player/media_player_panel.cpp +++ b/Telegram/SourceFiles/media/player/media_player_panel.cpp @@ -10,6 +10,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "media/player/media_player_cover.h" #include "media/player/media_player_instance.h" #include "info/media/info_media_list_widget.h" +#include "history/history.h" +#include "history/history_item.h" #include "history/history_media.h" #include "data/data_document.h" #include "ui/widgets/shadow.h" diff --git a/Telegram/SourceFiles/media/view/media_view_group_thumbs.cpp b/Telegram/SourceFiles/media/view/media_view_group_thumbs.cpp index c132972b9..d5f393f58 100644 --- a/Telegram/SourceFiles/media/view/media_view_group_thumbs.cpp +++ b/Telegram/SourceFiles/media/view/media_view_group_thumbs.cpp @@ -11,6 +11,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "data/data_user_photos.h" #include "data/data_photo.h" #include "data/data_document.h" +#include "history/history.h" #include "history/history_media.h" #include "styles/style_mediaview.h" diff --git a/Telegram/SourceFiles/mediaview.cpp b/Telegram/SourceFiles/mediaview.cpp index 70b06db17..027413c07 100644 --- a/Telegram/SourceFiles/mediaview.cpp +++ b/Telegram/SourceFiles/mediaview.cpp @@ -19,6 +19,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "media/view/media_clip_controller.h" #include "media/view/media_view_group_thumbs.h" #include "media/media_audio.h" +#include "history/history.h" #include "history/history_message.h" #include "history/history_media_types.h" #include "window/themes/window_theme_preview.h" @@ -915,7 +916,7 @@ void MediaView::clipCallback(Media::Clip::Notification notification) { } PeerData *MediaView::ui_getPeerForMouseAction() { - return _history ? _history->peer : nullptr; + return _history ? _history->peer.get() : nullptr; } void MediaView::onDownload() { diff --git a/Telegram/SourceFiles/messenger.cpp b/Telegram/SourceFiles/messenger.cpp index 98ea9e02b..83563c925 100644 --- a/Telegram/SourceFiles/messenger.cpp +++ b/Telegram/SourceFiles/messenger.cpp @@ -14,6 +14,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "storage/localstorage.h" #include "platform/platform_specific.h" #include "mainwindow.h" +#include "dialogs/dialogs_entry.h" +#include "history/history.h" #include "application.h" #include "shortcuts.h" #include "auth_session.h" diff --git a/Telegram/SourceFiles/mtproto/type_utils.h b/Telegram/SourceFiles/mtproto/type_utils.h index 8a9cc905c..b30e04a10 100644 --- a/Telegram/SourceFiles/mtproto/type_utils.h +++ b/Telegram/SourceFiles/mtproto/type_utils.h @@ -41,20 +41,20 @@ enum class MTPDmessage_ClientFlag : uint32 { // message is a group migrate (group -> supergroup) service message f_is_group_migrate = (1U << 29), - // message needs initDimensions() + resize() + paint() - f_pending_init_dimensions = (1U << 28), + //// message needs initDimensions() + resize() + paint() + //f_pending_init_dimensions = (1U << 28), - // message needs resize() + paint() - f_pending_resize = (1U << 27), + //// message needs resize() + paint() + //f_pending_resize = (1U << 27), - // message needs paint() - f_pending_paint = (1U << 26), + //// message needs paint() + //f_pending_paint = (1U << 26), - // message is attached to previous one when displaying the history - f_attach_to_previous = (1U << 25), + //// message is attached to previous one when displaying the history + //f_attach_to_previous = (1U << 25), - // message is attached to next one when displaying the history - f_attach_to_next = (1U << 24), + //// message is attached to next one when displaying the history + //f_attach_to_next = (1U << 24), // message was sent from inline bot, need to re-set media when sent f_from_inline_bot = (1U << 23), diff --git a/Telegram/SourceFiles/overview/overview_layout.cpp b/Telegram/SourceFiles/overview/overview_layout.cpp index cf1180cd6..2e916100e 100644 --- a/Telegram/SourceFiles/overview/overview_layout.cpp +++ b/Telegram/SourceFiles/overview/overview_layout.cpp @@ -9,6 +9,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "data/data_document.h" #include "data/data_session.h" +#include "data/data_web_page.h" #include "styles/style_overview.h" #include "styles/style_history.h" #include "core/file_utilities.h" diff --git a/Telegram/SourceFiles/overview/overview_layout.h b/Telegram/SourceFiles/overview/overview_layout.h index cb9e22245..fbe29e128 100644 --- a/Telegram/SourceFiles/overview/overview_layout.h +++ b/Telegram/SourceFiles/overview/overview_layout.h @@ -14,6 +14,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL struct HistoryTextState; struct HistoryStateRequest; +class HistoryMedia; namespace style { struct RoundCheckbox; diff --git a/Telegram/SourceFiles/platform/win/main_window_win.cpp b/Telegram/SourceFiles/platform/win/main_window_win.cpp index 83e727ff7..93bc5c4ae 100644 --- a/Telegram/SourceFiles/platform/win/main_window_win.cpp +++ b/Telegram/SourceFiles/platform/win/main_window_win.cpp @@ -18,6 +18,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "storage/localstorage.h" #include "ui/widgets/popup_menu.h" #include "window/themes/window_theme.h" +#include "history/history.h" #include diff --git a/Telegram/SourceFiles/platform/win/notifications_manager_win.cpp b/Telegram/SourceFiles/platform/win/notifications_manager_win.cpp index 19e2594be..9e994f539 100644 --- a/Telegram/SourceFiles/platform/win/notifications_manager_win.cpp +++ b/Telegram/SourceFiles/platform/win/notifications_manager_win.cpp @@ -11,6 +11,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "platform/win/windows_app_user_model_id.h" #include "platform/win/windows_event_filter.h" #include "platform/win/windows_dlls.h" +#include "history/history.h" #include "mainwindow.h" #include diff --git a/Telegram/SourceFiles/profile/profile_channel_controllers.cpp b/Telegram/SourceFiles/profile/profile_channel_controllers.cpp index c7cebd2e3..47d9cde2d 100644 --- a/Telegram/SourceFiles/profile/profile_channel_controllers.cpp +++ b/Telegram/SourceFiles/profile/profile_channel_controllers.cpp @@ -23,6 +23,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "data/data_session.h" #include "ui/widgets/popup_menu.h" #include "window/window_controller.h" +#include "history/history.h" namespace Profile { namespace { diff --git a/Telegram/SourceFiles/settings/settings_privacy_controllers.cpp b/Telegram/SourceFiles/settings/settings_privacy_controllers.cpp index 7c879a258..99decaad1 100644 --- a/Telegram/SourceFiles/settings/settings_privacy_controllers.cpp +++ b/Telegram/SourceFiles/settings/settings_privacy_controllers.cpp @@ -13,6 +13,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "mainwidget.h" #include "auth_session.h" #include "storage/localstorage.h" +#include "history/history.h" #include "boxes/peer_list_controllers.h" #include "boxes/confirm_box.h" diff --git a/Telegram/SourceFiles/storage/localstorage.cpp b/Telegram/SourceFiles/storage/localstorage.cpp index 285d838fe..415800ca4 100644 --- a/Telegram/SourceFiles/storage/localstorage.cpp +++ b/Telegram/SourceFiles/storage/localstorage.cpp @@ -27,6 +27,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "window/window_controller.h" #include "base/flags.h" #include "data/data_session.h" +#include "history/history.h" #include diff --git a/Telegram/SourceFiles/ui/images.cpp b/Telegram/SourceFiles/ui/images.cpp index de9af7a94..b29f6eeb8 100644 --- a/Telegram/SourceFiles/ui/images.cpp +++ b/Telegram/SourceFiles/ui/images.cpp @@ -12,6 +12,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "platform/platform_specific.h" #include "auth_session.h" #include "history/history_item.h" +#include "history/history.h" namespace Images { namespace { diff --git a/Telegram/SourceFiles/ui/text/text_entity.h b/Telegram/SourceFiles/ui/text/text_entity.h index 86f34d682..559f78128 100644 --- a/Telegram/SourceFiles/ui/text/text_entity.h +++ b/Telegram/SourceFiles/ui/text/text_entity.h @@ -123,6 +123,31 @@ enum { TextInstagramHashtags = 0x800, }; +struct TextWithTags { + struct Tag { + int offset, length; + QString id; + }; + using Tags = QVector; + + QString text; + Tags tags; +}; + +inline bool operator==(const TextWithTags::Tag &a, const TextWithTags::Tag &b) { + return (a.offset == b.offset) && (a.length == b.length) && (a.id == b.id); +} +inline bool operator!=(const TextWithTags::Tag &a, const TextWithTags::Tag &b) { + return !(a == b); +} + +inline bool operator==(const TextWithTags &a, const TextWithTags &b) { + return (a.text == b.text) && (a.tags == b.tags); +} +inline bool operator!=(const TextWithTags &a, const TextWithTags &b) { + return !(a == b); +} + // Parsing helpers. namespace TextUtilities { diff --git a/Telegram/SourceFiles/ui/text_options.cpp b/Telegram/SourceFiles/ui/text_options.cpp index fc5a817bd..79aa86d99 100644 --- a/Telegram/SourceFiles/ui/text_options.cpp +++ b/Telegram/SourceFiles/ui/text_options.cpp @@ -8,6 +8,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "ui/text_options.h" #include "styles/style_window.h" +#include "history/history.h" #include "history/history_item.h" namespace Ui { diff --git a/Telegram/SourceFiles/window/main_window.cpp b/Telegram/SourceFiles/window/main_window.cpp index c4e85d97c..e758a8d6a 100644 --- a/Telegram/SourceFiles/window/main_window.cpp +++ b/Telegram/SourceFiles/window/main_window.cpp @@ -10,6 +10,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "storage/localstorage.h" #include "styles/style_window.h" #include "platform/platform_window_title.h" +#include "history/history.h" #include "window/themes/window_theme.h" #include "window/window_controller.h" #include "mediaview.h" diff --git a/Telegram/SourceFiles/window/notifications_manager.cpp b/Telegram/SourceFiles/window/notifications_manager.cpp index d8d44f6a6..f1a33a005 100644 --- a/Telegram/SourceFiles/window/notifications_manager.cpp +++ b/Telegram/SourceFiles/window/notifications_manager.cpp @@ -11,6 +11,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "window/notifications_manager_default.h" #include "media/media_audio_track.h" #include "media/media_audio.h" +#include "history/history.h" #include "history/history_item_components.h" #include "lang/lang_keys.h" #include "mainwindow.h" @@ -56,7 +57,9 @@ void System::createManager() { void System::schedule(History *history, HistoryItem *item) { if (App::quitting() || !history->currentNotification() || !AuthSession::Exists()) return; - auto notifyByFrom = (!history->peer->isUser() && item->mentionsMe()) ? item->from() : nullptr; + auto notifyByFrom = (!history->peer->isUser() && item->mentionsMe()) + ? item->from().get() + : nullptr; if (item->isSilent()) { history->popNotification(item); diff --git a/Telegram/SourceFiles/window/notifications_manager_default.cpp b/Telegram/SourceFiles/window/notifications_manager_default.cpp index 09f4027b3..ef9183cee 100644 --- a/Telegram/SourceFiles/window/notifications_manager_default.cpp +++ b/Telegram/SourceFiles/window/notifications_manager_default.cpp @@ -20,6 +20,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "styles/style_window.h" #include "storage/file_download.h" #include "auth_session.h" +#include "history/history.h" #include "history/history_item.h" #include "platform/platform_specific.h" @@ -69,7 +70,7 @@ Manager::QueuedNotification::QueuedNotification( , int forwardedCount) : history(item->history()) , peer(history->peer) -, author((!peer->isUser() && !item->isPost()) ? item->author() : nullptr) +, author((!peer->isUser() && !item->isPost()) ? item->author().get() : nullptr) , item((forwardedCount < 2) ? item.get() : nullptr) , forwardedCount(forwardedCount) { } diff --git a/Telegram/SourceFiles/window/window_controller.cpp b/Telegram/SourceFiles/window/window_controller.cpp index aea7fe54d..70c39d050 100644 --- a/Telegram/SourceFiles/window/window_controller.cpp +++ b/Telegram/SourceFiles/window/window_controller.cpp @@ -12,6 +12,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "history/view/history_view_element.h" #include "history/view/history_view_message.h" #include "history/view/history_view_service_message.h" +#include "history/history.h" #include "history/history_item.h" #include "mainwidget.h" #include "mainwindow.h" diff --git a/Telegram/SourceFiles/window/window_peer_menu.cpp b/Telegram/SourceFiles/window/window_peer_menu.cpp index 557513232..95159458b 100644 --- a/Telegram/SourceFiles/window/window_peer_menu.cpp +++ b/Telegram/SourceFiles/window/window_peer_menu.cpp @@ -23,6 +23,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "mainwindow.h" #include "observer_peer.h" #include "styles/style_boxes.h" +#include "history/history.h" #include "window/window_controller.h" #include "data/data_session.h" #include "data/data_feed.h" diff --git a/Telegram/gyp/telegram_sources.txt b/Telegram/gyp/telegram_sources.txt index 4dfcc1019..f16ec59b4 100644 --- a/Telegram/gyp/telegram_sources.txt +++ b/Telegram/gyp/telegram_sources.txt @@ -233,6 +233,7 @@ <(src_loc)/history/view/history_view_list_widget.h <(src_loc)/history/view/history_view_message.cpp <(src_loc)/history/view/history_view_message.h +<(src_loc)/history/view/history_view_object.h <(src_loc)/history/view/history_view_service_message.cpp <(src_loc)/history/view/history_view_service_message.h <(src_loc)/history/view/history_view_top_bar_widget.cpp