kotatogram-desktop/Telegram/SourceFiles/info/profile/info_profile_actions.cpp

2225 lines
62 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/profile/info_profile_actions.h"
#include "kotato/kotato_lang.h"
#include "kotato/kotato_settings.h"
#include "api/api_blocked_peers.h"
#include "api/api_chat_participants.h"
#include "apiwrap.h"
#include "base/options.h"
#include "base/timer_rpl.h"
#include "base/unixtime.h"
#include "boxes/peers/add_bot_to_chat_box.h"
#include "boxes/peers/edit_contact_box.h"
#include "boxes/report_messages_box.h"
#include "boxes/share_box.h"
#include "boxes/translate_box.h"
#include "core/application.h"
#include "core/click_handler_types.h"
#include "data/business/data_business_common.h"
#include "data/business/data_business_info.h"
#include "data/data_changes.h"
#include "data/data_channel.h"
#include "data/data_chat.h"
#include "data/data_folder.h"
#include "data/data_forum_topic.h"
#include "data/data_peer_values.h"
#include "data/data_session.h"
#include "data/data_user.h"
#include "data/notify/data_notify_settings.h"
#include "dialogs/ui/dialogs_layout.h"
#include "dialogs/ui/dialogs_message_view.h"
#include "history/history.h"
#include "history/history_item.h"
#include "history/history_item_helpers.h"
#include "history/view/history_view_context_menu.h" // HistoryView::ShowReportPeerBox
#include "history/view/history_view_item_preview.h"
#include "info/info_controller.h"
#include "info/info_memento.h"
#include "info/profile/info_profile_icon.h"
#include "info/profile/info_profile_phone_menu.h"
#include "info/profile/info_profile_text.h"
#include "info/profile/info_profile_values.h"
#include "info/profile/info_profile_widget.h"
#include "lang/lang_keys.h"
#include "main/main_session.h"
#include "menu/menu_mute.h"
#include "support/support_helper.h"
#include "ui/boxes/report_box.h"
#include "ui/controls/userpic_button.h"
#include "ui/painter.h"
#include "ui/rect.h"
#include "ui/text/format_values.h"
#include "ui/text/text_utilities.h" // Ui::Text::ToUpper
#include "ui/text/text_variant.h"
#include "ui/toast/toast.h"
#include "ui/vertical_list.h"
#include "ui/widgets/buttons.h"
#include "ui/widgets/checkbox.h"
#include "ui/widgets/labels.h"
#include "ui/widgets/popup_menu.h"
#include "ui/widgets/shadow.h"
#include "ui/wrap/padding_wrap.h"
#include "ui/wrap/slide_wrap.h"
#include "ui/wrap/vertical_layout.h"
#include "window/window_controller.h" // Window::Controller::show.
#include "window/window_peer_menu.h"
#include "window/window_session_controller.h"
#include "styles/style_info.h"
#include "styles/style_layers.h"
#include "styles/style_menu_icons.h"
#include <QtGui/QGuiApplication>
#include <QtGui/QClipboard>
namespace Info {
namespace Profile {
namespace {
constexpr auto kDay = Data::WorkingInterval::kDay;
base::options::toggle ShowPeerIdBelowAbout({
.id = kOptionShowPeerIdBelowAbout,
.name = "Show Peer IDs in Profile",
.description = "Show peer IDs from API below their Bio / Description."
" Add contact IDs to exported data.",
});
[[nodiscard]] rpl::producer<TextWithEntities> UsernamesSubtext(
not_null<PeerData*> peer,
rpl::producer<QString> fallback) {
return rpl::combine(
UsernamesValue(peer),
std::move(fallback)
) | rpl::map([](std::vector<TextWithEntities> usernames, QString text) {
if (usernames.size() < 2) {
return TextWithEntities{ .text = text };
} else {
auto result = TextWithEntities();
result.append(tr::lng_info_usernames_label(tr::now));
result.append(' ');
auto &&subrange = ranges::make_subrange(
begin(usernames) + 1,
end(usernames));
for (auto &username : std::move(subrange)) {
const auto isLast = (usernames.back() == username);
result.append(Ui::Text::Link(
'@' + base::take(username.text),
username.entities.front().data()));
if (!isLast) {
result.append(u", "_q);
}
}
return result;
}
});
}
[[nodiscard]] Fn<void(QString)> UsernamesLinkCallback(
not_null<PeerData*> peer,
not_null<Window::SessionController*> controller,
const QString &addToLink) {
const auto weak = base::make_weak(controller);
return [=](QString link) {
if (link.startsWith(u"internal:"_q)) {
Core::App().openInternalUrl(link,
QVariant::fromValue(ClickHandlerContext{
.sessionWindow = weak,
}));
return;
} else if (!link.startsWith(u"https://"_q)) {
link = peer->session().createInternalLinkFull(peer->username())
+ addToLink;
}
if (!link.isEmpty()) {
if (const auto strong = weak.get()) {
FastShareLink(strong, link);
}
}
};
}
[[nodiscard]] object_ptr<Ui::RpWidget> CreateSkipWidget(
not_null<Ui::RpWidget*> parent) {
return Ui::CreateSkipWidget(parent, st::infoProfileSkip);
}
[[nodiscard]] object_ptr<Ui::SlideWrap<>> CreateSlideSkipWidget(
not_null<Ui::RpWidget*> parent) {
auto result = Ui::CreateSlideSkipWidget(
parent,
st::infoProfileSkip);
result->setDuration(st::infoSlideDuration);
return result;
}
[[nodiscard]] rpl::producer<TextWithEntities> AboutWithIdValue(
not_null<PeerData*> peer) {
return AboutValue(
peer
) | rpl::map([=](TextWithEntities &&value) {
if (!ShowPeerIdBelowAbout.value()) {
return std::move(value);
}
using namespace Ui::Text;
if (!value.empty()) {
value.append("\n");
}
value.append(Italic(u"id: "_q));
const auto raw = peer->id.value & PeerId::kChatTypeMask;
const auto id = QString::number(raw);
value.append(Link(Italic(id), "internal:copy:" + id));
return std::move(value);
});
}
[[nodiscard]] bool AreNonTrivialHours(const Data::WorkingHours &hours) {
if (!hours) {
return false;
}
const auto &intervals = hours.intervals.list;
for (auto i = 0; i != 7; ++i) {
const auto day = Data::WorkingInterval{ i * kDay, (i + 1) * kDay };
for (const auto &interval : intervals) {
const auto intersection = interval.intersected(day);
if (intersection && intersection != day) {
return true;
}
}
}
return false;
}
[[nodiscard]] TimeId OpensIn(
const Data::WorkingIntervals &intervals,
TimeId now) {
using namespace Data;
while (now < 0) {
now += WorkingInterval::kWeek;
}
while (now > WorkingInterval::kWeek) {
now -= WorkingInterval::kWeek;
}
auto closest = WorkingInterval::kWeek;
for (const auto &interval : intervals.list) {
if (interval.start <= now && interval.end > now) {
return TimeId(0);
} else if (interval.start > now && interval.start - now < closest) {
closest = interval.start - now;
} else if (interval.start < now) {
const auto next = interval.start + WorkingInterval::kWeek - now;
if (next < closest) {
closest = next;
}
}
}
return closest;
}
[[nodiscard]] rpl::producer<QString> OpensInText(
rpl::producer<TimeId> in,
rpl::producer<bool> hoursExpanded,
rpl::producer<QString> fallback) {
return rpl::combine(
std::move(in),
std::move(hoursExpanded),
std::move(fallback)
) | rpl::map([](TimeId in, bool hoursExpanded, QString fallback) {
return (!in || hoursExpanded)
? std::move(fallback)
: (in >= 86400)
? tr::lng_info_hours_opens_in_days(tr::now, lt_count, in / 86400)
: (in >= 3600)
? tr::lng_info_hours_opens_in_hours(tr::now, lt_count, in / 3600)
: tr::lng_info_hours_opens_in_minutes(
tr::now,
lt_count,
std::max(in / 60, 1));
});
}
[[nodiscard]] QString FormatDayTime(TimeId time) {
const auto wrap = [](TimeId value) {
const auto hours = value / 3600;
const auto minutes = (value % 3600) / 60;
return QString::number(hours).rightJustified(2, u'0')
+ ':'
+ QString::number(minutes).rightJustified(2, u'0');
};
return (time > kDay)
? tr::lng_info_hours_next_day(tr::now, lt_time, wrap(time - kDay))
: wrap(time == kDay ? 0 : time);
}
[[nodiscard]] QString JoinIntervals(const Data::WorkingIntervals &data) {
auto result = QStringList();
result.reserve(data.list.size());
for (const auto &interval : data.list) {
const auto start = FormatDayTime(interval.start);
const auto end = FormatDayTime(interval.end);
result.push_back(start + u" - "_q + end);
}
return result.join('\n');
}
[[nodiscard]] QString FormatDayHours(
const Data::WorkingHours &hours,
const Data::WorkingIntervals &mine,
bool my,
int day) {
using namespace Data;
const auto local = ExtractDayIntervals(hours.intervals, day);
if (IsFullOpen(local)) {
return tr::lng_info_hours_open_full(tr::now);
}
const auto use = my ? ExtractDayIntervals(mine, day) : local;
if (!use) {
return tr::lng_info_hours_closed(tr::now);
}
return JoinIntervals(use);
}
[[nodiscard]] Data::WorkingIntervals ShiftedIntervals(
Data::WorkingIntervals intervals,
int delta) {
auto &list = intervals.list;
if (!delta || list.empty()) {
return { std::move(list) };
}
for (auto &interval : list) {
interval.start += delta;
interval.end += delta;
}
while (list.front().start < 0) {
constexpr auto kWeek = Data::WorkingInterval::kWeek;
const auto first = list.front();
if (first.end > 0) {
list.push_back({ first.start + kWeek, kWeek });
list.front().start = 0;
} else {
list.push_back(first.shifted(kWeek));
list.erase(list.begin());
}
}
return intervals.normalized();
}
[[nodiscard]] object_ptr<Ui::SlideWrap<>> CreateWorkingHours(
not_null<QWidget*> parent,
not_null<UserData*> user) {
using namespace Data;
auto result = object_ptr<Ui::SlideWrap<Ui::RoundButton>>(
parent,
object_ptr<Ui::RoundButton>(
parent,
rpl::single(QString()),
st::infoHoursOuter),
st::infoProfileLabeledPadding - st::infoHoursOuterMargin);
const auto button = result->entity();
const auto inner = Ui::CreateChild<Ui::VerticalLayout>(button);
button->widthValue() | rpl::start_with_next([=](int width) {
const auto margin = st::infoHoursOuterMargin;
inner->resizeToWidth(width - margin.left() - margin.right());
inner->move(margin.left(), margin.top());
}, inner->lifetime());
inner->heightValue() | rpl::start_with_next([=](int height) {
const auto margin = st::infoHoursOuterMargin;
height += margin.top() + margin.bottom();
button->resize(button->width(), height);
}, inner->lifetime());
const auto info = &user->owner().businessInfo();
struct State {
rpl::variable<WorkingHours> hours;
rpl::variable<TimeId> time;
rpl::variable<int> day;
rpl::variable<int> timezoneDelta;
rpl::variable<WorkingIntervals> mine;
rpl::variable<WorkingIntervals> mineByDays;
rpl::variable<TimeId> opensIn;
rpl::variable<bool> opened;
rpl::variable<bool> expanded;
rpl::variable<bool> nonTrivial;
rpl::variable<bool> myTimezone;
rpl::event_stream<> recounts;
};
const auto state = inner->lifetime().make_state<State>();
auto recounts = state->recounts.events_starting_with_copy(rpl::empty);
const auto recount = [=] {
state->recounts.fire({});
};
state->hours = user->session().changes().peerFlagsValue(
user,
PeerUpdate::Flag::BusinessDetails
) | rpl::map([=] {
return user->businessDetails().hours;
});
state->nonTrivial = state->hours.value() | rpl::map(AreNonTrivialHours);
const auto seconds = QTime::currentTime().msecsSinceStartOfDay() / 1000;
const auto inMinute = seconds % 60;
const auto firstTick = inMinute ? (61 - inMinute) : 1;
state->time = rpl::single(rpl::empty) | rpl::then(
base::timer_once(firstTick * crl::time(1000))
) | rpl::then(
base::timer_each(60 * crl::time(1000))
) | rpl::map([] {
const auto local = QDateTime::currentDateTime();
const auto day = local.date().dayOfWeek() - 1;
const auto seconds = local.time().msecsSinceStartOfDay() / 1000;
return day * kDay + seconds;
});
state->day = state->time.value() | rpl::map([](TimeId time) {
return time / kDay;
});
state->timezoneDelta = rpl::combine(
state->hours.value(),
info->timezonesValue()
) | rpl::filter([](
const WorkingHours &hours,
const Timezones &timezones) {
return ranges::contains(
timezones.list,
hours.timezoneId,
&Timezone::id);
}) | rpl::map([](WorkingHours &&hours, const Timezones &timezones) {
const auto &list = timezones.list;
const auto closest = FindClosestTimezoneId(list);
const auto i = ranges::find(list, closest, &Timezone::id);
const auto j = ranges::find(list, hours.timezoneId, &Timezone::id);
Assert(i != end(list));
Assert(j != end(list));
return i->utcOffset - j->utcOffset;
});
state->mine = rpl::combine(
state->hours.value(),
state->timezoneDelta.value()
) | rpl::map([](WorkingHours &&hours, int delta) {
return ShiftedIntervals(hours.intervals, delta);
});
state->opensIn = rpl::combine(
state->mine.value(),
state->time.value()
) | rpl::map([](const WorkingIntervals &mine, TimeId time) {
return OpensIn(mine, time);
});
state->opened = state->opensIn.value() | rpl::map(rpl::mappers::_1 == 0);
state->mineByDays = rpl::combine(
state->hours.value(),
state->timezoneDelta.value()
) | rpl::map([](WorkingHours &&hours, int delta) {
auto full = std::array<bool, 7>();
auto withoutFullDays = hours.intervals;
for (auto i = 0; i != 7; ++i) {
if (IsFullOpen(ExtractDayIntervals(hours.intervals, i))) {
full[i] = true;
withoutFullDays = ReplaceDayIntervals(
withoutFullDays,
i,
Data::WorkingIntervals());
}
}
auto result = ShiftedIntervals(withoutFullDays, delta);
for (auto i = 0; i != 7; ++i) {
if (full[i]) {
result = ReplaceDayIntervals(
result,
i,
Data::WorkingIntervals{ { { 0, kDay } } });
}
}
return result;
});
const auto dayHoursText = [=](int day) {
return rpl::combine(
state->hours.value(),
state->mineByDays.value(),
state->myTimezone.value()
) | rpl::map([=](
const WorkingHours &hours,
const WorkingIntervals &mine,
bool my) {
return FormatDayHours(hours, mine, my, day);
});
};
const auto dayHoursTextValue = [=](rpl::producer<int> day) {
return std::move(day)
| rpl::map(dayHoursText)
| rpl::flatten_latest();
};
const auto openedWrap = inner->add(object_ptr<Ui::RpWidget>(inner));
const auto opened = Ui::CreateChild<Ui::FlatLabel>(
openedWrap,
rpl::conditional(
state->opened.value(),
tr::lng_info_work_open(),
tr::lng_info_work_closed()
) | rpl::after_next(recount),
st::infoHoursState);
opened->setAttribute(Qt::WA_TransparentForMouseEvents);
const auto timing = Ui::CreateChild<Ui::FlatLabel>(
openedWrap,
OpensInText(
state->opensIn.value(),
state->expanded.value(),
dayHoursTextValue(state->day.value())
) | rpl::after_next(recount),
st::infoHoursValue);
timing->setAttribute(Qt::WA_TransparentForMouseEvents);
state->opened.value() | rpl::start_with_next([=](bool value) {
opened->setTextColorOverride(value
? st::boxTextFgGood->c
: st::boxTextFgError->c);
}, opened->lifetime());
rpl::combine(
openedWrap->widthValue(),
opened->heightValue(),
timing->sizeValue()
) | rpl::start_with_next([=](int width, int h1, QSize size) {
opened->moveToLeft(0, 0, width);
timing->moveToRight(0, 0, width);
const auto margins = opened->getMargins();
const auto added = margins.top() + margins.bottom();
openedWrap->resize(width, std::max(h1, size.height()) - added);
}, openedWrap->lifetime());
const auto labelWrap = inner->add(object_ptr<Ui::RpWidget>(inner));
const auto label = Ui::CreateChild<Ui::FlatLabel>(
labelWrap,
tr::lng_info_hours_label(),
st::infoLabel);
label->setAttribute(Qt::WA_TransparentForMouseEvents);
const auto link = Ui::CreateChild<Ui::LinkButton>(
labelWrap,
QString());
rpl::combine(
state->nonTrivial.value(),
state->hours.value(),
state->mine.value(),
state->myTimezone.value()
) | rpl::map([=](
bool complex,
const WorkingHours &hours,
const WorkingIntervals &mine,
bool my) {
return (!complex || hours.intervals == mine)
? rpl::single(QString())
: my
? tr::lng_info_hours_my_time()
: tr::lng_info_hours_local_time();
}) | rpl::flatten_latest(
) | rpl::start_with_next([=](const QString &text) {
link->setText(text);
}, link->lifetime());
link->setClickedCallback([=] {
state->myTimezone = !state->myTimezone.current();
state->expanded = true;
});
rpl::combine(
labelWrap->widthValue(),
label->heightValue(),
link->sizeValue()
) | rpl::start_with_next([=](int width, int h1, QSize size) {
label->moveToLeft(0, 0, width);
link->moveToRight(0, 0, width);
const auto margins = label->getMargins();
const auto added = margins.top() + margins.bottom();
labelWrap->resize(width, std::max(h1, size.height()) - added);
}, labelWrap->lifetime());
const auto other = inner->add(
object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
inner,
object_ptr<Ui::VerticalLayout>(inner)));
other->toggleOn(state->expanded.value(), anim::type::normal);
other->finishAnimating();
const auto days = other->entity();
for (auto i = 1; i != 7; ++i) {
const auto dayWrap = days->add(
object_ptr<Ui::RpWidget>(other),
QMargins(0, st::infoHoursDaySkip, 0, 0));
auto label = state->day.value() | rpl::map([=](int day) {
switch ((day + i) % 7) {
case 0: return tr::lng_hours_monday();
case 1: return tr::lng_hours_tuesday();
case 2: return tr::lng_hours_wednesday();
case 3: return tr::lng_hours_thursday();
case 4: return tr::lng_hours_friday();
case 5: return tr::lng_hours_saturday();
case 6: return tr::lng_hours_sunday();
}
Unexpected("Index in working hours.");
}) | rpl::flatten_latest();
const auto dayLabel = Ui::CreateChild<Ui::FlatLabel>(
dayWrap,
std::move(label),
st::infoHoursDayLabel);
dayLabel->setAttribute(Qt::WA_TransparentForMouseEvents);
const auto dayHours = Ui::CreateChild<Ui::FlatLabel>(
dayWrap,
dayHoursTextValue(state->day.value()
| rpl::map((rpl::mappers::_1 + i) % 7)),
st::infoHoursValue);
dayHours->setAttribute(Qt::WA_TransparentForMouseEvents);
rpl::combine(
dayWrap->widthValue(),
dayLabel->heightValue(),
dayHours->sizeValue()
) | rpl::start_with_next([=](int width, int h1, QSize size) {
dayLabel->moveToLeft(0, 0, width);
dayHours->moveToRight(0, 0, width);
const auto margins = dayLabel->getMargins();
const auto added = margins.top() + margins.bottom();
dayWrap->resize(width, std::max(h1, size.height()) - added);
}, dayWrap->lifetime());
}
button->setClickedCallback([=] {
state->expanded = !state->expanded.current();
});
result->toggleOn(state->hours.value(
) | rpl::map([](const WorkingHours &data) {
return bool(data);
}));
return result;
}
[[nodiscard]] object_ptr<Ui::SlideWrap<>> CreateBirthday(
not_null<QWidget*> parent,
not_null<Window::SessionController*> controller,
not_null<UserData*> user) {
using namespace Data;
auto result = object_ptr<Ui::SlideWrap<Ui::RoundButton>>(
parent,
object_ptr<Ui::RoundButton>(
parent,
rpl::single(QString()),
st::infoHoursOuter),
st::infoProfileLabeledPadding - st::infoHoursOuterMargin);
result->setDuration(st::infoSlideDuration);
const auto button = result->entity();
auto outer = Ui::CreateChild<Ui::SlideWrap<Ui::VerticalLayout>>(
button,
object_ptr<Ui::VerticalLayout>(button),
st::infoHoursOuterMargin);
const auto layout = outer->entity();
layout->setAttribute(Qt::WA_TransparentForMouseEvents);
auto birthday = BirthdayValue(
user
) | rpl::start_spawning(result->lifetime());
auto label = BirthdayLabelText(rpl::duplicate(birthday));
auto text = BirthdayValueText(
rpl::duplicate(birthday)
) | Ui::Text::ToWithEntities();
const auto giftIcon = Ui::CreateChild<Ui::RpWidget>(layout);
giftIcon->resize(st::birthdayTodayIcon.size());
layout->sizeValue() | rpl::start_with_next([=](QSize size) {
giftIcon->moveToRight(
0,
(size.height() - giftIcon->height()) / 2,
size.width());
}, giftIcon->lifetime());
giftIcon->paintRequest() | rpl::start_with_next([=] {
auto p = QPainter(giftIcon);
st::birthdayTodayIcon.paint(p, 0, 0, giftIcon->width());
}, giftIcon->lifetime());
rpl::duplicate(
birthday
) | rpl::map([](Data::Birthday value) {
return Data::IsBirthdayTodayValue(value);
}) | rpl::flatten_latest(
) | rpl::distinct_until_changed(
) | rpl::start_with_next([=](bool today) {
const auto disable = !today && user->session().premiumCanBuy();
button->setDisabled(disable);
button->setAttribute(Qt::WA_TransparentForMouseEvents, disable);
button->clearState();
giftIcon->setVisible(!disable);
}, result->lifetime());
auto nonEmptyText = std::move(
text
) | rpl::before_next([slide = result.data()](
const TextWithEntities &value) {
if (value.text.isEmpty()) {
slide->hide(anim::type::normal);
}
}) | rpl::filter([](const TextWithEntities &value) {
return !value.text.isEmpty();
}) | rpl::after_next([slide = result.data()](
const TextWithEntities &value) {
slide->show(anim::type::normal);
});
layout->add(object_ptr<Ui::FlatLabel>(
layout,
std::move(nonEmptyText),
st::birthdayLabeled));
layout->add(Ui::CreateSkipWidget(layout, st::infoLabelSkip));
layout->add(object_ptr<Ui::FlatLabel>(
layout,
std::move(
label
) | rpl::after_next([=] {
layout->resizeToWidth(layout->widthNoMargins());
}),
st::birthdayLabel));
result->finishAnimating();
Ui::ResizeFitChild(button, outer);
button->setClickedCallback([=] {
if (!button->isDisabled()) {
controller->showGiftPremiumsBox(user, u"birthday"_q);
}
});
return result;
}
template <typename Text, typename ToggleOn, typename Callback>
auto AddActionButton(
not_null<Ui::VerticalLayout*> parent,
Text &&text,
ToggleOn &&toggleOn,
Callback &&callback,
const style::icon *icon,
const style::SettingsButton &st
= st::infoSharedMediaButton) {
auto result = parent->add(object_ptr<Ui::SlideWrap<Ui::SettingsButton>>(
parent,
object_ptr<Ui::SettingsButton>(
parent,
std::move(text),
st))
);
result->setDuration(
st::infoSlideDuration
)->toggleOn(
std::move(toggleOn)
)->entity()->addClickHandler(std::move(callback));
result->finishAnimating();
if (icon) {
object_ptr<Profile::FloatingIcon>(
result,
*icon,
st::infoSharedMediaButtonIconPosition);
}
return result;
};
template <typename Text, typename ToggleOn, typename Callback>
[[nodiscard]] auto AddMainButton(
not_null<Ui::VerticalLayout*> parent,
Text &&text,
ToggleOn &&toggleOn,
Callback &&callback,
Ui::MultiSlideTracker &tracker,
const style::SettingsButton &st = st::infoMainButton) {
tracker.track(AddActionButton(
parent,
std::move(text) | Ui::Text::ToUpper(),
std::move(toggleOn),
std::move(callback),
nullptr,
st));
}
class DetailsFiller {
public:
DetailsFiller(
not_null<Controller*> controller,
not_null<Ui::RpWidget*> parent,
not_null<PeerData*> peer,
Origin origin);
DetailsFiller(
not_null<Controller*> controller,
not_null<Ui::RpWidget*> parent,
not_null<Data::ForumTopic*> topic);
object_ptr<Ui::RpWidget> fill();
private:
object_ptr<Ui::RpWidget> setupPersonalChannel(not_null<UserData*> user);
object_ptr<Ui::RpWidget> setupInfo();
object_ptr<Ui::RpWidget> setupMuteToggle();
void setupMainButtons();
Ui::MultiSlideTracker fillTopicButtons();
Ui::MultiSlideTracker fillUserButtons(
not_null<UserData*> user);
Ui::MultiSlideTracker fillChannelButtons(
not_null<ChannelData*> channel);
void addReportReaction(Ui::MultiSlideTracker &tracker);
void addReportReaction(
GroupReactionOrigin data,
bool ban,
Ui::MultiSlideTracker &tracker);
template <
typename Widget,
typename = std::enable_if_t<
std::is_base_of_v<Ui::RpWidget, Widget>>>
Widget *add(
object_ptr<Widget> &&child,
const style::margins &margin = style::margins()) {
return _wrap->add(
std::move(child),
margin);
}
not_null<Controller*> _controller;
not_null<Ui::RpWidget*> _parent;
not_null<PeerData*> _peer;
Data::ForumTopic *_topic = nullptr;
Origin _origin;
object_ptr<Ui::VerticalLayout> _wrap;
};
class ActionsFiller {
public:
ActionsFiller(
not_null<Controller*> controller,
not_null<Ui::RpWidget*> parent,
not_null<PeerData*> peer);
object_ptr<Ui::RpWidget> fill();
private:
void addInviteToGroupAction(not_null<UserData*> user);
void addShareContactAction(not_null<UserData*> user);
void addEditContactAction(not_null<UserData*> user);
void addDeleteContactAction(not_null<UserData*> user);
void addBotCommandActions(not_null<UserData*> user);
void addReportAction();
void addBlockAction(not_null<UserData*> user);
void addLeaveChannelAction(not_null<ChannelData*> channel);
void addJoinChannelAction(not_null<ChannelData*> channel);
void fillUserActions(not_null<UserData*> user);
void fillChannelActions(not_null<ChannelData*> channel);
not_null<Controller*> _controller;
not_null<Ui::RpWidget*> _parent;
not_null<PeerData*> _peer;
object_ptr<Ui::VerticalLayout> _wrap = { nullptr };
};
void ReportReactionBox(
not_null<Ui::GenericBox*> box,
not_null<Window::SessionController*> controller,
not_null<PeerData*> participant,
GroupReactionOrigin data,
bool ban,
Fn<void()> sent) {
box->setTitle(tr::lng_report_reaction_title());
box->addRow(object_ptr<Ui::FlatLabel>(
box,
tr::lng_report_reaction_about(),
st::boxLabel));
const auto check = ban
? box->addRow(
object_ptr<Ui::Checkbox>(
box,
tr::lng_report_and_ban_button(tr::now),
true),
st::boxRowPadding + QMargins{ 0, st::boxLittleSkip, 0, 0 })
: nullptr;
box->addButton(tr::lng_report_button(), [=] {
const auto chat = data.group->asChat();
const auto channel = data.group->asMegagroup();
if (check && check->checked()) {
if (chat) {
chat->session().api().chatParticipants().kick(
chat,
participant);
} else if (channel) {
channel->session().api().chatParticipants().kick(
channel,
participant,
ChatRestrictionsInfo());
}
}
data.group->session().api().request(MTPmessages_ReportReaction(
data.group->input,
MTP_int(data.messageId.bare),
participant->input
)).done(crl::guard(controller, [=] {
controller->showToast(tr::lng_report_thanks(tr::now));
})).send();
sent();
box->closeBox();
}, st::attentionBoxButton);
box->addButton(tr::lng_cancel(), [=] {
box->closeBox();
});
}
DetailsFiller::DetailsFiller(
not_null<Controller*> controller,
not_null<Ui::RpWidget*> parent,
not_null<PeerData*> peer,
Origin origin)
: _controller(controller)
, _parent(parent)
, _peer(peer)
, _origin(origin)
, _wrap(_parent) {
}
DetailsFiller::DetailsFiller(
not_null<Controller*> controller,
not_null<Ui::RpWidget*> parent,
not_null<Data::ForumTopic*> topic)
: _controller(controller)
, _parent(parent)
, _peer(topic->peer())
, _topic(topic)
, _wrap(_parent) {
}
template <typename T>
bool SetClickContext(
const ClickHandlerPtr &handler,
const ClickContext &context) {
if (const auto casted = std::dynamic_pointer_cast<T>(handler)) {
casted->T::onClick(context);
return true;
}
return false;
}
object_ptr<Ui::RpWidget> DetailsFiller::setupInfo() {
auto result = object_ptr<Ui::VerticalLayout>(_wrap);
auto tracker = Ui::MultiSlideTracker();
// Fill context for a mention / hashtag / bot command link.
const auto infoClickFilter = [=,
peer = _peer.get(),
window = _controller->parentController()](
const ClickHandlerPtr &handler,
Qt::MouseButton button) {
const auto context = ClickContext{
button,
QVariant::fromValue(ClickHandlerContext{
.sessionWindow = base::make_weak(window),
.peer = peer,
})
};
if (SetClickContext<BotCommandClickHandler>(handler, context)) {
return false;
} else if (SetClickContext<MentionClickHandler>(handler, context)) {
return false;
} else if (SetClickContext<HashtagClickHandler>(handler, context)) {
return false;
} else if (SetClickContext<CashtagClickHandler>(handler, context)) {
return false;
} else if (SetClickContext<UrlClickHandler>(handler, context)) {
return false;
}
return true;
};
const auto addTranslateToMenu = [&,
peer = _peer.get(),
controller = _controller->parentController()](
not_null<Ui::FlatLabel*> label,
rpl::producer<TextWithEntities> &&text) {
struct State {
rpl::variable<TextWithEntities> labelText;
};
const auto state = label->lifetime().make_state<State>();
state->labelText = std::move(text);
label->setContextMenuHook([=](
Ui::FlatLabel::ContextMenuRequest request) {
label->fillContextMenu(request);
if (Ui::SkipTranslate(state->labelText.current())) {
return;
}
auto item = (request.selection.empty()
? tr::lng_context_translate
: tr::lng_context_translate_selected)(tr::now);
request.menu->addAction(std::move(item), [=] {
controller->window().show(Box(
Ui::TranslateBox,
peer,
MsgId(),
request.selection.empty()
? state->labelText.current()
: Ui::Text::Mid(
state->labelText.current(),
request.selection.from,
request.selection.to - request.selection.from),
false));
});
});
};
const auto addInfoLineGeneric = [&](
v::text::data &&label,
rpl::producer<TextWithEntities> &&text,
const style::FlatLabel &textSt = st::infoLabeled,
const style::margins &padding = st::infoProfileLabeledPadding) {
auto line = CreateTextWithLabel(
result,
v::text::take_marked(std::move(label)),
std::move(text),
st::infoLabel,
textSt,
padding);
tracker.track(result->add(std::move(line.wrap)));
line.text->setClickHandlerFilter(infoClickFilter);
return line;
};
const auto addInfoLine = [&](
v::text::data &&label,
rpl::producer<TextWithEntities> &&text,
const style::FlatLabel &textSt = st::infoLabeled,
const style::margins &padding = st::infoProfileLabeledPadding) {
return addInfoLineGeneric(
std::move(label),
std::move(text),
textSt,
padding);
};
const auto addInfoOneLine = [&](
v::text::data &&label,
rpl::producer<TextWithEntities> &&text,
const QString &contextCopyText,
const style::margins &padding = st::infoProfileLabeledPadding) {
auto result = addInfoLine(
std::move(label),
std::move(text),
st::infoLabeledOneLine,
padding);
result.text->setDoubleClickSelectsParagraph(true);
result.text->setContextCopyText(contextCopyText);
return result;
};
auto addInfoOneLineInline = [&](
rpl::producer<QString> &&label,
rpl::producer<TextWithEntities> &&text,
const QString &contextCopyText) {
auto result = addInfoLine(
std::move(label),
std::move(text),
st::infoLabeledOneLineInline);
result.text->setContextCopyText(contextCopyText);
return result;
};
if (const auto user = _peer->asUser()) {
const auto controller = _controller->parentController();
if (::Kotato::JsonSettings::GetInt("show_chat_id") != 0) {
auto idDrawableText = IDValue(
user
) | rpl::map([](TextWithEntities &&text) {
return Ui::Text::Link(text.text);
});
auto idInfo = addInfoOneLineInline(
(user->isBot()
? rktr("ktg_profile_bot_id")
: rktr("ktg_profile_user_id")),
std::move(idDrawableText),
ktr("ktg_profile_copy_id"));
idInfo.text->setClickHandlerFilter([user](auto&&...) {
const auto idText = IDString(user);
if (!idText.isEmpty()) {
QGuiApplication::clipboard()->setText(idText);
Ui::Toast::Show(user->isBot()
? ktr("ktg_bot_id_copied")
: ktr("ktg_user_id_copied"));
}
return false;
});
}
if (user->session().supportMode()) {
addInfoLineGeneric(
user->session().supportHelper().infoLabelValue(user),
user->session().supportHelper().infoTextValue(user));
}
auto phoneDrawableText = rpl::combine(
PhoneValue(user),
UsernameValue(user),
AboutValue(user),
tr::lng_info_mobile_hidden()
) | rpl::map([](
const TextWithEntities &phone,
const TextWithEntities &username,
const TextWithEntities &bio,
const QString &hidden) {
return (phone.text.isEmpty() && username.text.isEmpty() && bio.text.isEmpty())
? Ui::Text::WithEntities(hidden)
: Ui::Text::Link(phone.text);
});
{
const auto phoneLabel = addInfoOneLine(
tr::lng_info_mobile_label(),
PhoneOrHiddenValue(user),
tr::lng_profile_copy_phone(tr::now)).text;
const auto hook = [=](Ui::FlatLabel::ContextMenuRequest request) {
phoneLabel->fillContextMenu(request);
AddPhoneMenu(request.menu, user);
};
phoneLabel->setContextMenuHook(hook);
phoneLabel->setClickHandlerFilter([user](auto&&...) {
const auto phoneText = user->phone();
if (!phoneText.isEmpty()) {
QGuiApplication::clipboard()->setText(Ui::FormatPhone(phoneText));
Ui::Toast::Show(ktr("ktg_phone_copied"));
}
return false;
});
}
auto label = user->isBot()
? tr::lng_info_about_label()
: tr::lng_info_bio_label();
addTranslateToMenu(
addInfoLine(std::move(label), AboutWithIdValue(user)).text,
AboutWithIdValue(user));
const auto usernameLine = addInfoOneLine(
UsernamesSubtext(_peer, tr::lng_info_username_label()),
UsernameValue(user, true) | rpl::map([=](TextWithEntities u) {
return u.text.isEmpty()
? TextWithEntities()
: Ui::Text::Link(u, UsernameUrl(user, u.text.mid(1)));
}),
QString(),
st::infoProfileLabeledUsernamePadding);
const auto callback = UsernamesLinkCallback(
_peer,
controller,
QString());
const auto hook = [=](Ui::FlatLabel::ContextMenuRequest request) {
if (!request.link) {
return;
}
const auto text = request.link->copyToClipboardContextItemText();
if (text.isEmpty()) {
return;
}
const auto link = request.link->copyToClipboardText();
request.menu->addAction(
text,
[=] { QGuiApplication::clipboard()->setText(link); });
const auto last = link.lastIndexOf('/');
if (last < 0) {
return;
}
const auto mention = '@' + link.mid(last + 1);
if (mention.size() < 2) {
return;
}
request.menu->addAction(
tr::lng_context_copy_mention(tr::now),
[=] { QGuiApplication::clipboard()->setText(mention); });
};
usernameLine.text->overrideLinkClickHandler(callback);
usernameLine.subtext->overrideLinkClickHandler(callback);
usernameLine.text->setContextMenuHook(hook);
usernameLine.subtext->setContextMenuHook(hook);
const auto usernameLabel = usernameLine.text;
if (user->isBot()) {
const auto copyUsername = Ui::CreateChild<Ui::IconButton>(
usernameLabel->parentWidget(),
st::infoProfileLabeledButtonCopy);
result->sizeValue(
) | rpl::start_with_next([=] {
const auto s = usernameLabel->parentWidget()->size();
copyUsername->moveToRight(
0,
(s.height() - copyUsername->height()) / 2);
}, copyUsername->lifetime());
copyUsername->setClickedCallback([=] {
const auto link = user->session().createInternalLinkFull(
user->username());
if (!link.isEmpty()) {
QGuiApplication::clipboard()->setText(link);
controller->showToast(tr::lng_username_copied(tr::now));
}
return false;
});
} else {
tracker.track(result->add(
CreateBirthday(result, controller, user)));
tracker.track(result->add(CreateWorkingHours(result, user)));
auto locationText = user->session().changes().peerFlagsValue(
user,
Data::PeerUpdate::Flag::BusinessDetails
) | rpl::map([=] {
const auto &details = user->businessDetails();
if (!details.location) {
return TextWithEntities();
} else if (!details.location.point) {
return TextWithEntities{ details.location.address };
}
return Ui::Text::Link(
TextUtilities::SingleLine(details.location.address),
LocationClickHandler::Url(*details.location.point));
});
addInfoOneLine(
tr::lng_info_location_label(),
std::move(locationText),
QString()
).text->setLinksTrusted();
}
AddMainButton(
result,
tr::lng_info_add_as_contact(),
CanAddContactValue(user),
[=] { controller->window().show(Box(EditContactBox, controller, user)); },
tracker);
} else {
if (::Kotato::JsonSettings::GetInt("show_chat_id") != 0) {
auto idDrawableText = IDValue(
_peer
) | rpl::map([](TextWithEntities &&text) {
return Ui::Text::Link(text.text);
});
auto idInfo = addInfoOneLineInline(
(_peer->isChat()
? rktr("ktg_profile_group_id")
: _peer->isMegagroup()
? rktr("ktg_profile_supergroup_id")
: rktr("ktg_profile_channel_id")),
std::move(idDrawableText),
ktr("ktg_profile_copy_id"));
idInfo.text->setClickHandlerFilter([peer = _peer](auto&&...) {
const auto idText = IDString(peer);
if (!idText.isEmpty()) {
QGuiApplication::clipboard()->setText(idText);
Ui::Toast::Show(peer->isChat()
? ktr("ktg_group_id_copied")
: peer->isMegagroup()
? ktr("ktg_supergroup_id_copied")
: ktr("ktg_channel_id_copied"));
}
return false;
});
}
const auto topicRootId = _topic ? _topic->rootId() : 0;
const auto addToLink = topicRootId
? ('/' + QString::number(topicRootId.bare))
: QString();
auto linkText = LinkValue(
_peer,
true
) | rpl::map([=](const LinkWithUrl &link) {
const auto text = link.text;
return text.isEmpty()
? TextWithEntities()
: Ui::Text::Link(
(text.startsWith(u"https://"_q)
? text.mid(u"https://"_q.size())
: text) + addToLink,
(addToLink.isEmpty() ? link.url : (text + addToLink)));
});
auto linkLine = addInfoOneLine(
(topicRootId
? tr::lng_info_link_label(Ui::Text::WithEntities)
: UsernamesSubtext(_peer, tr::lng_info_link_label())),
std::move(linkText),
QString());
const auto controller = _controller->parentController();
const auto linkCallback = UsernamesLinkCallback(
_peer,
controller,
addToLink);
linkLine.text->overrideLinkClickHandler(linkCallback);
linkLine.subtext->overrideLinkClickHandler(linkCallback);
if (const auto channel = _topic ? nullptr : _peer->asChannel()) {
auto locationText = LocationValue(
channel
) | rpl::map([](const ChannelLocation *location) {
return location
? Ui::Text::Link(
TextUtilities::SingleLine(location->address),
LocationClickHandler::Url(location->point))
: TextWithEntities();
});
addInfoOneLine(
tr::lng_info_location_label(),
std::move(locationText),
QString()
).text->setLinksTrusted();
}
const auto about = addInfoLine(tr::lng_info_about_label(), _topic
? rpl::single(TextWithEntities())
: AboutWithIdValue(_peer));
if (!_topic) {
addTranslateToMenu(about.text, AboutWithIdValue(_peer));
}
}
if (!_peer->isSelf() && !::Kotato::JsonSettings::GetBool("profile_top_mute")) {
// No notifications toggle for Self => no separator.
result->add(object_ptr<Ui::SlideWrap<>>(
result,
object_ptr<Ui::PlainShadow>(result),
st::infoProfileSeparatorPadding)
)->setDuration(
st::infoSlideDuration
)->toggleOn(
std::move(tracker).atLeastOneShownValue()
);
}
object_ptr<FloatingIcon>(
result,
st::infoIconInformation,
st::infoInformationIconPosition);
return result;
}
object_ptr<Ui::RpWidget> DetailsFiller::setupPersonalChannel(
not_null<UserData*> user) {
auto result = object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
_wrap,
object_ptr<Ui::VerticalLayout>(_wrap));
const auto container = result->entity();
const auto window = _controller->parentController();
result->toggleOn(PersonalChannelValue(
user
) | rpl::map(rpl::mappers::_1 != nullptr));
result->finishAnimating();
auto channelToggleValue = PersonalChannelValue(
user
) | rpl::map([=] { return !!user->personalChannelId(); });
auto channel = PersonalChannelValue(
user
) | rpl::start_spawning(result->lifetime());
const auto channelLabelFactory = [=](rpl::producer<ChannelData*> c) {
return rpl::combine(
tr::lng_info_personal_channel_label(Ui::Text::WithEntities),
std::move(c)
) | rpl::map([](TextWithEntities &&text, ChannelData *channel) {
const auto count = channel ? channel->membersCount() : 0;
if (count > 1) {
text.append(
QString::fromUtf8(" \xE2\x80\xA2 ")
).append(tr::lng_chat_status_subscribers(
tr::now,
lt_count_decimal,
count));
}
return text;
});
};
{
const auto onlyChannelWrap = container->add(
object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
container,
object_ptr<Ui::VerticalLayout>(container)));
onlyChannelWrap->toggleOn(rpl::duplicate(channelToggleValue)
| rpl::map(!rpl::mappers::_1));
onlyChannelWrap->finishAnimating();
Ui::AddDivider(onlyChannelWrap->entity());
auto text = rpl::duplicate(
channel
) | rpl::map([=](ChannelData *channel) {
return channel ? NameValue(channel) : rpl::single(QString());
}) | rpl::flatten_latest() | rpl::map([](const QString &name) {
return name.isEmpty() ? TextWithEntities() : Ui::Text::Link(name);
});
auto line = CreateTextWithLabel(
result,
channelLabelFactory(rpl::duplicate(channel)),
std::move(text),
st::infoLabel,
st::infoLabeled,
st::infoProfileLabeledPadding);
onlyChannelWrap->entity()->add(std::move(line.wrap));
line.text->setClickHandlerFilter([=](
const ClickHandlerPtr &handler,
Qt::MouseButton button) {
if (const auto channelId = user->personalChannelId()) {
window->showPeerInfo(peerFromChannel(channelId));
}
return false;
});
object_ptr<FloatingIcon>(
onlyChannelWrap,
st::infoIconMediaChannel,
st::infoPersonalChannelIconPosition);
}
{
const auto messageChannelWrap = container->add(
object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
container,
object_ptr<Ui::VerticalLayout>(container)));
messageChannelWrap->toggleOn(rpl::duplicate(channelToggleValue));
messageChannelWrap->finishAnimating();
const auto clear = [=] {
while (messageChannelWrap->entity()->count()) {
delete messageChannelWrap->entity()->widgetAt(0);
}
};
const auto rebuild = [=](
not_null<HistoryItem*> item,
anim::type animated) {
const auto &stUserpic = st::infoPersonalChannelUserpic;
const auto &stLabeled = st::infoProfileLabeledPadding;
messageChannelWrap->toggle(false, anim::type::instant);
clear();
Ui::AddDivider(messageChannelWrap->entity());
Ui::AddSkip(messageChannelWrap->entity());
const auto inner = messageChannelWrap->entity()->add(
object_ptr<Ui::VerticalLayout>(messageChannelWrap->entity()));
const auto line = inner->add(
object_ptr<Ui::FixedHeightWidget>(
inner,
stUserpic.photoSize + rect::m::sum::v(stLabeled)));
const auto userpic = Ui::CreateChild<Ui::UserpicButton>(
line,
item->history()->peer,
st::infoPersonalChannelUserpic);
userpic->moveToLeft(
-st::infoPersonalChannelUserpicSkip
+ (stLabeled.left() - stUserpic.photoSize) / 2,
stLabeled.top());
userpic->setAttribute(Qt::WA_TransparentForMouseEvents);
const auto date = Ui::CreateChild<Ui::FlatLabel>(
line,
Ui::FormatDialogsDate(ItemDateTime(item)),
st::infoPersonalChannelDateLabel);
const auto name = Ui::CreateChild<Ui::FlatLabel>(
line,
NameValue(item->history()->peer),
st::infoPersonalChannelNameLabel);
const auto preview = Ui::CreateChild<Ui::RpWidget>(line);
auto &lifetime = preview->lifetime();
using namespace Dialogs::Ui;
const auto previewView = lifetime.make_state<MessageView>();
preview->resize(0, st::infoLabeled.style.font->height);
preview->paintRequest(
) | rpl::start_with_next([=, fullId = item->fullId()](
const QRect &rect) {
auto p = Painter(preview);
const auto item = user->session().data().message(fullId);
if (!item) {
p.setPen(st::infoPersonalChannelDateLabel.textFg);
p.setBrush(Qt::NoBrush);
p.setFont(st::infoPersonalChannelDateLabel.style.font);
p.drawText(
preview->rect(),
tr::lng_deleted_message(tr::now),
style::al_left);
return;
}
if (previewView->prepared(item, nullptr)) {
previewView->paint(p, preview->rect(), {
.st = &st::defaultDialogRow,
.currentBg = st::boxBg->b,
});
} else if (!previewView->dependsOn(item)) {
p.setPen(st::infoPersonalChannelDateLabel.textFg);
p.setBrush(Qt::NoBrush);
p.setFont(st::infoPersonalChannelDateLabel.style.font);
p.drawText(
preview->rect(),
tr::lng_contacts_loading(tr::now),
style::al_left);
previewView->prepare(
item,
nullptr,
[=] { preview->update(); },
{});
preview->update();
}
}, preview->lifetime());
line->sizeValue(
) | rpl::start_with_next([=](const QSize &size) {
const auto left = stLabeled.left();
const auto right = st::infoPersonalChannelDateSkip;
const auto top = stLabeled.top();
date->moveToRight(right, top);
name->resizeToWidth(size.width()
- left
- date->width()
- st::defaultVerticalListSkip
- right);
name->moveToLeft(left, top);
preview->resize(
size.width() - left - right,
st::infoLabeled.style.font->height);
preview->moveToLeft(
left,
size.height() - stLabeled.bottom() - preview->height());
}, preview->lifetime());
{
inner->add(
object_ptr<Ui::FlatLabel>(
inner,
channelLabelFactory(
rpl::single(item->history()->peer->asChannel())),
st::infoLabel),
QMargins(
st::infoProfileLabeledPadding.left(),
0,
st::infoProfileLabeledPadding.right(),
st::infoProfileLabeledPadding.bottom()));
}
{
const auto button = Ui::CreateChild<Ui::RippleButton>(
messageChannelWrap->entity(),
st::defaultRippleAnimation);
button->paintRequest(
) | rpl::start_with_next([=](const QRect &rect) {
auto p = QPainter(button);
button->paintRipple(p, 0, 0);
}, button->lifetime());
inner->geometryValue(
) | rpl::start_with_next([=](const QRect &rect) {
button->setGeometry(rect);
}, button->lifetime());
button->setClickedCallback([=, msg = item->fullId().msg] {
window->showPeerHistory(
item->history()->peer,
Window::SectionShow::Way::Forward,
msg);
});
button->lower();
}
inner->setAttribute(Qt::WA_TransparentForMouseEvents);
Ui::AddSkip(messageChannelWrap->entity());
Ui::ToggleChildrenVisibility(messageChannelWrap->entity(), true);
Ui::ToggleChildrenVisibility(line, true);
messageChannelWrap->toggle(true, animated);
};
rpl::duplicate(
channel
) | rpl::start_with_next([=](ChannelData *channel) {
clear();
if (!channel) {
return;
}
const auto id = FullMsgId(
channel->id,
user->personalChannelMessageId());
if (const auto item = user->session().data().message(id)) {
return rebuild(item, anim::type::instant);
}
user->session().api().requestMessageData(
channel,
user->personalChannelMessageId(),
crl::guard(container, [=] {
if (const auto i = user->session().data().message(id)) {
rebuild(i, anim::type::normal);
}
}));
}, messageChannelWrap->lifetime());
}
return result;
}
object_ptr<Ui::RpWidget> DetailsFiller::setupMuteToggle() {
const auto peer = _peer;
const auto topicRootId = _topic ? _topic->rootId() : MsgId();
const auto makeThread = [=] {
return topicRootId
? static_cast<Data::Thread*>(peer->forumTopicFor(topicRootId))
: peer->owner().history(peer).get();
};
auto result = object_ptr<Ui::SettingsButton>(
_wrap,
tr::lng_profile_enable_notifications(),
st::infoNotificationsButton);
result->toggleOn(_topic
? NotificationsEnabledValue(_topic)
: NotificationsEnabledValue(peer), true);
result->setAcceptBoth();
const auto notifySettings = &peer->owner().notifySettings();
MuteMenu::SetupMuteMenu(
result.data(),
result->clicks(
) | rpl::filter([=](Qt::MouseButton button) {
if (button == Qt::RightButton) {
return true;
}
const auto topic = topicRootId
? peer->forumTopicFor(topicRootId)
: nullptr;
Assert(!topicRootId || topic != nullptr);
const auto is = topic
? notifySettings->isMuted(topic)
: notifySettings->isMuted(peer);
if (is) {
if (topic) {
notifySettings->update(topic, { .unmute = true });
} else {
notifySettings->update(peer, { .unmute = true });
}
return false;
} else {
return true;
}
}) | rpl::to_empty,
makeThread,
_controller->uiShow());
object_ptr<FloatingIcon>(
result,
st::infoIconNotifications,
st::infoNotificationsIconPosition);
return result;
}
void DetailsFiller::setupMainButtons() {
auto wrapButtons = [=](auto &&callback) {
auto topSkip = _wrap->add(CreateSlideSkipWidget(_wrap));
auto tracker = callback();
topSkip->toggleOn(std::move(tracker).atLeastOneShownValue());
};
if (_topic) {
wrapButtons([=] {
return fillTopicButtons();
});
} else if (const auto user = _peer->asUser()) {
wrapButtons([=] {
return fillUserButtons(user);
});
} else if (const auto channel = _peer->asChannel()) {
if (!channel->isMegagroup()) {
wrapButtons([=] {
return fillChannelButtons(channel);
});
}
}
}
void DetailsFiller::addReportReaction(Ui::MultiSlideTracker &tracker) {
v::match(_origin.data, [&](GroupReactionOrigin data) {
const auto user = _peer->asUser();
if (_peer->isSelf()) {
return;
#if 0 // Only public groups allow reaction reports for now.
} else if (const auto chat = data.group->asChat()) {
const auto ban = chat->canBanMembers()
&& (!user || !chat->admins.contains(_peer))
&& (!user || chat->creator != user->id);
addReportReaction(data, ban, tracker);
#endif
} else if (const auto channel = data.group->asMegagroup()) {
if (channel->isPublic()) {
const auto ban = channel->canBanMembers()
&& (!user || !channel->mgInfo->admins.contains(user->id))
&& (!user || channel->mgInfo->creator != user);
addReportReaction(data, ban, tracker);
}
}
}, [](const auto &) {});
}
void DetailsFiller::addReportReaction(
GroupReactionOrigin data,
bool ban,
Ui::MultiSlideTracker &tracker) {
const auto peer = _peer;
if (!peer) {
return;
}
const auto controller = _controller->parentController();
const auto forceHidden = std::make_shared<rpl::variable<bool>>(false);
const auto user = peer->asUser();
auto shown = user
? rpl::combine(
Info::Profile::IsContactValue(user),
forceHidden->value(),
!rpl::mappers::_1 && !rpl::mappers::_2
) | rpl::type_erased()
: (forceHidden->value() | rpl::map(!rpl::mappers::_1));
const auto sent = [=] {
*forceHidden = true;
};
AddMainButton(
_wrap,
(ban
? tr::lng_report_and_ban()
: tr::lng_report_reaction()),
std::move(shown),
[=] { controller->show(
Box(ReportReactionBox, controller, peer, data, ban, sent)); },
tracker,
st::infoMainButtonAttention);
}
Ui::MultiSlideTracker DetailsFiller::fillTopicButtons() {
using namespace rpl::mappers;
Ui::MultiSlideTracker tracker;
const auto window = _controller->parentController();
const auto forum = _topic->forum();
auto showTopicsVisible = rpl::combine(
window->adaptive().oneColumnValue(),
window->shownForum().value(),
_1 || (_2 != forum));
AddMainButton(
_wrap,
tr::lng_forum_show_topics_list(),
std::move(showTopicsVisible),
[=] { window->showForum(forum); },
tracker);
return tracker;
}
Ui::MultiSlideTracker DetailsFiller::fillUserButtons(
not_null<UserData*> user) {
using namespace rpl::mappers;
Ui::MultiSlideTracker tracker;
auto window = _controller->parentController();
auto addSendMessageButton = [&] {
auto activePeerValue = window->activeChatValue(
) | rpl::map([](Dialogs::Key key) {
return key.peer();
});
auto sendMessageVisible = rpl::combine(
_controller->wrapValue(),
std::move(activePeerValue),
(_1 != Wrap::Side) || (_2 != user));
auto sendMessage = [window, user] {
window->showPeerHistory(
user,
Window::SectionShow::Way::Forward);
};
AddMainButton(
_wrap,
tr::lng_profile_send_message(),
std::move(sendMessageVisible),
std::move(sendMessage),
tracker);
};
if (user->isSelf()) {
auto separator = _wrap->add(object_ptr<Ui::SlideWrap<>>(
_wrap,
object_ptr<Ui::PlainShadow>(_wrap),
st::infoProfileSeparatorPadding)
)->setDuration(
st::infoSlideDuration
);
addSendMessageButton();
separator->toggleOn(
std::move(tracker).atLeastOneShownValue()
);
} else {
addSendMessageButton();
}
addReportReaction(tracker);
return tracker;
}
Ui::MultiSlideTracker DetailsFiller::fillChannelButtons(
not_null<ChannelData*> channel) {
using namespace rpl::mappers;
Ui::MultiSlideTracker tracker;
auto window = _controller->parentController();
auto activePeerValue = window->activeChatValue(
) | rpl::map([](Dialogs::Key key) {
return key.peer();
});
auto viewChannelVisible = rpl::combine(
_controller->wrapValue(),
std::move(activePeerValue),
(_1 != Wrap::Side) || (_2 != channel));
auto viewChannel = [=] {
window->showPeerHistory(
channel,
Window::SectionShow::Way::Forward);
};
AddMainButton(
_wrap,
tr::lng_profile_view_channel(),
std::move(viewChannelVisible),
std::move(viewChannel),
tracker);
return tracker;
}
object_ptr<Ui::RpWidget> DetailsFiller::fill() {
Expects(!_topic || !_topic->creating());
if (const auto user = _peer->asUser()) {
add(setupPersonalChannel(user));
}
add(object_ptr<Ui::BoxContentDivider>(_wrap));
add(CreateSkipWidget(_wrap));
add(setupInfo());
if (!_peer->isSelf() && !::Kotato::JsonSettings::GetBool("profile_top_mute")) {
add(setupMuteToggle());
}
setupMainButtons();
add(CreateSkipWidget(_wrap));
return std::move(_wrap);
}
ActionsFiller::ActionsFiller(
not_null<Controller*> controller,
not_null<Ui::RpWidget*> parent,
not_null<PeerData*> peer)
: _controller(controller)
, _parent(parent)
, _peer(peer) {
}
void ActionsFiller::addInviteToGroupAction(
not_null<UserData*> user) {
const auto notEmpty = [](const QString &value) {
return !value.isEmpty();
};
const auto controller = _controller->parentController();
AddActionButton(
_wrap,
InviteToChatButton(user) | rpl::filter(notEmpty),
InviteToChatButton(user) | rpl::map(notEmpty),
[=] { AddBotToGroupBoxController::Start(controller, user); },
&st::infoIconAddMember);
const auto about = _wrap->add(
object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
_wrap.data(),
object_ptr<Ui::VerticalLayout>(_wrap.data())));
about->toggleOn(InviteToChatAbout(user) | rpl::map(notEmpty));
Ui::AddSkip(about->entity());
Ui::AddDividerText(
about->entity(),
InviteToChatAbout(user) | rpl::filter(notEmpty));
Ui::AddSkip(about->entity());
about->finishAnimating();
}
void ActionsFiller::addShareContactAction(not_null<UserData*> user) {
const auto controller = _controller->parentController();
AddActionButton(
_wrap,
tr::lng_info_share_contact(),
CanShareContactValue(user),
[=] { Window::PeerMenuShareContactBox(controller, user); },
&st::infoIconShare);
}
void ActionsFiller::addEditContactAction(not_null<UserData*> user) {
const auto controller = _controller->parentController();
AddActionButton(
_wrap,
tr::lng_info_edit_contact(),
IsContactValue(user),
[=] { controller->window().show(Box(EditContactBox, controller, user)); },
&st::infoIconEdit);
}
void ActionsFiller::addDeleteContactAction(not_null<UserData*> user) {
const auto controller = _controller->parentController();
AddActionButton(
_wrap,
tr::lng_info_delete_contact(),
IsContactValue(user),
[=] { Window::PeerMenuDeleteContact(controller, user); },
&st::infoIconDelete);
}
void ActionsFiller::addBotCommandActions(not_null<UserData*> user) {
auto findBotCommand = [user](const QString &command) {
if (!user->isBot()) {
return QString();
}
for (const auto &data : user->botInfo->commands) {
const auto isSame = !data.command.compare(
command,
Qt::CaseInsensitive);
if (isSame) {
return data.command;
}
}
return QString();
};
auto hasBotCommandValue = [=](const QString &command) {
return user->session().changes().peerFlagsValue(
user,
Data::PeerUpdate::Flag::BotCommands
) | rpl::map([=] {
return !findBotCommand(command).isEmpty();
});
};
auto sendBotCommand = [=, window = _controller->parentController()](
const QString &command) {
const auto original = findBotCommand(command);
if (original.isEmpty()) {
return;
}
BotCommandClickHandler('/' + original).onClick(ClickContext{
Qt::LeftButton,
QVariant::fromValue(ClickHandlerContext{
.sessionWindow = base::make_weak(window),
.peer = user,
})
});
};
auto addBotCommand = [=](
rpl::producer<QString> text,
const QString &command,
const style::icon *icon = nullptr) {
AddActionButton(
_wrap,
std::move(text),
hasBotCommandValue(command),
[=] { sendBotCommand(command); },
icon);
};
addBotCommand(
tr::lng_profile_bot_help(),
u"help"_q,
&st::infoIconInformation);
addBotCommand(tr::lng_profile_bot_settings(), u"settings"_q);
addBotCommand(tr::lng_profile_bot_privacy(), u"privacy"_q);
}
void ActionsFiller::addReportAction() {
const auto peer = _peer;
const auto controller = _controller->parentController();
const auto report = [=] {
ShowReportPeerBox(controller, peer);
};
AddActionButton(
_wrap,
tr::lng_profile_report(),
rpl::single(true),
report,
&st::infoIconReport,
st::infoBlockButton);
}
void ActionsFiller::addBlockAction(not_null<UserData*> user) {
const auto controller = _controller->parentController();
const auto window = &controller->window();
auto text = user->session().changes().peerFlagsValue(
user,
Data::PeerUpdate::Flag::IsBlocked
) | rpl::map([=] {
switch (user->blockStatus()) {
case UserData::BlockStatus::Blocked:
return ((user->isBot() && !user->isSupport())
? tr::lng_profile_restart_bot
: tr::lng_profile_unblock_user)();
case UserData::BlockStatus::NotBlocked:
default:
return ((user->isBot() && !user->isSupport())
? tr::lng_profile_block_bot
: tr::lng_profile_block_user)();
}
}) | rpl::flatten_latest(
) | rpl::start_spawning(_wrap->lifetime());
auto toggleOn = rpl::duplicate(
text
) | rpl::map([](const QString &text) {
return !text.isEmpty();
});
auto callback = [=] {
if (user->isBlocked()) {
const auto show = controller->uiShow();
Window::PeerMenuUnblockUserWithBotRestart(show, user);
if (user->isBot()) {
controller->showPeerHistory(user);
}
} else if (user->isBot()) {
user->session().api().blockedPeers().block(user);
} else {
window->show(Box(
Window::PeerMenuBlockUserBox,
window,
user,
v::null,
v::null));
}
};
AddActionButton(
_wrap,
rpl::duplicate(text),
std::move(toggleOn),
std::move(callback),
&st::infoIconBlock,
st::infoBlockButton);
}
void ActionsFiller::addLeaveChannelAction(not_null<ChannelData*> channel) {
Expects(_controller->parentController());
AddActionButton(
_wrap,
tr::lng_profile_leave_channel(),
AmInChannelValue(channel),
Window::DeleteAndLeaveHandler(
_controller->parentController(),
channel),
&st::infoIconLeave);
}
void ActionsFiller::addJoinChannelAction(
not_null<ChannelData*> channel) {
using namespace rpl::mappers;
auto joinVisible = AmInChannelValue(channel)
| rpl::map(!_1)
| rpl::start_spawning(_wrap->lifetime());
AddActionButton(
_wrap,
tr::lng_profile_join_channel(),
rpl::duplicate(joinVisible),
[=] { channel->session().api().joinChannel(channel); },
&st::infoIconAddMember);
_wrap->add(object_ptr<Ui::SlideWrap<Ui::FixedHeightWidget>>(
_wrap,
CreateSkipWidget(
_wrap,
st::infoBlockButtonSkip))
)->setDuration(
st::infoSlideDuration
)->toggleOn(
rpl::duplicate(joinVisible)
);
}
void ActionsFiller::fillUserActions(not_null<UserData*> user) {
if (user->isBot()) {
addInviteToGroupAction(user);
}
addShareContactAction(user);
if (!user->isSelf()) {
addEditContactAction(user);
addDeleteContactAction(user);
}
if (!user->isSelf() && !user->isSupport()) {
if (user->isBot()) {
addBotCommandActions(user);
}
_wrap->add(CreateSkipWidget(
_wrap,
st::infoBlockButtonSkip));
if (user->isBot()) {
addReportAction();
}
addBlockAction(user);
}
}
void ActionsFiller::fillChannelActions(
not_null<ChannelData*> channel) {
using namespace rpl::mappers;
addJoinChannelAction(channel);
addLeaveChannelAction(channel);
if (!channel->amCreator()) {
addReportAction();
}
}
object_ptr<Ui::RpWidget> ActionsFiller::fill() {
auto wrapResult = [=](auto &&callback) {
_wrap = object_ptr<Ui::VerticalLayout>(_parent);
_wrap->add(CreateSkipWidget(_wrap));
callback();
_wrap->add(CreateSkipWidget(_wrap));
return std::move(_wrap);
};
if (auto user = _peer->asUser()) {
return wrapResult([=] {
fillUserActions(user);
});
} else if (auto channel = _peer->asChannel()) {
if (channel->isMegagroup()) {
return { nullptr };
}
return wrapResult([=] {
fillChannelActions(channel);
});
}
return { nullptr };
}
} // namespace
const char kOptionShowPeerIdBelowAbout[] = "show-peer-id-below-about";
object_ptr<Ui::RpWidget> SetupDetails(
not_null<Controller*> controller,
not_null<Ui::RpWidget*> parent,
not_null<PeerData*> peer,
Origin origin) {
DetailsFiller filler(controller, parent, peer, origin);
return filler.fill();
}
object_ptr<Ui::RpWidget> SetupDetails(
not_null<Controller*> controller,
not_null<Ui::RpWidget*> parent,
not_null<Data::ForumTopic*> topic) {
DetailsFiller filler(controller, parent, topic);
return filler.fill();
}
object_ptr<Ui::RpWidget> SetupActions(
not_null<Controller*> controller,
not_null<Ui::RpWidget*> parent,
not_null<PeerData*> peer) {
ActionsFiller filler(controller, parent, peer);
return filler.fill();
}
void SetupAddChannelMember(
not_null<Window::SessionNavigation*> navigation,
not_null<Ui::RpWidget*> parent,
not_null<ChannelData*> channel) {
auto add = Ui::CreateChild<Ui::IconButton>(
parent.get(),
st::infoMembersAddMember);
add->showOn(CanAddMemberValue(channel));
add->addClickHandler([=] {
Window::PeerMenuAddChannelMembers(navigation, channel);
});
parent->widthValue(
) | rpl::start_with_next([add](int newWidth) {
auto availableWidth = newWidth
- st::infoMembersButtonPosition.x();
add->moveToLeft(
availableWidth - add->width(),
st::infoMembersButtonPosition.y(),
newWidth);
}, add->lifetime());
}
object_ptr<Ui::RpWidget> SetupChannelMembers(
not_null<Controller*> controller,
not_null<Ui::RpWidget*> parent,
not_null<PeerData*> peer) {
using namespace rpl::mappers;
auto channel = peer->asChannel();
if (!channel || channel->isMegagroup()) {
return { nullptr };
}
auto membersShown = rpl::combine(
MembersCountValue(channel),
Data::PeerFlagValue(
channel,
ChannelDataFlag::CanViewParticipants),
(_1 > 0) && _2);
auto membersText = tr::lng_chat_status_subscribers(
lt_count_decimal,
MembersCountValue(channel) | tr::to_count());
auto membersCallback = [=] {
controller->showSection(std::make_shared<Info::Memento>(
channel,
Section::Type::Members));
};
auto result = object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
parent,
object_ptr<Ui::VerticalLayout>(parent));
result->setDuration(
st::infoSlideDuration
)->toggleOn(
std::move(membersShown)
);
auto members = result->entity();
members->add(object_ptr<Ui::BoxContentDivider>(members));
members->add(CreateSkipWidget(members));
auto button = AddActionButton(
members,
std::move(membersText),
rpl::single(true),
std::move(membersCallback),
nullptr)->entity();
SetupAddChannelMember(controller, button, channel);
object_ptr<FloatingIcon>(
members,
st::infoIconMembers,
st::infoChannelMembersIconPosition);
members->add(CreateSkipWidget(members));
return result;
}
} // namespace Profile
} // namespace Info