776 lines
		
	
	
	
		
			20 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			776 lines
		
	
	
	
		
			20 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 "calls/calls_top_bar.h"
 | |
| 
 | |
| #include "ui/effects/cross_line.h"
 | |
| #include "ui/paint/blobs_linear.h"
 | |
| #include "ui/widgets/buttons.h"
 | |
| #include "ui/widgets/labels.h"
 | |
| #include "ui/chat/group_call_userpics.h" // Ui::GroupCallUser.
 | |
| #include "ui/chat/group_call_bar.h" // Ui::GroupCallBarContent.
 | |
| #include "ui/layers/generic_box.h"
 | |
| #include "ui/wrap/padding_wrap.h"
 | |
| #include "ui/text/format_values.h"
 | |
| #include "ui/toast/toast.h"
 | |
| #include "ui/power_saving.h"
 | |
| #include "lang/lang_keys.h"
 | |
| #include "core/application.h"
 | |
| #include "calls/calls_call.h"
 | |
| #include "calls/calls_instance.h"
 | |
| #include "calls/calls_signal_bars.h"
 | |
| #include "calls/group/calls_group_call.h"
 | |
| #include "calls/group/calls_group_menu.h" // Group::LeaveBox.
 | |
| #include "history/view/history_view_group_call_bar.h" // ContentByCall.
 | |
| #include "data/data_user.h"
 | |
| #include "data/data_group_call.h"
 | |
| #include "data/data_peer.h"
 | |
| #include "data/data_changes.h"
 | |
| #include "main/main_session.h"
 | |
| #include "boxes/abstract_box.h"
 | |
| #include "base/timer.h"
 | |
| #include "styles/style_calls.h"
 | |
| #include "styles/style_chat_helpers.h" // style::GroupCallUserpics
 | |
| #include "styles/style_layers.h"
 | |
| 
 | |
| namespace Calls {
 | |
| 
 | |
| enum class BarState {
 | |
| 	Connecting,
 | |
| 	Active,
 | |
| 	Muted,
 | |
| 	ForceMuted,
 | |
| };
 | |
| 
 | |
| namespace {
 | |
| 
 | |
| constexpr auto kUpdateDebugTimeoutMs = crl::time(500);
 | |
| constexpr auto kSwitchStateDuration = 120;
 | |
| 
 | |
| constexpr auto kMinorBlobAlpha = 76. / 255.;
 | |
| 
 | |
| constexpr auto kHideBlobsDuration = crl::time(500);
 | |
| constexpr auto kBlobLevelDuration = crl::time(250);
 | |
| constexpr auto kBlobUpdateInterval = crl::time(100);
 | |
| 
 | |
| auto BarStateFromMuteState(
 | |
| 		MuteState state,
 | |
| 		GroupCall::InstanceState instanceState,
 | |
| 		TimeId scheduledDate) {
 | |
| 	return scheduledDate
 | |
| 		? BarState::ForceMuted
 | |
| 		: (instanceState == GroupCall::InstanceState::Disconnected)
 | |
| 		? BarState::Connecting
 | |
| 		: (state == MuteState::ForceMuted || state == MuteState::RaisedHand)
 | |
| 		? BarState::ForceMuted
 | |
| 		: (state == MuteState::Muted)
 | |
| 		? BarState::Muted
 | |
| 		: BarState::Active;
 | |
| };
 | |
| 
 | |
| auto LinearBlobs() {
 | |
| 	return std::vector<Ui::Paint::LinearBlobs::BlobData>{
 | |
| 		{
 | |
| 			.segmentsCount = 5,
 | |
| 			.minRadius = 0.,
 | |
| 			.maxRadius = (float)st::groupCallMajorBlobMaxRadius,
 | |
| 			.idleRadius = (float)st::groupCallMinorBlobIdleRadius,
 | |
| 			.speedScale = .3,
 | |
| 			.alpha = 1.,
 | |
| 		},
 | |
| 		{
 | |
| 			.segmentsCount = 7,
 | |
| 			.minRadius = 0.,
 | |
| 			.maxRadius = (float)st::groupCallMinorBlobMaxRadius,
 | |
| 			.idleRadius = (float)st::groupCallMinorBlobIdleRadius,
 | |
| 			.speedScale = .7,
 | |
| 			.alpha = kMinorBlobAlpha,
 | |
| 		},
 | |
| 		{
 | |
| 			.segmentsCount = 8,
 | |
| 			.minRadius = 0.,
 | |
| 			.maxRadius = (float)st::groupCallMinorBlobMaxRadius,
 | |
| 			.idleRadius = (float)st::groupCallMinorBlobIdleRadius,
 | |
| 			.speedScale = .7,
 | |
| 			.alpha = kMinorBlobAlpha,
 | |
| 		},
 | |
| 	};
 | |
| }
 | |
| 
 | |
| auto Colors() {
 | |
| 	using Vector = std::vector<QColor>;
 | |
| 	using Colors = anim::gradient_colors;
 | |
| 	return base::flat_map<BarState, Colors>{
 | |
| 		{
 | |
| 			BarState::ForceMuted,
 | |
| 			Colors(QGradientStops{
 | |
| 				{ 0.0, st::groupCallForceMutedBar1->c },
 | |
| 				{ .35, st::groupCallForceMutedBar2->c },
 | |
| 				{ 1.0, st::groupCallForceMutedBar3->c } })
 | |
| 		},
 | |
| 		{
 | |
| 			BarState::Active,
 | |
| 			Colors(Vector{ st::groupCallLive1->c, st::groupCallLive2->c })
 | |
| 		},
 | |
| 		{
 | |
| 			BarState::Muted,
 | |
| 			Colors(Vector{ st::groupCallMuted1->c, st::groupCallMuted2->c })
 | |
| 		},
 | |
| 		{
 | |
| 			BarState::Connecting,
 | |
| 			Colors(st::callBarBgMuted->c)
 | |
| 		},
 | |
| 	};
 | |
| }
 | |
| 
 | |
| class DebugInfoBox : public Ui::BoxContent {
 | |
| public:
 | |
| 	DebugInfoBox(QWidget*, base::weak_ptr<Call> call);
 | |
| 
 | |
| protected:
 | |
| 	void prepare() override;
 | |
| 
 | |
| private:
 | |
| 	void updateText();
 | |
| 
 | |
| 	base::weak_ptr<Call> _call;
 | |
| 	QPointer<Ui::FlatLabel> _text;
 | |
| 	base::Timer _updateTextTimer;
 | |
| 
 | |
| };
 | |
| 
 | |
| DebugInfoBox::DebugInfoBox(QWidget*, base::weak_ptr<Call> call)
 | |
| : _call(call) {
 | |
| }
 | |
| 
 | |
| void DebugInfoBox::prepare() {
 | |
| 	setTitle(rpl::single(u"Call Debug"_q));
 | |
| 
 | |
| 	addButton(tr::lng_close(), [this] { closeBox(); });
 | |
| 	_text = setInnerWidget(
 | |
| 		object_ptr<Ui::PaddingWrap<Ui::FlatLabel>>(
 | |
| 			this,
 | |
| 			object_ptr<Ui::FlatLabel>(this, st::callDebugLabel),
 | |
| 			st::callDebugPadding))->entity();
 | |
| 	_text->setSelectable(true);
 | |
| 	updateText();
 | |
| 	_updateTextTimer.setCallback([this] { updateText(); });
 | |
| 	_updateTextTimer.callEach(kUpdateDebugTimeoutMs);
 | |
| 	setDimensions(st::boxWideWidth, st::boxMaxListHeight);
 | |
| }
 | |
| 
 | |
| void DebugInfoBox::updateText() {
 | |
| 	if (auto call = _call.get()) {
 | |
| 		_text->setText(call->getDebugLog());
 | |
| 	}
 | |
| }
 | |
| 
 | |
| } // namespace
 | |
| 
 | |
| struct TopBar::User {
 | |
| 	Ui::GroupCallUser data;
 | |
| };
 | |
| 
 | |
| class Mute final : public Ui::IconButton {
 | |
| public:
 | |
| 	Mute(QWidget *parent, const style::IconButton &st)
 | |
| 	: Ui::IconButton(parent, st)
 | |
| 	, _st(st)
 | |
| 	, _crossLineMuteAnimation(st::callTopBarMuteCrossLine) {
 | |
| 		resize(_st.width, _st.height);
 | |
| 		installEventFilter(this);
 | |
| 
 | |
| 		style::PaletteChanged(
 | |
| 		) | rpl::start_with_next([=] {
 | |
| 			_crossLineMuteAnimation.invalidate();
 | |
| 		}, lifetime());
 | |
| 	}
 | |
| 
 | |
| 	void setProgress(float64 progress) {
 | |
| 		if (_progress == progress) {
 | |
| 			return;
 | |
| 		}
 | |
| 		_progress = progress;
 | |
| 		update();
 | |
| 	}
 | |
| 
 | |
| 	void setRippleColorOverride(const style::color *colorOverride) {
 | |
| 		_rippleColorOverride = colorOverride;
 | |
| 	}
 | |
| 
 | |
| protected:
 | |
| 	bool eventFilter(QObject *object, QEvent *event) {
 | |
| 		if (event->type() == QEvent::Paint) {
 | |
| 			auto p = QPainter(this);
 | |
| 			paintRipple(
 | |
| 				p,
 | |
| 				_st.rippleAreaPosition.x(),
 | |
| 				_st.rippleAreaPosition.y(),
 | |
| 				_rippleColorOverride ? &(*_rippleColorOverride)->c : nullptr);
 | |
| 			_crossLineMuteAnimation.paint(p, _st.iconPosition, _progress);
 | |
| 			return true;
 | |
| 		}
 | |
| 		return QObject::eventFilter(object, event);
 | |
| 	}
 | |
| 
 | |
| private:
 | |
| 	float64 _progress = 0.;
 | |
| 
 | |
| 	const style::IconButton &_st;
 | |
| 	Ui::CrossLineAnimation _crossLineMuteAnimation;
 | |
| 	const style::color *_rippleColorOverride = nullptr;
 | |
| 
 | |
| };
 | |
| 
 | |
| TopBar::TopBar(
 | |
| 	QWidget *parent,
 | |
| 	const base::weak_ptr<Call> &call,
 | |
| 	std::shared_ptr<Ui::Show> show)
 | |
| : TopBar(parent, show, call, nullptr) {
 | |
| }
 | |
| 
 | |
| TopBar::TopBar(
 | |
| 	QWidget *parent,
 | |
| 	const base::weak_ptr<GroupCall> &call,
 | |
| 	std::shared_ptr<Ui::Show> show)
 | |
| : TopBar(parent, show, nullptr, call) {
 | |
| }
 | |
| 
 | |
| TopBar::TopBar(
 | |
| 	QWidget *parent,
 | |
| 	std::shared_ptr<Ui::Show> show,
 | |
| 	const base::weak_ptr<Call> &call,
 | |
| 	const base::weak_ptr<GroupCall> &groupCall)
 | |
| : RpWidget(parent)
 | |
| , _call(call)
 | |
| , _groupCall(groupCall)
 | |
| , _show(show)
 | |
| , _userpics(call
 | |
| 	? nullptr
 | |
| 	: std::make_unique<Ui::GroupCallUserpics>(
 | |
| 		st::groupCallTopBarUserpics,
 | |
| 		rpl::single(true),
 | |
| 		[=] { updateUserpics(); }))
 | |
| , _durationLabel(_call
 | |
| 	? object_ptr<Ui::LabelSimple>(this, st::callBarLabel)
 | |
| 	: object_ptr<Ui::LabelSimple>(nullptr))
 | |
| , _signalBars(_call
 | |
| 	? object_ptr<SignalBars>(this, _call.get(), st::callBarSignalBars)
 | |
| 	: object_ptr<SignalBars>(nullptr))
 | |
| , _fullInfoLabel(this, st::callBarInfoLabel)
 | |
| , _shortInfoLabel(this, st::callBarInfoLabel)
 | |
| , _hangupLabel(_call
 | |
| 	? object_ptr<Ui::LabelSimple>(
 | |
| 		this,
 | |
| 		st::callBarLabel,
 | |
| 		tr::lng_call_bar_hangup(tr::now))
 | |
| 	: object_ptr<Ui::LabelSimple>(nullptr))
 | |
| , _mute(this, st::callBarMuteToggle)
 | |
| , _info(this)
 | |
| , _hangup(this, st::callBarHangup)
 | |
| , _gradients(Colors(), QPointF(), QPointF())
 | |
| , _updateDurationTimer([=] { updateDurationText(); }) {
 | |
| 	initControls();
 | |
| 	resize(width(), st::callBarHeight);
 | |
| 	setupInitialBrush();
 | |
| }
 | |
| 
 | |
| void TopBar::setupInitialBrush() {
 | |
| 	Expects(_switchStateCallback != nullptr);
 | |
| 
 | |
| 	_switchStateAnimation.stop();
 | |
| 	_switchStateCallback(1.);
 | |
| }
 | |
| 
 | |
| void TopBar::initControls() {
 | |
| 	_mute->setClickedCallback([=] {
 | |
| 		if (const auto call = _call.get()) {
 | |
| 			call->setMuted(!call->muted());
 | |
| 		} else if (const auto group = _groupCall.get()) {
 | |
| 			if (group->mutedByAdmin()) {
 | |
| 				_show->showToast(
 | |
| 					tr::lng_group_call_force_muted_sub(tr::now));
 | |
| 			} else {
 | |
| 				group->setMuted((group->muted() == MuteState::Muted)
 | |
| 					? MuteState::Active
 | |
| 					: MuteState::Muted);
 | |
| 			}
 | |
| 		}
 | |
| 	});
 | |
| 
 | |
| 	const auto mapToState = [](bool muted) {
 | |
| 		return muted ? MuteState::Muted : MuteState::Active;
 | |
| 	};
 | |
| 	const auto fromState = _mute->lifetime().make_state<BarState>(
 | |
| 		BarStateFromMuteState(
 | |
| 			_call
 | |
| 				? mapToState(_call->muted())
 | |
| 				: _groupCall->muted(),
 | |
| 			GroupCall::InstanceState::Connected,
 | |
| 			_call ? TimeId(0) : _groupCall->scheduleDate()));
 | |
| 	using namespace rpl::mappers;
 | |
| 	auto muted = _call
 | |
| 		? rpl::combine(
 | |
| 			_call->mutedValue() | rpl::map(mapToState),
 | |
| 			rpl::single(GroupCall::InstanceState::Connected),
 | |
| 			rpl::single(TimeId(0))
 | |
| 		) | rpl::type_erased()
 | |
| 		: rpl::combine(
 | |
| 			(_groupCall->mutedValue()
 | |
| 				| MapPushToTalkToActive()
 | |
| 				| rpl::distinct_until_changed()
 | |
| 				| rpl::type_erased()),
 | |
| 			rpl::single(
 | |
| 				_groupCall->instanceState()
 | |
| 			) | rpl::then(_groupCall->instanceStateValue() | rpl::filter(
 | |
| 				_1 != GroupCall::InstanceState::TransitionToRtc)),
 | |
| 			rpl::single(
 | |
| 				_groupCall->scheduleDate()
 | |
| 			) | rpl::then(_groupCall->real(
 | |
| 			) | rpl::map([](not_null<Data::GroupCall*> call) {
 | |
| 				return call->scheduleDateValue();
 | |
| 			}) | rpl::flatten_latest()));
 | |
| 	std::move(
 | |
| 		muted
 | |
| 	) | rpl::map(
 | |
| 		BarStateFromMuteState
 | |
| 	) | rpl::start_with_next([=](BarState state) {
 | |
| 		_isGroupConnecting = (state == BarState::Connecting);
 | |
| 		setMuted(state != BarState::Active);
 | |
| 		update();
 | |
| 
 | |
| 		const auto isForceMuted = (state == BarState::ForceMuted);
 | |
| 		if (isForceMuted) {
 | |
| 			_mute->clearState();
 | |
| 		}
 | |
| 		_mute->setPointerCursor(!isForceMuted);
 | |
| 
 | |
| 		const auto to = 1.;
 | |
| 		const auto from = _switchStateAnimation.animating()
 | |
| 			? (to - _switchStateAnimation.value(0.))
 | |
| 			: 0.;
 | |
| 		const auto fromMuted = *fromState;
 | |
| 		const auto toMuted = state;
 | |
| 		*fromState = state;
 | |
| 
 | |
| 		const auto crossFrom = (fromMuted != BarState::Active) ? 1. : 0.;
 | |
| 		const auto crossTo = (toMuted != BarState::Active) ? 1. : 0.;
 | |
| 
 | |
| 		_switchStateCallback = [=](float64 value) {
 | |
| 			if (_groupCall) {
 | |
| 				_groupBrush = QBrush(
 | |
| 					_gradients.gradient(fromMuted, toMuted, value));
 | |
| 				update();
 | |
| 			}
 | |
| 
 | |
| 			const auto crossProgress = (crossFrom == crossTo)
 | |
| 				? crossTo
 | |
| 				: anim::interpolateToF(crossFrom, crossTo, value);
 | |
| 			_mute->setProgress(crossProgress);
 | |
| 		};
 | |
| 
 | |
| 		_switchStateAnimation.stop();
 | |
| 		const auto duration = (to - from) * kSwitchStateDuration;
 | |
| 		_switchStateAnimation.start(
 | |
| 			_switchStateCallback,
 | |
| 			from,
 | |
| 			to,
 | |
| 			duration);
 | |
| 	}, _mute->lifetime());
 | |
| 
 | |
| 	if (const auto group = _groupCall.get()) {
 | |
| 		subscribeToMembersChanges(group);
 | |
| 
 | |
| 		_isGroupConnecting.value(
 | |
| 		) | rpl::start_with_next([=](bool isConnecting) {
 | |
| 			_mute->setAttribute(
 | |
| 				Qt::WA_TransparentForMouseEvents,
 | |
| 				isConnecting);
 | |
| 			updateInfoLabels();
 | |
| 		}, lifetime());
 | |
| 	}
 | |
| 
 | |
| 	if (const auto call = _call.get()) {
 | |
| 		call->user()->session().changes().peerUpdates(
 | |
| 			Data::PeerUpdate::Flag::Name
 | |
| 		) | rpl::filter([=](const Data::PeerUpdate &update) {
 | |
| 			// _user may change for the same Panel.
 | |
| 			return (_call != nullptr) && (update.peer == _call->user());
 | |
| 		}) | rpl::start_with_next([=] {
 | |
| 			updateInfoLabels();
 | |
| 		}, lifetime());
 | |
| 	}
 | |
| 
 | |
| 	setInfoLabels();
 | |
| 	_info->setClickedCallback([=] {
 | |
| 		if (const auto call = _call.get()) {
 | |
| 			if (Logs::DebugEnabled()
 | |
| 				&& (_info->clickModifiers() & Qt::ControlModifier)) {
 | |
| 				_show->showBox(
 | |
| 					Box<DebugInfoBox>(_call),
 | |
| 					Ui::LayerOption::CloseOther);
 | |
| 			} else {
 | |
| 				Core::App().calls().showInfoPanel(call);
 | |
| 			}
 | |
| 		} else if (const auto group = _groupCall.get()) {
 | |
| 			Core::App().calls().showInfoPanel(group);
 | |
| 		}
 | |
| 	});
 | |
| 	_hangup->setClickedCallback([this] {
 | |
| 		if (const auto call = _call.get()) {
 | |
| 			call->hangup();
 | |
| 		} else if (const auto group = _groupCall.get()) {
 | |
| 			if (!group->peer()->canManageGroupCall()) {
 | |
| 				group->hangup();
 | |
| 			} else {
 | |
| 				_show->showBox(
 | |
| 					Box(
 | |
| 						Group::LeaveBox,
 | |
| 						group,
 | |
| 						false,
 | |
| 						Group::BoxContext::MainWindow),
 | |
| 					Ui::LayerOption::CloseOther);
 | |
| 			}
 | |
| 		}
 | |
| 	});
 | |
| 	updateDurationText();
 | |
| }
 | |
| 
 | |
| void TopBar::initBlobsUnder(
 | |
| 		QWidget *blobsParent,
 | |
| 		rpl::producer<QRect> barGeometry) {
 | |
| 	const auto group = _groupCall.get();
 | |
| 	if (!group) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	struct State {
 | |
| 		Ui::Paint::LinearBlobs paint = {
 | |
| 			LinearBlobs(),
 | |
| 			kBlobLevelDuration,
 | |
| 			1.,
 | |
| 			Ui::Paint::LinearBlob::Direction::TopDown
 | |
| 		};
 | |
| 		Ui::Animations::Simple hideAnimation;
 | |
| 		Ui::Animations::Basic animation;
 | |
| 		base::Timer levelTimer;
 | |
| 		crl::time hideLastTime = 0;
 | |
| 		crl::time lastTime = 0;
 | |
| 		float lastLevel = 0.;
 | |
| 		float levelBeforeLast = 0.;
 | |
| 	};
 | |
| 
 | |
| 	_blobs = base::make_unique_q<Ui::RpWidget>(blobsParent);
 | |
| 
 | |
| 	const auto state = _blobs->lifetime().make_state<State>();
 | |
| 	state->levelTimer.setCallback([=] {
 | |
| 		state->levelBeforeLast = state->lastLevel;
 | |
| 		state->lastLevel = 0.;
 | |
| 		if (state->levelBeforeLast == 0.) {
 | |
| 			state->paint.setLevel(0.);
 | |
| 			state->levelTimer.cancel();
 | |
| 		}
 | |
| 	});
 | |
| 
 | |
| 	state->animation.init([=](crl::time now) {
 | |
| 		if (const auto last = state->hideLastTime; (last > 0)
 | |
| 			&& (now - last >= kHideBlobsDuration)) {
 | |
| 			state->animation.stop();
 | |
| 			return false;
 | |
| 		}
 | |
| 		state->paint.updateLevel(now - state->lastTime);
 | |
| 		state->lastTime = now;
 | |
| 
 | |
| 		_blobs->update();
 | |
| 		return true;
 | |
| 	});
 | |
| 
 | |
| 	group->stateValue(
 | |
| 	) | rpl::start_with_next([=](Calls::GroupCall::State state) {
 | |
| 		if (state == Calls::GroupCall::State::HangingUp) {
 | |
| 			_blobs->hide();
 | |
| 		}
 | |
| 	}, lifetime());
 | |
| 
 | |
| 	using namespace rpl::mappers;
 | |
| 	auto hideBlobs = rpl::combine(
 | |
| 		PowerSaving::OnValue(PowerSaving::kCalls),
 | |
| 		Core::App().appDeactivatedValue(),
 | |
| 		group->instanceStateValue()
 | |
| 	) | rpl::map(_1 || _2 || _3 == GroupCall::InstanceState::Disconnected);
 | |
| 
 | |
| 	std::move(
 | |
| 		hideBlobs
 | |
| 	) | rpl::distinct_until_changed(
 | |
| 	) | rpl::start_with_next([=](bool hide) {
 | |
| 		if (hide) {
 | |
| 			state->paint.setLevel(0.);
 | |
| 		}
 | |
| 		state->hideLastTime = hide ? crl::now() : 0;
 | |
| 		if (!hide && !state->animation.animating()) {
 | |
| 			state->animation.start();
 | |
| 		}
 | |
| 		if (hide) {
 | |
| 			state->levelTimer.cancel();
 | |
| 		} else {
 | |
| 			state->lastLevel = 0.;
 | |
| 		}
 | |
| 
 | |
| 		const auto from = hide ? 0. : 1.;
 | |
| 		const auto to = hide ? 1. : 0.;
 | |
| 		state->hideAnimation.start([=](float64) {
 | |
| 			_blobs->update();
 | |
| 		}, from, to, kHideBlobsDuration);
 | |
| 	}, lifetime());
 | |
| 
 | |
| 	std::move(
 | |
| 		barGeometry
 | |
| 	) | rpl::start_with_next([=](QRect rect) {
 | |
| 		_blobs->resize(
 | |
| 			rect.width(),
 | |
| 			(int)state->paint.maxRadius());
 | |
| 		_blobs->moveToLeft(rect.x(), rect.y() + rect.height());
 | |
| 	}, lifetime());
 | |
| 
 | |
| 	shownValue(
 | |
| 	) | rpl::start_with_next([=](bool shown) {
 | |
| 		_blobs->setVisible(shown);
 | |
| 	}, lifetime());
 | |
| 
 | |
| 	_blobs->paintRequest(
 | |
| 	) | rpl::start_with_next([=](QRect clip) {
 | |
| 		const auto hidden = state->hideAnimation.value(
 | |
| 			state->hideLastTime ? 1. : 0.);
 | |
| 		if (hidden == 1.) {
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		auto p = QPainter(_blobs);
 | |
| 		if (hidden > 0.) {
 | |
| 			p.setOpacity(1. - hidden);
 | |
| 		}
 | |
| 		const auto top = -_blobs->height() * hidden;
 | |
| 		const auto width = _blobs->width();
 | |
| 		p.translate(0, top);
 | |
| 		state->paint.paint(p, _groupBrush, width);
 | |
| 	}, _blobs->lifetime());
 | |
| 
 | |
| 	group->levelUpdates(
 | |
| 	) | rpl::filter([=](const LevelUpdate &update) {
 | |
| 		return !state->hideLastTime && (update.value > state->lastLevel);
 | |
| 	}) | rpl::start_with_next([=](const LevelUpdate &update) {
 | |
| 		if (state->lastLevel == 0.) {
 | |
| 			state->levelTimer.callEach(kBlobUpdateInterval);
 | |
| 		}
 | |
| 		state->lastLevel = update.value;
 | |
| 		state->paint.setLevel(update.value);
 | |
| 	}, _blobs->lifetime());
 | |
| 
 | |
| 	_blobs->setAttribute(Qt::WA_TransparentForMouseEvents);
 | |
| 	_blobs->show();
 | |
| 
 | |
| 	if (!state->hideLastTime) {
 | |
| 		state->animation.start();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void TopBar::subscribeToMembersChanges(not_null<GroupCall*> call) {
 | |
| 	const auto peer = call->peer();
 | |
| 	peer->session().changes().peerFlagsValue(
 | |
| 		peer,
 | |
| 		Data::PeerUpdate::Flag::GroupCall
 | |
| 	) | rpl::map([=] {
 | |
| 		return peer->groupCall();
 | |
| 	}) | rpl::filter([=](Data::GroupCall *real) {
 | |
| 		const auto call = _groupCall.get();
 | |
| 		return call && real && (real->id() == call->id());
 | |
| 	}) | rpl::take(
 | |
| 		1
 | |
| 	) | rpl::before_next([=](not_null<Data::GroupCall*> real) {
 | |
| 		real->titleValue() | rpl::start_with_next([=] {
 | |
| 			updateInfoLabels();
 | |
| 		}, lifetime());
 | |
| 	}) | rpl::map([=](not_null<Data::GroupCall*> real) {
 | |
| 
 | |
| 		return HistoryView::GroupCallBarContentByCall(
 | |
| 			real,
 | |
| 			st::groupCallTopBarUserpics.size);
 | |
| 	}) | rpl::flatten_latest(
 | |
| 	) | rpl::filter([=](const Ui::GroupCallBarContent &content) {
 | |
| 		if (_users.size() != content.users.size()) {
 | |
| 			return true;
 | |
| 		}
 | |
| 		for (auto i = 0, count = int(_users.size()); i != count; ++i) {
 | |
| 			if (_users[i].userpicKey != content.users[i].userpicKey
 | |
| 				|| _users[i].id != content.users[i].id) {
 | |
| 				return true;
 | |
| 			}
 | |
| 		}
 | |
| 		return false;
 | |
| 	}) | rpl::start_with_next([=](const Ui::GroupCallBarContent &content) {
 | |
| 		_users = content.users;
 | |
| 		for (auto &user : _users) {
 | |
| 			user.speaking = false;
 | |
| 		}
 | |
| 		_userpics->update(_users, !isHidden());
 | |
| 	}, lifetime());
 | |
| 
 | |
| 	_userpics->widthValue(
 | |
| 	) | rpl::start_with_next([=](int width) {
 | |
| 		_userpicsWidth = width;
 | |
| 		updateControlsGeometry();
 | |
| 	}, lifetime());
 | |
| 
 | |
| 	call->peer()->session().changes().peerUpdates(
 | |
| 		Data::PeerUpdate::Flag::Name
 | |
| 	) | rpl::filter([=](const Data::PeerUpdate &update) {
 | |
| 		// _peer may change for the same Panel.
 | |
| 		const auto call = _groupCall.get();
 | |
| 		return (call != nullptr) && (update.peer == call->peer());
 | |
| 	}) | rpl::start_with_next([=] {
 | |
| 		updateInfoLabels();
 | |
| 	}, lifetime());
 | |
| }
 | |
| 
 | |
| void TopBar::updateUserpics() {
 | |
| 	update(_mute->width(), 0, _userpics->maxWidth(), height());
 | |
| }
 | |
| 
 | |
| void TopBar::updateInfoLabels() {
 | |
| 	setInfoLabels();
 | |
| 	updateControlsGeometry();
 | |
| }
 | |
| 
 | |
| void TopBar::setInfoLabels() {
 | |
| 	if (const auto call = _call.get()) {
 | |
| 		const auto user = call->user();
 | |
| 		const auto fullName = user->name();
 | |
| 		const auto shortName = user->firstName;
 | |
| 		_fullInfoLabel->setText(fullName);
 | |
| 		_shortInfoLabel->setText(shortName);
 | |
| 	} else if (const auto group = _groupCall.get()) {
 | |
| 		const auto peer = group->peer();
 | |
| 		const auto real = peer->groupCall();
 | |
| 		const auto name = peer->name();
 | |
| 		const auto text = _isGroupConnecting.current()
 | |
| 			? tr::lng_group_call_connecting(tr::now)
 | |
| 			: (real && real->id() == group->id() && !real->title().isEmpty())
 | |
| 			? real->title()
 | |
| 			: name;
 | |
| 		_fullInfoLabel->setText(text);
 | |
| 		_shortInfoLabel->setText(text);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void TopBar::setMuted(bool mute) {
 | |
| 	_mute->setRippleColorOverride(&st::shadowFg);
 | |
| 	_hangup->setRippleColorOverride(&st::shadowFg);
 | |
| 	_muted = mute;
 | |
| }
 | |
| 
 | |
| void TopBar::updateDurationText() {
 | |
| 	if (!_call || !_durationLabel) {
 | |
| 		return;
 | |
| 	}
 | |
| 	auto wasWidth = _durationLabel->width();
 | |
| 	auto durationMs = _call->getDurationMs();
 | |
| 	auto durationSeconds = durationMs / 1000;
 | |
| 	startDurationUpdateTimer(durationMs);
 | |
| 	_durationLabel->setText(Ui::FormatDurationText(durationSeconds));
 | |
| 	if (_durationLabel->width() != wasWidth) {
 | |
| 		updateControlsGeometry();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void TopBar::startDurationUpdateTimer(crl::time currentDuration) {
 | |
| 	auto msTillNextSecond = 1000 - (currentDuration % 1000);
 | |
| 	_updateDurationTimer.callOnce(msTillNextSecond + 5);
 | |
| }
 | |
| 
 | |
| void TopBar::resizeEvent(QResizeEvent *e) {
 | |
| 	updateControlsGeometry();
 | |
| }
 | |
| 
 | |
| void TopBar::updateControlsGeometry() {
 | |
| 	auto left = 0;
 | |
| 	_mute->moveToLeft(left, 0);
 | |
| 	left += _mute->width();
 | |
| 	if (_durationLabel) {
 | |
| 		_durationLabel->moveToLeft(left, st::callBarLabelTop);
 | |
| 		left += _durationLabel->width() + st::callBarSkip;
 | |
| 	}
 | |
| 	if (_userpicsWidth) {
 | |
| 		const auto single = st::groupCallTopBarUserpics.size;
 | |
| 		const auto skip = anim::interpolate(
 | |
| 			0,
 | |
| 			st::callBarSkip,
 | |
| 			std::min(_userpicsWidth, single) / float64(single));
 | |
| 		left += _userpicsWidth + skip;
 | |
| 	}
 | |
| 	if (_signalBars) {
 | |
| 		_signalBars->moveToLeft(left, (height() - _signalBars->height()) / 2);
 | |
| 		left += _signalBars->width() + st::callBarSkip;
 | |
| 	}
 | |
| 
 | |
| 	auto right = st::callBarRightSkip;
 | |
| 	if (_hangupLabel) {
 | |
| 		_hangupLabel->moveToRight(right, st::callBarLabelTop);
 | |
| 		right += _hangupLabel->width();
 | |
| 	} else {
 | |
| 		//right -= st::callBarRightSkip;
 | |
| 	}
 | |
| 	right += st::callBarHangup.width;
 | |
| 	_hangup->setGeometryToRight(0, 0, right, height());
 | |
| 	_info->setGeometryToLeft(
 | |
| 		_mute->width(),
 | |
| 		0,
 | |
| 		width() - _mute->width() - _hangup->width(),
 | |
| 		height());
 | |
| 
 | |
| 	auto fullWidth = _fullInfoLabel->textMaxWidth();
 | |
| 	auto showFull = (left + fullWidth + right <= width());
 | |
| 	_fullInfoLabel->setVisible(showFull);
 | |
| 	_shortInfoLabel->setVisible(!showFull);
 | |
| 
 | |
| 	auto setInfoLabelGeometry = [this, left, right](auto &&infoLabel) {
 | |
| 		auto minPadding = qMax(left, right);
 | |
| 		auto infoWidth = infoLabel->textMaxWidth();
 | |
| 		auto infoLeft = (width() - infoWidth) / 2;
 | |
| 		if (infoLeft < minPadding) {
 | |
| 			infoLeft = left;
 | |
| 			infoWidth = width() - left - right;
 | |
| 		}
 | |
| 		infoLabel->setGeometryToLeft(infoLeft, st::callBarLabelTop, infoWidth, st::callBarInfoLabel.style.font->height);
 | |
| 	};
 | |
| 	setInfoLabelGeometry(_fullInfoLabel);
 | |
| 	setInfoLabelGeometry(_shortInfoLabel);
 | |
| 
 | |
| 	_gradients.set_points(
 | |
| 		QPointF(0, st::callBarHeight / 2),
 | |
| 		QPointF(width(), st::callBarHeight / 2));
 | |
| 	if (!_switchStateAnimation.animating()) {
 | |
| 		_switchStateCallback(1.);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void TopBar::paintEvent(QPaintEvent *e) {
 | |
| 	auto p = QPainter(this);
 | |
| 	auto brush = _groupCall
 | |
| 		? _groupBrush
 | |
| 		: (_muted ? st::callBarBgMuted : st::callBarBg);
 | |
| 	p.fillRect(e->rect(), std::move(brush));
 | |
| 
 | |
| 	if (_userpicsWidth) {
 | |
| 		const auto size = st::groupCallTopBarUserpics.size;
 | |
| 		const auto top = (height() - size) / 2;
 | |
| 		_userpics->paint(p, _mute->width(), top, size);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| TopBar::~TopBar() = default;
 | |
| 
 | |
| } // namespace Calls
 | 
