mirror of
https://github.com/AyuGram/AyuGramDesktop.git
synced 2025-04-13 04:37:11 +02:00
2070 lines
57 KiB
C++
2070 lines
57 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 "ui/chat/attach/attach_bot_webview.h"
|
|
|
|
#include "core/file_utilities.h"
|
|
#include "ui/boxes/confirm_box.h"
|
|
#include "ui/chat/attach/attach_bot_downloads.h"
|
|
#include "ui/effects/radial_animation.h"
|
|
#include "ui/effects/ripple_animation.h"
|
|
#include "ui/layers/box_content.h"
|
|
#include "ui/style/style_core_palette.h"
|
|
#include "ui/text/text_utilities.h"
|
|
#include "ui/widgets/separate_panel.h"
|
|
#include "ui/widgets/buttons.h"
|
|
#include "ui/widgets/labels.h"
|
|
#include "ui/widgets/menu/menu_add_action_callback.h"
|
|
#include "ui/wrap/fade_wrap.h"
|
|
#include "ui/integration.h"
|
|
#include "ui/painter.h"
|
|
#include "ui/rect.h"
|
|
#include "ui/ui_utility.h"
|
|
#include "lang/lang_keys.h"
|
|
#include "webview/webview_embed.h"
|
|
#include "webview/webview_dialog.h"
|
|
#include "webview/webview_interface.h"
|
|
#include "base/debug_log.h"
|
|
#include "base/invoke_queued.h"
|
|
#include "base/qt_signal_producer.h"
|
|
#include "styles/style_chat.h"
|
|
#include "styles/style_payments.h"
|
|
#include "styles/style_layers.h"
|
|
#include "styles/style_menu_icons.h"
|
|
|
|
#include <QtCore/QJsonDocument>
|
|
#include <QtCore/QJsonObject>
|
|
#include <QtCore/QJsonArray>
|
|
#include <QtGui/QGuiApplication>
|
|
#include <QtGui/QClipboard>
|
|
#include <QtGui/QWindow>
|
|
#include <QtGui/QScreen>
|
|
#include <QtGui/qpa/qplatformscreen.h>
|
|
|
|
// AyuGram includes
|
|
#include "ayu/ayu_settings.h"
|
|
#include "styles/style_ayu_styles.h"
|
|
|
|
|
|
namespace Ui::BotWebView {
|
|
namespace {
|
|
|
|
constexpr auto kProcessClickTimeout = crl::time(1000);
|
|
constexpr auto kProgressDuration = crl::time(200);
|
|
constexpr auto kProgressOpacity = 0.3;
|
|
constexpr auto kLightnessThreshold = 128;
|
|
constexpr auto kLightnessDelta = 32;
|
|
|
|
struct ButtonArgs {
|
|
bool isActive = false;
|
|
bool isVisible = false;
|
|
bool isProgressVisible = false;
|
|
QString text;
|
|
};
|
|
|
|
[[nodiscard]] RectPart ParsePosition(const QString &position) {
|
|
if (position == u"left"_q) {
|
|
return RectPart::Left;
|
|
} else if (position == u"top"_q) {
|
|
return RectPart::Top;
|
|
} else if (position == u"right"_q) {
|
|
return RectPart::Right;
|
|
} else if (position == u"bottom"_q) {
|
|
return RectPart::Bottom;
|
|
}
|
|
return RectPart::Left;
|
|
}
|
|
|
|
[[nodiscard]] QJsonObject ParseMethodArgs(const QString &json) {
|
|
if (json.isEmpty()) {
|
|
return {};
|
|
}
|
|
auto error = QJsonParseError();
|
|
const auto dictionary = QJsonDocument::fromJson(json.toUtf8(), &error);
|
|
if (error.error != QJsonParseError::NoError) {
|
|
LOG(("BotWebView Error: Could not parse \"%1\".").arg(json));
|
|
return {};
|
|
}
|
|
return dictionary.object();
|
|
}
|
|
|
|
[[nodiscard]] std::optional<QColor> ParseColor(const QString &text) {
|
|
if (!text.startsWith('#') || text.size() != 7) {
|
|
return {};
|
|
}
|
|
const auto data = text.data() + 1;
|
|
const auto hex = [&](int from) -> std::optional<int> {
|
|
const auto parse = [](QChar ch) -> std::optional<int> {
|
|
const auto code = ch.unicode();
|
|
return (code >= 'a' && code <= 'f')
|
|
? std::make_optional(10 + (code - 'a'))
|
|
: (code >= 'A' && code <= 'F')
|
|
? std::make_optional(10 + (code - 'A'))
|
|
: (code >= '0' && code <= '9')
|
|
? std::make_optional(code - '0')
|
|
: std::nullopt;
|
|
};
|
|
const auto h = parse(data[from]), l = parse(data[from + 1]);
|
|
return (h && l) ? std::make_optional(*h * 16 + *l) : std::nullopt;
|
|
};
|
|
const auto r = hex(0), g = hex(2), b = hex(4);
|
|
return (r && g && b) ? QColor(*r, *g, *b) : std::optional<QColor>();
|
|
}
|
|
|
|
[[nodiscard]] QColor ResolveRipple(QColor background) {
|
|
auto hue = 0;
|
|
auto saturation = 0;
|
|
auto lightness = 0;
|
|
auto alpha = 0;
|
|
background.getHsv(&hue, &saturation, &lightness, &alpha);
|
|
return QColor::fromHsv(
|
|
hue,
|
|
saturation,
|
|
lightness - (lightness > kLightnessThreshold
|
|
? kLightnessDelta
|
|
: -kLightnessDelta),
|
|
alpha);
|
|
}
|
|
|
|
[[nodiscard]] const style::color *LookupNamedColor(const QString &key) {
|
|
if (key == u"secondary_bg_color"_q) {
|
|
return &st::boxDividerBg;
|
|
} else if (key == u"bottom_bar_bg_color"_q) {
|
|
return &st::windowBg;
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
} // namespace
|
|
|
|
class Panel::Button final : public RippleButton {
|
|
public:
|
|
Button(QWidget *parent, const style::RoundButton &st);
|
|
~Button();
|
|
|
|
void updateBg(QColor bg);
|
|
void updateBg(not_null<const style::color*> paletteBg);
|
|
void updateFg(QColor fg);
|
|
void updateFg(not_null<const style::color*> paletteFg);
|
|
|
|
void updateArgs(ButtonArgs &&args);
|
|
|
|
private:
|
|
void paintEvent(QPaintEvent *e) override;
|
|
|
|
QImage prepareRippleMask() const override;
|
|
QPoint prepareRippleStartPosition() const override;
|
|
|
|
void toggleProgress(bool shown);
|
|
void setupProgressGeometry();
|
|
|
|
std::unique_ptr<Progress> _progress;
|
|
rpl::variable<QString> _textFull;
|
|
Ui::Text::String _text;
|
|
|
|
const style::RoundButton &_st;
|
|
QColor _fg;
|
|
style::owned_color _bg;
|
|
RoundRect _roundRect;
|
|
|
|
rpl::lifetime _bgLifetime;
|
|
rpl::lifetime _fgLifetime;
|
|
|
|
};
|
|
|
|
struct Panel::Progress {
|
|
Progress(QWidget *parent, Fn<QRect()> rect);
|
|
|
|
RpWidget widget;
|
|
InfiniteRadialAnimation animation;
|
|
Animations::Simple shownAnimation;
|
|
bool shown = true;
|
|
rpl::lifetime geometryLifetime;
|
|
};
|
|
|
|
struct Panel::WebviewWithLifetime {
|
|
WebviewWithLifetime(
|
|
QWidget *parent = nullptr,
|
|
Webview::WindowConfig config = Webview::WindowConfig());
|
|
|
|
Webview::Window window;
|
|
std::vector<QPointer<RpWidget>> boxes;
|
|
rpl::lifetime boxesLifetime;
|
|
rpl::lifetime lifetime;
|
|
};
|
|
|
|
Panel::Button::Button(QWidget *parent, const style::RoundButton &st)
|
|
: RippleButton(parent, st.ripple)
|
|
, _st(st)
|
|
, _bg(st::windowBgActive->c)
|
|
, _roundRect(st::callRadius, st::windowBgActive) {
|
|
_textFull.value(
|
|
) | rpl::start_with_next([=](const QString &text) {
|
|
_text.setText(st::semiboldTextStyle, text);
|
|
update();
|
|
}, lifetime());
|
|
|
|
resize(
|
|
_st.padding.left() + _text.maxWidth() + _st.padding.right(),
|
|
_st.padding.top() + _st.height + _st.padding.bottom());
|
|
}
|
|
|
|
Panel::Button::~Button() = default;
|
|
|
|
void Panel::Button::updateBg(QColor bg) {
|
|
_bg.update(bg);
|
|
_roundRect.setColor(_bg.color());
|
|
_bgLifetime.destroy();
|
|
update();
|
|
}
|
|
|
|
void Panel::Button::updateBg(not_null<const style::color*> paletteBg) {
|
|
updateBg((*paletteBg)->c);
|
|
_bgLifetime = style::PaletteChanged(
|
|
) | rpl::start_with_next([=] {
|
|
updateBg((*paletteBg)->c);
|
|
});
|
|
}
|
|
|
|
void Panel::Button::updateFg(QColor fg) {
|
|
_fg = fg;
|
|
_fgLifetime.destroy();
|
|
update();
|
|
}
|
|
|
|
void Panel::Button::updateFg(not_null<const style::color*> paletteFg) {
|
|
updateFg((*paletteFg)->c);
|
|
_fgLifetime = style::PaletteChanged(
|
|
) | rpl::start_with_next([=] {
|
|
updateFg((*paletteFg)->c);
|
|
});
|
|
}
|
|
|
|
void Panel::Button::updateArgs(ButtonArgs &&args) {
|
|
_textFull = std::move(args.text);
|
|
setDisabled(!args.isActive);
|
|
setPointerCursor(false);
|
|
setCursor(args.isActive ? style::cur_pointer : Qt::ForbiddenCursor);
|
|
setVisible(args.isVisible);
|
|
toggleProgress(args.isProgressVisible);
|
|
update();
|
|
}
|
|
|
|
void Panel::Button::toggleProgress(bool shown) {
|
|
if (!_progress) {
|
|
if (!shown) {
|
|
return;
|
|
}
|
|
_progress = std::make_unique<Progress>(
|
|
this,
|
|
[=] { return _progress->widget.rect(); });
|
|
_progress->widget.paintRequest(
|
|
) | rpl::start_with_next([=](QRect clip) {
|
|
auto p = QPainter(&_progress->widget);
|
|
p.setOpacity(
|
|
_progress->shownAnimation.value(_progress->shown ? 1. : 0.));
|
|
auto thickness = st::paymentsLoading.thickness;
|
|
const auto rect = _progress->widget.rect().marginsRemoved(
|
|
{ thickness, thickness, thickness, thickness });
|
|
InfiniteRadialAnimation::Draw(
|
|
p,
|
|
_progress->animation.computeState(),
|
|
rect.topLeft(),
|
|
rect.size() - QSize(),
|
|
_progress->widget.width(),
|
|
_fg,
|
|
thickness);
|
|
}, _progress->widget.lifetime());
|
|
_progress->widget.show();
|
|
_progress->animation.start();
|
|
} else if (_progress->shown == shown) {
|
|
return;
|
|
}
|
|
const auto callback = [=] {
|
|
if (!_progress->shownAnimation.animating() && !_progress->shown) {
|
|
_progress = nullptr;
|
|
} else {
|
|
_progress->widget.update();
|
|
}
|
|
};
|
|
_progress->shown = shown;
|
|
_progress->shownAnimation.start(
|
|
callback,
|
|
shown ? 0. : 1.,
|
|
shown ? 1. : 0.,
|
|
kProgressDuration);
|
|
if (shown) {
|
|
setupProgressGeometry();
|
|
}
|
|
}
|
|
|
|
void Panel::Button::setupProgressGeometry() {
|
|
if (!_progress || !_progress->shown) {
|
|
return;
|
|
}
|
|
_progress->geometryLifetime.destroy();
|
|
sizeValue(
|
|
) | rpl::start_with_next([=](QSize outer) {
|
|
const auto height = outer.height();
|
|
const auto size = st::paymentsLoading.size;
|
|
const auto skip = (height - size.height()) / 2;
|
|
const auto right = outer.width();
|
|
const auto top = outer.height() - height;
|
|
_progress->widget.setGeometry(QRect{
|
|
QPoint(right - skip - size.width(), top + skip),
|
|
size });
|
|
}, _progress->geometryLifetime);
|
|
|
|
_progress->widget.show();
|
|
_progress->widget.raise();
|
|
if (_progress->shown
|
|
&& Ui::AppInFocus()
|
|
&& Ui::InFocusChain(_progress->widget.window())) {
|
|
_progress->widget.setFocus();
|
|
}
|
|
}
|
|
|
|
void Panel::Button::paintEvent(QPaintEvent *e) {
|
|
Painter p(this);
|
|
|
|
_roundRect.paint(p, rect());
|
|
|
|
if (!isDisabled()) {
|
|
const auto ripple = ResolveRipple(_bg.color()->c);
|
|
paintRipple(p, rect().topLeft(), &ripple);
|
|
}
|
|
|
|
p.setFont(_st.style.font);
|
|
|
|
const auto height = rect().height();
|
|
const auto progress = st::paymentsLoading.size;
|
|
const auto skip = (height - progress.height()) / 2;
|
|
const auto padding = skip + progress.width() + skip;
|
|
|
|
const auto space = width() - padding * 2;
|
|
const auto textWidth = std::min(space, _text.maxWidth());
|
|
const auto textTop = _st.padding.top() + _st.textTop;
|
|
const auto textLeft = padding + (space - textWidth) / 2;
|
|
p.setPen(_fg);
|
|
_text.drawLeftElided(p, textLeft, textTop, textWidth, width());
|
|
}
|
|
|
|
QImage Panel::Button::prepareRippleMask() const {
|
|
return RippleAnimation::RoundRectMask(size(), st::callRadius);
|
|
}
|
|
|
|
QPoint Panel::Button::prepareRippleStartPosition() const {
|
|
return mapFromGlobal(QCursor::pos())
|
|
- QPoint(_st.padding.left(), _st.padding.top());
|
|
}
|
|
|
|
Panel::WebviewWithLifetime::WebviewWithLifetime(
|
|
QWidget *parent,
|
|
Webview::WindowConfig config)
|
|
: window(parent, std::move(config)) {
|
|
}
|
|
|
|
Panel::Progress::Progress(QWidget *parent, Fn<QRect()> rect)
|
|
: widget(parent)
|
|
, animation(
|
|
[=] { if (!anim::Disabled()) widget.update(rect()); },
|
|
st::paymentsLoading) {
|
|
}
|
|
|
|
Panel::Panel(Args &&args)
|
|
: _storageId(args.storageId)
|
|
, _delegate(args.delegate)
|
|
, _menuButtons(args.menuButtons)
|
|
, _widget(std::make_unique<SeparatePanel>(Ui::SeparatePanelArgs{
|
|
.menuSt = &st::botWebViewMenu,
|
|
}))
|
|
, _fullscreen(args.fullscreen)
|
|
, _allowClipboardRead(args.allowClipboardRead) {
|
|
_widget->setWindowFlag(Qt::WindowStaysOnTopHint, false);
|
|
|
|
const auto settings = &AyuSettings::getInstance();
|
|
auto size = QSize(st::botWebViewPanelSize);
|
|
if (settings->increaseWebviewHeight) {
|
|
size.setHeight(st::botWebViewPanelHeightIncreased);
|
|
}
|
|
if (settings->increaseWebviewWidth) {
|
|
size.setWidth(st::botWebViewPanelWidthIncreased);
|
|
}
|
|
|
|
_widget->setInnerSize(size, true);
|
|
|
|
const auto params = _delegate->botThemeParams();
|
|
updateColorOverrides(params);
|
|
|
|
_fullscreen.value(
|
|
) | rpl::start_with_next([=](bool fullscreen) {
|
|
_widget->toggleFullScreen(fullscreen);
|
|
layoutButtons();
|
|
sendFullScreen();
|
|
sendSafeArea();
|
|
sendContentSafeArea();
|
|
}, _widget->lifetime());
|
|
|
|
_widget->fullScreenValue(
|
|
) | rpl::start_with_next([=](bool fullscreen) {
|
|
_fullscreen = fullscreen;
|
|
}, _widget->lifetime());
|
|
|
|
_widget->closeRequests(
|
|
) | rpl::start_with_next([=] {
|
|
if (_closeNeedConfirmation) {
|
|
scheduleCloseWithConfirmation();
|
|
} else {
|
|
_delegate->botClose();
|
|
}
|
|
}, _widget->lifetime());
|
|
|
|
_widget->closeEvents(
|
|
) | rpl::filter([=] {
|
|
return !_hiddenForPayment;
|
|
}) | rpl::start_with_next([=] {
|
|
_delegate->botClose();
|
|
}, _widget->lifetime());
|
|
|
|
_widget->backRequests(
|
|
) | rpl::start_with_next([=] {
|
|
postEvent("back_button_pressed");
|
|
}, _widget->lifetime());
|
|
|
|
rpl::merge(
|
|
style::PaletteChanged(),
|
|
_themeUpdateForced.events()
|
|
) | rpl::filter([=] {
|
|
return !_themeUpdateScheduled;
|
|
}) | rpl::start_with_next([=] {
|
|
_themeUpdateScheduled = true;
|
|
crl::on_main(_widget.get(), [=] {
|
|
_themeUpdateScheduled = false;
|
|
updateThemeParams(_delegate->botThemeParams());
|
|
});
|
|
}, _widget->lifetime());
|
|
|
|
setTitle(std::move(args.title));
|
|
_widget->setTitleBadge(std::move(args.titleBadge));
|
|
|
|
if (!showWebview(std::move(args), params)) {
|
|
const auto available = Webview::Availability();
|
|
if (available.error != Webview::Available::Error::None) {
|
|
showWebviewError(tr::lng_bot_no_webview(tr::now), available);
|
|
} else {
|
|
showCriticalError({ "Error: Could not initialize WebView." });
|
|
}
|
|
}
|
|
}
|
|
|
|
Panel::~Panel() {
|
|
base::take(_webview);
|
|
_progress = nullptr;
|
|
_widget = nullptr;
|
|
}
|
|
|
|
void Panel::setupDownloadsProgress(
|
|
not_null<RpWidget*> button,
|
|
rpl::producer<DownloadsProgress> progress,
|
|
bool fullscreen) {
|
|
const auto widget = Ui::CreateChild<RpWidget>(button.get());
|
|
widget->show();
|
|
widget->setAttribute(Qt::WA_TransparentForMouseEvents);
|
|
|
|
button->sizeValue() | rpl::start_with_next([=](QSize size) {
|
|
widget->setGeometry(QRect(QPoint(), size));
|
|
}, widget->lifetime());
|
|
|
|
struct State {
|
|
State(QWidget *parent)
|
|
: animation([=](crl::time now) {
|
|
const auto total = progress.total;
|
|
const auto current = total
|
|
? (progress.ready / float64(total))
|
|
: 0.;
|
|
const auto updated = animation.update(current, false, now);
|
|
if (!anim::Disabled() || updated) {
|
|
parent->update();
|
|
}
|
|
}) {
|
|
}
|
|
|
|
DownloadsProgress progress;
|
|
RadialAnimation animation;
|
|
Animations::Simple fade;
|
|
bool shown = false;
|
|
};
|
|
const auto state = widget->lifetime().make_state<State>(widget);
|
|
std::move(
|
|
progress
|
|
) | rpl::start_with_next([=](DownloadsProgress progress) {
|
|
const auto toggle = [&](bool shown) {
|
|
if (state->shown == shown) {
|
|
return;
|
|
}
|
|
state->shown = shown;
|
|
if (shown && !state->fade.animating()) {
|
|
return;
|
|
}
|
|
state->fade.start([=] {
|
|
widget->update();
|
|
if (!state->shown
|
|
&& !state->fade.animating()
|
|
&& (!state->progress.total
|
|
|| (state->progress.ready
|
|
== state->progress.total))) {
|
|
state->animation.stop();
|
|
}
|
|
}, shown ? 0. : 2., shown ? 2. : 0., st::radialDuration * 2);
|
|
};
|
|
if (!state->shown && progress.loading) {
|
|
if (!state->animation.animating()) {
|
|
state->animation.start(0.);
|
|
}
|
|
toggle(true);
|
|
} else if ((state->progress.total && !progress.total)
|
|
|| (state->progress.ready < state->progress.total
|
|
&& progress.ready == progress.total)) {
|
|
state->animation.update(1., false, crl::now());
|
|
toggle(false);
|
|
}
|
|
state->progress = progress;
|
|
}, widget->lifetime());
|
|
|
|
widget->paintRequest() | rpl::start_with_next([=] {
|
|
const auto opacity = std::clamp(
|
|
state->fade.value(state->shown ? 2. : 0.) - 1.,
|
|
0.,
|
|
1.);
|
|
if (!opacity) {
|
|
return;
|
|
}
|
|
auto p = QPainter(widget);
|
|
p.setOpacity(opacity);
|
|
const auto palette = _widget->titleOverridePalette();
|
|
const auto color = fullscreen
|
|
? st::radialFg
|
|
: palette
|
|
? palette->boxTitleCloseFg()
|
|
: st::paymentsLoading.color;
|
|
const auto &st = fullscreen
|
|
? st::fullScreenPanelMenu
|
|
: st::separatePanelMenu;
|
|
const auto size = st.rippleAreaSize;
|
|
const auto rect = QRect(st.rippleAreaPosition, QSize(size, size));
|
|
const auto stroke = st::botWebViewRadialStroke;
|
|
const auto shift = stroke * 1.5;
|
|
const auto inner = QRectF(rect).marginsRemoved(
|
|
QMarginsF{ shift, shift, shift, shift });
|
|
state->animation.draw(p, inner, stroke, color);
|
|
}, widget->lifetime());
|
|
}
|
|
|
|
void Panel::requestActivate() {
|
|
_widget->showAndActivate();
|
|
if (const auto widget = _webview ? _webview->window.widget() : nullptr) {
|
|
InvokeQueued(widget, [=] {
|
|
if (widget->isVisible()) {
|
|
_webview->window.focus();
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
void Panel::toggleProgress(bool shown) {
|
|
if (!_progress) {
|
|
if (!shown) {
|
|
return;
|
|
}
|
|
_progress = std::make_unique<Progress>(
|
|
_widget.get(),
|
|
[=] { return progressRect(); });
|
|
_progress->widget.paintRequest(
|
|
) | rpl::start_with_next([=](QRect clip) {
|
|
auto p = QPainter(&_progress->widget);
|
|
p.setOpacity(
|
|
_progress->shownAnimation.value(_progress->shown ? 1. : 0.));
|
|
const auto thickness = st::paymentsLoading.thickness;
|
|
if (progressWithBackground()) {
|
|
auto color = st::windowBg->c;
|
|
color.setAlphaF(kProgressOpacity);
|
|
p.fillRect(clip, color);
|
|
}
|
|
const auto rect = progressRect() - Margins(thickness);
|
|
InfiniteRadialAnimation::Draw(
|
|
p,
|
|
_progress->animation.computeState(),
|
|
rect.topLeft(),
|
|
rect.size() - QSize(),
|
|
_progress->widget.width(),
|
|
st::paymentsLoading.color,
|
|
anim::Disabled() ? (thickness / 2.) : thickness);
|
|
}, _progress->widget.lifetime());
|
|
_progress->widget.show();
|
|
_progress->animation.start();
|
|
} else if (_progress->shown == shown) {
|
|
return;
|
|
}
|
|
const auto callback = [=] {
|
|
if (!_progress->shownAnimation.animating() && !_progress->shown) {
|
|
_progress = nullptr;
|
|
} else {
|
|
_progress->widget.update();
|
|
}
|
|
};
|
|
_progress->shown = shown;
|
|
_progress->shownAnimation.start(
|
|
callback,
|
|
shown ? 0. : 1.,
|
|
shown ? 1. : 0.,
|
|
kProgressDuration);
|
|
if (shown) {
|
|
setupProgressGeometry();
|
|
}
|
|
}
|
|
|
|
bool Panel::progressWithBackground() const {
|
|
return (_progress->widget.width() == _widget->innerGeometry().width());
|
|
}
|
|
|
|
QRect Panel::progressRect() const {
|
|
const auto rect = _progress->widget.rect();
|
|
if (!progressWithBackground()) {
|
|
return rect;
|
|
}
|
|
const auto size = st::defaultBoxButton.height;
|
|
return QRect(
|
|
rect.x() + (rect.width() - size) / 2,
|
|
rect.y() + (rect.height() - size) / 2,
|
|
size,
|
|
size);
|
|
}
|
|
|
|
void Panel::setupProgressGeometry() {
|
|
if (!_progress || !_progress->shown) {
|
|
return;
|
|
}
|
|
_progress->geometryLifetime.destroy();
|
|
if (_webviewBottom) {
|
|
_webviewBottom->geometryValue(
|
|
) | rpl::start_with_next([=](QRect bottom) {
|
|
const auto height = bottom.height();
|
|
const auto size = st::paymentsLoading.size;
|
|
const auto skip = (height - size.height()) / 2;
|
|
const auto inner = _widget->innerGeometry();
|
|
const auto right = inner.x() + inner.width();
|
|
const auto top = inner.y() + inner.height() - height;
|
|
// This doesn't work, because first we get the correct bottom
|
|
// geometry and after that we get the previous event (which
|
|
// triggered the 'fire' of correct geometry before getting here).
|
|
//const auto right = bottom.x() + bottom.width();
|
|
//const auto top = bottom.y();
|
|
_progress->widget.setGeometry(QRect{
|
|
QPoint(right - skip - size.width(), top + skip),
|
|
size });
|
|
}, _progress->geometryLifetime);
|
|
}
|
|
_progress->widget.show();
|
|
_progress->widget.raise();
|
|
if (_progress->shown) {
|
|
_progress->widget.setFocus();
|
|
}
|
|
}
|
|
|
|
void Panel::showWebviewProgress() {
|
|
if (_webviewProgress && _progress && _progress->shown) {
|
|
return;
|
|
}
|
|
_webviewProgress = true;
|
|
toggleProgress(true);
|
|
}
|
|
|
|
void Panel::hideWebviewProgress() {
|
|
if (!_webviewProgress) {
|
|
return;
|
|
}
|
|
_webviewProgress = false;
|
|
toggleProgress(false);
|
|
}
|
|
|
|
bool Panel::showWebview(Args &&args, const Webview::ThemeParams ¶ms) {
|
|
_bottomText = std::move(args.bottom);
|
|
if (!_webview && !createWebview(params)) {
|
|
return false;
|
|
}
|
|
const auto allowBack = false;
|
|
showWebviewProgress();
|
|
_widget->hideLayer(anim::type::instant);
|
|
updateThemeParams(params);
|
|
const auto url = args.url;
|
|
_webview->window.navigate(url);
|
|
_widget->setBackAllowed(allowBack);
|
|
|
|
rpl::duplicate(args.downloadsProgress) | rpl::start_with_next([=] {
|
|
_downloadsUpdated.fire({});
|
|
}, lifetime());
|
|
|
|
_widget->setMenuAllowed([=](
|
|
const Ui::Menu::MenuCallback &callback) {
|
|
auto list = _delegate->botDownloads(true);
|
|
if (!list.empty()) {
|
|
auto value = rpl::single(
|
|
std::move(list)
|
|
) | rpl::then(_downloadsUpdated.events(
|
|
) | rpl::map([=] {
|
|
return _delegate->botDownloads();
|
|
}));
|
|
const auto action = [=](uint32 id, DownloadsAction type) {
|
|
_delegate->botDownloadsAction(id, type);
|
|
};
|
|
callback(Ui::Menu::MenuCallback::Args{
|
|
.text = tr::lng_downloads_section(tr::now),
|
|
.icon = &st::menuIconDownload,
|
|
.fillSubmenu = FillAttachBotDownloadsSubmenu(
|
|
std::move(value),
|
|
action),
|
|
});
|
|
callback({
|
|
.separatorSt = &st::expandedMenuSeparator,
|
|
.isSeparator = true,
|
|
});
|
|
}
|
|
if (_webview && _webview->window.widget() && _hasSettingsButton) {
|
|
callback(tr::lng_bot_settings(tr::now), [=] {
|
|
postEvent("settings_button_pressed");
|
|
}, &st::menuIconSettings);
|
|
}
|
|
if (_menuButtons & MenuButton::OpenBot) {
|
|
callback(tr::lng_bot_open(tr::now), [=] {
|
|
_delegate->botHandleMenuButton(MenuButton::OpenBot);
|
|
}, &st::menuIconLeave);
|
|
}
|
|
callback(tr::lng_bot_reload_page(tr::now), [=] {
|
|
if (_webview && _webview->window.widget()) {
|
|
_webview->window.reload();
|
|
} else if (const auto params = _delegate->botThemeParams()
|
|
; createWebview(params)) {
|
|
showWebviewProgress();
|
|
updateThemeParams(params);
|
|
_webview->window.navigate(url);
|
|
}
|
|
}, &st::menuIconRestore);
|
|
if (_menuButtons & MenuButton::ShareGame) {
|
|
callback(tr::lng_iv_share(tr::now), [=] {
|
|
_delegate->botHandleMenuButton(MenuButton::ShareGame);
|
|
}, &st::menuIconShare);
|
|
} else {
|
|
callback(tr::lng_bot_terms(tr::now), [=] {
|
|
File::OpenUrl(tr::lng_mini_apps_tos_url(tr::now));
|
|
}, &st::menuIconGroupLog);
|
|
callback(tr::lng_bot_privacy(tr::now), [=] {
|
|
_delegate->botOpenPrivacyPolicy();
|
|
}, &st::menuIconAntispam);
|
|
}
|
|
const auto main = (_menuButtons & MenuButton::RemoveFromMainMenu);
|
|
if (main || (_menuButtons & MenuButton::RemoveFromMenu)) {
|
|
const auto handler = [=] {
|
|
_delegate->botHandleMenuButton(main
|
|
? MenuButton::RemoveFromMainMenu
|
|
: MenuButton::RemoveFromMenu);
|
|
};
|
|
callback({
|
|
.text = (main
|
|
? tr::lng_bot_remove_from_side_menu
|
|
: tr::lng_bot_remove_from_menu)(tr::now),
|
|
.handler = handler,
|
|
.icon = &st::menuIconDeleteAttention,
|
|
.isAttention = true,
|
|
});
|
|
}
|
|
}, [=, progress = std::move(args.downloadsProgress)](
|
|
not_null<RpWidget*> button,
|
|
bool fullscreen) {
|
|
setupDownloadsProgress(
|
|
button,
|
|
rpl::duplicate(progress),
|
|
fullscreen);
|
|
});
|
|
|
|
return true;
|
|
}
|
|
|
|
void Panel::createWebviewBottom() {
|
|
_webviewBottom = std::make_unique<RpWidget>(_widget.get());
|
|
const auto bottom = _webviewBottom.get();
|
|
bottom->setVisible(!_fullscreen.current());
|
|
|
|
const auto &padding = st::paymentsPanelPadding;
|
|
const auto label = CreateChild<FlatLabel>(
|
|
_webviewBottom.get(),
|
|
_bottomText.value(),
|
|
st::paymentsWebviewBottom);
|
|
_webviewBottomLabel = label;
|
|
|
|
const auto height = padding.top()
|
|
+ label->heightNoMargins()
|
|
+ padding.bottom();
|
|
rpl::combine(
|
|
_webviewBottom->widthValue(),
|
|
label->widthValue()
|
|
) | rpl::start_with_next([=](int outerWidth, int width) {
|
|
label->move((outerWidth - width) / 2, padding.top());
|
|
}, label->lifetime());
|
|
label->show();
|
|
_webviewBottom->resize(_webviewBottom->width(), height);
|
|
|
|
rpl::combine(
|
|
_webviewParent->geometryValue() | rpl::map([=] {
|
|
return _widget->innerGeometry();
|
|
}),
|
|
bottom->heightValue()
|
|
) | rpl::start_with_next([=](QRect inner, int height) {
|
|
bottom->move(inner.x(), inner.y() + inner.height() - height);
|
|
bottom->resizeToWidth(inner.width());
|
|
layoutButtons();
|
|
}, bottom->lifetime());
|
|
}
|
|
|
|
bool Panel::createWebview(const Webview::ThemeParams ¶ms) {
|
|
auto outer = base::make_unique_q<RpWidget>(_widget.get());
|
|
const auto container = outer.get();
|
|
_widget->showInner(std::move(outer));
|
|
_webviewParent = container;
|
|
|
|
_headerColorReceived = false;
|
|
_bodyColorReceived = false;
|
|
_bottomColorReceived = false;
|
|
updateColorOverrides(params);
|
|
createWebviewBottom();
|
|
|
|
container->show();
|
|
_webview = std::make_unique<WebviewWithLifetime>(
|
|
container,
|
|
Webview::WindowConfig{
|
|
.opaqueBg = params.bodyBg,
|
|
.storageId = _storageId,
|
|
});
|
|
const auto raw = &_webview->window;
|
|
|
|
const auto bottom = _webviewBottom.get();
|
|
QObject::connect(container, &QObject::destroyed, [=] {
|
|
if (_webview && &_webview->window == raw) {
|
|
base::take(_webview);
|
|
if (_webviewProgress) {
|
|
hideWebviewProgress();
|
|
if (_progress && !_progress->shown) {
|
|
_progress = nullptr;
|
|
}
|
|
}
|
|
}
|
|
if (_webviewBottom.get() == bottom) {
|
|
_webviewBottomLabel = nullptr;
|
|
_webviewBottom = nullptr;
|
|
_secondaryButton = nullptr;
|
|
_mainButton = nullptr;
|
|
_bottomButtonsBg = nullptr;
|
|
}
|
|
});
|
|
if (!raw->widget()) {
|
|
return false;
|
|
}
|
|
|
|
#if !defined Q_OS_WIN && !defined Q_OS_MAC
|
|
_widget->allowChildFullScreenControls(
|
|
!raw->widget()->inherits("QWindowContainer"));
|
|
#endif // !Q_OS_WIN && !Q_OS_MAC
|
|
|
|
QObject::connect(raw->widget(), &QObject::destroyed, [=] {
|
|
const auto parent = _webviewParent.data();
|
|
if (!_webview
|
|
|| &_webview->window != raw
|
|
|| !parent
|
|
|| _widget->inner() != parent) {
|
|
// If we destroyed _webview ourselves,
|
|
// or if we changed _widget->inner ourselves,
|
|
// we don't show any message, nothing crashed.
|
|
return;
|
|
}
|
|
crl::on_main(this, [=] {
|
|
showCriticalError({ "Error: WebView has crashed." });
|
|
});
|
|
});
|
|
|
|
rpl::combine(
|
|
container->geometryValue(),
|
|
_footerHeight.value()
|
|
) | rpl::start_with_next([=](QRect geometry, int footer) {
|
|
if (const auto view = raw->widget()) {
|
|
view->setGeometry(geometry.marginsRemoved({ 0, 0, 0, footer }));
|
|
crl::on_main(view, [=] {
|
|
sendViewport();
|
|
InvokeQueued(view, [=] { sendViewport(); });
|
|
});
|
|
}
|
|
}, _webview->lifetime);
|
|
|
|
raw->setMessageHandler([=](const QJsonDocument &message) {
|
|
if (!message.isArray()) {
|
|
LOG(("BotWebView Error: "
|
|
"Not an array received in buy_callback arguments."));
|
|
return;
|
|
}
|
|
const auto list = message.array();
|
|
const auto command = list.at(0).toString();
|
|
const auto arguments = ParseMethodArgs(list.at(1).toString());
|
|
if (command == "web_app_close") {
|
|
_delegate->botClose();
|
|
} else if (command == "web_app_data_send") {
|
|
sendDataMessage(arguments);
|
|
} else if (command == "web_app_switch_inline_query") {
|
|
switchInlineQueryMessage(arguments);
|
|
} else if (command == "web_app_setup_main_button") {
|
|
processButtonMessage(_mainButton, arguments);
|
|
} else if (command == "web_app_setup_secondary_button") {
|
|
processButtonMessage(_secondaryButton, arguments);
|
|
} else if (command == "web_app_setup_back_button") {
|
|
processBackButtonMessage(arguments);
|
|
} else if (command == "web_app_setup_settings_button") {
|
|
processSettingsButtonMessage(arguments);
|
|
} else if (command == "web_app_request_theme") {
|
|
_themeUpdateForced.fire({});
|
|
} else if (command == "web_app_request_viewport") {
|
|
sendViewport();
|
|
} else if (command == "web_app_request_safe_area") {
|
|
sendSafeArea();
|
|
} else if (command == "web_app_request_content_safe_area") {
|
|
sendContentSafeArea();
|
|
} else if (command == "web_app_request_fullscreen") {
|
|
if (!_fullscreen.current()) {
|
|
_fullscreen = true;
|
|
} else {
|
|
sendFullScreen();
|
|
}
|
|
} else if (command == "web_app_request_file_download") {
|
|
processDownloadRequest(arguments);
|
|
} else if (command == "web_app_exit_fullscreen") {
|
|
if (_fullscreen.current()) {
|
|
_fullscreen = false;
|
|
} else {
|
|
sendFullScreen();
|
|
}
|
|
} else if (command == "web_app_check_home_screen") {
|
|
postEvent("home_screen_checked", "{ status: \"unsupported\" }");
|
|
} else if (command == "web_app_start_accelerometer") {
|
|
postEvent("accelerometer_failed", "{ error: \"UNSUPPORTED\" }");
|
|
} else if (command == "web_app_start_device_orientation") {
|
|
postEvent(
|
|
"device_orientation_failed",
|
|
"{ error: \"UNSUPPORTED\" }");
|
|
} else if (command == "web_app_start_gyroscope") {
|
|
postEvent("gyroscope_failed", "{ error: \"UNSUPPORTED\" }");
|
|
} else if (command == "web_app_check_location") {
|
|
postEvent("location_checked", "{ available: false }");
|
|
} else if (command == "web_app_request_location") {
|
|
postEvent("location_requested", "{ available: false }");
|
|
} else if (command == "web_app_biometry_get_info") {
|
|
postEvent("biometry_info_received", "{ available: false }");
|
|
} else if (command == "web_app_open_tg_link") {
|
|
openTgLink(arguments);
|
|
} else if (command == "web_app_open_link") {
|
|
openExternalLink(arguments);
|
|
} else if (command == "web_app_open_invoice") {
|
|
openInvoice(arguments);
|
|
} else if (command == "web_app_open_popup") {
|
|
openPopup(arguments);
|
|
} else if (command == "web_app_open_scan_qr_popup") {
|
|
openScanQrPopup(arguments);
|
|
} else if (command == "web_app_share_to_story") {
|
|
openShareStory(arguments);
|
|
} else if (command == "web_app_request_write_access") {
|
|
requestWriteAccess();
|
|
} else if (command == "web_app_request_phone") {
|
|
requestPhone();
|
|
} else if (command == "web_app_invoke_custom_method") {
|
|
invokeCustomMethod(arguments);
|
|
} else if (command == "web_app_setup_closing_behavior") {
|
|
setupClosingBehaviour(arguments);
|
|
} else if (command == "web_app_read_text_from_clipboard") {
|
|
requestClipboardText(arguments);
|
|
} else if (command == "web_app_set_header_color") {
|
|
processHeaderColor(arguments);
|
|
} else if (command == "web_app_set_background_color") {
|
|
processBackgroundColor(arguments);
|
|
} else if (command == "web_app_set_bottom_bar_color") {
|
|
processBottomBarColor(arguments);
|
|
} else if (command == "web_app_send_prepared_message") {
|
|
processSendMessageRequest(arguments);
|
|
} else if (command == "web_app_set_emoji_status") {
|
|
processEmojiStatusRequest(arguments);
|
|
} else if (command == "web_app_request_emoji_status_access") {
|
|
processEmojiStatusAccessRequest();
|
|
} else if (command == "share_score") {
|
|
_delegate->botHandleMenuButton(MenuButton::ShareGame);
|
|
}
|
|
});
|
|
|
|
raw->setNavigationStartHandler([=](const QString &uri, bool newWindow) {
|
|
if (_delegate->botHandleLocalUri(uri, false)) {
|
|
return false;
|
|
} else if (newWindow) {
|
|
return true;
|
|
}
|
|
showWebviewProgress();
|
|
return true;
|
|
});
|
|
raw->setNavigationDoneHandler([=](bool success) {
|
|
hideWebviewProgress();
|
|
});
|
|
|
|
raw->init(R"(
|
|
window.TelegramWebviewProxy = {
|
|
postEvent: function(eventType, eventData) {
|
|
if (window.external && window.external.invoke) {
|
|
window.external.invoke(JSON.stringify([eventType, eventData]));
|
|
}
|
|
}
|
|
};)");
|
|
|
|
if (!_webview) {
|
|
return false;
|
|
}
|
|
|
|
layoutButtons();
|
|
setupProgressGeometry();
|
|
|
|
base::qt_signal_producer(
|
|
qApp,
|
|
&QGuiApplication::focusWindowChanged
|
|
) | rpl::filter([=](QWindow *focused) {
|
|
const auto handle = _widget->window()->windowHandle();
|
|
const auto widget = _webview ? _webview->window.widget() : nullptr;
|
|
return widget
|
|
&& !widget->isHidden()
|
|
&& handle
|
|
&& (focused == handle);
|
|
}) | rpl::start_with_next([=] {
|
|
_webview->window.focus();
|
|
}, _webview->lifetime);
|
|
|
|
return true;
|
|
}
|
|
|
|
void Panel::sendViewport() {
|
|
postEvent("viewport_changed", "{ "
|
|
"height: window.innerHeight, "
|
|
"is_state_stable: true, "
|
|
"is_expanded: true }");
|
|
}
|
|
|
|
void Panel::sendFullScreen() {
|
|
postEvent("fullscreen_changed", _fullscreen.current()
|
|
? "{ is_fullscreen: true }"
|
|
: "{ is_fullscreen: false }");
|
|
}
|
|
|
|
void Panel::sendSafeArea() {
|
|
postEvent("safe_area_changed",
|
|
"{ top: 0, right: 0, bottom: 0, left: 0 }");
|
|
}
|
|
|
|
void Panel::sendContentSafeArea() {
|
|
const auto shift = st::separatePanelClose.rippleAreaPosition.y();
|
|
const auto top = _fullscreen.current()
|
|
? (shift + st::fullScreenPanelClose.height + (shift / 2))
|
|
: 0;
|
|
const auto scaled = top * style::DevicePixelRatio();
|
|
auto report = 0;
|
|
if (const auto screen = QGuiApplication::primaryScreen()) {
|
|
const auto dpi = screen->logicalDotsPerInch();
|
|
const auto ratio = screen->devicePixelRatio();
|
|
const auto basePair = screen->handle()->logicalBaseDpi();
|
|
const auto base = (basePair.first + basePair.second) * 0.5;
|
|
const auto systemScreenScale = dpi * ratio / base;
|
|
report = int(base::SafeRound(scaled / systemScreenScale));
|
|
}
|
|
postEvent("content_safe_area_changed",
|
|
u"{ top: %1, right: 0, bottom: 0, left: 0 }"_q.arg(report));
|
|
}
|
|
|
|
void Panel::setTitle(rpl::producer<QString> title) {
|
|
_widget->setTitle(std::move(title));
|
|
}
|
|
|
|
void Panel::sendDataMessage(const QJsonObject &args) {
|
|
if (args.isEmpty()) {
|
|
_delegate->botClose();
|
|
return;
|
|
}
|
|
const auto data = args["data"].toString();
|
|
if (data.isEmpty()) {
|
|
LOG(("BotWebView Error: Bad 'data' in sendDataMessage."));
|
|
_delegate->botClose();
|
|
return;
|
|
}
|
|
_delegate->botSendData(data.toUtf8());
|
|
}
|
|
|
|
void Panel::switchInlineQueryMessage(const QJsonObject &args) {
|
|
if (args.isEmpty()) {
|
|
_delegate->botClose();
|
|
return;
|
|
}
|
|
const auto query = args["query"].toString();
|
|
if (query.isEmpty()) {
|
|
LOG(("BotWebView Error: Bad 'query' in switchInlineQueryMessage."));
|
|
_delegate->botClose();
|
|
return;
|
|
}
|
|
const auto valid = base::flat_set<QString>{
|
|
u"users"_q,
|
|
u"bots"_q,
|
|
u"groups"_q,
|
|
u"channels"_q,
|
|
};
|
|
const auto typeArray = args["chat_types"].toArray();
|
|
auto types = std::vector<QString>();
|
|
for (const auto &value : typeArray) {
|
|
const auto type = value.toString();
|
|
if (valid.contains(type)) {
|
|
types.push_back(type);
|
|
} else {
|
|
LOG(("BotWebView Error: "
|
|
"Bad chat type in switchInlineQueryMessage: %1.").arg(type));
|
|
types.clear();
|
|
break;
|
|
}
|
|
}
|
|
_delegate->botSwitchInlineQuery(types, query);
|
|
}
|
|
|
|
void Panel::processSendMessageRequest(const QJsonObject &args) {
|
|
if (args.isEmpty()) {
|
|
_delegate->botClose();
|
|
return;
|
|
}
|
|
const auto id = args["id"].toString();
|
|
auto callback = crl::guard(this, [=](QString error) {
|
|
if (error.isEmpty()) {
|
|
postEvent("prepared_message_sent");
|
|
} else {
|
|
postEvent(
|
|
"prepared_message_failed",
|
|
u"{ error: \"%1\" }"_q.arg(error));
|
|
}
|
|
});
|
|
_delegate->botSendPreparedMessage({
|
|
.id = id,
|
|
.callback = std::move(callback),
|
|
});
|
|
}
|
|
|
|
void Panel::processEmojiStatusRequest(const QJsonObject &args) {
|
|
if (args.isEmpty()) {
|
|
_delegate->botClose();
|
|
return;
|
|
}
|
|
const auto emojiId = args["custom_emoji_id"].toString().toULongLong();
|
|
const auto duration = TimeId(base::SafeRound(
|
|
args["duration"].toDouble()));
|
|
if (!emojiId) {
|
|
postEvent(
|
|
"emoji_status_failed",
|
|
"{ error: \"SUGGESTED_EMOJI_INVALID\" }");
|
|
return;
|
|
} else if (duration < 0) {
|
|
postEvent(
|
|
"emoji_status_failed",
|
|
"{ error: \"DURATION_INVALID\" }");
|
|
return;
|
|
}
|
|
auto callback = crl::guard(this, [=](QString error) {
|
|
if (error.isEmpty()) {
|
|
postEvent("emoji_status_set");
|
|
} else {
|
|
postEvent(
|
|
"emoji_status_failed",
|
|
u"{ error: \"%1\" }"_q.arg(error));
|
|
}
|
|
});
|
|
_delegate->botSetEmojiStatus({
|
|
.customEmojiId = emojiId,
|
|
.duration = duration,
|
|
.callback = std::move(callback),
|
|
});
|
|
}
|
|
|
|
void Panel::processEmojiStatusAccessRequest() {
|
|
auto callback = crl::guard(this, [=](bool allowed) {
|
|
postEvent("emoji_status_access_requested", allowed
|
|
? "{ status: \"allowed\" }"
|
|
: "{ status: \"cancelled\" }");
|
|
});
|
|
_delegate->botRequestEmojiStatusAccess(std::move(callback));
|
|
}
|
|
|
|
void Panel::openTgLink(const QJsonObject &args) {
|
|
if (args.isEmpty()) {
|
|
LOG(("BotWebView Error: Bad arguments in 'web_app_open_tg_link'."));
|
|
_delegate->botClose();
|
|
return;
|
|
}
|
|
const auto path = args["path_full"].toString();
|
|
if (path.isEmpty()) {
|
|
LOG(("BotWebView Error: Bad 'path_full' in 'web_app_open_tg_link'."));
|
|
_delegate->botClose();
|
|
return;
|
|
}
|
|
_delegate->botHandleLocalUri("https://t.me" + path, true);
|
|
}
|
|
|
|
void Panel::openExternalLink(const QJsonObject &args) {
|
|
if (args.isEmpty()) {
|
|
_delegate->botClose();
|
|
return;
|
|
}
|
|
const auto iv = args["try_instant_view"].toBool();
|
|
const auto url = args["url"].toString();
|
|
if (!_delegate->botValidateExternalLink(url)) {
|
|
LOG(("BotWebView Error: Bad url in openExternalLink: %1").arg(url));
|
|
_delegate->botClose();
|
|
return;
|
|
} else if (!allowOpenLink()) {
|
|
return;
|
|
} else if (iv) {
|
|
_delegate->botOpenIvLink(url);
|
|
} else {
|
|
File::OpenUrl(url);
|
|
}
|
|
}
|
|
|
|
void Panel::openInvoice(const QJsonObject &args) {
|
|
if (args.isEmpty()) {
|
|
_delegate->botClose();
|
|
return;
|
|
}
|
|
const auto slug = args["slug"].toString();
|
|
if (slug.isEmpty()) {
|
|
LOG(("BotWebView Error: Bad 'slug' in openInvoice."));
|
|
_delegate->botClose();
|
|
return;
|
|
}
|
|
_delegate->botHandleInvoice(slug);
|
|
}
|
|
|
|
void Panel::openPopup(const QJsonObject &args) {
|
|
if (args.isEmpty()) {
|
|
_delegate->botClose();
|
|
return;
|
|
}
|
|
using Button = Webview::PopupArgs::Button;
|
|
using Type = Button::Type;
|
|
const auto message = args["message"].toString();
|
|
const auto types = base::flat_map<QString, Button::Type>{
|
|
{ "default", Type::Default },
|
|
{ "ok", Type::Ok },
|
|
{ "close", Type::Close },
|
|
{ "cancel", Type::Cancel },
|
|
{ "destructive", Type::Destructive },
|
|
};
|
|
const auto buttonArray = args["buttons"].toArray();
|
|
auto buttons = std::vector<Webview::PopupArgs::Button>();
|
|
for (const auto button : buttonArray) {
|
|
const auto fields = button.toObject();
|
|
const auto i = types.find(fields["type"].toString());
|
|
if (i == end(types)) {
|
|
LOG(("BotWebView Error: Bad 'type' in openPopup buttons."));
|
|
_delegate->botClose();
|
|
return;
|
|
}
|
|
buttons.push_back({
|
|
.id = fields["id"].toString(),
|
|
.text = fields["text"].toString(),
|
|
.type = i->second,
|
|
});
|
|
}
|
|
if (message.isEmpty()) {
|
|
LOG(("BotWebView Error: Bad 'message' in openPopup."));
|
|
_delegate->botClose();
|
|
return;
|
|
} else if (buttons.empty()) {
|
|
LOG(("BotWebView Error: Bad 'buttons' in openPopup."));
|
|
_delegate->botClose();
|
|
return;
|
|
}
|
|
const auto widget = _webview->window.widget();
|
|
const auto weak = base::make_weak(this);
|
|
const auto result = Webview::ShowBlockingPopup({
|
|
.parent = widget ? widget->window() : nullptr,
|
|
.title = args["title"].toString(),
|
|
.text = message,
|
|
.buttons = std::move(buttons),
|
|
});
|
|
if (weak) {
|
|
postEvent("popup_closed", result.id
|
|
? QJsonObject{ { u"button_id"_q, *result.id } }
|
|
: EventData());
|
|
}
|
|
}
|
|
|
|
void Panel::openScanQrPopup(const QJsonObject &args) {
|
|
const auto widget = _webview->window.widget();
|
|
[[maybe_unused]] const auto ok = Webview::ShowBlockingPopup({
|
|
.parent = widget ? widget->window() : nullptr,
|
|
.text = tr::lng_bot_no_scan_qr(tr::now),
|
|
.buttons = { {
|
|
.id = "ok",
|
|
.text = tr::lng_box_ok(tr::now),
|
|
.type = Webview::PopupArgs::Button::Type::Ok,
|
|
}},
|
|
});
|
|
}
|
|
|
|
void Panel::openShareStory(const QJsonObject &args) {
|
|
const auto widget = _webview->window.widget();
|
|
[[maybe_unused]] const auto ok = Webview::ShowBlockingPopup({
|
|
.parent = widget ? widget->window() : nullptr,
|
|
.text = tr::lng_bot_no_share_story(tr::now),
|
|
.buttons = { {
|
|
.id = "ok",
|
|
.text = tr::lng_box_ok(tr::now),
|
|
.type = Webview::PopupArgs::Button::Type::Ok,
|
|
}},
|
|
});
|
|
}
|
|
|
|
void Panel::requestWriteAccess() {
|
|
if (_inBlockingRequest) {
|
|
replyRequestWriteAccess(false);
|
|
return;
|
|
}
|
|
_inBlockingRequest = true;
|
|
const auto finish = [=](bool allowed) {
|
|
_inBlockingRequest = false;
|
|
replyRequestWriteAccess(allowed);
|
|
};
|
|
const auto weak = base::make_weak(this);
|
|
_delegate->botCheckWriteAccess([=](bool allowed) {
|
|
if (!weak) {
|
|
return;
|
|
} else if (allowed) {
|
|
finish(true);
|
|
return;
|
|
}
|
|
using Button = Webview::PopupArgs::Button;
|
|
const auto widget = _webview->window.widget();
|
|
const auto integration = &Ui::Integration::Instance();
|
|
const auto result = Webview::ShowBlockingPopup({
|
|
.parent = widget ? widget->window() : nullptr,
|
|
.title = integration->phraseBotAllowWriteTitle(),
|
|
.text = integration->phraseBotAllowWrite(),
|
|
.buttons = {
|
|
{
|
|
.id = "allow",
|
|
.text = integration->phraseBotAllowWriteConfirm(),
|
|
},
|
|
{ .id = "cancel", .type = Button::Type::Cancel },
|
|
},
|
|
});
|
|
if (!weak) {
|
|
return;
|
|
} else if (result.id == "allow") {
|
|
_delegate->botAllowWriteAccess(crl::guard(this, finish));
|
|
} else {
|
|
finish(false);
|
|
}
|
|
});
|
|
}
|
|
|
|
void Panel::replyRequestWriteAccess(bool allowed) {
|
|
postEvent("write_access_requested", QJsonObject{
|
|
{ u"status"_q, allowed ? u"allowed"_q : u"cancelled"_q }
|
|
});
|
|
}
|
|
|
|
void Panel::requestPhone() {
|
|
if (_inBlockingRequest) {
|
|
replyRequestPhone(false);
|
|
return;
|
|
}
|
|
_inBlockingRequest = true;
|
|
const auto finish = [=](bool shared) {
|
|
_inBlockingRequest = false;
|
|
replyRequestPhone(shared);
|
|
};
|
|
using Button = Webview::PopupArgs::Button;
|
|
const auto widget = _webview->window.widget();
|
|
const auto weak = base::make_weak(this);
|
|
const auto integration = &Ui::Integration::Instance();
|
|
const auto result = Webview::ShowBlockingPopup({
|
|
.parent = widget ? widget->window() : nullptr,
|
|
.title = integration->phraseBotSharePhoneTitle(),
|
|
.text = integration->phraseBotSharePhone(),
|
|
.buttons = {
|
|
{
|
|
.id = "share",
|
|
.text = integration->phraseBotSharePhoneConfirm(),
|
|
},
|
|
{ .id = "cancel", .type = Button::Type::Cancel },
|
|
},
|
|
});
|
|
if (!weak) {
|
|
return;
|
|
} else if (result.id == "share") {
|
|
_delegate->botSharePhone(crl::guard(this, finish));
|
|
} else {
|
|
finish(false);
|
|
}
|
|
}
|
|
|
|
void Panel::replyRequestPhone(bool shared) {
|
|
postEvent("phone_requested", QJsonObject{
|
|
{ u"status"_q, shared ? u"sent"_q : u"cancelled"_q }
|
|
});
|
|
}
|
|
|
|
void Panel::invokeCustomMethod(const QJsonObject &args) {
|
|
const auto requestId = args["req_id"];
|
|
if (requestId.isUndefined()) {
|
|
return;
|
|
}
|
|
const auto finish = [=](QJsonObject response) {
|
|
replyCustomMethod(requestId, std::move(response));
|
|
};
|
|
auto callback = crl::guard(this, [=](CustomMethodResult result) {
|
|
if (result) {
|
|
auto error = QJsonParseError();
|
|
const auto parsed = QJsonDocument::fromJson(
|
|
"{ \"result\": " + *result + '}',
|
|
&error);
|
|
if (error.error != QJsonParseError::NoError
|
|
|| !parsed.isObject()
|
|
|| parsed.object().size() != 1) {
|
|
finish({ { u"error"_q, u"Could not parse response."_q } });
|
|
} else {
|
|
finish(parsed.object());
|
|
}
|
|
} else {
|
|
finish({ { u"error"_q, result.error() } });
|
|
}
|
|
});
|
|
const auto params = QJsonDocument(
|
|
args["params"].toObject()
|
|
).toJson(QJsonDocument::Compact);
|
|
_delegate->botInvokeCustomMethod({
|
|
.method = args["method"].toString(),
|
|
.params = params,
|
|
.callback = std::move(callback),
|
|
});
|
|
}
|
|
|
|
void Panel::replyCustomMethod(QJsonValue requestId, QJsonObject response) {
|
|
response["req_id"] = requestId;
|
|
postEvent(u"custom_method_invoked"_q, response);
|
|
}
|
|
|
|
void Panel::requestClipboardText(const QJsonObject &args) {
|
|
const auto requestId = args["req_id"];
|
|
if (requestId.isUndefined()) {
|
|
return;
|
|
}
|
|
auto result = QJsonObject();
|
|
result["req_id"] = requestId;
|
|
if (allowClipboardQuery()) {
|
|
result["data"] = QGuiApplication::clipboard()->text();
|
|
}
|
|
postEvent(u"clipboard_text_received"_q, result);
|
|
}
|
|
|
|
bool Panel::allowOpenLink() const {
|
|
//const auto now = crl::now();
|
|
//if (_mainButtonLastClick
|
|
// && _mainButtonLastClick + kProcessClickTimeout >= now) {
|
|
// _mainButtonLastClick = 0;
|
|
// return true;
|
|
//}
|
|
return true;
|
|
}
|
|
|
|
bool Panel::allowClipboardQuery() const {
|
|
if (!_allowClipboardRead) {
|
|
return false;
|
|
}
|
|
//const auto now = crl::now();
|
|
//if (_mainButtonLastClick
|
|
// && _mainButtonLastClick + kProcessClickTimeout >= now) {
|
|
// _mainButtonLastClick = 0;
|
|
// return true;
|
|
//}
|
|
return true;
|
|
}
|
|
|
|
void Panel::scheduleCloseWithConfirmation() {
|
|
if (!_closeWithConfirmationScheduled) {
|
|
_closeWithConfirmationScheduled = true;
|
|
InvokeQueued(_widget.get(), [=] { closeWithConfirmation(); });
|
|
}
|
|
}
|
|
|
|
void Panel::closeWithConfirmation() {
|
|
using Button = Webview::PopupArgs::Button;
|
|
const auto widget = _webview->window.widget();
|
|
const auto weak = base::make_weak(this);
|
|
const auto integration = &Ui::Integration::Instance();
|
|
const auto result = Webview::ShowBlockingPopup({
|
|
.parent = widget ? widget->window() : nullptr,
|
|
.title = integration->phrasePanelCloseWarning(),
|
|
.text = integration->phrasePanelCloseUnsaved(),
|
|
.buttons = {
|
|
{
|
|
.id = "close",
|
|
.text = integration->phrasePanelCloseAnyway(),
|
|
.type = Button::Type::Destructive,
|
|
},
|
|
{ .id = "cancel", .type = Button::Type::Cancel },
|
|
},
|
|
.ignoreFloodCheck = true,
|
|
});
|
|
if (!weak) {
|
|
return;
|
|
} else if (result.id == "close") {
|
|
_delegate->botClose();
|
|
} else {
|
|
_closeWithConfirmationScheduled = false;
|
|
}
|
|
}
|
|
|
|
void Panel::setupClosingBehaviour(const QJsonObject &args) {
|
|
_closeNeedConfirmation = args["need_confirmation"].toBool();
|
|
}
|
|
|
|
void Panel::processButtonMessage(
|
|
std::unique_ptr<Button> &button,
|
|
const QJsonObject &args) {
|
|
if (args.isEmpty()) {
|
|
_delegate->botClose();
|
|
return;
|
|
}
|
|
|
|
const auto shown = [&] {
|
|
return button && !button->isHidden();
|
|
};
|
|
const auto wasShown = shown();
|
|
const auto guard = gsl::finally([&] {
|
|
if (shown() != wasShown) {
|
|
crl::on_main(this, [=] {
|
|
sendViewport();
|
|
});
|
|
}
|
|
});
|
|
|
|
const auto text = args["text"].toString().trimmed();
|
|
const auto visible = args["is_visible"].toBool() && !text.isEmpty();
|
|
if (!button) {
|
|
if (visible) {
|
|
createButton(button);
|
|
_bottomButtonsBg->show();
|
|
} else {
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (const auto bg = ParseColor(args["color"].toString())) {
|
|
button->updateBg(*bg);
|
|
} else {
|
|
button->updateBg(&st::windowBgActive);
|
|
}
|
|
|
|
if (const auto fg = ParseColor(args["text_color"].toString())) {
|
|
button->updateFg(*fg);
|
|
} else {
|
|
button->updateFg(&st::windowFgActive);
|
|
}
|
|
|
|
button->updateArgs({
|
|
.isActive = args["is_active"].toBool(),
|
|
.isVisible = visible,
|
|
.isProgressVisible = args["is_progress_visible"].toBool(),
|
|
.text = args["text"].toString(),
|
|
});
|
|
if (button.get() == _secondaryButton.get()) {
|
|
const auto position = ParsePosition(args["position"].toString());
|
|
if (_secondaryPosition != position) {
|
|
_secondaryPosition = position;
|
|
layoutButtons();
|
|
}
|
|
}
|
|
}
|
|
|
|
void Panel::processBackButtonMessage(const QJsonObject &args) {
|
|
_widget->setBackAllowed(args["is_visible"].toBool());
|
|
}
|
|
|
|
void Panel::processSettingsButtonMessage(const QJsonObject &args) {
|
|
_hasSettingsButton = args["is_visible"].toBool();
|
|
}
|
|
|
|
void Panel::processHeaderColor(const QJsonObject &args) {
|
|
_headerColorReceived = true;
|
|
if (const auto color = ParseColor(args["color"].toString())) {
|
|
_widget->overrideTitleColor(color);
|
|
_headerColorLifetime.destroy();
|
|
} else if (const auto color = LookupNamedColor(
|
|
args["color_key"].toString())) {
|
|
_widget->overrideTitleColor((*color)->c);
|
|
_headerColorLifetime = style::PaletteChanged(
|
|
) | rpl::start_with_next([=] {
|
|
_widget->overrideTitleColor((*color)->c);
|
|
});
|
|
} else {
|
|
_widget->overrideTitleColor(std::nullopt);
|
|
_headerColorLifetime.destroy();
|
|
}
|
|
}
|
|
|
|
void Panel::overrideBodyColor(std::optional<QColor> color) {
|
|
_widget->overrideBodyColor(color);
|
|
const auto raw = _webviewBottomLabel.data();
|
|
if (!raw) {
|
|
return;
|
|
} else if (!color) {
|
|
raw->setTextColorOverride(std::nullopt);
|
|
return;
|
|
}
|
|
const auto contrast = 2.5;
|
|
const auto luminance = 0.2126 * color->redF()
|
|
+ 0.7152 * color->greenF()
|
|
+ 0.0722 * color->blueF();
|
|
const auto textColor = (luminance > 0.5)
|
|
? QColor(0, 0, 0)
|
|
: QColor(255, 255, 255);
|
|
const auto textLuminance = (luminance > 0.5) ? 0 : 1;
|
|
const auto adaptiveOpacity = (luminance - textLuminance + contrast)
|
|
/ contrast;
|
|
const auto opacity = std::clamp(adaptiveOpacity, 0.5, 0.64);
|
|
auto buttonColor = textColor;
|
|
buttonColor.setAlphaF(opacity);
|
|
raw->setTextColorOverride(buttonColor);
|
|
}
|
|
|
|
void Panel::processBackgroundColor(const QJsonObject &args) {
|
|
_bodyColorReceived = true;
|
|
if (const auto color = ParseColor(args["color"].toString())) {
|
|
overrideBodyColor(*color);
|
|
_bodyColorLifetime.destroy();
|
|
} else if (const auto color = LookupNamedColor(
|
|
args["color_key"].toString())) {
|
|
overrideBodyColor((*color)->c);
|
|
_bodyColorLifetime = style::PaletteChanged(
|
|
) | rpl::start_with_next([=] {
|
|
overrideBodyColor((*color)->c);
|
|
});
|
|
} else {
|
|
overrideBodyColor(std::nullopt);
|
|
_bodyColorLifetime.destroy();
|
|
}
|
|
if (const auto raw = _bottomButtonsBg.get()) {
|
|
raw->update();
|
|
}
|
|
if (const auto raw = _webviewBottom.get()) {
|
|
raw->update();
|
|
}
|
|
}
|
|
|
|
void Panel::processBottomBarColor(const QJsonObject &args) {
|
|
_bottomColorReceived = true;
|
|
if (const auto color = ParseColor(args["color"].toString())) {
|
|
_widget->overrideBottomBarColor(color);
|
|
_bottomBarColor = color;
|
|
_bottomBarColorLifetime.destroy();
|
|
} else if (const auto color = LookupNamedColor(
|
|
args["color_key"].toString())) {
|
|
_widget->overrideBottomBarColor((*color)->c);
|
|
_bottomBarColor = (*color)->c;
|
|
_bottomBarColorLifetime = style::PaletteChanged(
|
|
) | rpl::start_with_next([=] {
|
|
_widget->overrideBottomBarColor((*color)->c);
|
|
_bottomBarColor = (*color)->c;
|
|
});
|
|
} else {
|
|
_widget->overrideBottomBarColor(std::nullopt);
|
|
_bottomBarColor = std::nullopt;
|
|
_bottomBarColorLifetime.destroy();
|
|
}
|
|
if (const auto raw = _bottomButtonsBg.get()) {
|
|
raw->update();
|
|
}
|
|
}
|
|
|
|
void Panel::processDownloadRequest(const QJsonObject &args) {
|
|
if (args.isEmpty()) {
|
|
_delegate->botClose();
|
|
return;
|
|
}
|
|
const auto url = args["url"].toString();
|
|
const auto name = args["file_name"].toString();
|
|
if (url.isEmpty()) {
|
|
LOG(("BotWebView Error: Bad 'url' in download request."));
|
|
_delegate->botClose();
|
|
return;
|
|
} else if (name.isEmpty()) {
|
|
LOG(("BotWebView Error: Bad 'file_name' in download request."));
|
|
_delegate->botClose();
|
|
return;
|
|
}
|
|
const auto done = crl::guard(this, [=](bool started) {
|
|
postEvent("file_download_requested", started
|
|
? "{ status: \"downloading\" }"
|
|
: "{ status: \"cancelled\" }");
|
|
});
|
|
_delegate->botDownloadFile({
|
|
.url = url,
|
|
.name = name,
|
|
.callback = done,
|
|
});
|
|
}
|
|
|
|
void Panel::createButton(std::unique_ptr<Button> &button) {
|
|
if (!_bottomButtonsBg) {
|
|
_bottomButtonsBg = std::make_unique<RpWidget>(_widget.get());
|
|
|
|
const auto raw = _bottomButtonsBg.get();
|
|
raw->paintRequest() | rpl::start_with_next([=] {
|
|
auto p = QPainter(raw);
|
|
auto hq = PainterHighQualityEnabler(p);
|
|
p.setPen(Qt::NoPen);
|
|
p.setBrush(_bottomBarColor.value_or(st::windowBg->c));
|
|
p.drawRoundedRect(
|
|
raw->rect().marginsAdded({ 0, 2 * st::callRadius, 0, 0 }),
|
|
st::callRadius,
|
|
st::callRadius);
|
|
}, raw->lifetime());
|
|
}
|
|
button = std::make_unique<Button>(
|
|
_bottomButtonsBg.get(),
|
|
st::botWebViewBottomButton);
|
|
const auto raw = button.get();
|
|
|
|
raw->setClickedCallback([=] {
|
|
if (!raw->isDisabled()) {
|
|
if (raw == _mainButton.get()) {
|
|
postEvent("main_button_pressed");
|
|
} else if (raw == _secondaryButton.get()) {
|
|
postEvent("secondary_button_pressed");
|
|
}
|
|
}
|
|
});
|
|
raw->hide();
|
|
|
|
rpl::combine(
|
|
raw->shownValue(),
|
|
raw->heightValue()
|
|
) | rpl::start_with_next([=] {
|
|
layoutButtons();
|
|
}, raw->lifetime());
|
|
}
|
|
|
|
void Panel::layoutButtons() {
|
|
if (!_webviewBottom) {
|
|
return;
|
|
}
|
|
const auto inner = _widget->innerGeometry();
|
|
const auto shown = [](std::unique_ptr<Button> &button) {
|
|
return button && !button->isHidden();
|
|
};
|
|
const auto any = shown(_mainButton) || shown(_secondaryButton);
|
|
_webviewBottom->setVisible(!any
|
|
&& !_fullscreen.current()
|
|
&& !_layerShown);
|
|
if (any) {
|
|
_bottomButtonsBg->setVisible(!_layerShown);
|
|
|
|
const auto one = shown(_mainButton)
|
|
? _mainButton.get()
|
|
: _secondaryButton.get();
|
|
const auto both = shown(_mainButton) && shown(_secondaryButton);
|
|
const auto vertical = both
|
|
&& ((_secondaryPosition == RectPart::Top)
|
|
|| (_secondaryPosition == RectPart::Bottom));
|
|
const auto padding = st::botWebViewBottomPadding;
|
|
const auto height = padding.top()
|
|
+ (vertical
|
|
? (_mainButton->height()
|
|
+ st::botWebViewBottomSkip.y()
|
|
+ _secondaryButton->height())
|
|
: one->height())
|
|
+ padding.bottom();
|
|
_bottomButtonsBg->setGeometry(
|
|
inner.x(),
|
|
inner.y() + inner.height() - height,
|
|
inner.width(),
|
|
height);
|
|
auto left = padding.left();
|
|
auto bottom = height - padding.bottom();
|
|
auto available = inner.width() - padding.left() - padding.right();
|
|
if (!both) {
|
|
one->resizeToWidth(available);
|
|
one->move(left, bottom - one->height());
|
|
} else if (_secondaryPosition == RectPart::Top) {
|
|
_mainButton->resizeToWidth(available);
|
|
bottom -= _mainButton->height();
|
|
_mainButton->move(left, bottom);
|
|
bottom -= st::botWebViewBottomSkip.y();
|
|
_secondaryButton->resizeToWidth(available);
|
|
bottom -= _secondaryButton->height();
|
|
_secondaryButton->move(left, bottom);
|
|
} else if (_secondaryPosition == RectPart::Bottom) {
|
|
_secondaryButton->resizeToWidth(available);
|
|
bottom -= _secondaryButton->height();
|
|
_secondaryButton->move(left, bottom);
|
|
bottom -= st::botWebViewBottomSkip.y();
|
|
_mainButton->resizeToWidth(available);
|
|
bottom -= _mainButton->height();
|
|
_mainButton->move(left, bottom);
|
|
} else if (_secondaryPosition == RectPart::Left) {
|
|
available = (available - st::botWebViewBottomSkip.x()) / 2;
|
|
_secondaryButton->resizeToWidth(available);
|
|
bottom -= _secondaryButton->height();
|
|
_secondaryButton->move(left, bottom);
|
|
_mainButton->resizeToWidth(available);
|
|
_mainButton->move(
|
|
inner.width() - padding.right() - available,
|
|
bottom);
|
|
} else {
|
|
available = (available - st::botWebViewBottomSkip.x()) / 2;
|
|
_mainButton->resizeToWidth(available);
|
|
bottom -= _mainButton->height();
|
|
_mainButton->move(left, bottom);
|
|
_secondaryButton->resizeToWidth(available);
|
|
_secondaryButton->move(
|
|
inner.width() - padding.right() - available,
|
|
bottom);
|
|
}
|
|
} else if (_bottomButtonsBg) {
|
|
_bottomButtonsBg->hide();
|
|
}
|
|
const auto footer = _layerShown
|
|
? 0
|
|
: any
|
|
? _bottomButtonsBg->height()
|
|
: _fullscreen.current()
|
|
? 0
|
|
: _webviewBottom->height();
|
|
_widget->setBottomBarHeight((!_layerShown && any) ? footer : 0);
|
|
_footerHeight = footer;
|
|
}
|
|
|
|
void Panel::showBox(object_ptr<BoxContent> box) {
|
|
showBox(std::move(box), LayerOption::KeepOther, anim::type::normal);
|
|
}
|
|
|
|
void Panel::showBox(
|
|
object_ptr<BoxContent> box,
|
|
LayerOptions options,
|
|
anim::type animated) {
|
|
if (const auto widget = _webview ? _webview->window.widget() : nullptr) {
|
|
_layerShown = true;
|
|
const auto hideNow = !widget->isHidden();
|
|
const auto raw = box.data();
|
|
_webview->boxes.push_back(raw);
|
|
raw->boxClosing(
|
|
) | rpl::filter([=] {
|
|
return _webview != nullptr;
|
|
}) | rpl::start_with_next([=] {
|
|
auto &list = _webview->boxes;
|
|
list.erase(ranges::remove_if(list, [&](QPointer<RpWidget> b) {
|
|
return !b || (b == raw);
|
|
}), end(list));
|
|
if (list.empty()) {
|
|
_webview->boxesLifetime.destroy();
|
|
_layerShown = false;
|
|
const auto widget = _webview
|
|
? _webview->window.widget()
|
|
: nullptr;
|
|
if (widget && widget->isHidden()) {
|
|
widget->show();
|
|
layoutButtons();
|
|
}
|
|
}
|
|
}, _webview->boxesLifetime);
|
|
|
|
if (hideNow) {
|
|
widget->hide();
|
|
layoutButtons();
|
|
}
|
|
}
|
|
const auto raw = box.data();
|
|
|
|
InvokeQueued(raw, [=] {
|
|
if (raw->window()->isActiveWindow()) {
|
|
// In case focus is somewhat in a native child window,
|
|
// like a webview, Qt glitches here with input fields showing
|
|
// focused state, but not receiving any keyboard input:
|
|
//
|
|
// window()->windowHandle()->isActive() == false.
|
|
//
|
|
// Steps were: SeparatePanel with a WebView2 child,
|
|
// some interaction with mouse inside the WebView2,
|
|
// so that WebView2 gets focus and active window state,
|
|
// then we call setSearchAllowed() and after animation
|
|
// is finished try typing -> nothing happens.
|
|
//
|
|
// With this workaround it works fine.
|
|
_widget->activateWindow();
|
|
}
|
|
});
|
|
|
|
_widget->showBox(
|
|
std::move(box),
|
|
LayerOption::KeepOther,
|
|
anim::type::normal);
|
|
}
|
|
|
|
void Panel::showToast(TextWithEntities &&text) {
|
|
_widget->showToast(std::move(text));
|
|
}
|
|
|
|
not_null<QWidget*> Panel::toastParent() const {
|
|
return _widget->uiShow()->toastParent();
|
|
}
|
|
|
|
void Panel::hideLayer(anim::type animated) {
|
|
_widget->hideLayer(animated);
|
|
}
|
|
|
|
void Panel::showCriticalError(const TextWithEntities &text) {
|
|
_progress = nullptr;
|
|
_webviewProgress = false;
|
|
auto error = base::make_unique_q<PaddingWrap<FlatLabel>>(
|
|
_widget.get(),
|
|
object_ptr<FlatLabel>(
|
|
_widget.get(),
|
|
rpl::single(text),
|
|
st::paymentsCriticalError),
|
|
st::paymentsCriticalErrorPadding);
|
|
error->entity()->setClickHandlerFilter([=](
|
|
const ClickHandlerPtr &handler,
|
|
Qt::MouseButton) {
|
|
const auto entity = handler->getTextEntity();
|
|
if (entity.type != EntityType::CustomUrl) {
|
|
return true;
|
|
}
|
|
File::OpenUrl(entity.data);
|
|
return false;
|
|
});
|
|
_widget->showInner(std::move(error));
|
|
}
|
|
|
|
void Panel::updateThemeParams(const Webview::ThemeParams ¶ms) {
|
|
updateColorOverrides(params);
|
|
if (!_webview || !_webview->window.widget()) {
|
|
return;
|
|
}
|
|
_webview->window.updateTheme(
|
|
params.bodyBg,
|
|
params.scrollBg,
|
|
params.scrollBgOver,
|
|
params.scrollBarBg,
|
|
params.scrollBarBgOver);
|
|
postEvent("theme_changed", "{\"theme_params\": " + params.json + "}");
|
|
}
|
|
|
|
void Panel::updateColorOverrides(const Webview::ThemeParams ¶ms) {
|
|
if (!_headerColorReceived && params.titleBg.alpha() == 255) {
|
|
_widget->overrideTitleColor(params.titleBg);
|
|
}
|
|
if (!_bodyColorReceived && params.bodyBg.alpha() == 255) {
|
|
overrideBodyColor(params.bodyBg);
|
|
}
|
|
}
|
|
|
|
void Panel::invoiceClosed(const QString &slug, const QString &status) {
|
|
if (!_webview || !_webview->window.widget()) {
|
|
return;
|
|
}
|
|
postEvent("invoice_closed", QJsonObject{
|
|
{ u"slug"_q, slug },
|
|
{ u"status"_q, status },
|
|
});
|
|
if (_hiddenForPayment) {
|
|
_hiddenForPayment = false;
|
|
_widget->showAndActivate();
|
|
}
|
|
}
|
|
|
|
void Panel::hideForPayment() {
|
|
_hiddenForPayment = true;
|
|
_widget->hideGetDuration();
|
|
}
|
|
|
|
void Panel::postEvent(const QString &event) {
|
|
postEvent(event, {});
|
|
}
|
|
|
|
void Panel::postEvent(const QString &event, EventData data) {
|
|
if (!_webview) {
|
|
LOG(("BotWebView Error: Post event \"%1\" on crashed webview."
|
|
).arg(event));
|
|
return;
|
|
}
|
|
auto written = v::is<QString>(data)
|
|
? v::get<QString>(data).toUtf8()
|
|
: QJsonDocument(
|
|
v::get<QJsonObject>(data)).toJson(QJsonDocument::Compact);
|
|
_webview->window.eval(R"(
|
|
if (window.TelegramGameProxy) {
|
|
window.TelegramGameProxy.receiveEvent(
|
|
")"
|
|
+ event.toUtf8()
|
|
+ '"' + (written.isEmpty() ? QByteArray() : ", " + written)
|
|
+ R"();
|
|
}
|
|
)");
|
|
}
|
|
|
|
TextWithEntities ErrorText(const Webview::Available &info) {
|
|
Expects(info.error != Webview::Available::Error::None);
|
|
|
|
using Error = Webview::Available::Error;
|
|
switch (info.error) {
|
|
case Error::NoWebview2:
|
|
return tr::lng_payments_webview_install_edge(
|
|
tr::now,
|
|
lt_link,
|
|
Text::Link(
|
|
"Microsoft Edge WebView2 Runtime",
|
|
"https://go.microsoft.com/fwlink/p/?LinkId=2124703"),
|
|
Ui::Text::WithEntities);
|
|
case Error::NoWebKitGTK:
|
|
return { tr::lng_payments_webview_install_webkit(tr::now) };
|
|
case Error::NoOpenGL:
|
|
return { tr::lng_payments_webview_enable_opengl(tr::now) };
|
|
case Error::NonX11:
|
|
return { tr::lng_payments_webview_switch_x11(tr::now) };
|
|
case Error::OldWindows:
|
|
return { tr::lng_payments_webview_update_windows(tr::now) };
|
|
default:
|
|
return { QString::fromStdString(info.details) };
|
|
}
|
|
}
|
|
|
|
void Panel::showWebviewError(
|
|
const QString &text,
|
|
const Webview::Available &information) {
|
|
showCriticalError(TextWithEntities{ text }.append(
|
|
"\n\n"
|
|
).append(ErrorText(information)));
|
|
}
|
|
|
|
rpl::lifetime &Panel::lifetime() {
|
|
return _widget->lifetime();
|
|
}
|
|
|
|
std::unique_ptr<Panel> Show(Args &&args) {
|
|
return std::make_unique<Panel>(std::move(args));
|
|
}
|
|
|
|
} // namespace Ui::BotWebView
|