mirror of
				https://github.com/telegramdesktop/tdesktop
				synced 2025-10-21 14:38:26 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			238 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			238 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
| This file is part of Telegram Desktop,
 | |
| the official desktop application for the Telegram messaging service.
 | |
| 
 | |
| For license and copyright information please follow this link:
 | |
| https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
 | |
| */
 | |
| #include "data/data_web_page.h"
 | |
| 
 | |
| #include "auth_session.h"
 | |
| #include "apiwrap.h"
 | |
| #include "mainwidget.h"
 | |
| #include "data/data_session.h"
 | |
| #include "data/data_photo.h"
 | |
| #include "data/data_document.h"
 | |
| #include "ui/image/image.h"
 | |
| #include "ui/image/image_source.h"
 | |
| #include "ui/text/text_entity.h"
 | |
| 
 | |
| namespace {
 | |
| 
 | |
| QString SiteNameFromUrl(const QString &url) {
 | |
| 	QUrl u(url);
 | |
| 	QString pretty = u.isValid() ? u.toDisplayString() : url;
 | |
| 	QRegularExpressionMatch m = QRegularExpression(qsl("^[a-zA-Z0-9]+://")).match(pretty);
 | |
| 	if (m.hasMatch()) pretty = pretty.mid(m.capturedLength());
 | |
| 	int32 slash = pretty.indexOf('/');
 | |
| 	if (slash > 0) pretty = pretty.mid(0, slash);
 | |
| 	QStringList components = pretty.split('.', QString::SkipEmptyParts);
 | |
| 	if (components.size() >= 2) {
 | |
| 		components = components.mid(components.size() - 2);
 | |
| 		return components.at(0).at(0).toUpper() + components.at(0).mid(1) + '.' + components.at(1);
 | |
| 	}
 | |
| 	return QString();
 | |
| }
 | |
| 
 | |
| WebPageCollage ExtractCollage(
 | |
| 		const QVector<MTPPageBlock> &items,
 | |
| 		const QVector<MTPPhoto> &photos,
 | |
| 		const QVector<MTPDocument> &documents) {
 | |
| 	const auto count = items.size();
 | |
| 	if (count < 2) {
 | |
| 		return {};
 | |
| 	}
 | |
| 	const auto bad = ranges::find_if(items, [](mtpTypeId type) {
 | |
| 		return (type != mtpc_pageBlockPhoto && type != mtpc_pageBlockVideo);
 | |
| 	}, [](const MTPPageBlock &item) {
 | |
| 		return item.type();
 | |
| 	});
 | |
| 	if (bad != items.end()) {
 | |
| 		return {};
 | |
| 	}
 | |
| 
 | |
| 	auto &storage = Auth().data();
 | |
| 	for (const auto &photo : photos) {
 | |
| 		storage.photo(photo);
 | |
| 	}
 | |
| 	for (const auto &document : documents) {
 | |
| 		storage.document(document);
 | |
| 	}
 | |
| 	auto result = WebPageCollage();
 | |
| 	result.items.reserve(count);
 | |
| 	for (const auto &item : items) {
 | |
| 		const auto good = item.match([&](const MTPDpageBlockPhoto &data) {
 | |
| 			const auto photo = storage.photo(data.vphoto_id.v);
 | |
| 			if (photo->full->isNull()) {
 | |
| 				return false;
 | |
| 			}
 | |
| 			result.items.push_back(photo);
 | |
| 			return true;
 | |
| 		}, [&](const MTPDpageBlockVideo &data) {
 | |
| 			const auto document = storage.document(data.vvideo_id.v);
 | |
| 			if (!document->isVideoFile()) {
 | |
| 				return false;
 | |
| 			}
 | |
| 			result.items.push_back(document);
 | |
| 			return true;
 | |
| 		}, [](const auto &) -> bool {
 | |
| 			Unexpected("Type of block in Collage.");
 | |
| 		});
 | |
| 		if (!good) {
 | |
| 			return {};
 | |
| 		}
 | |
| 	}
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| WebPageCollage ExtractCollage(const MTPDwebPage &data) {
 | |
| 	if (!data.has_cached_page()) {
 | |
| 		return {};
 | |
| 	}
 | |
| 	const auto parseMedia = [&] {
 | |
| 		if (data.has_photo()) {
 | |
| 			Auth().data().photo(data.vphoto);
 | |
| 		}
 | |
| 		if (data.has_document()) {
 | |
| 			Auth().data().document(data.vdocument);
 | |
| 		}
 | |
| 	};
 | |
| 	return data.vcached_page.match([&](const auto &page) {
 | |
| 		for (const auto &block : page.vblocks.v) {
 | |
| 			switch (block.type()) {
 | |
| 			case mtpc_pageBlockPhoto:
 | |
| 			case mtpc_pageBlockVideo:
 | |
| 			case mtpc_pageBlockCover:
 | |
| 			case mtpc_pageBlockEmbed:
 | |
| 			case mtpc_pageBlockEmbedPost:
 | |
| 			case mtpc_pageBlockAudio:
 | |
| 				return WebPageCollage();
 | |
| 			case mtpc_pageBlockSlideshow:
 | |
| 				parseMedia();
 | |
| 				return ExtractCollage(
 | |
| 					block.c_pageBlockSlideshow().vitems.v,
 | |
| 					page.vphotos.v,
 | |
| 					page.vdocuments.v);
 | |
| 			case mtpc_pageBlockCollage:
 | |
| 				parseMedia();
 | |
| 				return ExtractCollage(
 | |
| 					block.c_pageBlockCollage().vitems.v,
 | |
| 					page.vphotos.v,
 | |
| 					page.vdocuments.v);
 | |
| 			default: break;
 | |
| 			}
 | |
| 		}
 | |
| 		return WebPageCollage();
 | |
| 	});
 | |
| }
 | |
| 
 | |
| } // namespace
 | |
| 
 | |
| WebPageType ParseWebPageType(const MTPDwebPage &page) {
 | |
| 	const auto type = page.has_type() ? qs(page.vtype) : QString();
 | |
| 	if (type == qstr("photo")) return WebPageType::Photo;
 | |
| 	if (type == qstr("video")) return WebPageType::Video;
 | |
| 	if (type == qstr("profile")) return WebPageType::Profile;
 | |
| 	return page.has_cached_page()
 | |
| 		? WebPageType::ArticleWithIV
 | |
| 		: WebPageType::Article;
 | |
| }
 | |
| 
 | |
| WebPageCollage::WebPageCollage(const MTPDwebPage &data)
 | |
| : WebPageCollage(ExtractCollage(data)) {
 | |
| }
 | |
| 
 | |
| bool WebPageData::applyChanges(
 | |
| 		WebPageType newType,
 | |
| 		const QString &newUrl,
 | |
| 		const QString &newDisplayUrl,
 | |
| 		const QString &newSiteName,
 | |
| 		const QString &newTitle,
 | |
| 		const TextWithEntities &newDescription,
 | |
| 		PhotoData *newPhoto,
 | |
| 		DocumentData *newDocument,
 | |
| 		WebPageCollage &&newCollage,
 | |
| 		int newDuration,
 | |
| 		const QString &newAuthor,
 | |
| 		int newPendingTill) {
 | |
| 	if (newPendingTill != 0
 | |
| 		&& (!url.isEmpty() || pendingTill < 0)
 | |
| 		&& (!pendingTill
 | |
| 			|| pendingTill == newPendingTill
 | |
| 			|| newPendingTill < -1)) {
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	const auto resultUrl = TextUtilities::Clean(newUrl);
 | |
| 	const auto resultDisplayUrl = TextUtilities::Clean(
 | |
| 		newDisplayUrl);
 | |
| 	const auto possibleSiteName = TextUtilities::Clean(
 | |
| 		newSiteName);
 | |
| 	const auto resultTitle = TextUtilities::SingleLine(
 | |
| 		newTitle);
 | |
| 	const auto resultAuthor = TextUtilities::Clean(newAuthor);
 | |
| 
 | |
| 	const auto viewTitleText = resultTitle.isEmpty()
 | |
| 		? TextUtilities::SingleLine(resultAuthor)
 | |
| 		: resultTitle;
 | |
| 	const auto resultSiteName = [&] {
 | |
| 		if (!possibleSiteName.isEmpty()) {
 | |
| 			return possibleSiteName;
 | |
| 		} else if (!newDescription.text.isEmpty()
 | |
| 			&& viewTitleText.isEmpty()
 | |
| 			&& !resultUrl.isEmpty()) {
 | |
| 			return SiteNameFromUrl(resultUrl);
 | |
| 		}
 | |
| 		return QString();
 | |
| 	}();
 | |
| 
 | |
| 	if (type == newType
 | |
| 		&& url == resultUrl
 | |
| 		&& displayUrl == resultDisplayUrl
 | |
| 		&& siteName == resultSiteName
 | |
| 		&& title == resultTitle
 | |
| 		&& description.text == newDescription.text
 | |
| 		&& photo == newPhoto
 | |
| 		&& document == newDocument
 | |
| 		&& collage.items == newCollage.items
 | |
| 		&& duration == newDuration
 | |
| 		&& author == resultAuthor
 | |
| 		&& pendingTill == newPendingTill) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	if (pendingTill > 0 && newPendingTill <= 0) {
 | |
| 		Auth().api().clearWebPageRequest(this);
 | |
| 	}
 | |
| 	type = newType;
 | |
| 	url = resultUrl;
 | |
| 	displayUrl = resultDisplayUrl;
 | |
| 	siteName = resultSiteName;
 | |
| 	title = resultTitle;
 | |
| 	description = newDescription;
 | |
| 	photo = newPhoto;
 | |
| 	document = newDocument;
 | |
| 	collage = std::move(newCollage);
 | |
| 	duration = newDuration;
 | |
| 	author = resultAuthor;
 | |
| 	pendingTill = newPendingTill;
 | |
| 	++version;
 | |
| 
 | |
| 	replaceDocumentGoodThumbnail();
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| void WebPageData::replaceDocumentGoodThumbnail() {
 | |
| 	if (!document || !photo || !document->goodThumbnail()) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto &location = photo->full->location();
 | |
| 	if (!location.isNull()) {
 | |
| 		document->replaceGoodThumbnail(
 | |
| 			std::make_unique<Images::StorageSource>(
 | |
| 				location,
 | |
| 				photo->full->bytesSize()));
 | |
| 	}
 | |
| 
 | |
| }
 |