1557 lines
		
	
	
	
		
			45 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1557 lines
		
	
	
	
		
			45 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/settings_privacy_controllers.h"
 | |
| 
 | |
| #include "kotato/kotato_lang.h"
 | |
| #include "api/api_global_privacy.h"
 | |
| #include "api/api_peer_photo.h"
 | |
| #include "apiwrap.h"
 | |
| #include "base/call_delayed.h"
 | |
| #include "base/event_filter.h"
 | |
| #include "base/unixtime.h"
 | |
| #include "boxes/abstract_box.h"
 | |
| #include "boxes/peer_list_controllers.h"
 | |
| #include "boxes/peers/peer_short_info_box.h"
 | |
| #include "boxes/peers/prepare_short_info_box.h"
 | |
| #include "calls/calls_instance.h"
 | |
| #include "core/application.h"
 | |
| #include "data/data_changes.h"
 | |
| #include "data/data_file_origin.h"
 | |
| #include "data/data_peer_values.h" // Data::AmPremiumValue.
 | |
| #include "data/data_photo_media.h"
 | |
| #include "data/data_session.h"
 | |
| #include "data/data_user.h"
 | |
| #include "data/data_user_photos.h" // UserPhotosViewer.
 | |
| #include "editor/photo_editor_common.h"
 | |
| #include "editor/photo_editor_layer_widget.h"
 | |
| #include "history/admin_log/history_admin_log_item.h"
 | |
| #include "history/history.h"
 | |
| #include "history/history_item_components.h"
 | |
| #include "history/view/history_view_message.h"
 | |
| #include "lang/lang_keys.h"
 | |
| #include "main/main_session.h"
 | |
| #include "settings/settings_premium.h"
 | |
| #include "settings/settings_privacy_security.h"
 | |
| #include "ui/boxes/confirm_box.h"
 | |
| #include "ui/chat/chat_style.h"
 | |
| #include "ui/chat/chat_theme.h"
 | |
| #include "ui/painter.h"
 | |
| #include "ui/vertical_list.h"
 | |
| #include "ui/text/format_values.h" // Ui::FormatPhone
 | |
| #include "ui/text/text_utilities.h"
 | |
| #include "ui/toast/toast.h"
 | |
| #include "ui/widgets/checkbox.h"
 | |
| #include "ui/wrap/slide_wrap.h"
 | |
| #include "window/section_widget.h"
 | |
| #include "window/window_controller.h"
 | |
| #include "window/window_session_controller.h"
 | |
| #include "styles/style_chat.h"
 | |
| #include "styles/style_chat_helpers.h"
 | |
| #include "styles/style_settings.h"
 | |
| #include "styles/style_info.h"
 | |
| #include "styles/style_layers.h"
 | |
| #include "styles/style_menu_icons.h"
 | |
| 
 | |
| #include <QtGui/QGuiApplication>
 | |
| #include <QtGui/QClipboard>
 | |
| 
 | |
| namespace Settings {
 | |
| namespace {
 | |
| 
 | |
| using UserPrivacy = Api::UserPrivacy;
 | |
| using PrivacyRule = Api::UserPrivacy::Rule;
 | |
| using Option = EditPrivacyBox::Option;
 | |
| 
 | |
| [[nodiscard]] QString PublicLinkByPhone(not_null<UserData*> user) {
 | |
| 	return user->session().createInternalLinkFull('+' + user->phone());
 | |
| }
 | |
| 
 | |
| class BlockPeerBoxController final : public ChatsListBoxController {
 | |
| public:
 | |
| 	explicit BlockPeerBoxController(not_null<Main::Session*> session);
 | |
| 
 | |
| 	Main::Session &session() const override;
 | |
| 	void rowClicked(not_null<PeerListRow*> row) override;
 | |
| 
 | |
| 	void setBlockPeerCallback(Fn<void(not_null<PeerData*> peer)> callback) {
 | |
| 		_blockPeerCallback = std::move(callback);
 | |
| 	}
 | |
| 
 | |
| protected:
 | |
| 	void prepareViewHook() override;
 | |
| 	std::unique_ptr<Row> createRow(not_null<History*> history) override;
 | |
| 	void updateRowHook(not_null<Row*> row) override {
 | |
| 		updateIsBlocked(row, row->peer());
 | |
| 		delegate()->peerListUpdateRow(row);
 | |
| 	}
 | |
| 
 | |
| private:
 | |
| 	void updateIsBlocked(not_null<PeerListRow*> row, PeerData *peer) const;
 | |
| 
 | |
| 	const not_null<Main::Session*> _session;
 | |
| 	Fn<void(not_null<PeerData*> peer)> _blockPeerCallback;
 | |
| 
 | |
| };
 | |
| 
 | |
| BlockPeerBoxController::BlockPeerBoxController(
 | |
| 	not_null<Main::Session*> session)
 | |
| : ChatsListBoxController(session)
 | |
| , _session(session) {
 | |
| }
 | |
| 
 | |
| Main::Session &BlockPeerBoxController::session() const {
 | |
| 	return *_session;
 | |
| }
 | |
| 
 | |
| void BlockPeerBoxController::prepareViewHook() {
 | |
| 	delegate()->peerListSetTitle(tr::lng_blocked_list_add_title());
 | |
| 	session().changes().peerUpdates(
 | |
| 		Data::PeerUpdate::Flag::IsBlocked
 | |
| 	) | rpl::start_with_next([=](const Data::PeerUpdate &update) {
 | |
| 		if (auto row = delegate()->peerListFindRow(update.peer->id.value)) {
 | |
| 			updateIsBlocked(row, update.peer);
 | |
| 			delegate()->peerListUpdateRow(row);
 | |
| 		}
 | |
| 	}, lifetime());
 | |
| }
 | |
| 
 | |
| void BlockPeerBoxController::updateIsBlocked(
 | |
| 		not_null<PeerListRow*> row,
 | |
| 		PeerData *peer) const {
 | |
| 	if (!peer) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto blocked = peer->isBlocked();
 | |
| 	row->setDisabledState(blocked
 | |
| 		? PeerListRow::State::DisabledChecked
 | |
| 		: PeerListRow::State::Active);
 | |
| 	if (blocked) {
 | |
| 		row->setCustomStatus(tr::lng_blocked_list_already_blocked(tr::now));
 | |
| 	} else {
 | |
| 		row->clearCustomStatus();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void BlockPeerBoxController::rowClicked(not_null<PeerListRow*> row) {
 | |
| 	_blockPeerCallback(row->peer());
 | |
| }
 | |
| 
 | |
| auto BlockPeerBoxController::createRow(not_null<History*> history)
 | |
| -> std::unique_ptr<BlockPeerBoxController::Row> {
 | |
| 	if (!history->peer->isUser()
 | |
| 		|| history->peer->isServiceUser()
 | |
| 		|| history->peer->isSelf()
 | |
| 		|| history->peer->isRepliesChat()) {
 | |
| 		return nullptr;
 | |
| 	}
 | |
| 	auto row = std::make_unique<Row>(history);
 | |
| 	updateIsBlocked(row.get(), history->peer);
 | |
| 	return row;
 | |
| }
 | |
| 
 | |
| AdminLog::OwnedItem GenerateForwardedItem(
 | |
| 		not_null<HistoryView::ElementDelegate*> delegate,
 | |
| 		not_null<History*> history,
 | |
| 		const QString &text) {
 | |
| 	Expects(history->peer->isUser());
 | |
| 
 | |
| 	using Flag = MTPDmessage::Flag;
 | |
| 	const auto flags = Flag::f_from_id | Flag::f_fwd_from;
 | |
| 	const auto item = MTP_message(
 | |
| 		MTP_flags(flags),
 | |
| 		MTP_int(0), // Not used (would've been trimmed to 32 bits).
 | |
| 		peerToMTP(history->peer->id),
 | |
| 		MTPint(), // from_boosts_applied
 | |
| 		peerToMTP(history->peer->id),
 | |
| 		MTPPeer(), // saved_peer_id
 | |
| 		MTP_messageFwdHeader(
 | |
| 			MTP_flags(MTPDmessageFwdHeader::Flag::f_from_id),
 | |
| 			peerToMTP(history->session().userPeerId()),
 | |
| 			MTPstring(), // from_name
 | |
| 			MTP_int(base::unixtime::now()),
 | |
| 			MTPint(), // channel_post
 | |
| 			MTPstring(), // post_author
 | |
| 			MTPPeer(), // saved_from_peer
 | |
| 			MTPint(), // saved_from_msg_id
 | |
| 			MTPPeer(), // saved_from_id
 | |
| 			MTPstring(), // saved_from_name
 | |
| 			MTPint(), // saved_date
 | |
| 			MTPstring()), // psa_type
 | |
| 		MTPlong(), // via_bot_id
 | |
| 		MTPlong(), // via_business_bot_id
 | |
| 		MTPMessageReplyHeader(),
 | |
| 		MTP_int(base::unixtime::now()), // date
 | |
| 		MTP_string(text),
 | |
| 		MTPMessageMedia(),
 | |
| 		MTPReplyMarkup(),
 | |
| 		MTPVector<MTPMessageEntity>(),
 | |
| 		MTPint(), // views
 | |
| 		MTPint(), // forwards
 | |
| 		MTPMessageReplies(),
 | |
| 		MTPint(), // edit_date
 | |
| 		MTPstring(), // post_author
 | |
| 		MTPlong(), // grouped_id
 | |
| 		MTPMessageReactions(),
 | |
| 		MTPVector<MTPRestrictionReason>(),
 | |
| 		MTPint(), // ttl_period
 | |
| 		MTPint() // quick_reply_shortcut_id
 | |
| 	).match([&](const MTPDmessage &data) {
 | |
| 		return history->makeMessage(
 | |
| 			history->nextNonHistoryEntryId(),
 | |
| 			data,
 | |
| 			MessageFlag::FakeHistoryItem);
 | |
| 	}, [](auto &&) -> not_null<HistoryItem*> {
 | |
| 		Unexpected("Type in GenerateForwardedItem.");
 | |
| 	});
 | |
| 
 | |
| 	return AdminLog::OwnedItem(delegate, item);
 | |
| }
 | |
| 
 | |
| struct ForwardedTooltip {
 | |
| 	QRect geometry;
 | |
| 	Fn<void(QPainter&)> paint;
 | |
| };
 | |
| [[nodiscard]] ForwardedTooltip PrepareForwardedTooltip(
 | |
| 		not_null<HistoryView::Element*> view,
 | |
| 		Option value) {
 | |
| 	// This breaks HistoryView::Element encapsulation :(
 | |
| 	const auto forwarded = view->data()->Get<HistoryMessageForwarded>();
 | |
| 	const auto availableWidth = view->width()
 | |
| 		- st::msgMargin.left()
 | |
| 		- st::msgMargin.right();
 | |
| 	const auto bubbleWidth = ranges::min({
 | |
| 		availableWidth,
 | |
| 		view->maxWidth(),
 | |
| 		st::msgMaxWidth
 | |
| 	});
 | |
| 	const auto innerWidth = bubbleWidth
 | |
| 		- st::msgPadding.left()
 | |
| 		- st::msgPadding.right();
 | |
| 	const auto phrase = tr::lng_forwarded(
 | |
| 		tr::now,
 | |
| 		lt_user,
 | |
| 		view->history()->session().user()->name());
 | |
| 	const auto kReplacementPosition = QChar(0x0001);
 | |
| 	const auto possiblePosition = tr::lng_forwarded(
 | |
| 		tr::now,
 | |
| 		lt_user,
 | |
| 		QString(1, kReplacementPosition)
 | |
| 	).indexOf(kReplacementPosition);
 | |
| 	const auto position = (possiblePosition >= 0
 | |
| 		&& possiblePosition < phrase.size())
 | |
| 		? possiblePosition
 | |
| 		: 0;
 | |
| 	const auto before = phrase.mid(0, position);
 | |
| 	const auto skip = st::msgMargin.left() + st::msgPadding.left();
 | |
| 	const auto small = forwarded->text.countHeight(innerWidth)
 | |
| 		< 2 * st::msgServiceFont->height;
 | |
| 	const auto nameLeft = skip
 | |
| 		+ (small ? st::msgServiceFont->width(before) : 0);
 | |
| 	const auto right = skip + innerWidth;
 | |
| 	const auto text = [&] {
 | |
| 		switch (value) {
 | |
| 		case Option::Everyone:
 | |
| 			return tr::lng_edit_privacy_forwards_sample_everyone(tr::now);
 | |
| 		case Option::Contacts:
 | |
| 		case Option::CloseFriends:
 | |
| 			return tr::lng_edit_privacy_forwards_sample_contacts(tr::now);
 | |
| 		case Option::Nobody:
 | |
| 			return tr::lng_edit_privacy_forwards_sample_nobody(tr::now);
 | |
| 		}
 | |
| 		Unexpected("Option value in ForwardsPrivacyController.");
 | |
| 	}();
 | |
| 	const auto &font = st::defaultToast.style.font;
 | |
| 	const auto textWidth = font->width(text);
 | |
| 	const auto arrowSkip = st::settingsForwardPrivacyArrowSkip;
 | |
| 	const auto arrowSize = st::settingsForwardPrivacyArrowSize;
 | |
| 	const auto padding = st::settingsForwardPrivacyTooltipPadding;
 | |
| 	const auto rect = QRect(0, 0, textWidth, font->height).marginsAdded(
 | |
| 		padding
 | |
| 	).translated(padding.left(), padding.top());
 | |
| 
 | |
| 	const auto top = st::settingsForwardPrivacyPadding
 | |
| 		+ view->marginTop()
 | |
| 		+ st::msgPadding.top()
 | |
| 		- arrowSize
 | |
| 		- rect.height();
 | |
| 	const auto left1 = std::min(nameLeft, right - rect.width());
 | |
| 	const auto left2 = std::max(left1, skip);
 | |
| 	const auto left = left2;
 | |
| 	const auto arrowLeft1 = nameLeft + arrowSkip;
 | |
| 	const auto arrowLeft2 = std::min(
 | |
| 		arrowLeft1,
 | |
| 		std::max((left + right) / 2, right - arrowSkip));
 | |
| 	const auto arrowLeft = arrowLeft2;
 | |
| 	const auto geometry = rect.translated(left, top);
 | |
| 
 | |
| 	const auto line = st::lineWidth;
 | |
| 	const auto full = geometry.marginsAdded(
 | |
| 		{ line, line, line, line + arrowSize });
 | |
| 	const auto origin = full.topLeft();
 | |
| 
 | |
| 	const auto rounded = std::make_shared<Ui::RoundRect>(
 | |
| 		ImageRoundRadius::Large,
 | |
| 		st::toastBg);
 | |
| 	const auto paint = [=](QPainter &p) {
 | |
| 		p.translate(-origin);
 | |
| 
 | |
| 		rounded->paint(p, geometry);
 | |
| 
 | |
| 		p.setFont(font);
 | |
| 		p.setPen(st::toastFg);
 | |
| 		p.drawText(
 | |
| 			geometry.x() + padding.left(),
 | |
| 			geometry.y() + padding.top() + font->ascent,
 | |
| 			text);
 | |
| 
 | |
| 		const auto bottom = full.y() + full.height() - line;
 | |
| 
 | |
| 		QPainterPath path;
 | |
| 		path.moveTo(arrowLeft - arrowSize, bottom - arrowSize);
 | |
| 		path.lineTo(arrowLeft, bottom);
 | |
| 		path.lineTo(arrowLeft + arrowSize, bottom - arrowSize);
 | |
| 		path.lineTo(arrowLeft - arrowSize, bottom - arrowSize);
 | |
| 		{
 | |
| 			PainterHighQualityEnabler hq(p);
 | |
| 			p.setPen(Qt::NoPen);
 | |
| 			p.fillPath(path, st::toastBg);
 | |
| 		}
 | |
| 	};
 | |
| 	return { .geometry = full, .paint = paint };
 | |
| }
 | |
| 
 | |
| } // namespace
 | |
| 
 | |
| BlockedBoxController::BlockedBoxController(
 | |
| 	not_null<Window::SessionController*> window)
 | |
| : _window(window) {
 | |
| }
 | |
| 
 | |
| Main::Session &BlockedBoxController::session() const {
 | |
| 	return _window->session();
 | |
| }
 | |
| 
 | |
| void BlockedBoxController::prepare() {
 | |
| 	delegate()->peerListSetTitle(tr::lng_blocked_list_title());
 | |
| 	setDescriptionText(tr::lng_contacts_loading(tr::now));
 | |
| 	delegate()->peerListRefreshRows();
 | |
| 
 | |
| 	session().changes().peerUpdates(
 | |
| 		Data::PeerUpdate::Flag::IsBlocked
 | |
| 	) | rpl::start_with_next([=](const Data::PeerUpdate &update) {
 | |
| 		handleBlockedEvent(update.peer);
 | |
| 	}, lifetime());
 | |
| 
 | |
| 	session().api().blockedPeers().slice(
 | |
| 	) | rpl::take(
 | |
| 		1
 | |
| 	) | rpl::start_with_next([=](const Api::BlockedPeers::Slice &result) {
 | |
| 		setDescriptionText(tr::lng_blocked_list_about(tr::now));
 | |
| 		applySlice(result);
 | |
| 		loadMoreRows();
 | |
| 	}, lifetime());
 | |
| }
 | |
| 
 | |
| void BlockedBoxController::loadMoreRows() {
 | |
| 	if (_allLoaded) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	session().api().blockedPeers().request(
 | |
| 		_offset,
 | |
| 		crl::guard(&_guard, [=](const Api::BlockedPeers::Slice &slice) {
 | |
| 			applySlice(slice);
 | |
| 		}));
 | |
| }
 | |
| 
 | |
| void BlockedBoxController::rowClicked(not_null<PeerListRow*> row) {
 | |
| 	const auto peer = row->peer();
 | |
| 	const auto window = _window;
 | |
| 	crl::on_main(window, [=] {
 | |
| 		window->showPeerHistory(peer);
 | |
| 	});
 | |
| }
 | |
| 
 | |
| void BlockedBoxController::rowRightActionClicked(not_null<PeerListRow*> row) {
 | |
| 	session().api().blockedPeers().unblock(row->peer());
 | |
| }
 | |
| 
 | |
| void BlockedBoxController::applySlice(const Api::BlockedPeers::Slice &slice) {
 | |
| 	if (slice.list.empty()) {
 | |
| 		_allLoaded = true;
 | |
| 	}
 | |
| 
 | |
| 	_offset += slice.list.size();
 | |
| 	for (const auto &item : slice.list) {
 | |
| 		if (const auto peer = session().data().peerLoaded(item.id)) {
 | |
| 			appendRow(peer);
 | |
| 			peer->setIsBlocked(true);
 | |
| 		}
 | |
| 	}
 | |
| 	if (_offset >= slice.total) {
 | |
| 		_allLoaded = true;
 | |
| 	}
 | |
| 	delegate()->peerListRefreshRows();
 | |
| }
 | |
| 
 | |
| void BlockedBoxController::handleBlockedEvent(not_null<PeerData*> user) {
 | |
| 	if (user->isBlocked()) {
 | |
| 		if (prependRow(user)) {
 | |
| 			delegate()->peerListRefreshRows();
 | |
| 			delegate()->peerListScrollToTop();
 | |
| 		}
 | |
| 	} else if (auto row = delegate()->peerListFindRow(user->id.value)) {
 | |
| 		delegate()->peerListRemoveRow(row);
 | |
| 		delegate()->peerListRefreshRows();
 | |
| 		_rowsCountChanges.fire(delegate()->peerListFullRowsCount());
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void BlockedBoxController::BlockNewPeer(
 | |
| 		not_null<Window::SessionController*> window) {
 | |
| 	auto controller = std::make_unique<BlockPeerBoxController>(
 | |
| 		&window->session());
 | |
| 	auto initBox = [=, controller = controller.get()](
 | |
| 			not_null<PeerListBox*> box) {
 | |
| 		controller->setBlockPeerCallback([=](not_null<PeerData*> peer) {
 | |
| 			window->session().api().blockedPeers().block(peer);
 | |
| 			box->closeBox();
 | |
| 		});
 | |
| 		box->addButton(tr::lng_cancel(), [box] { box->closeBox(); });
 | |
| 	};
 | |
| 	window->show(
 | |
| 		Box<PeerListBox>(std::move(controller), std::move(initBox)));
 | |
| }
 | |
| 
 | |
| bool BlockedBoxController::appendRow(not_null<PeerData*> peer) {
 | |
| 	if (delegate()->peerListFindRow(peer->id.value)) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	delegate()->peerListAppendRow(createRow(peer));
 | |
| 	_rowsCountChanges.fire(delegate()->peerListFullRowsCount());
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| bool BlockedBoxController::prependRow(not_null<PeerData*> peer) {
 | |
| 	if (delegate()->peerListFindRow(peer->id.value)) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	delegate()->peerListPrependRow(createRow(peer));
 | |
| 	_rowsCountChanges.fire(delegate()->peerListFullRowsCount());
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| std::unique_ptr<PeerListRow> BlockedBoxController::createRow(
 | |
| 		not_null<PeerData*> peer) const {
 | |
| 	auto row = std::make_unique<PeerListRowWithLink>(peer);
 | |
| 	row->setActionLink(tr::lng_blocked_list_unblock(tr::now));
 | |
| 	const auto status = [&] {
 | |
| 		const auto user = peer->asUser();
 | |
| 		if (!user) {
 | |
| 			return tr::lng_group_status(tr::now);
 | |
| 		} else if (user->isInaccessible()) {
 | |
| 			return ktr("ktg_user_status_unaccessible");
 | |
| 		} else if (!user->phone().isEmpty()) {
 | |
| 			return Ui::FormatPhone(user->phone());
 | |
| 		} else if (!user->username().isEmpty()) {
 | |
| 			return '@' + user->username();
 | |
| 		} else if (user->isBot()) {
 | |
| 			return tr::lng_status_bot(tr::now);
 | |
| 		}
 | |
| 		return tr::lng_blocked_list_unknown_phone(tr::now);
 | |
| 	}();
 | |
| 	row->setCustomStatus(status);
 | |
| 	return row;
 | |
| }
 | |
| 
 | |
| rpl::producer<int> BlockedBoxController::rowsCountChanges() const {
 | |
| 	return _rowsCountChanges.events();
 | |
| }
 | |
| 
 | |
| PhoneNumberPrivacyController::PhoneNumberPrivacyController(
 | |
| 	not_null<Window::SessionController*> controller)
 | |
| : _controller(controller) {
 | |
| }
 | |
| 
 | |
| UserPrivacy::Key PhoneNumberPrivacyController::key() const {
 | |
| 	return Key::PhoneNumber;
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> PhoneNumberPrivacyController::title() const {
 | |
| 	return tr::lng_edit_privacy_phone_number_title();
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> PhoneNumberPrivacyController::optionsTitleKey() const {
 | |
| 	return tr::lng_edit_privacy_phone_number_header();
 | |
| }
 | |
| 
 | |
| auto PhoneNumberPrivacyController::warning() const
 | |
| -> rpl::producer<TextWithEntities> {
 | |
| 	using namespace rpl::mappers;
 | |
| 	const auto self = _controller->session().user();
 | |
| 	return rpl::combine(
 | |
| 		_phoneNumberOption.value(),
 | |
| 		_addedByPhone.value(),
 | |
| 		(_1 == Option::Nobody) && (_2 != Option::Everyone)
 | |
| 	) | rpl::map([=](bool onlyContactsSee) {
 | |
| 		return onlyContactsSee
 | |
| 			? tr::lng_edit_privacy_phone_number_contacts(
 | |
| 				Ui::Text::WithEntities)
 | |
| 			: rpl::combine(
 | |
| 				tr::lng_edit_privacy_phone_number_warning(),
 | |
| 				tr::lng_username_link()
 | |
| 			) | rpl::map([=](const QString &warning, const QString &added) {
 | |
| 				auto base = TextWithEntities{
 | |
| 					warning + "\n\n" + added + "\n",
 | |
| 				};
 | |
| 				const auto link = PublicLinkByPhone(self);
 | |
| 				return base.append(Ui::Text::Link(link, link));
 | |
| 			});
 | |
| 	}) | rpl::flatten_latest();
 | |
| }
 | |
| 
 | |
| void PhoneNumberPrivacyController::prepareWarningLabel(
 | |
| 		not_null<Ui::FlatLabel*> warning) const {
 | |
| 	warning->overrideLinkClickHandler([=] {
 | |
| 		QGuiApplication::clipboard()->setText(PublicLinkByPhone(
 | |
| 			_controller->session().user()));
 | |
| 		_controller->window().showToast(
 | |
| 			tr::lng_username_copied(tr::now));
 | |
| 	});
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> PhoneNumberPrivacyController::exceptionButtonTextKey(
 | |
| 		Exception exception) const {
 | |
| 	switch (exception) {
 | |
| 	case Exception::Always:
 | |
| 		return tr::lng_edit_privacy_phone_number_always_empty();
 | |
| 	case Exception::Never:
 | |
| 		return tr::lng_edit_privacy_phone_number_never_empty();
 | |
| 	}
 | |
| 	Unexpected("Invalid exception value.");
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> PhoneNumberPrivacyController::exceptionBoxTitle(
 | |
| 		Exception exception) const {
 | |
| 	switch (exception) {
 | |
| 	case Exception::Always: {
 | |
| 		return tr::lng_edit_privacy_phone_number_always_title();
 | |
| 	};
 | |
| 	case Exception::Never: {
 | |
| 		return tr::lng_edit_privacy_phone_number_never_title();
 | |
| 	};
 | |
| 	}
 | |
| 	Unexpected("Invalid exception value.");
 | |
| }
 | |
| 
 | |
| auto PhoneNumberPrivacyController::exceptionsDescription() const
 | |
| -> rpl::producer<QString> {
 | |
| 	return tr::lng_edit_privacy_phone_number_exceptions();
 | |
| }
 | |
| 
 | |
| object_ptr<Ui::RpWidget> PhoneNumberPrivacyController::setupMiddleWidget(
 | |
| 		not_null<Window::SessionController*> controller,
 | |
| 		not_null<QWidget*> parent,
 | |
| 		rpl::producer<Option> optionValue) {
 | |
| 	const auto key = UserPrivacy::Key::AddedByPhone;
 | |
| 	controller->session().api().userPrivacy().reload(key);
 | |
| 
 | |
| 	_phoneNumberOption = std::move(optionValue);
 | |
| 
 | |
| 	auto widget = object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
 | |
| 		parent,
 | |
| 		object_ptr<Ui::VerticalLayout>(parent));
 | |
| 
 | |
| 	const auto container = widget->entity();
 | |
| 	Ui::AddSkip(container);
 | |
| 	Ui::AddSubsectionTitle(
 | |
| 		container,
 | |
| 		tr::lng_edit_privacy_phone_number_find());
 | |
| 	const auto group = std::make_shared<Ui::RadioenumGroup<Option>>();
 | |
| 	group->setChangedCallback([=](Option value) {
 | |
| 		_addedByPhone = value;
 | |
| 	});
 | |
| 	controller->session().api().userPrivacy().value(
 | |
| 		key
 | |
| 	) | rpl::take(
 | |
| 		1
 | |
| 	) | rpl::start_with_next([=](const PrivacyRule &value) {
 | |
| 		group->setValue(value.option);
 | |
| 	}, widget->lifetime());
 | |
| 
 | |
| 	const auto addOption = [&](Option option) {
 | |
| 		return EditPrivacyBox::AddOption(container, this, group, option);
 | |
| 	};
 | |
| 	addOption(Option::Everyone);
 | |
| 	addOption(Option::Contacts);
 | |
| 	Ui::AddSkip(
 | |
| 		container,
 | |
| 		st::defaultVerticalListSkip + st::settingsPrivacySkipTop);
 | |
| 	Ui::AddDivider(container);
 | |
| 
 | |
| 	using namespace rpl::mappers;
 | |
| 	widget->toggleOn(_phoneNumberOption.value(
 | |
| 	) | rpl::map(
 | |
| 		_1 == Option::Nobody
 | |
| 	));
 | |
| 
 | |
| 	_saveAdditional = [=] {
 | |
| 		controller->session().api().userPrivacy().save(
 | |
| 			Api::UserPrivacy::Key::AddedByPhone,
 | |
| 			Api::UserPrivacy::Rule{ .option = group->current() });
 | |
| 	};
 | |
| 
 | |
| 	return widget;
 | |
| }
 | |
| 
 | |
| void PhoneNumberPrivacyController::saveAdditional() {
 | |
| 	if (_saveAdditional) {
 | |
| 		_saveAdditional();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| LastSeenPrivacyController::LastSeenPrivacyController(
 | |
| 	not_null<::Main::Session*> session)
 | |
| : _session(session) {
 | |
| }
 | |
| 
 | |
| UserPrivacy::Key LastSeenPrivacyController::key() const {
 | |
| 	return Key::LastSeen;
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> LastSeenPrivacyController::title() const {
 | |
| 	return tr::lng_edit_privacy_lastseen_title();
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> LastSeenPrivacyController::optionsTitleKey() const {
 | |
| 	return tr::lng_edit_privacy_lastseen_header();
 | |
| }
 | |
| 
 | |
| auto LastSeenPrivacyController::warning() const
 | |
| -> rpl::producer<TextWithEntities> {
 | |
| 	return tr::lng_edit_privacy_lastseen_warning(Ui::Text::WithEntities);
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> LastSeenPrivacyController::exceptionButtonTextKey(
 | |
| 		Exception exception) const {
 | |
| 	switch (exception) {
 | |
| 	case Exception::Always:
 | |
| 		return tr::lng_edit_privacy_lastseen_always_empty();
 | |
| 	case Exception::Never:
 | |
| 		return tr::lng_edit_privacy_lastseen_never_empty();
 | |
| 	}
 | |
| 	Unexpected("Invalid exception value.");
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> LastSeenPrivacyController::exceptionBoxTitle(
 | |
| 		Exception exception) const {
 | |
| 	switch (exception) {
 | |
| 	case Exception::Always: {
 | |
| 		return tr::lng_edit_privacy_lastseen_always_title();
 | |
| 	};
 | |
| 	case Exception::Never: {
 | |
| 		return tr::lng_edit_privacy_lastseen_never_title();
 | |
| 	};
 | |
| 	}
 | |
| 	Unexpected("Invalid exception value.");
 | |
| }
 | |
| 
 | |
| auto LastSeenPrivacyController::exceptionsDescription() const
 | |
| -> rpl::producer<QString>{
 | |
| 	return tr::lng_edit_privacy_lastseen_exceptions();
 | |
| }
 | |
| 
 | |
| object_ptr<Ui::RpWidget> LastSeenPrivacyController::setupBelowWidget(
 | |
| 		not_null<Window::SessionController*> controller,
 | |
| 		not_null<QWidget*> parent,
 | |
| 		rpl::producer<Option> option) {
 | |
| 	using namespace rpl::mappers;
 | |
| 
 | |
| 	auto result = object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
 | |
| 		parent,
 | |
| 		object_ptr<Ui::VerticalLayout>(parent));
 | |
| 
 | |
| 	_option = std::move(option);
 | |
| 
 | |
| 	const auto content = result->entity();
 | |
| 
 | |
| 	Ui::AddSkip(content);
 | |
| 
 | |
| 	const auto privacy = &controller->session().api().globalPrivacy();
 | |
| 	content->add(object_ptr<Ui::SettingsButton>(
 | |
| 		content,
 | |
| 		tr::lng_edit_lastseen_hide_read_time(),
 | |
| 		st::settingsButtonNoIcon
 | |
| 	))->toggleOn(privacy->hideReadTime())->toggledValue(
 | |
| 	) | rpl::start_with_next([=](bool value) {
 | |
| 		_hideReadTime = value;
 | |
| 	}, content->lifetime());
 | |
| 
 | |
| 	Ui::AddSkip(content);
 | |
| 	Ui::AddDividerText(
 | |
| 		content,
 | |
| 		tr::lng_edit_lastseen_hide_read_time_about());
 | |
| 	if (!controller->session().premium()) {
 | |
| 		Ui::AddSkip(content);
 | |
| 		content->add(object_ptr<Ui::SettingsButton>(
 | |
| 			content,
 | |
| 			tr::lng_edit_lastseen_subscribe(),
 | |
| 			st::settingsButtonLightNoIcon
 | |
| 		))->setClickedCallback([=] {
 | |
| 			Settings::ShowPremium(controller, u"lastseen"_q);
 | |
| 		});
 | |
| 		Ui::AddSkip(content);
 | |
| 		Ui::AddDividerText(
 | |
| 			content,
 | |
| 			tr::lng_edit_lastseen_subscribe_about());
 | |
| 	}
 | |
| 
 | |
| 	result->toggleOn(rpl::combine(
 | |
| 		_option.value(),
 | |
| 		_exceptionsNever.value(),
 | |
| 		(_1 != Option::Everyone) || (_2 > 0)));
 | |
| 
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| void LastSeenPrivacyController::handleExceptionsChange(
 | |
| 		Exception exception,
 | |
| 		rpl::producer<int> value) {
 | |
| 	if (exception == Exception::Never) {
 | |
| 		_exceptionsNever = std::move(value);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void LastSeenPrivacyController::confirmSave(
 | |
| 		bool someAreDisallowed,
 | |
| 		Fn<void()> saveCallback) {
 | |
| 	if (someAreDisallowed && !Core::App().settings().lastSeenWarningSeen()) {
 | |
| 		auto callback = [
 | |
| 			=,
 | |
| 			saveCallback = std::move(saveCallback)
 | |
| 		](Fn<void()> &&close) {
 | |
| 			close();
 | |
| 			saveCallback();
 | |
| 			Core::App().settings().setLastSeenWarningSeen(true);
 | |
| 			Core::App().saveSettingsDelayed();
 | |
| 		};
 | |
| 		auto box = Ui::MakeConfirmBox({
 | |
| 			.text = tr::lng_edit_privacy_lastseen_warning(),
 | |
| 			.confirmed = std::move(callback),
 | |
| 			.confirmText = tr::lng_continue(),
 | |
| 		});
 | |
| 		Ui::show(std::move(box), Ui::LayerOption::KeepOther);
 | |
| 	} else {
 | |
| 		saveCallback();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void LastSeenPrivacyController::saveAdditional() {
 | |
| 	if (_option.current() == Option::Everyone
 | |
| 		&& !_exceptionsNever.current()) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto privacy = &_session->api().globalPrivacy();
 | |
| 	if (privacy->hideReadTimeCurrent() != _hideReadTime) {
 | |
| 		privacy->updateHideReadTime(_hideReadTime);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| UserPrivacy::Key GroupsInvitePrivacyController::key() const {
 | |
| 	return Key::Invites;
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> GroupsInvitePrivacyController::title() const {
 | |
| 	return tr::lng_edit_privacy_groups_title();
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> GroupsInvitePrivacyController::optionsTitleKey(
 | |
| 		) const {
 | |
| 	return tr::lng_edit_privacy_groups_header();
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> GroupsInvitePrivacyController::exceptionButtonTextKey(
 | |
| 		Exception exception) const {
 | |
| 	switch (exception) {
 | |
| 	case Exception::Always: return tr::lng_edit_privacy_groups_always_empty();
 | |
| 	case Exception::Never: return tr::lng_edit_privacy_groups_never_empty();
 | |
| 	}
 | |
| 	Unexpected("Invalid exception value.");
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> GroupsInvitePrivacyController::exceptionBoxTitle(
 | |
| 		Exception exception) const {
 | |
| 	switch (exception) {
 | |
| 	case Exception::Always: return tr::lng_edit_privacy_groups_always_title();
 | |
| 	case Exception::Never: return tr::lng_edit_privacy_groups_never_title();
 | |
| 	}
 | |
| 	Unexpected("Invalid exception value.");
 | |
| }
 | |
| 
 | |
| auto GroupsInvitePrivacyController::exceptionsDescription() const
 | |
| -> rpl::producer<QString> {
 | |
| 	return tr::lng_edit_privacy_groups_exceptions();
 | |
| }
 | |
| 
 | |
| bool GroupsInvitePrivacyController::allowPremiumsToggle(
 | |
| 		Exception exception) const {
 | |
| 	return (exception == Exception::Always);
 | |
| }
 | |
| 
 | |
| UserPrivacy::Key CallsPrivacyController::key() const {
 | |
| 	return Key::Calls;
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> CallsPrivacyController::title() const {
 | |
| 	return tr::lng_edit_privacy_calls_title();
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> CallsPrivacyController::optionsTitleKey() const {
 | |
| 	return tr::lng_edit_privacy_calls_header();
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> CallsPrivacyController::exceptionButtonTextKey(
 | |
| 		Exception exception) const {
 | |
| 	switch (exception) {
 | |
| 	case Exception::Always: return tr::lng_edit_privacy_calls_always_empty();
 | |
| 	case Exception::Never: return tr::lng_edit_privacy_calls_never_empty();
 | |
| 	}
 | |
| 	Unexpected("Invalid exception value.");
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> CallsPrivacyController::exceptionBoxTitle(
 | |
| 		Exception exception) const {
 | |
| 	switch (exception) {
 | |
| 	case Exception::Always: return tr::lng_edit_privacy_calls_always_title();
 | |
| 	case Exception::Never: return tr::lng_edit_privacy_calls_never_title();
 | |
| 	}
 | |
| 	Unexpected("Invalid exception value.");
 | |
| }
 | |
| 
 | |
| auto CallsPrivacyController::exceptionsDescription() const
 | |
| -> rpl::producer<QString>{
 | |
| 	return tr::lng_edit_privacy_calls_exceptions();
 | |
| }
 | |
| 
 | |
| object_ptr<Ui::RpWidget> CallsPrivacyController::setupBelowWidget(
 | |
| 		not_null<Window::SessionController*> controller,
 | |
| 		not_null<QWidget*> parent,
 | |
| 		rpl::producer<Option> option) {
 | |
| 	auto result = object_ptr<Ui::VerticalLayout>(parent);
 | |
| 	const auto content = result.data();
 | |
| 
 | |
| 	Ui::AddSkip(content, st::settingsPeerToPeerSkip);
 | |
| 	Ui::AddSubsectionTitle(
 | |
| 		content,
 | |
| 		tr::lng_settings_calls_peer_to_peer_title());
 | |
| 	Settings::AddPrivacyButton(
 | |
| 		controller,
 | |
| 		content,
 | |
| 		tr::lng_settings_calls_peer_to_peer_button(),
 | |
| 		{ &st::menuIconNetwork },
 | |
| 		UserPrivacy::Key::CallsPeer2Peer,
 | |
| 		[] { return std::make_unique<CallsPeer2PeerPrivacyController>(); },
 | |
| 		&st::settingsButton);
 | |
| 	Ui::AddSkip(content);
 | |
| 
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| UserPrivacy::Key CallsPeer2PeerPrivacyController::key() const {
 | |
| 	return Key::CallsPeer2Peer;
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> CallsPeer2PeerPrivacyController::title() const {
 | |
| 	return tr::lng_edit_privacy_calls_p2p_title();
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> CallsPeer2PeerPrivacyController::optionsTitleKey() const {
 | |
| 	return tr::lng_edit_privacy_calls_p2p_header();
 | |
| }
 | |
| 
 | |
| QString CallsPeer2PeerPrivacyController::optionLabel(
 | |
| 		EditPrivacyBox::Option option) const {
 | |
| 	switch (option) {
 | |
| 	case Option::Everyone:
 | |
| 		return tr::lng_edit_privacy_calls_p2p_everyone(tr::now);
 | |
| 	case Option::Contacts:
 | |
| 		return tr::lng_edit_privacy_calls_p2p_contacts(tr::now);
 | |
| 	case Option::CloseFriends:
 | |
| 		return tr::lng_edit_privacy_close_friends(tr::now); // unused
 | |
| 	case Option::Nobody:
 | |
| 		return tr::lng_edit_privacy_calls_p2p_nobody(tr::now);
 | |
| 	}
 | |
| 	Unexpected("Option value in optionsLabelKey.");
 | |
| }
 | |
| 
 | |
| auto CallsPeer2PeerPrivacyController::warning() const
 | |
| -> rpl::producer<TextWithEntities> {
 | |
| 	return tr::lng_settings_peer_to_peer_about(Ui::Text::WithEntities);
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> CallsPeer2PeerPrivacyController::exceptionButtonTextKey(
 | |
| 		Exception exception) const {
 | |
| 	switch (exception) {
 | |
| 	case Exception::Always: {
 | |
| 		return tr::lng_edit_privacy_calls_p2p_always_empty();
 | |
| 	};
 | |
| 	case Exception::Never: {
 | |
| 		return tr::lng_edit_privacy_calls_p2p_never_empty();
 | |
| 	};
 | |
| 	}
 | |
| 	Unexpected("Invalid exception value.");
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> CallsPeer2PeerPrivacyController::exceptionBoxTitle(
 | |
| 		Exception exception) const {
 | |
| 	switch (exception) {
 | |
| 	case Exception::Always: {
 | |
| 		return tr::lng_edit_privacy_calls_p2p_always_title();
 | |
| 	};
 | |
| 	case Exception::Never: {
 | |
| 		return tr::lng_edit_privacy_calls_p2p_never_title();
 | |
| 	};
 | |
| 	}
 | |
| 	Unexpected("Invalid exception value.");
 | |
| }
 | |
| 
 | |
| auto CallsPeer2PeerPrivacyController::exceptionsDescription() const
 | |
| -> rpl::producer<QString>{
 | |
| 	return tr::lng_edit_privacy_calls_p2p_exceptions();
 | |
| }
 | |
| 
 | |
| ForwardsPrivacyController::ForwardsPrivacyController(
 | |
| 	not_null<Window::SessionController*> controller)
 | |
| : SimpleElementDelegate(controller, [] {})
 | |
| , _controller(controller)
 | |
| , _chatStyle(
 | |
| 	std::make_unique<Ui::ChatStyle>(
 | |
| 		controller->session().colorIndicesValue())) {
 | |
| 	_chatStyle->apply(controller->defaultChatTheme().get());
 | |
| }
 | |
| 
 | |
| UserPrivacy::Key ForwardsPrivacyController::key() const {
 | |
| 	return Key::Forwards;
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> ForwardsPrivacyController::title() const {
 | |
| 	return tr::lng_edit_privacy_forwards_title();
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> ForwardsPrivacyController::optionsTitleKey() const {
 | |
| 	return tr::lng_edit_privacy_forwards_header();
 | |
| }
 | |
| 
 | |
| auto ForwardsPrivacyController::warning() const
 | |
| -> rpl::producer<TextWithEntities> {
 | |
| 	return tr::lng_edit_privacy_forwards_warning(Ui::Text::WithEntities);
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> ForwardsPrivacyController::exceptionButtonTextKey(
 | |
| 		Exception exception) const {
 | |
| 	switch (exception) {
 | |
| 	case Exception::Always: {
 | |
| 		return tr::lng_edit_privacy_forwards_always_empty();
 | |
| 	};
 | |
| 	case Exception::Never: {
 | |
| 		return tr::lng_edit_privacy_forwards_never_empty();
 | |
| 	};
 | |
| 	}
 | |
| 	Unexpected("Invalid exception value.");
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> ForwardsPrivacyController::exceptionBoxTitle(
 | |
| 		Exception exception) const {
 | |
| 	switch (exception) {
 | |
| 	case Exception::Always: {
 | |
| 		return tr::lng_edit_privacy_forwards_always_title();
 | |
| 	};
 | |
| 	case Exception::Never: {
 | |
| 		return tr::lng_edit_privacy_forwards_never_title();
 | |
| 	};
 | |
| 	}
 | |
| 	Unexpected("Invalid exception value.");
 | |
| }
 | |
| 
 | |
| auto ForwardsPrivacyController::exceptionsDescription() const
 | |
| -> rpl::producer<QString> {
 | |
| 	return tr::lng_edit_privacy_forwards_exceptions();
 | |
| }
 | |
| 
 | |
| object_ptr<Ui::RpWidget> ForwardsPrivacyController::setupAboveWidget(
 | |
| 		not_null<Window::SessionController*> controller,
 | |
| 		not_null<QWidget*> parent,
 | |
| 		rpl::producer<Option> optionValue,
 | |
| 		not_null<QWidget*> outerContainer) {
 | |
| 	using namespace rpl::mappers;
 | |
| 
 | |
| 	auto message = GenerateForwardedItem(
 | |
| 		delegate(),
 | |
| 		controller->session().data().history(
 | |
| 			PeerData::kServiceNotificationsId),
 | |
| 		tr::lng_edit_privacy_forwards_sample_message(tr::now));
 | |
| 	const auto view = message.get();
 | |
| 
 | |
| 	auto result = object_ptr<Ui::PaddingWrap<Ui::RpWidget>>(
 | |
| 		parent,
 | |
| 		object_ptr<Ui::RpWidget>(parent),
 | |
| 		style::margins(
 | |
| 			0,
 | |
| 			st::defaultVerticalListSkip,
 | |
| 			0,
 | |
| 			st::settingsPrivacySkipTop));
 | |
| 	const auto widget = result->entity();
 | |
| 
 | |
| 	struct State {
 | |
| 		AdminLog::OwnedItem item;
 | |
| 		Option option = {};
 | |
| 		base::unique_qptr<Ui::RpWidget> tooltip;
 | |
| 		ForwardedTooltip info;
 | |
| 		Fn<void()> refreshGeometry;
 | |
| 	};
 | |
| 	const auto state = widget->lifetime().make_state<State>();
 | |
| 	state->item = std::move(message);
 | |
| 	state->tooltip = base::make_unique_q<Ui::RpWidget>(outerContainer);
 | |
| 	state->tooltip->paintRequest(
 | |
| 	) | rpl::start_with_next([=] {
 | |
| 		if (state->info.paint) {
 | |
| 			auto p = QPainter(state->tooltip.get());
 | |
| 			state->info.paint(p);
 | |
| 		}
 | |
| 	}, state->tooltip->lifetime());
 | |
| 	state->refreshGeometry = [=] {
 | |
| 		state->tooltip->show();
 | |
| 		state->tooltip->raise();
 | |
| 		auto position = state->info.geometry.topLeft();
 | |
| 		auto parent = (QWidget*)widget;
 | |
| 		while (parent && parent != outerContainer) {
 | |
| 			position += parent->pos();
 | |
| 			parent = parent->parentWidget();
 | |
| 		}
 | |
| 		state->tooltip->move(position);
 | |
| 	};
 | |
| 	const auto watch = [&](QWidget *widget, const auto &self) -> void {
 | |
| 		if (!widget) {
 | |
| 			return;
 | |
| 		}
 | |
| 		base::install_event_filter(state->tooltip, widget, [=](
 | |
| 				not_null<QEvent*> e) {
 | |
| 			if (e->type() == QEvent::Move
 | |
| 				|| e->type() == QEvent::Show
 | |
| 				|| e->type() == QEvent::ShowToParent
 | |
| 				|| e->type() == QEvent::ZOrderChange) {
 | |
| 				state->refreshGeometry();
 | |
| 			}
 | |
| 			return base::EventFilterResult::Continue;
 | |
| 		});
 | |
| 		if (widget == outerContainer) {
 | |
| 			return;
 | |
| 		}
 | |
| 		self(widget->parentWidget(), self);
 | |
| 	};
 | |
| 	watch(widget, watch);
 | |
| 
 | |
| 	const auto padding = st::settingsForwardPrivacyPadding;
 | |
| 	widget->widthValue(
 | |
| 	) | rpl::filter(
 | |
| 		_1 >= (st::historyMinimalWidth / 2)
 | |
| 	) | rpl::start_with_next([=](int width) {
 | |
| 		const auto height = view->resizeGetHeight(width);
 | |
| 		const auto top = view->marginTop();
 | |
| 		const auto bottom = view->marginBottom();
 | |
| 		const auto full = padding + top + height + bottom + padding;
 | |
| 		widget->resize(width, full);
 | |
| 	}, widget->lifetime());
 | |
| 
 | |
| 	rpl::combine(
 | |
| 		widget->widthValue(),
 | |
| 		std::move(optionValue)
 | |
| 	) | rpl::start_with_next([=](int width, Option value) {
 | |
| 		state->info = PrepareForwardedTooltip(view, value);
 | |
| 		state->tooltip->resize(state->info.geometry.size());
 | |
| 		state->refreshGeometry();
 | |
| 		state->tooltip->update();
 | |
| 	}, state->tooltip->lifetime());
 | |
| 
 | |
| 	widget->paintRequest(
 | |
| 	) | rpl::start_with_next([=](QRect rect) {
 | |
| 		// #TODO themes
 | |
| 		Window::SectionWidget::PaintBackground(
 | |
| 			controller,
 | |
| 			controller->defaultChatTheme().get(), // #TODO themes
 | |
| 			widget,
 | |
| 			rect);
 | |
| 
 | |
| 		Painter p(widget);
 | |
| 		const auto theme = controller->defaultChatTheme().get();
 | |
| 		auto context = theme->preparePaintContext(
 | |
| 			_chatStyle.get(),
 | |
| 			widget->rect(),
 | |
| 			widget->rect(),
 | |
| 			controller->isGifPausedAtLeastFor(
 | |
| 				Window::GifPauseReason::Layer));
 | |
| 		p.translate(padding / 2, padding + view->marginBottom());
 | |
| 		context.outbg = view->hasOutLayout();
 | |
| 		view->draw(p, context);
 | |
| 	}, widget->lifetime());
 | |
| 
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| auto ForwardsPrivacyController::delegate()
 | |
| -> not_null<HistoryView::ElementDelegate*> {
 | |
| 	return static_cast<HistoryView::ElementDelegate*>(this);
 | |
| }
 | |
| 
 | |
| HistoryView::Context ForwardsPrivacyController::elementContext() {
 | |
| 	return HistoryView::Context::ContactPreview;
 | |
| }
 | |
| 
 | |
| UserPrivacy::Key ProfilePhotoPrivacyController::key() const {
 | |
| 	return Key::ProfilePhoto;
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> ProfilePhotoPrivacyController::title() const {
 | |
| 	return tr::lng_edit_privacy_profile_photo_title();
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> ProfilePhotoPrivacyController::optionsTitleKey() const {
 | |
| 	return tr::lng_edit_privacy_profile_photo_header();
 | |
| }
 | |
| 
 | |
| object_ptr<Ui::RpWidget> ProfilePhotoPrivacyController::setupAboveWidget(
 | |
| 		not_null<Window::SessionController*> controller,
 | |
| 		not_null<QWidget*> parent,
 | |
| 		rpl::producer<Option> optionValue,
 | |
| 		not_null<QWidget*> outerContainer) {
 | |
| 	_option = std::move(optionValue);
 | |
| 	return nullptr;
 | |
| }
 | |
| 
 | |
| object_ptr<Ui::RpWidget> ProfilePhotoPrivacyController::setupMiddleWidget(
 | |
| 		not_null<Window::SessionController*> controller,
 | |
| 		not_null<QWidget*> parent,
 | |
| 		rpl::producer<Option> optionValue) {
 | |
| 	const auto self = controller->session().user();
 | |
| 	auto widget = object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
 | |
| 		parent,
 | |
| 		object_ptr<Ui::VerticalLayout>(parent));
 | |
| 
 | |
| 	const auto container = widget->entity();
 | |
| 	struct State {
 | |
| 		void updatePhoto(QImage &&image, bool local) {
 | |
| 			auto result = image.scaled(
 | |
| 				userpicSize * style::DevicePixelRatio(),
 | |
| 				Qt::KeepAspectRatio,
 | |
| 				Qt::SmoothTransformation);
 | |
| 			result = Images::Round(
 | |
| 				std::move(result),
 | |
| 				ImageRoundRadius::Ellipse);
 | |
| 			result.setDevicePixelRatio(style::DevicePixelRatio());
 | |
| 			(local ? localPhoto : photo) = std::move(result);
 | |
| 			if (local) {
 | |
| 				localOriginal = std::move(image);
 | |
| 			}
 | |
| 			hasPhoto.fire(!localPhoto.isNull() || !photo.isNull());
 | |
| 		}
 | |
| 
 | |
| 		rpl::event_stream<bool> hasPhoto;
 | |
| 		rpl::variable<bool> hiddenByUser = false;
 | |
| 		rpl::variable<QString> setUserpicButtonText;
 | |
| 		QSize userpicSize;
 | |
| 		QImage photo;
 | |
| 		QImage localPhoto;
 | |
| 		QImage localOriginal;
 | |
| 	};
 | |
| 	const auto state = container->lifetime().make_state<State>();
 | |
| 	state->userpicSize = QSize(
 | |
| 		st::inviteLinkUserpics.size,
 | |
| 		st::inviteLinkUserpics.size);
 | |
| 
 | |
| 	Ui::AddSkip(container);
 | |
| 	const auto setUserpicButton = AddButtonWithIcon(
 | |
| 		container,
 | |
| 		state->setUserpicButtonText.value(),
 | |
| 		st::settingsButtonLight,
 | |
| 		{ &st::menuBlueIconPhotoSet });
 | |
| 	const auto &stRemoveButton = st::settingsAttentionButtonWithIcon;
 | |
| 	const auto removeButton = container->add(
 | |
| 		object_ptr<Ui::SlideWrap<Ui::SettingsButton>>(
 | |
| 			container,
 | |
| 			object_ptr<Ui::SettingsButton>(
 | |
| 				parent,
 | |
| 				tr::lng_edit_privacy_profile_photo_public_remove(),
 | |
| 				stRemoveButton)));
 | |
| 	Ui::AddSkip(container);
 | |
| 	Ui::AddDividerText(
 | |
| 		container,
 | |
| 		tr::lng_edit_privacy_profile_photo_public_about());
 | |
| 
 | |
| 	const auto userpic = Ui::CreateChild<Ui::RpWidget>(
 | |
| 		removeButton->entity());
 | |
| 	userpic->resize(state->userpicSize);
 | |
| 	userpic->paintRequest(
 | |
| 	) | rpl::start_with_next([=](const QRect &r) {
 | |
| 		auto p = QPainter(userpic);
 | |
| 		p.fillRect(r, Qt::transparent);
 | |
| 		if (!state->localPhoto.isNull()) {
 | |
| 			p.drawImage(0, 0, state->localPhoto);
 | |
| 		} else if (!state->photo.isNull()) {
 | |
| 			p.drawImage(0, 0, state->photo);
 | |
| 		}
 | |
| 	}, userpic->lifetime());
 | |
| 	removeButton->entity()->heightValue(
 | |
| 	) | rpl::start_with_next([=,
 | |
| 			left = stRemoveButton.iconLeft,
 | |
| 			width = st::menuBlueIconPhotoSet.width()](int height) {
 | |
| 		userpic->moveToLeft(
 | |
| 			left + (width - userpic->width()) / 2,
 | |
| 			(height - userpic->height()) / 2);
 | |
| 	}, userpic->lifetime());
 | |
| 	removeButton->toggleOn(rpl::combine(
 | |
| 		state->hasPhoto.events_starting_with(false),
 | |
| 		state->hiddenByUser.value()
 | |
| 	) | rpl::map(rpl::mappers::_1 && !rpl::mappers::_2));
 | |
| 
 | |
| 	(
 | |
| 		PrepareShortInfoFallbackUserpic(self, st::shortInfoCover).value
 | |
| 	) | rpl::start_with_next([=](PeerShortInfoUserpic info) {
 | |
| 		state->updatePhoto(base::take(info.photo), false);
 | |
| 		userpic->update();
 | |
| 	}, userpic->lifetime());
 | |
| 	setUserpicButton->setClickedCallback([=] {
 | |
| 		base::call_delayed(
 | |
| 			st::settingsButton.ripple.hideDuration,
 | |
| 			crl::guard(container, [=] {
 | |
| 				using namespace Editor;
 | |
| 				PrepareProfilePhotoFromFile(
 | |
| 					container,
 | |
| 					&controller->window(),
 | |
| 					{
 | |
| 						.confirm = tr::lng_profile_set_photo_button(tr::now),
 | |
| 						.cropType = EditorData::CropType::Ellipse,
 | |
| 						.keepAspectRatio = true,
 | |
| 					},
 | |
| 					[=](QImage &&image) {
 | |
| 						state->updatePhoto(std::move(image), true);
 | |
| 						state->hiddenByUser = false;
 | |
| 						userpic->update();
 | |
| 					});
 | |
| 			}));
 | |
| 	});
 | |
| 	removeButton->entity()->setClickedCallback([=] {
 | |
| 		state->hiddenByUser = true;
 | |
| 	});
 | |
| 	state->setUserpicButtonText = removeButton->toggledValue(
 | |
| 	) | rpl::map([](bool toggled) {
 | |
| 		return !toggled
 | |
| 			? tr::lng_edit_privacy_profile_photo_public_set()
 | |
| 			: tr::lng_edit_privacy_profile_photo_public_update();
 | |
| 	}) | rpl::flatten_latest();
 | |
| 
 | |
| 	_saveAdditional = [=] {
 | |
| 		if (removeButton->isHidden()) {
 | |
| 			if (const auto photoId = SyncUserFallbackPhotoViewer(self)) {
 | |
| 				if (const auto photo = self->owner().photo(*photoId)) {
 | |
| 					controller->session().api().peerPhoto().clear(photo);
 | |
| 				}
 | |
| 			}
 | |
| 		} else if (!state->localOriginal.isNull()) {
 | |
| 			controller->session().api().peerPhoto().uploadFallback(
 | |
| 				self,
 | |
| 				{ base::take(state->localOriginal) });
 | |
| 		}
 | |
| 	};
 | |
| 
 | |
| 	widget->toggleOn(rpl::combine(
 | |
| 		std::move(optionValue),
 | |
| 		_exceptionsNever.value()
 | |
| 	) | rpl::map(rpl::mappers::_1 != Option::Everyone || rpl::mappers::_2));
 | |
| 
 | |
| 	return widget;
 | |
| }
 | |
| 
 | |
| void ProfilePhotoPrivacyController::saveAdditional() {
 | |
| 	if (_saveAdditional) {
 | |
| 		_saveAdditional();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> ProfilePhotoPrivacyController::exceptionButtonTextKey(
 | |
| 		Exception exception) const {
 | |
| 	switch (exception) {
 | |
| 	case Exception::Always: {
 | |
| 		return tr::lng_edit_privacy_profile_photo_always_empty();
 | |
| 	};
 | |
| 	case Exception::Never: {
 | |
| 		return tr::lng_edit_privacy_profile_photo_never_empty();
 | |
| 	};
 | |
| 	}
 | |
| 	Unexpected("Invalid exception value.");
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> ProfilePhotoPrivacyController::exceptionBoxTitle(
 | |
| 		Exception exception) const {
 | |
| 	switch (exception) {
 | |
| 	case Exception::Always: {
 | |
| 		return tr::lng_edit_privacy_profile_photo_always_title();
 | |
| 	};
 | |
| 	case Exception::Never: {
 | |
| 		return tr::lng_edit_privacy_profile_photo_never_title();
 | |
| 	};
 | |
| 	}
 | |
| 	Unexpected("Invalid exception value.");
 | |
| }
 | |
| 
 | |
| auto ProfilePhotoPrivacyController::exceptionsDescription() const
 | |
| -> rpl::producer<QString> {
 | |
| 	return _option.value(
 | |
| 	) | rpl::map([](Option option) {
 | |
| 		switch (option) {
 | |
| 		case Option::Everyone:
 | |
| 			return tr::lng_edit_privacy_forwards_exceptions_everyone();
 | |
| 		case Option::Contacts:
 | |
| 		case Option::CloseFriends:
 | |
| 			return tr::lng_edit_privacy_forwards_exceptions();
 | |
| 		case Option::Nobody:
 | |
| 			return tr::lng_edit_privacy_forwards_exceptions_nobody();
 | |
| 		}
 | |
| 		Unexpected("Option value in exceptionsDescription.");
 | |
| 	}) | rpl::flatten_latest();
 | |
| }
 | |
| 
 | |
| 
 | |
| void ProfilePhotoPrivacyController::handleExceptionsChange(
 | |
| 		Exception exception,
 | |
| 		rpl::producer<int> value) {
 | |
| 	if (exception == Exception::Never) {
 | |
| 		_exceptionsNever = std::move(value);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| VoicesPrivacyController::VoicesPrivacyController(
 | |
| 		not_null<::Main::Session*> session) {
 | |
| 	Data::AmPremiumValue(
 | |
| 		session
 | |
| 	) | rpl::start_with_next([=](bool premium) {
 | |
| 		if (!premium) {
 | |
| 			if (const auto box = view()) {
 | |
| 				box->closeBox();
 | |
| 			}
 | |
| 		}
 | |
| 	}, _lifetime);
 | |
| }
 | |
| 
 | |
| UserPrivacy::Key VoicesPrivacyController::key() const {
 | |
| 	return Key::Voices;
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> VoicesPrivacyController::title() const {
 | |
| 	return tr::lng_edit_privacy_voices_title();
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> VoicesPrivacyController::optionsTitleKey() const {
 | |
| 	return tr::lng_edit_privacy_voices_header();
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> VoicesPrivacyController::exceptionButtonTextKey(
 | |
| 		Exception exception) const {
 | |
| 	switch (exception) {
 | |
| 	case Exception::Always: return tr::lng_edit_privacy_voices_always_empty();
 | |
| 	case Exception::Never: return tr::lng_edit_privacy_voices_never_empty();
 | |
| 	}
 | |
| 	Unexpected("Invalid exception value.");
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> VoicesPrivacyController::exceptionBoxTitle(
 | |
| 		Exception exception) const {
 | |
| 	switch (exception) {
 | |
| 	case Exception::Always: return tr::lng_edit_privacy_voices_always_title();
 | |
| 	case Exception::Never: return tr::lng_edit_privacy_voices_never_title();
 | |
| 	}
 | |
| 	Unexpected("Invalid exception value.");
 | |
| }
 | |
| 
 | |
| auto VoicesPrivacyController::exceptionsDescription() const
 | |
| -> rpl::producer<QString> {
 | |
| 	return tr::lng_edit_privacy_voices_exceptions();
 | |
| }
 | |
| 
 | |
| object_ptr<Ui::RpWidget> VoicesPrivacyController::setupBelowWidget(
 | |
| 		not_null<Window::SessionController*> controller,
 | |
| 		not_null<QWidget*> parent,
 | |
| 		rpl::producer<Option> option) {
 | |
| 	using namespace rpl::mappers;
 | |
| 
 | |
| 	auto result = object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
 | |
| 		parent,
 | |
| 		object_ptr<Ui::VerticalLayout>(parent));
 | |
| 	result->toggleOn(
 | |
| 		Data::AmPremiumValue(&controller->session()) | rpl::map(!_1),
 | |
| 		anim::type::instant);
 | |
| 
 | |
| 	const auto content = result->entity();
 | |
| 
 | |
| 	Ui::AddSkip(content);
 | |
| 	Settings::AddButtonWithIcon(
 | |
| 		content,
 | |
| 		tr::lng_messages_privacy_premium_button(),
 | |
| 		st::messagePrivacySubscribe,
 | |
| 		{ .icon = &st::menuBlueIconPremium }
 | |
| 	)->setClickedCallback([=] {
 | |
| 		Settings::ShowPremium(
 | |
| 			controller,
 | |
| 			u"voice_restrictions_require_premium"_q);
 | |
| 	});
 | |
| 	Ui::AddSkip(content);
 | |
| 	Ui::AddDividerText(content, tr::lng_messages_privacy_premium_about());
 | |
| 
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| Fn<void()> VoicesPrivacyController::premiumClickedCallback(
 | |
| 		Option option,
 | |
| 		not_null<Window::SessionController*> controller) {
 | |
| 	if (option == Option::Everyone) {
 | |
| 		return nullptr;
 | |
| 	}
 | |
| 	const auto showToast = [=] {
 | |
| 		auto link = Ui::Text::Link(
 | |
| 			Ui::Text::Semibold(
 | |
| 				tr::lng_settings_privacy_premium_link(tr::now)));
 | |
| 		_toastInstance = controller->showToast({
 | |
| 			.text = tr::lng_settings_privacy_premium(
 | |
| 				tr::now,
 | |
| 				lt_link,
 | |
| 				link,
 | |
| 				Ui::Text::WithEntities),
 | |
| 			.st = &st::defaultMultilineToast,
 | |
| 			.duration = Ui::Toast::kDefaultDuration * 2,
 | |
| 			.multiline = true,
 | |
| 			.filter = crl::guard(&controller->session(), [=](
 | |
| 					const ClickHandlerPtr &,
 | |
| 					Qt::MouseButton button) {
 | |
| 				if (button == Qt::LeftButton) {
 | |
| 					if (const auto strong = _toastInstance.get()) {
 | |
| 						strong->hideAnimated();
 | |
| 						_toastInstance = nullptr;
 | |
| 						Settings::ShowPremium(
 | |
| 							controller,
 | |
| 							u"voice_restrictions_require_premium"_q);
 | |
| 						return true;
 | |
| 					}
 | |
| 				}
 | |
| 				return false;
 | |
| 			}),
 | |
| 		});
 | |
| 	};
 | |
| 
 | |
| 	return showToast;
 | |
| }
 | |
| 
 | |
| UserPrivacy::Key AboutPrivacyController::key() const {
 | |
| 	return Key::About;
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> AboutPrivacyController::title() const {
 | |
| 	return tr::lng_edit_privacy_about_title();
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> AboutPrivacyController::optionsTitleKey() const {
 | |
| 	return tr::lng_edit_privacy_about_header();
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> AboutPrivacyController::exceptionButtonTextKey(
 | |
| 		Exception exception) const {
 | |
| 	switch (exception) {
 | |
| 	case Exception::Always: return tr::lng_edit_privacy_about_always_empty();
 | |
| 	case Exception::Never: return tr::lng_edit_privacy_about_never_empty();
 | |
| 	}
 | |
| 	Unexpected("Invalid exception value.");
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> AboutPrivacyController::exceptionBoxTitle(
 | |
| 		Exception exception) const {
 | |
| 	switch (exception) {
 | |
| 	case Exception::Always: return tr::lng_edit_privacy_about_always_title();
 | |
| 	case Exception::Never: return tr::lng_edit_privacy_about_never_title();
 | |
| 	}
 | |
| 	Unexpected("Invalid exception value.");
 | |
| }
 | |
| 
 | |
| auto AboutPrivacyController::exceptionsDescription() const
 | |
| -> rpl::producer<QString> {
 | |
| 	return tr::lng_edit_privacy_birthday_exceptions();
 | |
| }
 | |
| 
 | |
| UserPrivacy::Key BirthdayPrivacyController::key() const {
 | |
| 	return Key::Birthday;
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> BirthdayPrivacyController::title() const {
 | |
| 	return tr::lng_edit_privacy_birthday_title();
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> BirthdayPrivacyController::optionsTitleKey() const {
 | |
| 	return tr::lng_edit_privacy_birthday_header();
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> BirthdayPrivacyController::exceptionButtonTextKey(
 | |
| 	Exception exception) const {
 | |
| 	switch (exception) {
 | |
| 	case Exception::Always: return tr::lng_edit_privacy_birthday_always_empty();
 | |
| 	case Exception::Never: return tr::lng_edit_privacy_birthday_never_empty();
 | |
| 	}
 | |
| 	Unexpected("Invalid exception value.");
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> BirthdayPrivacyController::exceptionBoxTitle(
 | |
| 	Exception exception) const {
 | |
| 	switch (exception) {
 | |
| 	case Exception::Always: return tr::lng_edit_privacy_birthday_always_title();
 | |
| 	case Exception::Never: return tr::lng_edit_privacy_birthday_never_title();
 | |
| 	}
 | |
| 	Unexpected("Invalid exception value.");
 | |
| }
 | |
| 
 | |
| auto BirthdayPrivacyController::exceptionsDescription() const
 | |
| -> rpl::producer<QString> {
 | |
| 	return tr::lng_edit_privacy_birthday_exceptions();
 | |
| }
 | |
| 
 | |
| object_ptr<Ui::RpWidget> BirthdayPrivacyController::setupAboveWidget(
 | |
| 		not_null<Window::SessionController*> controller,
 | |
| 		not_null<QWidget*> parent,
 | |
| 		rpl::producer<Option> optionValue,
 | |
| 		not_null<QWidget*> outerContainer) {
 | |
| 	const auto session = &controller->session();
 | |
| 	const auto user = session->user();
 | |
| 	auto result = object_ptr<Ui::SlideWrap<Ui::DividerLabel>>(
 | |
| 		parent,
 | |
| 		object_ptr<Ui::DividerLabel>(
 | |
| 			parent,
 | |
| 			object_ptr<Ui::FlatLabel>(
 | |
| 				parent,
 | |
| 				tr::lng_edit_privacy_birthday_yet(
 | |
| 					lt_link,
 | |
| 					tr::lng_edit_privacy_birthday_yet_link(
 | |
| 					) | Ui::Text::ToLink("internal:edit_birthday"),
 | |
| 					Ui::Text::WithEntities),
 | |
| 				st::boxDividerLabel),
 | |
| 			st::defaultBoxDividerLabelPadding));
 | |
| 	result->toggleOn(session->changes().peerFlagsValue(
 | |
| 		user,
 | |
| 		Data::PeerUpdate::Flag::Birthday
 | |
| 	) | rpl::map([=] {
 | |
| 		return !user->birthday();
 | |
| 	}));
 | |
| 	result->finishAnimating();
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| } // namespace Settings
 |