tdesktop/Telegram/SourceFiles/settings/business/settings_chatbots.cpp

517 lines
14 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 "settings/business/settings_chatbots.h"
#include "apiwrap.h"
#include "boxes/peers/prepare_short_info_box.h"
#include "boxes/peer_list_box.h"
#include "core/application.h"
#include "data/business/data_business_chatbots.h"
#include "data/data_session.h"
#include "data/data_user.h"
#include "lang/lang_keys.h"
#include "main/main_session.h"
#include "settings/business/settings_recipients_helper.h"
#include "ui/effects/ripple_animation.h"
#include "ui/text/text_utilities.h"
#include "ui/widgets/fields/input_field.h"
#include "ui/widgets/buttons.h"
#include "ui/wrap/slide_wrap.h"
#include "ui/wrap/vertical_layout.h"
#include "ui/painter.h"
#include "ui/vertical_list.h"
#include "window/window_session_controller.h"
#include "styles/style_boxes.h"
#include "styles/style_layers.h"
#include "styles/style_settings.h"
namespace Settings {
namespace {
constexpr auto kDebounceTimeout = crl::time(400);
enum class LookupState {
Empty,
Loading,
Unsupported,
Ready,
};
struct BotState {
UserData *bot = nullptr;
LookupState state = LookupState::Empty;
};
class Chatbots final : public BusinessSection<Chatbots> {
public:
Chatbots(
QWidget *parent,
not_null<Window::SessionController*> controller);
~Chatbots();
[[nodiscard]] bool closeByOutsideClick() const override;
[[nodiscard]] rpl::producer<QString> title() override;
const Ui::RoundRect *bottomSkipRounding() const override {
return &_bottomSkipRounding;
}
private:
void setupContent(not_null<Window::SessionController*> controller);
void save();
Ui::RoundRect _bottomSkipRounding;
rpl::variable<Data::BusinessRecipients> _recipients;
rpl::variable<QString> _usernameValue;
rpl::variable<BotState> _botValue;
rpl::variable<bool> _repliesAllowed = true;
};
class PreviewController final : public PeerListController {
public:
PreviewController(not_null<PeerData*> peer, Fn<void()> resetBot);
void prepare() override;
void loadMoreRows() override;
void rowClicked(not_null<PeerListRow*> row) override;
void rowRightActionClicked(not_null<PeerListRow*> row) override;
Main::Session &session() const override;
private:
const not_null<PeerData*> _peer;
const Fn<void()> _resetBot;
rpl::lifetime _lifetime;
};
class PreviewRow final : public PeerListRow {
public:
using PeerListRow::PeerListRow;
QSize rightActionSize() const override;
QMargins rightActionMargins() const override;
void rightActionPaint(
Painter &p,
int x,
int y,
int outerWidth,
bool selected,
bool actionSelected) override;
void rightActionAddRipple(
QPoint point,
Fn<void()> updateCallback) override;
void rightActionStopLastRipple() override;
private:
std::unique_ptr<Ui::RippleAnimation> _actionRipple;
};
QSize PreviewRow::rightActionSize() const {
return QSize(
st::settingsChatbotsDeleteIcon.width(),
st::settingsChatbotsDeleteIcon.height()) * 2;
}
QMargins PreviewRow::rightActionMargins() const {
const auto itemHeight = st::peerListSingleRow.item.height;
const auto skip = (itemHeight - rightActionSize().height()) / 2;
return QMargins(0, skip, skip, 0);
}
void PreviewRow::rightActionPaint(
Painter &p,
int x,
int y,
int outerWidth,
bool selected,
bool actionSelected) {
if (_actionRipple) {
_actionRipple->paint(
p,
x,
y,
outerWidth);
if (_actionRipple->empty()) {
_actionRipple.reset();
}
}
const auto rect = QRect(QPoint(x, y), PreviewRow::rightActionSize());
(actionSelected
? st::settingsChatbotsDeleteIconOver
: st::settingsChatbotsDeleteIcon).paintInCenter(p, rect);
}
void PreviewRow::rightActionAddRipple(
QPoint point,
Fn<void()> updateCallback) {
if (!_actionRipple) {
auto mask = Ui::RippleAnimation::EllipseMask(rightActionSize());
_actionRipple = std::make_unique<Ui::RippleAnimation>(
st::defaultRippleAnimation,
std::move(mask),
std::move(updateCallback));
}
_actionRipple->add(point);
}
void PreviewRow::rightActionStopLastRipple() {
if (_actionRipple) {
_actionRipple->lastStop();
}
}
PreviewController::PreviewController(
not_null<PeerData*> peer,
Fn<void()> resetBot)
: _peer(peer)
, _resetBot(std::move(resetBot)) {
}
void PreviewController::prepare() {
delegate()->peerListAppendRow(std::make_unique<PreviewRow>(_peer));
delegate()->peerListRefreshRows();
}
void PreviewController::loadMoreRows() {
}
void PreviewController::rowClicked(not_null<PeerListRow*> row) {
}
void PreviewController::rowRightActionClicked(not_null<PeerListRow*> row) {
_resetBot();
}
Main::Session &PreviewController::session() const {
return _peer->session();
}
[[nodiscard]] rpl::producer<QString> DebouncedValue(
not_null<Ui::InputField*> field) {
return [=](auto consumer) {
auto result = rpl::lifetime();
struct State {
base::Timer timer;
QString lastText;
};
const auto state = result.make_state<State>();
const auto push = [=] {
state->timer.cancel();
consumer.put_next_copy(state->lastText);
};
state->timer.setCallback(push);
state->lastText = field->getLastText();
consumer.put_next_copy(field->getLastText());
field->changes() | rpl::start_with_next([=] {
const auto &text = field->getLastText();
const auto was = std::exchange(state->lastText, text);
if (std::abs(int(text.size()) - int(was.size())) == 1) {
state->timer.callOnce(kDebounceTimeout);
} else {
push();
}
}, result);
return result;
};
}
[[nodiscard]] QString ExtractUsername(QString text) {
text = text.trimmed();
static const auto expression = QRegularExpression(
"^(https://)?([a-zA-Z0-9\\.]+/)?([a-zA-Z0-9_\\.]+)");
const auto match = expression.match(text);
return match.hasMatch() ? match.captured(3) : text;
}
[[nodiscard]] rpl::producer<BotState> LookupBot(
not_null<Main::Session*> session,
rpl::producer<QString> usernameChanges) {
using Cache = base::flat_map<QString, UserData*>;
const auto cache = std::make_shared<Cache>();
return std::move(
usernameChanges
) | rpl::map([=](const QString &username) -> rpl::producer<BotState> {
const auto extracted = ExtractUsername(username);
const auto owner = &session->data();
static const auto expression = QRegularExpression(
"^[a-zA-Z0-9_\\.]+$");
if (!expression.match(extracted).hasMatch()) {
return rpl::single(BotState());
} else if (const auto peer = owner->peerByUsername(extracted)) {
if (const auto user = peer->asUser(); user && user->isBot()) {
if (user->botInfo->supportsBusiness) {
return rpl::single(BotState{
.bot = user,
.state = LookupState::Ready,
});
}
return rpl::single(BotState{
.state = LookupState::Unsupported,
});
}
return rpl::single(BotState{
.state = LookupState::Ready,
});
} else if (const auto i = cache->find(extracted); i != end(*cache)) {
return rpl::single(BotState{
.bot = i->second,
.state = LookupState::Ready,
});
}
return [=](auto consumer) {
auto result = rpl::lifetime();
const auto requestId = result.make_state<mtpRequestId>();
*requestId = session->api().request(MTPcontacts_ResolveUsername(
MTP_string(extracted)
)).done([=](const MTPcontacts_ResolvedPeer &result) {
const auto &data = result.data();
session->data().processUsers(data.vusers());
session->data().processChats(data.vchats());
const auto peerId = peerFromMTP(data.vpeer());
const auto peer = session->data().peer(peerId);
if (const auto user = peer->asUser()) {
if (user->isBot()) {
cache->emplace(extracted, user);
consumer.put_next(BotState{
.bot = user,
.state = LookupState::Ready,
});
return;
}
}
cache->emplace(extracted, nullptr);
consumer.put_next(BotState{ .state = LookupState::Ready });
}).fail([=] {
cache->emplace(extracted, nullptr);
consumer.put_next(BotState{ .state = LookupState::Ready });
}).send();
result.add([=] {
session->api().request(*requestId).cancel();
});
return result;
};
}) | rpl::flatten_latest();
}
[[nodiscard]] object_ptr<Ui::RpWidget> MakeBotPreview(
not_null<Ui::RpWidget*> parent,
rpl::producer<BotState> state,
Fn<void()> resetBot) {
auto result = object_ptr<Ui::SlideWrap<>>(
parent.get(),
object_ptr<Ui::RpWidget>(parent.get()));
const auto raw = result.data();
const auto inner = raw->entity();
raw->hide(anim::type::instant);
const auto child = inner->lifetime().make_state<Ui::RpWidget*>(nullptr);
std::move(state) | rpl::filter([=](BotState state) {
return state.state != LookupState::Loading;
}) | rpl::start_with_next([=](BotState state) {
raw->toggle(
(state.state == LookupState::Ready
|| state.state == LookupState::Unsupported),
anim::type::normal);
if (state.bot) {
const auto delegate = parent->lifetime().make_state<
PeerListContentDelegateSimple
>();
const auto controller = parent->lifetime().make_state<
PreviewController
>(state.bot, resetBot);
controller->setStyleOverrides(&st::peerListSingleRow);
const auto content = Ui::CreateChild<PeerListContent>(
inner,
controller);
delegate->setContent(content);
controller->setDelegate(delegate);
delete base::take(*child);
*child = content;
} else if (state.state == LookupState::Ready
|| state.state == LookupState::Unsupported) {
const auto content = Ui::CreateChild<Ui::RpWidget>(inner);
const auto label = Ui::CreateChild<Ui::FlatLabel>(
content,
(state.state == LookupState::Unsupported
? tr::lng_chatbots_not_supported()
: tr::lng_chatbots_not_found()),
st::settingsChatbotsNotFound);
content->resize(
inner->width(),
st::peerListSingleRow.item.height);
rpl::combine(
content->sizeValue(),
label->sizeValue()
) | rpl::start_with_next([=](QSize size, QSize inner) {
label->move(
(size.width() - inner.width()) / 2,
(size.height() - inner.height()) / 2);
}, label->lifetime());
delete base::take(*child);
*child = content;
} else {
return;
}
(*child)->show();
inner->widthValue() | rpl::start_with_next([=](int width) {
(*child)->resizeToWidth(width);
}, (*child)->lifetime());
(*child)->heightValue() | rpl::start_with_next([=](int height) {
inner->resize(inner->width(), height + st::contactSkip);
}, inner->lifetime());
}, inner->lifetime());
raw->finishAnimating();
return result;
}
Chatbots::Chatbots(
QWidget *parent,
not_null<Window::SessionController*> controller)
: BusinessSection(parent, controller)
, _bottomSkipRounding(st::boxRadius, st::boxDividerBg) {
setupContent(controller);
}
Chatbots::~Chatbots() {
if (!Core::Quitting()) {
save();
}
}
bool Chatbots::closeByOutsideClick() const {
return false;
}
rpl::producer<QString> Chatbots::title() {
return tr::lng_chatbots_title();
}
void Chatbots::setupContent(
not_null<Window::SessionController*> controller) {
using namespace rpl::mappers;
const auto content = Ui::CreateChild<Ui::VerticalLayout>(this);
const auto current = controller->session().data().chatbots().current();
_recipients = Data::BusinessRecipients::MakeValid(current.recipients);
_repliesAllowed = current.repliesAllowed;
AddDividerTextWithLottie(content, {
.lottie = u"robot"_q,
.lottieSize = st::settingsCloudPasswordIconSize,
.lottieMargins = st::peerAppearanceIconPadding,
.showFinished = showFinishes(),
.about = tr::lng_chatbots_about(
lt_link,
tr::lng_chatbots_about_link(
) | Ui::Text::ToLink(tr::lng_chatbots_info_url(tr::now)),
Ui::Text::WithEntities),
.aboutMargins = st::peerAppearanceCoverLabelMargin,
});
const auto username = content->add(
object_ptr<Ui::InputField>(
content,
st::settingsChatbotsUsername,
tr::lng_chatbots_placeholder(),
(current.bot
? current.bot->session().createInternalLink(
current.bot->username())
: QString())),
st::settingsChatbotsUsernameMargins);
_usernameValue = DebouncedValue(username);
_botValue = rpl::single(BotState{
current.bot,
current.bot ? LookupState::Ready : LookupState::Empty
}) | rpl::then(
LookupBot(&controller->session(), _usernameValue.changes())
);
const auto resetBot = [=] {
username->setText(QString());
username->setFocus();
};
content->add(object_ptr<Ui::SlideWrap<Ui::RpWidget>>(
content,
MakeBotPreview(content, _botValue.value(), resetBot)));
Ui::AddDividerText(
content,
tr::lng_chatbots_add_about(),
st::peerAppearanceDividerTextMargin);
AddBusinessRecipientsSelector(content, {
.controller = controller,
.title = tr::lng_chatbots_access_title(),
.data = &_recipients,
.type = Data::BusinessRecipientsType::Bots,
});
Ui::AddSkip(content, st::settingsChatbotsAccessSkip);
Ui::AddDividerText(
content,
tr::lng_chatbots_exclude_about(),
st::peerAppearanceDividerTextMargin);
Ui::AddSkip(content);
Ui::AddSubsectionTitle(content, tr::lng_chatbots_permissions_title());
content->add(object_ptr<Ui::SettingsButton>(
content,
tr::lng_chatbots_reply(),
st::settingsButtonNoIcon
))->toggleOn(_repliesAllowed.value())->toggledChanges(
) | rpl::start_with_next([=](bool value) {
_repliesAllowed = value;
}, content->lifetime());
Ui::AddSkip(content);
Ui::AddDividerText(
content,
tr::lng_chatbots_reply_about(),
st::settingsChatbotsBottomTextMargin,
RectPart::Top);
Ui::ResizeFitChild(this, content);
}
void Chatbots::save() {
const auto show = controller()->uiShow();
const auto fail = [=](QString error) {
if (error == u"BUSINESS_RECIPIENTS_EMPTY"_q) {
show->showToast(tr::lng_greeting_recipients_empty(tr::now));
} else if (error == u"BOT_BUSINESS_MISSING"_q) {
show->showToast(tr::lng_chatbots_not_supported(tr::now));
}
};
controller()->session().data().chatbots().save({
.bot = _botValue.current().bot,
.recipients = _recipients.current(),
.repliesAllowed = _repliesAllowed.current(),
}, [=] {
}, fail);
}
} // namespace
Type ChatbotsId() {
return Chatbots::Id();
}
} // namespace Settings