mirror of https://github.com/procxx/kepka.git
				
				
				
			exif fixed, media overview redesign done
This commit is contained in:
		
							parent
							
								
									5108685123
								
							
						
					
					
						commit
						af9908f076
					
				|  | @ -2315,36 +2315,45 @@ namespace App { | ||||||
|         if (!format) { |         if (!format) { | ||||||
|             format = &tmpFormat; |             format = &tmpFormat; | ||||||
|         } |         } | ||||||
|  | 		{ | ||||||
| 			QImageReader reader(&buffer, *format); | 			QImageReader reader(&buffer, *format); | ||||||
|  | #if QT_VERSION >= QT_VERSION_CHECK(5, 5, 0) | ||||||
|  | 			reader.setAutoTransform(true); | ||||||
|  | #endif | ||||||
| 			if (animated) *animated = reader.supportsAnimation() && reader.imageCount() > 1; | 			if (animated) *animated = reader.supportsAnimation() && reader.imageCount() > 1; | ||||||
|  | 			QByteArray fmt = reader.format(); | ||||||
|  | 			if (!fmt.isEmpty()) *format = fmt; | ||||||
| 			if (!reader.read(&result)) { | 			if (!reader.read(&result)) { | ||||||
| 				return QImage(); | 				return QImage(); | ||||||
| 			} | 			} | ||||||
| 
 | 			fmt = reader.format(); | ||||||
|  | 			if (!fmt.isEmpty()) *format = fmt; | ||||||
|  | 		} | ||||||
| 		buffer.seek(0); | 		buffer.seek(0); | ||||||
|         *format = reader.format(); |         QString fmt = QString::fromUtf8(*format).toLower(); | ||||||
|         QString fmt = QString::fromUtf8(*format).toLower() ; |  | ||||||
| 		if (fmt == "jpg" || fmt == "jpeg") { | 		if (fmt == "jpg" || fmt == "jpeg") { | ||||||
| 			//ExifData *exifData = exif_data_new_from_data((const uchar*)(data.constData()), data.size());
 | #if QT_VERSION < QT_VERSION_CHECK(5, 5, 0) | ||||||
| 			//if (exifData) {
 | 			ExifData *exifData = exif_data_new_from_data((const uchar*)(data.constData()), data.size()); | ||||||
| 			//	ExifByteOrder byteOrder = exif_data_get_byte_order(exifData);
 | 			if (exifData) { | ||||||
| 			//	ExifEntry *exifEntry = exif_data_get_entry(exifData, EXIF_TAG_ORIENTATION);
 | 				ExifByteOrder byteOrder = exif_data_get_byte_order(exifData); | ||||||
| 			//	if (exifEntry) {
 | 				ExifEntry *exifEntry = exif_data_get_entry(exifData, EXIF_TAG_ORIENTATION); | ||||||
| 			//		QTransform orientationFix;
 | 				if (exifEntry) { | ||||||
| 			//		int orientation = exif_get_short(exifEntry->data, byteOrder);
 | 					QTransform orientationFix; | ||||||
| 			//		switch (orientation) {
 | 					int orientation = exif_get_short(exifEntry->data, byteOrder); | ||||||
| 			//		case 2: orientationFix = QTransform(-1, 0, 0, 1, 0, 0); break;
 | 					switch (orientation) { | ||||||
| 			//		case 3: orientationFix = QTransform(-1, 0, 0, -1, 0, 0); break;
 | 					case 2: orientationFix = QTransform(-1, 0, 0, 1, 0, 0); break; | ||||||
| 			//		case 4: orientationFix = QTransform(1, 0, 0, -1, 0, 0); break;
 | 					case 3: orientationFix = QTransform(-1, 0, 0, -1, 0, 0); break; | ||||||
| 			//		case 5: orientationFix = QTransform(0, -1, -1, 0, 0, 0); break;
 | 					case 4: orientationFix = QTransform(1, 0, 0, -1, 0, 0); break; | ||||||
| 			//		case 6: orientationFix = QTransform(0, 1, -1, 0, 0, 0); break;
 | 					case 5: orientationFix = QTransform(0, -1, -1, 0, 0, 0); break; | ||||||
| 			//		case 7: orientationFix = QTransform(0, 1, 1, 0, 0, 0); break;
 | 					case 6: orientationFix = QTransform(0, 1, -1, 0, 0, 0); break; | ||||||
| 			//		case 8: orientationFix = QTransform(0, -1, 1, 0, 0, 0); break;
 | 					case 7: orientationFix = QTransform(0, 1, 1, 0, 0, 0); break; | ||||||
| 			//		}
 | 					case 8: orientationFix = QTransform(0, -1, 1, 0, 0, 0); break; | ||||||
| 			//		result = result.transformed(orientationFix);
 | 					} | ||||||
| 			//	}
 | 					result = result.transformed(orientationFix); | ||||||
| 			//	exif_data_free(exifData);
 | 				} | ||||||
| 			//}
 | 				exif_data_free(exifData); | ||||||
|  | 			} | ||||||
|  | #endif | ||||||
| 		} else if (opaque && result.hasAlphaChannel()) { | 		} else if (opaque && result.hasAlphaChannel()) { | ||||||
| 			QImage solid(result.width(), result.height(), QImage::Format_ARGB32_Premultiplied); | 			QImage solid(result.width(), result.height(), QImage::Format_ARGB32_Premultiplied); | ||||||
| 			solid.fill(st::white->c); | 			solid.fill(st::white->c); | ||||||
|  |  | ||||||
|  | @ -363,6 +363,9 @@ private: | ||||||
| 		delete _reader; | 		delete _reader; | ||||||
| 		initDevice(); | 		initDevice(); | ||||||
| 		_reader = new QImageReader(_device); | 		_reader = new QImageReader(_device); | ||||||
|  | #if QT_VERSION >= QT_VERSION_CHECK(5, 5, 0) | ||||||
|  | 		_reader->setAutoTransform(true); | ||||||
|  | #endif | ||||||
| 		if (!_reader->canRead() || !_reader->supportsAnimation()) { | 		if (!_reader->canRead() || !_reader->supportsAnimation()) { | ||||||
| 			return false; | 			return false; | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | @ -138,6 +138,9 @@ protected: | ||||||
| 	void doRestore() const {  | 	void doRestore() const {  | ||||||
| 		QBuffer buffer(&saved); | 		QBuffer buffer(&saved); | ||||||
| 		QImageReader reader(&buffer, format); | 		QImageReader reader(&buffer, format); | ||||||
|  | #if QT_VERSION >= QT_VERSION_CHECK(5, 5, 0) | ||||||
|  | 		reader.setAutoTransform(true); | ||||||
|  | #endif | ||||||
| 		data = QPixmap::fromImageReader(&reader, Qt::ColorOnly); | 		data = QPixmap::fromImageReader(&reader, Qt::ColorOnly); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | @ -206,6 +209,9 @@ protected: | ||||||
| 	void doRestore() const {  | 	void doRestore() const {  | ||||||
| 		QBuffer buffer(&saved); | 		QBuffer buffer(&saved); | ||||||
| 		QImageReader reader(&buffer, format); | 		QImageReader reader(&buffer, format); | ||||||
|  | #if QT_VERSION >= QT_VERSION_CHECK(5, 5, 0) | ||||||
|  | 		reader.setAutoTransform(true); | ||||||
|  | #endif | ||||||
| 		data = QPixmap::fromImageReader(&reader, Qt::ColorOnly); | 		data = QPixmap::fromImageReader(&reader, Qt::ColorOnly); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -323,6 +323,8 @@ public: | ||||||
| 	virtual bool fullDisplayed() const { | 	virtual bool fullDisplayed() const { | ||||||
| 		return true; | 		return true; | ||||||
| 	} | 	} | ||||||
|  | 	virtual void setFullDisplayed(bool full) { | ||||||
|  | 	} | ||||||
| 	virtual QString encoded() const { | 	virtual QString encoded() const { | ||||||
| 		return QString(); | 		return QString(); | ||||||
| 	} | 	} | ||||||
|  | @ -364,6 +366,10 @@ public: | ||||||
| 		return _fullDisplayed; | 		return _fullDisplayed; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	void setFullDisplayed(bool full) { | ||||||
|  | 		_fullDisplayed = full; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	QString encoded() const { | 	QString encoded() const { | ||||||
| 		QUrl u(_url), good(u.isValid() ? u.toEncoded() : QString()); | 		QUrl u(_url), good(u.isValid() ? u.toEncoded() : QString()); | ||||||
| 		QString result(good.isValid() ? QString::fromUtf8(good.toEncoded()) : _url); | 		QString result(good.isValid() ? QString::fromUtf8(good.toEncoded()) : _url); | ||||||
|  |  | ||||||
|  | @ -5350,6 +5350,15 @@ void HistoryWebPage::getState(TextLinkPtr &lnk, HistoryCursorState &state, int32 | ||||||
| 			int32 attachLeft = lshift - bubble.left(), attachTop = tshift - bubble.top(); | 			int32 attachLeft = lshift - bubble.left(), attachTop = tshift - bubble.top(); | ||||||
| 			if (rtl()) attachLeft = _width - attachLeft - _attach->currentWidth(); | 			if (rtl()) attachLeft = _width - attachLeft - _attach->currentWidth(); | ||||||
| 			_attach->getState(lnk, state, x - attachLeft, y - attachTop, parent); | 			_attach->getState(lnk, state, x - attachLeft, y - attachTop, parent); | ||||||
|  | 			if (lnk && _data->photo) { | ||||||
|  | 				if (_data->type == WebPageProfile || _data->type == WebPageVideo) { | ||||||
|  | 					lnk = _openl; | ||||||
|  | 				} else if (_data->type == WebPagePhoto || _data->siteName == qstr("Twitter") || _data->siteName == qstr("Facebook")) { | ||||||
|  | 					// leave photo link
 | ||||||
|  | 				} else { | ||||||
|  | 					lnk = _openl; | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  | @ -5538,6 +5547,9 @@ void ImageLinkManager::onFinished(QNetworkReply *reply) { | ||||||
| 			{ | 			{ | ||||||
| 				QBuffer buffer(&data); | 				QBuffer buffer(&data); | ||||||
| 				QImageReader reader(&buffer); | 				QImageReader reader(&buffer); | ||||||
|  | #if QT_VERSION >= QT_VERSION_CHECK(5, 5, 0) | ||||||
|  | 				reader.setAutoTransform(true); | ||||||
|  | #endif | ||||||
| 				thumb = QPixmap::fromImageReader(&reader, Qt::ColorOnly); | 				thumb = QPixmap::fromImageReader(&reader, Qt::ColorOnly); | ||||||
| 				format = reader.format(); | 				format = reader.format(); | ||||||
| 				thumb.setDevicePixelRatio(cRetinaFactor()); | 				thumb.setDevicePixelRatio(cRetinaFactor()); | ||||||
|  |  | ||||||
|  | @ -534,8 +534,9 @@ void LayoutOverviewVideo::updateStatusText() const { | ||||||
| 
 | 
 | ||||||
| LayoutOverviewAudio::LayoutOverviewAudio(AudioData *audio, HistoryItem *parent) : LayoutAbstractFileItem(parent) | LayoutOverviewAudio::LayoutOverviewAudio(AudioData *audio, HistoryItem *parent) : LayoutAbstractFileItem(parent) | ||||||
| , _data(audio) { | , _data(audio) { | ||||||
|  | 	setLinks(new AudioOpenLink(_data), new AudioSaveLink(_data), new AudioCancelLink(_data)); | ||||||
| 	updateName(); | 	updateName(); | ||||||
| 	_details.setText(st::normalFont, lng_date_and_duration(lt_date, textcmdLink(1, langDateTime(date(_data->date))), lt_duration, formatDurationText(_data->duration)), _textNameOptions); | 	_details.setText(st::normalFont, lng_date_and_duration(lt_date, textcmdLink(1, langDateTime(date(_data->date))), lt_duration, formatDurationText(_data->duration)), _defaultOptions); | ||||||
| 	_details.setLink(1, TextLinkPtr(new MessageLink(parent))); | 	_details.setLink(1, TextLinkPtr(new MessageLink(parent))); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -547,6 +548,11 @@ void LayoutOverviewAudio::initDimensions() { | ||||||
| void LayoutOverviewAudio::paint(Painter &p, const QRect &clip, uint32 selection, const PaintContext *context) const { | void LayoutOverviewAudio::paint(Painter &p, const QRect &clip, uint32 selection, const PaintContext *context) const { | ||||||
| 	bool selected = (selection == FullSelection); | 	bool selected = (selection == FullSelection); | ||||||
| 	bool already = !_data->already().isEmpty(), hasdata = !_data->data.isEmpty(); | 	bool already = !_data->already().isEmpty(), hasdata = !_data->data.isEmpty(); | ||||||
|  | 
 | ||||||
|  | 	if (!_data->loader && _data->status == FileReady && !already && !hasdata && _data->size < AudioVoiceMsgInMemory) { | ||||||
|  | 		_data->save(QString()); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	if (_data->loader) { | 	if (_data->loader) { | ||||||
| 		ensureRadial(); | 		ensureRadial(); | ||||||
| 		if (!_radial->animating()) { | 		if (!_radial->animating()) { | ||||||
|  | @ -563,11 +569,16 @@ void LayoutOverviewAudio::paint(Painter &p, const QRect &clip, uint32 selection, | ||||||
| 
 | 
 | ||||||
| 	int32 nameleft = 0, nametop = 0, nameright = 0, statustop = 0, datetop = -1; | 	int32 nameleft = 0, nametop = 0, nameright = 0, statustop = 0, datetop = -1; | ||||||
| 
 | 
 | ||||||
| 	nameleft = st::msgFileSize + st::msgFilePadding.right(); | 	nameleft = st::msgFilePadding.left() + st::msgFileSize + st::msgFilePadding.right(); | ||||||
|  | 	nameright = st::msgFilePadding.left(); | ||||||
| 	nametop = st::msgFileNameTop; | 	nametop = st::msgFileNameTop; | ||||||
| 	statustop = st::msgFileStatusTop; | 	statustop = st::msgFileStatusTop; | ||||||
| 
 | 
 | ||||||
| 	QRect inner(rtlrect(0, st::msgFilePadding.top(), st::msgFileSize, st::msgFileSize, _width)); | 	if (selected) { | ||||||
|  | 		p.fillRect(clip.intersected(QRect(0, 0, _width, _height)), st::msgInBgSelected); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	QRect inner(rtlrect(st::msgFilePadding.left(), st::msgFilePadding.top(), st::msgFileSize, st::msgFileSize, _width)); | ||||||
| 	if (clip.intersects(inner)) { | 	if (clip.intersects(inner)) { | ||||||
| 		p.setPen(Qt::NoPen); | 		p.setPen(Qt::NoPen); | ||||||
| 		if (selected) { | 		if (selected) { | ||||||
|  | @ -576,7 +587,7 @@ void LayoutOverviewAudio::paint(Painter &p, const QRect &clip, uint32 selection, | ||||||
| 			float64 over = a_iconOver.current(); | 			float64 over = a_iconOver.current(); | ||||||
| 			p.setBrush(style::interpolate(st::msgFileInBg, st::msgFileInBgOver, over)); | 			p.setBrush(style::interpolate(st::msgFileInBg, st::msgFileInBgOver, over)); | ||||||
| 		} else { | 		} else { | ||||||
| 			bool over = textlnkDrawOver(already ? _openl : (_data->loader ? _cancell : _savel)); | 			bool over = textlnkDrawOver((already || hasdata) ? _openl : (_data->loader ? _cancell : _savel)); | ||||||
| 			p.setBrush(over ? st::msgFileInBgOver : st::msgFileInBg); | 			p.setBrush(over ? st::msgFileInBgOver : st::msgFileInBg); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
|  | @ -611,7 +622,7 @@ void LayoutOverviewAudio::paint(Painter &p, const QRect &clip, uint32 selection, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (clip.intersects(rtlrect(nameleft, statustop, namewidth, st::normalFont->height, _width))) { | 	if (clip.intersects(rtlrect(nameleft, statustop, namewidth, st::normalFont->height, _width))) { | ||||||
| 		p.setPen(st::mediaInFg); | 		p.setPen(selected ? st::msgInDateFgSelected : st::mediaInFg); | ||||||
| 		_details.drawLeftElided(p, nameleft, statustop, namewidth, _width); | 		_details.drawLeftElided(p, nameleft, statustop, namewidth, _width); | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  | @ -623,15 +634,21 @@ void LayoutOverviewAudio::getState(TextLinkPtr &link, HistoryCursorState &cursor | ||||||
| 
 | 
 | ||||||
| 	int32 nameleft = 0, nametop = 0, nameright = 0, statustop = 0, datetop = 0; | 	int32 nameleft = 0, nametop = 0, nameright = 0, statustop = 0, datetop = 0; | ||||||
| 
 | 
 | ||||||
| 	nameleft = st::msgFileSize + st::msgFilePadding.right(); | 	nameleft = st::msgFilePadding.left() + st::msgFileSize + st::msgFilePadding.right(); | ||||||
|  | 	nameright = st::msgFilePadding.left(); | ||||||
| 	nametop = st::msgFileNameTop; | 	nametop = st::msgFileNameTop; | ||||||
| 	statustop = st::msgFileStatusTop; | 	statustop = st::msgFileStatusTop; | ||||||
| 
 | 
 | ||||||
| 	QRect inner(rtlrect(0, st::msgFilePadding.top(), st::msgFileSize, st::msgFileSize, _width)); | 	QRect inner(rtlrect(st::msgFilePadding.left(), st::msgFilePadding.top(), st::msgFileSize, st::msgFileSize, _width)); | ||||||
| 	if (inner.contains(x, y)) { | 	if (inner.contains(x, y)) { | ||||||
| 		link = (already || hasdata) ? _openl : ((_data->loader || _data->status == FileUploading) ? _cancell : _savel); | 		link = (already || hasdata) ? _openl : ((_data->loader || _data->status == FileUploading) ? _cancell : _savel); | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
|  | 	if (rtlrect(nameleft, statustop, _width - nameleft - nameright, st::normalFont->height, _width).contains(x, y)) { | ||||||
|  | 		bool inText = false; | ||||||
|  | 		_details.getStateLeft(link, inText, x - nameleft, y - statustop, _width, _width); | ||||||
|  | 		cursor = inText ? HistoryInTextCursorState : HistoryDefaultCursorState; | ||||||
|  | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void LayoutOverviewAudio::updateName() const { | void LayoutOverviewAudio::updateName() const { | ||||||
|  | @ -672,8 +689,6 @@ bool LayoutOverviewAudio::updateStatusText() const { | ||||||
| 	} | 	} | ||||||
| 	if (statusSize != _statusSize) { | 	if (statusSize != _statusSize) { | ||||||
| 		setStatusSize(statusSize, _data->size, _data->duration, realDuration); | 		setStatusSize(statusSize, _data->size, _data->duration, realDuration); | ||||||
| 		_details.setText(st::normalFont, lng_date_and_duration(lt_date, textcmdLink(1, langDateTime(date(_data->date))), lt_duration, formatDurationText(_data->duration)), _textNameOptions); |  | ||||||
| 		_details.setLink(1, TextLinkPtr(new MessageLink(_parent))); |  | ||||||
| 	} | 	} | ||||||
| 	return showPause; | 	return showPause; | ||||||
| } | } | ||||||
|  | @ -734,11 +749,16 @@ void LayoutOverviewDocument::paint(Painter &p, const QRect &clip, uint32 selecti | ||||||
| 	bool wthumb = withThumb(); | 	bool wthumb = withThumb(); | ||||||
| 
 | 
 | ||||||
| 	if (_data->song()) { | 	if (_data->song()) { | ||||||
| 		nameleft = st::msgFileSize + st::msgFilePadding.right(); | 		nameleft = st::msgFilePadding.left() + st::msgFileSize + st::msgFilePadding.right(); | ||||||
|  | 		nameright = st::msgFilePadding.left(); | ||||||
| 		nametop = st::msgFileNameTop; | 		nametop = st::msgFileNameTop; | ||||||
| 		statustop = st::msgFileStatusTop; | 		statustop = st::msgFileStatusTop; | ||||||
| 
 | 
 | ||||||
| 		QRect inner(rtlrect(0, st::msgFilePadding.top(), st::msgFileSize, st::msgFileSize, _width)); | 		if (selected) { | ||||||
|  | 			p.fillRect(QRect(0, 0, _width, _height), st::msgInBgSelected); | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		QRect inner(rtlrect(st::msgFilePadding.left(), st::msgFilePadding.top(), st::msgFileSize, st::msgFileSize, _width)); | ||||||
| 		if (clip.intersects(inner)) { | 		if (clip.intersects(inner)) { | ||||||
| 			p.setPen(Qt::NoPen); | 			p.setPen(Qt::NoPen); | ||||||
| 			if (selected) { | 			if (selected) { | ||||||
|  | @ -890,11 +910,12 @@ void LayoutOverviewDocument::getState(TextLinkPtr &link, HistoryCursorState &cur | ||||||
| 	bool wthumb = withThumb(); | 	bool wthumb = withThumb(); | ||||||
| 
 | 
 | ||||||
| 	if (_data->song()) { | 	if (_data->song()) { | ||||||
| 		nameleft = st::msgFileSize + st::msgFilePadding.right(); | 		nameleft = st::msgFilePadding.left() + st::msgFileSize + st::msgFilePadding.right(); | ||||||
|  | 		nameright = st::msgFilePadding.left(); | ||||||
| 		nametop = st::msgFileNameTop; | 		nametop = st::msgFileNameTop; | ||||||
| 		statustop = st::msgFileStatusTop; | 		statustop = st::msgFileStatusTop; | ||||||
| 
 | 
 | ||||||
| 		QRect inner(rtlrect(0, st::msgFilePadding.top(), st::msgFileSize, st::msgFileSize, _width)); | 		QRect inner(rtlrect(st::msgFilePadding.left(), st::msgFilePadding.top(), st::msgFileSize, st::msgFileSize, _width)); | ||||||
| 		if (inner.contains(x, y)) { | 		if (inner.contains(x, y)) { | ||||||
| 			link = (already || hasdata) ? _openl : ((_data->loader || _data->status == FileUploading) ? _cancell : _savel); | 			link = (already || hasdata) ? _openl : ((_data->loader || _data->status == FileUploading) ? _cancell : _savel); | ||||||
| 			return; | 			return; | ||||||
|  | @ -972,3 +993,265 @@ bool LayoutOverviewDocument::updateStatusText() const { | ||||||
| 	} | 	} | ||||||
| 	return showPause; | 	return showPause; | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | namespace { | ||||||
|  | 	ITextLink *linkFromUrl(const QString &url) { | ||||||
|  | 		int32 at = url.indexOf('@'), slash = url.indexOf('/'); | ||||||
|  | 		if ((at > 0) && (slash < 0 || slash > at)) { | ||||||
|  | 			return new EmailLink(url); | ||||||
|  | 		} | ||||||
|  | 		return new TextLink(url); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | LayoutOverviewLink::LayoutOverviewLink(HistoryMedia *media, HistoryItem *parent) : LayoutMediaItem(parent) | ||||||
|  | , _titlew(0) | ||||||
|  | , _page(0) | ||||||
|  | , _pixw(0) | ||||||
|  | , _pixh(0) | ||||||
|  | , _text(st::msgMinWidth) { | ||||||
|  | 	QString text = _parent->originalText(); | ||||||
|  | 	EntitiesInText entities = _parent->originalEntities(); | ||||||
|  | 
 | ||||||
|  | 	int32 from = 0, till = text.size(), lnk = entities.size(); | ||||||
|  | 	for (int32 i = 0; i < lnk; ++i) { | ||||||
|  | 		if (entities[i].type != EntityInTextUrl && entities[i].type != EntityInTextCustomUrl && entities[i].type != EntityInTextEmail) { | ||||||
|  | 			continue; | ||||||
|  | 		} | ||||||
|  | 		QString u = entities[i].text, t = text.mid(entities[i].offset, entities[i].length); | ||||||
|  | 		_links.push_back(Link(u.isEmpty() ? t : u, t)); | ||||||
|  | 	} | ||||||
|  | 	while (lnk > 0 && till > from) { | ||||||
|  | 		--lnk; | ||||||
|  | 		if (entities[lnk].type != EntityInTextUrl && entities[lnk].type != EntityInTextCustomUrl && entities[lnk].type != EntityInTextEmail) { | ||||||
|  | 			++lnk; | ||||||
|  | 			break; | ||||||
|  | 		} | ||||||
|  | 		int32 afterLinkStart = entities[lnk].offset + entities[lnk].length; | ||||||
|  | 		if (till > afterLinkStart) { | ||||||
|  | 			if (!QRegularExpression(qsl("^[,.\\s_=+\\-;:`'\"\\(\\)\\[\\]\\{\\}<>*&^%\\$#@!\\\\/]+$")).match(text.mid(afterLinkStart, till - afterLinkStart)).hasMatch()) { | ||||||
|  | 				++lnk; | ||||||
|  | 				break; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		till = entities[lnk].offset; | ||||||
|  | 	} | ||||||
|  | 	if (!lnk) { | ||||||
|  | 		if (QRegularExpression(qsl("^[,.\\s\\-;:`'\"\\(\\)\\[\\]\\{\\}<>*&^%\\$#@!\\\\/]+$")).match(text.mid(from, till - from)).hasMatch()) { | ||||||
|  | 			till = from; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	_page = (media && media->type() == MediaTypeWebPage) ? static_cast<HistoryWebPage*>(media)->webpage() : 0; | ||||||
|  | 	if (_page) { | ||||||
|  | 		if (_page->doc) { | ||||||
|  | 			_photol = TextLinkPtr(new DocumentOpenLink(_page->doc)); | ||||||
|  | 		} else if (_page->photo) { | ||||||
|  | 			if (_page->type == WebPageProfile || _page->type == WebPageVideo) { | ||||||
|  | 				_photol = TextLinkPtr(linkFromUrl(_page->url)); | ||||||
|  | 			} else if (_page->type == WebPagePhoto || _page->siteName == qstr("Twitter") || _page->siteName == qstr("Facebook")) { | ||||||
|  | 				_photol = TextLinkPtr(new PhotoLink(_page->photo)); | ||||||
|  | 			} else { | ||||||
|  | 				_photol = TextLinkPtr(linkFromUrl(_page->url)); | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			_photol = TextLinkPtr(linkFromUrl(_page->url)); | ||||||
|  | 		} | ||||||
|  | 	} else if (!_links.isEmpty()) { | ||||||
|  | 		_photol = TextLinkPtr(linkFromUrl(_links.at(0).lnk->text())); | ||||||
|  | 	} | ||||||
|  | 	if (from >= till && _page) { | ||||||
|  | 		text = _page->description; | ||||||
|  | 		from = 0; | ||||||
|  | 		till = text.size(); | ||||||
|  | 	} | ||||||
|  | 	if (till > from) { | ||||||
|  | 		TextParseOptions opts = { TextParseMultiline, int32(st::linksMaxWidth), 3 * st::normalFont->height, Qt::LayoutDirectionAuto }; | ||||||
|  | 		_text.setText(st::normalFont, text.mid(from, till - from), opts); | ||||||
|  | 	} | ||||||
|  | 	int32 tw = 0, th = 0; | ||||||
|  | 	if (_page && _page->photo) { | ||||||
|  | 		if (!_page->photo->full->loaded()) _page->photo->thumb->load(false, false); | ||||||
|  | 
 | ||||||
|  | 		tw = convertScale(_page->photo->thumb->width()); | ||||||
|  | 		th = convertScale(_page->photo->thumb->height()); | ||||||
|  | 	} else if (_page && _page->doc) { | ||||||
|  | 		if (!_page->doc->thumb->loaded()) _page->doc->thumb->load(false, false); | ||||||
|  | 
 | ||||||
|  | 		tw = convertScale(_page->doc->thumb->width()); | ||||||
|  | 		th = convertScale(_page->doc->thumb->height()); | ||||||
|  | 	} | ||||||
|  | 	if (tw > st::dlgPhotoSize) { | ||||||
|  | 		if (th > tw) { | ||||||
|  | 			th = th * st::dlgPhotoSize / tw; | ||||||
|  | 			tw = st::dlgPhotoSize; | ||||||
|  | 		} else if (th > st::dlgPhotoSize) { | ||||||
|  | 			tw = tw * st::dlgPhotoSize / th; | ||||||
|  | 			th = st::dlgPhotoSize; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	_pixw = qMax(tw, 1); | ||||||
|  | 	_pixh = qMax(th, 1); | ||||||
|  | 
 | ||||||
|  | 	if (_page) { | ||||||
|  | 		_title = _page->title; | ||||||
|  | 	} | ||||||
|  | 	QVector<QStringRef> parts = (_page ? _page->url : (_links.isEmpty() ? QString() : _links.at(0).lnk->text())).splitRef('/'); | ||||||
|  | 	if (!parts.isEmpty()) { | ||||||
|  | 		QStringRef domain = parts.at(0); | ||||||
|  | 		if (parts.size() > 2 && domain.endsWith(':') && parts.at(1).isEmpty()) { // http:// and others
 | ||||||
|  | 			domain = parts.at(2); | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		parts = domain.split('@').back().split('.'); | ||||||
|  | 		if (parts.size() > 1) { | ||||||
|  | 			_letter = parts.at(parts.size() - 2).at(0).toUpper(); | ||||||
|  | 			if (_title.isEmpty()) { | ||||||
|  | 				_title.reserve(parts.at(parts.size() - 2).size()); | ||||||
|  | 				_title.append(_letter).append(parts.at(parts.size() - 2).mid(1)); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	_titlew = st::semiboldFont->width(_title); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void LayoutOverviewLink::initDimensions() { | ||||||
|  | 	_maxw = st::linksMaxWidth; | ||||||
|  | 	_minh = 0; | ||||||
|  | 	if (!_title.isEmpty()) { | ||||||
|  | 		_minh += st::semiboldFont->height; | ||||||
|  | 	} | ||||||
|  | 	if (!_text.isEmpty()) { | ||||||
|  | 		_minh += qMin(3 * st::normalFont->height, _text.countHeight(_maxw - st::dlgPhotoSize - st::dlgPhotoPadding)); | ||||||
|  | 	} | ||||||
|  | 	_minh += _links.size() * st::normalFont->height; | ||||||
|  | 	_minh = qMax(_minh, int32(st::dlgPhotoSize)) + st::linksMargin * 2 + st::linksBorder; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int32 LayoutOverviewLink::resizeGetHeight(int32 width) { | ||||||
|  | 	_width = qMin(width, _maxw); | ||||||
|  | 	int32 w = _width - st::dlgPhotoSize - st::dlgPhotoPadding; | ||||||
|  | 	for (int32 i = 0, l = _links.size(); i < l; ++i) { | ||||||
|  | 		_links.at(i).lnk->setFullDisplayed(w >= _links.at(i).width); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	_height = 0; | ||||||
|  | 	if (!_title.isEmpty()) { | ||||||
|  | 		_height += st::semiboldFont->height; | ||||||
|  | 	} | ||||||
|  | 	if (!_text.isEmpty()) { | ||||||
|  | 		_height += qMin(3 * st::normalFont->height, _text.countHeight(_width - st::dlgPhotoSize - st::dlgPhotoPadding)); | ||||||
|  | 	} | ||||||
|  | 	_height += _links.size() * st::normalFont->height; | ||||||
|  | 	_height = qMax(_height, int32(st::dlgPhotoSize)) + st::linksMargin * 2 + st::linksBorder; | ||||||
|  | 	return _height; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void LayoutOverviewLink::paint(Painter &p, const QRect &clip, uint32 selection, const PaintContext *context) const { | ||||||
|  | 	int32 left = st::dlgPhotoSize + st::dlgPhotoPadding, top = st::linksMargin + st::linksBorder, w = _width - left; | ||||||
|  | 	if (clip.intersects(rtlrect(0, top, st::dlgPhotoSize, st::dlgPhotoSize, _width))) { | ||||||
|  | 		if (_page && _page->photo) { | ||||||
|  | 			QPixmap pix; | ||||||
|  | 			if (_page->photo->full->loaded()) { | ||||||
|  | 				pix = _page->photo->full->pixSingle(_pixw, _pixh, st::dlgPhotoSize, st::dlgPhotoSize); | ||||||
|  | 			} else if (_page->photo->medium->loaded()) { | ||||||
|  | 				pix = _page->photo->medium->pixSingle(_pixw, _pixh, st::dlgPhotoSize, st::dlgPhotoSize); | ||||||
|  | 			} else { | ||||||
|  | 				pix = _page->photo->thumb->pixSingle(_pixw, _pixh, st::dlgPhotoSize, st::dlgPhotoSize); | ||||||
|  | 			} | ||||||
|  | 			p.drawPixmapLeft(0, top, _width, pix); | ||||||
|  | 		} else if (_page && _page->doc && !_page->doc->thumb->isNull()) { | ||||||
|  | 			p.drawPixmapLeft(0, top, _width, _page->doc->thumb->pixSingle(_pixw, _pixh, st::dlgPhotoSize, st::dlgPhotoSize)); | ||||||
|  | 		} else { | ||||||
|  | 			int32 index = _letter.isEmpty() ? 0 : (_letter.at(0).unicode() % 4); | ||||||
|  | 			switch (index) { | ||||||
|  | 			case 0: App::roundRect(p, rtlrect(0, top, st::dlgPhotoSize, st::dlgPhotoSize, _width), st::msgFileRedColor, DocRedCorners); break; | ||||||
|  | 			case 1: App::roundRect(p, rtlrect(0, top, st::dlgPhotoSize, st::dlgPhotoSize, _width), st::msgFileYellowColor, DocYellowCorners); break; | ||||||
|  | 			case 2: App::roundRect(p, rtlrect(0, top, st::dlgPhotoSize, st::dlgPhotoSize, _width), st::msgFileGreenColor, DocGreenCorners); break; | ||||||
|  | 			case 3: App::roundRect(p, rtlrect(0, top, st::dlgPhotoSize, st::dlgPhotoSize, _width), st::msgFileBlueColor, DocBlueCorners); break; | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			if (!_letter.isEmpty()) { | ||||||
|  | 				p.setFont(st::linksLetterFont->f); | ||||||
|  | 				p.setPen(st::white->p); | ||||||
|  | 				p.drawText(rtlrect(0, top, st::dlgPhotoSize, st::dlgPhotoSize, _width), _letter, style::al_center); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if (selection == FullSelection) { | ||||||
|  | 			App::roundRect(p, rtlrect(0, top, st::dlgPhotoSize, st::dlgPhotoSize, _width), st::overviewPhotoSelectOverlay, PhotoSelectOverlayCorners); | ||||||
|  | 			p.drawSpriteLeft(QPoint(st::dlgPhotoSize - st::linksPhotoCheck.pxWidth(), top + st::dlgPhotoSize - st::linksPhotoCheck.pxHeight()), _width, st::linksPhotoChecked); | ||||||
|  | 		} else if (context->selecting) { | ||||||
|  | 			p.drawSpriteLeft(QPoint(st::dlgPhotoSize - st::linksPhotoCheck.pxWidth(), top + st::dlgPhotoSize - st::linksPhotoCheck.pxHeight()), _width, st::linksPhotoCheck); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (!_title.isEmpty() && _text.isEmpty() && _links.size() == 1) { | ||||||
|  | 		top += (st::dlgPhotoSize - st::semiboldFont->height - st::normalFont->height) / 2; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	p.setPen(st::black); | ||||||
|  | 	p.setFont(st::semiboldFont); | ||||||
|  | 	if (!_title.isEmpty()) { | ||||||
|  | 		if (clip.intersects(rtlrect(left, top, qMin(w, _titlew), st::semiboldFont->height, _width))) { | ||||||
|  | 			p.drawTextLeft(left, top, _width, (w < _titlew) ? st::semiboldFont->elided(_title, w) : _title); | ||||||
|  | 		} | ||||||
|  | 		top += st::semiboldFont->height; | ||||||
|  | 	} | ||||||
|  | 	p.setFont(st::msgFont->f); | ||||||
|  | 	if (!_text.isEmpty()) { | ||||||
|  | 		int32 h = qMin(st::normalFont->height * 3, _text.countHeight(w)); | ||||||
|  | 		if (clip.intersects(rtlrect(left, top, w, h, _width))) { | ||||||
|  | 			_text.drawLeftElided(p, left, top, w, _width, 3); | ||||||
|  | 		} | ||||||
|  | 		top += h; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	p.setPen(st::btnYesColor); | ||||||
|  | 	for (int32 i = 0, l = _links.size(); i < l; ++i) { | ||||||
|  | 		if (clip.intersects(rtlrect(left, top, qMin(w, _links.at(i).width), st::normalFont->height, _width))) { | ||||||
|  | 			p.setFont(textlnkDrawOver(_links.at(i).lnk) ? st::normalFont->underline() : st::normalFont); | ||||||
|  | 			p.drawTextLeft(left, top, _width, (w < _links.at(i).width) ? st::normalFont->elided(_links.at(i).text, w) : _links.at(i).text); | ||||||
|  | 		} | ||||||
|  | 		top += st::normalFont->height; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (clip.intersects(rtlrect(left, 0, w, st::linksBorder, _width))) { | ||||||
|  | 		p.fillRect(clip.intersected(rtlrect(left, 0, w, st::linksBorder, _width)), st::linksBorderColor); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void LayoutOverviewLink::getState(TextLinkPtr &link, HistoryCursorState &cursor, int32 x, int32 y) const { | ||||||
|  | 	int32 left = st::dlgPhotoSize + st::dlgPhotoPadding, top = st::linksMargin + st::linksBorder, w = _width - left; | ||||||
|  | 	if (rtlrect(0, top, st::dlgPhotoSize, st::dlgPhotoSize, _width).contains(x, y)) { | ||||||
|  | 		link = _photol; | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (!_title.isEmpty() && _text.isEmpty() && _links.size() == 1) { | ||||||
|  | 		top += (st::dlgPhotoSize - st::semiboldFont->height - st::normalFont->height) / 2; | ||||||
|  | 	} | ||||||
|  | 	if (!_title.isEmpty()) { | ||||||
|  | 		if (rtlrect(left, top, qMin(w, _titlew), st::semiboldFont->height, _width).contains(x, y)) { | ||||||
|  | 			link = _photol; | ||||||
|  | 			return; | ||||||
|  | 		} | ||||||
|  | 		top += st::webPageTitleFont->height; | ||||||
|  | 	} | ||||||
|  | 	if (!_text.isEmpty()) { | ||||||
|  | 		top += qMin(st::normalFont->height * 3, _text.countHeight(w)); | ||||||
|  | 	} | ||||||
|  | 	for (int32 i = 0, l = _links.size(); i < l; ++i) { | ||||||
|  | 		if (rtlrect(left, top, qMin(w, _links.at(i).width), st::normalFont->height, _width).contains(x, y)) { | ||||||
|  | 			link = _links.at(i).lnk; | ||||||
|  | 			return; | ||||||
|  | 		} | ||||||
|  | 		top += st::normalFont->height; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | LayoutOverviewLink::Link::Link(const QString &url, const QString &text)  | ||||||
|  | : text(text) | ||||||
|  | , width(st::normalFont->width(text)) | ||||||
|  | , lnk(linkFromUrl(url)) { | ||||||
|  | } | ||||||
|  | @ -369,7 +369,7 @@ protected: | ||||||
| 		return !_data->already().isEmpty() || !_data->data.isEmpty(); | 		return !_data->already().isEmpty() || !_data->data.isEmpty(); | ||||||
| 	} | 	} | ||||||
| 	virtual bool iconAnimated() const { | 	virtual bool iconAnimated() const { | ||||||
| 		return !dataLoaded() || (_radial && _radial->animating()); | 		return true; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
|  | @ -434,9 +434,10 @@ private: | ||||||
| 
 | 
 | ||||||
| class LayoutOverviewLink : public LayoutMediaItem { | class LayoutOverviewLink : public LayoutMediaItem { | ||||||
| public: | public: | ||||||
| 	LayoutOverviewLink(HistoryItem *parent); | 	LayoutOverviewLink(HistoryMedia *media, HistoryItem *parent); | ||||||
| 
 | 
 | ||||||
| 	virtual void initDimensions(); | 	virtual void initDimensions(); | ||||||
|  | 	virtual int32 resizeGetHeight(int32 width); | ||||||
| 	virtual void paint(Painter &p, const QRect &clip, uint32 selection, const PaintContext *context) const; | 	virtual void paint(Painter &p, const QRect &clip, uint32 selection, const PaintContext *context) const; | ||||||
| 	virtual void getState(TextLinkPtr &link, HistoryCursorState &cursor, int32 x, int32 y) const; | 	virtual void getState(TextLinkPtr &link, HistoryCursorState &cursor, int32 x, int32 y) const; | ||||||
| 
 | 
 | ||||||
|  | @ -449,26 +450,22 @@ public: | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
| 	OverviewItemInfo _info; | 	OverviewItemInfo _info; | ||||||
| 	TextLinkPtr _msgl; | 	TextLinkPtr _photol; | ||||||
| 
 | 
 | ||||||
| 	QString title, letter; | 	QString _title, _letter; | ||||||
| 	int32 titleWidth; | 	int32 _titlew; | ||||||
| 	WebPageData *page; | 	WebPageData *_page; | ||||||
| 	int32 pixw, pixh; | 	int32 _pixw, _pixh; | ||||||
| 	Text text; | 	Text _text; | ||||||
| 
 | 
 | ||||||
| 	struct Link { | 	struct Link { | ||||||
| 		Link() : width(0) { | 		Link() : width(0) { | ||||||
| 		} | 		} | ||||||
| 		Link(const QString &url, const QString &text) | 		Link(const QString &url, const QString &text); | ||||||
| 			: text(text) |  | ||||||
| 			, width(st::normalFont->width(text)) |  | ||||||
| 			, lnk(new TextLink(url)) { |  | ||||||
| 		} |  | ||||||
| 		QString text; | 		QString text; | ||||||
| 		int32 width; | 		int32 width; | ||||||
| 		TextLinkPtr lnk; | 		TextLinkPtr lnk; | ||||||
| 	}; | 	}; | ||||||
| 	QVector<Link> urls; | 	QVector<Link> _links; | ||||||
| 
 | 
 | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | @ -2886,6 +2886,9 @@ namespace Local { | ||||||
| 		QImage img; | 		QImage img; | ||||||
| 		QBuffer buf(&pngData); | 		QBuffer buf(&pngData); | ||||||
| 		QImageReader reader(&buf); | 		QImageReader reader(&buf); | ||||||
|  | #if QT_VERSION >= QT_VERSION_CHECK(5, 5, 0) | ||||||
|  | 		reader.setAutoTransform(true); | ||||||
|  | #endif | ||||||
| 		if (reader.read(&img)) { | 		if (reader.read(&img)) { | ||||||
| 			App::initBackground(id, img, true); | 			App::initBackground(id, img, true); | ||||||
| 			return true; | 			return true; | ||||||
|  |  | ||||||
|  | @ -1522,7 +1522,7 @@ void MainWidget::loadMediaBack(PeerData *peer, MediaOverviewType type, bool many | ||||||
| 	if (history->overviewLoaded(type)) return; | 	if (history->overviewLoaded(type)) return; | ||||||
| 
 | 
 | ||||||
| 	MsgId minId = history->overviewMinId(type); | 	MsgId minId = history->overviewMinId(type); | ||||||
| 	int32 limit = many ? SearchManyPerPage : (history->overview[type].size() > MediaOverviewStartPerPage) ? SearchPerPage : MediaOverviewStartPerPage; | 	int32 limit = (many || history->overview[type].size() > MediaOverviewStartPerPage) ? SearchPerPage : MediaOverviewStartPerPage; | ||||||
| 	MTPMessagesFilter filter = typeToMediaFilter(type); | 	MTPMessagesFilter filter = typeToMediaFilter(type); | ||||||
| 	if (type == OverviewCount) return; | 	if (type == OverviewCount) return; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -465,8 +465,7 @@ void MediaView::step_state(uint64 ms, bool timer) { | ||||||
| 			if (!_doc->already().isEmpty() && _doc->size < MediaViewImageSizeLimit) { | 			if (!_doc->already().isEmpty() && _doc->size < MediaViewImageSizeLimit) { | ||||||
| 				const FileLocation &location(_doc->location(true)); | 				const FileLocation &location(_doc->location(true)); | ||||||
| 				if (location.accessEnable()) { | 				if (location.accessEnable()) { | ||||||
| 					QImageReader reader(location.name()); | 					if (QImageReader(location.name()).canRead()) { | ||||||
| 					if (reader.canRead()) { |  | ||||||
| 						displayDocument(_doc, App::histItemById(_msgmigrated ? 0 : _channel, _msgid)); | 						displayDocument(_doc, App::histItemById(_msgmigrated ? 0 : _channel, _msgid)); | ||||||
| 					} | 					} | ||||||
| 					location.accessDisable(); | 					location.accessDisable(); | ||||||
|  | @ -962,8 +961,7 @@ void MediaView::displayDocument(DocumentData *doc, HistoryItem *item) { // empty | ||||||
| 					_gif = new ClipReader(location, _doc->data); | 					_gif = new ClipReader(location, _doc->data); | ||||||
| 				} | 				} | ||||||
| 			} else { | 			} else { | ||||||
| 				QImageReader reader(location.name()); | 				if (QImageReader(location.name()).canRead()) { | ||||||
| 				if (reader.canRead()) { |  | ||||||
| 					_current = QPixmap::fromImage(App::readImage(location.name(), 0, false), Qt::ColorOnly); | 					_current = QPixmap::fromImage(App::readImage(location.name(), 0, false), Qt::ColorOnly); | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							|  | @ -57,7 +57,7 @@ public: | ||||||
| 	void touchScrollUpdated(const QPoint &screenPos); | 	void touchScrollUpdated(const QPoint &screenPos); | ||||||
| 	QPoint mapMouseToItem(QPoint p, MsgId itemId, int32 itemIndex); | 	QPoint mapMouseToItem(QPoint p, MsgId itemId, int32 itemIndex); | ||||||
| 
 | 
 | ||||||
| 	int32 resizeToWidth(int32 nwidth, int32 scrollTop, int32 minHeight); // returns new scroll top
 | 	int32 resizeToWidth(int32 nwidth, int32 scrollTop, int32 minHeight, bool force = false); // returns new scroll top
 | ||||||
| 	void dropResizeIndex(); | 	void dropResizeIndex(); | ||||||
| 
 | 
 | ||||||
| 	PeerData *peer() const; | 	PeerData *peer() const; | ||||||
|  | @ -67,7 +67,7 @@ public: | ||||||
| 
 | 
 | ||||||
| 	void setSelectMode(bool enabled); | 	void setSelectMode(bool enabled); | ||||||
| 
 | 
 | ||||||
| 	void mediaOverviewUpdated(bool fromResize = false); | 	void mediaOverviewUpdated(); | ||||||
| 	void changingMsgId(HistoryItem *row, MsgId newId); | 	void changingMsgId(HistoryItem *row, MsgId newId); | ||||||
| 	void redrawItem(const HistoryItem *msg); | 	void redrawItem(const HistoryItem *msg); | ||||||
| 	void itemRemoved(HistoryItem *item); | 	void itemRemoved(HistoryItem *item); | ||||||
|  | @ -133,8 +133,8 @@ private: | ||||||
| 	void applyDragSelection(); | 	void applyDragSelection(); | ||||||
| 	void addSelectionRange(int32 selFrom, int32 selTo, History *history); | 	void addSelectionRange(int32 selFrom, int32 selTo, History *history); | ||||||
| 
 | 
 | ||||||
| 	void showAll(bool recountHeights = false); | 	void recountMargins(); | ||||||
| 	int32 recountHeight(); | 	int32 countHeight(); | ||||||
| 
 | 
 | ||||||
| 	OverviewWidget *_overview; | 	OverviewWidget *_overview; | ||||||
| 	ScrollArea *_scroll; | 	ScrollArea *_scroll; | ||||||
|  | @ -149,40 +149,25 @@ private: | ||||||
| 	bool _selMode; | 	bool _selMode; | ||||||
| 	uint32 itemSelectedValue(int32 index) const; | 	uint32 itemSelectedValue(int32 index) const; | ||||||
| 
 | 
 | ||||||
| 	// for audio files, files, voice messages and links
 |  | ||||||
| 	int32 _rowsLeft, _rowWidth; | 	int32 _rowsLeft, _rowWidth; | ||||||
| 
 | 
 | ||||||
| 	// photos
 | 	typedef QVector<LayoutItem*> Items; | ||||||
| 	int32 _photosInRow, _photosToAdd; | 	Items _items; | ||||||
|  | 	typedef QMap<HistoryItem*, LayoutMediaItem*> LayoutItems; | ||||||
|  | 	LayoutItems _layoutItems; | ||||||
|  | 	typedef QMap<int32, LayoutOverviewDate*> LayoutDates; | ||||||
|  | 	LayoutDates _layoutDates; | ||||||
|  | 	LayoutMediaItem *getItemLayout(HistoryItem *item); | ||||||
|  | 	LayoutItem *getDateLayout(const QDate &date, bool month); | ||||||
|  | 	int32 setLayoutItem(int32 index, LayoutItem *item, int32 top); | ||||||
| 
 | 
 | ||||||
| 	// shared links
 |  | ||||||
| 	struct Link { |  | ||||||
| 		Link() : width(0) { |  | ||||||
| 		} |  | ||||||
| 		Link(const QString &url, const QString &text) : url(url), text(text), width(st::msgFont->width(text)) { |  | ||||||
| 		} |  | ||||||
| 		QString url, text; |  | ||||||
| 		int32 width; |  | ||||||
| 	}; |  | ||||||
| 	struct CachedLink { |  | ||||||
| 		CachedLink() : titleWidth(0), page(0), pixw(0), pixh(0), text(st::msgMinWidth) { |  | ||||||
| 		} |  | ||||||
| 		CachedLink(HistoryItem *item); |  | ||||||
| 		int32 countHeight(int32 w); |  | ||||||
| 
 |  | ||||||
| 		QString title, letter; |  | ||||||
| 		int32 titleWidth; |  | ||||||
| 		WebPageData *page; |  | ||||||
| 		int32 pixw, pixh; |  | ||||||
| 		Text text; |  | ||||||
| 		QVector<Link> urls; |  | ||||||
| 	}; |  | ||||||
| 	typedef QMap<MsgId, CachedLink*> CachedLinks; |  | ||||||
| 	CachedLinks _links; |  | ||||||
| 	FlatInput _search; | 	FlatInput _search; | ||||||
| 	IconedButton _cancelSearch; | 	IconedButton _cancelSearch; | ||||||
| 	QVector<MsgId> _results; | 	QVector<MsgId> _results; | ||||||
| 	int32 _cachedItemsToBeLoaded; | 	int32 _itemsToBeLoaded; | ||||||
|  | 
 | ||||||
|  | 	// photos
 | ||||||
|  | 	int32 _photosInRow, _photosToAdd; | ||||||
| 
 | 
 | ||||||
| 	QTimer _searchTimer; | 	QTimer _searchTimer; | ||||||
| 	QString _searchQuery; | 	QString _searchQuery; | ||||||
|  | @ -206,31 +191,7 @@ private: | ||||||
| 	typedef QMap<mtpRequestId, QString> SearchQueries; | 	typedef QMap<mtpRequestId, QString> SearchQueries; | ||||||
| 	SearchQueries _searchQueries; | 	SearchQueries _searchQueries; | ||||||
| 
 | 
 | ||||||
| 	CachedLink *cachedLink(HistoryItem *item); | 	int32 _width, _height, _minHeight, _marginTop, _marginBottom; | ||||||
| 
 |  | ||||||
| 	typedef QVector<LayoutItem*> Items; |  | ||||||
| 	Items _items; |  | ||||||
| 	typedef QMap<HistoryItem*, LayoutMediaItem*> LayoutItems; |  | ||||||
| 	LayoutItems _layoutItems; |  | ||||||
| 	LayoutMediaItem *getItemLayout(HistoryItem *item); |  | ||||||
| 	LayoutItem *getDateLayout(const QDate &date, bool month); |  | ||||||
| 	int32 setLayoutItem(int32 index, LayoutItem *item, int32 top); |  | ||||||
| 
 |  | ||||||
| 	// other
 |  | ||||||
| 	struct CachedItem { |  | ||||||
| 		CachedItem() : msgid(0), y(0), link(0) { |  | ||||||
| 		} |  | ||||||
| 		CachedItem(MsgId msgid, const QDate &date, int32 y) : msgid(msgid), date(date), y(y), link(0) { |  | ||||||
| 		} |  | ||||||
| 		MsgId msgid; |  | ||||||
| 		QDate date; |  | ||||||
| 		int32 y; |  | ||||||
| 		CachedLink *link; |  | ||||||
| 	}; |  | ||||||
| 	typedef QVector<CachedItem> CachedItems; |  | ||||||
| 	CachedItems _cachedItems; |  | ||||||
| 
 |  | ||||||
| 	int32 _width, _height, _minHeight, _addToY; |  | ||||||
| 
 | 
 | ||||||
| 	QTimer _linkTipTimer; | 	QTimer _linkTipTimer; | ||||||
| 
 | 
 | ||||||
|  | @ -252,14 +213,9 @@ private: | ||||||
| 	int32 _dragItemIndex; | 	int32 _dragItemIndex; | ||||||
| 	MsgId _mousedItem; | 	MsgId _mousedItem; | ||||||
| 	int32 _mousedItemIndex; | 	int32 _mousedItemIndex; | ||||||
| 	int32 _lnkOverIndex, _lnkDownIndex; // for OverviewLinks, 0 - none, -1 - photo or title, > 0 - lnk index
 |  | ||||||
| 	uint16 _dragSymbol; | 	uint16 _dragSymbol; | ||||||
| 	bool _dragWasInactive; | 	bool _dragWasInactive; | ||||||
| 
 | 
 | ||||||
| 	QString urlByIndex(MsgId msgid, int32 index, int32 lnkIndex, bool *fullShown = 0) const; |  | ||||||
| 	bool urlIsEmail(const QString &url) const; |  | ||||||
| 
 |  | ||||||
| 	QString _contextMenuUrl; |  | ||||||
| 	TextLinkPtr _contextMenuLnk; | 	TextLinkPtr _contextMenuLnk; | ||||||
| 
 | 
 | ||||||
| 	MsgId _dragSelFrom, _dragSelTo; | 	MsgId _dragSelFrom, _dragSelTo; | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue