mirror of https://github.com/procxx/kepka.git
				
				
				
			Ui::MultiSelect control ready.
This commit is contained in:
		
							parent
							
								
									3455344c62
								
							
						
					
					
						commit
						48332c0c6b
					
				
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 39 KiB After Width: | Height: | Size: 39 KiB | 
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 83 KiB After Width: | Height: | Size: 82 KiB | 
|  | @ -193,6 +193,7 @@ defaultInputArea: InputArea { | |||
| 	heightMax: 128px; | ||||
| } | ||||
| defaultInputField: InputField { | ||||
| 	textBg: white; | ||||
| 	textFg: black; | ||||
| 	textMargins: margins(0px, 6px, 0px, 4px); | ||||
| 	textAlign: align(topleft); | ||||
|  | @ -216,15 +217,6 @@ defaultInputField: InputField { | |||
| 
 | ||||
| 	height: 32px; | ||||
| } | ||||
| dialogsSearchField: InputField(defaultInputField) { | ||||
| 	textMargins: margins(34px, 7px, 34px, 7px); | ||||
| 
 | ||||
| 	iconSprite: sprite(227px, 21px, 24px, 24px); | ||||
| 	iconPosition: point(6px, 5px); | ||||
| 
 | ||||
| 	width: 240px; | ||||
| 	height: 34px; | ||||
| } | ||||
| defaultCheckbox: Checkbox { | ||||
| 	textFg: black; | ||||
| 	textBg: white; | ||||
|  | @ -746,12 +738,11 @@ dlgFilter: flatInput(inpDefGray) { | |||
| 	bgColor: #f2f2f2; | ||||
| 	phColor: #949494; | ||||
| 	phFocusColor: #a4a4a4; | ||||
| 	imgRect: sprite(227px, 21px, 24px, 24px); | ||||
| 	icon: icon {{ "box_search_icon", #aaaaaa, point(10px, 9px) }}; | ||||
| 
 | ||||
| 	width: 240px; | ||||
| 	height: 34px; | ||||
| 	textMrg: margins(34px, 2px, 34px, 4px); | ||||
| 	imgPos: point(6px, 5px); | ||||
| } | ||||
| 
 | ||||
| topBarHeight: 54px; | ||||
|  |  | |||
|  | @ -130,8 +130,7 @@ flatInput { | |||
| 	font: font; | ||||
| 	cursor: cursor; | ||||
| 
 | ||||
| 	imgRect: sprite; | ||||
| 	imgPos: point; | ||||
| 	icon: icon; | ||||
| 
 | ||||
| 	borderWidth: pixels; | ||||
| 	borderColor: color; | ||||
|  | @ -394,6 +393,7 @@ InputArea { | |||
| } | ||||
| 
 | ||||
| InputField { | ||||
| 	textBg: color; | ||||
| 	textFg: color; | ||||
| 	textMargins: margins; | ||||
| 	textAlign: align; | ||||
|  | @ -418,9 +418,6 @@ InputField { | |||
| 
 | ||||
| 	width: pixels; | ||||
| 	height: pixels; | ||||
| 
 | ||||
| 	iconSprite: sprite; | ||||
| 	iconPosition: point; | ||||
| } | ||||
| 
 | ||||
| PeerAvatarButton { | ||||
|  |  | |||
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 350 B | 
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 707 B | 
|  | @ -69,20 +69,19 @@ aboutRevokePublicLabel: flatLabel(labelDefFlat) { | |||
| } | ||||
| 
 | ||||
| boxSearchField: InputField(defaultInputField) { | ||||
| 	textMargins: margins(41px, 16px, 41px, 0px); | ||||
| 	textBg: transparent; | ||||
| 	textMargins: margins(2px, 7px, 2px, 0px); | ||||
| 
 | ||||
| 	placeholderFg: #999; | ||||
| 	placeholderFgActive: #aaa; | ||||
| 	placeholderMargins: margins(4px, 0px, 4px, 0px); | ||||
| 	placeholderMargins: margins(2px, 0px, 2px, 0px); | ||||
| 	duration: 150; | ||||
| 
 | ||||
| 	border: 0px; | ||||
| 	borderActive: 0px; | ||||
| 	borderError: 0px; | ||||
| 
 | ||||
| 	height: 48px; | ||||
| 
 | ||||
| 	iconSprite: sprite(227px, 21px, 24px, 24px); | ||||
| 	iconPosition: point(15px, 14px); | ||||
| 	height: 32px; | ||||
| 
 | ||||
| 	font: normalFont; | ||||
| } | ||||
|  | @ -95,15 +94,46 @@ boxSearchCancel: IconButton { | |||
| 
 | ||||
| 	icon: icon {{ "box_search_cancel", #000000 }}; | ||||
| 	iconPosition: point(8px, 18px); | ||||
| 	downIconPosition: point(8px, 18px); | ||||
| 	downIconPosition: point(8px, 19px); | ||||
| 
 | ||||
| 	duration: 150; | ||||
| } | ||||
| 
 | ||||
| contactsMultiSelect: MultiSelect { | ||||
| 	field: boxSearchField; | ||||
| 	cancel: boxSearchCancel; | ||||
| 	padding: margins(8px, 8px, 8px, 8px); | ||||
| 	maxHeight: 104px; | ||||
| 	scroll: flatScroll(solidScroll) { | ||||
| 		deltat: 3px; | ||||
| 		deltab: 3px; | ||||
| 		round: 1px; | ||||
| 		width: 8px; | ||||
| 		deltax: 3px; | ||||
| 		hiding: 1000; | ||||
| 	} | ||||
| 
 | ||||
| 	item: MultiSelectItem { | ||||
| 		padding: margins(6px, 7px, 12px, 0px); | ||||
| 		maxWidth: 128px; | ||||
| 		height: 32px; | ||||
| 		font: normalFont; | ||||
| 		textBg: contactsBgOver; | ||||
| 		textFg: windowTextFg; | ||||
| 		textActiveBg: titleBg; | ||||
| 		textActiveFg: white; | ||||
| 		deleteFg: white; | ||||
| 		deleteLeft: 9px; | ||||
| 		deleteStroke: 3px; | ||||
| 		duration: 150; | ||||
| 		minScale: 0.3; | ||||
| 	} | ||||
| 	itemSkip: 8px; | ||||
| 
 | ||||
| 	field: boxSearchField; | ||||
| 	fieldIcon: icon {{ "box_search_icon", #aaaaaa, point(11px, 9px) }}; | ||||
| 	fieldIconSkip: 36px; | ||||
| 	fieldCancel: boxSearchCancel; | ||||
| 	fieldCancelSkip: 34px; | ||||
| 	fieldMinWidth: 42px; | ||||
| } | ||||
| contactsPhotoCheckbox: RoundImageCheckbox { | ||||
| 	imageRadius: 21px; | ||||
|  |  | |||
|  | @ -23,6 +23,7 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org | |||
| 
 | ||||
| #include "dialogs/dialogs_indexed_list.h" | ||||
| #include "styles/style_dialogs.h" | ||||
| #include "styles/style_history.h" | ||||
| #include "styles/style_boxes.h" | ||||
| #include "lang.h" | ||||
| #include "boxes/addcontactbox.h" | ||||
|  | @ -97,10 +98,18 @@ ContactsBox::ContactsBox(UserData *bot) : ItemListBox(st::contactsScroll) | |||
| } | ||||
| 
 | ||||
| void ContactsBox::init() { | ||||
| 	ItemListBox::init(_inner); | ||||
| 	_select->resizeToWidth(st::boxWideWidth); | ||||
| 
 | ||||
| 	auto inviting = (_inner->creating() == CreatingGroupGroup) || (_inner->channel() && _inner->membersFilter() == MembersFilter::Recent) || _inner->chat(); | ||||
| 	auto topSkip = st::boxTitleHeight + _select->height(); | ||||
| 	auto bottomSkip = inviting ? (st::boxButtonPadding.top() + _next.height() + st::boxButtonPadding.bottom()) : st::boxScrollSkip; | ||||
| 	ItemListBox::init(_inner, bottomSkip, topSkip); | ||||
| 
 | ||||
| 	connect(_inner, SIGNAL(chosenChanged()), this, SLOT(onChosenChanged())); | ||||
| 	connect(_inner, SIGNAL(addRequested()), App::wnd(), SLOT(onShowAddContact())); | ||||
| 	_inner->setPeerSelectedChangedCallback([this](PeerData *peer, bool checked) { | ||||
| 		onPeerSelectedChanged(peer, checked); | ||||
| 	}); | ||||
| 
 | ||||
| 	if (_inner->channel() && _inner->membersFilter() == MembersFilter::Admins) { | ||||
| 		_next.hide(); | ||||
| 		_cancel.hide(); | ||||
|  | @ -120,7 +129,14 @@ void ContactsBox::init() { | |||
| 	connect(&_cancel, SIGNAL(clicked()), this, SLOT(onClose())); | ||||
| 	connect(scrollArea(), SIGNAL(scrolled()), this, SLOT(onScroll())); | ||||
| 	_select->setQueryChangedCallback([this](const QString &query) { onFilterUpdate(query); }); | ||||
| 	_select->setItemRemovedCallback([this](uint64 itemId) { | ||||
| 		if (auto peer = App::peerLoaded(itemId)) { | ||||
| 			_inner->peerUnselected(peer); | ||||
| 			update(); | ||||
| 		} | ||||
| 	}); | ||||
| 	_select->setSubmittedCallback([this](bool) { onSubmit(); }); | ||||
| 	_select->setResizedCallback([this] { updateScrollSkips(); }); | ||||
| 	connect(_inner, SIGNAL(mustScrollTo(int, int)), scrollArea(), SLOT(scrollToY(int, int))); | ||||
| 	connect(_inner, SIGNAL(searchByUsername()), this, SLOT(onNeedSearchByUsername())); | ||||
| 	connect(_inner, SIGNAL(adminAdded()), this, SIGNAL(adminAdded())); | ||||
|  | @ -184,7 +200,7 @@ void ContactsBox::peopleReceived(const MTPcontacts_Found &result, mtpRequestId r | |||
| 		} | ||||
| 
 | ||||
| 		_peopleRequest = 0; | ||||
| 		_inner->updateSel(); | ||||
| 		_inner->updateSelection(); | ||||
| 		onScroll(); | ||||
| 	} | ||||
| } | ||||
|  | @ -266,21 +282,31 @@ void ContactsBox::paintEvent(QPaintEvent *e) { | |||
| 	} | ||||
| } | ||||
| 
 | ||||
| void ContactsBox::updateScrollSkips() { | ||||
| 	auto oldScrollHeight = scrollArea()->height(); | ||||
| 	auto inviting = (_inner->creating() == CreatingGroupGroup) || (_inner->channel() && _inner->membersFilter() == MembersFilter::Recent) || _inner->chat(); | ||||
| 	auto topSkip = st::boxTitleHeight + _select->height(); | ||||
| 	auto bottomSkip = inviting ? (st::boxButtonPadding.top() + _next.height() + st::boxButtonPadding.bottom()) : st::boxScrollSkip; | ||||
| 	setScrollSkips(bottomSkip, topSkip); | ||||
| 	auto scrollHeightDelta = scrollArea()->height() - oldScrollHeight; | ||||
| 	if (scrollHeightDelta) { | ||||
| 		scrollArea()->scrollToY(scrollArea()->scrollTop() - scrollHeightDelta); | ||||
| 	} | ||||
| 
 | ||||
| 	_topShadow.setGeometry(0, st::boxTitleHeight + _select->height(), width(), st::lineWidth); | ||||
| } | ||||
| 
 | ||||
| void ContactsBox::resizeEvent(QResizeEvent *e) { | ||||
| 	ItemListBox::resizeEvent(e); | ||||
| 
 | ||||
| 	_select->resizeToWidth(width()); | ||||
| 	_select->moveToLeft(0, st::boxTitleHeight); | ||||
| 
 | ||||
| 	auto inviting = (_inner->creating() == CreatingGroupGroup) || (_inner->channel() && _inner->membersFilter() == MembersFilter::Recent) || _inner->chat(); | ||||
| 	auto topSkip = st::boxTitleHeight + _select->height(); | ||||
| 	auto bottomSkip = inviting ? (st::boxButtonPadding.top() + _next.height() + st::boxButtonPadding.bottom()) : st::boxScrollSkip; | ||||
| 	setScrollSkips(bottomSkip, topSkip); | ||||
| 	updateScrollSkips(); | ||||
| 
 | ||||
| 	_inner->resize(width(), _inner->height()); | ||||
| 	_next.moveToRight(st::boxButtonPadding.right(), height() - st::boxButtonPadding.bottom() - _next.height()); | ||||
| 	_cancel.moveToRight(st::boxButtonPadding.right() + _next.width() + st::boxButtonPadding.left(), _next.y()); | ||||
| 	_topShadow.setGeometry(0, st::boxTitleHeight + _select->height(), width(), st::lineWidth); | ||||
| 	if (_bottomShadow) _bottomShadow->setGeometry(0, height() - st::boxButtonPadding.bottom() - _next.height() - st::boxButtonPadding.top() - st::lineWidth, width(), st::lineWidth); | ||||
| } | ||||
| 
 | ||||
|  | @ -295,7 +321,25 @@ void ContactsBox::onFilterUpdate(const QString &filter) { | |||
| 	_inner->updateFilter(filter); | ||||
| } | ||||
| 
 | ||||
| void ContactsBox::onChosenChanged() { | ||||
| void ContactsBox::onPeerSelectedChanged(PeerData *peer, bool checked) { | ||||
| 	if (checked) { | ||||
| 		auto getColor = [peer]() -> const style::color &{ | ||||
| 			switch (peer->colorIndex) { | ||||
| 			case 1: return st::historyPeer2UserpicFg; | ||||
| 			case 2: return st::historyPeer3UserpicFg; | ||||
| 			case 3: return st::historyPeer4UserpicFg; | ||||
| 			case 4: return st::historyPeer5UserpicFg; | ||||
| 			case 5: return st::historyPeer6UserpicFg; | ||||
| 			case 6: return st::historyPeer7UserpicFg; | ||||
| 			case 7: return st::historyPeer8UserpicFg; | ||||
| 			default: return st::historyPeer1UserpicFg; | ||||
| 			} | ||||
| 		}; | ||||
| 		_select->addItem(peer->id, peer->shortName(), getColor(), PaintUserpicCallback(peer)); | ||||
| 		_select->clearQuery(); | ||||
| 	} else { | ||||
| 		_select->removeItem(peer->id); | ||||
| 	} | ||||
| 	update(); | ||||
| } | ||||
| 
 | ||||
|  | @ -482,12 +526,8 @@ bool ContactsBox::creationFail(const RPCError &error) { | |||
| 	return false; | ||||
| } | ||||
| 
 | ||||
| ContactsBox::Inner::ContactData::ContactData() : name(st::boxWideWidth) { | ||||
| } | ||||
| 
 | ||||
| ContactsBox::Inner::ContactData::ContactData(PeerData *peer, Ui::RoundImageCheckbox::UpdateCallback &&updateCallback) | ||||
| : checkbox(std_::make_unique<Ui::RoundImageCheckbox>(st::contactsPhotoCheckbox, std_::move(updateCallback), PaintUserpicCallback(peer))) | ||||
| , name(st::boxWideWidth) { | ||||
| ContactsBox::Inner::ContactData::ContactData(PeerData *peer, base::lambda_wrap<void()> updateCallback) | ||||
| : checkbox(std_::make_unique<Ui::RoundImageCheckbox>(st::contactsPhotoCheckbox, std_::move(updateCallback), PaintUserpicCallback(peer))) { | ||||
| } | ||||
| 
 | ||||
| ContactsBox::Inner::Inner(QWidget *parent, CreatingGroupType creating) : ScrolledWidget(parent) | ||||
|  | @ -613,7 +653,7 @@ void ContactsBox::Inner::initList() { | |||
| 		if (i.key()->id == peerFromUser(_chat->creator)) continue; | ||||
| 		if (!_allAdmins.checked() && _chat->admins.contains(i.key())) { | ||||
| 			admins.push_back(i.key()); | ||||
| 			_checkedContacts.insert(i.key(), true); | ||||
| 			_checkedContacts.insert(i.key()); | ||||
| 		} else { | ||||
| 			others.push_back(i.key()); | ||||
| 		} | ||||
|  | @ -1186,13 +1226,13 @@ void ContactsBox::Inner::leaveEvent(QEvent *e) { | |||
| void ContactsBox::Inner::mouseMoveEvent(QMouseEvent *e) { | ||||
| 	_mouseSel = true; | ||||
| 	_lastMousePos = e->globalPos(); | ||||
| 	updateSel(); | ||||
| 	updateSelection(); | ||||
| } | ||||
| 
 | ||||
| void ContactsBox::Inner::mousePressEvent(QMouseEvent *e) { | ||||
| 	_mouseSel = true; | ||||
| 	_lastMousePos = e->globalPos(); | ||||
| 	updateSel(); | ||||
| 	updateSelection(); | ||||
| 	if (e->button() == Qt::LeftButton) { | ||||
| 		chooseParticipant(); | ||||
| 	} | ||||
|  | @ -1205,29 +1245,35 @@ void ContactsBox::Inner::chooseParticipant() { | |||
| 		_time = unixtime(); | ||||
| 		if (_filter.isEmpty()) { | ||||
| 			if (_byUsernameSel >= 0 && _byUsernameSel < _byUsername.size()) { | ||||
| 				if (d_byUsername[_byUsernameSel]->disabledChecked) return; | ||||
| 				changeCheckState(d_byUsername[_byUsernameSel], _byUsername[_byUsernameSel]); | ||||
| 			} else { | ||||
| 				if (!_sel || contactData(_sel)->disabledChecked) return; | ||||
| 				auto data = d_byUsername[_byUsernameSel]; | ||||
| 				auto peer = _byUsername[_byUsernameSel]; | ||||
| 				if (data->disabledChecked) return; | ||||
| 
 | ||||
| 				changeCheckState(data, peer); | ||||
| 			} else if (_sel) { | ||||
| 				auto data = contactData(_sel); | ||||
| 				auto peer = _sel->history()->peer; | ||||
| 				if (data->disabledChecked) return; | ||||
| 
 | ||||
| 				changeCheckState(_sel); | ||||
| 			} | ||||
| 		} else { | ||||
| 			if (_byUsernameSel >= 0 && _byUsernameSel < _byUsernameFiltered.size()) { | ||||
| 				if (d_byUsernameFiltered[_byUsernameSel]->disabledChecked) return; | ||||
| 				changeCheckState(d_byUsernameFiltered[_byUsernameSel], _byUsernameFiltered[_byUsernameSel]); | ||||
| 				auto data = d_byUsernameFiltered[_byUsernameSel]; | ||||
| 				auto peer = _byUsernameFiltered[_byUsernameSel]; | ||||
| 				if (data->disabledChecked) return; | ||||
| 
 | ||||
| 				ContactData *moving = d_byUsernameFiltered[_byUsernameSel]; | ||||
| 				int32 i = 0, l = d_byUsername.size(); | ||||
| 				int i = 0, l = d_byUsername.size(); | ||||
| 				for (; i < l; ++i) { | ||||
| 					if (d_byUsername[i] == moving) { | ||||
| 					if (d_byUsername[i] == data) { | ||||
| 						break; | ||||
| 					} | ||||
| 				} | ||||
| 				if (i == l) { | ||||
| 					d_byUsername.push_back(moving); | ||||
| 					_byUsername.push_back(_byUsernameFiltered[_byUsernameSel]); | ||||
| 					d_byUsername.push_back(data); | ||||
| 					_byUsername.push_back(peer); | ||||
| 					for (i = 0, l = _byUsernameDatas.size(); i < l;) { | ||||
| 						if (_byUsernameDatas[i] == moving) { | ||||
| 						if (_byUsernameDatas[i] == data) { | ||||
| 							_byUsernameDatas.removeAt(i); | ||||
| 							--l; | ||||
| 						} else { | ||||
|  | @ -1235,9 +1281,14 @@ void ContactsBox::Inner::chooseParticipant() { | |||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 			} else { | ||||
| 				if (_filteredSel < 0 || _filteredSel >= _filtered.size() || contactData(_filtered[_filteredSel])->disabledChecked) return; | ||||
| 				changeCheckState(_filtered[_filteredSel]); | ||||
| 
 | ||||
| 				changeCheckState(data, peer); | ||||
| 			} else if (_filteredSel >= 0 && _filteredSel < _filtered.size()) { | ||||
| 				auto data = contactData(_filtered[_filteredSel]); | ||||
| 				auto peer = _filtered[_filteredSel]->history()->peer; | ||||
| 				if (data->disabledChecked) return; | ||||
| 
 | ||||
| 				changeCheckState(data, peer); | ||||
| 			} | ||||
| 		} | ||||
| 	} else { | ||||
|  | @ -1304,25 +1355,43 @@ void ContactsBox::Inner::changeCheckState(Dialogs::Row *row) { | |||
| void ContactsBox::Inner::changeCheckState(ContactData *data, PeerData *peer) { | ||||
| 	t_assert(usingMultiSelect()); | ||||
| 
 | ||||
| 	int32 cnt = _selCount; | ||||
| 	if (data->checkbox->checked()) { | ||||
| 		data->checkbox->setChecked(false); | ||||
| 		_checkedContacts.remove(peer); | ||||
| 		--_selCount; | ||||
| 		changePeerCheckState(data, peer, false); | ||||
| 	} else if (selectedCount() < ((_channel && _channel->isMegagroup()) ? Global::MegagroupSizeMax() : Global::ChatSizeMax())) { | ||||
| 		data->checkbox->setChecked(true); | ||||
| 		_checkedContacts.insert(peer, true); | ||||
| 		++_selCount; | ||||
| 		changePeerCheckState(data, peer, true); | ||||
| 	} else if (_channel && !_channel->isMegagroup()) { | ||||
| 		Ui::showLayer(new MaxInviteBox(_channel->inviteLink()), KeepOtherLayers); | ||||
| 	} else if (!_channel && selectedCount() >= Global::ChatSizeMax() && selectedCount() < Global::MegagroupSizeMax()) { | ||||
| 		Ui::showLayer(new InformBox(lng_profile_add_more_after_upgrade(lt_count, Global::MegagroupSizeMax())), KeepOtherLayers); | ||||
| 	} | ||||
| 	if (cnt != _selCount) emit chosenChanged(); | ||||
| } | ||||
| 
 | ||||
| void ContactsBox::Inner::peerUnselected(PeerData *peer) { | ||||
| 	// If data is nullptr we simply won't do anything.
 | ||||
| 	auto data = _contactsData.value(peer, nullptr); | ||||
| 	changePeerCheckState(data, peer, false, ChangeStateWay::SkipCallback); | ||||
| } | ||||
| 
 | ||||
| void ContactsBox::Inner::setPeerSelectedChangedCallback(base::lambda_unique<void(PeerData *peer, bool selected)> callback) { | ||||
| 	_peerSelectedChangedCallback = std_::move(callback); | ||||
| } | ||||
| 
 | ||||
| void ContactsBox::Inner::changePeerCheckState(ContactData *data, PeerData *peer, bool checked, ChangeStateWay useCallback) { | ||||
| 	if (data) { | ||||
| 		data->checkbox->setChecked(checked); | ||||
| 	} | ||||
| 	if (checked) { | ||||
| 		_checkedContacts.insert(peer); | ||||
| 	} else { | ||||
| 		_checkedContacts.remove(peer); | ||||
| 	} | ||||
| 	if (useCallback != ChangeStateWay::SkipCallback) { | ||||
| 		_peerSelectedChangedCallback(peer, checked); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| int32 ContactsBox::Inner::selectedCount() const { | ||||
| 	int32 result = _selCount; | ||||
| 	auto result = _checkedContacts.size(); | ||||
| 	if (_chat) { | ||||
| 		result += qMax(_chat->count, 1); | ||||
| 	} else if (_channel) { | ||||
|  | @ -1333,7 +1402,7 @@ int32 ContactsBox::Inner::selectedCount() const { | |||
| 	return result; | ||||
| } | ||||
| 
 | ||||
| void ContactsBox::Inner::updateSel() { | ||||
| void ContactsBox::Inner::updateSelection() { | ||||
| 	if (!_mouseSel) return; | ||||
| 
 | ||||
| 	QPoint p(mapFromGlobal(_lastMousePos)); | ||||
|  |  | |||
|  | @ -56,8 +56,7 @@ public: | |||
| signals: | ||||
| 	void adminAdded(); | ||||
| 
 | ||||
| public slots: | ||||
| 	void onChosenChanged(); | ||||
| private slots: | ||||
| 	void onScroll(); | ||||
| 
 | ||||
| 	void onInvite(); | ||||
|  | @ -80,7 +79,9 @@ protected: | |||
| 
 | ||||
| private: | ||||
| 	void init(); | ||||
| 	void updateScrollSkips(); | ||||
| 	void onFilterUpdate(const QString &filter); | ||||
| 	void onPeerSelectedChanged(PeerData *peer, bool checked); | ||||
| 
 | ||||
| 	class Inner; | ||||
| 	ChildWidget<Inner> _inner; | ||||
|  | @ -136,10 +137,11 @@ public: | |||
| 	Inner(QWidget *parent, ChatData *chat, MembersFilter membersFilter); | ||||
| 	Inner(QWidget *parent, UserData *bot); | ||||
| 
 | ||||
| 	void init(); | ||||
| 	void initList(); | ||||
| 	void setPeerSelectedChangedCallback(base::lambda_unique<void(PeerData *peer, bool selected)> callback); | ||||
| 	void peerUnselected(PeerData *peer); | ||||
| 
 | ||||
| 	void updateFilter(QString filter = QString()); | ||||
| 	void updateSelection(); | ||||
| 
 | ||||
| 	void selectSkip(int32 dir); | ||||
| 	void selectSkipPage(int32 h, int32 dir); | ||||
|  | @ -177,14 +179,12 @@ public: | |||
| signals: | ||||
| 	void mustScrollTo(int ymin, int ymax); | ||||
| 	void searchByUsername(); | ||||
| 	void chosenChanged(); | ||||
| 	void adminAdded(); | ||||
| 	void addRequested(); | ||||
| 
 | ||||
| public slots: | ||||
| private slots: | ||||
| 	void onDialogRowReplaced(Dialogs::Row *oldRow, Dialogs::Row *newRow); | ||||
| 
 | ||||
| 	void updateSel(); | ||||
| 	void peerUpdated(PeerData *peer); | ||||
| 	void onPeerNameChanged(PeerData *peer, const PeerData::Names &oldNames, const PeerData::NameFirstChars &oldChars); | ||||
| 
 | ||||
|  | @ -204,8 +204,8 @@ protected: | |||
| 
 | ||||
| private: | ||||
| 	struct ContactData { | ||||
| 		ContactData(); | ||||
| 		ContactData(PeerData *peer, Ui::RoundImageCheckbox::UpdateCallback &&updateCallback); | ||||
| 		ContactData() = default; | ||||
| 		ContactData(PeerData *peer, base::lambda_wrap<void()> updateCallback); | ||||
| 
 | ||||
| 		std_::unique_ptr<Ui::RoundImageCheckbox> checkbox; | ||||
| 		Text name; | ||||
|  | @ -214,6 +214,9 @@ private: | |||
| 		bool disabledChecked = false; | ||||
| 	}; | ||||
| 
 | ||||
| 	void init(); | ||||
| 	void initList(); | ||||
| 
 | ||||
| 	void updateRowWithTop(int rowTop); | ||||
| 	int getSelectedRowTop() const; | ||||
| 	void updateSelectedRow(); | ||||
|  | @ -227,6 +230,11 @@ private: | |||
| 
 | ||||
| 	void changeCheckState(Dialogs::Row *row); | ||||
| 	void changeCheckState(ContactData *data, PeerData *peer); | ||||
| 	enum class ChangeStateWay { | ||||
| 		Default, | ||||
| 		SkipCallback, | ||||
| 	}; | ||||
| 	void changePeerCheckState(ContactData *data, PeerData *peer, bool checked, ChangeStateWay useCallback = ChangeStateWay::Default); | ||||
| 
 | ||||
| 	template <typename FilterCallback> | ||||
| 	void addDialogsToList(FilterCallback callback); | ||||
|  | @ -235,6 +243,8 @@ private: | |||
| 		return (_chat != nullptr) || (_creating != CreatingGroupNone && (!_channel || _membersFilter != MembersFilter::Admins)); | ||||
| 	} | ||||
| 
 | ||||
| 	base::lambda_unique<void(PeerData *peer, bool selected)> _peerSelectedChangedCallback; | ||||
| 
 | ||||
| 	int32 _rowHeight; | ||||
| 	int _newItemHeight = 0; | ||||
| 	bool _newItemSel = false; | ||||
|  | @ -261,24 +271,22 @@ private: | |||
| 	Dialogs::IndexedList *_contacts = nullptr; | ||||
| 	Dialogs::Row *_sel = nullptr; | ||||
| 	QString _filter; | ||||
| 	typedef QVector<Dialogs::Row*> FilteredDialogs; | ||||
| 	using FilteredDialogs = QVector<Dialogs::Row*>; | ||||
| 	FilteredDialogs _filtered; | ||||
| 	int _filteredSel = -1; | ||||
| 	bool _mouseSel = false; | ||||
| 
 | ||||
| 	int _selCount = 0; | ||||
| 
 | ||||
| 	typedef QMap<PeerData*, ContactData*> ContactsData; | ||||
| 	using ContactsData = QMap<PeerData*, ContactData*>; | ||||
| 	ContactsData _contactsData; | ||||
| 	typedef QMap<PeerData*, bool> CheckedContacts; | ||||
| 	using CheckedContacts = OrderedSet<PeerData*>; | ||||
| 	CheckedContacts _checkedContacts; | ||||
| 
 | ||||
| 	ContactData *contactData(Dialogs::Row *row); | ||||
| 
 | ||||
| 	bool _searching = false; | ||||
| 	QString _lastQuery; | ||||
| 	typedef QVector<PeerData*> ByUsernameRows; | ||||
| 	typedef QVector<ContactData*> ByUsernameDatas; | ||||
| 	using ByUsernameRows = QVector<PeerData*>; | ||||
| 	using ByUsernameDatas = QVector<ContactData*>; | ||||
| 	ByUsernameRows _byUsername, _byUsernameFiltered; | ||||
| 	ByUsernameDatas d_byUsername, d_byUsernameFiltered; // filtered is partly subset of d_byUsername, partly subset of _byUsernameDatas
 | ||||
| 	ByUsernameDatas _byUsernameDatas; | ||||
|  |  | |||
|  | @ -473,7 +473,7 @@ void ShareBox::Inner::paintChat(Painter &p, Chat *chat, int index) { | |||
| 	chat->name.drawLeftElided(p, x + nameLeft, y + nameTop, nameWidth, outerWidth, 2, style::al_top, 0, -1, 0, true); | ||||
| } | ||||
| 
 | ||||
| ShareBox::Inner::Chat::Chat(PeerData *peer, Ui::RoundImageCheckbox::UpdateCallback &&updateCallback) | ||||
| ShareBox::Inner::Chat::Chat(PeerData *peer, base::lambda_wrap<void()> updateCallback) | ||||
| : peer(peer) | ||||
| , checkbox(st::sharePhotoCheckbox, std_::move(updateCallback), PaintUserpicCallback(peer)) | ||||
| , name(st::sharePhotoCheckbox.imageRadius * 2) { | ||||
|  |  | |||
|  | @ -153,7 +153,7 @@ private: | |||
| 	int displayedChatsCount() const; | ||||
| 
 | ||||
| 	struct Chat { | ||||
| 		Chat(PeerData *peer, Ui::RoundImageCheckbox::UpdateCallback &&updateCallback); | ||||
| 		Chat(PeerData *peer, base::lambda_wrap<void()> updateCallback); | ||||
| 
 | ||||
| 		PeerData *peer; | ||||
| 		Ui::RoundImageCheckbox checkbox; | ||||
|  |  | |||
|  | @ -359,6 +359,10 @@ public: | |||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	lambda_wrap clone() const { | ||||
| 		return *this; | ||||
| 	} | ||||
| 
 | ||||
| 	template <typename Lambda, typename = IsOther<Lambda>> | ||||
| 	lambda_wrap(const Lambda &other) : Parent(&internal::lambda_wrap_helper_copy<Lambda, Return, Args...>::instance, typename Parent::Private()) { | ||||
| 		internal::lambda_wrap_helper_copy<Lambda, Return, Args...>::construct_copy_lambda_method(this->storage_, &other); | ||||
|  |  | |||
|  | @ -102,6 +102,22 @@ namespace anim { | |||
|     float64 easeInQuint(const float64 &delta, const float64 &dt); | ||||
|     float64 easeOutQuint(const float64 &delta, const float64 &dt); | ||||
| 
 | ||||
| 	template <int BumpRatioNumerator, int BumpRatioDenominator> | ||||
| 	float64 bumpy(const float64 &delta, const float64 &dt) { | ||||
| 		struct Bumpy { | ||||
| 			Bumpy() | ||||
| 				: bump(BumpRatioNumerator / float64(BumpRatioDenominator)) | ||||
| 				, dt0(bump - sqrt(bump * (bump - 1.))) | ||||
| 				, k(1 / (2 * dt0 - 1)) { | ||||
| 			} | ||||
| 			float64 bump; | ||||
| 			float64 dt0; | ||||
| 			float64 k; | ||||
| 		}; | ||||
| 		static Bumpy data; | ||||
| 		return delta * (data.bump - data.k * (dt - data.dt0) * (dt - data.dt0)); | ||||
| 	} | ||||
| 
 | ||||
| 	class fvalue { // float animated value
 | ||||
| 	public: | ||||
| 		using ValueType = float64; | ||||
|  | @ -499,7 +515,7 @@ public: | |||
| 	template <typename Lambda> | ||||
| 	void start(Lambda &&updateCallback, const ValueType &from, const ValueType &to, float64 duration, anim::transition transition = anim::linear) { | ||||
| 		if (!_data) { | ||||
| 			_data = std_::make_unique<Data>(from, std_::move(updateCallback)); | ||||
| 			_data = std_::make_unique<Data>(from, std_::forward<Lambda>(updateCallback)); | ||||
| 		} | ||||
| 		_data->value.start(to); | ||||
| 		_data->duration = duration; | ||||
|  | @ -522,6 +538,11 @@ private: | |||
| 			, a_animation(animation(this, &Data::step)) | ||||
| 			, updateCallback(std_::move(updateCallback)) { | ||||
| 		} | ||||
| 		Data(const ValueType &from, const base::lambda_wrap<void()> &updateCallback) | ||||
| 			: value(from, from) | ||||
| 			, a_animation(animation(this, &Data::step)) | ||||
| 			, updateCallback(base::lambda_wrap<void()>(updateCallback)) { | ||||
| 		} | ||||
| 		void step(float64 ms, bool timer) { | ||||
| 			auto dt = (ms >= duration) ? 1. : (ms / duration); | ||||
| 			if (dt >= 1) { | ||||
|  |  | |||
|  | @ -56,25 +56,9 @@ void prepareCheckCaches(const style::RoundImageCheckbox *st, QPixmap &checkBgCac | |||
| 	checkFullCache.setDevicePixelRatio(cRetinaFactor()); | ||||
| } | ||||
| 
 | ||||
| struct AnimBumpy { | ||||
| 	AnimBumpy(float64 bump) : bump(bump) | ||||
| 		, dt0(bump - sqrt(bump * (bump - 1.))) | ||||
| 		, k(1 / (2 * dt0 - 1)) { | ||||
| 	} | ||||
| 	float64 bump; | ||||
| 	float64 dt0; | ||||
| 	float64 k; | ||||
| }; | ||||
| 
 | ||||
| template <int BumpRatioPercent> | ||||
| float64 anim_bumpy(const float64 &delta, const float64 &dt) { | ||||
| 	static AnimBumpy data = { BumpRatioPercent / 100. }; | ||||
| 	return delta * (data.bump - data.k * (dt - data.dt0) * (dt - data.dt0)); | ||||
| } | ||||
| 
 | ||||
| } // namespace
 | ||||
| 
 | ||||
| RoundImageCheckbox::RoundImageCheckbox(const style::RoundImageCheckbox &st, UpdateCallback &&updateCallback, PaintRoundImage &&paintRoundImage) | ||||
| RoundImageCheckbox::RoundImageCheckbox(const style::RoundImageCheckbox &st, base::lambda_wrap<void()> updateCallback, PaintRoundImage paintRoundImage) | ||||
| : _st(st) | ||||
| , _updateCallback(std_::move(updateCallback)) | ||||
| , _paintRoundImage(std_::move(paintRoundImage)) { | ||||
|  | @ -84,13 +68,14 @@ RoundImageCheckbox::RoundImageCheckbox(const style::RoundImageCheckbox &st, Upda | |||
| void RoundImageCheckbox::paint(Painter &p, int x, int y, int outerWidth) { | ||||
| 	auto selectionLevel = _selection.current(_checked ? 1. : 0.); | ||||
| 	if (_selection.animating()) { | ||||
| 		p.setRenderHint(QPainter::SmoothPixmapTransform, true); | ||||
| 		auto userpicRadius = qRound(kWideScale * (_st.imageRadius + (_st.imageSmallRadius - _st.imageRadius) * selectionLevel)); | ||||
| 		auto userpicShift = kWideScale * _st.imageRadius - userpicRadius; | ||||
| 		auto userpicLeft = x - (kWideScale - 1) * _st.imageRadius + userpicShift; | ||||
| 		auto userpicTop = y - (kWideScale - 1) * _st.imageRadius + userpicShift; | ||||
| 		auto to = QRect(userpicLeft, userpicTop, userpicRadius * 2, userpicRadius * 2); | ||||
| 		auto from = QRect(QPoint(0, 0), _wideCache.size()); | ||||
| 
 | ||||
| 		p.setRenderHint(QPainter::SmoothPixmapTransform, true); | ||||
| 		p.drawPixmapLeft(to, outerWidth, _wideCache, from); | ||||
| 		p.setRenderHint(QPainter::SmoothPixmapTransform, false); | ||||
| 	} else { | ||||
|  | @ -159,7 +144,7 @@ void RoundImageCheckbox::setChecked(bool checked, SetStyle speed) { | |||
| 	_checked = checked; | ||||
| 	if (_checked) { | ||||
| 		_icons.push_back(Icon()); | ||||
| 		_icons.back().fadeIn.start(UpdateCallback(_updateCallback), 0, 1, _st.selectDuration); | ||||
| 		_icons.back().fadeIn.start(_updateCallback, 0, 1, _st.selectDuration); | ||||
| 		if (speed != SetStyle::Animated) { | ||||
| 			_icons.back().fadeIn.finish(); | ||||
| 		} | ||||
|  | @ -176,7 +161,7 @@ void RoundImageCheckbox::setChecked(bool checked, SetStyle speed) { | |||
| 	} | ||||
| 	if (speed == SetStyle::Animated) { | ||||
| 		prepareWideCache(); | ||||
| 		_selection.start(UpdateCallback(_updateCallback), _checked ? 0 : 1, _checked ? 1 : 0, _st.selectDuration, anim_bumpy<125>); | ||||
| 		_selection.start(_updateCallback, _checked ? 0 : 1, _checked ? 1 : 0, _st.selectDuration, anim::bumpy<125, 100>); | ||||
| 	} else { | ||||
| 		_selection.finish(); | ||||
| 	} | ||||
|  |  | |||
|  | @ -27,8 +27,7 @@ namespace Ui { | |||
| class RoundImageCheckbox { | ||||
| public: | ||||
| 	using PaintRoundImage = base::lambda_unique<void(Painter &p, int x, int y, int outerWidth, int size)>; | ||||
| 	using UpdateCallback = base::lambda_wrap<void()>; | ||||
| 	RoundImageCheckbox(const style::RoundImageCheckbox &st, UpdateCallback &&updateCallback, PaintRoundImage &&paintRoundImage); | ||||
| 	RoundImageCheckbox(const style::RoundImageCheckbox &st, base::lambda_wrap<void()> updateCallback, PaintRoundImage paintRoundImage); | ||||
| 
 | ||||
| 	void paint(Painter &p, int x, int y, int outerWidth); | ||||
| 	float64 checkedAnimationRatio() const; | ||||
|  | @ -53,7 +52,7 @@ private: | |||
| 	void prepareWideCheckIconCache(Icon *icon); | ||||
| 
 | ||||
| 	const style::RoundImageCheckbox &_st; | ||||
| 	UpdateCallback _updateCallback; | ||||
| 	base::lambda_wrap<void()> _updateCallback; | ||||
| 	PaintRoundImage _paintRoundImage; | ||||
| 
 | ||||
| 	bool _checked = false; | ||||
|  |  | |||
|  | @ -184,8 +184,8 @@ void FlatInput::paintEvent(QPaintEvent *e) { | |||
| 	p.drawRoundedRect(QRectF(0, 0, width(), height()).marginsRemoved(QMarginsF(_st.borderWidth / 2., _st.borderWidth / 2., _st.borderWidth / 2., _st.borderWidth / 2.)), st::buttonRadius - (_st.borderWidth / 2.), st::buttonRadius - (_st.borderWidth / 2.)); | ||||
| 	p.setRenderHint(QPainter::HighQualityAntialiasing, false); | ||||
| 
 | ||||
| 	if (_st.imgRect.pxWidth()) { | ||||
| 		p.drawSprite(_st.imgPos, _st.imgRect); | ||||
| 	if (!_st.icon.empty()) { | ||||
| 		_st.icon.paint(p, 0, 0, width()); | ||||
| 	} | ||||
| 
 | ||||
| 	bool phDraw = _phVisible; | ||||
|  | @ -683,10 +683,10 @@ void InputArea::checkContentHeight() { | |||
| 	} | ||||
| } | ||||
| 
 | ||||
| InputArea::InputAreaInner::InputAreaInner(InputArea *parent) : QTextEdit(parent) { | ||||
| InputArea::Inner::Inner(InputArea *parent) : QTextEdit(parent) { | ||||
| } | ||||
| 
 | ||||
| bool InputArea::InputAreaInner::viewportEvent(QEvent *e) { | ||||
| bool InputArea::Inner::viewportEvent(QEvent *e) { | ||||
| 	if (e->type() == QEvent::TouchBegin || e->type() == QEvent::TouchUpdate || e->type() == QEvent::TouchEnd || e->type() == QEvent::TouchCancel) { | ||||
| 		QTouchEvent *ev = static_cast<QTouchEvent*>(e); | ||||
| 		if (ev->device()->type() == QTouchDevice::TouchScreen) { | ||||
|  | @ -790,7 +790,7 @@ void InputArea::contextMenuEvent(QContextMenuEvent *e) { | |||
| 	_inner.contextMenuEvent(e); | ||||
| } | ||||
| 
 | ||||
| void InputArea::InputAreaInner::focusInEvent(QFocusEvent *e) { | ||||
| void InputArea::Inner::focusInEvent(QFocusEvent *e) { | ||||
| 	f()->focusInInner(); | ||||
| 	QTextEdit::focusInEvent(e); | ||||
| 	emit f()->focused(); | ||||
|  | @ -807,7 +807,7 @@ void InputArea::focusInInner() { | |||
| 	} | ||||
| } | ||||
| 
 | ||||
| void InputArea::InputAreaInner::focusOutEvent(QFocusEvent *e) { | ||||
| void InputArea::Inner::focusOutEvent(QFocusEvent *e) { | ||||
| 	f()->focusOutInner(); | ||||
| 	QTextEdit::focusOutEvent(e); | ||||
| 	emit f()->blurred(); | ||||
|  | @ -943,7 +943,7 @@ void InputArea::insertEmoji(EmojiPtr emoji, QTextCursor c) { | |||
| 	c.insertText(objectReplacement, imageFormat); | ||||
| } | ||||
| 
 | ||||
| QVariant InputArea::InputAreaInner::loadResource(int type, const QUrl &name) { | ||||
| QVariant InputArea::Inner::loadResource(int type, const QUrl &name) { | ||||
| 	QString imageName = name.toDisplayString(); | ||||
| 	if (imageName.startsWith(qstr("emoji://e."))) { | ||||
| 		if (EmojiPtr emoji = emojiFromUrl(imageName)) { | ||||
|  | @ -1193,7 +1193,7 @@ void InputArea::updatePlaceholder() { | |||
| 	} | ||||
| } | ||||
| 
 | ||||
| QMimeData *InputArea::InputAreaInner::createMimeDataFromSelection() const { | ||||
| QMimeData *InputArea::Inner::createMimeDataFromSelection() const { | ||||
| 	QMimeData *result = new QMimeData(); | ||||
| 	QTextCursor c(textCursor()); | ||||
| 	int32 start = c.selectionStart(), end = c.selectionEnd(); | ||||
|  | @ -1211,7 +1211,7 @@ void InputArea::setCtrlEnterSubmit(CtrlEnterSubmit ctrlEnterSubmit) { | |||
| 	_ctrlEnterSubmit = ctrlEnterSubmit; | ||||
| } | ||||
| 
 | ||||
| void InputArea::InputAreaInner::keyPressEvent(QKeyEvent *e) { | ||||
| void InputArea::Inner::keyPressEvent(QKeyEvent *e) { | ||||
| 	bool shift = e->modifiers().testFlag(Qt::ShiftModifier), alt = e->modifiers().testFlag(Qt::AltModifier); | ||||
| 	bool macmeta = (cPlatform() == dbipMac || cPlatform() == dbipMacOld) && e->modifiers().testFlag(Qt::ControlModifier) && !e->modifiers().testFlag(Qt::MetaModifier) && !e->modifiers().testFlag(Qt::AltModifier); | ||||
| 	bool ctrl = e->modifiers().testFlag(Qt::ControlModifier) || e->modifiers().testFlag(Qt::MetaModifier); | ||||
|  | @ -1276,11 +1276,11 @@ void InputArea::InputAreaInner::keyPressEvent(QKeyEvent *e) { | |||
| 	} | ||||
| } | ||||
| 
 | ||||
| void InputArea::InputAreaInner::paintEvent(QPaintEvent *e) { | ||||
| void InputArea::Inner::paintEvent(QPaintEvent *e) { | ||||
| 	return QTextEdit::paintEvent(e); | ||||
| } | ||||
| 
 | ||||
| void InputArea::InputAreaInner::contextMenuEvent(QContextMenuEvent *e) { | ||||
| void InputArea::Inner::contextMenuEvent(QContextMenuEvent *e) { | ||||
| 	if (QMenu *menu = createStandardContextMenu()) { | ||||
| 		(new PopupMenu(menu))->popup(e->globalPos()); | ||||
| 	} | ||||
|  | @ -1338,7 +1338,9 @@ InputField::InputField(QWidget *parent, const style::InputField &st, const QStri | |||
| 
 | ||||
| 	_inner.setWordWrapMode(QTextOption::NoWrap); | ||||
| 
 | ||||
| 	setAttribute(Qt::WA_OpaquePaintEvent); | ||||
| 	if (_st.textBg->c.alphaF() >= 1.) { | ||||
| 		setAttribute(Qt::WA_OpaquePaintEvent); | ||||
| 	} | ||||
| 
 | ||||
| 	_inner.setFont(_st.font->f); | ||||
| 	_inner.setAlignment(_st.textAlign); | ||||
|  | @ -1380,10 +1382,10 @@ void InputField::onTouchTimer() { | |||
| 	_touchRightButton = true; | ||||
| } | ||||
| 
 | ||||
| InputField::InputFieldInner::InputFieldInner(InputField *parent) : QTextEdit(parent) { | ||||
| InputField::Inner::Inner(InputField *parent) : QTextEdit(parent) { | ||||
| } | ||||
| 
 | ||||
| bool InputField::InputFieldInner::viewportEvent(QEvent *e) { | ||||
| bool InputField::Inner::viewportEvent(QEvent *e) { | ||||
| 	if (e->type() == QEvent::TouchBegin || e->type() == QEvent::TouchUpdate || e->type() == QEvent::TouchEnd || e->type() == QEvent::TouchCancel) { | ||||
| 		QTouchEvent *ev = static_cast<QTouchEvent*>(e); | ||||
| 		if (ev->device()->type() == QTouchDevice::TouchScreen) { | ||||
|  | @ -1437,7 +1439,9 @@ void InputField::paintEvent(QPaintEvent *e) { | |||
| 	Painter p(this); | ||||
| 
 | ||||
| 	QRect r(rect().intersected(e->rect())); | ||||
| 	p.fillRect(r, st::white->b); | ||||
| 	if (_st.textBg->c.alphaF() > 0.) { | ||||
| 		p.fillRect(r, _st.textBg); | ||||
| 	} | ||||
| 	if (_st.border) { | ||||
| 		p.fillRect(0, height() - _st.border, width(), _st.border, _st.borderFg->b); | ||||
| 	} | ||||
|  | @ -1446,9 +1450,6 @@ void InputField::paintEvent(QPaintEvent *e) { | |||
| 		p.fillRect(0, height() - _st.borderActive, width(), _st.borderActive, a_borderFg.current()); | ||||
| 		p.setOpacity(1); | ||||
| 	} | ||||
| 	if (_st.iconSprite.pxWidth()) { | ||||
| 		p.drawSpriteLeft(_st.iconPosition, width(), _st.iconSprite); | ||||
| 	} | ||||
| 
 | ||||
| 	bool drawPlaceholder = _placeholderVisible; | ||||
| 	if (_a_placeholderShift.animating()) { | ||||
|  | @ -1490,7 +1491,7 @@ void InputField::contextMenuEvent(QContextMenuEvent *e) { | |||
| 	_inner.contextMenuEvent(e); | ||||
| } | ||||
| 
 | ||||
| void InputField::InputFieldInner::focusInEvent(QFocusEvent *e) { | ||||
| void InputField::Inner::focusInEvent(QFocusEvent *e) { | ||||
| 	f()->focusInInner(); | ||||
| 	QTextEdit::focusInEvent(e); | ||||
| 	emit f()->focused(); | ||||
|  | @ -1507,7 +1508,7 @@ void InputField::focusInInner() { | |||
| 	} | ||||
| } | ||||
| 
 | ||||
| void InputField::InputFieldInner::focusOutEvent(QFocusEvent *e) { | ||||
| void InputField::Inner::focusOutEvent(QFocusEvent *e) { | ||||
| 	f()->focusOutInner(); | ||||
| 	QTextEdit::focusOutEvent(e); | ||||
| 	emit f()->blurred(); | ||||
|  | @ -1643,7 +1644,7 @@ void InputField::insertEmoji(EmojiPtr emoji, QTextCursor c) { | |||
| 	c.insertText(objectReplacement, imageFormat); | ||||
| } | ||||
| 
 | ||||
| QVariant InputField::InputFieldInner::loadResource(int type, const QUrl &name) { | ||||
| QVariant InputField::Inner::loadResource(int type, const QUrl &name) { | ||||
| 	QString imageName = name.toDisplayString(); | ||||
| 	if (imageName.startsWith(qstr("emoji://e."))) { | ||||
| 		if (EmojiPtr emoji = emojiFromUrl(imageName)) { | ||||
|  | @ -1928,7 +1929,7 @@ void InputField::setPlaceholderHidden(bool forcePlaceholderHidden) { | |||
| 	updatePlaceholder(); | ||||
| } | ||||
| 
 | ||||
| QMimeData *InputField::InputFieldInner::createMimeDataFromSelection() const { | ||||
| QMimeData *InputField::Inner::createMimeDataFromSelection() const { | ||||
| 	QMimeData *result = new QMimeData(); | ||||
| 	QTextCursor c(textCursor()); | ||||
| 	int32 start = c.selectionStart(), end = c.selectionEnd(); | ||||
|  | @ -1942,7 +1943,7 @@ void InputField::customUpDown(bool custom) { | |||
| 	_customUpDown = custom; | ||||
| } | ||||
| 
 | ||||
| void InputField::InputFieldInner::keyPressEvent(QKeyEvent *e) { | ||||
| void InputField::Inner::keyPressEvent(QKeyEvent *e) { | ||||
| 	bool shift = e->modifiers().testFlag(Qt::ShiftModifier), alt = e->modifiers().testFlag(Qt::AltModifier); | ||||
| 	bool macmeta = (cPlatform() == dbipMac || cPlatform() == dbipMacOld) && e->modifiers().testFlag(Qt::ControlModifier) && !e->modifiers().testFlag(Qt::MetaModifier) && !e->modifiers().testFlag(Qt::AltModifier); | ||||
| 	bool ctrl = e->modifiers().testFlag(Qt::ControlModifier) || e->modifiers().testFlag(Qt::MetaModifier), ctrlGood = true; | ||||
|  | @ -1979,36 +1980,41 @@ void InputField::InputFieldInner::keyPressEvent(QKeyEvent *e) { | |||
| 		} | ||||
| #endif // Q_OS_MAC
 | ||||
| 	} else { | ||||
| 		QTextCursor tc(textCursor()); | ||||
| 		auto oldCursorPosition = textCursor().position(); | ||||
| 		if (enter && ctrl) { | ||||
| 			e->setModifiers(e->modifiers() & ~Qt::ControlModifier); | ||||
| 		} | ||||
| 		QTextEdit::keyPressEvent(e); | ||||
| 		if (tc == textCursor()) { | ||||
| 		auto currentCursor = textCursor(); | ||||
| 		if (textCursor().position() == oldCursorPosition) { | ||||
| 			bool check = false; | ||||
| 			if (e->key() == Qt::Key_PageUp || e->key() == Qt::Key_Up) { | ||||
| 				tc.movePosition(QTextCursor::Start, e->modifiers().testFlag(Qt::ShiftModifier) ? QTextCursor::KeepAnchor : QTextCursor::MoveAnchor); | ||||
| 				oldCursorPosition = currentCursor.position(); | ||||
| 				currentCursor.movePosition(QTextCursor::Start, e->modifiers().testFlag(Qt::ShiftModifier) ? QTextCursor::KeepAnchor : QTextCursor::MoveAnchor); | ||||
| 				check = true; | ||||
| 			} else if (e->key() == Qt::Key_PageDown || e->key() == Qt::Key_Down) { | ||||
| 				tc.movePosition(QTextCursor::End, e->modifiers().testFlag(Qt::ShiftModifier) ? QTextCursor::KeepAnchor : QTextCursor::MoveAnchor); | ||||
| 				oldCursorPosition = currentCursor.position(); | ||||
| 				currentCursor.movePosition(QTextCursor::End, e->modifiers().testFlag(Qt::ShiftModifier) ? QTextCursor::KeepAnchor : QTextCursor::MoveAnchor); | ||||
| 				check = true; | ||||
| 			} else if (e->key() == Qt::Key_Left || e->key() == Qt::Key_Right || e->key() == Qt::Key_Backspace) { | ||||
| 				e->ignore(); | ||||
| 			} | ||||
| 			if (check) { | ||||
| 				if (tc == textCursor()) { | ||||
| 				if (oldCursorPosition == currentCursor.position()) { | ||||
| 					e->ignore(); | ||||
| 				} else { | ||||
| 					setTextCursor(tc); | ||||
| 					setTextCursor(currentCursor); | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| void InputField::InputFieldInner::paintEvent(QPaintEvent *e) { | ||||
| void InputField::Inner::paintEvent(QPaintEvent *e) { | ||||
| 	return QTextEdit::paintEvent(e); | ||||
| } | ||||
| 
 | ||||
| void InputField::InputFieldInner::contextMenuEvent(QContextMenuEvent *e) { | ||||
| void InputField::Inner::contextMenuEvent(QContextMenuEvent *e) { | ||||
| 	if (QMenu *menu = createStandardContextMenu()) { | ||||
| 		(new PopupMenu(menu))->popup(e->globalPos()); | ||||
| 	} | ||||
|  | @ -2167,9 +2173,6 @@ void MaskedInputField::paintEvent(QPaintEvent *e) { | |||
| 		p.fillRect(0, height() - _st.borderActive, width(), _st.borderActive, a_borderFg.current()); | ||||
| 		p.setOpacity(1); | ||||
| 	} | ||||
| 	if (_st.iconSprite.pxWidth()) { | ||||
| 		p.drawSpriteLeft(_st.iconPosition, width(), _st.iconSprite); | ||||
| 	} | ||||
| 
 | ||||
| 	p.setClipRect(r); | ||||
| 	paintPlaceholder(p); | ||||
|  |  | |||
|  | @ -262,10 +262,9 @@ private: | |||
| 	bool heightAutoupdated(); | ||||
| 	void checkContentHeight(); | ||||
| 
 | ||||
| 	friend class InputAreaInner; | ||||
| 	class InputAreaInner : public QTextEdit { | ||||
| 	class Inner : public QTextEdit { | ||||
| 	public: | ||||
| 		InputAreaInner(InputArea *parent); | ||||
| 		Inner(InputArea *parent); | ||||
| 
 | ||||
| 		QVariant loadResource(int type, const QUrl &name) override; | ||||
| 
 | ||||
|  | @ -286,6 +285,7 @@ private: | |||
| 		friend class InputArea; | ||||
| 
 | ||||
| 	}; | ||||
| 	friend class Inner; | ||||
| 
 | ||||
| 	void focusInInner(); | ||||
| 	void focusOutInner(); | ||||
|  | @ -294,7 +294,7 @@ private: | |||
| 
 | ||||
| 	void startBorderAnimation(); | ||||
| 
 | ||||
| 	InputAreaInner _inner; | ||||
| 	Inner _inner; | ||||
| 
 | ||||
| 	QString _oldtext; | ||||
| 
 | ||||
|  | @ -431,10 +431,9 @@ private: | |||
| 	int32 _maxLength; | ||||
| 	bool _forcePlaceholderHidden = false; | ||||
| 
 | ||||
| 	friend class InputFieldInner; | ||||
| 	class InputFieldInner : public QTextEdit { | ||||
| 	class Inner : public QTextEdit { | ||||
| 	public: | ||||
| 		InputFieldInner(InputField *parent); | ||||
| 		Inner(InputField *parent); | ||||
| 
 | ||||
| 		QVariant loadResource(int type, const QUrl &name) override; | ||||
| 
 | ||||
|  | @ -455,6 +454,7 @@ private: | |||
| 		friend class InputField; | ||||
| 
 | ||||
| 	}; | ||||
| 	friend class Inner; | ||||
| 
 | ||||
| 	void focusInInner(); | ||||
| 	void focusOutInner(); | ||||
|  | @ -463,7 +463,7 @@ private: | |||
| 
 | ||||
| 	void startBorderAnimation(); | ||||
| 
 | ||||
| 	InputFieldInner _inner; | ||||
| 	Inner _inner; | ||||
| 
 | ||||
| 	QString _oldtext; | ||||
| 
 | ||||
|  |  | |||
|  | @ -105,6 +105,9 @@ public: | |||
| 		return std_::make_unique<Icon>(ColoredCopy { *this, colors }); | ||||
| 	} | ||||
| 
 | ||||
| 	bool empty() const { | ||||
| 		return _parts.empty(); | ||||
| 	} | ||||
| 	void paint(QPainter &p, const QPoint &pos, int outerw) const; | ||||
| 	void paint(QPainter &p, int x, int y, int outerw) const { | ||||
| 		paint(p, QPoint(x, y), outerw); | ||||
|  |  | |||
|  | @ -26,91 +26,476 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org | |||
| #include "lang.h" | ||||
| 
 | ||||
| namespace Ui { | ||||
| namespace { | ||||
| 
 | ||||
| constexpr int kWideScale = 3; | ||||
| 
 | ||||
| } // namespace
 | ||||
| 
 | ||||
| class MultiSelect::Inner::Item { | ||||
| public: | ||||
| 	Item(const style::MultiSelectItem &st, uint64 id, const QString &text, const style::color &color, PaintRoundImage paintRoundImage); | ||||
| 
 | ||||
| 	uint64 id() const { | ||||
| 		return _id; | ||||
| 	} | ||||
| 	int getWidth() const { | ||||
| 		return _width; | ||||
| 	} | ||||
| 	QRect rect() const { | ||||
| 		return QRect(_x, _y, _width, _st.height); | ||||
| 	} | ||||
| 	bool isOverDelete() const { | ||||
| 		return _overDelete; | ||||
| 	} | ||||
| 	void setActive(bool active) { | ||||
| 		_active = active; | ||||
| 	} | ||||
| 	void setPosition(int x, int y, int outerWidth, int maxVisiblePadding); | ||||
| 	QRect paintArea(int outerWidth) const; | ||||
| 
 | ||||
| 	void setUpdateCallback(base::lambda_wrap<void()> updateCallback) { | ||||
| 		_updateCallback = std_::move(updateCallback); | ||||
| 	} | ||||
| 	void setText(const QString &text); | ||||
| 	void paint(Painter &p, int outerWidth, uint64 ms); | ||||
| 
 | ||||
| 	void mouseMoveEvent(QPoint point); | ||||
| 	void leaveEvent(); | ||||
| 
 | ||||
| 	void showAnimated() { | ||||
| 		setVisibleAnimated(true); | ||||
| 	} | ||||
| 	void hideAnimated() { | ||||
| 		setVisibleAnimated(false); | ||||
| 	} | ||||
| 	bool hideFinished() const { | ||||
| 		return (_hiding && !_visibility.animating()); | ||||
| 	} | ||||
| 
 | ||||
| 
 | ||||
| private: | ||||
| 	void setOver(bool over); | ||||
| 	void paintOnce(Painter &p, int x, int y, int outerWidth, uint64 ms); | ||||
| 	void paintDeleteButton(Painter &p, int x, int y, int outerWidth, float64 overOpacity); | ||||
| 	bool paintCached(Painter &p, int x, int y, int outerWidth); | ||||
| 	void prepareCache(); | ||||
| 	void setVisibleAnimated(bool visible); | ||||
| 
 | ||||
| 	const style::MultiSelectItem &_st; | ||||
| 
 | ||||
| 	uint64 _id; | ||||
| 	struct SlideAnimation { | ||||
| 		SlideAnimation(base::lambda_wrap<void()> updateCallback, int fromX, int toX, int y, float64 duration) | ||||
| 			: fromX(fromX) | ||||
| 			, toX(toX) | ||||
| 			, y(y) { | ||||
| 			x.start(std_::move(updateCallback), fromX, toX, duration); | ||||
| 		} | ||||
| 		IntAnimation x; | ||||
| 		int fromX, toX; | ||||
| 		int y; | ||||
| 	}; | ||||
| 	std_::vector_of_moveable<SlideAnimation> _copies; | ||||
| 	int _x = -1; | ||||
| 	int _y = -1; | ||||
| 	int _width = 0; | ||||
| 	Text _text; | ||||
| 	const style::color &_color; | ||||
| 	bool _over = false; | ||||
| 	QPixmap _cache; | ||||
| 	FloatAnimation _visibility; | ||||
| 	FloatAnimation _overOpacity; | ||||
| 	bool _overDelete = false; | ||||
| 	bool _active = false; | ||||
| 	PaintRoundImage _paintRoundImage; | ||||
| 	base::lambda_wrap<void()> _updateCallback; | ||||
| 	bool _hiding = false; | ||||
| 
 | ||||
| }; | ||||
| 
 | ||||
| MultiSelect::Inner::Item::Item(const style::MultiSelectItem &st, uint64 id, const QString &text, const style::color &color, PaintRoundImage paintRoundImage) | ||||
| : _st(st) | ||||
| , _id(id) | ||||
| , _color(color) | ||||
| , _paintRoundImage(std_::move(paintRoundImage)) { | ||||
| 	setText(text); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::Item::setText(const QString &text) { | ||||
| 	_text.setText(_st.font, text, _textNameOptions); | ||||
| 	_width = _st.height + _st.padding.left() + _text.maxWidth() + _st.padding.right(); | ||||
| 	accumulate_min(_width, _st.maxWidth); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::Item::paint(Painter &p, int outerWidth, uint64 ms) { | ||||
| 	if (!_cache.isNull() && !_visibility.animating(ms)) { | ||||
| 		if (_hiding) { | ||||
| 			return; | ||||
| 		} else { | ||||
| 			_cache = QPixmap(); | ||||
| 		} | ||||
| 	} | ||||
| 	if (_copies.empty()) { | ||||
| 		paintOnce(p, _x, _y, outerWidth, ms); | ||||
| 	} else { | ||||
| 		for (auto i = _copies.begin(), e = _copies.end(); i != e;) { | ||||
| 			auto x = i->x.current(getms(), _x); | ||||
| 			auto y = i->y; | ||||
| 			auto animating = i->x.animating(); | ||||
| 			if (animating || (y == _y)) { | ||||
| 				paintOnce(p, x, y, outerWidth, ms); | ||||
| 			} | ||||
| 			if (animating) { | ||||
| 				++i; | ||||
| 			} else { | ||||
| 				i = _copies.erase(i); | ||||
| 				e = _copies.end(); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::Item::paintOnce(Painter &p, int x, int y, int outerWidth, uint64 ms) { | ||||
| 	if (!_cache.isNull()) { | ||||
| 		paintCached(p, x, y, outerWidth); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	auto radius = _st.height / 2; | ||||
| 	auto inner = rtlrect(x + radius, y, _width - radius, _st.height, outerWidth); | ||||
| 
 | ||||
| 	auto clipEnabled = p.hasClipping(); | ||||
| 	auto clip = clipEnabled ? p.clipRegion() : QRegion(); | ||||
| 	p.setRenderHint(QPainter::HighQualityAntialiasing); | ||||
| 	p.setClipRect(inner); | ||||
| 
 | ||||
| 	p.setPen(Qt::NoPen); | ||||
| 	p.setBrush(_active ? _st.textActiveBg : _st.textBg); | ||||
| 	p.drawRoundedRect(rtlrect(x, y, _width, _st.height, outerWidth), radius, radius); | ||||
| 
 | ||||
| 	if (clipEnabled) { | ||||
| 		p.setClipRegion(clip); | ||||
| 	} else { | ||||
| 		p.setClipping(false); | ||||
| 	} | ||||
| 	p.setRenderHint(QPainter::HighQualityAntialiasing, false); | ||||
| 
 | ||||
| 	auto overOpacity = _overOpacity.current(ms, _over ? 1. : 0.); | ||||
| 	if (overOpacity < 1.) { | ||||
| 		_paintRoundImage(p, x, y, outerWidth, _st.height); | ||||
| 	} | ||||
| 	if (overOpacity > 0.) { | ||||
| 		paintDeleteButton(p, x, y, outerWidth, overOpacity); | ||||
| 	} | ||||
| 
 | ||||
| 	auto textLeft = _st.height + _st.padding.left(); | ||||
| 	auto textWidth = _width - textLeft - _st.padding.right(); | ||||
| 	p.setPen(_active ? _st.textActiveFg : _st.textFg); | ||||
| 	_text.drawLeftElided(p, x + textLeft, y + _st.padding.top(), textWidth, outerWidth); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::Item::paintDeleteButton(Painter &p, int x, int y, int outerWidth, float64 overOpacity) { | ||||
| 	p.setOpacity(overOpacity); | ||||
| 	p.setRenderHint(QPainter::HighQualityAntialiasing); | ||||
| 	p.setPen(Qt::NoPen); | ||||
| 	p.setBrush(_color); | ||||
| 	p.drawEllipse(rtlrect(x, y, _st.height, _st.height, outerWidth)); | ||||
| 
 | ||||
| 	auto deleteScale = overOpacity + _st.minScale * (1. - overOpacity); | ||||
| 	auto deleteSkip = deleteScale * _st.deleteLeft + (1. - deleteScale) * (_st.height / 2); | ||||
| 	auto sqrt2 = sqrt(2.); | ||||
| 	auto deleteLeft = rtlpoint(x + deleteSkip, 0, outerWidth).x() + 0.; | ||||
| 	auto deleteTop = y + deleteSkip + 0.; | ||||
| 	auto deleteWidth = _st.height - 2 * deleteSkip; | ||||
| 	auto deleteHeight = _st.height - 2 * deleteSkip; | ||||
| 	auto deleteStroke = _st.deleteStroke / sqrt2; | ||||
| 	QPointF pathDelete[] = { | ||||
| 		{ deleteLeft, deleteTop + deleteStroke }, | ||||
| 		{ deleteLeft + deleteStroke, deleteTop }, | ||||
| 		{ deleteLeft + (deleteWidth / 2.), deleteTop + (deleteHeight / 2.) - deleteStroke }, | ||||
| 		{ deleteLeft + deleteWidth - deleteStroke, deleteTop }, | ||||
| 		{ deleteLeft + deleteWidth, deleteTop + deleteStroke }, | ||||
| 		{ deleteLeft + (deleteWidth / 2.) + deleteStroke, deleteTop + (deleteHeight / 2.) }, | ||||
| 		{ deleteLeft + deleteWidth, deleteTop + deleteHeight - deleteStroke }, | ||||
| 		{ deleteLeft + deleteWidth - deleteStroke, deleteTop + deleteHeight }, | ||||
| 		{ deleteLeft + (deleteWidth / 2.), deleteTop + (deleteHeight / 2.) + deleteStroke }, | ||||
| 		{ deleteLeft + deleteStroke, deleteTop + deleteHeight }, | ||||
| 		{ deleteLeft, deleteTop + deleteHeight - deleteStroke }, | ||||
| 		{ deleteLeft + (deleteWidth / 2.) - deleteStroke, deleteTop + (deleteHeight / 2.) }, | ||||
| 	}; | ||||
| 	if (overOpacity < 1.) { | ||||
| 		auto alpha = -(overOpacity - 1.) * M_PI_2; | ||||
| 		auto cosalpha = cos(alpha); | ||||
| 		auto sinalpha = sin(alpha); | ||||
| 		auto shiftx = deleteLeft + (deleteWidth / 2.); | ||||
| 		auto shifty = deleteTop + (deleteHeight / 2.); | ||||
| 		for (auto &point : pathDelete) { | ||||
| 			auto x = point.x() - shiftx; | ||||
| 			auto y = point.y() - shifty; | ||||
| 			point.setX(shiftx + x * cosalpha - y * sinalpha); | ||||
| 			point.setY(shifty + y * cosalpha + x * sinalpha); | ||||
| 		} | ||||
| 	} | ||||
| 	QPainterPath path; | ||||
| 	path.moveTo(pathDelete[0]); | ||||
| 	for (int i = 1; i != base::array_size(pathDelete); ++i) { | ||||
| 		path.lineTo(pathDelete[i]); | ||||
| 	} | ||||
| 	p.fillPath(path, _st.deleteFg); | ||||
| 
 | ||||
| 	p.setRenderHint(QPainter::HighQualityAntialiasing, false); | ||||
| 	p.setOpacity(1.); | ||||
| } | ||||
| 
 | ||||
| bool MultiSelect::Inner::Item::paintCached(Painter &p, int x, int y, int outerWidth) { | ||||
| 	auto opacity = _visibility.current(_hiding ? 0. : 1.); | ||||
| 	auto scale = opacity + _st.minScale * (1. - opacity); | ||||
| 	auto height = opacity * _cache.height() / _cache.devicePixelRatio(); | ||||
| 	auto width = opacity * _cache.width() / _cache.devicePixelRatio(); | ||||
| 
 | ||||
| 	p.setOpacity(opacity); | ||||
| 	p.setRenderHint(QPainter::SmoothPixmapTransform, true); | ||||
| 	p.drawPixmap(rtlrect(x + (_width - width) / 2., y + (_st.height - height) / 2., width, height, outerWidth), _cache); | ||||
| 	p.setRenderHint(QPainter::SmoothPixmapTransform, false); | ||||
| 	p.setOpacity(1.); | ||||
| 	return true; | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::Item::mouseMoveEvent(QPoint point) { | ||||
| 	if (!_cache.isNull()) return; | ||||
| 	_overDelete = QRect(0, 0, _st.height, _st.height).contains(point); | ||||
| 	setOver(true); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::Item::leaveEvent() { | ||||
| 	_overDelete = false; | ||||
| 	setOver(false); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::Item::setPosition(int x, int y, int outerWidth, int maxVisiblePadding) { | ||||
| 	if (_x >= 0 && _y >= 0 && (_x != x || _y != y)) { | ||||
| 		// Make an animation if it is not the first setPosition().
 | ||||
| 		auto found = false; | ||||
| 		auto leftHidden = -_width - maxVisiblePadding; | ||||
| 		auto rightHidden = outerWidth + maxVisiblePadding; | ||||
| 		for (auto i = _copies.begin(), e = _copies.end(); i != e;) { | ||||
| 			if (i->x.animating()) { | ||||
| 				if (i->y == y) { | ||||
| 					i->x.start(_updateCallback, i->toX, x, _st.duration); | ||||
| 					found = true; | ||||
| 				} else { | ||||
| 					i->x.start(_updateCallback, i->fromX, (i->toX > i->fromX) ? rightHidden : leftHidden, _st.duration); | ||||
| 				} | ||||
| 				++i; | ||||
| 			} else { | ||||
| 				i = _copies.erase(i); | ||||
| 				e = _copies.end(); | ||||
| 			} | ||||
| 		} | ||||
| 		if (_copies.empty()) { | ||||
| 			if (_y == y) { | ||||
| 				auto copy = SlideAnimation(_updateCallback, _x, x, _y, _st.duration); | ||||
| 				_copies.push_back(std_::move(copy)); | ||||
| 			} else { | ||||
| 				auto copyHiding = SlideAnimation(_updateCallback, _x, (y > _y) ? rightHidden : leftHidden, _y, _st.duration); | ||||
| 				_copies.push_back(std_::move(copyHiding)); | ||||
| 				auto copyShowing = SlideAnimation(_updateCallback, (y > _y) ? leftHidden : rightHidden, x, y, _st.duration); | ||||
| 				_copies.push_back(std_::move(copyShowing)); | ||||
| 			} | ||||
| 		} else if (!found) { | ||||
| 			auto copy = SlideAnimation(_updateCallback, (y > _y) ? leftHidden : rightHidden, x, y, _st.duration); | ||||
| 			_copies.push_back(std_::move(copy)); | ||||
| 		} | ||||
| 	} | ||||
| 	_x = x; | ||||
| 	_y = y; | ||||
| } | ||||
| 
 | ||||
| QRect MultiSelect::Inner::Item::paintArea(int outerWidth) const { | ||||
| 	if (_copies.empty()) { | ||||
| 		return rect(); | ||||
| 	} | ||||
| 	auto yMin = 0, yMax = 0; | ||||
| 	for_const (auto ©, _copies) { | ||||
| 		accumulate_max(yMax, copy.y); | ||||
| 		if (yMin) { | ||||
| 			accumulate_min(yMin, copy.y); | ||||
| 		} else { | ||||
| 			yMin = copy.y; | ||||
| 		} | ||||
| 	} | ||||
| 	return QRect(0, yMin, outerWidth, yMax - yMin + _st.height); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::Item::prepareCache() { | ||||
| 	if (!_cache.isNull()) return; | ||||
| 
 | ||||
| 	t_assert(!_visibility.animating()); | ||||
| 	auto cacheWidth = _width * kWideScale * cIntRetinaFactor(); | ||||
| 	auto cacheHeight = _st.height * kWideScale * cIntRetinaFactor(); | ||||
| 	auto data = QImage(cacheWidth, cacheHeight, QImage::Format_ARGB32_Premultiplied); | ||||
| 	data.fill(Qt::transparent); | ||||
| 	data.setDevicePixelRatio(cRetinaFactor()); | ||||
| 	{ | ||||
| 		Painter p(&data); | ||||
| 		paintOnce(p, _width * (kWideScale - 1) / 2, _st.height  * (kWideScale - 1) / 2, cacheWidth, getms()); | ||||
| 	} | ||||
| 	_cache = App::pixmapFromImageInPlace(std_::move(data)); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::Item::setVisibleAnimated(bool visible) { | ||||
| 	_hiding = !visible; | ||||
| 	prepareCache(); | ||||
| 	auto from = visible ? 0. : 1.; | ||||
| 	auto to = visible ? 1. : 0.; | ||||
| 	auto transition = visible ? anim::bumpy<1125, 1000> : anim::linear; | ||||
| 	_visibility.start(_updateCallback, from, to, _st.duration, transition); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::Item::setOver(bool over) { | ||||
| 	if (over != _over) { | ||||
| 		_over = over; | ||||
| 		_overOpacity.start(_updateCallback, _over ? 0. : 1., _over ? 1. : 0., _st.duration); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| MultiSelect::MultiSelect(QWidget *parent, const style::MultiSelect &st, const QString &placeholder) : TWidget(parent) | ||||
| , _st(st) | ||||
| , _scroll(this, st::boxScroll) | ||||
| , _inner(this, st, placeholder) { | ||||
| , _scroll(this, _st.scroll) | ||||
| , _inner(this, st, placeholder, [this](int activeTop, int activeBottom) { scrollTo(activeTop, activeBottom); }) { | ||||
| 	_scroll->setOwnedWidget(_inner); | ||||
| 	_scroll->installEventFilter(this); | ||||
| 	_inner->setResizedCallback([this](int innerHeightDelta) { | ||||
| 		auto newHeight = resizeGetHeight(width()); | ||||
| 		if (innerHeightDelta > 0) { | ||||
| 			_scroll->scrollToY(_scroll->scrollTop() + innerHeightDelta); | ||||
| 		} | ||||
| 		if (newHeight != height()) { | ||||
| 			resize(width(), newHeight); | ||||
| 			if (_resizedCallback) { | ||||
| 				_resizedCallback(); | ||||
| 			} | ||||
| 		} | ||||
| 	}); | ||||
| 	_inner->setQueryChangedCallback([this](const QString &query) { | ||||
| 		_scroll->scrollToY(_scroll->scrollTopMax()); | ||||
| 		if (_queryChangedCallback) { | ||||
| 			_queryChangedCallback(query); | ||||
| 		} | ||||
| 	}); | ||||
| 
 | ||||
| 	setAttribute(Qt::WA_OpaquePaintEvent); | ||||
| } | ||||
| 
 | ||||
| bool MultiSelect::eventFilter(QObject *o, QEvent *e) { | ||||
| 	if (o == _scroll && e->type() == QEvent::KeyPress) { | ||||
| 		e->ignore(); | ||||
| 		return true; | ||||
| 	} | ||||
| 	return false; | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::scrollTo(int activeTop, int activeBottom) { | ||||
| 	auto scrollTop = _scroll->scrollTop(); | ||||
| 	auto scrollHeight = _scroll->height(); | ||||
| 	auto scrollBottom = scrollTop + scrollHeight; | ||||
| 	if (scrollTop > activeTop) { | ||||
| 		_scroll->scrollToY(activeTop); | ||||
| 	} else if (scrollBottom < activeBottom) { | ||||
| 		_scroll->scrollToY(activeBottom - scrollHeight); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::setQueryChangedCallback(base::lambda_unique<void(const QString &query)> callback) { | ||||
| 	_inner->setQueryChangedCallback(std_::move(callback)); | ||||
| 	_queryChangedCallback = std_::move(callback); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::setSubmittedCallback(base::lambda_unique<void(bool ctrlShiftEnter)> callback) { | ||||
| 	_inner->setSubmittedCallback(std_::move(callback)); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::setResizedCallback(base::lambda_unique<void()> callback) { | ||||
| 	_resizedCallback = std_::move(callback); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::setInnerFocus() { | ||||
| 	if (_inner->setInnerFocus()) { | ||||
| 		_scroll->scrollToY(_scroll->scrollTopMax()); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::clearQuery() { | ||||
| 	_inner->clearQuery(); | ||||
| } | ||||
| 
 | ||||
| QString MultiSelect::getQuery() const { | ||||
| 	return _inner->getQuery(); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::addItem(std_::unique_ptr<Item> item) { | ||||
| 	_inner->addItem(std_::move(item)); | ||||
| void MultiSelect::addItem(uint64 itemId, const QString &text, const style::color &color, PaintRoundImage paintRoundImage) { | ||||
| 	_inner->addItem(std_::make_unique<Inner::Item>(_st.item, itemId, text, color, std_::move(paintRoundImage))); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::setItemRemovedCallback(base::lambda_unique<void(uint64 itemId)> callback) { | ||||
| 	_inner->setItemRemovedCallback(std_::move(callback)); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::removeItem(uint64 itemId) { | ||||
| 	_inner->removeItem(itemId); | ||||
| } | ||||
| 
 | ||||
| int MultiSelect::resizeGetHeight(int newWidth) { | ||||
| 	_inner->resizeToWidth(newWidth); | ||||
| 	if (newWidth != _inner->width()) { | ||||
| 		_inner->resizeToWidth(newWidth); | ||||
| 	} | ||||
| 	auto newHeight = qMin(_inner->height(), _st.maxHeight); | ||||
| 	_scroll->resize(newWidth, newHeight); | ||||
| 	_scroll->setGeometryToLeft(0, 0, newWidth, newHeight); | ||||
| 	return newHeight; | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::resizeEvent(QResizeEvent *e) { | ||||
| 	_scroll->moveToLeft(0, 0); | ||||
| } | ||||
| 
 | ||||
| MultiSelect::Item::Item(uint64 id, const QString &text, const style::color &color) | ||||
| : _id(id) { | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Item::setText(const QString &text) { | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Item::paint(Painter &p, int x, int y) { | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| MultiSelect::Inner::Inner(QWidget *parent, const style::MultiSelect &st, const QString &placeholder) : ScrolledWidget(parent) | ||||
| MultiSelect::Inner::Inner(QWidget *parent, const style::MultiSelect &st, const QString &placeholder, ScrollCallback callback) : ScrolledWidget(parent) | ||||
| , _st(st) | ||||
| , _filter(this, _st.field, placeholder) | ||||
| , _cancel(this, _st.cancel) { | ||||
| 	connect(_filter, SIGNAL(changed()), this, SLOT(onQueryChanged())); | ||||
| 	connect(_filter, SIGNAL(submitted(bool)), this, SLOT(onSubmitted(bool))); | ||||
| , _scrollCallback(std_::move(callback)) | ||||
| , _field(this, _st.field, placeholder) | ||||
| , _cancel(this, _st.fieldCancel) { | ||||
| 	_field->customUpDown(true); | ||||
| 	connect(_field, SIGNAL(focused()), this, SLOT(onFieldFocused())); | ||||
| 	connect(_field, SIGNAL(changed()), this, SLOT(onQueryChanged())); | ||||
| 	connect(_field, SIGNAL(submitted(bool)), this, SLOT(onSubmitted(bool))); | ||||
| 	_cancel->hide(); | ||||
| 	_cancel->setClickedCallback([this] { | ||||
| 		_filter->setText(QString()); | ||||
| 		_filter->setFocus(); | ||||
| 		clearQuery(); | ||||
| 		_field->setFocus(); | ||||
| 	}); | ||||
| 	setMouseTracking(true); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::onQueryChanged() { | ||||
| 	auto query = getQuery(); | ||||
| 	_cancel->setVisible(!query.isEmpty()); | ||||
| 	updateFieldGeometry(); | ||||
| 	if (_queryChangedCallback) { | ||||
| 		_queryChangedCallback(query); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| QString MultiSelect::Inner::getQuery() const { | ||||
| 	return _field->getLastText().trimmed(); | ||||
| } | ||||
| 
 | ||||
| bool MultiSelect::Inner::setInnerFocus() { | ||||
| 	if (!_filter->hasFocus()) { | ||||
| 		_filter->setFocus(); | ||||
| 	if (_active >= 0) { | ||||
| 		setFocus(); | ||||
| 	} else if (!_field->hasFocus()) { | ||||
| 		_field->setFocus(); | ||||
| 		return true; | ||||
| 	} | ||||
| 	return false; | ||||
| } | ||||
| 
 | ||||
| QString MultiSelect::Inner::getQuery() const { | ||||
| 	return _filter->getLastText().trimmed(); | ||||
| void MultiSelect::Inner::clearQuery() { | ||||
| 	_field->setText(QString()); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::setQueryChangedCallback(base::lambda_unique<void(const QString &query)> callback) { | ||||
|  | @ -121,41 +506,275 @@ void MultiSelect::Inner::setSubmittedCallback(base::lambda_unique<void(bool ctrl | |||
| 	_submittedCallback = std_::move(callback); | ||||
| } | ||||
| 
 | ||||
| int MultiSelect::Inner::resizeGetHeight(int newWidth) { | ||||
| 	_filter->resizeToWidth(newWidth); | ||||
| 	return _filter->height(); | ||||
| void MultiSelect::Inner::updateFieldGeometry() { | ||||
| 	auto fieldFinalWidth = _fieldWidth; | ||||
| 	if (!_cancel->isHidden()) { | ||||
| 		fieldFinalWidth -= _st.fieldCancelSkip; | ||||
| 	} | ||||
| 	_field->resizeToWidth(fieldFinalWidth); | ||||
| 	_field->moveToLeft(_st.padding.left() + _fieldLeft, _st.padding.top() + _fieldTop); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::resizeEvent(QResizeEvent *e) { | ||||
| 	_filter->moveToLeft(0, 0); | ||||
| 	_cancel->moveToRight(0, 0); | ||||
| void MultiSelect::Inner::updateHasAnyItems(bool hasAnyItems) { | ||||
| 	_field->setPlaceholderHidden(hasAnyItems); | ||||
| 	updateCursor(); | ||||
| 	_iconOpacity.start([this] { | ||||
| 		rtlupdate(_st.padding.left(), _st.padding.top(), _st.fieldIcon.width(), _st.fieldIcon.height()); | ||||
| 	}, hasAnyItems ? 1. : 0., hasAnyItems ? 0. : 1., _st.item.duration); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::updateCursor() { | ||||
| 	setCursor(_items.empty() ? style::cur_text : (_overDelete ? style::cur_pointer : style::cur_default)); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::setActiveItem(int active, ChangeActiveWay skipSetFocus) { | ||||
| 	if (_active == active) return; | ||||
| 
 | ||||
| 	if (_active >= 0) { | ||||
| 		t_assert(_active < _items.size()); | ||||
| 		_items[_active]->setActive(false); | ||||
| 	} | ||||
| 	_active = active; | ||||
| 	if (_active >= 0) { | ||||
| 		t_assert(_active < _items.size()); | ||||
| 		_items[_active]->setActive(true); | ||||
| 	} | ||||
| 	if (skipSetFocus != ChangeActiveWay::SkipSetFocus) { | ||||
| 		setInnerFocus(); | ||||
| 	} | ||||
| 	if (_scrollCallback) { | ||||
| 		auto rect = (_active >= 0) ? _items[_active]->rect() : _field->geometry().translated(-_st.padding.left(), -_st.padding.top()); | ||||
| 		_scrollCallback(rect.y(), rect.y() + rect.height() + _st.padding.top() + _st.padding.bottom()); | ||||
| 	} | ||||
| 	update(); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::setActiveItemPrevious() { | ||||
| 	if (_active > 0) { | ||||
| 		setActiveItem(_active - 1); | ||||
| 	} else if (_active < 0 && !_items.empty()) { | ||||
| 		setActiveItem(_items.size() - 1); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::setActiveItemNext() { | ||||
| 	if (_active >= 0 && _active + 1 < _items.size()) { | ||||
| 		setActiveItem(_active + 1); | ||||
| 	} else { | ||||
| 		setActiveItem(-1); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| int MultiSelect::Inner::resizeGetHeight(int newWidth) { | ||||
| 	computeItemsGeometry(newWidth); | ||||
| 	updateFieldGeometry(); | ||||
| 
 | ||||
| 	auto cancelLeft = _fieldLeft + _fieldWidth + _st.padding.right() - _cancel->width(); | ||||
| 	auto cancelTop = _fieldTop - _st.padding.top(); | ||||
| 	_cancel->moveToLeft(_st.padding.left() + cancelLeft, _st.padding.top() + cancelTop); | ||||
| 
 | ||||
| 	return _field->y() + _field->height() + _st.padding.bottom(); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::paintEvent(QPaintEvent *e) { | ||||
| 	Painter p(this); | ||||
| 
 | ||||
| 	auto paintRect = e->rect(); | ||||
| 	p.fillRect(paintRect, st::windowBg); | ||||
| 
 | ||||
| 	auto offset = QPoint(rtl() ? _st.padding.right() : _st.padding.left(), _st.padding.top()); | ||||
| 	p.translate(offset); | ||||
| 	paintRect.translate(-offset); | ||||
| 
 | ||||
| 	auto ms = getms(); | ||||
| 	auto outerWidth = width() - _st.padding.left() - _st.padding.right(); | ||||
| 	auto iconOpacity = _iconOpacity.current(ms, _items.empty() ? 1. : 0.); | ||||
| 	if (iconOpacity > 0.) { | ||||
| 		p.setOpacity(iconOpacity); | ||||
| 		_st.fieldIcon.paint(p, 0, 0, outerWidth); | ||||
| 		p.setOpacity(1.); | ||||
| 	} | ||||
| 
 | ||||
| 	auto checkRect = myrtlrect(paintRect); | ||||
| 	auto paintMargins = itemPaintMargins(); | ||||
| 	for (auto i = _removingItems.begin(), e = _removingItems.end(); i != e;) { | ||||
| 		auto item = *i; | ||||
| 		auto itemRect = item->paintArea(outerWidth); | ||||
| 		itemRect = itemRect.marginsAdded(paintMargins); | ||||
| 		if (checkRect.intersects(itemRect)) { | ||||
| 			item->paint(p, outerWidth, ms); | ||||
| 		} | ||||
| 		if (item->hideFinished()) { | ||||
| 			i = _removingItems.erase(i); | ||||
| 			e = _removingItems.end(); | ||||
| 		} else { | ||||
| 			++i; | ||||
| 		} | ||||
| 	} | ||||
| 	for_const (auto item, _items) { | ||||
| 		auto itemRect = item->paintArea(outerWidth); | ||||
| 		itemRect = itemRect.marginsAdded(paintMargins); | ||||
| 		if (checkRect.y() + checkRect.height() <= itemRect.y()) { | ||||
| 			break; | ||||
| 		} else if (checkRect.intersects(itemRect)) { | ||||
| 			item->paint(p, outerWidth, ms); | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| QMargins MultiSelect::Inner::itemPaintMargins() const { | ||||
| 	return { | ||||
| 		qMax(_st.itemSkip, _st.padding.left()), | ||||
| 		_st.itemSkip, | ||||
| 		qMax(_st.itemSkip, _st.padding.right()), | ||||
| 		_st.itemSkip, | ||||
| 	}; | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::leaveEvent(QEvent *e) { | ||||
| 	clearSelection(); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::mouseMoveEvent(QMouseEvent *e) { | ||||
| 	updateSelection(e->pos()); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::keyPressEvent(QKeyEvent *e) { | ||||
| 	if (_active >= 0) { | ||||
| 		t_assert(_active < _items.size()); | ||||
| 		if (e->key() == Qt::Key_Delete || e->key() == Qt::Key_Backspace) { | ||||
| 			auto itemId = _items[_active]->id(); | ||||
| 			setActiveItemNext(); | ||||
| 			removeItem(itemId); | ||||
| 		} else if (e->key() == Qt::Key_Left) { | ||||
| 			setActiveItemPrevious(); | ||||
| 		} else if (e->key() == Qt::Key_Right) { | ||||
| 			setActiveItemNext(); | ||||
| 		} else if (e->key() == Qt::Key_Escape) { | ||||
| 			setActiveItem(-1); | ||||
| 		} else { | ||||
| 			e->ignore(); | ||||
| 		} | ||||
| 	} else if (e->key() == Qt::Key_Left || e->key() == Qt::Key_Backspace) { | ||||
| 		setActiveItemPrevious(); | ||||
| 	} else { | ||||
| 		e->ignore(); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::onFieldFocused() { | ||||
| 	setActiveItem(-1, ChangeActiveWay::SkipSetFocus); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::updateSelection(QPoint mousePosition) { | ||||
| 	auto point = myrtlpoint(mousePosition) - QPoint(_st.padding.left(), _st.padding.right()); | ||||
| 	auto selected = -1; | ||||
| 	for (auto i = 0, size = _items.size(); i != size; ++i) { | ||||
| 		auto itemRect = _items[i]->rect(); | ||||
| 		if (itemRect.y() > point.y()) { | ||||
| 			break; | ||||
| 		} else if (itemRect.contains(point)) { | ||||
| 			point -= itemRect.topLeft(); | ||||
| 			selected = i; | ||||
| 			break; | ||||
| 		} | ||||
| 	} | ||||
| 	if (_selected != selected) { | ||||
| 		if (_selected >= 0) { | ||||
| 			t_assert(_selected < _items.size()); | ||||
| 			_items[_selected]->leaveEvent(); | ||||
| 		} | ||||
| 		_selected = selected; | ||||
| 		update(); | ||||
| 	} | ||||
| 	auto overDelete = false; | ||||
| 	if (_selected >= 0) { | ||||
| 		_items[_selected]->mouseMoveEvent(point); | ||||
| 		overDelete = _items[_selected]->isOverDelete(); | ||||
| 	} | ||||
| 	if (_overDelete != overDelete) { | ||||
| 		_overDelete = overDelete; | ||||
| 		updateCursor(); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::mousePressEvent(QMouseEvent *e) { | ||||
| 	if (_overDelete) { | ||||
| 		t_assert(_selected >= 0); | ||||
| 		t_assert(_selected < _items.size()); | ||||
| 		removeItem(_items[_selected]->id()); | ||||
| 	} else if (_selected >= 0) { | ||||
| 		setActiveItem(_selected); | ||||
| 	} else { | ||||
| 		setInnerFocus(); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::addItem(std_::unique_ptr<Item> item) { | ||||
| 	auto wasEmpty = _items.empty(); | ||||
| 	item->setUpdateCallback([this, item = item.get()] { | ||||
| 		auto itemRect = item->paintArea(width() - _st.padding.left() - _st.padding.top()); | ||||
| 		itemRect = itemRect.translated(_st.padding.left(), _st.padding.top()); | ||||
| 		itemRect = itemRect.marginsAdded(itemPaintMargins()); | ||||
| 		rtlupdate(itemRect); | ||||
| 	}); | ||||
| 	_items.push_back(item.release()); | ||||
| 	refreshItemsGeometry(nullptr); | ||||
| 	updateItemsGeometry(); | ||||
| 	if (wasEmpty) { | ||||
| 		updateHasAnyItems(true); | ||||
| 	} | ||||
| 	_items.back()->showAnimated(); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::refreshItemsGeometry(Item *startingFromRowWithItem) { | ||||
| 	int startingFromRow = 0; | ||||
| 	int startingFromIndex = 0; | ||||
| 	for (int row = 1, rowsCount = qMin(_rows.size(), 1); row != rowsCount; ++row) { | ||||
| 		if (startingFromRowWithItem) { | ||||
| 			if (_rows[row - 1].contains(startingFromRowWithItem)) { | ||||
| 				break; | ||||
| 			} | ||||
| void MultiSelect::Inner::computeItemsGeometry(int newWidth) { | ||||
| 	newWidth -= _st.padding.left() + _st.padding.right(); | ||||
| 
 | ||||
| 	auto itemLeft = 0; | ||||
| 	auto itemTop = 0; | ||||
| 	auto widthLeft = newWidth; | ||||
| 	auto maxVisiblePadding = qMax(_st.padding.left(), _st.padding.right()); | ||||
| 	for_const (auto item, _items) { | ||||
| 		auto itemWidth = item->getWidth(); | ||||
| 		t_assert(itemWidth <= newWidth); | ||||
| 		if (itemWidth > widthLeft) { | ||||
| 			itemLeft = 0; | ||||
| 			itemTop += _st.item.height + _st.itemSkip; | ||||
| 			widthLeft = newWidth; | ||||
| 		} | ||||
| 		startingFromIndex += _rows[row - 1].size(); | ||||
| 		++startingFromRow; | ||||
| 		item->setPosition(itemLeft, itemTop, newWidth, maxVisiblePadding); | ||||
| 		itemLeft += itemWidth + _st.itemSkip; | ||||
| 		widthLeft -= itemWidth + _st.itemSkip; | ||||
| 	} | ||||
| 	while (_rows.size() > startingFromRow) { | ||||
| 		_rows.pop_back(); | ||||
| 	} | ||||
| 	for (int i = startingFromIndex, count = _items.size(); i != count; ++i) { | ||||
| 		Row row; | ||||
| 		row.append(_items[i]); | ||||
| 		_rows.append(row); | ||||
| 
 | ||||
| 	auto fieldMinWidth = _st.fieldMinWidth + _st.fieldCancelSkip; | ||||
| 	t_assert(fieldMinWidth <= newWidth); | ||||
| 	if (fieldMinWidth > widthLeft) { | ||||
| 		_fieldLeft = 0; | ||||
| 		_fieldTop = itemTop + _st.item.height + _st.itemSkip; | ||||
| 	} else { | ||||
| 		_fieldLeft = itemLeft + (_items.empty() ? _st.fieldIconSkip : 0); | ||||
| 		_fieldTop = itemTop; | ||||
| 	} | ||||
| 	_fieldWidth = newWidth - _fieldLeft; | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::updateItemsGeometry() { | ||||
| 	computeItemsGeometry(width()); | ||||
| 	updateFieldGeometry(); | ||||
| 	auto newHeight = resizeGetHeight(width()); | ||||
| 	if (newHeight == _newHeight) return; | ||||
| 
 | ||||
| 	_newHeight = newHeight; | ||||
| 	_height.start([this] { | ||||
| 		auto newHeight = _height.current(_newHeight); | ||||
| 		if (auto heightDelta = newHeight - height()) { | ||||
| 			resize(width(), newHeight); | ||||
| 			if (_resizedCallback) { | ||||
| 				_resizedCallback(heightDelta); | ||||
| 			} | ||||
| 			update(); | ||||
| 		} | ||||
| 	}, height(), _newHeight, _st.item.duration); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::setItemText(uint64 itemId, const QString &text) { | ||||
|  | @ -163,7 +782,7 @@ void MultiSelect::Inner::setItemText(uint64 itemId, const QString &text) { | |||
| 		auto item = _items[i]; | ||||
| 		if (item->id() == itemId) { | ||||
| 			item->setText(text); | ||||
| 			refreshItemsGeometry(item); | ||||
| 			updateItemsGeometry(); | ||||
| 			return; | ||||
| 		} | ||||
| 	} | ||||
|  | @ -173,26 +792,50 @@ void MultiSelect::Inner::setItemRemovedCallback(base::lambda_unique<void(uint64 | |||
| 	_itemRemovedCallback = std_::move(callback); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::setResizedCallback(base::lambda_unique<void(int heightDelta)> callback) { | ||||
| 	_resizedCallback = std_::move(callback); | ||||
| } | ||||
| 
 | ||||
| void MultiSelect::Inner::removeItem(uint64 itemId) { | ||||
| 	for (int i = 0, count = _items.size(); i != count; ++i) { | ||||
| 		auto item = _items[i]; | ||||
| 		if (item->id() == itemId) { | ||||
| 			clearSelection(); | ||||
| 			_items.removeAt(i); | ||||
| 			refreshItemsGeometry(item); | ||||
| 			delete item; | ||||
| 			if (_active == i) { | ||||
| 				_active = -1; | ||||
| 			} else if (_active > i) { | ||||
| 				--_active; | ||||
| 			} | ||||
| 			_removingItems.insert(item); | ||||
| 			item->hideAnimated(); | ||||
| 
 | ||||
| 			updateItemsGeometry(); | ||||
| 			if (_items.empty()) { | ||||
| 				updateHasAnyItems(false); | ||||
| 			} | ||||
| 			auto point = QCursor::pos(); | ||||
| 			if (auto parent = parentWidget()) { | ||||
| 				if (parent->rect().contains(parent->mapFromGlobal(point))) { | ||||
| 					updateSelection(mapFromGlobal(point)); | ||||
| 				} | ||||
| 			} | ||||
| 			break; | ||||
| 		} | ||||
| 	} | ||||
| 	if (_itemRemovedCallback) { | ||||
| 		_itemRemovedCallback(itemId); | ||||
| 	} | ||||
| 	setInnerFocus(); | ||||
| } | ||||
| 
 | ||||
| MultiSelect::Inner::~Inner() { | ||||
| 	base::take(_rows); | ||||
| 	for (auto item : base::take(_items)) { | ||||
| 		delete item; | ||||
| 	} | ||||
| 	for (auto item : base::take(_removingItems)) { | ||||
| 		delete item; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| } // namespace Ui
 | ||||
|  |  | |||
|  | @ -34,6 +34,49 @@ public: | |||
| 
 | ||||
| 	QString getQuery() const; | ||||
| 	void setInnerFocus(); | ||||
| 	void clearQuery(); | ||||
| 
 | ||||
| 	void setQueryChangedCallback(base::lambda_unique<void(const QString &query)> callback); | ||||
| 	void setSubmittedCallback(base::lambda_unique<void(bool ctrlShiftEnter)> callback); | ||||
| 	void setResizedCallback(base::lambda_unique<void()> callback); | ||||
| 
 | ||||
| 	using PaintRoundImage = base::lambda_unique<void(Painter &p, int x, int y, int outerWidth, int size)>; | ||||
| 	void addItem(uint64 itemId, const QString &text, const style::color &color, PaintRoundImage paintRoundImage); | ||||
| 	void setItemText(uint64 itemId, const QString &text); | ||||
| 
 | ||||
| 	void setItemRemovedCallback(base::lambda_unique<void(uint64 itemId)> callback); | ||||
| 	void removeItem(uint64 itemId); | ||||
| 
 | ||||
| protected: | ||||
| 	int resizeGetHeight(int newWidth) override; | ||||
| 	bool eventFilter(QObject *o, QEvent *e) override; | ||||
| 
 | ||||
| private: | ||||
| 	void scrollTo(int activeTop, int activeBottom); | ||||
| 
 | ||||
| 	const style::MultiSelect &_st; | ||||
| 
 | ||||
| 	ChildWidget<ScrollArea> _scroll; | ||||
| 
 | ||||
| 	class Inner; | ||||
| 	ChildWidget<Inner> _inner; | ||||
| 
 | ||||
| 	base::lambda_unique<void()> _resizedCallback; | ||||
| 	base::lambda_unique<void(const QString &query)> _queryChangedCallback; | ||||
| 
 | ||||
| }; | ||||
| 
 | ||||
| // This class is hold in header because it requires Qt preprocessing.
 | ||||
| class MultiSelect::Inner : public ScrolledWidget { | ||||
| 	Q_OBJECT | ||||
| 
 | ||||
| public: | ||||
| 	using ScrollCallback = base::lambda_unique<void(int activeTop, int activeBottom)>; | ||||
| 	Inner(QWidget *parent, const style::MultiSelect &st, const QString &placeholder, ScrollCallback callback); | ||||
| 
 | ||||
| 	QString getQuery() const; | ||||
| 	bool setInnerFocus(); | ||||
| 	void clearQuery(); | ||||
| 
 | ||||
| 	void setQueryChangedCallback(base::lambda_unique<void(const QString &query)> callback); | ||||
| 	void setSubmittedCallback(base::lambda_unique<void(bool ctrlShiftEnter)> callback); | ||||
|  | @ -43,68 +86,20 @@ public: | |||
| 	void setItemText(uint64 itemId, const QString &text); | ||||
| 
 | ||||
| 	void setItemRemovedCallback(base::lambda_unique<void(uint64 itemId)> callback); | ||||
| 	void removeItem(uint64 itemId); // Always calls the itemRemovedCallback().
 | ||||
| 	void removeItem(uint64 itemId); | ||||
| 
 | ||||
| protected: | ||||
| 	int resizeGetHeight(int newWidth) override; | ||||
| 
 | ||||
| 	void resizeEvent(QResizeEvent *e) override; | ||||
| 
 | ||||
| private: | ||||
| 	ChildWidget<ScrollArea> _scroll; | ||||
| 
 | ||||
| 	class Inner; | ||||
| 	ChildWidget<Inner> _inner; | ||||
| 
 | ||||
| 	const style::MultiSelect &_st; | ||||
| 
 | ||||
| }; | ||||
| 
 | ||||
| class MultiSelect::Item { | ||||
| public: | ||||
| 	Item(uint64 id, const QString &text, const style::color &color); | ||||
| 
 | ||||
| 	uint64 id() const { | ||||
| 		return _id; | ||||
| 	} | ||||
| 	void setText(const QString &text); | ||||
| 	void paint(Painter &p, int x, int y); | ||||
| 
 | ||||
| 	virtual ~Item() = default; | ||||
| 
 | ||||
| protected: | ||||
| 	virtual void paintImage(Painter &p, int x, int y, int outerWidth, int size) = 0; | ||||
| 
 | ||||
| private: | ||||
| 	uint64 _id; | ||||
| 
 | ||||
| }; | ||||
| 
 | ||||
| // This class is hold in header because it requires Qt preprocessing.
 | ||||
| class MultiSelect::Inner : public ScrolledWidget { | ||||
| 	Q_OBJECT | ||||
| 
 | ||||
| public: | ||||
| 	Inner(QWidget *parent, const style::MultiSelect &st, const QString &placeholder); | ||||
| 
 | ||||
| 	QString getQuery() const; | ||||
| 	bool setInnerFocus(); | ||||
| 
 | ||||
| 	void setQueryChangedCallback(base::lambda_unique<void(const QString &query)> callback); | ||||
| 	void setSubmittedCallback(base::lambda_unique<void(bool ctrlShiftEnter)> callback); | ||||
| 
 | ||||
| 	void addItem(std_::unique_ptr<Item> item); | ||||
| 	void setItemText(uint64 itemId, const QString &text); | ||||
| 
 | ||||
| 	void setItemRemovedCallback(base::lambda_unique<void(uint64 itemId)> callback); | ||||
| 	void removeItem(uint64 itemId); // Always calls the itemRemovedCallback().
 | ||||
| 	void setResizedCallback(base::lambda_unique<void(int heightDelta)> callback); | ||||
| 
 | ||||
| 	~Inner(); | ||||
| 
 | ||||
| protected: | ||||
| 	int resizeGetHeight(int newWidth) override; | ||||
| 
 | ||||
| 	void resizeEvent(QResizeEvent *e) override; | ||||
| 	void paintEvent(QPaintEvent *e) override; | ||||
| 	void leaveEvent(QEvent *e) override; | ||||
| 	void mouseMoveEvent(QMouseEvent *e) override; | ||||
| 	void mousePressEvent(QMouseEvent *e) override; | ||||
| 	void keyPressEvent(QKeyEvent *e) override; | ||||
| 
 | ||||
| private slots: | ||||
| 	void onQueryChanged(); | ||||
|  | @ -113,25 +108,55 @@ private slots: | |||
| 			_submittedCallback(ctrlShiftEnter); | ||||
| 		} | ||||
| 	} | ||||
| 	void onFieldFocused(); | ||||
| 
 | ||||
| private: | ||||
| 	void refreshItemsGeometry(Item *startingFromItem); | ||||
| 	void computeItemsGeometry(int newWidth); | ||||
| 	void updateItemsGeometry(); | ||||
| 	void updateFieldGeometry(); | ||||
| 	void updateHasAnyItems(bool hasAnyItems); | ||||
| 	void updateSelection(QPoint mousePosition); | ||||
| 	void clearSelection() { | ||||
| 		updateSelection(QPoint(-1, -1)); | ||||
| 	} | ||||
| 	void updateCursor(); | ||||
| 	enum class ChangeActiveWay { | ||||
| 		Default, | ||||
| 		SkipSetFocus, | ||||
| 	}; | ||||
| 	void setActiveItem(int active, ChangeActiveWay skipSetFocus = ChangeActiveWay::Default); | ||||
| 	void setActiveItemPrevious(); | ||||
| 	void setActiveItemNext(); | ||||
| 
 | ||||
| 	QMargins itemPaintMargins() const; | ||||
| 
 | ||||
| 	const style::MultiSelect &_st; | ||||
| 	FloatAnimation _iconOpacity; | ||||
| 
 | ||||
| 	using Row = QList<Item*>; | ||||
| 	using Rows = QList<Row>; | ||||
| 	Rows _rows; | ||||
| 	ScrollCallback _scrollCallback; | ||||
| 
 | ||||
| 	using Items = QList<Item*>; | ||||
| 	Items _items; | ||||
| 	using RemovingItems = OrderedSet<Item*>; | ||||
| 	RemovingItems _removingItems; | ||||
| 
 | ||||
| 	ChildWidget<InputField> _filter; | ||||
| 	int _selected = -1; | ||||
| 	int _active = -1; | ||||
| 	bool _overDelete = false; | ||||
| 
 | ||||
| 	int _fieldLeft = 0; | ||||
| 	int _fieldTop = 0; | ||||
| 	int _fieldWidth = 0; | ||||
| 	ChildWidget<InputField> _field; | ||||
| 	ChildWidget<Ui::IconButton> _cancel; | ||||
| 
 | ||||
| 	int _newHeight = 0; | ||||
| 	IntAnimation _height; | ||||
| 
 | ||||
| 	base::lambda_unique<void(const QString &query)> _queryChangedCallback; | ||||
| 	base::lambda_unique<void(bool ctrlShiftEnter)> _submittedCallback; | ||||
| 	base::lambda_unique<void(uint64 itemId)> _itemRemovedCallback; | ||||
| 	base::lambda_unique<void(int heightDelta)> _resizedCallback; | ||||
| 
 | ||||
| }; | ||||
| 
 | ||||
|  |  | |||
|  | @ -67,10 +67,36 @@ RoundImageCheckbox { | |||
| 	checkIcon: icon; | ||||
| } | ||||
| 
 | ||||
| MultiSelectItem { | ||||
| 	padding: margins; | ||||
| 	maxWidth: pixels; | ||||
| 	height: pixels; | ||||
| 	font: font; | ||||
| 	textBg: color; | ||||
| 	textFg: color; | ||||
| 	textActiveBg: color; | ||||
| 	textActiveFg: color; | ||||
| 	deleteFg: color; | ||||
| 	deleteLeft: pixels; | ||||
| 	deleteStroke: pixels; | ||||
| 	duration: int; | ||||
| 	minScale: double; | ||||
| } | ||||
| 
 | ||||
| MultiSelect { | ||||
| 	field: InputField; | ||||
| 	cancel: IconButton; | ||||
| 	padding: margins; | ||||
| 	maxHeight: pixels; | ||||
| 	scroll: flatScroll; | ||||
| 
 | ||||
| 	item: MultiSelectItem; | ||||
| 	itemSkip: pixels; | ||||
| 
 | ||||
| 	field: InputField; | ||||
| 	fieldIcon: icon; | ||||
| 	fieldIconSkip: pixels; | ||||
| 	fieldCancel: IconButton; | ||||
| 	fieldCancelSkip: pixels; | ||||
| 	fieldMinWidth: pixels; | ||||
| } | ||||
| 
 | ||||
| widgetSlideDuration: 200; | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue