335 lines
8.6 KiB
C++
335 lines
8.6 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 "ui/chat/choose_send_as.h"
|
|
|
|
#include "boxes/peer_list_box.h"
|
|
#include "data/data_peer.h"
|
|
#include "data/data_channel.h"
|
|
#include "data/data_peer_values.h"
|
|
#include "history/history.h"
|
|
#include "ui/controls/send_as_button.h"
|
|
#include "ui/text/text_utilities.h"
|
|
#include "ui/toast/toast.h"
|
|
#include "ui/painter.h"
|
|
#include "window/window_session_controller.h"
|
|
#include "main/main_session.h"
|
|
#include "main/session/send_as_peers.h"
|
|
#include "lang/lang_keys.h"
|
|
#include "settings/settings_premium.h"
|
|
#include "styles/style_calls.h"
|
|
#include "styles/style_boxes.h"
|
|
#include "styles/style_chat.h"
|
|
#include "styles/style_chat_helpers.h"
|
|
|
|
namespace Ui {
|
|
namespace {
|
|
|
|
class Row final : public PeerListRow {
|
|
public:
|
|
explicit Row(const Main::SendAsPeer &sendAsPeer);
|
|
|
|
int paintNameIconGetWidth(
|
|
Painter &p,
|
|
Fn<void()> repaint,
|
|
crl::time now,
|
|
int nameLeft,
|
|
int nameTop,
|
|
int nameWidth,
|
|
int availableWidth,
|
|
int outerWidth,
|
|
bool selected) override;
|
|
|
|
private:
|
|
bool _premiumRequired = false;
|
|
|
|
};
|
|
|
|
class ListController final : public PeerListController {
|
|
public:
|
|
ListController(
|
|
std::vector<Main::SendAsPeer> list,
|
|
not_null<PeerData*> selected);
|
|
|
|
Main::Session &session() const override;
|
|
void prepare() override;
|
|
void rowClicked(not_null<PeerListRow*> row) override;
|
|
|
|
[[nodiscard]] rpl::producer<not_null<PeerData*>> clicked() const;
|
|
|
|
private:
|
|
std::unique_ptr<Row> createRow(const Main::SendAsPeer &sendAsPeer);
|
|
|
|
std::vector<Main::SendAsPeer> _list;
|
|
not_null<PeerData*> _selected;
|
|
rpl::event_stream<not_null<PeerData*>> _clicked;
|
|
|
|
};
|
|
|
|
Row::Row(const Main::SendAsPeer &sendAsPeer)
|
|
: PeerListRow(sendAsPeer.peer)
|
|
, _premiumRequired(sendAsPeer.premiumRequired) {
|
|
}
|
|
|
|
int Row::paintNameIconGetWidth(
|
|
Painter &p,
|
|
Fn<void()> repaint,
|
|
crl::time now,
|
|
int nameLeft,
|
|
int nameTop,
|
|
int nameWidth,
|
|
int availableWidth,
|
|
int outerWidth,
|
|
bool selected) {
|
|
if (_premiumRequired && !peer()->session().premium()) {
|
|
const auto &icon = st::emojiPremiumRequired;
|
|
availableWidth -= icon.width();
|
|
const auto x = nameLeft + std::min(nameWidth, availableWidth);
|
|
icon.paint(p, x, nameTop, outerWidth);
|
|
return icon.width();
|
|
}
|
|
return PeerListRow::paintNameIconGetWidth(
|
|
p,
|
|
std::move(repaint),
|
|
now,
|
|
nameLeft,
|
|
nameTop,
|
|
nameWidth,
|
|
availableWidth,
|
|
outerWidth,
|
|
selected);
|
|
}
|
|
|
|
ListController::ListController(
|
|
std::vector<Main::SendAsPeer> list,
|
|
not_null<PeerData*> selected)
|
|
: PeerListController()
|
|
, _list(std::move(list))
|
|
, _selected(selected) {
|
|
Data::AmPremiumValue(
|
|
&selected->session()
|
|
) | rpl::skip(1) | rpl::start_with_next([=] {
|
|
const auto count = delegate()->peerListFullRowsCount();
|
|
for (auto i = 0; i != count; ++i) {
|
|
delegate()->peerListUpdateRow(
|
|
delegate()->peerListRowAt(i));
|
|
}
|
|
}, lifetime());
|
|
}
|
|
|
|
Main::Session &ListController::session() const {
|
|
return _selected->session();
|
|
}
|
|
|
|
std::unique_ptr<Row> ListController::createRow(
|
|
const Main::SendAsPeer &sendAsPeer) {
|
|
auto result = std::make_unique<Row>(sendAsPeer);
|
|
if (sendAsPeer.peer->isSelf()) {
|
|
result->setCustomStatus(
|
|
tr::lng_group_call_join_as_personal(tr::now));
|
|
} else if (sendAsPeer.peer->isMegagroup()) {
|
|
result->setCustomStatus(tr::lng_send_as_anonymous_admin(tr::now));
|
|
} else if (const auto channel = sendAsPeer.peer->asChannel()) {
|
|
result->setCustomStatus(tr::lng_chat_status_subscribers(
|
|
tr::now,
|
|
lt_count,
|
|
channel->membersCount()));
|
|
}
|
|
return result;
|
|
}
|
|
|
|
void ListController::prepare() {
|
|
delegate()->peerListSetSearchMode(PeerListSearchMode::Disabled);
|
|
for (const auto &sendAsPeer : _list) {
|
|
auto row = createRow(sendAsPeer);
|
|
const auto raw = row.get();
|
|
delegate()->peerListAppendRow(std::move(row));
|
|
if (sendAsPeer.peer == _selected) {
|
|
delegate()->peerListSetRowChecked(raw, true);
|
|
raw->finishCheckedAnimation();
|
|
}
|
|
}
|
|
delegate()->peerListRefreshRows();
|
|
}
|
|
|
|
void ListController::rowClicked(not_null<PeerListRow*> row) {
|
|
const auto peer = row->peer();
|
|
if (peer == _selected) {
|
|
return;
|
|
}
|
|
_clicked.fire_copy(peer);
|
|
}
|
|
|
|
rpl::producer<not_null<PeerData*>> ListController::clicked() const {
|
|
return _clicked.events();
|
|
}
|
|
|
|
void ShowPremiumPromoToast(not_null<Window::SessionController*> controller) {
|
|
using WeakToast = base::weak_ptr<Ui::Toast::Instance>;
|
|
const auto toast = std::make_shared<WeakToast>();
|
|
|
|
auto link = Ui::Text::Link(
|
|
tr::lng_send_as_premium_required_link(tr::now));
|
|
link.entities.push_back(
|
|
EntityInText(EntityType::Semibold, 0, link.text.size()));
|
|
const auto config = Ui::Toast::Config{
|
|
.text = tr::lng_send_as_premium_required(
|
|
tr::now,
|
|
lt_link,
|
|
link,
|
|
Ui::Text::WithEntities),
|
|
.st = &st::defaultMultilineToast,
|
|
.durationMs = Ui::Toast::kDefaultDuration * 2,
|
|
.multiline = true,
|
|
.filter = crl::guard(&controller->session(), [=](
|
|
const ClickHandlerPtr &,
|
|
Qt::MouseButton button) {
|
|
if (button == Qt::LeftButton) {
|
|
if (const auto strong = toast->get()) {
|
|
strong->hideAnimated();
|
|
(*toast) = nullptr;
|
|
Settings::ShowPremium(controller, "send_as");
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}),
|
|
};
|
|
(*toast) = Ui::Toast::Show(
|
|
Window::Show(controller).toastParent(),
|
|
config);
|
|
}
|
|
|
|
} // namespace
|
|
|
|
void ChooseSendAsBox(
|
|
not_null<GenericBox*> box,
|
|
std::vector<Main::SendAsPeer> list,
|
|
not_null<PeerData*> chosen,
|
|
Fn<bool(not_null<PeerData*>)> done) {
|
|
Expects(ranges::contains(list, chosen, &Main::SendAsPeer::peer));
|
|
Expects(done != nullptr);
|
|
|
|
box->setWidth(st::groupCallJoinAsWidth);
|
|
box->setTitle(tr::lng_send_as_title());
|
|
const auto &labelSt = st::confirmPhoneAboutLabel;
|
|
box->addRow(object_ptr<Ui::FlatLabel>(
|
|
box,
|
|
tr::lng_group_call_join_as_about(),
|
|
labelSt));
|
|
|
|
auto &lifetime = box->lifetime();
|
|
const auto delegate = lifetime.make_state<
|
|
PeerListContentDelegateSimple
|
|
>();
|
|
const auto controller = lifetime.make_state<ListController>(
|
|
list,
|
|
chosen);
|
|
controller->setStyleOverrides(
|
|
&st::peerListJoinAsList,
|
|
nullptr);
|
|
|
|
controller->clicked(
|
|
) | rpl::start_with_next([=](not_null<PeerData*> peer) {
|
|
const auto weak = MakeWeak(box);
|
|
if (done(peer) && weak) {
|
|
box->closeBox();
|
|
}
|
|
}, box->lifetime());
|
|
|
|
const auto content = box->addRow(
|
|
object_ptr<PeerListContent>(box, controller),
|
|
style::margins());
|
|
delegate->setContent(content);
|
|
controller->setDelegate(delegate);
|
|
box->addButton(tr::lng_box_done(), [=] { box->closeBox(); });
|
|
}
|
|
|
|
void SetupSendAsButton(
|
|
not_null<SendAsButton*> button,
|
|
rpl::producer<PeerData*> active,
|
|
not_null<Window::SessionController*> window) {
|
|
using namespace rpl::mappers;
|
|
const auto current = button->lifetime().make_state<
|
|
rpl::variable<PeerData*>
|
|
>(std::move(active));
|
|
button->setClickedCallback([=] {
|
|
const auto peer = current->current();
|
|
if (!peer) {
|
|
return;
|
|
}
|
|
const auto session = &peer->session();
|
|
const auto &list = session->sendAsPeers().list(peer);
|
|
if (list.size() < 2) {
|
|
return;
|
|
}
|
|
const auto done = [=](not_null<PeerData*> sendAs) {
|
|
const auto i = ranges::find(
|
|
list,
|
|
sendAs,
|
|
&Main::SendAsPeer::peer);
|
|
if (i != end(list)
|
|
&& i->premiumRequired
|
|
&& !sendAs->session().premium()) {
|
|
ShowPremiumPromoToast(window);
|
|
return false;
|
|
}
|
|
session->sendAsPeers().saveChosen(peer, sendAs);
|
|
return true;
|
|
};
|
|
window->show(Box(
|
|
Ui::ChooseSendAsBox,
|
|
list,
|
|
session->sendAsPeers().resolveChosen(peer),
|
|
done));
|
|
});
|
|
|
|
auto userpic = current->value(
|
|
) | rpl::filter([=](PeerData *peer) {
|
|
return peer && peer->isMegagroup();
|
|
}) | rpl::map([=](not_null<PeerData*> peer) {
|
|
const auto channel = peer->asMegagroup();
|
|
|
|
auto updates = rpl::single(
|
|
rpl::empty
|
|
) | rpl::then(channel->session().sendAsPeers().updated(
|
|
) | rpl::filter(
|
|
_1 == channel
|
|
) | rpl::to_empty);
|
|
|
|
return rpl::combine(
|
|
std::move(updates),
|
|
channel->adminRightsValue()
|
|
) | rpl::map([=] {
|
|
return channel->session().sendAsPeers().resolveChosen(channel);
|
|
}) | rpl::distinct_until_changed(
|
|
) | rpl::map([=](not_null<PeerData*> chosen) {
|
|
return Data::PeerUserpicImageValue(
|
|
chosen,
|
|
st::sendAsButton.size * style::DevicePixelRatio());
|
|
}) | rpl::flatten_latest();
|
|
}) | rpl::flatten_latest();
|
|
|
|
std::move(
|
|
userpic
|
|
) | rpl::start_with_next([=](QImage &&userpic) {
|
|
button->setUserpic(std::move(userpic));
|
|
}, button->lifetime());
|
|
}
|
|
|
|
void SetupSendAsButton(
|
|
not_null<SendAsButton*> button,
|
|
not_null<Window::SessionController*> window) {
|
|
auto active = window->activeChatValue(
|
|
) | rpl::map([=](const Dialogs::Key &key) {
|
|
return key.history() ? key.history()->peer.get() : nullptr;
|
|
});
|
|
SetupSendAsButton(button, std::move(active), window);
|
|
}
|
|
|
|
} // namespace Ui
|