mirror of https://github.com/procxx/kepka.git
				
				
				
			some thread work fixes, no contacts msgs added, username info msgs added
This commit is contained in:
		
							parent
							
								
									9e1d5da5a1
								
							
						
					
					
						commit
						8b743ae751
					
				|  | @ -134,6 +134,9 @@ lng_dlg_conversations: "Conversations"; | ||||||
| lng_dlg_messages: "Messages"; | lng_dlg_messages: "Messages"; | ||||||
| lng_dlg_new_group_name: "Group name"; | lng_dlg_new_group_name: "Group name"; | ||||||
| lng_dlg_create_group: "Create"; | lng_dlg_create_group: "Create"; | ||||||
|  | lng_no_contacts: "You have no contacts"; | ||||||
|  | lng_contacts_loading: "Loading.."; | ||||||
|  | lng_contacts_not_found: "No contacts found"; | ||||||
| 
 | 
 | ||||||
| lng_settings_profile: "Profile"; | lng_settings_profile: "Profile"; | ||||||
| lng_settings_edit: "Edit"; | lng_settings_edit: "Edit"; | ||||||
|  | @ -155,6 +158,8 @@ Minimum length is 5 characters."; | ||||||
| lng_username_invalid: "This name is invalid."; | lng_username_invalid: "This name is invalid."; | ||||||
| lng_username_occupied: "This name is already occupied."; | lng_username_occupied: "This name is already occupied."; | ||||||
| lng_username_too_short: "This name is too short."; | lng_username_too_short: "This name is too short."; | ||||||
|  | lng_username_bad_symbols: "This name has bad symbols."; | ||||||
|  | lng_username_available: "This name is available."; | ||||||
| 
 | 
 | ||||||
| lng_settings_section_contact_info: "Contact info"; | lng_settings_section_contact_info: "Contact info"; | ||||||
| lng_settings_phone_number: "Phone number:"; | lng_settings_phone_number: "Phone number:"; | ||||||
|  |  | ||||||
|  | @ -535,6 +535,7 @@ setErrBG: #ffa5a5; | ||||||
| setErrColor: #800000; | setErrColor: #800000; | ||||||
| setErrHeight: 30px; | setErrHeight: 30px; | ||||||
| setErrFont: font(fsize); | setErrFont: font(fsize); | ||||||
|  | setGoodColor: #008000; | ||||||
| 
 | 
 | ||||||
| btnSetUpload: flatButton(btnDefNext, btnDefBig) { | btnSetUpload: flatButton(btnDefNext, btnDefBig) { | ||||||
| 	width: 206px; | 	width: 206px; | ||||||
|  | @ -598,22 +599,9 @@ dlgPaddingVer: 8px; | ||||||
| dlgHeight: 62px; | dlgHeight: 62px; | ||||||
| dlgPhotoPadding: 12px; | dlgPhotoPadding: 12px; | ||||||
| 
 | 
 | ||||||
| dlgState: switcher { | noContactsHeight: 100px; | ||||||
| 	border: 2px; | noContactsFont: font(fsize); | ||||||
| 	borderColor: btnNextBG; | noContactsColor: #777; | ||||||
| 
 |  | ||||||
| 	bgColor: #fff; |  | ||||||
| 	bgHovered: btnWhiteHover; |  | ||||||
| 	bgActive: btnNextBG; |  | ||||||
| 
 |  | ||||||
| 	height: 34px; |  | ||||||
| 
 |  | ||||||
| 	font: font(fsize); |  | ||||||
| 	textColor: btnYesColor; |  | ||||||
| 	activeColor: #fff; |  | ||||||
| 
 |  | ||||||
| 	duration: 200; |  | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| dlgSep: 8px; | dlgSep: 8px; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 9.9 KiB | 
|  | @ -23,19 +23,26 @@ Copyright (c) 2014 John Preston, https://tdesktop.com | ||||||
| #include "window.h" | #include "window.h" | ||||||
| 
 | 
 | ||||||
| AddParticipantInner::AddParticipantInner(ChatData *chat) : _chat(chat), | AddParticipantInner::AddParticipantInner(ChatData *chat) : _chat(chat), | ||||||
| 	_contacts(&App::main()->contactsList()), _sel(0), _filteredSel(-1), _mouseSel(false), _selCount(0) { | _contacts(&App::main()->contactsList()), | ||||||
| 	 | _sel(0), | ||||||
| 	_filter = qsl("a"); | _filteredSel(-1), | ||||||
| 	updateFilter(); | _mouseSel(false), | ||||||
|  | _selCount(0), | ||||||
|  | _addContactLnk(this, lang(lng_add_contact_button)) { | ||||||
|  | 
 | ||||||
|  | 	connect(&_addContactLnk, SIGNAL(clicked()), App::wnd(), SLOT(onShowAddContact())); | ||||||
| 
 | 
 | ||||||
| 	for (DialogRow *r = _contacts->list.begin; r != _contacts->list.end; r = r->next) { | 	for (DialogRow *r = _contacts->list.begin; r != _contacts->list.end; r = r->next) { | ||||||
| 		r->attached = 0; | 		r->attached = 0; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	connect(App::main(), SIGNAL(dialogRowReplaced(DialogRow *, DialogRow *)), this, SLOT(onDialogRowReplaced(DialogRow *, DialogRow *))); | 	_filter = qsl("a"); | ||||||
| 	connect(App::main(), SIGNAL(peerUpdated(PeerData*)), this, SLOT(peerUpdated(PeerData *))); | 	updateFilter(); | ||||||
| 	connect(App::main(), SIGNAL(peerNameChanged(PeerData *, const PeerData::Names &, const PeerData::NameFirstChars &)), this, SLOT(peerUpdated(PeerData *))); | 
 | ||||||
| 	connect(App::main(), SIGNAL(peerPhotoChanged(PeerData *)), this, SLOT(peerUpdated(PeerData *))); | 	connect(App::main(), SIGNAL(dialogRowReplaced(DialogRow*,DialogRow*)), this, SLOT(onDialogRowReplaced(DialogRow*,DialogRow*))); | ||||||
|  | 	connect(App::main(), SIGNAL(peerUpdated(PeerData*)), this, SLOT(peerUpdated(PeerData*))); | ||||||
|  | 	connect(App::main(), SIGNAL(peerNameChanged(PeerData*,const PeerData::Names&,const PeerData::NameFirstChars&)), this, SLOT(peerUpdated(PeerData*))); | ||||||
|  | 	connect(App::main(), SIGNAL(peerPhotoChanged(PeerData*)), this, SLOT(peerUpdated(PeerData*))); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void AddParticipantInner::peerUpdated(PeerData *peer) { | void AddParticipantInner::peerUpdated(PeerData *peer) { | ||||||
|  | @ -184,11 +191,15 @@ void AddParticipantInner::paintEvent(QPaintEvent *e) { | ||||||
| 				drawFrom = drawFrom->next; | 				drawFrom = drawFrom->next; | ||||||
| 			} | 			} | ||||||
| 		} else { | 		} else { | ||||||
| 			// ..
 | 			p.setFont(st::noContactsFont->f); | ||||||
|  | 			p.setPen(st::noContactsColor->p); | ||||||
|  | 			p.drawText(QRect(0, 0, width(), st::noContactsHeight - (cContactsReceived() ? st::noContactsFont->height : 0)), lang(cContactsReceived() ? lng_no_contacts : lng_contacts_loading), style::al_center); | ||||||
| 		} | 		} | ||||||
| 	} else { | 	} else { | ||||||
| 		if (_filtered.isEmpty()) { | 		if (_filtered.isEmpty()) { | ||||||
| 			// ..
 | 			p.setFont(st::noContactsFont->f); | ||||||
|  | 			p.setPen(st::noContactsColor->p); | ||||||
|  | 			p.drawText(QRect(0, 0, width(), st::noContactsHeight), lang(lng_contacts_not_found), style::al_center); | ||||||
| 		} else { | 		} else { | ||||||
| 			int32 from = yFrom / rh; | 			int32 from = yFrom / rh; | ||||||
| 			if (from < 0) from = 0; | 			if (from < 0) from = 0; | ||||||
|  | @ -312,14 +323,23 @@ void AddParticipantInner::updateFilter(QString filter) { | ||||||
| 		int32 rh = (st::profileListPhotoSize + st::profileListPadding.height() * 2); | 		int32 rh = (st::profileListPhotoSize + st::profileListPadding.height() * 2); | ||||||
| 		_filter = filter; | 		_filter = filter; | ||||||
| 		if (_filter.isEmpty()) { | 		if (_filter.isEmpty()) { | ||||||
| 			resize(width(), _contacts->list.count * rh); |  | ||||||
| 			if (_contacts->list.count) { | 			if (_contacts->list.count) { | ||||||
|  | 				if (!_addContactLnk.isHidden()) _addContactLnk.hide(); | ||||||
|  | 				resize(width(), _contacts->list.count * rh); | ||||||
| 				_sel = _contacts->list.begin; | 				_sel = _contacts->list.begin; | ||||||
| 				while (_sel->next->next &&& contactData(_sel)->inchat) { | 				while (_sel->next->next &&& contactData(_sel)->inchat) { | ||||||
| 					_sel = _sel->next; | 					_sel = _sel->next; | ||||||
| 				} | 				} | ||||||
|  | 			} else { | ||||||
|  | 				resize(width(), st::noContactsHeight); | ||||||
|  | 				if (cContactsReceived()) { | ||||||
|  | 					if (_addContactLnk.isHidden()) _addContactLnk.show(); | ||||||
|  | 				} else { | ||||||
|  | 					if (!_addContactLnk.isHidden()) _addContactLnk.hide(); | ||||||
|  | 				} | ||||||
| 			} | 			} | ||||||
| 		} else { | 		} else { | ||||||
|  | 			if (!_addContactLnk.isHidden()) _addContactLnk.hide(); | ||||||
| 			QStringList::const_iterator fb = f.cbegin(), fe = f.cend(), fi; | 			QStringList::const_iterator fb = f.cbegin(), fe = f.cend(), fi; | ||||||
| 
 | 
 | ||||||
| 			_filtered.clear(); | 			_filtered.clear(); | ||||||
|  | @ -365,7 +385,11 @@ void AddParticipantInner::updateFilter(QString filter) { | ||||||
| 				++_filteredSel; | 				++_filteredSel; | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			resize(width(), _filtered.size() * rh); | 			if (!_filtered.isEmpty()) { | ||||||
|  | 				resize(width(), _filtered.size() * rh); | ||||||
|  | 			} else { | ||||||
|  | 				resize(width(), st::noContactsHeight); | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
| 		if (parentWidget()) parentWidget()->update(); | 		if (parentWidget()) parentWidget()->update(); | ||||||
| 		loadProfilePhotos(0); | 		loadProfilePhotos(0); | ||||||
|  | @ -406,6 +430,10 @@ AddParticipantInner::~AddParticipantInner() { | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void AddParticipantInner::resizeEvent(QResizeEvent *e) { | ||||||
|  | 	_addContactLnk.move((width() - _addContactLnk.width()) / 2, (st::noContactsHeight + st::noContactsFont->height) / 2); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void AddParticipantInner::selectSkip(int32 dir) { | void AddParticipantInner::selectSkip(int32 dir) { | ||||||
| 	_time = unixtime(); | 	_time = unixtime(); | ||||||
| 	_mouseSel = false; | 	_mouseSel = false; | ||||||
|  | @ -520,7 +548,7 @@ AddParticipantBox::AddParticipantBox(ChatData *chat) : | ||||||
| 	connect(&_scroll, SIGNAL(scrolled()), &_inner, SLOT(updateSel())); | 	connect(&_scroll, SIGNAL(scrolled()), &_inner, SLOT(updateSel())); | ||||||
| 	connect(&_scroll, SIGNAL(scrolled()), this, SLOT(onScroll())); | 	connect(&_scroll, SIGNAL(scrolled()), this, SLOT(onScroll())); | ||||||
| 	connect(&_filter, SIGNAL(changed()), this, SLOT(onFilterUpdate())); | 	connect(&_filter, SIGNAL(changed()), this, SLOT(onFilterUpdate())); | ||||||
| 	connect(&_filter, SIGNAL(cancelled()), this, SIGNAL(onClose())); | 	connect(&_filter, SIGNAL(cancelled()), this, SLOT(onClose())); | ||||||
| 	connect(&_inner, SIGNAL(mustScrollTo(int,int)), &_scroll, SLOT(scrollToY(int,int))); | 	connect(&_inner, SIGNAL(mustScrollTo(int,int)), &_scroll, SLOT(scrollToY(int,int))); | ||||||
| 	connect(&_inner, SIGNAL(selectAllQuery()), &_filter, SLOT(selectAll())); | 	connect(&_inner, SIGNAL(selectAllQuery()), &_filter, SLOT(selectAll())); | ||||||
| 
 | 
 | ||||||
|  | @ -578,6 +606,7 @@ void AddParticipantBox::paintEvent(QPaintEvent *e) { | ||||||
| 
 | 
 | ||||||
| 			// paint shadows
 | 			// paint shadows
 | ||||||
| 			p.fillRect(0, st::participantFilter.height, _width, st::scrollDef.topsh, st::scrollDef.shColor->b); | 			p.fillRect(0, st::participantFilter.height, _width, st::scrollDef.topsh, st::scrollDef.shColor->b); | ||||||
|  | 			p.fillRect(0, size().height() - st::btnSelectCancel.height - st::scrollDef.bottomsh, _width, st::scrollDef.bottomsh, st::scrollDef.shColor->b); | ||||||
| 
 | 
 | ||||||
| 			// paint button sep
 | 			// paint button sep
 | ||||||
| 			p.fillRect(st::btnSelectCancel.width, size().height() - st::btnSelectCancel.height, st::lineWidth, st::btnSelectCancel.height, st::btnSelectSep->b); | 			p.fillRect(st::btnSelectCancel.width, size().height() - st::btnSelectCancel.height, st::lineWidth, st::btnSelectCancel.height, st::btnSelectSep->b); | ||||||
|  |  | ||||||
|  | @ -31,6 +31,7 @@ public: | ||||||
| 	void leaveEvent(QEvent *e); | 	void leaveEvent(QEvent *e); | ||||||
| 	void mouseMoveEvent(QMouseEvent *e); | 	void mouseMoveEvent(QMouseEvent *e); | ||||||
| 	void mousePressEvent(QMouseEvent *e); | 	void mousePressEvent(QMouseEvent *e); | ||||||
|  | 	void resizeEvent(QResizeEvent *e); | ||||||
| 	 | 	 | ||||||
| 	void paintDialog(QPainter &p, DialogRow *row, bool sel); | 	void paintDialog(QPainter &p, DialogRow *row, bool sel); | ||||||
| 	void updateFilter(QString filter = QString()); | 	void updateFilter(QString filter = QString()); | ||||||
|  | @ -87,6 +88,7 @@ private: | ||||||
| 	ContactData *contactData(DialogRow *row); | 	ContactData *contactData(DialogRow *row); | ||||||
| 
 | 
 | ||||||
| 	QPoint _lastMousePos; | 	QPoint _lastMousePos; | ||||||
|  | 	LinkButton _addContactLnk; | ||||||
| 
 | 
 | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -23,14 +23,22 @@ Copyright (c) 2014 John Preston, https://tdesktop.com | ||||||
| #include "mainwidget.h" | #include "mainwidget.h" | ||||||
| #include "window.h" | #include "window.h" | ||||||
| 
 | 
 | ||||||
| ContactsInner::ContactsInner() : _contacts(&App::main()->contactsList()), _sel(0), _filteredSel(-1), _mouseSel(false) { | ContactsInner::ContactsInner() : | ||||||
| 	_filter = qsl("a"); | _contacts(&App::main()->contactsList()), | ||||||
| 	updateFilter(); | _sel(0), | ||||||
|  | _filteredSel(-1), | ||||||
|  | _mouseSel(false), | ||||||
|  | _addContactLnk(this, lang(lng_add_contact_button)) { | ||||||
|  | 
 | ||||||
|  | 	connect(&_addContactLnk, SIGNAL(clicked()), App::wnd(), SLOT(onShowAddContact())); | ||||||
| 
 | 
 | ||||||
| 	for (DialogRow *r = _contacts->list.begin; r != _contacts->list.end; r = r->next) { | 	for (DialogRow *r = _contacts->list.begin; r != _contacts->list.end; r = r->next) { | ||||||
| 		r->attached = 0; | 		r->attached = 0; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	_filter = qsl("a"); | ||||||
|  | 	updateFilter(); | ||||||
|  | 
 | ||||||
| 	connect(App::main(), SIGNAL(dialogRowReplaced(DialogRow *, DialogRow *)), this, SLOT(onDialogRowReplaced(DialogRow *, DialogRow *))); | 	connect(App::main(), SIGNAL(dialogRowReplaced(DialogRow *, DialogRow *)), this, SLOT(onDialogRowReplaced(DialogRow *, DialogRow *))); | ||||||
| 	connect(App::main(), SIGNAL(peerUpdated(PeerData*)), this, SLOT(peerUpdated(PeerData *))); | 	connect(App::main(), SIGNAL(peerUpdated(PeerData*)), this, SLOT(peerUpdated(PeerData *))); | ||||||
| 	connect(App::main(), SIGNAL(peerNameChanged(PeerData *, const PeerData::Names &, const PeerData::NameFirstChars &)), this, SLOT(peerUpdated(PeerData *))); | 	connect(App::main(), SIGNAL(peerNameChanged(PeerData *, const PeerData::Names &, const PeerData::NameFirstChars &)), this, SLOT(peerUpdated(PeerData *))); | ||||||
|  | @ -153,11 +161,15 @@ void ContactsInner::paintEvent(QPaintEvent *e) { | ||||||
| 				drawFrom = drawFrom->next; | 				drawFrom = drawFrom->next; | ||||||
| 			} | 			} | ||||||
| 		} else { | 		} else { | ||||||
| 			// ..
 | 			p.setFont(st::noContactsFont->f); | ||||||
|  | 			p.setPen(st::noContactsColor->p); | ||||||
|  | 			p.drawText(QRect(0, 0, width(), st::noContactsHeight - (cContactsReceived() ? st::noContactsFont->height : 0)), lang(cContactsReceived() ? lng_no_contacts : lng_contacts_loading), style::al_center); | ||||||
| 		} | 		} | ||||||
| 	} else { | 	} else { | ||||||
| 		if (_filtered.isEmpty()) { | 		if (_filtered.isEmpty()) { | ||||||
| 			// ..
 | 			p.setFont(st::noContactsFont->f); | ||||||
|  | 			p.setPen(st::noContactsColor->p); | ||||||
|  | 			p.drawText(QRect(0, 0, width(), st::noContactsHeight), lang(lng_contacts_not_found), style::al_center); | ||||||
| 		} else { | 		} else { | ||||||
| 			int32 from = yFrom / rh; | 			int32 from = yFrom / rh; | ||||||
| 			if (from < 0) from = 0; | 			if (from < 0) from = 0; | ||||||
|  | @ -255,11 +267,20 @@ void ContactsInner::updateFilter(QString filter) { | ||||||
| 		int32 rh = (st::profileListPhotoSize + st::profileListPadding.height() * 2); | 		int32 rh = (st::profileListPhotoSize + st::profileListPadding.height() * 2); | ||||||
| 		_filter = filter; | 		_filter = filter; | ||||||
| 		if (_filter.isEmpty()) { | 		if (_filter.isEmpty()) { | ||||||
| 			resize(width(), _contacts->list.count * rh + st::contactsClose.height); |  | ||||||
| 			if (_contacts->list.count) { | 			if (_contacts->list.count) { | ||||||
|  | 				if (!_addContactLnk.isHidden()) _addContactLnk.hide(); | ||||||
|  | 				resize(width(), _contacts->list.count * rh + st::contactsClose.height); | ||||||
| 				_sel = _contacts->list.begin; | 				_sel = _contacts->list.begin; | ||||||
|  | 			} else { | ||||||
|  | 				resize(width(), st::noContactsHeight); | ||||||
|  | 				if (cContactsReceived()) { | ||||||
|  | 					if (_addContactLnk.isHidden()) _addContactLnk.show(); | ||||||
|  | 				} else { | ||||||
|  | 					if (!_addContactLnk.isHidden()) _addContactLnk.hide(); | ||||||
|  | 				} | ||||||
| 			} | 			} | ||||||
| 		} else { | 		} else { | ||||||
|  | 			if (!_addContactLnk.isHidden()) _addContactLnk.hide(); | ||||||
| 			QStringList::const_iterator fb = f.cbegin(), fe = f.cend(), fi; | 			QStringList::const_iterator fb = f.cbegin(), fe = f.cend(), fi; | ||||||
| 
 | 
 | ||||||
| 			_filtered.clear(); | 			_filtered.clear(); | ||||||
|  | @ -302,7 +323,11 @@ void ContactsInner::updateFilter(QString filter) { | ||||||
| 			} | 			} | ||||||
| 			_filteredSel = _filtered.isEmpty() ? -1 : 0; | 			_filteredSel = _filtered.isEmpty() ? -1 : 0; | ||||||
| 
 | 
 | ||||||
| 			resize(width(), _filtered.size() * rh + st::contactsClose.height); | 			if (!_filtered.isEmpty()) { | ||||||
|  | 				resize(width(), _filtered.size() * rh + st::contactsClose.height); | ||||||
|  | 			} else { | ||||||
|  | 				resize(width(), st::noContactsHeight); | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
| 		if (parentWidget()) parentWidget()->update(); | 		if (parentWidget()) parentWidget()->update(); | ||||||
| 		loadProfilePhotos(0); | 		loadProfilePhotos(0); | ||||||
|  | @ -333,7 +358,8 @@ void ContactsInner::onDialogRowReplaced(DialogRow *oldRow, DialogRow *newRow) { | ||||||
| 	} | 	} | ||||||
| 	_mouseSel = false; | 	_mouseSel = false; | ||||||
| 	int32 rh = (st::profileListPhotoSize + st::profileListPadding.height() * 2); | 	int32 rh = (st::profileListPhotoSize + st::profileListPadding.height() * 2); | ||||||
| 	int32 newh = (_filter.isEmpty() ? _contacts->list.count : _filtered.size()) * rh; | 	int32 cnt = (_filter.isEmpty() ? _contacts->list.count : _filtered.size()); | ||||||
|  | 	int32 newh = cnt ? (cnt * rh + st::contactsClose.height) : st::noContactsHeight; | ||||||
| 	resize(width(), newh); | 	resize(width(), newh); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -343,6 +369,10 @@ ContactsInner::~ContactsInner() { | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void ContactsInner::resizeEvent(QResizeEvent *e) { | ||||||
|  | 	_addContactLnk.move((width() - _addContactLnk.width()) / 2, (st::noContactsHeight + st::noContactsFont->height) / 2); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void ContactsInner::selectSkip(int32 dir) { | void ContactsInner::selectSkip(int32 dir) { | ||||||
| 	_mouseSel = false; | 	_mouseSel = false; | ||||||
| 	int32 rh = st::profileListPhotoSize + st::profileListPadding.height() * 2, origDir = dir; | 	int32 rh = st::profileListPhotoSize + st::profileListPadding.height() * 2, origDir = dir; | ||||||
|  | @ -409,12 +439,12 @@ ContactsBox::ContactsBox() : _scroll(this, st::newGroupScroll), _inner(), | ||||||
| 	_scroll.setWidget(&_inner); | 	_scroll.setWidget(&_inner); | ||||||
| 	_scroll.setFocusPolicy(Qt::NoFocus); | 	_scroll.setFocusPolicy(Qt::NoFocus); | ||||||
| 
 | 
 | ||||||
| 	connect(&_addContact, SIGNAL(clicked()), this, SLOT(onAdd())); | 	connect(&_addContact, SIGNAL(clicked()), App::wnd(), SLOT(onShowAddContact())); | ||||||
| 	connect(&_close, SIGNAL(clicked()), this, SLOT(onClose())); | 	connect(&_close, SIGNAL(clicked()), this, SLOT(onClose())); | ||||||
| 	connect(&_scroll, SIGNAL(scrolled()), &_inner, SLOT(updateSel())); | 	connect(&_scroll, SIGNAL(scrolled()), &_inner, SLOT(updateSel())); | ||||||
| 	connect(&_scroll, SIGNAL(scrolled()), this, SLOT(onScroll())); | 	connect(&_scroll, SIGNAL(scrolled()), this, SLOT(onScroll())); | ||||||
| 	connect(&_filter, SIGNAL(changed()), this, SLOT(onFilterUpdate())); | 	connect(&_filter, SIGNAL(changed()), this, SLOT(onFilterUpdate())); | ||||||
| 	connect(&_filter, SIGNAL(cancelled()), this, SIGNAL(onClose())); | 	connect(&_filter, SIGNAL(cancelled()), this, SLOT(onClose())); | ||||||
| 	connect(&_inner, SIGNAL(mustScrollTo(int,int)), &_scroll, SLOT(scrollToY(int,int))); | 	connect(&_inner, SIGNAL(mustScrollTo(int,int)), &_scroll, SLOT(scrollToY(int,int))); | ||||||
| 
 | 
 | ||||||
| 	showAll(); | 	showAll(); | ||||||
|  |  | ||||||
|  | @ -31,6 +31,7 @@ public: | ||||||
| 	void leaveEvent(QEvent *e); | 	void leaveEvent(QEvent *e); | ||||||
| 	void mouseMoveEvent(QMouseEvent *e); | 	void mouseMoveEvent(QMouseEvent *e); | ||||||
| 	void mousePressEvent(QMouseEvent *e); | 	void mousePressEvent(QMouseEvent *e); | ||||||
|  | 	void resizeEvent(QResizeEvent *e); | ||||||
| 	 | 	 | ||||||
| 	void paintDialog(QPainter &p, DialogRow *row, bool sel); | 	void paintDialog(QPainter &p, DialogRow *row, bool sel); | ||||||
| 	void updateFilter(QString filter = QString()); | 	void updateFilter(QString filter = QString()); | ||||||
|  | @ -77,6 +78,7 @@ private: | ||||||
| 	ContactData *contactData(DialogRow *row); | 	ContactData *contactData(DialogRow *row); | ||||||
| 
 | 
 | ||||||
| 	QPoint _lastMousePos; | 	QPoint _lastMousePos; | ||||||
|  | 	LinkButton _addContactLnk; | ||||||
| 
 | 
 | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -22,7 +22,19 @@ Copyright (c) 2014 John Preston, https://tdesktop.com | ||||||
| #include "mainwidget.h" | #include "mainwidget.h" | ||||||
| #include "window.h" | #include "window.h" | ||||||
| 
 | 
 | ||||||
| NewGroupInner::NewGroupInner() : _contacts(&App::main()->contactsList()), _sel(0), _filteredSel(-1), _mouseSel(false), _selCount(0) { | NewGroupInner::NewGroupInner() : | ||||||
|  | _contacts(&App::main()->contactsList()), | ||||||
|  | _sel(0), | ||||||
|  | _filteredSel(-1), | ||||||
|  | _mouseSel(false), | ||||||
|  | _selCount(0), | ||||||
|  | _addContactLnk(this, lang(lng_add_contact_button)) { | ||||||
|  | 
 | ||||||
|  | 	connect(&_addContactLnk, SIGNAL(clicked()), App::wnd(), SLOT(onShowAddContact())); | ||||||
|  | 
 | ||||||
|  | 	for (DialogRow *r = _contacts->list.begin; r != _contacts->list.end; r = r->next) { | ||||||
|  | 		r->attached = 0; | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
| 	_filter = qsl("a"); | 	_filter = qsl("a"); | ||||||
| 	updateFilter(); | 	updateFilter(); | ||||||
|  | @ -165,11 +177,15 @@ void NewGroupInner::paintEvent(QPaintEvent *e) { | ||||||
| 				drawFrom = drawFrom->next; | 				drawFrom = drawFrom->next; | ||||||
| 			} | 			} | ||||||
| 		} else { | 		} else { | ||||||
| 			// ..
 | 			p.setFont(st::noContactsFont->f); | ||||||
|  | 			p.setPen(st::noContactsColor->p); | ||||||
|  | 			p.drawText(QRect(0, 0, width(), st::noContactsHeight - (cContactsReceived() ? st::noContactsFont->height : 0)), lang(cContactsReceived() ? lng_no_contacts : lng_contacts_loading), style::al_center); | ||||||
| 		} | 		} | ||||||
| 	} else { | 	} else { | ||||||
| 		if (_filtered.isEmpty()) { | 		if (_filtered.isEmpty()) { | ||||||
| 			// ..
 | 			p.setFont(st::noContactsFont->f); | ||||||
|  | 			p.setPen(st::noContactsColor->p); | ||||||
|  | 			p.drawText(QRect(0, 0, width(), st::noContactsHeight), lang(lng_contacts_not_found), style::al_center); | ||||||
| 		} else { | 		} else { | ||||||
| 			int32 from = yFrom / rh; | 			int32 from = yFrom / rh; | ||||||
| 			if (from < 0) from = 0; | 			if (from < 0) from = 0; | ||||||
|  | @ -279,9 +295,19 @@ void NewGroupInner::updateFilter(QString filter) { | ||||||
| 		if (_filter.isEmpty()) { | 		if (_filter.isEmpty()) { | ||||||
| 			resize(width(), _contacts->list.count * rh); | 			resize(width(), _contacts->list.count * rh); | ||||||
| 			if (_contacts->list.count) { | 			if (_contacts->list.count) { | ||||||
|  | 				if (!_addContactLnk.isHidden()) _addContactLnk.hide(); | ||||||
|  | 				resize(width(), _contacts->list.count * rh); | ||||||
| 				_sel = _contacts->list.begin; | 				_sel = _contacts->list.begin; | ||||||
|  | 			} else { | ||||||
|  | 				resize(width(), st::noContactsHeight); | ||||||
|  | 				if (cContactsReceived()) { | ||||||
|  | 					if (_addContactLnk.isHidden()) _addContactLnk.show(); | ||||||
|  | 				} else { | ||||||
|  | 					if (!_addContactLnk.isHidden()) _addContactLnk.hide(); | ||||||
|  | 				} | ||||||
| 			} | 			} | ||||||
| 		} else { | 		} else { | ||||||
|  | 			if (!_addContactLnk.isHidden()) _addContactLnk.hide(); | ||||||
| 			QStringList::const_iterator fb = f.cbegin(), fe = f.cend(), fi; | 			QStringList::const_iterator fb = f.cbegin(), fe = f.cend(), fi; | ||||||
| 
 | 
 | ||||||
| 			_filtered.clear(); | 			_filtered.clear(); | ||||||
|  | @ -324,7 +350,11 @@ void NewGroupInner::updateFilter(QString filter) { | ||||||
| 			} | 			} | ||||||
| 			_filteredSel = _filtered.isEmpty() ? -1 : 0; | 			_filteredSel = _filtered.isEmpty() ? -1 : 0; | ||||||
| 
 | 
 | ||||||
| 			resize(width(), _filtered.size() * rh); | 			if (!_filtered.isEmpty()) { | ||||||
|  | 				resize(width(), _filtered.size() * rh); | ||||||
|  | 			} else { | ||||||
|  | 				resize(width(), st::noContactsHeight); | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
| 		if (parentWidget()) parentWidget()->update(); | 		if (parentWidget()) parentWidget()->update(); | ||||||
| 		loadProfilePhotos(0); | 		loadProfilePhotos(0); | ||||||
|  | @ -365,6 +395,10 @@ NewGroupInner::~NewGroupInner() { | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void NewGroupInner::resizeEvent(QResizeEvent *e) { | ||||||
|  | 	_addContactLnk.move((width() - _addContactLnk.width()) / 2, (st::noContactsHeight + st::noContactsFont->height) / 2); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void NewGroupInner::selectSkip(int32 dir) { | void NewGroupInner::selectSkip(int32 dir) { | ||||||
| 	_mouseSel = false; | 	_mouseSel = false; | ||||||
| 	int32 rh = st::profileListPhotoSize + st::profileListPadding.height() * 2, origDir = dir; | 	int32 rh = st::profileListPhotoSize + st::profileListPadding.height() * 2, origDir = dir; | ||||||
|  | @ -456,7 +490,7 @@ NewGroupBox::NewGroupBox() : _scroll(this, st::newGroupScroll), _inner(), | ||||||
| 	connect(&_scroll, SIGNAL(scrolled()), &_inner, SLOT(updateSel())); | 	connect(&_scroll, SIGNAL(scrolled()), &_inner, SLOT(updateSel())); | ||||||
| 	connect(&_scroll, SIGNAL(scrolled()), this, SLOT(onScroll())); | 	connect(&_scroll, SIGNAL(scrolled()), this, SLOT(onScroll())); | ||||||
| 	connect(&_filter, SIGNAL(changed()), this, SLOT(onFilterUpdate())); | 	connect(&_filter, SIGNAL(changed()), this, SLOT(onFilterUpdate())); | ||||||
| 	connect(&_filter, SIGNAL(cancelled()), this, SIGNAL(onClose())); | 	connect(&_filter, SIGNAL(cancelled()), this, SLOT(onClose())); | ||||||
| 	connect(&_inner, SIGNAL(mustScrollTo(int,int)), &_scroll, SLOT(scrollToY(int,int))); | 	connect(&_inner, SIGNAL(mustScrollTo(int,int)), &_scroll, SLOT(scrollToY(int,int))); | ||||||
| 	connect(&_inner, SIGNAL(selectAllQuery()), &_filter, SLOT(selectAll())); | 	connect(&_inner, SIGNAL(selectAllQuery()), &_filter, SLOT(selectAll())); | ||||||
| 
 | 
 | ||||||
|  | @ -518,6 +552,7 @@ void NewGroupBox::paintEvent(QPaintEvent *e) { | ||||||
| 
 | 
 | ||||||
| 			// paint shadows
 | 			// paint shadows
 | ||||||
| 			p.fillRect(0, st::participantFilter.height, _width, st::scrollDef.topsh, st::scrollDef.shColor->b); | 			p.fillRect(0, st::participantFilter.height, _width, st::scrollDef.topsh, st::scrollDef.shColor->b); | ||||||
|  | 			p.fillRect(0, size().height() - st::btnSelectCancel.height - st::scrollDef.bottomsh, _width, st::scrollDef.bottomsh, st::scrollDef.shColor->b); | ||||||
| 
 | 
 | ||||||
| 			// paint button sep
 | 			// paint button sep
 | ||||||
| 			p.fillRect(st::btnSelectCancel.width, size().height() - st::btnSelectCancel.height, st::lineWidth, st::btnSelectCancel.height, st::btnSelectSep->b); | 			p.fillRect(st::btnSelectCancel.width, size().height() - st::btnSelectCancel.height, st::lineWidth, st::btnSelectCancel.height, st::btnSelectSep->b); | ||||||
|  |  | ||||||
|  | @ -31,7 +31,8 @@ public: | ||||||
| 	void leaveEvent(QEvent *e); | 	void leaveEvent(QEvent *e); | ||||||
| 	void mouseMoveEvent(QMouseEvent *e); | 	void mouseMoveEvent(QMouseEvent *e); | ||||||
| 	void mousePressEvent(QMouseEvent *e); | 	void mousePressEvent(QMouseEvent *e); | ||||||
| 	 | 	void resizeEvent(QResizeEvent *e); | ||||||
|  | 
 | ||||||
| 	void paintDialog(QPainter &p, DialogRow *row, bool sel); | 	void paintDialog(QPainter &p, DialogRow *row, bool sel); | ||||||
| 	void updateFilter(QString filter = QString()); | 	void updateFilter(QString filter = QString()); | ||||||
| 
 | 
 | ||||||
|  | @ -85,6 +86,7 @@ private: | ||||||
| 	ContactData *contactData(DialogRow *row); | 	ContactData *contactData(DialogRow *row); | ||||||
| 
 | 
 | ||||||
| 	QPoint _lastMousePos; | 	QPoint _lastMousePos; | ||||||
|  | 	LinkButton _addContactLnk; | ||||||
| 
 | 
 | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -28,24 +28,23 @@ UsernameInput::UsernameInput(QWidget *parent, const style::flatInput &st, const | ||||||
| 
 | 
 | ||||||
| void UsernameInput::correctValue(QKeyEvent *e, const QString &was) { | void UsernameInput::correctValue(QKeyEvent *e, const QString &was) { | ||||||
| 	QString oldText(text()), newText; | 	QString oldText(text()), newText; | ||||||
| 	int32 oldPos(cursorPosition()), newPos(-1), oldLen(oldText.length()); | 	int32 newPos = cursorPosition(), from, len = oldText.size(); | ||||||
| 	newText.reserve(oldLen); | 	for (from = 0; from < len; ++from) { | ||||||
| 
 | 		if (!oldText.at(from).isSpace()) { | ||||||
| 	for (int32 i = 0; i < oldLen; ++i) { | 			break; | ||||||
| 		if (i == oldPos) { |  | ||||||
| 			newPos = newText.length(); |  | ||||||
| 		} | 		} | ||||||
| 
 | 		if (newPos > 0) --newPos; | ||||||
| 		QChar ch = oldText[i]; | 	} | ||||||
| 		if ((ch >= 'A' && ch <= 'Z') || (ch >= 'a' && ch <= 'z') || (ch >= '0' && ch <= '9') || ch == '_' || (ch == '@' && !i)) { | 	len -= from; | ||||||
| 			if (newText.size() < MaxUsernameLength) { | 	if (len > MaxUsernameLength) len = MaxUsernameLength + (oldText.at(from) == '@' ? 1 : 0); | ||||||
| 				newText.append(ch); | 	for (int32 to = from + len; to > from;) { | ||||||
| 			} | 		--to; | ||||||
|  | 		if (!oldText.at(to).isSpace()) { | ||||||
|  | 			break; | ||||||
| 		} | 		} | ||||||
|  | 		--len; | ||||||
| 	} | 	} | ||||||
| 	if (newPos < 0) { | 	newText = oldText.mid(from, len); | ||||||
| 		newPos = newText.length(); |  | ||||||
| 	} |  | ||||||
| 	if (newText != oldText) { | 	if (newText != oldText) { | ||||||
| 		setText(newText); | 		setText(newText); | ||||||
| 		setCursorPosition(newPos); | 		setCursorPosition(newPos); | ||||||
|  | @ -59,6 +58,7 @@ _usernameInput(this, st::inpAddContact, qsl("@username"), App::self()->username) | ||||||
| _saveRequest(0), _checkRequest(0), _about(st::usernameWidth - 2 * st::addContactTitlePos.x()), | _saveRequest(0), _checkRequest(0), _about(st::usernameWidth - 2 * st::addContactTitlePos.x()), | ||||||
| a_opacity(0, 1), _hiding(false) { | a_opacity(0, 1), _hiding(false) { | ||||||
| 	_about.setRichText(st::usernameFont, lang(lng_username_about)); | 	_about.setRichText(st::usernameFont, lang(lng_username_about)); | ||||||
|  | 	_goodText = App::self()->username.isEmpty() ? QString() : lang(lng_username_available); | ||||||
| 	initBox(); | 	initBox(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -135,6 +135,11 @@ void UsernameBox::paintEvent(QPaintEvent *e) { | ||||||
| 				p.setFont(st::setErrFont->f); | 				p.setFont(st::setErrFont->f); | ||||||
| 				int32 w = st::setErrFont->m.width(_errorText); | 				int32 w = st::setErrFont->m.width(_errorText); | ||||||
| 				p.drawText((_width - w) / 2, _usernameInput.y() + _usernameInput.height() + ((st::usernameSkip - st::setErrFont->height) / 2) + st::setErrFont->ascent, _errorText); | 				p.drawText((_width - w) / 2, _usernameInput.y() + _usernameInput.height() + ((st::usernameSkip - st::setErrFont->height) / 2) + st::setErrFont->ascent, _errorText); | ||||||
|  | 			} else if (!_goodText.isEmpty()) { | ||||||
|  | 				p.setPen(st::setGoodColor->p); | ||||||
|  | 				p.setFont(st::setErrFont->f); | ||||||
|  | 				int32 w = st::setErrFont->m.width(_goodText); | ||||||
|  | 				p.drawText((_width - w) / 2, _usernameInput.y() + _usernameInput.height() + ((st::usernameSkip - st::setErrFont->height) / 2) + st::setErrFont->ascent, _goodText); | ||||||
| 			} | 			} | ||||||
| 			p.setPen(st::usernameColor->p); | 			p.setPen(st::usernameColor->p); | ||||||
| 			_about.draw(p, st::addContactTitlePos.x(), _usernameInput.y() + _usernameInput.height() + st::usernameSkip, width() - 2 * st::addContactTitlePos.x()); | 			_about.draw(p, st::addContactTitlePos.x(), _usernameInput.y() + _usernameInput.height() + st::usernameSkip, width() - 2 * st::addContactTitlePos.x()); | ||||||
|  | @ -180,23 +185,37 @@ void UsernameBox::onCheck() { | ||||||
| void UsernameBox::onChanged() { | void UsernameBox::onChanged() { | ||||||
| 	QString name = getName(); | 	QString name = getName(); | ||||||
| 	if (name.isEmpty()) { | 	if (name.isEmpty()) { | ||||||
| 		if (!_errorText.isEmpty()) { | 		if (!_errorText.isEmpty() || !_goodText.isEmpty()) { | ||||||
| 			_errorText = QString(); | 			_errorText = _goodText = QString(); | ||||||
| 			update(); |  | ||||||
| 		} |  | ||||||
| 		_checkTimer.stop(); |  | ||||||
| 	} else if (name.size() < MinUsernameLength) { |  | ||||||
| 		if (_errorText != lang(lng_username_too_short)) { |  | ||||||
| 			_errorText = lang(lng_username_too_short); |  | ||||||
| 			update(); | 			update(); | ||||||
| 		} | 		} | ||||||
| 		_checkTimer.stop(); | 		_checkTimer.stop(); | ||||||
| 	} else { | 	} else { | ||||||
| 		if (!_errorText.isEmpty()) { | 		int32 i, len = name.size(); | ||||||
| 			_errorText = QString(); | 		for (int32 i = 0; i < len; ++i) { | ||||||
| 			update(); | 			QChar ch = name.at(i); | ||||||
|  | 			if ((ch < 'A' || ch > 'Z') && (ch < 'a' || ch > 'z') && (ch < '0' || ch > '9') && ch != '_' && (ch != '@' || i > 0)) { | ||||||
|  | 				if (_errorText != lang(lng_username_bad_symbols)) { | ||||||
|  | 					_errorText = lang(lng_username_bad_symbols); | ||||||
|  | 					update(); | ||||||
|  | 				} | ||||||
|  | 				_checkTimer.stop(); | ||||||
|  | 				return; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		if (name.size() < MinUsernameLength) { | ||||||
|  | 			if (_errorText != lang(lng_username_too_short)) { | ||||||
|  | 				_errorText = lang(lng_username_too_short); | ||||||
|  | 				update(); | ||||||
|  | 			} | ||||||
|  | 			_checkTimer.stop(); | ||||||
|  | 		} else { | ||||||
|  | 			if (!_errorText.isEmpty() || !_goodText.isEmpty()) { | ||||||
|  | 				_errorText = _goodText = QString(); | ||||||
|  | 				update(); | ||||||
|  | 			} | ||||||
|  | 			_checkTimer.start(UsernameCheckTimeout); | ||||||
| 		} | 		} | ||||||
| 		_checkTimer.start(UsernameCheckTimeout); |  | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -230,8 +249,10 @@ bool UsernameBox::onUpdateFail(const RPCError &error) { | ||||||
| void UsernameBox::onCheckDone(const MTPBool &result) { | void UsernameBox::onCheckDone(const MTPBool &result) { | ||||||
| 	_checkRequest = 0; | 	_checkRequest = 0; | ||||||
| 	QString newError = (result.v || _checkUsername == App::self()->username) ? QString() : lang(lng_username_occupied); | 	QString newError = (result.v || _checkUsername == App::self()->username) ? QString() : lang(lng_username_occupied); | ||||||
| 	if (_errorText != newError) { | 	QString newGood = newError.isEmpty() ? lang(lng_username_available) : QString(); | ||||||
|  | 	if (_errorText != newError || _goodText != newGood) { | ||||||
| 		_errorText = newError; | 		_errorText = newError; | ||||||
|  | 		_goodText = newGood; | ||||||
| 		update(); | 		update(); | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  | @ -248,6 +269,7 @@ bool UsernameBox::onCheckFail(const RPCError &error) { | ||||||
| 		update(); | 		update(); | ||||||
| 		return true; | 		return true; | ||||||
| 	} | 	} | ||||||
|  | 	_goodText = QString(); | ||||||
| 	_usernameInput.setFocus(); | 	_usernameInput.setFocus(); | ||||||
| 	return true; | 	return true; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -72,7 +72,7 @@ private: | ||||||
| 	QPixmap _cache; | 	QPixmap _cache; | ||||||
| 
 | 
 | ||||||
| 	mtpRequestId _saveRequest, _checkRequest; | 	mtpRequestId _saveRequest, _checkRequest; | ||||||
| 	QString _sentUsername, _checkUsername, _errorText; | 	QString _sentUsername, _checkUsername, _errorText, _goodText; | ||||||
| 
 | 
 | ||||||
| 	Text _about; | 	Text _about; | ||||||
| 	QTimer _checkTimer; | 	QTimer _checkTimer; | ||||||
|  |  | ||||||
|  | @ -26,11 +26,25 @@ Copyright (c) 2014 John Preston, https://tdesktop.com | ||||||
| #include "boxes/newgroupbox.h" | #include "boxes/newgroupbox.h" | ||||||
| 
 | 
 | ||||||
| DialogsListWidget::DialogsListWidget(QWidget *parent, MainWidget *main) : QWidget(parent), | DialogsListWidget::DialogsListWidget(QWidget *parent, MainWidget *main) : QWidget(parent), | ||||||
| dialogs(false), contactsNoDialogs(true), contacts(true), sel(0), contactSel(false), selByMouse(false), filteredSel(-1), searchedCount(0), searchedSel(-1), peopleSel(-1), _lastSearchId(0), _state(DefaultState) { | dialogs(false), | ||||||
| 	connect(main, SIGNAL(dialogToTop(const History::DialogLinks &)), this, SLOT(onDialogToTop(const History::DialogLinks &))); | contactsNoDialogs(true), | ||||||
| 	connect(main, SIGNAL(peerNameChanged(PeerData *, const PeerData::Names &, const PeerData::NameFirstChars &)), this, SLOT(onPeerNameChanged(PeerData *, const PeerData::Names &, const PeerData::NameFirstChars &))); | contacts(true), | ||||||
| 	connect(main, SIGNAL(peerPhotoChanged(PeerData *)), this, SLOT(onPeerPhotoChanged(PeerData *))); | sel(0), | ||||||
| 	connect(main, SIGNAL(dialogRowReplaced(DialogRow *, DialogRow *)), this, SLOT(onDialogRowReplaced(DialogRow *, DialogRow *))); | contactSel(false), | ||||||
|  | selByMouse(false), | ||||||
|  | filteredSel(-1), | ||||||
|  | searchedCount(0), | ||||||
|  | searchedSel(-1), | ||||||
|  | peopleSel(-1), | ||||||
|  | _lastSearchId(0), | ||||||
|  | _state(DefaultState), | ||||||
|  | _addContactLnk(this, lang(lng_add_contact_button)) { | ||||||
|  | 	connect(main, SIGNAL(dialogToTop(const History::DialogLinks&)), this, SLOT(onDialogToTop(const History::DialogLinks&))); | ||||||
|  | 	connect(main, SIGNAL(peerNameChanged(PeerData*,const PeerData::Names&,const PeerData::NameFirstChars&)), this, SLOT(onPeerNameChanged(PeerData*,const PeerData::Names&,const PeerData::NameFirstChars&))); | ||||||
|  | 	connect(main, SIGNAL(peerPhotoChanged(PeerData*)), this, SLOT(onPeerPhotoChanged(PeerData*))); | ||||||
|  | 	connect(main, SIGNAL(dialogRowReplaced(DialogRow*,DialogRow*)), this, SLOT(onDialogRowReplaced(DialogRow*,DialogRow*))); | ||||||
|  | 	connect(&_addContactLnk, SIGNAL(clicked()), App::wnd(), SLOT(onShowAddContact())); | ||||||
|  | 	refresh(false); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void DialogsListWidget::paintEvent(QPaintEvent *e) { | void DialogsListWidget::paintEvent(QPaintEvent *e) { | ||||||
|  | @ -51,7 +65,9 @@ void DialogsListWidget::paintEvent(QPaintEvent *e) { | ||||||
| 		if (contactsNoDialogs.list.count) { | 		if (contactsNoDialogs.list.count) { | ||||||
| 			contactsNoDialogs.list.paint(p, width(), r.top() - otherStart, r.bottom() - otherStart, active, selected); | 			contactsNoDialogs.list.paint(p, width(), r.top() - otherStart, r.bottom() - otherStart, active, selected); | ||||||
| 		} else if (!otherStart) { | 		} else if (!otherStart) { | ||||||
| 			// .. paint no dialogs found
 | 			p.setFont(st::noContactsFont->f); | ||||||
|  | 			p.setPen(st::noContactsColor->p); | ||||||
|  | 			p.drawText(QRect(0, 0, width(), st::noContactsHeight - (cContactsReceived() ? st::noContactsFont->height : 0)), lang(cContactsReceived() ? lng_no_contacts : lng_contacts_loading), style::al_center); | ||||||
| 		} | 		} | ||||||
| 	} else if (_state == FilteredState || _state == SearchedState) { | 	} else if (_state == FilteredState || _state == SearchedState) { | ||||||
| 		if (filterResults.isEmpty()) { | 		if (filterResults.isEmpty()) { | ||||||
|  | @ -255,6 +271,10 @@ void DialogsListWidget::mousePressEvent(QMouseEvent *e) { | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void DialogsListWidget::resizeEvent(QResizeEvent *e) { | ||||||
|  | 	_addContactLnk.move((width() - _addContactLnk.width()) / 2, (st::noContactsHeight + st::noContactsFont->height) / 2); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void DialogsListWidget::onDialogRowReplaced(DialogRow *oldRow, DialogRow *newRow) { | void DialogsListWidget::onDialogRowReplaced(DialogRow *oldRow, DialogRow *newRow) { | ||||||
| 	if (_state == FilteredState || _state == SearchedState) { | 	if (_state == FilteredState || _state == SearchedState) { | ||||||
| 		for (FilteredDialogs::iterator i = filterResults.begin(); i != filterResults.end();) { | 		for (FilteredDialogs::iterator i = filterResults.begin(); i != filterResults.end();) { | ||||||
|  | @ -615,6 +635,7 @@ void DialogsListWidget::peopleReceived(const QString &query, const QVector<MTPCo | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void DialogsListWidget::contactsReceived(const QVector<MTPContact> &contacts) { | void DialogsListWidget::contactsReceived(const QVector<MTPContact> &contacts) { | ||||||
|  | 	cSetContactsReceived(true); | ||||||
| 	for (QVector<MTPContact>::const_iterator i = contacts.cbegin(), e = contacts.cend(); i != e; ++i) { | 	for (QVector<MTPContact>::const_iterator i = contacts.cbegin(), e = contacts.cend(); i != e; ++i) { | ||||||
| 		addNewContact(i->c_contact().vuser_id.v); | 		addNewContact(i->c_contact().vuser_id.v); | ||||||
| 	} | 	} | ||||||
|  | @ -652,10 +673,23 @@ void DialogsListWidget::refresh(bool toTop) { | ||||||
| 	int32 h = 0; | 	int32 h = 0; | ||||||
| 	if (_state == DefaultState) { | 	if (_state == DefaultState) { | ||||||
| 		h = (dialogs.list.count + contactsNoDialogs.list.count) * st::dlgHeight; | 		h = (dialogs.list.count + contactsNoDialogs.list.count) * st::dlgHeight; | ||||||
| 	} else if (_state == FilteredState) { | 		if (h) { | ||||||
| 		h = (filterResults.count() + peopleResults.count() + searchResults.count()) * st::dlgHeight + (peopleResults.isEmpty() ? 0 : st::searchedBarHeight) + (searchResults.isEmpty() ? 0 : st::searchedBarHeight); | 			if (!_addContactLnk.isHidden()) _addContactLnk.hide(); | ||||||
| 	} else if (_state == SearchedState) { | 		} else { | ||||||
| 		h = (filterResults.count() + peopleResults.count() + searchResults.count()) * st::dlgHeight + (peopleResults.isEmpty() ? 0 : st::searchedBarHeight) + st::searchedBarHeight; | 			h = st::noContactsHeight; | ||||||
|  | 			if (cContactsReceived()) { | ||||||
|  | 				if (_addContactLnk.isHidden()) _addContactLnk.show(); | ||||||
|  | 			} else { | ||||||
|  | 				if (!_addContactLnk.isHidden()) _addContactLnk.hide(); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		if (!_addContactLnk.isHidden()) _addContactLnk.hide(); | ||||||
|  | 		if (_state == FilteredState) { | ||||||
|  | 			h = (filterResults.count() + peopleResults.count() + searchResults.count()) * st::dlgHeight + (peopleResults.isEmpty() ? 0 : st::searchedBarHeight) + (searchResults.isEmpty() ? 0 : st::searchedBarHeight); | ||||||
|  | 		} else if (_state == SearchedState) { | ||||||
|  | 			h = (filterResults.count() + peopleResults.count() + searchResults.count()) * st::dlgHeight + (peopleResults.isEmpty() ? 0 : st::searchedBarHeight) + st::searchedBarHeight; | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 	resize(width(), h); | 	resize(width(), h); | ||||||
| 	if (toTop) { | 	if (toTop) { | ||||||
|  | @ -1607,7 +1641,7 @@ DialogsIndexed &DialogsWidget::contactsList() { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void DialogsWidget::onAddContact() { | void DialogsWidget::onAddContact() { | ||||||
| 	App::wnd()->showLayer(new AddContactBox()); | 	App::wnd()->replaceLayer(new AddContactBox()); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void DialogsWidget::onNewGroup() { | void DialogsWidget::onNewGroup() { | ||||||
|  |  | ||||||
|  | @ -39,6 +39,7 @@ public: | ||||||
| 	void paintEvent(QPaintEvent *e); | 	void paintEvent(QPaintEvent *e); | ||||||
| 	void mouseMoveEvent(QMouseEvent *e); | 	void mouseMoveEvent(QMouseEvent *e); | ||||||
| 	void mousePressEvent(QMouseEvent *e); | 	void mousePressEvent(QMouseEvent *e); | ||||||
|  | 	void resizeEvent(QResizeEvent *e); | ||||||
| 	void enterEvent(QEvent *e); | 	void enterEvent(QEvent *e); | ||||||
| 	void leaveEvent(QEvent *e); | 	void leaveEvent(QEvent *e); | ||||||
| 
 | 
 | ||||||
|  | @ -139,6 +140,8 @@ private: | ||||||
| 
 | 
 | ||||||
| 	void paintDialog(QPainter &p, DialogRow *dialog); | 	void paintDialog(QPainter &p, DialogRow *dialog); | ||||||
| 
 | 
 | ||||||
|  | 	LinkButton _addContactLnk; | ||||||
|  | 
 | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| class DialogsWidget : public QWidget, public Animated, public RPCSender { | class DialogsWidget : public QWidget, public Animated, public RPCSender { | ||||||
|  |  | ||||||
|  | @ -64,18 +64,19 @@ void BackgroundWidget::onClose() { | ||||||
| 	startHide(); | 	startHide(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void BackgroundWidget::onInnerClose() { | bool BackgroundWidget::onInnerClose() { | ||||||
| 	if (_hidden) { | 	if (!_hidden) { | ||||||
| 		w->deleteLater(); |  | ||||||
| 		w = _hidden; |  | ||||||
| 		_hidden = 0; |  | ||||||
| 		w->show(); |  | ||||||
| 		resizeEvent(0); |  | ||||||
| 		w->animStep(1); |  | ||||||
| 		update(); |  | ||||||
| 	} else { |  | ||||||
| 		onClose(); | 		onClose(); | ||||||
|  | 		return true; | ||||||
| 	} | 	} | ||||||
|  | 	w->deleteLater(); | ||||||
|  | 	w = _hidden; | ||||||
|  | 	_hidden = 0; | ||||||
|  | 	w->show(); | ||||||
|  | 	resizeEvent(0); | ||||||
|  | 	w->animStep(1); | ||||||
|  | 	update(); | ||||||
|  | 	return false; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void BackgroundWidget::startHide() { | void BackgroundWidget::startHide() { | ||||||
|  |  | ||||||
|  | @ -68,7 +68,7 @@ public: | ||||||
| public slots: | public slots: | ||||||
| 
 | 
 | ||||||
| 	void onClose(); | 	void onClose(); | ||||||
| 	void onInnerClose(); | 	bool onInnerClose(); | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -769,6 +769,10 @@ void MainWidget::checkLastUpdate(bool afterSleep) { | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void MainWidget::showAddContact() { | ||||||
|  | 	dialogs.onAddContact(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void MainWidget::showNewGroup() { | void MainWidget::showNewGroup() { | ||||||
| 	dialogs.onNewGroup(); | 	dialogs.onNewGroup(); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -285,6 +285,7 @@ public: | ||||||
| 	void peerUsernameChanged(PeerData *peer); | 	void peerUsernameChanged(PeerData *peer); | ||||||
| 
 | 
 | ||||||
| 	void checkLastUpdate(bool afterSleep); | 	void checkLastUpdate(bool afterSleep); | ||||||
|  | 	void showAddContact(); | ||||||
| 	void showNewGroup(); | 	void showNewGroup(); | ||||||
| 
 | 
 | ||||||
| 	~MainWidget(); | 	~MainWidget(); | ||||||
|  |  | ||||||
|  | @ -1090,7 +1090,6 @@ MTProtoConnectionPrivate::MTProtoConnectionPrivate(QThread *thread, MTProtoConne | ||||||
| 	oldConnectionTimer.moveToThread(thread); | 	oldConnectionTimer.moveToThread(thread); | ||||||
| 	connCheckTimer.moveToThread(thread); | 	connCheckTimer.moveToThread(thread); | ||||||
| 	retryTimer.moveToThread(thread); | 	retryTimer.moveToThread(thread); | ||||||
| 	pinger.moveToThread(thread); |  | ||||||
| 	moveToThread(thread); | 	moveToThread(thread); | ||||||
| 
 | 
 | ||||||
| //	createConn();
 | //	createConn();
 | ||||||
|  | @ -1118,10 +1117,22 @@ MTProtoConnectionPrivate::MTProtoConnectionPrivate(QThread *thread, MTProtoConne | ||||||
| 	connect(this, SIGNAL(needToReceive()), sessionData->owner(), SLOT(tryToReceive()), Qt::QueuedConnection); | 	connect(this, SIGNAL(needToReceive()), sessionData->owner(), SLOT(tryToReceive()), Qt::QueuedConnection); | ||||||
| 	connect(this, SIGNAL(stateChanged(qint32)), sessionData->owner(), SLOT(onConnectionStateChange(qint32)), Qt::QueuedConnection); | 	connect(this, SIGNAL(stateChanged(qint32)), sessionData->owner(), SLOT(onConnectionStateChange(qint32)), Qt::QueuedConnection); | ||||||
| 	connect(sessionData->owner(), SIGNAL(needToSend()), this, SLOT(tryToSend()), Qt::QueuedConnection); | 	connect(sessionData->owner(), SIGNAL(needToSend()), this, SLOT(tryToSend()), Qt::QueuedConnection); | ||||||
| 	connect(this, SIGNAL(needToSendAsync()), sessionData->owner(), SIGNAL(needToSend()), Qt::QueuedConnection); |  | ||||||
| 	connect(this, SIGNAL(sendHttpWait()), sessionData->owner(), SLOT(sendHttpWait()), Qt::QueuedConnection); |  | ||||||
| 	connect(this, SIGNAL(sessionResetDone()), sessionData->owner(), SLOT(onResetDone()), Qt::QueuedConnection); | 	connect(this, SIGNAL(sessionResetDone()), sessionData->owner(), SLOT(onResetDone()), Qt::QueuedConnection); | ||||||
|  | 
 | ||||||
|  | 	static bool _registered = false; | ||||||
|  | 	if (!_registered) { | ||||||
|  | 		_registered = true; | ||||||
|  | 		qRegisterMetaType<QVector<quint64>>("QVector<quint64>"); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	connect(this, SIGNAL(needToSendAsync()), sessionData->owner(), SIGNAL(needToSend()), Qt::QueuedConnection); | ||||||
| 	connect(this, SIGNAL(sendAnythingAsync(quint64)), sessionData->owner(), SLOT(sendAnything(quint64)), Qt::QueuedConnection); | 	connect(this, SIGNAL(sendAnythingAsync(quint64)), sessionData->owner(), SLOT(sendAnything(quint64)), Qt::QueuedConnection); | ||||||
|  | 	connect(this, SIGNAL(sendHttpWaitAsync()), sessionData->owner(), SLOT(sendHttpWait()), Qt::QueuedConnection); | ||||||
|  | 	connect(this, SIGNAL(sendPongAsync(quint64,quint64)), sessionData->owner(), SLOT(sendPong(quint64,quint64)), Qt::QueuedConnection); | ||||||
|  | 	connect(this, SIGNAL(sendMsgsStateInfoAsync(quint64, QByteArray)), sessionData->owner(), SLOT(sendMsgsStateInfo(quint64,QByteArray)), Qt::QueuedConnection); | ||||||
|  | 	connect(this, SIGNAL(resendAsync(quint64,quint64,bool,bool)), sessionData->owner(), SLOT(resend(quint64,quint64,bool,bool)), Qt::QueuedConnection); | ||||||
|  | 	connect(this, SIGNAL(resendManyAsync(QVector<quint64>,quint64,bool,bool)), sessionData->owner(), SLOT(resendMany(QVector<quint64>,quint64,bool,bool)), Qt::QueuedConnection); | ||||||
|  | 	connect(this, SIGNAL(resendAllAsync()), sessionData->owner(), SLOT(resendAll())); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void MTProtoConnectionPrivate::onConfigLoaded() { | void MTProtoConnectionPrivate::onConfigLoaded() { | ||||||
|  | @ -1809,7 +1820,6 @@ void MTProtoConnectionPrivate::doDisconnect() { | ||||||
| 
 | 
 | ||||||
| 	unlockKey(); | 	unlockKey(); | ||||||
| 
 | 
 | ||||||
| 	pinger.stop(); |  | ||||||
| 	clearAuthKeyData(); | 	clearAuthKeyData(); | ||||||
| 
 | 
 | ||||||
| 	setState(MTProtoConnection::Disconnected); | 	setState(MTProtoConnection::Disconnected); | ||||||
|  | @ -1914,7 +1924,7 @@ void MTProtoConnectionPrivate::handleReceived() { | ||||||
| 				sessionData->setSalt(serverSalt); | 				sessionData->setSalt(serverSalt); | ||||||
| 				if (setState(MTProtoConnection::Connected, MTProtoConnection::Connecting)) { // only connected
 | 				if (setState(MTProtoConnection::Connected, MTProtoConnection::Connecting)) { // only connected
 | ||||||
| 					if (restarted) { | 					if (restarted) { | ||||||
| 						sessionData->owner()->resendAll(); | 						emit resendAllAsync(); | ||||||
| 						restarted = false; | 						restarted = false; | ||||||
| 					} | 					} | ||||||
| 				} | 				} | ||||||
|  | @ -1987,7 +1997,7 @@ void MTProtoConnectionPrivate::handleReceived() { | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	if (conn->needHttpWait()) { | 	if (conn->needHttpWait()) { | ||||||
| 		emit sendHttpWait(); | 		emit sendHttpWaitAsync(); | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -2165,7 +2175,7 @@ int32 MTProtoConnectionPrivate::handleOneReceived(const mtpPrime *from, const mt | ||||||
| 
 | 
 | ||||||
| 		if (setState(MTProtoConnection::Connected, MTProtoConnection::Connecting)) { // maybe only connected
 | 		if (setState(MTProtoConnection::Connected, MTProtoConnection::Connecting)) { // maybe only connected
 | ||||||
| 			if (restarted) { | 			if (restarted) { | ||||||
| 				sessionData->owner()->resendAll(); | 				emit resendAllAsync(); | ||||||
| 				restarted = false; | 				restarted = false; | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  | @ -2187,10 +2197,7 @@ int32 MTProtoConnectionPrivate::handleOneReceived(const mtpPrime *from, const mt | ||||||
| 		DEBUG_LOG(("Message Info: msgs_state_req received, ids: %1").arg(logVectorLong(ids))); | 		DEBUG_LOG(("Message Info: msgs_state_req received, ids: %1").arg(logVectorLong(ids))); | ||||||
| 		if (!idsCount) return 1; | 		if (!idsCount) return 1; | ||||||
| 
 | 
 | ||||||
| 		MTPMsgsStateInfo req(MTP_msgs_state_info(MTP_long(msgId), MTPstring())); | 		QByteArray info(idsCount, Qt::Uninitialized); | ||||||
| 		string &info(req._msgs_state_info().vinfo._string().v); |  | ||||||
| 		info.resize(idsCount); |  | ||||||
| 
 |  | ||||||
| 		{ | 		{ | ||||||
| 			QReadLocker lock(sessionData->receivedIdsMutex()); | 			QReadLocker lock(sessionData->receivedIdsMutex()); | ||||||
| 			const mtpMsgIdsMap &receivedIds(sessionData->receivedIdsSet()); | 			const mtpMsgIdsMap &receivedIds(sessionData->receivedIdsSet()); | ||||||
|  | @ -2227,8 +2234,7 @@ int32 MTProtoConnectionPrivate::handleOneReceived(const mtpPrime *from, const mt | ||||||
| 				info[i] = state; | 				info[i] = state; | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 
 | 		emit sendMsgsStateInfoAsync(msgId, info); | ||||||
| 		sessionData->owner()->send(req); |  | ||||||
| 	} return 1; | 	} return 1; | ||||||
| 
 | 
 | ||||||
| 	case mtpc_msgs_state_info: { | 	case mtpc_msgs_state_info: { | ||||||
|  | @ -2366,9 +2372,11 @@ int32 MTProtoConnectionPrivate::handleOneReceived(const mtpPrime *from, const mt | ||||||
| 		DEBUG_LOG(("Message Info: resend of msgs requested, ids: %1").arg(logVectorLong(ids))); | 		DEBUG_LOG(("Message Info: resend of msgs requested, ids: %1").arg(logVectorLong(ids))); | ||||||
| 		if (!idsCount) return (badTime ? 0 : 1); | 		if (!idsCount) return (badTime ? 0 : 1); | ||||||
| 
 | 
 | ||||||
| 		for (QVector<MTPlong>::const_iterator i = ids.cbegin(), e = ids.cend(); i != e; ++i) { | 		QVector<quint64> toResend(ids.size(), Qt::Uninitialized); | ||||||
| 			resend(i->v, 0, false, true); | 		for (int32 i = 0, l = ids.size(); i < l; ++i) { | ||||||
|  | 			toResend[i] = ids.at(i).v; | ||||||
| 		} | 		} | ||||||
|  | 		resendMany(toResend, 0, false, true); | ||||||
| 	} return 1; | 	} return 1; | ||||||
| 
 | 
 | ||||||
| 	case mtpc_rpc_result: { | 	case mtpc_rpc_result: { | ||||||
|  | @ -2425,7 +2433,7 @@ int32 MTProtoConnectionPrivate::handleOneReceived(const mtpPrime *from, const mt | ||||||
| 		sessionData->setSalt(data.vserver_salt.v); | 		sessionData->setSalt(data.vserver_salt.v); | ||||||
| 
 | 
 | ||||||
| 		mtpMsgId firstMsgId = data.vfirst_msg_id.v; | 		mtpMsgId firstMsgId = data.vfirst_msg_id.v; | ||||||
| 		QVector<mtpMsgId> toResend; | 		QVector<quint64> toResend; | ||||||
| 		{ | 		{ | ||||||
| 			QReadLocker locker(sessionData->haveSentMutex()); | 			QReadLocker locker(sessionData->haveSentMutex()); | ||||||
| 			const mtpRequestMap &haveSent(sessionData->haveSentMap()); | 			const mtpRequestMap &haveSent(sessionData->haveSentMap()); | ||||||
|  | @ -2435,9 +2443,7 @@ int32 MTProtoConnectionPrivate::handleOneReceived(const mtpPrime *from, const mt | ||||||
| 				if (i.value()->requestId) toResend.push_back(i.key()); | 				if (i.value()->requestId) toResend.push_back(i.key()); | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		for (uint32 i = 0, l = toResend.size(); i < l; ++i) { | 		resendMany(toResend, 10, true); | ||||||
| 			resend(toResend[i], 10, true); |  | ||||||
| 		} |  | ||||||
| 
 | 
 | ||||||
| 		mtpBuffer update(end - from); | 		mtpBuffer update(end - from); | ||||||
| 		if (end > from) memcpy(update.data(), from, (end - from) * sizeof(mtpPrime)); | 		if (end > from) memcpy(update.data(), from, (end - from) * sizeof(mtpPrime)); | ||||||
|  | @ -2454,7 +2460,7 @@ int32 MTProtoConnectionPrivate::handleOneReceived(const mtpPrime *from, const mt | ||||||
| 		MTPPing msg(from, end); | 		MTPPing msg(from, end); | ||||||
| 		DEBUG_LOG(("Message Info: ping received, ping_id: %1, sending pong..").arg(msg.vping_id.v)); | 		DEBUG_LOG(("Message Info: ping received, ping_id: %1, sending pong..").arg(msg.vping_id.v)); | ||||||
| 
 | 
 | ||||||
| 		sessionData->owner()->send(MTP_pong(MTP_long(msgId), msg.vping_id)); | 		emit sendPongAsync(msgId, msg.vping_id.v); | ||||||
| 	} return 1; | 	} return 1; | ||||||
| 
 | 
 | ||||||
| 	case mtpc_pong: { | 	case mtpc_pong: { | ||||||
|  | @ -2714,9 +2720,19 @@ void MTProtoConnectionPrivate::handleMsgsStates(const QVector<MTPlong> &ids, con | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| mtpRequestId MTProtoConnectionPrivate::resend(mtpMsgId msgId, uint64 msCanWait, bool forceContainer, bool sendMsgStateInfo) { | void MTProtoConnectionPrivate::resend(quint64 msgId, quint64 msCanWait, bool forceContainer, bool sendMsgStateInfo) { | ||||||
| 	if (msgId == pingMsgId) return 0xFFFFFFFF; | 	if (msgId == pingMsgId) return; | ||||||
| 	return sessionData->owner()->resend(msgId, msCanWait, forceContainer, sendMsgStateInfo); | 	emit resendAsync(msgId, msCanWait, forceContainer, sendMsgStateInfo); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void MTProtoConnectionPrivate::resendMany(QVector<quint64> msgIds, quint64 msCanWait, bool forceContainer, bool sendMsgStateInfo) { | ||||||
|  | 	for (int32 i = 0, l = msgIds.size(); i < l; ++i) { | ||||||
|  | 		if (msgIds.at(i) == pingMsgId) { | ||||||
|  | 			msgIds.remove(i); | ||||||
|  | 			--l; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	emit resendManyAsync(msgIds, msCanWait, forceContainer, sendMsgStateInfo); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void MTProtoConnectionPrivate::onConnected() { | void MTProtoConnectionPrivate::onConnected() { | ||||||
|  | @ -3186,7 +3202,7 @@ void MTProtoConnectionPrivate::authKeyCreated() { | ||||||
| 	if (sessionData->getSalt()) { // else receive salt in bad_server_salt first, then try to send all the requests
 | 	if (sessionData->getSalt()) { // else receive salt in bad_server_salt first, then try to send all the requests
 | ||||||
| 		setState(MTProtoConnection::Connected); | 		setState(MTProtoConnection::Connected); | ||||||
| 		if (restarted) { | 		if (restarted) { | ||||||
| 			sessionData->owner()->resendAll(); | 			emit resendAllAsync(); | ||||||
| 			restarted = false; | 			restarted = false; | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | @ -3194,10 +3210,6 @@ void MTProtoConnectionPrivate::authKeyCreated() { | ||||||
| 	toSendPingId = MTP::nonce<uint64>(); // get server_salt
 | 	toSendPingId = MTP::nonce<uint64>(); // get server_salt
 | ||||||
| 
 | 
 | ||||||
| 	emit needToSendAsync(); | 	emit needToSendAsync(); | ||||||
| 
 |  | ||||||
| //	disconnect(&pinger, SIGNAL(timeout()), 0, 0);
 |  | ||||||
| //	connect(&pinger, SIGNAL(timeout()), this, SLOT(sendPing()));
 |  | ||||||
| //	pinger.start(30000);
 |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void MTProtoConnectionPrivate::clearAuthKeyData() { | void MTProtoConnectionPrivate::clearAuthKeyData() { | ||||||
|  | @ -3218,10 +3230,6 @@ void MTProtoConnectionPrivate::clearAuthKeyData() { | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void MTProtoConnectionPrivate::sendPing() { |  | ||||||
| 	sessionData->owner()->send(MTPPing(MTP::nonce<MTPlong>())); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void MTProtoConnectionPrivate::onError(bool mayBeBadKey) { | void MTProtoConnectionPrivate::onError(bool mayBeBadKey) { | ||||||
| 	MTP_LOG(dc, ("Restarting after error, maybe bad key: %1..").arg(logBool(mayBeBadKey))); | 	MTP_LOG(dc, ("Restarting after error, maybe bad key: %1..").arg(logBool(mayBeBadKey))); | ||||||
| 	return restart(mayBeBadKey); | 	return restart(mayBeBadKey); | ||||||
|  |  | ||||||
|  | @ -308,10 +308,15 @@ signals: | ||||||
| 	void needToRestart(); | 	void needToRestart(); | ||||||
| 	void stateChanged(qint32 newState); | 	void stateChanged(qint32 newState); | ||||||
| 	void sessionResetDone(); | 	void sessionResetDone(); | ||||||
| 	void needToSendAsync(); |  | ||||||
| 	void sendAnythingAsync(quint64); |  | ||||||
| 
 | 
 | ||||||
| 	void sendHttpWait(); | 	void needToSendAsync(); | ||||||
|  | 	void sendAnythingAsync(quint64 msWait); | ||||||
|  | 	void sendHttpWaitAsync(); | ||||||
|  | 	void sendPongAsync(quint64 msgId, quint64 pingId); | ||||||
|  | 	void sendMsgsStateInfoAsync(quint64 msgId, QByteArray data); | ||||||
|  | 	void resendAsync(quint64 msgId, quint64 msCanWait, bool forceContainer, bool sendMsgStateInfo); | ||||||
|  | 	void resendManyAsync(QVector<quint64> msgIds, quint64 msCanWait, bool forceContainer, bool sendMsgStateInfo); | ||||||
|  | 	void resendAllAsync(); | ||||||
| 
 | 
 | ||||||
| public slots: | public slots: | ||||||
| 
 | 
 | ||||||
|  | @ -344,8 +349,6 @@ public slots: | ||||||
| 
 | 
 | ||||||
| 	bool updateAuthKey(); | 	bool updateAuthKey(); | ||||||
| 
 | 
 | ||||||
| 	void sendPing(); |  | ||||||
| 
 |  | ||||||
| 	void onConfigLoaded(); | 	void onConfigLoaded(); | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
|  | @ -398,7 +401,8 @@ private: | ||||||
| 	mtpPingId pingId, toSendPingId; | 	mtpPingId pingId, toSendPingId; | ||||||
| 	mtpMsgId pingMsgId; | 	mtpMsgId pingMsgId; | ||||||
| 
 | 
 | ||||||
| 	mtpRequestId resend(mtpMsgId msgId, uint64 msCanWait = 0, bool forceContainer = false, bool sendMsgStateInfo = false); | 	void resend(quint64 msgId, quint64 msCanWait = 0, bool forceContainer = false, bool sendMsgStateInfo = false); | ||||||
|  | 	void resendMany(QVector<quint64> msgIds, quint64 msCanWait = 0, bool forceContainer = false, bool sendMsgStateInfo = false); | ||||||
| 
 | 
 | ||||||
| 	template <typename TRequest> | 	template <typename TRequest> | ||||||
| 	void sendRequestNotSecure(const TRequest &request); | 	void sendRequestNotSecure(const TRequest &request); | ||||||
|  | @ -457,6 +461,4 @@ private: | ||||||
| 	void authKeyCreated(); | 	void authKeyCreated(); | ||||||
| 	void clearAuthKeyData(); | 	void clearAuthKeyData(); | ||||||
| 
 | 
 | ||||||
| 	QTimer pinger; |  | ||||||
| 
 |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | @ -113,8 +113,8 @@ void MTProtoSession::start(int32 dcenter, uint32 connects) { | ||||||
| 			if (lock && dc->connectionInited()) { | 			if (lock && dc->connectionInited()) { | ||||||
| 				data.setLayerWasInited(true); | 				data.setLayerWasInited(true); | ||||||
| 			} | 			} | ||||||
| 			connect(dc.data(), SIGNAL(authKeyCreated()), this, SLOT(authKeyCreatedForDC())); | 			connect(dc.data(), SIGNAL(authKeyCreated()), this, SLOT(authKeyCreatedForDC()), Qt::QueuedConnection); | ||||||
| 			connect(dc.data(), SIGNAL(layerWasInited(bool)), this, SLOT(layerWasInitedForDC(bool))); | 			connect(dc.data(), SIGNAL(layerWasInited(bool)), this, SLOT(layerWasInitedForDC(bool)), Qt::QueuedConnection); | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  | @ -160,6 +160,20 @@ void MTProtoSession::sendHttpWait() { | ||||||
| 	send(MTPHttpWait(MTP_http_wait(MTP_int(100), MTP_int(30), MTP_int(25000))), RPCResponseHandler(), 50); | 	send(MTPHttpWait(MTP_http_wait(MTP_int(100), MTP_int(30), MTP_int(25000))), RPCResponseHandler(), 50); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void MTProtoSession::sendPong(quint64 msgId, quint64 pingId) { | ||||||
|  | 	send(MTP_pong(MTP_long(msgId), MTP_long(pingId))); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void MTProtoSession::sendMsgsStateInfo(quint64 msgId, QByteArray data) { | ||||||
|  | 	MTPMsgsStateInfo req(MTP_msgs_state_info(MTP_long(msgId), MTPstring())); | ||||||
|  | 	string &info(req._msgs_state_info().vinfo._string().v); | ||||||
|  | 	info.resize(data.size()); | ||||||
|  | 	if (!data.isEmpty()) { | ||||||
|  | 		memcpy(&info[0], data.constData(), data.size()); | ||||||
|  | 	} | ||||||
|  | 	send(req); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void MTProtoSession::checkRequestsByTimer() { | void MTProtoSession::checkRequestsByTimer() { | ||||||
| 	QVector<mtpMsgId> resendingIds; | 	QVector<mtpMsgId> resendingIds; | ||||||
| 	QVector<mtpMsgId> removingIds; // remove very old (10 minutes) containers and resend requests
 | 	QVector<mtpMsgId> removingIds; // remove very old (10 minutes) containers and resend requests
 | ||||||
|  | @ -308,7 +322,7 @@ QString MTProtoSession::transport() const { | ||||||
| 	return QString(); | 	return QString(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| mtpRequestId MTProtoSession::resend(mtpMsgId msgId, uint64 msCanWait, bool forceContainer, bool sendMsgStateInfo) { | mtpRequestId MTProtoSession::resend(quint64 msgId, quint64 msCanWait, bool forceContainer, bool sendMsgStateInfo) { | ||||||
| 	mtpRequest request; | 	mtpRequest request; | ||||||
| 	{ | 	{ | ||||||
| 		QWriteLocker locker(data.haveSentMutex()); | 		QWriteLocker locker(data.haveSentMutex()); | ||||||
|  | @ -348,6 +362,12 @@ mtpRequestId MTProtoSession::resend(mtpMsgId msgId, uint64 msCanWait, bool force | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void MTProtoSession::resendMany(QVector<quint64> msgIds, quint64 msCanWait, bool forceContainer, bool sendMsgStateInfo) { | ||||||
|  | 	for (int32 i = 0, l = msgIds.size(); i < l; ++i) { | ||||||
|  | 		resend(msgIds.at(i), msCanWait, forceContainer, sendMsgStateInfo); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void MTProtoSession::resendAll() { | void MTProtoSession::resendAll() { | ||||||
| 	QVector<mtpMsgId> toResend; | 	QVector<mtpMsgId> toResend; | ||||||
| 	{ | 	{ | ||||||
|  | @ -427,6 +447,7 @@ int32 MTProtoSession::getDC() const { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void MTProtoSession::tryToReceive() { | void MTProtoSession::tryToReceive() { | ||||||
|  | 	int32 cnt = 0; | ||||||
| 	while (true) { | 	while (true) { | ||||||
| 		mtpRequestId requestId; | 		mtpRequestId requestId; | ||||||
| 		mtpResponse response; | 		mtpResponse response; | ||||||
|  | @ -445,6 +466,7 @@ void MTProtoSession::tryToReceive() { | ||||||
| 		} else { | 		} else { | ||||||
| 			_mtp_internal::execCallback(requestId, response.constData(), response.constData() + response.size()); | 			_mtp_internal::execCallback(requestId, response.constData(), response.constData() + response.size()); | ||||||
| 		} | 		} | ||||||
|  | 		++cnt; | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -241,9 +241,6 @@ public: | ||||||
| 	int32 getState() const; | 	int32 getState() const; | ||||||
| 	QString transport() const; | 	QString transport() const; | ||||||
| 
 | 
 | ||||||
| 	mtpRequestId resend(mtpMsgId msgId, uint64 msCanWait = 0, bool forceContainer = false, bool sendMsgStateInfo = false); |  | ||||||
| 	void resendAll(); // after connection restart
 |  | ||||||
| 
 |  | ||||||
| 	void sendPrepared(const mtpRequest &request, uint64 msCanWait = 0, bool newRequest = true); // nulls msgId and seqNo in request, if newRequest = true
 | 	void sendPrepared(const mtpRequest &request, uint64 msCanWait = 0, bool newRequest = true); // nulls msgId and seqNo in request, if newRequest = true
 | ||||||
| 	void sendPreparedWithInit(const mtpRequest &request, uint64 msCanWait = 0); | 	void sendPreparedWithInit(const mtpRequest &request, uint64 msCanWait = 0); | ||||||
| 
 | 
 | ||||||
|  | @ -255,6 +252,10 @@ signals: | ||||||
| 
 | 
 | ||||||
| public slots: | public slots: | ||||||
| 
 | 
 | ||||||
|  | 	mtpRequestId resend(quint64 msgId, quint64 msCanWait = 0, bool forceContainer = false, bool sendMsgStateInfo = false); | ||||||
|  | 	void resendMany(QVector<quint64> msgIds, quint64 msCanWait, bool forceContainer, bool sendMsgStateInfo); | ||||||
|  | 	void resendAll(); // after connection restart
 | ||||||
|  | 
 | ||||||
| 	void authKeyCreatedForDC(); | 	void authKeyCreatedForDC(); | ||||||
| 	void layerWasInitedForDC(bool wasInited); | 	void layerWasInitedForDC(bool wasInited); | ||||||
| 
 | 
 | ||||||
|  | @ -264,8 +265,9 @@ public slots: | ||||||
| 	void onResetDone(); | 	void onResetDone(); | ||||||
| 
 | 
 | ||||||
| 	void sendAnything(quint64 msCanWait); | 	void sendAnything(quint64 msCanWait); | ||||||
| 
 |  | ||||||
| 	void sendHttpWait(); | 	void sendHttpWait(); | ||||||
|  | 	void sendPong(quint64 msgId, quint64 pingId); | ||||||
|  | 	void sendMsgsStateInfo(quint64 msgId, QByteArray data); | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
| 	 | 	 | ||||||
|  |  | ||||||
|  | @ -96,6 +96,8 @@ QUrl gUpdateURL = QUrl(qsl("http://tdesktop.com/linux/tupdates/current")); | ||||||
| #error Unknown platform | #error Unknown platform | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
|  | bool gContactsReceived = false; | ||||||
|  | 
 | ||||||
| void settingsParseArgs(int argc, char *argv[]) { | void settingsParseArgs(int argc, char *argv[]) { | ||||||
| 	if (cPlatform() == dbipMac) { | 	if (cPlatform() == dbipMac) { | ||||||
| 		gCustomNotifies = false; | 		gCustomNotifies = false; | ||||||
|  |  | ||||||
|  | @ -156,4 +156,6 @@ DeclareReadSetting(uint64, Instance); | ||||||
| DeclareReadSetting(DBIPlatform, Platform); | DeclareReadSetting(DBIPlatform, Platform); | ||||||
| DeclareReadSetting(QUrl, UpdateURL); | DeclareReadSetting(QUrl, UpdateURL); | ||||||
| 
 | 
 | ||||||
|  | DeclareSetting(bool, ContactsReceived); | ||||||
|  | 
 | ||||||
| void settingsParseArgs(int argc, char *argv[]); | void settingsParseArgs(int argc, char *argv[]); | ||||||
|  |  | ||||||
|  | @ -434,6 +434,7 @@ void Window::clearWidgets() { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void Window::setupIntro(bool anim) { | void Window::setupIntro(bool anim) { | ||||||
|  | 	cSetContactsReceived(false); | ||||||
| 	if (intro && (intro->animating() || intro->isVisible()) && !main) return; | 	if (intro && (intro->animating() || intro->isVisible()) && !main) return; | ||||||
| 
 | 
 | ||||||
| 	QPixmap bg = myGrab(this, QRect(0, st::titleHeight, width(), height() - st::titleHeight)); | 	QPixmap bg = myGrab(this, QRect(0, st::titleHeight, width(), height() - st::titleHeight)); | ||||||
|  | @ -636,6 +637,13 @@ void Window::hideLayer() { | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | bool Window::hideInnerLayer() { | ||||||
|  | 	if (layerBG) { | ||||||
|  | 		return layerBG->onInnerClose(); | ||||||
|  | 	} | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| bool Window::layerShown() { | bool Window::layerShown() { | ||||||
| 	return !!layerBG || !!_topWidget; | 	return !!layerBG || !!_topWidget; | ||||||
| } | } | ||||||
|  | @ -810,6 +818,12 @@ void Window::updateTrayMenu(bool force) { | ||||||
| #endif | #endif | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void Window::onShowAddContact() { | ||||||
|  | 	if (isHidden()) showFromTray(); | ||||||
|  | 
 | ||||||
|  | 	if (main) main->showAddContact(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void Window::onShowNewGroup() { | void Window::onShowNewGroup() { | ||||||
| 	if (isHidden()) showFromTray(); | 	if (isHidden()) showFromTray(); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -172,6 +172,7 @@ public: | ||||||
| 	void showLayer(LayeredWidget *w); | 	void showLayer(LayeredWidget *w); | ||||||
| 	void replaceLayer(LayeredWidget *w); | 	void replaceLayer(LayeredWidget *w); | ||||||
| 	void hideLayer(); | 	void hideLayer(); | ||||||
|  | 	bool hideInnerLayer(); | ||||||
| 
 | 
 | ||||||
| 	bool layerShown(); | 	bool layerShown(); | ||||||
| 
 | 
 | ||||||
|  | @ -240,6 +241,7 @@ public slots: | ||||||
| 	void notifyFire(); | 	void notifyFire(); | ||||||
| 	void updateTrayMenu(bool force = false); | 	void updateTrayMenu(bool force = false); | ||||||
| 
 | 
 | ||||||
|  | 	void onShowAddContact(); | ||||||
| 	void onShowNewGroup(); | 	void onShowNewGroup(); | ||||||
| 	void onLogout(); | 	void onLogout(); | ||||||
| 	void onLogoutSure(); | 	void onLogoutSure(); | ||||||
|  |  | ||||||
|  | @ -1742,7 +1742,6 @@ | ||||||
|     </CustomBuild> |     </CustomBuild> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <Image Include="sourcefiles\art\icon2.ico" /> |  | ||||||
|     <Image Include="SourceFiles\art\icon256.ico" /> |     <Image Include="SourceFiles\art\icon256.ico" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|  |  | ||||||
|  | @ -1023,7 +1023,6 @@ | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <Image Include="SourceFiles\art\icon256.ico" /> |     <Image Include="SourceFiles\art\icon256.ico" /> | ||||||
|     <Image Include="sourcefiles\art\icon2.ico" /> |  | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <ResourceCompile Include="Telegram.rc" /> |     <ResourceCompile Include="Telegram.rc" /> | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue