234 lines
6.0 KiB
C++
234 lines
6.0 KiB
C++
/*
|
|
This file is part of Telegram Desktop,
|
|
the official desktop version of Telegram messaging app, see https://telegram.org
|
|
|
|
Telegram Desktop is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
It is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
In addition, as a special exception, the copyright holders give permission
|
|
to link the code of portions of this program with the OpenSSL library.
|
|
|
|
Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
|
|
Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
|
|
*/
|
|
#include "inline_bots/inline_bot_layout_item.h"
|
|
|
|
#include "core/click_handler_types.h"
|
|
#include "inline_bots/inline_bot_result.h"
|
|
#include "inline_bots/inline_bot_layout_internal.h"
|
|
#include "storage/localstorage.h"
|
|
#include "mainwidget.h"
|
|
|
|
namespace InlineBots {
|
|
namespace Layout {
|
|
|
|
void ItemBase::setPosition(int32 position) {
|
|
_position = position;
|
|
}
|
|
|
|
int32 ItemBase::position() const {
|
|
return _position;
|
|
}
|
|
|
|
Result *ItemBase::getResult() const {
|
|
return _result;
|
|
}
|
|
|
|
DocumentData *ItemBase::getDocument() const {
|
|
return _doc;
|
|
}
|
|
|
|
PhotoData *ItemBase::getPhoto() const {
|
|
return _photo;
|
|
}
|
|
|
|
DocumentData *ItemBase::getPreviewDocument() const {
|
|
auto previewDocument = [this]() -> DocumentData* {
|
|
if (_doc) {
|
|
return _doc;
|
|
}
|
|
if (_result) {
|
|
return _result->_document;
|
|
}
|
|
return nullptr;
|
|
};
|
|
if (DocumentData *result = previewDocument()) {
|
|
if (result->sticker() || result->loaded()) {
|
|
return result;
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
PhotoData *ItemBase::getPreviewPhoto() const {
|
|
if (_photo) {
|
|
return _photo;
|
|
}
|
|
if (_result) {
|
|
return _result->_photo;
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
void ItemBase::preload() const {
|
|
if (_result) {
|
|
if (_result->_photo) {
|
|
_result->_photo->thumb->load();
|
|
} else if (_result->_document) {
|
|
_result->_document->thumb->load();
|
|
} else if (!_result->_thumb->isNull()) {
|
|
_result->_thumb->load();
|
|
}
|
|
} else if (_doc) {
|
|
_doc->thumb->load();
|
|
} else if (_photo) {
|
|
_photo->medium->load();
|
|
}
|
|
}
|
|
|
|
void ItemBase::update() {
|
|
if (_position >= 0) {
|
|
Ui::repaintInlineItem(this);
|
|
}
|
|
}
|
|
|
|
std::unique_ptr<ItemBase> ItemBase::createLayout(Result *result, bool forceThumb) {
|
|
using Type = Result::Type;
|
|
|
|
switch (result->_type) {
|
|
case Type::Photo: return std::make_unique<internal::Photo>(result); break;
|
|
case Type::Audio:
|
|
case Type::File: return std::make_unique<internal::File>(result); break;
|
|
case Type::Video: return std::make_unique<internal::Video>(result); break;
|
|
case Type::Sticker: return std::make_unique<internal::Sticker>(result); break;
|
|
case Type::Gif: return std::make_unique<internal::Gif>(result); break;
|
|
case Type::Article:
|
|
case Type::Geo:
|
|
case Type::Venue: return std::make_unique<internal::Article>(result, forceThumb); break;
|
|
case Type::Game: return std::make_unique<internal::Game>(result); break;
|
|
case Type::Contact: return std::make_unique<internal::Contact>(result); break;
|
|
}
|
|
return std::unique_ptr<ItemBase>();
|
|
}
|
|
|
|
std::unique_ptr<ItemBase> ItemBase::createLayoutGif(DocumentData *document) {
|
|
return std::make_unique<internal::Gif>(document, true);
|
|
}
|
|
|
|
DocumentData *ItemBase::getResultDocument() const {
|
|
return _result ? _result->_document : nullptr;
|
|
}
|
|
|
|
PhotoData *ItemBase::getResultPhoto() const {
|
|
return _result ? _result->_photo : nullptr;
|
|
}
|
|
|
|
ImagePtr ItemBase::getResultThumb() const {
|
|
if (_result) {
|
|
if (_result->_photo && !_result->_photo->thumb->isNull()) {
|
|
return _result->_photo->thumb;
|
|
}
|
|
if (!_result->_thumb->isNull()) {
|
|
return _result->_thumb;
|
|
}
|
|
return _result->_locationThumb;
|
|
}
|
|
return ImagePtr();
|
|
}
|
|
|
|
QPixmap ItemBase::getResultContactAvatar(int width, int height) const {
|
|
if (_result->_type == Result::Type::Contact) {
|
|
auto result = EmptyUserpic(qHash(_result->_id) % kUserColorsCount, _result->getLayoutTitle()).generate(width);
|
|
if (result.height() != height * cIntRetinaFactor()) {
|
|
result = result.scaled(QSize(width, height) * cIntRetinaFactor(), Qt::IgnoreAspectRatio, Qt::SmoothTransformation);
|
|
}
|
|
return result;
|
|
}
|
|
return QPixmap();
|
|
}
|
|
|
|
int ItemBase::getResultDuration() const {
|
|
return _result->_duration;
|
|
}
|
|
|
|
QString ItemBase::getResultUrl() const {
|
|
return _result->_url;
|
|
}
|
|
|
|
ClickHandlerPtr ItemBase::getResultUrlHandler() const {
|
|
if (!_result->_url.isEmpty()) {
|
|
return MakeShared<UrlClickHandler>(_result->_url);
|
|
}
|
|
return ClickHandlerPtr();
|
|
}
|
|
|
|
ClickHandlerPtr ItemBase::getResultContentUrlHandler() const {
|
|
if (!_result->_content_url.isEmpty()) {
|
|
return MakeShared<UrlClickHandler>(_result->_content_url);
|
|
}
|
|
return ClickHandlerPtr();
|
|
}
|
|
|
|
QString ItemBase::getResultThumbLetter() const {
|
|
#ifndef OS_MAC_OLD
|
|
auto parts = _result->_url.splitRef('/');
|
|
#else // OS_MAC_OLD
|
|
auto parts = _result->_url.split('/');
|
|
#endif // OS_MAC_OLD
|
|
if (!parts.isEmpty()) {
|
|
auto 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) {
|
|
return parts.at(parts.size() - 2).at(0).toUpper();
|
|
}
|
|
}
|
|
if (!_result->_title.isEmpty()) {
|
|
return _result->_title.at(0).toUpper();
|
|
}
|
|
return QString();
|
|
}
|
|
|
|
namespace {
|
|
|
|
NeverFreedPointer<DocumentItems> documentItemsMap;
|
|
|
|
} // namespace
|
|
|
|
const DocumentItems *documentItems() {
|
|
return documentItemsMap.data();
|
|
}
|
|
|
|
namespace internal {
|
|
|
|
void regDocumentItem(DocumentData *document, ItemBase *item) {
|
|
documentItemsMap.createIfNull();
|
|
(*documentItemsMap)[document].insert(item);
|
|
}
|
|
|
|
void unregDocumentItem(DocumentData *document, ItemBase *item) {
|
|
if (documentItemsMap) {
|
|
auto i = documentItemsMap->find(document);
|
|
if (i != documentItemsMap->cend()) {
|
|
if (i->remove(item) && i->isEmpty()) {
|
|
documentItemsMap->erase(i);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
} // namespace internal
|
|
|
|
} // namespace Layout
|
|
} // namespace InlineBots
|