391 lines
9.9 KiB
C++
391 lines
9.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 "ui/widgets/discrete_sliders.h"
|
|
|
|
#include "ui/effects/ripple_animation.h"
|
|
#include "ui/painter.h"
|
|
|
|
namespace Ui {
|
|
|
|
DiscreteSlider::DiscreteSlider(QWidget *parent, bool snapToLabel)
|
|
: RpWidget(parent)
|
|
, _snapToLabel(snapToLabel) {
|
|
setCursor(style::cur_pointer);
|
|
}
|
|
|
|
void DiscreteSlider::setActiveSection(int index) {
|
|
_activeIndex = index;
|
|
activateCallback();
|
|
setSelectedSection(index);
|
|
}
|
|
|
|
void DiscreteSlider::activateCallback() {
|
|
if (_timerId >= 0) {
|
|
killTimer(_timerId);
|
|
_timerId = -1;
|
|
}
|
|
auto ms = crl::now();
|
|
if (ms >= _callbackAfterMs) {
|
|
_sectionActivated.fire_copy(_activeIndex);
|
|
} else {
|
|
_timerId = startTimer(_callbackAfterMs - ms, Qt::PreciseTimer);
|
|
}
|
|
}
|
|
|
|
void DiscreteSlider::timerEvent(QTimerEvent *e) {
|
|
activateCallback();
|
|
}
|
|
|
|
void DiscreteSlider::setActiveSectionFast(int index) {
|
|
setActiveSection(index);
|
|
finishAnimating();
|
|
}
|
|
|
|
void DiscreteSlider::finishAnimating() {
|
|
_a_left.stop();
|
|
update();
|
|
_callbackAfterMs = 0;
|
|
if (_timerId >= 0) {
|
|
activateCallback();
|
|
}
|
|
}
|
|
|
|
void DiscreteSlider::setSelectOnPress(bool selectOnPress) {
|
|
_selectOnPress = selectOnPress;
|
|
}
|
|
|
|
void DiscreteSlider::addSection(const QString &label) {
|
|
_sections.push_back(Section(label, getLabelStyle()));
|
|
resizeToWidth(width());
|
|
}
|
|
|
|
void DiscreteSlider::setSections(const std::vector<QString> &labels) {
|
|
Assert(!labels.empty());
|
|
|
|
_sections.clear();
|
|
for (const auto &label : labels) {
|
|
_sections.push_back(Section(label, getLabelStyle()));
|
|
}
|
|
stopAnimation();
|
|
if (_activeIndex >= _sections.size()) {
|
|
_activeIndex = 0;
|
|
}
|
|
if (_selected >= _sections.size()) {
|
|
_selected = 0;
|
|
}
|
|
resizeToWidth(width());
|
|
}
|
|
|
|
DiscreteSlider::Range DiscreteSlider::getFinalActiveRange() const {
|
|
const auto raw = _sections.empty() ? nullptr : &_sections[_selected];
|
|
if (!raw) {
|
|
return { 0, 0 };
|
|
}
|
|
const auto width = _snapToLabel
|
|
? std::min(raw->width, raw->label.maxWidth())
|
|
: raw->width;
|
|
return { raw->left + ((raw->width - width) / 2), width };
|
|
}
|
|
|
|
DiscreteSlider::Range DiscreteSlider::getCurrentActiveRange() const {
|
|
const auto to = getFinalActiveRange();
|
|
return {
|
|
int(base::SafeRound(_a_left.value(to.left))),
|
|
int(base::SafeRound(_a_width.value(to.width))),
|
|
};
|
|
}
|
|
|
|
template <typename Lambda>
|
|
void DiscreteSlider::enumerateSections(Lambda callback) {
|
|
for (auto §ion : _sections) {
|
|
if (!callback(section)) {
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
template <typename Lambda>
|
|
void DiscreteSlider::enumerateSections(Lambda callback) const {
|
|
for (auto §ion : _sections) {
|
|
if (!callback(section)) {
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
void DiscreteSlider::mousePressEvent(QMouseEvent *e) {
|
|
auto index = getIndexFromPosition(e->pos());
|
|
if (_selectOnPress) {
|
|
setSelectedSection(index);
|
|
}
|
|
startRipple(index);
|
|
_pressed = index;
|
|
}
|
|
|
|
void DiscreteSlider::mouseMoveEvent(QMouseEvent *e) {
|
|
if (_pressed < 0) return;
|
|
if (_selectOnPress) {
|
|
setSelectedSection(getIndexFromPosition(e->pos()));
|
|
}
|
|
}
|
|
|
|
void DiscreteSlider::mouseReleaseEvent(QMouseEvent *e) {
|
|
auto pressed = std::exchange(_pressed, -1);
|
|
if (pressed < 0) return;
|
|
|
|
auto index = getIndexFromPosition(e->pos());
|
|
if (pressed < _sections.size()) {
|
|
if (_sections[pressed].ripple) {
|
|
_sections[pressed].ripple->lastStop();
|
|
}
|
|
}
|
|
if (_selectOnPress || index == pressed) {
|
|
setActiveSection(index);
|
|
}
|
|
}
|
|
|
|
void DiscreteSlider::setSelectedSection(int index) {
|
|
if (index < 0 || index >= _sections.size()) return;
|
|
|
|
if (_selected != index) {
|
|
const auto from = getFinalActiveRange();
|
|
_selected = index;
|
|
const auto to = getFinalActiveRange();
|
|
const auto duration = getAnimationDuration();
|
|
const auto updater = [=] { update(); };
|
|
_a_left.start(updater, from.left, to.left, duration);
|
|
_a_width.start(updater, from.width, to.width, duration);
|
|
_callbackAfterMs = crl::now() + duration;
|
|
}
|
|
}
|
|
|
|
int DiscreteSlider::getIndexFromPosition(QPoint pos) {
|
|
int count = _sections.size();
|
|
for (int i = 0; i != count; ++i) {
|
|
if (_sections[i].left + _sections[i].width > pos.x()) {
|
|
return i;
|
|
}
|
|
}
|
|
return count - 1;
|
|
}
|
|
|
|
DiscreteSlider::Section::Section(
|
|
const QString &label,
|
|
const style::TextStyle &st)
|
|
: label(st, label) {
|
|
}
|
|
|
|
SettingsSlider::SettingsSlider(
|
|
QWidget *parent,
|
|
const style::SettingsSlider &st)
|
|
: DiscreteSlider(parent, st.barSnapToLabel)
|
|
, _st(st) {
|
|
if (_st.barRadius > 0) {
|
|
_bar.emplace(_st.barRadius, _st.barFg);
|
|
_barActive.emplace(_st.barRadius, _st.barFgActive);
|
|
}
|
|
setSelectOnPress(_st.ripple.showDuration == 0);
|
|
}
|
|
|
|
void SettingsSlider::setRippleTopRoundRadius(int radius) {
|
|
_rippleTopRoundRadius = radius;
|
|
}
|
|
|
|
const style::TextStyle &SettingsSlider::getLabelStyle() const {
|
|
return _st.labelStyle;
|
|
}
|
|
|
|
int SettingsSlider::getAnimationDuration() const {
|
|
return _st.duration;
|
|
}
|
|
|
|
void SettingsSlider::resizeSections(int newWidth) {
|
|
auto count = getSectionsCount();
|
|
if (!count) return;
|
|
|
|
auto sectionWidths = countSectionsWidths(newWidth);
|
|
|
|
auto skip = 0;
|
|
auto x = 0.;
|
|
auto sectionWidth = sectionWidths.begin();
|
|
enumerateSections([&](Section §ion) {
|
|
Expects(sectionWidth != sectionWidths.end());
|
|
|
|
section.left = std::floor(x) + skip;
|
|
x += *sectionWidth;
|
|
section.width = qRound(x) - (section.left - skip);
|
|
skip += _st.barSkip;
|
|
++sectionWidth;
|
|
return true;
|
|
});
|
|
stopAnimation();
|
|
}
|
|
|
|
std::vector<float64> SettingsSlider::countSectionsWidths(
|
|
int newWidth) const {
|
|
auto count = getSectionsCount();
|
|
auto sectionsWidth = newWidth - (count - 1) * _st.barSkip;
|
|
auto sectionWidth = sectionsWidth / float64(count);
|
|
|
|
auto result = std::vector<float64>(count, sectionWidth);
|
|
auto labelsWidth = 0;
|
|
auto commonWidth = true;
|
|
enumerateSections([&](const Section §ion) {
|
|
labelsWidth += section.label.maxWidth();
|
|
if (section.label.maxWidth() >= sectionWidth) {
|
|
commonWidth = false;
|
|
}
|
|
return true;
|
|
});
|
|
// If labelsWidth > sectionsWidth we're screwed anyway.
|
|
if (!commonWidth && labelsWidth <= sectionsWidth) {
|
|
auto padding = (sectionsWidth - labelsWidth) / (2. * count);
|
|
auto currentWidth = result.begin();
|
|
enumerateSections([&](const Section §ion) {
|
|
Expects(currentWidth != result.end());
|
|
|
|
*currentWidth = padding + section.label.maxWidth() + padding;
|
|
++currentWidth;
|
|
return true;
|
|
});
|
|
}
|
|
return result;
|
|
}
|
|
|
|
int SettingsSlider::resizeGetHeight(int newWidth) {
|
|
resizeSections(newWidth);
|
|
return _st.height;
|
|
}
|
|
|
|
void SettingsSlider::startRipple(int sectionIndex) {
|
|
if (!_st.ripple.showDuration) return;
|
|
auto index = 0;
|
|
enumerateSections([this, &index, sectionIndex](Section §ion) {
|
|
if (index++ == sectionIndex) {
|
|
if (!section.ripple) {
|
|
auto mask = prepareRippleMask(sectionIndex, section);
|
|
section.ripple = std::make_unique<RippleAnimation>(
|
|
_st.ripple,
|
|
std::move(mask),
|
|
[this] { update(); });
|
|
}
|
|
const auto point = mapFromGlobal(QCursor::pos());
|
|
section.ripple->add(point - QPoint(section.left, 0));
|
|
return false;
|
|
}
|
|
return true;
|
|
});
|
|
}
|
|
|
|
QImage SettingsSlider::prepareRippleMask(
|
|
int sectionIndex,
|
|
const Section §ion) {
|
|
auto size = QSize(section.width, height() - _st.rippleBottomSkip);
|
|
if (!_rippleTopRoundRadius
|
|
|| (sectionIndex > 0 && sectionIndex + 1 < getSectionsCount())) {
|
|
return RippleAnimation::RectMask(size);
|
|
}
|
|
return RippleAnimation::MaskByDrawer(size, false, [&](QPainter &p) {
|
|
auto plusRadius = _rippleTopRoundRadius + 1;
|
|
p.drawRoundedRect(
|
|
0,
|
|
0,
|
|
section.width,
|
|
height() + plusRadius,
|
|
_rippleTopRoundRadius,
|
|
_rippleTopRoundRadius);
|
|
if (sectionIndex > 0) {
|
|
p.fillRect(0, 0, plusRadius, plusRadius, p.brush());
|
|
}
|
|
if (sectionIndex + 1 < getSectionsCount()) {
|
|
p.fillRect(
|
|
section.width - plusRadius,
|
|
0,
|
|
plusRadius,
|
|
plusRadius, p.brush());
|
|
}
|
|
});
|
|
}
|
|
|
|
void SettingsSlider::paintEvent(QPaintEvent *e) {
|
|
Painter p(this);
|
|
|
|
auto clip = e->rect();
|
|
auto range = getCurrentActiveRange();
|
|
|
|
const auto drawRect = [&](QRect rect, bool active = false) {
|
|
const auto &bar = active ? _barActive : _bar;
|
|
if (bar) {
|
|
bar->paint(p, rect);
|
|
} else {
|
|
p.fillRect(rect, active ? _st.barFgActive : _st.barFg);
|
|
}
|
|
};
|
|
enumerateSections([&](Section §ion) {
|
|
const auto activeWidth = _st.barSnapToLabel
|
|
? section.label.maxWidth()
|
|
: section.width;
|
|
const auto activeLeft = section.left
|
|
+ (section.width - activeWidth) / 2;
|
|
auto active = 1.
|
|
- std::clamp(
|
|
qAbs(range.left - activeLeft) / float64(section.width),
|
|
0.,
|
|
1.);
|
|
if (section.ripple) {
|
|
auto color = anim::color(_st.rippleBg, _st.rippleBgActive, active);
|
|
section.ripple->paint(p, section.left, 0, width(), &color);
|
|
if (section.ripple->empty()) {
|
|
section.ripple.reset();
|
|
}
|
|
}
|
|
if (!_st.barSnapToLabel) {
|
|
auto from = activeLeft, tofill = activeWidth;
|
|
if (range.left > from) {
|
|
auto fill = qMin(tofill, range.left - from);
|
|
drawRect(myrtlrect(from, _st.barTop, fill, _st.barStroke));
|
|
from += fill;
|
|
tofill -= fill;
|
|
}
|
|
if (range.left + activeWidth > from) {
|
|
if (auto fill = qMin(tofill, range.left + activeWidth - from)) {
|
|
drawRect(
|
|
myrtlrect(from, _st.barTop, fill, _st.barStroke),
|
|
true);
|
|
from += fill;
|
|
tofill -= fill;
|
|
}
|
|
}
|
|
if (tofill) {
|
|
drawRect(myrtlrect(from, _st.barTop, tofill, _st.barStroke));
|
|
}
|
|
}
|
|
const auto labelLeft = section.left + (section.width - section.label.maxWidth()) / 2;
|
|
if (myrtlrect(labelLeft, _st.labelTop, section.label.maxWidth(), _st.labelStyle.font->height).intersects(clip)) {
|
|
p.setPen(anim::pen(_st.labelFg, _st.labelFgActive, active));
|
|
section.label.drawLeft(
|
|
p,
|
|
labelLeft,
|
|
_st.labelTop,
|
|
section.label.maxWidth(),
|
|
width());
|
|
}
|
|
return true;
|
|
});
|
|
if (_st.barSnapToLabel) {
|
|
const auto add = _st.barStroke / 2;
|
|
const auto from = std::max(range.left - add, 0);
|
|
const auto till = std::min(range.left + range.width + add, width());
|
|
if (from < till) {
|
|
drawRect(myrtlrect(from, _st.barTop, till - from, _st.barStroke), true);
|
|
}
|
|
}
|
|
}
|
|
|
|
} // namespace Ui
|