tdesktop/Telegram/SourceFiles/media/stories/media_stories_slider.cpp
2023-07-20 07:20:12 +04:00

153 lines
4 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 "media/stories/media_stories_slider.h"
#include "media/stories/media_stories_controller.h"
#include "media/view/media_view_playback_progress.h"
#include "media/audio/media_audio.h"
#include "ui/painter.h"
#include "ui/rp_widget.h"
#include "styles/style_widgets.h"
#include "styles/style_media_view.h"
namespace Media::Stories {
namespace {
constexpr auto kOpacityInactive = 0.4;
constexpr auto kOpacityActive = 1.;
} // namespace
Slider::Slider(not_null<Controller*> controller)
: _controller(controller)
, _progress(std::make_unique<View::PlaybackProgress>()) {
}
Slider::~Slider() {
}
void Slider::show(SliderData data) {
resetProgress();
data.total = std::max(data.total, 1);
data.index = std::clamp(data.index, 0, data.total - 1);
if (_data == data) {
return;
}
_data = data;
const auto parent = _controller->wrap();
auto widget = std::make_unique<Ui::RpWidget>(parent);
const auto raw = widget.get();
_rects.resize(_data.total);
raw->widthValue() | rpl::filter([=](int width) {
return (width >= st::storiesSliderWidth);
}) | rpl::start_with_next([=](int width) {
layout(width);
}, raw->lifetime());
raw->paintRequest(
) | rpl::filter([=] {
return (raw->width() >= st::storiesSliderWidth);
}) | rpl::start_with_next([=](QRect clip) {
paint(QRectF(clip));
}, raw->lifetime());
raw->show();
_widget = std::move(widget);
_progress->setValueChangedCallback([=](float64, float64) {
_widget->update(_activeBoundingRect);
});
_controller->layoutValue(
) | rpl::start_with_next([=](const Layout &layout) {
raw->setGeometry(layout.slider - st::storiesSliderMargin);
}, raw->lifetime());
}
void Slider::raise() {
if (_widget) {
_widget->raise();
}
}
void Slider::updatePlayback(const Player::TrackState &state) {
_progress->updateState(state);
}
void Slider::resetProgress() {
_progress->updateState({});
}
void Slider::layout(int width) {
const auto single = st::storiesSliderWidth;
const auto skip = st::storiesSliderSkip;
// width == single * max + skip * (max - 1);
// max == (width + skip) / (single + skip);
const auto max = (width + skip) / (single + skip);
Assert(max > 0);
const auto count = std::clamp(_data.total, 1, max);
const auto one = (width - (count - 1) * skip) / float64(count);
auto left = 0.;
for (auto i = 0; i != count; ++i) {
_rects[i] = QRectF(left, 0, one, single);
if (i == _data.index) {
const auto from = int(std::floor(left));
const auto size = int(std::ceil(left + one)) - from;
_activeBoundingRect = QRect(from, 0, size, single);
}
left += one + skip;
}
for (auto i = count; i != _rects.size(); ++i) {
_rects[i] = QRectF();
}
}
void Slider::paint(QRectF clip) {
auto p = QPainter(_widget.get());
auto hq = PainterHighQualityEnabler(p);
p.setBrush(st::mediaviewControlFg);
p.setPen(Qt::NoPen);
const auto radius = st::storiesSliderWidth / 2.;
for (auto i = 0; i != int(_rects.size()); ++i) {
if (_rects[i].isEmpty()) {
break;
} else if (!_rects[i].intersects(clip)) {
continue;
} else if (i == _data.index) {
const auto progress = _progress->value();
const auto full = _rects[i].width();
const auto min = _rects[i].height();
const auto activeWidth = std::max(full * progress, min);
const auto inactiveWidth = full - activeWidth + min;
const auto activeLeft = _rects[i].left();
const auto inactiveLeft = activeLeft + activeWidth - min;
p.setOpacity(kOpacityInactive);
p.drawRoundedRect(
QRectF(inactiveLeft, 0, inactiveWidth, min),
radius,
radius);
p.setOpacity(kOpacityActive);
p.drawRoundedRect(
QRectF(activeLeft, 0, activeWidth, min),
radius,
radius);
} else {
p.setOpacity((i < _data.index)
? kOpacityActive
: kOpacityInactive);
p.drawRoundedRect(_rects[i], radius, radius);
}
}
}
} // namespace Media::Stories