242 lines
		
	
	
	
		
			6.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			242 lines
		
	
	
	
		
			6.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						|
This file is part of Telegram Desktop,
 | 
						|
the official desktop application for the Telegram messaging service.
 | 
						|
 | 
						|
For license and copyright information please follow this link:
 | 
						|
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
 | 
						|
*/
 | 
						|
#include "ui/chat/requests_bar.h"
 | 
						|
 | 
						|
#include "ui/chat/group_call_userpics.h"
 | 
						|
#include "ui/widgets/shadow.h"
 | 
						|
#include "ui/text/text_options.h"
 | 
						|
#include "lang/lang_keys.h"
 | 
						|
#include "styles/style_chat.h"
 | 
						|
#include "styles/style_calls.h"
 | 
						|
#include "styles/style_info.h" // st::topBarArrowPadding, like TopBarWidget.
 | 
						|
#include "styles/palette.h"
 | 
						|
 | 
						|
#include <QtGui/QtEvents>
 | 
						|
 | 
						|
namespace Ui {
 | 
						|
 | 
						|
RequestsBar::RequestsBar(
 | 
						|
	not_null<QWidget*> parent,
 | 
						|
	rpl::producer<RequestsBarContent> content)
 | 
						|
: _wrap(parent, object_ptr<RpWidget>(parent))
 | 
						|
, _inner(_wrap.entity())
 | 
						|
, _shadow(std::make_unique<PlainShadow>(_wrap.parentWidget()))
 | 
						|
, _userpics(std::make_unique<GroupCallUserpics>(
 | 
						|
		st::historyRequestsUserpics,
 | 
						|
		rpl::single(false),
 | 
						|
		[=] { _inner->update(); })) {
 | 
						|
	_wrap.hide(anim::type::instant);
 | 
						|
	_shadow->hide();
 | 
						|
 | 
						|
	_wrap.entity()->paintRequest(
 | 
						|
	) | rpl::start_with_next([=](QRect clip) {
 | 
						|
		QPainter(_wrap.entity()).fillRect(clip, st::historyPinnedBg);
 | 
						|
	}, lifetime());
 | 
						|
	_wrap.setAttribute(Qt::WA_OpaquePaintEvent);
 | 
						|
 | 
						|
	auto copy = std::move(
 | 
						|
		content
 | 
						|
	) | rpl::start_spawning(_wrap.lifetime());
 | 
						|
 | 
						|
	rpl::duplicate(
 | 
						|
		copy
 | 
						|
	) | rpl::start_with_next([=](RequestsBarContent &&content) {
 | 
						|
		_content = content;
 | 
						|
		if (_content.count > 0) {
 | 
						|
			if (_content.count == 1 && !_content.nameFull.isEmpty()) {
 | 
						|
				_textFull.setText(
 | 
						|
					st::defaultMessageBar.title,
 | 
						|
					tr::lng_group_requests_pending_user(
 | 
						|
						tr::now,
 | 
						|
						lt_user,
 | 
						|
						_content.nameFull),
 | 
						|
					Ui::NameTextOptions());
 | 
						|
				_textShort.setText(
 | 
						|
					st::defaultMessageBar.title,
 | 
						|
					tr::lng_group_requests_pending_user(
 | 
						|
						tr::now,
 | 
						|
						lt_user,
 | 
						|
						_content.nameShort),
 | 
						|
					Ui::NameTextOptions());
 | 
						|
			} else {
 | 
						|
				_textShort.setText(
 | 
						|
					st::defaultMessageBar.title,
 | 
						|
					tr::lng_group_requests_pending(
 | 
						|
						tr::now,
 | 
						|
						lt_count_decimal,
 | 
						|
						_content.count),
 | 
						|
					Ui::NameTextOptions());
 | 
						|
				_textFull.clear();
 | 
						|
			}
 | 
						|
		}
 | 
						|
		_userpics->update(_content.users, !_wrap.isHidden());
 | 
						|
		_inner->update();
 | 
						|
	}, lifetime());
 | 
						|
 | 
						|
	std::move(
 | 
						|
		copy
 | 
						|
	) | rpl::map([=](const RequestsBarContent &content) {
 | 
						|
		return !content.count;
 | 
						|
	}) | rpl::start_with_next_done([=](bool hidden) {
 | 
						|
		_shouldBeShown = !hidden;
 | 
						|
		if (!_forceHidden) {
 | 
						|
			_wrap.toggle(_shouldBeShown, anim::type::normal);
 | 
						|
		}
 | 
						|
	}, [=] {
 | 
						|
		_forceHidden = true;
 | 
						|
		_wrap.toggle(false, anim::type::normal);
 | 
						|
	}, lifetime());
 | 
						|
 | 
						|
	_userpics->widthValue(
 | 
						|
	) | rpl::start_with_next([=](int width) {
 | 
						|
		_userpicsWidth = width;
 | 
						|
	}, lifetime());
 | 
						|
 | 
						|
	setupInner();
 | 
						|
}
 | 
						|
 | 
						|
RequestsBar::~RequestsBar() = default;
 | 
						|
 | 
						|
void RequestsBar::setupInner() {
 | 
						|
	_inner->resize(0, st::historyRequestsHeight);
 | 
						|
	_inner->paintRequest(
 | 
						|
	) | rpl::start_with_next([=](QRect rect) {
 | 
						|
		auto p = Painter(_inner);
 | 
						|
		paint(p);
 | 
						|
	}, _inner->lifetime());
 | 
						|
 | 
						|
	// Clicks.
 | 
						|
	_inner->setCursor(style::cur_pointer);
 | 
						|
	_inner->events(
 | 
						|
	) | rpl::filter([=](not_null<QEvent*> event) {
 | 
						|
		return (event->type() == QEvent::MouseButtonPress);
 | 
						|
	}) | rpl::map([=] {
 | 
						|
		return _inner->events(
 | 
						|
		) | rpl::filter([=](not_null<QEvent*> event) {
 | 
						|
			return (event->type() == QEvent::MouseButtonRelease);
 | 
						|
		}) | rpl::take(1) | rpl::filter([=](not_null<QEvent*> event) {
 | 
						|
			return _inner->rect().contains(
 | 
						|
				static_cast<QMouseEvent*>(event.get())->pos());
 | 
						|
		});
 | 
						|
	}) | rpl::flatten_latest(
 | 
						|
	) | rpl::to_empty | rpl::start_to_stream(_barClicks, _inner->lifetime());
 | 
						|
 | 
						|
	_wrap.geometryValue(
 | 
						|
	) | rpl::start_with_next([=](QRect rect) {
 | 
						|
		updateShadowGeometry(rect);
 | 
						|
		updateControlsGeometry(rect);
 | 
						|
	}, _inner->lifetime());
 | 
						|
}
 | 
						|
 | 
						|
void RequestsBar::paint(Painter &p) {
 | 
						|
	p.fillRect(_inner->rect(), st::historyComposeAreaBg);
 | 
						|
 | 
						|
	const auto userpicsSize = st::historyRequestsUserpics.size;
 | 
						|
	const auto userpicsTop = st::lineWidth + (st::historyRequestsHeight
 | 
						|
		- st::lineWidth
 | 
						|
		- userpicsSize) / 2;
 | 
						|
	const auto userpicsLeft = userpicsTop * 2;
 | 
						|
	const auto textTop = st::lineWidth + (st::historyRequestsHeight
 | 
						|
		- st::lineWidth
 | 
						|
		- st::msgServiceNameFont->height) / 2;
 | 
						|
	const auto width = _inner->width();
 | 
						|
	const auto &font = st::defaultMessageBar.title.font;
 | 
						|
	p.setPen(st::defaultMessageBar.titleFg);
 | 
						|
	p.setFont(font);
 | 
						|
 | 
						|
	const auto textLeft = userpicsLeft + _userpicsWidth + userpicsLeft;
 | 
						|
	const auto available = width - textLeft - userpicsLeft;
 | 
						|
	if (_textFull.isEmpty() || available < _textFull.maxWidth()) {
 | 
						|
		_textShort.drawElided(p, textLeft, textTop, available);
 | 
						|
	} else {
 | 
						|
		_textFull.drawElided(p, textLeft, textTop, available);
 | 
						|
	}
 | 
						|
 | 
						|
	// Skip shadow of the bar above.
 | 
						|
	_userpics->paint(p, userpicsLeft, userpicsTop, userpicsSize);
 | 
						|
}
 | 
						|
 | 
						|
void RequestsBar::updateControlsGeometry(QRect wrapGeometry) {
 | 
						|
	const auto hidden = _wrap.isHidden() || !wrapGeometry.height();
 | 
						|
	if (_shadow->isHidden() != hidden) {
 | 
						|
		_shadow->setVisible(!hidden);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void RequestsBar::setShadowGeometryPostprocess(Fn<QRect(QRect)> postprocess) {
 | 
						|
	_shadowGeometryPostprocess = std::move(postprocess);
 | 
						|
	updateShadowGeometry(_wrap.geometry());
 | 
						|
}
 | 
						|
 | 
						|
void RequestsBar::updateShadowGeometry(QRect wrapGeometry) {
 | 
						|
	const auto regular = QRect(
 | 
						|
		wrapGeometry.x(),
 | 
						|
		wrapGeometry.y() + wrapGeometry.height(),
 | 
						|
		wrapGeometry.width(),
 | 
						|
		st::lineWidth);
 | 
						|
	_shadow->setGeometry(_shadowGeometryPostprocess
 | 
						|
		? _shadowGeometryPostprocess(regular)
 | 
						|
		: regular);
 | 
						|
}
 | 
						|
 | 
						|
void RequestsBar::show() {
 | 
						|
	if (!_forceHidden) {
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	_forceHidden = false;
 | 
						|
	if (_shouldBeShown) {
 | 
						|
		_wrap.show(anim::type::instant);
 | 
						|
		_shadow->show();
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void RequestsBar::hide() {
 | 
						|
	if (_forceHidden) {
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	_forceHidden = true;
 | 
						|
	_wrap.hide(anim::type::instant);
 | 
						|
	_shadow->hide();
 | 
						|
}
 | 
						|
 | 
						|
void RequestsBar::raise() {
 | 
						|
	_wrap.raise();
 | 
						|
	_shadow->raise();
 | 
						|
}
 | 
						|
 | 
						|
void RequestsBar::finishAnimating() {
 | 
						|
	_wrap.finishAnimating();
 | 
						|
}
 | 
						|
 | 
						|
void RequestsBar::move(int x, int y) {
 | 
						|
	_wrap.move(x, y);
 | 
						|
}
 | 
						|
 | 
						|
void RequestsBar::resizeToWidth(int width) {
 | 
						|
	_wrap.entity()->resizeToWidth(width);
 | 
						|
	_inner->resizeToWidth(width);
 | 
						|
}
 | 
						|
 | 
						|
int RequestsBar::height() const {
 | 
						|
	return !_forceHidden
 | 
						|
		? _wrap.height()
 | 
						|
		: _shouldBeShown
 | 
						|
		? st::historyRequestsHeight
 | 
						|
		: 0;
 | 
						|
}
 | 
						|
 | 
						|
rpl::producer<int> RequestsBar::heightValue() const {
 | 
						|
	return _wrap.heightValue();
 | 
						|
}
 | 
						|
 | 
						|
rpl::producer<> RequestsBar::barClicks() const {
 | 
						|
	return _barClicks.events();
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
} // namespace Ui
 |