286 lines
7.0 KiB
C++
286 lines
7.0 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
|
|
*/
|
|
#pragma once
|
|
|
|
#include "api/api_common.h"
|
|
#include "ui/effects/animations.h"
|
|
#include "ui/rp_widget.h"
|
|
#include "base/timer.h"
|
|
#include "base/object_ptr.h"
|
|
|
|
namespace Ui {
|
|
class PopupMenu;
|
|
class ScrollArea;
|
|
} // namespace Ui
|
|
|
|
namespace Lottie {
|
|
class SinglePlayer;
|
|
class FrameRenderer;
|
|
} // namespace Lottie;
|
|
|
|
namespace Window {
|
|
class SessionController;
|
|
} // namespace Window
|
|
|
|
namespace Data {
|
|
class DocumentMedia;
|
|
class CloudImageView;
|
|
} // namespace Data
|
|
|
|
namespace internal {
|
|
|
|
struct StickerSuggestion {
|
|
not_null<DocumentData*> document;
|
|
std::shared_ptr<Data::DocumentMedia> documentMedia;
|
|
std::unique_ptr<Lottie::SinglePlayer> animated;
|
|
};
|
|
|
|
struct MentionRow {
|
|
not_null<UserData*> user;
|
|
std::shared_ptr<Data::CloudImageView> userpic;
|
|
};
|
|
|
|
struct BotCommandRow {
|
|
not_null<UserData*> user;
|
|
not_null<const BotCommand*> command;
|
|
std::shared_ptr<Data::CloudImageView> userpic;
|
|
};
|
|
|
|
using HashtagRows = std::vector<QString>;
|
|
using BotCommandRows = std::vector<BotCommandRow>;
|
|
using StickerRows = std::vector<StickerSuggestion>;
|
|
using MentionRows = std::vector<MentionRow>;
|
|
|
|
class FieldAutocompleteInner;
|
|
|
|
} // namespace internal
|
|
|
|
class FieldAutocomplete final : public Ui::RpWidget {
|
|
|
|
public:
|
|
FieldAutocomplete(
|
|
QWidget *parent,
|
|
not_null<Window::SessionController*> controller);
|
|
~FieldAutocomplete();
|
|
|
|
bool clearFilteredBotCommands();
|
|
void showFiltered(
|
|
not_null<PeerData*> peer,
|
|
QString query,
|
|
bool addInlineBots);
|
|
void showStickers(EmojiPtr emoji);
|
|
void setBoundings(QRect boundings);
|
|
|
|
const QString &filter() const;
|
|
ChatData *chat() const;
|
|
ChannelData *channel() const;
|
|
UserData *user() const;
|
|
|
|
int32 innerTop();
|
|
int32 innerBottom();
|
|
|
|
bool eventFilter(QObject *obj, QEvent *e) override;
|
|
|
|
enum class ChooseMethod {
|
|
ByEnter,
|
|
ByTab,
|
|
ByClick,
|
|
};
|
|
struct MentionChosen {
|
|
not_null<UserData*> user;
|
|
ChooseMethod method;
|
|
};
|
|
struct HashtagChosen {
|
|
QString hashtag;
|
|
ChooseMethod method;
|
|
};
|
|
struct BotCommandChosen {
|
|
QString command;
|
|
ChooseMethod method;
|
|
};
|
|
struct StickerChosen {
|
|
not_null<DocumentData*> sticker;
|
|
Api::SendOptions options;
|
|
ChooseMethod method;
|
|
};
|
|
|
|
bool chooseSelected(ChooseMethod method) const;
|
|
|
|
bool stickersShown() const {
|
|
return !_srows.empty();
|
|
}
|
|
|
|
bool overlaps(const QRect &globalRect) {
|
|
if (isHidden() || !testAttribute(Qt::WA_OpaquePaintEvent)) return false;
|
|
|
|
return rect().contains(QRect(mapFromGlobal(globalRect.topLeft()), globalRect.size()));
|
|
}
|
|
|
|
void setModerateKeyActivateCallback(Fn<bool(int)> callback) {
|
|
_moderateKeyActivateCallback = std::move(callback);
|
|
}
|
|
|
|
void hideFast();
|
|
|
|
rpl::producer<MentionChosen> mentionChosen() const;
|
|
rpl::producer<HashtagChosen> hashtagChosen() const;
|
|
rpl::producer<BotCommandChosen> botCommandChosen() const;
|
|
rpl::producer<StickerChosen> stickerChosen() const;
|
|
|
|
public slots:
|
|
void showAnimated();
|
|
void hideAnimated();
|
|
|
|
protected:
|
|
void paintEvent(QPaintEvent *e) override;
|
|
|
|
private:
|
|
void animationCallback();
|
|
void hideFinish();
|
|
|
|
void updateFiltered(bool resetScroll = false);
|
|
void recount(bool resetScroll = false);
|
|
internal::StickerRows getStickerSuggestions();
|
|
|
|
const not_null<Window::SessionController*> _controller;
|
|
QPixmap _cache;
|
|
internal::MentionRows _mrows;
|
|
internal::HashtagRows _hrows;
|
|
internal::BotCommandRows _brows;
|
|
internal::StickerRows _srows;
|
|
|
|
void rowsUpdated(
|
|
internal::MentionRows &&mrows,
|
|
internal::HashtagRows &&hrows,
|
|
internal::BotCommandRows &&brows,
|
|
internal::StickerRows &&srows,
|
|
bool resetScroll);
|
|
|
|
object_ptr<Ui::ScrollArea> _scroll;
|
|
QPointer<internal::FieldAutocompleteInner> _inner;
|
|
|
|
ChatData *_chat = nullptr;
|
|
UserData *_user = nullptr;
|
|
ChannelData *_channel = nullptr;
|
|
EmojiPtr _emoji;
|
|
uint64 _stickersSeed = 0;
|
|
enum class Type {
|
|
Mentions,
|
|
Hashtags,
|
|
BotCommands,
|
|
Stickers,
|
|
};
|
|
Type _type = Type::Mentions;
|
|
QString _filter;
|
|
QRect _boundings;
|
|
bool _addInlineBots;
|
|
|
|
bool _hiding = false;
|
|
|
|
Ui::Animations::Simple _a_opacity;
|
|
|
|
Fn<bool(int)> _moderateKeyActivateCallback;
|
|
|
|
friend class internal::FieldAutocompleteInner;
|
|
|
|
};
|
|
|
|
namespace internal {
|
|
|
|
class FieldAutocompleteInner final
|
|
: public Ui::RpWidget
|
|
, private base::Subscriber {
|
|
|
|
public:
|
|
struct ScrollTo {
|
|
int top;
|
|
int bottom;
|
|
};
|
|
|
|
FieldAutocompleteInner(
|
|
not_null<Window::SessionController*> controller,
|
|
not_null<FieldAutocomplete*> parent,
|
|
not_null<MentionRows*> mrows,
|
|
not_null<HashtagRows*> hrows,
|
|
not_null<BotCommandRows*> brows,
|
|
not_null<StickerRows*> srows);
|
|
|
|
void clearSel(bool hidden = false);
|
|
bool moveSel(int key);
|
|
bool chooseSelected(FieldAutocomplete::ChooseMethod method) const;
|
|
bool chooseAtIndex(
|
|
FieldAutocomplete::ChooseMethod method,
|
|
int index,
|
|
Api::SendOptions options = Api::SendOptions()) const;
|
|
|
|
void setRecentInlineBotsInRows(int32 bots);
|
|
void rowsUpdated();
|
|
|
|
rpl::producer<FieldAutocomplete::MentionChosen> mentionChosen() const;
|
|
rpl::producer<FieldAutocomplete::HashtagChosen> hashtagChosen() const;
|
|
rpl::producer<FieldAutocomplete::BotCommandChosen>
|
|
botCommandChosen() const;
|
|
rpl::producer<FieldAutocomplete::StickerChosen> stickerChosen() const;
|
|
rpl::producer<ScrollTo> scrollToRequested() const;
|
|
|
|
void onParentGeometryChanged();
|
|
|
|
private:
|
|
void paintEvent(QPaintEvent *e) override;
|
|
void resizeEvent(QResizeEvent *e) override;
|
|
|
|
void enterEventHook(QEvent *e) override;
|
|
void leaveEventHook(QEvent *e) override;
|
|
|
|
void mousePressEvent(QMouseEvent *e) override;
|
|
void mouseMoveEvent(QMouseEvent *e) override;
|
|
void mouseReleaseEvent(QMouseEvent *e) override;
|
|
void contextMenuEvent(QContextMenuEvent *e) override;
|
|
|
|
void updateSelectedRow();
|
|
void setSel(int sel, bool scroll = false);
|
|
void showPreview();
|
|
void selectByMouse(QPoint global);
|
|
|
|
QSize stickerBoundingBox() const;
|
|
void setupLottie(StickerSuggestion &suggestion);
|
|
void repaintSticker(not_null<DocumentData*> document);
|
|
std::shared_ptr<Lottie::FrameRenderer> getLottieRenderer();
|
|
|
|
const not_null<Window::SessionController*> _controller;
|
|
const not_null<FieldAutocomplete*> _parent;
|
|
const not_null<MentionRows*> _mrows;
|
|
const not_null<HashtagRows*> _hrows;
|
|
const not_null<BotCommandRows*> _brows;
|
|
const not_null<StickerRows*> _srows;
|
|
rpl::lifetime _stickersLifetime;
|
|
std::weak_ptr<Lottie::FrameRenderer> _lottieRenderer;
|
|
base::unique_qptr<Ui::PopupMenu> _menu;
|
|
int _stickersPerRow = 1;
|
|
int _recentInlineBotsInRows = 0;
|
|
int _sel = -1;
|
|
int _down = -1;
|
|
std::optional<QPoint> _lastMousePosition;
|
|
bool _mouseSelection = false;
|
|
|
|
bool _overDelete = false;
|
|
|
|
bool _previewShown = false;
|
|
|
|
rpl::event_stream<FieldAutocomplete::MentionChosen> _mentionChosen;
|
|
rpl::event_stream<FieldAutocomplete::HashtagChosen> _hashtagChosen;
|
|
rpl::event_stream<FieldAutocomplete::BotCommandChosen> _botCommandChosen;
|
|
rpl::event_stream<FieldAutocomplete::StickerChosen> _stickerChosen;
|
|
rpl::event_stream<ScrollTo> _scrollToRequested;
|
|
|
|
base::Timer _previewTimer;
|
|
|
|
};
|
|
|
|
} // namespace internal
|