mirror of https://github.com/procxx/kepka.git
				
				
				
			
		
			
				
	
	
		
			574 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			574 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			C++
		
	
	
	
| //
 | |
| // This file is part of Kepka,
 | |
| // an unofficial desktop version of Telegram messaging app,
 | |
| // see https://github.com/procxx/kepka
 | |
| //
 | |
| // Kepka is free software: you can redistribute it and/or modify
 | |
| // it under the terms of the GNU General Public License as published by
 | |
| // the Free Software Foundation, either version 3 of the License, or
 | |
| // (at your option) any later version.
 | |
| //
 | |
| // It is distributed in the hope that it will be useful,
 | |
| // but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
| // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | |
| // GNU General Public License for more details.
 | |
| //
 | |
| // In addition, as a special exception, the copyright holders give permission
 | |
| // to link the code of portions of this program with the OpenSSL library.
 | |
| //
 | |
| // Full license: https://github.com/procxx/kepka/blob/master/LICENSE
 | |
| // Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
 | |
| // Copyright (c) 2017- Kepka Contributors, https://github.com/procxx
 | |
| //
 | |
| #include "profile/profile_cover.h"
 | |
| 
 | |
| #include "apiwrap.h"
 | |
| #include "auth_session.h"
 | |
| #include "boxes/confirm_box.h"
 | |
| #include "boxes/peer_list_controllers.h"
 | |
| #include "boxes/photo_crop_box.h"
 | |
| #include "core/file_utilities.h"
 | |
| #include "lang/lang_keys.h"
 | |
| #include "mainwidget.h"
 | |
| #include "mainwindow.h"
 | |
| #include "messenger.h"
 | |
| #include "observer_peer.h"
 | |
| #include "platform/platform_file_utilities.h"
 | |
| #include "profile/profile_cover_drop_area.h"
 | |
| #include "profile/profile_userpic_button.h"
 | |
| #include "styles/style_profile.h"
 | |
| #include "styles/style_widgets.h"
 | |
| #include "styles/style_window.h"
 | |
| #include "ui/widgets/buttons.h"
 | |
| #include "ui/widgets/labels.h"
 | |
| 
 | |
| #include <QWidget>
 | |
| 
 | |
