AyuGramDesktop/Telegram/SourceFiles/calls/group/calls_group_viewport_tile.cpp
2021-05-24 22:45:16 +04:00

156 lines
3.9 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 "calls/group/calls_group_viewport_tile.h"
#include "webrtc/webrtc_video_track.h"
#include "lang/lang_keys.h"
#include "styles/style_calls.h"
#include <QtGui/QOpenGLFunctions>
namespace Calls::Group {
Viewport::VideoTile::VideoTile(
const VideoEndpoint &endpoint,
LargeVideoTrack track,
rpl::producer<bool> pinned,
Fn<void()> update)
: _endpoint(endpoint)
, _update(std::move(update))
, _track(track) {
Expects(track.track != nullptr);
Expects(track.row != nullptr);
setup(std::move(pinned));
}
Viewport::VideoTile::~VideoTile() {
Expects(!_textures);
}
QRect Viewport::VideoTile::pinInner() const {
return _pinInner.translated(0, -pinSlide());
}
QRect Viewport::VideoTile::pinOuter() const {
return _pinOuter;
}
int Viewport::VideoTile::pinSlide() const {
return anim::interpolate(
st::groupCallLargeVideoWide.pinPosition.y() + _pinInner.height(),
0,
_pinShownAnimation.value(_pinShown ? 1. : 0.));
}
void Viewport::VideoTile::setGeometry(QRect geometry) {
_geometry = geometry;
updatePinnedGeometry();
}
void Viewport::VideoTile::togglePinShown(bool shown) {
if (_pinShown == shown) {
return;
}
_pinShown = shown;
_pinShownAnimation.start(
_update,
shown ? 0. : 1.,
shown ? 1. : 0.,
st::slideWrapDuration);
}
bool Viewport::VideoTile::updateRequestedQuality(VideoQuality quality) {
if (_quality && *_quality == quality) {
return false;
}
_quality = quality;
return true;
}
void Viewport::VideoTile::updatePinnedGeometry() {
const auto &st = st::groupCallLargeVideoWide;
const auto &icon = st::groupCallLargeVideoPin.icon;
const auto innerWidth = icon.width()
+ st.pinTextPosition.x()
+ st::semiboldFont->width(_pinned
? tr::lng_pinned_unpin(tr::now)
: tr::lng_pinned_pin(tr::now));
const auto innerHeight = icon.height();
const auto buttonWidth = st.pinPadding.left()
+ innerWidth
+ st.pinPadding.right();
const auto buttonHeight = st.pinPadding.top()
+ innerHeight
+ st.pinPadding.bottom();
const auto fullWidth = st.pinPosition.x() * 2 + buttonWidth;
const auto fullHeight = st.pinPosition.y() * 2 + buttonHeight;
_pinInner = QRect(
_geometry.width() - st.pinPosition.x() - buttonWidth,
st.pinPosition.y(),
buttonWidth,
buttonHeight);
_pinOuter = QRect(
_geometry.width() - fullWidth,
0,
fullWidth,
fullHeight);
}
void Viewport::VideoTile::setup(rpl::producer<bool> pinned) {
std::move(
pinned
) | rpl::filter([=](bool pinned) {
return (_pinned != pinned);
}) | rpl::start_with_next([=](bool pinned) {
_pinned = pinned;
updatePinnedGeometry();
_update();
}, _lifetime);
_track.track->renderNextFrame(
) | rpl::start_with_next([=] {
const auto size = _track.track->frameSize();
if (size.isEmpty()) {
_track.track->markFrameShown();
} else {
_trackSize = size;
}
_update();
}, _lifetime);
if (const auto size = _track.track->frameSize(); !size.isEmpty()) {
_trackSize = size;
}
}
void Viewport::VideoTile::ensureTexturesCreated(
not_null<QOpenGLFunctions*> f) {
if (_textures) {
return;
}
f->glGenTextures(_textures.values.size(), _textures.values.data());
for (const auto texture : _textures.values) {
f->glBindTexture(GL_TEXTURE_2D, texture);
const auto clamp = GL_CLAMP_TO_EDGE;
f->glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, clamp);
f->glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, clamp);
f->glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
f->glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
}
}
const Viewport::Textures &Viewport::VideoTile::textures() const {
return _textures;
}
Viewport::Textures Viewport::VideoTile::takeTextures() {
return base::take(_textures);
}
} // namespace Calls::Group