mirror of
https://github.com/AyuGram/AyuGramDesktop.git
synced 2025-04-11 11:47:09 +02:00
305 lines
8.5 KiB
C++
305 lines
8.5 KiB
C++
/*
|
|
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
|
|
*/
|
|
#include "info/media/info_media_buttons.h"
|
|
|
|
#include "base/call_delayed.h"
|
|
#include "base/qt/qt_key_modifiers.h"
|
|
#include "core/application.h"
|
|
#include "data/data_channel.h"
|
|
#include "data/data_saved_messages.h"
|
|
#include "data/data_session.h"
|
|
#include "data/data_stories_ids.h"
|
|
#include "data/data_user.h"
|
|
#include "history/view/history_view_sublist_section.h"
|
|
#include "info/info_controller.h"
|
|
#include "info/info_memento.h"
|
|
#include "info/profile/info_profile_values.h"
|
|
#include "info/stories/info_stories_widget.h"
|
|
#include "ui/widgets/buttons.h"
|
|
#include "ui/widgets/popup_menu.h"
|
|
#include "ui/wrap/slide_wrap.h"
|
|
#include "ui/wrap/vertical_layout.h"
|
|
#include "window/window_separate_id.h"
|
|
#include "window/window_session_controller.h"
|
|
#include "styles/style_info.h"
|
|
#include "styles/style_menu_icons.h"
|
|
|
|
namespace Info::Media {
|
|
namespace {
|
|
|
|
[[nodiscard]] Window::SeparateSharedMediaType ToSeparateType(
|
|
Storage::SharedMediaType type) {
|
|
using Type = Storage::SharedMediaType;
|
|
using SeparatedType = Window::SeparateSharedMediaType;
|
|
return (type == Type::Photo)
|
|
? SeparatedType::Photos
|
|
: (type == Type::Video)
|
|
? SeparatedType::Videos
|
|
: (type == Type::File)
|
|
? SeparatedType::Files
|
|
: (type == Type::MusicFile)
|
|
? SeparatedType::Audio
|
|
: (type == Type::Link)
|
|
? SeparatedType::Links
|
|
: (type == Type::RoundVoiceFile)
|
|
? SeparatedType::Voices
|
|
: (type == Type::GIF)
|
|
? SeparatedType::GIF
|
|
: SeparatedType::None;
|
|
}
|
|
|
|
[[nodiscard]] Window::SeparateId SeparateId(
|
|
not_null<PeerData*> peer,
|
|
MsgId topicRootId,
|
|
Storage::SharedMediaType type) {
|
|
if (peer->isSelf()) {
|
|
return { nullptr };
|
|
}
|
|
const auto separateType = ToSeparateType(type);
|
|
if (separateType == Window::SeparateSharedMediaType::None) {
|
|
return { nullptr };
|
|
}
|
|
return { Window::SeparateSharedMedia(separateType, peer, topicRootId) };
|
|
}
|
|
|
|
void AddContextMenuToButton(
|
|
not_null<Ui::AbstractButton*> button,
|
|
Fn<void()> openInWindow) {
|
|
if (!openInWindow) {
|
|
return;
|
|
}
|
|
button->setAcceptBoth();
|
|
struct State final {
|
|
base::unique_qptr<Ui::PopupMenu> menu;
|
|
};
|
|
const auto state = button->lifetime().make_state<State>();
|
|
button->addClickHandler([=](Qt::MouseButton mouse) {
|
|
if (mouse != Qt::RightButton) {
|
|
return;
|
|
}
|
|
state->menu = base::make_unique_q<Ui::PopupMenu>(
|
|
button.get(),
|
|
st::popupMenuWithIcons);
|
|
state->menu->addAction(tr::lng_context_new_window(tr::now), [=] {
|
|
base::call_delayed(
|
|
st::popupMenuWithIcons.showDuration,
|
|
crl::guard(button, openInWindow));
|
|
}, &st::menuIconNewWindow);
|
|
state->menu->popup(QCursor::pos());
|
|
});
|
|
}
|
|
|
|
} // namespace
|
|
|
|
tr::phrase<lngtag_count> MediaTextPhrase(Type type) {
|
|
switch (type) {
|
|
case Type::Photo: return tr::lng_profile_photos;
|
|
case Type::GIF: return tr::lng_profile_gifs;
|
|
case Type::Video: return tr::lng_profile_videos;
|
|
case Type::File: return tr::lng_profile_files;
|
|
case Type::MusicFile: return tr::lng_profile_songs;
|
|
case Type::Link: return tr::lng_profile_shared_links;
|
|
case Type::RoundVoiceFile: return tr::lng_profile_audios;
|
|
}
|
|
Unexpected("Type in MediaTextPhrase()");
|
|
};
|
|
|
|
Fn<QString(int)> MediaText(Type type) {
|
|
return [phrase = MediaTextPhrase(type)](int count) {
|
|
return phrase(tr::now, lt_count, count);
|
|
};
|
|
}
|
|
|
|
not_null<Ui::SlideWrap<Ui::SettingsButton>*> AddCountedButton(
|
|
Ui::VerticalLayout *parent,
|
|
rpl::producer<int> &&count,
|
|
Fn<QString(int)> &&textFromCount,
|
|
Ui::MultiSlideTracker &tracker) {
|
|
using namespace ::Settings;
|
|
auto forked = std::move(count)
|
|
| start_spawning(parent->lifetime());
|
|
auto text = rpl::duplicate(
|
|
forked
|
|
) | rpl::map([textFromCount](int count) {
|
|
return (count > 0)
|
|
? textFromCount(count)
|
|
: QString();
|
|
});
|
|
auto button = parent->add(object_ptr<Ui::SlideWrap<Ui::SettingsButton>>(
|
|
parent,
|
|
object_ptr<Ui::SettingsButton>(
|
|
parent,
|
|
std::move(text),
|
|
st::infoSharedMediaButton))
|
|
)->setDuration(
|
|
st::infoSlideDuration
|
|
)->toggleOn(
|
|
rpl::duplicate(forked) | rpl::map(rpl::mappers::_1 > 0)
|
|
);
|
|
tracker.track(button);
|
|
return button;
|
|
};
|
|
|
|
not_null<Ui::SettingsButton*> AddButton(
|
|
Ui::VerticalLayout *parent,
|
|
not_null<Window::SessionNavigation*> navigation,
|
|
not_null<PeerData*> peer,
|
|
MsgId topicRootId,
|
|
PeerData *migrated,
|
|
Type type,
|
|
Ui::MultiSlideTracker &tracker) {
|
|
auto result = AddCountedButton(
|
|
parent,
|
|
Profile::SharedMediaCountValue(peer, topicRootId, migrated, type),
|
|
MediaText(type),
|
|
tracker)->entity();
|
|
const auto separateId = SeparateId(peer, topicRootId, type);
|
|
const auto openInWindow = separateId
|
|
? [=] { navigation->parentController()->showInNewWindow(separateId); }
|
|
: Fn<void()>(nullptr);
|
|
AddContextMenuToButton(result, openInWindow);
|
|
result->addClickHandler([=](Qt::MouseButton mouse) {
|
|
if (mouse == Qt::RightButton) {
|
|
return;
|
|
}
|
|
if (openInWindow
|
|
&& (base::IsCtrlPressed() || mouse == Qt::MiddleButton)) {
|
|
return openInWindow();
|
|
}
|
|
const auto topic = topicRootId
|
|
? peer->forumTopicFor(topicRootId)
|
|
: nullptr;
|
|
if (topicRootId && !topic) {
|
|
return;
|
|
}
|
|
const auto separateId = SeparateId(peer, topicRootId, type);
|
|
if (Core::App().separateWindowFor(separateId) && openInWindow) {
|
|
openInWindow();
|
|
} else {
|
|
navigation->showSection(topicRootId
|
|
? std::make_shared<Info::Memento>(topic, Section(type))
|
|
: std::make_shared<Info::Memento>(peer, Section(type)));
|
|
}
|
|
});
|
|
return result;
|
|
};
|
|
|
|
not_null<Ui::SettingsButton*> AddCommonGroupsButton(
|
|
Ui::VerticalLayout *parent,
|
|
not_null<Window::SessionNavigation*> navigation,
|
|
not_null<UserData*> user,
|
|
Ui::MultiSlideTracker &tracker) {
|
|
auto result = AddCountedButton(
|
|
parent,
|
|
Profile::CommonGroupsCountValue(user),
|
|
[](int count) {
|
|
return tr::lng_profile_common_groups(tr::now, lt_count, count);
|
|
},
|
|
tracker)->entity();
|
|
result->addClickHandler([=] {
|
|
navigation->showSection(
|
|
std::make_shared<Info::Memento>(
|
|
user,
|
|
Section::Type::CommonGroups));
|
|
});
|
|
return result;
|
|
}
|
|
|
|
not_null<Ui::SettingsButton*> AddSimilarPeersButton(
|
|
Ui::VerticalLayout *parent,
|
|
not_null<Window::SessionNavigation*> navigation,
|
|
not_null<PeerData*> peer,
|
|
Ui::MultiSlideTracker &tracker) {
|
|
auto result = AddCountedButton(
|
|
parent,
|
|
Profile::SimilarPeersCountValue(peer),
|
|
[=](int count) {
|
|
return peer->isBroadcast()
|
|
? tr::lng_profile_similar_channels(tr::now, lt_count, count)
|
|
: tr::lng_profile_similar_bots(tr::now, lt_count, count);
|
|
},
|
|
tracker)->entity();
|
|
result->addClickHandler([=] {
|
|
navigation->showSection(
|
|
std::make_shared<Info::Memento>(
|
|
peer,
|
|
Section::Type::SimilarPeers));
|
|
});
|
|
return result;
|
|
}
|
|
|
|
not_null<Ui::SettingsButton*> AddStoriesButton(
|
|
Ui::VerticalLayout *parent,
|
|
not_null<Window::SessionNavigation*> navigation,
|
|
not_null<PeerData*> peer,
|
|
Ui::MultiSlideTracker &tracker) {
|
|
auto count = rpl::single(0) | rpl::then(Data::SavedStoriesIds(
|
|
peer,
|
|
ServerMaxStoryId - 1,
|
|
0
|
|
) | rpl::map([](const Data::StoriesIdsSlice &slice) {
|
|
return slice.fullCount().value_or(0);
|
|
}));
|
|
const auto phrase = peer->isChannel() ? (+[](int count) {
|
|
return tr::lng_profile_posts(tr::now, lt_count, count);
|
|
}) : (+[](int count) {
|
|
return tr::lng_profile_saved_stories(tr::now, lt_count, count);
|
|
});
|
|
auto result = AddCountedButton(
|
|
parent,
|
|
std::move(count),
|
|
phrase,
|
|
tracker)->entity();
|
|
result->addClickHandler([=] {
|
|
navigation->showSection(Info::Stories::Make(peer));
|
|
});
|
|
return result;
|
|
}
|
|
|
|
not_null<Ui::SettingsButton*> AddSavedSublistButton(
|
|
Ui::VerticalLayout *parent,
|
|
not_null<Window::SessionNavigation*> navigation,
|
|
not_null<PeerData*> peer,
|
|
Ui::MultiSlideTracker &tracker) {
|
|
auto result = AddCountedButton(
|
|
parent,
|
|
Profile::SavedSublistCountValue(peer),
|
|
[](int count) {
|
|
return tr::lng_profile_saved_messages(tr::now, lt_count, count);
|
|
},
|
|
tracker)->entity();
|
|
result->addClickHandler([=] {
|
|
navigation->showSection(
|
|
std::make_shared<HistoryView::SublistMemento>(
|
|
peer->owner().savedMessages().sublist(peer)));
|
|
});
|
|
return result;
|
|
}
|
|
|
|
not_null<Ui::SettingsButton*> AddPeerGiftsButton(
|
|
Ui::VerticalLayout *parent,
|
|
not_null<Window::SessionNavigation*> navigation,
|
|
not_null<PeerData*> peer,
|
|
Ui::MultiSlideTracker &tracker) {
|
|
auto result = AddCountedButton(
|
|
parent,
|
|
Profile::PeerGiftsCountValue(peer),
|
|
[](int count) {
|
|
return tr::lng_profile_peer_gifts(tr::now, lt_count, count);
|
|
},
|
|
tracker)->entity();
|
|
result->addClickHandler([=] {
|
|
navigation->showSection(
|
|
std::make_shared<Info::Memento>(
|
|
peer,
|
|
Section::Type::PeerGifts));
|
|
});
|
|
return result;
|
|
}
|
|
|
|
} // namespace Info::Media
|