| namespace Profile {
 | |
| namespace {
 | |
| 
 | |
| using UpdateFlag = Notify::PeerUpdate::Flag;
 | |
| const auto ButtonsUpdateFlags = UpdateFlag::UserCanShareContact | UpdateFlag::BotCanAddToGroups |
 | |
|                                 UpdateFlag::ChatCanEdit | UpdateFlag::ChannelRightsChanged | UpdateFlag::ChannelAmIn;
 | |
| 
 | |
| } // namespace
 | |
| 
 | |
| CoverWidget::CoverWidget(QWidget *parent, PeerData *peer)
 | |
|     : TWidget(parent)
 | |
|     , _peer(peer)
 | |
|     , _peerUser(peer->asUser())
 | |
|     , _peerChat(peer->asChat())
 | |
|     , _peerChannel(peer->asChannel())
 | |
|     , _peerMegagroup(peer->isMegagroup() ? _peerChannel : nullptr)
 | |
|     , _userpicButton(this, peer)
 | |
|     , _name(this, st::profileNameLabel) {
 | |
| 	_peer->updateFull();
 | |
| 
 | |
| 	subscribe(Lang::Current().updated(), [this] { refreshLang(); });
 | |
| 
 | |
| 	setAttribute(Qt::WA_OpaquePaintEvent);
 | |
| 	setAcceptDrops(true);
 | |
| 
 | |
| 	_name->setSelectable(true);
 | |
| 	_name->setContextCopyText(lang(lng_profile_copy_fullname));
 | |
| 
 | |
| 	auto observeEvents = ButtonsUpdateFlags | UpdateFlag::NameChanged | UpdateFlag::UserOnlineChanged |
 | |
| 	                     UpdateFlag::MembersChanged | UpdateFlag::PhotoChanged;
 | |
| 	subscribe(Notify::PeerUpdated(),
 | |
| 	          Notify::PeerUpdatedHandler(observeEvents,
 | |
| 	                                     [this](const Notify::PeerUpdate &update) { notifyPeerUpdated(update); }));
 | |
| 
 | |
| 	connect(&Messenger::Instance(), SIGNAL(peerPhotoDone(PeerId)), this, SLOT(onPhotoUploadStatusChanged(PeerId)));
 | |
| 	connect(&Messenger::Instance(), SIGNAL(peerPhotoFail(PeerId)), this, SLOT(onPhotoUploadStatusChanged(PeerId)));
 | |
| 
 | |
| 	connect(_userpicButton, SIGNAL(clicked()), this, SLOT(onPhotoShow()));
 | |
| 	validatePhoto();
 | |
| 
 | |
| 	refreshNameText();
 | |
| 	refreshStatusText();
 | |
| 
 | |
| 	refreshButtons();
 | |
| }
 | |
| 
 | |
| void CoverWidget::refreshLang() {
 | |
| 	InvokeQueued(this, [this] { moveAndToggleButtons(width()); });
 | |
| }
 | |
| 
 | |
| PhotoData *CoverWidget::validatePhoto() const {
 | |
| 	auto photo = (_peer->photoId && _peer->photoId != UnknownPeerPhotoId) ? App::photo(_peer->photoId) : nullptr;
 | |
| 	_userpicButton->setPointerCursor(photo != nullptr && photo->date != 0);
 | |
| 	if ((_peer->photoId == UnknownPeerPhotoId) || (_peer->photoId && (!photo || !photo->date))) {
 | |
| 		Auth().api().requestFullPeer(_peer);
 | |
| 		return nullptr;
 | |
| 	}
 | |
| 	return photo;
 | |
| }
 | |
| 
 | |
| void CoverWidget::onPhotoShow() {
 | |
| 	if (auto photo = validatePhoto()) {
 | |
| 		Messenger::Instance().showPhoto(photo, _peer);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void CoverWidget::onCancelPhotoUpload() {
 | |
| 	Messenger::Instance().cancelPhotoUpdate(_peer->id);
 | |
| 	refreshStatusText();
 | |
| }
 | |
| 
 | |
| int CoverWidget::countPhotoLeft(int newWidth) const {
 | |
| 	int result = st::profilePhotoLeftMin;
 | |
| 	result += (newWidth - st::windowMinWidth) / 2;
 | |
| 	return std::min(result, st::profilePhotoLeftMax);
 | |
| }
 | |
| 
 | |
| int CoverWidget::resizeGetHeight(int newWidth) {
 | |
| 	int newHeight = 0;
 | |
| 
 | |
| 	newHeight += st::profileMarginTop;
 | |
| 
 | |
| 	_photoLeft = countPhotoLeft(newWidth);
 | |
| 	_userpicButton->moveToLeft(_photoLeft, newHeight);
 | |
| 
 | |
| 	refreshNameGeometry(newWidth);
 | |
| 
 | |
| 	int infoLeft = _userpicButton->x() + _userpicButton->width();
 | |
| 	_statusPosition = QPoint(infoLeft + st::profileStatusLeft, _userpicButton->y() + st::profileStatusTop);
 | |
| 	if (_cancelPhotoUpload) {
 | |
| 		_cancelPhotoUpload->moveToLeft(_statusPosition.x() + st::profileStatusFont->width(_statusText) +
 | |
| 		                                   st::profileStatusFont->spacew,
 | |
| 		                               _statusPosition.y());
 | |
| 	}
 | |
| 
 | |
| 	moveAndToggleButtons(newWidth);
 | |
| 
 | |
| 	newHeight += st::profilePhotoSize;
 | |
| 	newHeight += st::profileMarginBottom;
 | |
| 
 | |
| 	_dividerTop = newHeight;
 | |
| 	newHeight += st::profileDividerLeft.height();
 | |
| 
 | |
| 	newHeight += st::profileBlocksTop;
 | |
| 
 | |
| 	resizeDropArea(newWidth);
 | |
| 	return newHeight;
 | |
| }
 | |
| 
 | |
| void CoverWidget::refreshNameGeometry(int newWidth) {
 | |
| 	int infoLeft = _userpicButton->x() + _userpicButton->width();
 | |
| 	int nameLeft = infoLeft + st::profileNameLeft - st::profileNameLabel.margin.left();
 | |
| 	int nameTop = _userpicButton->y() + st::profileNameTop - st::profileNameLabel.margin.top();
 | |
| 	int nameWidth = newWidth - infoLeft - st::profileNameLeft;
 | |
| 	if (_peer->isVerified()) {
 | |
| 		nameWidth -= st::profileVerifiedCheckShift + st::profileVerifiedCheck.width();
 | |
| 	}
 | |
| 	int marginsAdd = st::profileNameLabel.margin.left() + st::profileNameLabel.margin.right();
 | |
| 	_name->resizeToWidth(std::min(nameWidth - marginsAdd, _name->naturalWidth()) + marginsAdd);
 | |
| 	_name->moveToLeft(nameLeft, nameTop);
 | |
| }
 | |
| 
 | |
| // A more generic solution would be allowing an optional icon button
 | |
| // for each text button. But currently we use only one, so it is done easily:
 | |
| // There can be primary + secondary + icon buttons. If primary + secondary fit,
 | |
| // then icon is hidden, otherwise secondary is hidden and icon is shown.
 | |
| void CoverWidget::moveAndToggleButtons(int newWidth) {
 | |
| 	int buttonLeft = _userpicButton->x() + _userpicButton->width() + st::profileButtonLeft;
 | |
| 	int buttonsRight = newWidth - st::profileButtonSkip;
 | |
| 	for (int i = 0, count = _buttons.size(); i < count; ++i) {
 | |
| 		auto &button = _buttons.at(i);
 | |
| 		button.widget->moveToLeft(buttonLeft, st::profileButtonTop);
 | |
| 		if (button.replacement) {
 | |
| 			button.replacement->moveToLeft(buttonLeft, st::profileButtonTop);
 | |
| 			if (buttonLeft + button.widget->width() > buttonsRight) {
 | |
| 				button.widget->hide();
 | |
| 				button.replacement->show();
 | |
| 				buttonLeft += button.replacement->width() + st::profileButtonSkip;
 | |
| 			} else {
 | |
| 				button.widget->show();
 | |
| 				button.replacement->hide();
 | |
| 				buttonLeft += button.widget->width() + st::profileButtonSkip;
 | |
| 			}
 | |
| 		} else if (i == 1 && (buttonLeft + button.widget->width() > buttonsRight)) {
 | |
| 			// If second button is not fitting.
 | |
| 			button.widget->hide();
 | |
| 		} else {
 | |
| 			button.widget->show();
 | |
| 			buttonLeft += button.widget->width() + st::profileButtonSkip;
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void CoverWidget::showFinished() {
 | |
| 	_userpicButton->showFinished();
 | |
| }
 | |
| 
 | |
| bool CoverWidget::shareContactButtonShown() const {
 | |
| 	return _peerUser && (_buttons.size() > 1) && !(_buttons.at(1).widget->isHidden());
 | |
| }
 | |
| 
 | |
| void CoverWidget::paintEvent(QPaintEvent *e) {
 | |
| 	Painter p(this);
 | |
| 
 | |
| 	p.fillRect(e->rect(), st::profileBg);
 | |
| 
 | |
| 	p.setFont(st::profileStatusFont);
 | |
| 	p.setPen(_statusTextIsOnline ? st::profileStatusFgActive : st::profileStatusFg);
 | |
| 	p.drawTextLeft(_statusPosition.x(), _statusPosition.y(), width(), _statusText);
 | |
| 
 | |
| 	if (_peer->isVerified()) {
 | |
| 		st::profileVerifiedCheck.paint(p, _name->x() + _name->width() + st::profileVerifiedCheckShift, _name->y(),
 | |
| 		                               width());
 | |
| 	}
 | |
| 
 | |
| 	paintDivider(p);
 | |
| }
 | |
| 
 | |
| void CoverWidget::resizeDropArea(int newWidth) {
 | |
| 	if (_dropArea) {
 | |
| 		_dropArea->setGeometry(0, 0, newWidth, _dividerTop);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void CoverWidget::dropAreaHidden(CoverDropArea *dropArea) {
 | |
| 	if (_dropArea == dropArea) {
 | |
| 		_dropArea.destroyDelayed();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool CoverWidget::canEditPhoto() const {
 | |
| 	if (_peerChat && _peerChat->canEdit()) {
 | |
| 		return true;
 | |
| 	} else if (_peerMegagroup && _peerMegagroup->canEditInformation()) {
 | |
| 		return true;
 | |
| 	} else if (_peerChannel && _peerChannel->canEditInformation()) {
 | |
| 		return true;
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| bool CoverWidget::mimeDataHasImage(const QMimeData *mimeData) const {
 | |
| 	if (!mimeData) return false;
 | |
| 
 | |
| 	if (mimeData->hasImage()) return true;
 | |
| 
 | |
| 	auto uriListFormat = qsl("text/uri-list");
 | |
| 	if (!mimeData->hasFormat(uriListFormat)) return false;
 | |
| 
 | |
| 	const auto &urls = mimeData->urls();
 | |
| 	if (urls.size() != 1) return false;
 | |
| 
 | |
| 	auto &url = urls.at(0);
 | |
| 	if (!url.isLocalFile()) return false;
 | |
| 
 | |
| 	auto file = Platform::File::UrlToLocal(url);
 | |
| 
 | |
| 	QFileInfo info(file);
 | |
| 	if (info.isDir()) return false;
 | |
| 
 | |
| 	if (info.size() > App::kImageSizeLimit) return false;
 | |
| 
 | |
| 	for (auto &ext : cImgExtensions()) {
 | |
| 		if (file.endsWith(ext, Qt::CaseInsensitive)) {
 | |
| 			return true;
 | |
| 		}
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| void CoverWidget::dragEnterEvent(QDragEnterEvent *e) {
 | |
| 	if (!canEditPhoto() || !mimeDataHasImage(e->mimeData())) {
 | |
| 		e->ignore();
 | |
| 		return;
 | |
| 	}
 | |
| 	if (!_dropArea) {
 | |
| 		auto title = lang(lng_profile_drop_area_title);
 | |
| 		QString subtitle;
 | |
| 		if (_peerChat || _peerMegagroup) {
 | |
| 			subtitle = lang(lng_profile_drop_area_subtitle);
 | |
| 		} else {
 | |
| 			subtitle = lang(lng_profile_drop_area_subtitle_channel);
 | |
| 		}
 | |
| 		_dropArea.create(this, title, subtitle);
 | |
| 		resizeDropArea(width());
 | |
| 	}
 | |
| 	_dropArea->showAnimated();
 | |
| 	e->setDropAction(Qt::CopyAction);
 | |
| 	e->accept();
 | |
| }
 | |
| 
 | |
| void CoverWidget::dragLeaveEvent(QDragLeaveEvent *e) {
 | |
| 	if (_dropArea && !_dropArea->hiding()) {
 | |
| 		_dropArea->hideAnimated([this](CoverDropArea *area) { dropAreaHidden(area); });
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void CoverWidget::dropEvent(QDropEvent *e) {
 | |
| 	auto mimeData = e->mimeData();
 | |
| 
 | |
| 	QImage img;
 | |
| 	if (mimeData->hasImage()) {
 | |
| 		img = qvariant_cast<QImage>(mimeData->imageData());
 | |
| 	} else {
 | |
| 		const auto &urls = mimeData->urls();
 | |
| 		if (urls.size() == 1) {
 | |
| 			auto &url = urls.at(0);
 | |
| 			if (url.isLocalFile()) {
 | |
| 				img = App::readImage(Platform::File::UrlToLocal(url));
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (!_dropArea->hiding()) {
 | |
| 		_dropArea->hideAnimated([this](CoverDropArea *area) { dropAreaHidden(area); });
 | |
| 	}
 | |
| 	e->acceptProposedAction();
 | |
| 
 | |
| 	showSetPhotoBox(img);
 | |
| }
 | |
| 
 | |
| void CoverWidget::paintDivider(Painter &p) {
 | |
| 	auto dividerHeight = st::profileDividerLeft.height();
 | |
| 	auto dividerLeft = Adaptive::OneColumn() ? 0 : st::lineWidth;
 | |
| 	auto divider = rtlrect(dividerLeft, _dividerTop, width() - dividerLeft, dividerHeight, width());
 | |
| 	p.fillRect(divider, st::profileDividerBg);
 | |
| 	if (!Adaptive::OneColumn()) {
 | |
| 		st::profileDividerLeft.paint(p, QPoint(dividerLeft, _dividerTop), width());
 | |
| 	}
 | |
| 	auto dividerFillLeft = Adaptive::OneColumn() ? 0 : (st::lineWidth + st::profileDividerLeft.width());
 | |
| 	auto dividerFillTop =
 | |
| 	    rtlrect(dividerFillLeft, _dividerTop, width() - dividerFillLeft, st::profileDividerTop.height(), width());
 | |
| 	st::profileDividerTop.fill(p, dividerFillTop);
 | |
| 	auto dividerFillBottom = rtlrect(dividerFillLeft, _dividerTop + dividerHeight - st::profileDividerBottom.height(),
 | |
| 	                                 width() - dividerFillLeft, st::profileDividerBottom.height(), width());
 | |
| 	st::profileDividerBottom.fill(p, dividerFillBottom);
 | |
| }
 | |
| 
 | |
| void CoverWidget::notifyPeerUpdated(const Notify::PeerUpdate &update) {
 | |
| 	if (update.peer != _peer) {
 | |
| 		return;
 | |
| 	}
 | |
| 	if ((update.flags & ButtonsUpdateFlags) != 0) {
 | |
| 		refreshButtons();
 | |
| 	}
 | |
| 	if (update.flags & UpdateFlag::NameChanged) {
 | |
| 		refreshNameText();
 | |
| 	}
 | |
| 	if (update.flags & UpdateFlag::PhotoChanged) {
 | |
| 		validatePhoto();
 | |
| 	}
 | |
| 	if (update.flags & (UpdateFlag::UserOnlineChanged | UpdateFlag::MembersChanged)) {
 | |
| 		refreshStatusText();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void CoverWidget::refreshNameText() {
 | |
| 	_name->setText(App::peerName(_peer));
 | |
| 	refreshNameGeometry(width());
 | |
| }
 | |
| 
 | |
| void CoverWidget::refreshStatusText() {
 | |
| 	if (Messenger::Instance().isPhotoUpdating(_peer->id)) {
 | |
| 		_statusText = lang(lng_settings_uploading_photo);
 | |
| 		_statusTextIsOnline = false;
 | |
| 		if (!_cancelPhotoUpload) {
 | |
| 			_cancelPhotoUpload.create(this, lang(lng_cancel), st::defaultLinkButton);
 | |
| 			connect(_cancelPhotoUpload, SIGNAL(clicked()), this, SLOT(onCancelPhotoUpload()));
 | |
| 			_cancelPhotoUpload->show();
 | |
| 			_cancelPhotoUpload->moveToLeft(_statusPosition.x() + st::profileStatusFont->width(_statusText) +
 | |
| 			                                   st::profileStatusFont->spacew,
 | |
| 			                               _statusPosition.y());
 | |
| 		}
 | |
| 		update();
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	_cancelPhotoUpload.destroy();
 | |
| 	auto currentTime = unixtime();
 | |
| 	if (_peerUser) {
 | |
| 		_statusText = App::onlineText(_peerUser, currentTime, true);
 | |
| 		_statusTextIsOnline = App::onlineColorUse(_peerUser, currentTime);
 | |
| 	} else if (_peerChat && _peerChat->amIn()) {
 | |
| 		auto fullCount = std::max(_peerChat->count, _peerChat->participants.size());
 | |
| 		if (_onlineCount > 0 && _onlineCount <= fullCount) {
 | |
| 			auto membersCount = lng_chat_status_members(lt_count, fullCount);
 | |
| 			auto onlineCount = lng_chat_status_online(lt_count, _onlineCount);
 | |
| 			_statusText = lng_chat_status_members_online(lt_members_count, membersCount, lt_online_count, onlineCount);
 | |
| 		} else if (_peerChat->count > 0) {
 | |
| 			_statusText = lng_chat_status_members(lt_count, _peerChat->count);
 | |
| 		} else {
 | |
| 			_statusText = lang(lng_group_status);
 | |
| 		}
 | |
| 	} else if (_peerChannel) {
 | |
| 		auto fullCount = _peerChannel->membersCount();
 | |
| 		if (_onlineCount > 0 && _onlineCount <= fullCount) {
 | |
| 			auto membersCount = lng_chat_status_members(lt_count, fullCount);
 | |
| 			auto onlineCount = lng_chat_status_online(lt_count, _onlineCount);
 | |
| 			_statusText = lng_chat_status_members_online(lt_members_count, membersCount, lt_online_count, onlineCount);
 | |
| 		} else if (fullCount > 0) {
 | |
| 			_statusText = lng_chat_status_members(lt_count, fullCount);
 | |
| 		} else {
 | |
| 			_statusText = lang(_peerChannel->isMegagroup() ? lng_group_status : lng_channel_status);
 | |
| 		}
 | |
| 	} else {
 | |
| 		_statusText = lang(lng_chat_status_unaccessible);
 | |
| 	}
 | |
| 	update();
 | |
| }
 | |
| 
 | |
| void CoverWidget::refreshButtons() {
 | |
| 	clearButtons();
 | |
| 	if (_peerUser) {
 | |
| 		setUserButtons();
 | |
| 	} else if (_peerChat) {
 | |
| 		setChatButtons();
 | |
| 	} else if (_peerMegagroup) {
 | |
| 		setMegagroupButtons();
 | |
| 	} else if (_peerChannel) {
 | |
| 		setChannelButtons();
 | |
| 	}
 | |
| 	resizeToWidth(width());
 | |
| }
 | |
| 
 | |
| void CoverWidget::setUserButtons() {
 | |
| 	addButton(langFactory(lng_profile_send_message), SLOT(onSendMessage()));
 | |
| 	if (_peerUser->botInfo && !_peerUser->botInfo->cantJoinGroups) {
 | |
| 		addButton(langFactory(lng_profile_invite_to_group), SLOT(onAddBotToGroup()), &st::profileAddMemberButton);
 | |
| 	} else if (_peerUser->canShareThisContact()) {
 | |
| 		addButton(langFactory(lng_profile_share_contact), SLOT(onShareContact()));
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void CoverWidget::setChatButtons() {
 | |
| 	if (_peerChat->canEdit()) {
 | |
| 		addButton(langFactory(lng_profile_set_group_photo), SLOT(onSetPhoto()));
 | |
| 		addButton(langFactory(lng_profile_add_participant), SLOT(onAddMember()), &st::profileAddMemberButton);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void CoverWidget::setMegagroupButtons() {
 | |
| 	if (_peerMegagroup->amIn()) {
 | |
| 		if (canEditPhoto()) {
 | |
| 			addButton(langFactory(lng_profile_set_group_photo), SLOT(onSetPhoto()));
 | |
| 		}
 | |
| 	} else {
 | |
| 		addButton(langFactory(lng_profile_join_channel), SLOT(onJoin()));
 | |
| 	}
 | |
| 	if (_peerMegagroup->canAddMembers()) {
 | |
| 		addButton(langFactory(lng_profile_add_participant), SLOT(onAddMember()), &st::profileAddMemberButton);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void CoverWidget::setChannelButtons() {
 | |
| 	if (canEditPhoto()) {
 | |
| 		addButton(langFactory(lng_profile_set_group_photo), SLOT(onSetPhoto()));
 | |
| 	} else if (_peerChannel->amIn()) {
 | |
| 		addButton(langFactory(lng_profile_view_channel), SLOT(onViewChannel()));
 | |
| 	} else {
 | |
| 		addButton(langFactory(lng_profile_join_channel), SLOT(onJoin()));
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void CoverWidget::clearButtons() {
 | |
| 	auto buttons = base::take(_buttons);
 | |
| 	for_const (auto button, buttons) {
 | |
| 		delete button.widget;
 | |
| 		delete button.replacement;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void CoverWidget::addButton(Fn<QString()> textFactory, const char *slot, const style::RoundButton *replacementStyle) {
 | |
| 	auto &buttonStyle = _buttons.isEmpty() ? st::profilePrimaryButton : st::profileSecondaryButton;
 | |
| 	auto button = new Ui::RoundButton(this, std::move(textFactory), buttonStyle);
 | |
| 	connect(button, SIGNAL(clicked()), this, slot);
 | |
| 	button->show();
 | |
| 
 | |
| 	auto replacement = replacementStyle ? new Ui::RoundButton(this, Fn<QString()>(), *replacementStyle) : nullptr;
 | |
| 	if (replacement) {
 | |
| 		connect(replacement, SIGNAL(clicked()), this, slot);
 | |
| 		replacement->hide();
 | |
| 	}
 | |
| 
 | |
| 	_buttons.push_back({button, replacement});
 | |
| }
 | |
| 
 | |
| void CoverWidget::onOnlineCountUpdated(int onlineCount) {
 | |
| 	_onlineCount = onlineCount;
 | |
| 	refreshStatusText();
 | |
| }
 | |
| 
 | |
| void CoverWidget::onSendMessage() {
 | |
| 	Ui::showPeerHistory(_peer, ShowAtUnreadMsgId, Ui::ShowWay::Forward);
 | |
| }
 | |
| 
 | |
| void CoverWidget::onShareContact() {
 | |
| 	App::main()->shareContactLayer(_peerUser);
 | |
| }
 | |
| 
 | |
| void CoverWidget::onSetPhoto() {
 | |
| 	App::CallDelayed(st::profilePrimaryButton.ripple.hideDuration, this, [this] {
 | |
| 		auto imgExtensions = cImgExtensions();
 | |
| 		auto filter = qsl("Image files (*") + imgExtensions.join(qsl(" *")) + qsl(");;") + FileDialog::AllFilesFilter();
 | |
| 		FileDialog::GetOpenPath(lang(lng_choose_image), filter,
 | |
| 		                        base::lambda_guarded(this, [this](const FileDialog::OpenResult &result) {
 | |
| 			                        if (result.paths.isEmpty() && result.remoteContent.isEmpty()) {
 | |
| 				                        return;
 | |
| 			                        }
 | |
| 
 | |
| 			                        QImage img;
 | |
| 			                        if (!result.remoteContent.isEmpty()) {
 | |
| 				                        img = App::readImage(result.remoteContent);
 | |
| 			                        } else {
 | |
| 				                        img = App::readImage(result.paths.front());
 | |
| 			                        }
 | |
| 
 | |
| 			                        showSetPhotoBox(img);
 | |
| 		                        }));
 | |
| 	});
 | |
| }
 | |
| 
 | |
| void CoverWidget::showSetPhotoBox(const QImage &img) {
 | |
| 	if (img.isNull() || img.width() > 10 * img.height() || img.height() > 10 * img.width()) {
 | |
| 		Ui::show(Box<InformBox>(lang(lng_bad_photo)));
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	auto box = Ui::show(Box<PhotoCropBox>(img, _peer));
 | |
| 	subscribe(box->boxClosing, [this] { onPhotoUploadStatusChanged(); });
 | |
| }
 | |
| 
 | |
| void CoverWidget::onPhotoUploadStatusChanged(PeerId peerId) {
 | |
| 	if (!peerId || peerId == _peer->id) {
 | |
| 		refreshStatusText();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void CoverWidget::onAddMember() {
 | |
| 	if (_peerChat) {
 | |
| 		if (_peerChat->count >= Global::ChatSizeMax() && _peerChat->amCreator()) {
 | |
| 			Ui::show(Box<ConvertToSupergroupBox>(_peerChat));
 | |
| 		} else {
 | |
| 			AddParticipantsBoxController::Start(_peerChat);
 | |
| 		}
 | |
| 	} else if (_peerChannel && _peerChannel->mgInfo) {
 | |
| 		auto &participants = _peerChannel->mgInfo->lastParticipants;
 | |
| 		AddParticipantsBoxController::Start(_peerChannel, {participants.cbegin(), participants.cend()});
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void CoverWidget::onAddBotToGroup() {
 | |
| 	if (_peerUser && _peerUser->botInfo) {
 | |
| 		AddBotToGroupBoxController::Start(_peerUser);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void CoverWidget::onJoin() {
 | |
| 	if (!_peerChannel) return;
 | |
| 
 | |
| 	Auth().api().joinChannel(_peerChannel);
 | |
| }
 | |
| 
 | |
| void CoverWidget::onViewChannel() {
 | |
| 	Ui::showPeerHistory(_peer, ShowAtUnreadMsgId);
 | |
| }
 | |
| 
 | |
| } // namespace Profile
 |