1104 lines
30 KiB
C++
1104 lines
30 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/player/media_player_widget.h"
|
|
|
|
#include "platform/platform_specific.h"
|
|
#include "data/data_document.h"
|
|
#include "data/data_session.h"
|
|
#include "data/data_peer.h"
|
|
#include "core/application.h"
|
|
#include "core/core_settings.h"
|
|
#include "ui/widgets/labels.h"
|
|
#include "ui/widgets/continuous_sliders.h"
|
|
#include "ui/widgets/shadow.h"
|
|
#include "ui/widgets/buttons.h"
|
|
#include "ui/widgets/popup_menu.h"
|
|
#include "ui/widgets/dropdown_menu.h"
|
|
#include "ui/widgets/menu/menu_action.h"
|
|
#include "ui/wrap/fade_wrap.h"
|
|
#include "ui/effects/ripple_animation.h"
|
|
#include "ui/text/format_values.h"
|
|
#include "ui/text/format_song_document_name.h"
|
|
#include "lang/lang_keys.h"
|
|
#include "media/audio/media_audio.h"
|
|
#include "media/view/media_view_playback_progress.h"
|
|
#include "media/player/media_player_button.h"
|
|
#include "media/player/media_player_instance.h"
|
|
#include "media/player/media_player_dropdown.h"
|
|
#include "media/player/media_player_volume_controller.h"
|
|
#include "styles/style_media_player.h"
|
|
#include "styles/style_media_view.h"
|
|
#include "history/history_item.h"
|
|
#include "storage/storage_account.h"
|
|
#include "main/main_session.h"
|
|
#include "facades.h"
|
|
|
|
namespace Media {
|
|
namespace Player {
|
|
|
|
class WithDropdownController {
|
|
public:
|
|
WithDropdownController(
|
|
not_null<Ui::IconButton*> button,
|
|
not_null<Ui::RpWidget*> menuParent,
|
|
Fn<void(bool)> menuOverCallback);
|
|
virtual ~WithDropdownController() = default;
|
|
|
|
[[nodiscard]] not_null<Ui::IconButton*> button() const;
|
|
Ui::DropdownMenu *menu() const;
|
|
|
|
void updateDropdownGeometry();
|
|
|
|
void hideTemporarily();
|
|
void showBack();
|
|
|
|
protected:
|
|
void showMenu();
|
|
|
|
private:
|
|
virtual void fillMenu(not_null<Ui::DropdownMenu*> menu) = 0;
|
|
|
|
const not_null<Ui::IconButton*> _button;
|
|
const not_null<Ui::RpWidget*> _menuParent;
|
|
const Fn<void(bool)> _menuOverCallback;
|
|
base::unique_qptr<Ui::DropdownMenu> _menu;
|
|
bool _temporarilyHidden = false;
|
|
bool _overButton = false;
|
|
|
|
};
|
|
|
|
class Widget::OrderController final : public WithDropdownController {
|
|
public:
|
|
OrderController(
|
|
not_null<Ui::IconButton*> button,
|
|
not_null<Ui::RpWidget*> menuParent,
|
|
Fn<void(bool)> menuOverCallback);
|
|
|
|
private:
|
|
void fillMenu(not_null<Ui::DropdownMenu*> menu) override;
|
|
void updateIcon();
|
|
|
|
};
|
|
|
|
class Widget::SpeedController final : public WithDropdownController {
|
|
public:
|
|
SpeedController(
|
|
not_null<Ui::IconButton*> button,
|
|
not_null<Ui::RpWidget*> menuParent,
|
|
Fn<void(bool)> menuOverCallback);
|
|
|
|
[[nodiscard]] rpl::producer<> saved() const;
|
|
|
|
private:
|
|
void fillMenu(not_null<Ui::DropdownMenu*> menu) override;
|
|
void updateIcon();
|
|
|
|
[[nodiscard]] float64 speed() const;
|
|
[[nodiscard]] bool isDefault() const;
|
|
[[nodiscard]] float64 lastNonDefaultSpeed() const;
|
|
void toggleDefault();
|
|
void setSpeed(float64 newSpeed);
|
|
void save();
|
|
|
|
float64 _speed = 2.;
|
|
bool _isDefault = true;
|
|
rpl::event_stream<float64> _speedChanged;
|
|
rpl::event_stream<> _saved;
|
|
|
|
};
|
|
|
|
WithDropdownController::WithDropdownController(
|
|
not_null<Ui::IconButton*> button,
|
|
not_null<Ui::RpWidget*> menuParent,
|
|
Fn<void(bool)> menuOverCallback)
|
|
: _button(button)
|
|
, _menuParent(menuParent)
|
|
, _menuOverCallback(std::move(menuOverCallback)) {
|
|
button->events(
|
|
) | rpl::filter([=](not_null<QEvent*> e) {
|
|
return (e->type() == QEvent::Enter)
|
|
|| (e->type() == QEvent::Leave);
|
|
}) | rpl::start_with_next([=](not_null<QEvent*> e) {
|
|
_overButton = (e->type() == QEvent::Enter);
|
|
if (_overButton) {
|
|
InvokeQueued(button, [=] {
|
|
if (_overButton) {
|
|
showMenu();
|
|
}
|
|
});
|
|
}
|
|
}, button->lifetime());
|
|
}
|
|
|
|
not_null<Ui::IconButton*> WithDropdownController::button() const {
|
|
return _button;
|
|
}
|
|
|
|
Ui::DropdownMenu *WithDropdownController::menu() const {
|
|
return _menu.get();
|
|
}
|
|
|
|
void WithDropdownController::updateDropdownGeometry() {
|
|
if (!_menu) {
|
|
return;
|
|
}
|
|
const auto position = _menu->parentWidget()->mapFromGlobal(
|
|
_button->mapToGlobal(
|
|
QPoint(_button->width(), _button->height())));
|
|
const auto padding = st::mediaPlayerMenu.wrap.padding;
|
|
_menu->move(position
|
|
- QPoint(_menu->width(), 0)
|
|
+ QPoint(padding.right(), -padding.top())
|
|
+ st::mediaPlayerMenuPosition);
|
|
}
|
|
|
|
void WithDropdownController::hideTemporarily() {
|
|
if (_menu && !_menu->isHidden()) {
|
|
_temporarilyHidden = true;
|
|
_menu->hide();
|
|
}
|
|
}
|
|
|
|
void WithDropdownController::showBack() {
|
|
if (_temporarilyHidden) {
|
|
_temporarilyHidden = false;
|
|
if (_menu && _menu->isHidden()) {
|
|
_menu->show();
|
|
}
|
|
}
|
|
}
|
|
|
|
void WithDropdownController::showMenu() {
|
|
if (_menu) {
|
|
return;
|
|
}
|
|
_menu.emplace(_menuParent, st::mediaPlayerMenu);
|
|
const auto raw = _menu.get();
|
|
_menu->events(
|
|
) | rpl::start_with_next([this](not_null<QEvent*> e) {
|
|
const auto type = e->type();
|
|
if (type == QEvent::Enter) {
|
|
_menuOverCallback(true);
|
|
} else if (type == QEvent::Leave) {
|
|
_menuOverCallback(false);
|
|
}
|
|
}, _menu->lifetime());
|
|
_menu->setHiddenCallback([=]{
|
|
Ui::PostponeCall(raw, [this] {
|
|
_menu = nullptr;
|
|
});
|
|
});
|
|
_button->installEventFilter(raw);
|
|
fillMenu(raw);
|
|
updateDropdownGeometry();
|
|
_menu->showAnimated(Ui::PanelAnimation::Origin::TopRight);
|
|
}
|
|
|
|
Widget::OrderController::OrderController(
|
|
not_null<Ui::IconButton*> button,
|
|
not_null<Ui::RpWidget*> menuParent,
|
|
Fn<void(bool)> menuOverCallback)
|
|
: WithDropdownController(button, menuParent, std::move(menuOverCallback)) {
|
|
button->setClickedCallback([=] {
|
|
showMenu();
|
|
});
|
|
|
|
Core::App().settings().playerOrderModeValue(
|
|
) | rpl::start_with_next([=] {
|
|
updateIcon();
|
|
}, button->lifetime());
|
|
}
|
|
|
|
void Widget::OrderController::fillMenu(not_null<Ui::DropdownMenu*> menu) {
|
|
const auto addOrderAction = [&](OrderMode mode) {
|
|
struct Fields {
|
|
QString label;
|
|
const style::icon &icon;
|
|
const style::icon &activeIcon;
|
|
};
|
|
const auto current = Core::App().settings().playerOrderMode();
|
|
const auto active = (current == mode);
|
|
const auto callback = [=] {
|
|
Core::App().settings().setPlayerOrderMode(active
|
|
? OrderMode::Default
|
|
: mode);
|
|
Core::App().saveSettingsDelayed();
|
|
};
|
|
const auto fields = [&]() -> Fields {
|
|
switch (mode) {
|
|
case OrderMode::Reverse: return {
|
|
.label = tr::lng_audio_player_reverse(tr::now),
|
|
.icon = st::mediaPlayerOrderIconReverse,
|
|
.activeIcon = st::mediaPlayerOrderIconReverseActive,
|
|
};
|
|
case OrderMode::Shuffle: return {
|
|
.label = tr::lng_audio_player_shuffle(tr::now),
|
|
.icon = st::mediaPlayerOrderIconShuffle,
|
|
.activeIcon = st::mediaPlayerOrderIconShuffleActive,
|
|
};
|
|
}
|
|
Unexpected("Order mode in addOrderAction.");
|
|
}();
|
|
menu->addAction(base::make_unique_q<Ui::Menu::Action>(
|
|
menu,
|
|
(active
|
|
? st::mediaPlayerOrderMenuActive
|
|
: st::mediaPlayerOrderMenu),
|
|
Ui::Menu::CreateAction(menu, fields.label, callback),
|
|
&(active ? fields.activeIcon : fields.icon),
|
|
&(active ? fields.activeIcon : fields.icon)));
|
|
};
|
|
addOrderAction(OrderMode::Reverse);
|
|
addOrderAction(OrderMode::Shuffle);
|
|
}
|
|
|
|
void Widget::OrderController::updateIcon() {
|
|
switch (Core::App().settings().playerOrderMode()) {
|
|
case OrderMode::Default:
|
|
button()->setIconOverride(
|
|
&st::mediaPlayerReverseDisabledIcon,
|
|
&st::mediaPlayerReverseDisabledIconOver);
|
|
button()->setRippleColorOverride(
|
|
&st::mediaPlayerRepeatDisabledRippleBg);
|
|
break;
|
|
case OrderMode::Reverse:
|
|
button()->setIconOverride(&st::mediaPlayerReverseIcon);
|
|
button()->setRippleColorOverride(nullptr);
|
|
break;
|
|
case OrderMode::Shuffle:
|
|
button()->setIconOverride(&st::mediaPlayerShuffleIcon);
|
|
button()->setRippleColorOverride(nullptr);
|
|
break;
|
|
}
|
|
}
|
|
|
|
Widget::SpeedController::SpeedController(
|
|
not_null<Ui::IconButton*> button,
|
|
not_null<Ui::RpWidget*> menuParent,
|
|
Fn<void(bool)> menuOverCallback)
|
|
: WithDropdownController(button, menuParent, std::move(menuOverCallback)) {
|
|
button->setClickedCallback([=] {
|
|
toggleDefault();
|
|
save();
|
|
if (const auto current = menu()) {
|
|
current->otherEnter();
|
|
}
|
|
});
|
|
|
|
setSpeed(Core::App().settings().voicePlaybackSpeed());
|
|
_speed = Core::App().settings().voicePlaybackSpeed(true);
|
|
|
|
_speedChanged.events_starting_with(
|
|
speed()
|
|
) | rpl::start_with_next([=] {
|
|
updateIcon();
|
|
}, button->lifetime());
|
|
}
|
|
|
|
void Widget::SpeedController::updateIcon() {
|
|
const auto isDefaultSpeed = isDefault();
|
|
const auto nonDefaultSpeed = lastNonDefaultSpeed();
|
|
|
|
if (nonDefaultSpeed == .5) {
|
|
button()->setIconOverride(
|
|
(isDefaultSpeed
|
|
? &st::mediaPlayerSpeedSlowDisabledIcon
|
|
: &st::mediaPlayerSpeedSlowIcon),
|
|
(isDefaultSpeed
|
|
? &st::mediaPlayerSpeedSlowDisabledIconOver
|
|
: &st::mediaPlayerSpeedSlowIcon));
|
|
} else if (nonDefaultSpeed == 1.5) {
|
|
button()->setIconOverride(
|
|
(isDefaultSpeed
|
|
? &st::mediaPlayerSpeedFastDisabledIcon
|
|
: &st::mediaPlayerSpeedFastIcon),
|
|
(isDefaultSpeed
|
|
? &st::mediaPlayerSpeedFastDisabledIconOver
|
|
: &st::mediaPlayerSpeedFastIcon));
|
|
} else {
|
|
button()->setIconOverride(
|
|
isDefaultSpeed ? &st::mediaPlayerSpeedDisabledIcon : nullptr,
|
|
(isDefaultSpeed
|
|
? &st::mediaPlayerSpeedDisabledIconOver
|
|
: nullptr));
|
|
}
|
|
button()->setRippleColorOverride(isDefaultSpeed
|
|
? &st::mediaPlayerSpeedDisabledRippleBg
|
|
: nullptr);
|
|
}
|
|
|
|
rpl::producer<> Widget::SpeedController::saved() const {
|
|
return _saved.events();
|
|
}
|
|
|
|
float64 Widget::SpeedController::speed() const {
|
|
return _isDefault ? 1. : _speed;
|
|
}
|
|
|
|
bool Widget::SpeedController::isDefault() const {
|
|
return _isDefault;
|
|
}
|
|
|
|
float64 Widget::SpeedController::lastNonDefaultSpeed() const {
|
|
return _speed;
|
|
}
|
|
|
|
void Widget::SpeedController::toggleDefault() {
|
|
_isDefault = !_isDefault;
|
|
_speedChanged.fire(speed());
|
|
}
|
|
|
|
void Widget::SpeedController::setSpeed(float64 newSpeed) {
|
|
if (!(_isDefault = (newSpeed == 1.))) {
|
|
_speed = newSpeed;
|
|
}
|
|
_speedChanged.fire(speed());
|
|
}
|
|
|
|
void Widget::SpeedController::save() {
|
|
Core::App().settings().setVoicePlaybackSpeed(speed());
|
|
Core::App().saveSettingsDelayed();
|
|
_saved.fire({});
|
|
}
|
|
|
|
void Widget::SpeedController::fillMenu(not_null<Ui::DropdownMenu*> menu) {
|
|
const auto currentSpeed = speed();
|
|
const auto addSpeedAction = [&](float64 speed, QString text) {
|
|
const auto callback = [=] {
|
|
setSpeed(speed);
|
|
save();
|
|
};
|
|
const auto icon = (speed == currentSpeed)
|
|
? &st::mediaPlayerMenuCheck
|
|
: nullptr;
|
|
auto action = base::make_unique_q<Ui::Menu::Action>(
|
|
menu,
|
|
st::mediaPlayerSpeedMenu,
|
|
Ui::Menu::CreateAction(menu, text, callback),
|
|
icon,
|
|
icon);
|
|
const auto raw = action.get();
|
|
_speedChanged.events(
|
|
) | rpl::start_with_next([=](float64 updatedSpeed) {
|
|
const auto icon = (speed == updatedSpeed)
|
|
? &st::mediaPlayerMenuCheck
|
|
: nullptr;
|
|
raw->setIcon(icon, icon);
|
|
}, raw->lifetime());
|
|
menu->addAction(std::move(action));
|
|
};
|
|
addSpeedAction(0.5, tr::lng_voice_speed_slow(tr::now));
|
|
addSpeedAction(1., tr::lng_voice_speed_normal(tr::now));
|
|
addSpeedAction(1.5, tr::lng_voice_speed_fast(tr::now));
|
|
addSpeedAction(2., tr::lng_voice_speed_very_fast(tr::now));
|
|
}
|
|
|
|
Widget::Widget(
|
|
QWidget *parent,
|
|
not_null<Ui::RpWidget*> dropdownsParent,
|
|
not_null<Window::SessionController*> controller)
|
|
: RpWidget(parent)
|
|
, _controller(controller)
|
|
, _orderMenuParent(dropdownsParent)
|
|
, _nameLabel(this, st::mediaPlayerName)
|
|
, _rightControls(this, object_ptr<Ui::RpWidget>(this))
|
|
, _timeLabel(rightControls(), st::mediaPlayerTime)
|
|
, _playPause(this, st::mediaPlayerPlayButton)
|
|
, _volumeToggle(rightControls(), st::mediaPlayerVolumeToggle)
|
|
, _repeatToggle(rightControls(), st::mediaPlayerRepeatButton)
|
|
, _orderToggle(rightControls(), st::mediaPlayerRepeatButton)
|
|
, _speedToggle(rightControls(), st::mediaPlayerSpeedButton)
|
|
, _close(this, st::mediaPlayerClose)
|
|
, _shadow(this)
|
|
, _playbackSlider(this, st::mediaPlayerPlayback)
|
|
, _volume(std::in_place, dropdownsParent.get())
|
|
, _playbackProgress(std::make_unique<View::PlaybackProgress>())
|
|
, _orderController(
|
|
std::make_unique<OrderController>(
|
|
_orderToggle.data(),
|
|
dropdownsParent,
|
|
[=](bool over) { markOver(over); }))
|
|
, _speedController(
|
|
std::make_unique<SpeedController>(
|
|
_speedToggle.data(),
|
|
dropdownsParent,
|
|
[=](bool over) { markOver(over); })) {
|
|
setAttribute(Qt::WA_OpaquePaintEvent);
|
|
setMouseTracking(true);
|
|
resize(width(), st::mediaPlayerHeight + st::lineWidth);
|
|
|
|
setupRightControls();
|
|
|
|
_nameLabel->setAttribute(Qt::WA_TransparentForMouseEvents);
|
|
_timeLabel->setAttribute(Qt::WA_TransparentForMouseEvents);
|
|
|
|
_playbackProgress->setInLoadingStateChangedCallback([=](bool loading) {
|
|
_playbackSlider->setDisabled(loading);
|
|
});
|
|
_playbackProgress->setValueChangedCallback([=](float64 value, float64) {
|
|
_playbackSlider->setValue(value);
|
|
});
|
|
_playbackSlider->setChangeProgressCallback([=](float64 value) {
|
|
if (_type != AudioMsgId::Type::Song) {
|
|
return; // Round video seek is not supported for now :(
|
|
}
|
|
_playbackProgress->setValue(value, false);
|
|
handleSeekProgress(value);
|
|
});
|
|
_playbackSlider->setChangeFinishedCallback([=](float64 value) {
|
|
if (_type != AudioMsgId::Type::Song) {
|
|
return; // Round video seek is not supported for now :(
|
|
}
|
|
_playbackProgress->setValue(value, false);
|
|
handleSeekFinished(value);
|
|
});
|
|
_playPause->setClickedCallback([=] {
|
|
instance()->playPauseCancelClicked(_type);
|
|
});
|
|
|
|
updateVolumeToggleIcon();
|
|
_volumeToggle->setClickedCallback([=] {
|
|
const auto volume = (Core::App().settings().songVolume() > 0)
|
|
? 0.
|
|
: Core::App().settings().rememberedSongVolume();
|
|
Core::App().settings().setSongVolume(volume);
|
|
Core::App().saveSettingsDelayed();
|
|
mixer()->setSongVolume(volume);
|
|
});
|
|
Core::App().settings().songVolumeChanges(
|
|
) | rpl::start_with_next([=] {
|
|
updateVolumeToggleIcon();
|
|
}, lifetime());
|
|
|
|
Core::App().settings().playerRepeatModeValue(
|
|
) | rpl::start_with_next([=] {
|
|
updateRepeatToggleIcon();
|
|
}, lifetime());
|
|
|
|
_repeatToggle->setClickedCallback([=] {
|
|
auto &settings = Core::App().settings();
|
|
settings.setPlayerRepeatMode([&] {
|
|
switch (settings.playerRepeatMode()) {
|
|
case RepeatMode::None: return RepeatMode::One;
|
|
case RepeatMode::One: return RepeatMode::All;
|
|
case RepeatMode::All: return RepeatMode::None;
|
|
}
|
|
Unexpected("Repeat mode in Settings.");
|
|
}());
|
|
Core::App().saveSettingsDelayed();
|
|
});
|
|
|
|
_speedController->saved(
|
|
) | rpl::start_with_next([=] {
|
|
instance()->updateVoicePlaybackSpeed();
|
|
}, lifetime());
|
|
|
|
instance()->trackChanged(
|
|
) | rpl::filter([=](AudioMsgId::Type type) {
|
|
return (type == _type);
|
|
}) | rpl::start_with_next([=](AudioMsgId::Type type) {
|
|
handleSongChange();
|
|
updateControlsVisibility();
|
|
updateLabelsGeometry();
|
|
}, lifetime());
|
|
|
|
instance()->tracksFinished(
|
|
) | rpl::filter([=](AudioMsgId::Type type) {
|
|
return (type == AudioMsgId::Type::Voice);
|
|
}) | rpl::start_with_next([=](AudioMsgId::Type type) {
|
|
_voiceIsActive = false;
|
|
const auto currentSong = instance()->current(AudioMsgId::Type::Song);
|
|
const auto songState = instance()->getState(AudioMsgId::Type::Song);
|
|
if (currentSong == songState.id && !IsStoppedOrStopping(songState.state)) {
|
|
setType(AudioMsgId::Type::Song);
|
|
}
|
|
}, lifetime());
|
|
|
|
instance()->updatedNotifier(
|
|
) | rpl::start_with_next([=](const TrackState &state) {
|
|
handleSongUpdate(state);
|
|
}, lifetime());
|
|
|
|
PrepareVolumeDropdown(_volume.get(), controller, _volumeToggle->events(
|
|
) | rpl::filter([=](not_null<QEvent*> e) {
|
|
return (e->type() == QEvent::Wheel);
|
|
}) | rpl::map([=](not_null<QEvent*> e) {
|
|
return not_null{ static_cast<QWheelEvent*>(e.get()) };
|
|
}));
|
|
_volumeToggle->installEventFilter(_volume.get());
|
|
_volume->events(
|
|
) | rpl::start_with_next([=](not_null<QEvent*> e) {
|
|
if (e->type() == QEvent::Enter) {
|
|
markOver(true);
|
|
} else if (e->type() == QEvent::Leave) {
|
|
markOver(false);
|
|
}
|
|
}, _volume->lifetime());
|
|
|
|
hidePlaylistOn(_playPause);
|
|
hidePlaylistOn(_close);
|
|
hidePlaylistOn(_rightControls);
|
|
|
|
setType(AudioMsgId::Type::Song);
|
|
}
|
|
|
|
void Widget::hidePlaylistOn(not_null<Ui::RpWidget*> widget) {
|
|
widget->events(
|
|
) | rpl::filter([=](not_null<QEvent*> e) {
|
|
return (e->type() == QEvent::Enter);
|
|
}) | rpl::start_with_next([=] {
|
|
updateOverLabelsState(false);
|
|
}, widget->lifetime());
|
|
}
|
|
|
|
void Widget::setupRightControls() {
|
|
const auto raw = rightControls();
|
|
raw->paintRequest(
|
|
) | rpl::start_with_next([=](QRect clip) {
|
|
auto p = QPainter(raw);
|
|
const auto &icon = st::mediaPlayerControlsFade;
|
|
const auto fade = QRect(0, 0, icon.width(), raw->height());
|
|
if (fade.intersects(clip)) {
|
|
icon.fill(p, fade);
|
|
}
|
|
const auto fill = clip.intersected(
|
|
{ icon.width(), 0, raw->width() - icon.width(), raw->height() });
|
|
if (!fill.isEmpty()) {
|
|
p.fillRect(fill, st::mediaPlayerBg);
|
|
}
|
|
}, raw->lifetime());
|
|
_rightControls->show(anim::type::instant);
|
|
}
|
|
|
|
void Widget::updateVolumeToggleIcon() {
|
|
_volumeToggle->setIconOverride([] {
|
|
const auto volume = Core::App().settings().songVolume();
|
|
return (volume == 0.)
|
|
? &st::mediaPlayerVolumeIcon0
|
|
: (volume < 0.66)
|
|
? &st::mediaPlayerVolumeIcon1
|
|
: nullptr;
|
|
}());
|
|
}
|
|
|
|
void Widget::setCloseCallback(Fn<void()> callback) {
|
|
_closeCallback = std::move(callback);
|
|
_close->setClickedCallback([this] { stopAndClose(); });
|
|
}
|
|
|
|
void Widget::setShowItemCallback(
|
|
Fn<void(not_null<const HistoryItem*>)> callback) {
|
|
_showItemCallback = std::move(callback);
|
|
}
|
|
|
|
void Widget::stopAndClose() {
|
|
_voiceIsActive = false;
|
|
if (_type == AudioMsgId::Type::Voice) {
|
|
const auto songData = instance()->current(AudioMsgId::Type::Song);
|
|
const auto songState = instance()->getState(AudioMsgId::Type::Song);
|
|
if (songData == songState.id && !IsStoppedOrStopping(songState.state)) {
|
|
instance()->stop(AudioMsgId::Type::Voice);
|
|
return;
|
|
}
|
|
}
|
|
if (_closeCallback) {
|
|
_closeCallback();
|
|
}
|
|
}
|
|
|
|
void Widget::setShadowGeometryToLeft(int x, int y, int w, int h) {
|
|
_shadow->setGeometryToLeft(x, y, w, h);
|
|
}
|
|
|
|
void Widget::showShadowAndDropdowns() {
|
|
_shadow->show();
|
|
_playbackSlider->setVisible(_type == AudioMsgId::Type::Song);
|
|
if (_volumeHidden) {
|
|
_volumeHidden = false;
|
|
_volume->show();
|
|
}
|
|
_speedController->showBack();
|
|
_orderController->showBack();
|
|
}
|
|
|
|
void Widget::updateDropdownsGeometry() {
|
|
const auto dropdownWidth = st::mediaPlayerVolumeSize.width();
|
|
const auto position = _volume->parentWidget()->mapFromGlobal(
|
|
_volumeToggle->mapToGlobal(
|
|
QPoint(
|
|
(_volumeToggle->width() - dropdownWidth) / 2,
|
|
height())));
|
|
const auto playerMargins = _volume->getMargin();
|
|
const auto shift = QPoint(playerMargins.left(), playerMargins.top());
|
|
_volume->move(position - shift);
|
|
|
|
_orderController->updateDropdownGeometry();
|
|
_speedController->updateDropdownGeometry();
|
|
}
|
|
|
|
void Widget::hideShadowAndDropdowns() {
|
|
_shadow->hide();
|
|
_playbackSlider->hide();
|
|
if (!_volume->isHidden()) {
|
|
_volumeHidden = true;
|
|
_volume->hide();
|
|
}
|
|
_speedController->hideTemporarily();
|
|
_orderController->hideTemporarily();
|
|
}
|
|
|
|
void Widget::raiseDropdowns() {
|
|
_volume->raise();
|
|
}
|
|
|
|
Widget::~Widget() = default;
|
|
|
|
not_null<Ui::RpWidget*> Widget::rightControls() {
|
|
return _rightControls->entity();
|
|
}
|
|
|
|
void Widget::handleSeekProgress(float64 progress) {
|
|
if (!_lastDurationMs) return;
|
|
|
|
const auto positionMs = std::clamp(
|
|
static_cast<crl::time>(progress * _lastDurationMs),
|
|
crl::time(0),
|
|
_lastDurationMs);
|
|
if (_seekPositionMs != positionMs) {
|
|
_seekPositionMs = positionMs;
|
|
updateTimeLabel();
|
|
|
|
instance()->startSeeking(_type);
|
|
}
|
|
}
|
|
|
|
void Widget::handleSeekFinished(float64 progress) {
|
|
if (!_lastDurationMs) return;
|
|
|
|
_seekPositionMs = -1;
|
|
|
|
instance()->finishSeeking(_type, progress);
|
|
}
|
|
|
|
void Widget::resizeEvent(QResizeEvent *e) {
|
|
updateControlsGeometry();
|
|
_narrow = (width() < st::mediaPlayerWideWidth);
|
|
updateControlsWrapVisibility();
|
|
}
|
|
|
|
void Widget::updateControlsGeometry() {
|
|
_close->moveToRight(st::mediaPlayerCloseRight, st::mediaPlayerPlayTop);
|
|
auto right = 0;
|
|
if (hasPlaybackSpeedControl()) {
|
|
_speedToggle->moveToRight(right, 0); right += _speedToggle->width();
|
|
}
|
|
_repeatToggle->moveToRight(right, 0); right += _repeatToggle->width();
|
|
_orderToggle->moveToRight(right, 0); right += _orderToggle->width();
|
|
_volumeToggle->moveToRight(right, 0); right += _volumeToggle->width();
|
|
|
|
updateControlsWrapGeometry();
|
|
|
|
updatePlayPrevNextPositions();
|
|
|
|
_playbackSlider->setGeometry(
|
|
0,
|
|
height() - st::mediaPlayerPlayback.fullWidth,
|
|
width(),
|
|
st::mediaPlayerPlayback.fullWidth);
|
|
|
|
updateDropdownsGeometry();
|
|
}
|
|
|
|
void Widget::updateControlsWrapGeometry() {
|
|
const auto fade = st::mediaPlayerControlsFade.width();
|
|
const auto controls = getTimeRight() + _timeLabel->width() + fade;
|
|
rightControls()->resize(controls, _repeatToggle->height());
|
|
_rightControls->move(
|
|
width() - st::mediaPlayerCloseRight - _close->width() - controls,
|
|
st::mediaPlayerPlayTop);
|
|
}
|
|
|
|
void Widget::updateControlsWrapVisibility() {
|
|
_rightControls->toggle(
|
|
_over || !_narrow,
|
|
isHidden() ? anim::type::instant : anim::type::normal);
|
|
}
|
|
|
|
void Widget::paintEvent(QPaintEvent *e) {
|
|
Painter p(this);
|
|
auto fill = e->rect().intersected(QRect(0, 0, width(), st::mediaPlayerHeight));
|
|
if (!fill.isEmpty()) {
|
|
p.fillRect(fill, st::mediaPlayerBg);
|
|
}
|
|
}
|
|
|
|
void Widget::enterEventHook(QEnterEvent *e) {
|
|
markOver(true);
|
|
}
|
|
|
|
void Widget::leaveEventHook(QEvent *e) {
|
|
markOver(false);
|
|
}
|
|
|
|
void Widget::markOver(bool over) {
|
|
if (over) {
|
|
_over = true;
|
|
_wontBeOver = false;
|
|
InvokeQueued(this, [=] {
|
|
updateControlsWrapVisibility();
|
|
});
|
|
} else {
|
|
_wontBeOver = true;
|
|
InvokeQueued(this, [=] {
|
|
if (!_wontBeOver) {
|
|
return;
|
|
}
|
|
_wontBeOver = false;
|
|
_over = false;
|
|
updateControlsWrapVisibility();
|
|
});
|
|
updateOverLabelsState(false);
|
|
}
|
|
}
|
|
|
|
void Widget::mouseMoveEvent(QMouseEvent *e) {
|
|
updateOverLabelsState(e->pos());
|
|
}
|
|
|
|
void Widget::mousePressEvent(QMouseEvent *e) {
|
|
_labelsDown = _labelsOver;
|
|
}
|
|
|
|
void Widget::mouseReleaseEvent(QMouseEvent *e) {
|
|
if (auto downLabels = base::take(_labelsDown)) {
|
|
if (_labelsOver != downLabels) {
|
|
return;
|
|
}
|
|
if (_type == AudioMsgId::Type::Voice) {
|
|
const auto current = instance()->current(_type);
|
|
const auto document = current.audio();
|
|
const auto context = current.contextId();
|
|
if (document && context && _showItemCallback) {
|
|
if (const auto item = document->owner().message(context)) {
|
|
_showItemCallback(item);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void Widget::updateOverLabelsState(QPoint pos) {
|
|
const auto left = getNameLeft();
|
|
const auto right = width()
|
|
- _rightControls->x()
|
|
- _rightControls->width()
|
|
+ getTimeRight();
|
|
const auto labels = myrtlrect(left, 0, width() - right - left, height() - st::mediaPlayerPlayback.fullWidth);
|
|
const auto over = labels.contains(pos);
|
|
updateOverLabelsState(over);
|
|
}
|
|
|
|
void Widget::updateOverLabelsState(bool over) {
|
|
_labelsOver = over;
|
|
auto pressShowsItem = _labelsOver && (_type == AudioMsgId::Type::Voice);
|
|
setCursor(pressShowsItem ? style::cur_pointer : style::cur_default);
|
|
_togglePlaylistRequests.fire(over && (_type == AudioMsgId::Type::Song));
|
|
}
|
|
|
|
void Widget::updatePlayPrevNextPositions() {
|
|
auto left = st::mediaPlayerPlayLeft;
|
|
auto top = st::mediaPlayerPlayTop;
|
|
if (_previousTrack) {
|
|
_previousTrack->moveToLeft(left, top); left += _previousTrack->width() + st::mediaPlayerPlaySkip;
|
|
_playPause->moveToLeft(left, top); left += _playPause->width() + st::mediaPlayerPlaySkip;
|
|
_nextTrack->moveToLeft(left, top);
|
|
} else {
|
|
_playPause->moveToLeft(left, top);
|
|
}
|
|
updateLabelsGeometry();
|
|
}
|
|
|
|
int Widget::getNameLeft() const {
|
|
auto result = st::mediaPlayerPlayLeft + _playPause->width();
|
|
if (_previousTrack) {
|
|
result += _previousTrack->width() + st::mediaPlayerPlaySkip + _nextTrack->width() + st::mediaPlayerPlaySkip;
|
|
}
|
|
result += st::mediaPlayerPadding;
|
|
return result;
|
|
}
|
|
|
|
int Widget::getNameRight() const {
|
|
return st::mediaPlayerCloseRight
|
|
+ _close->width()
|
|
+ st::mediaPlayerPadding;
|
|
}
|
|
|
|
int Widget::getTimeRight() const {
|
|
auto result = 0;
|
|
if (_type == AudioMsgId::Type::Song) {
|
|
result += _repeatToggle->width()
|
|
+ _orderToggle->width()
|
|
+ _volumeToggle->width();
|
|
}
|
|
if (hasPlaybackSpeedControl()) {
|
|
result += _speedToggle->width();
|
|
}
|
|
result += st::mediaPlayerPadding;
|
|
return result;
|
|
}
|
|
|
|
void Widget::updateLabelsGeometry() {
|
|
const auto left = getNameLeft();
|
|
const auto widthForName = width()
|
|
- left
|
|
- getNameRight();
|
|
_nameLabel->resizeToWidth(widthForName);
|
|
_nameLabel->moveToLeft(left, st::mediaPlayerNameTop - st::mediaPlayerName.style.font->ascent);
|
|
|
|
const auto right = getTimeRight();
|
|
_timeLabel->moveToRight(right, st::mediaPlayerNameTop - st::mediaPlayerTime.font->ascent);
|
|
|
|
updateControlsWrapGeometry();
|
|
}
|
|
|
|
void Widget::updateRepeatToggleIcon() {
|
|
switch (Core::App().settings().playerRepeatMode()) {
|
|
case RepeatMode::None:
|
|
_repeatToggle->setIconOverride(
|
|
&st::mediaPlayerRepeatDisabledIcon,
|
|
&st::mediaPlayerRepeatDisabledIconOver);
|
|
_repeatToggle->setRippleColorOverride(
|
|
&st::mediaPlayerRepeatDisabledRippleBg);
|
|
break;
|
|
case RepeatMode::One:
|
|
_repeatToggle->setIconOverride(&st::mediaPlayerRepeatOneIcon);
|
|
_repeatToggle->setRippleColorOverride(nullptr);
|
|
break;
|
|
case RepeatMode::All:
|
|
_repeatToggle->setIconOverride(nullptr);
|
|
_repeatToggle->setRippleColorOverride(nullptr);
|
|
break;
|
|
}
|
|
}
|
|
|
|
void Widget::checkForTypeChange() {
|
|
auto hasActiveType = [](AudioMsgId::Type type) {
|
|
const auto current = instance()->current(type);
|
|
const auto state = instance()->getState(type);
|
|
return (current == state.id && !IsStoppedOrStopping(state.state));
|
|
};
|
|
if (hasActiveType(AudioMsgId::Type::Voice)) {
|
|
_voiceIsActive = true;
|
|
setType(AudioMsgId::Type::Voice);
|
|
} else if (!_voiceIsActive && hasActiveType(AudioMsgId::Type::Song)) {
|
|
setType(AudioMsgId::Type::Song);
|
|
}
|
|
}
|
|
|
|
bool Widget::hasPlaybackSpeedControl() const {
|
|
return _lastSongId.changeablePlaybackSpeed()
|
|
&& Media::Audio::SupportsSpeedControl();
|
|
}
|
|
|
|
void Widget::updateControlsVisibility() {
|
|
_repeatToggle->setVisible(_type == AudioMsgId::Type::Song);
|
|
_orderToggle->setVisible(_type == AudioMsgId::Type::Song);
|
|
_volumeToggle->setVisible(_type == AudioMsgId::Type::Song);
|
|
_speedToggle->setVisible(hasPlaybackSpeedControl());
|
|
if (!_shadow->isHidden()) {
|
|
_playbackSlider->setVisible(_type == AudioMsgId::Type::Song);
|
|
}
|
|
updateControlsGeometry();
|
|
}
|
|
|
|
void Widget::setType(AudioMsgId::Type type) {
|
|
if (_type != type) {
|
|
_type = type;
|
|
handleSongChange();
|
|
updateControlsVisibility();
|
|
updateLabelsGeometry();
|
|
handleSongUpdate(instance()->getState(_type));
|
|
updateOverLabelsState(_labelsOver);
|
|
_playlistChangesLifetime = instance()->playlistChanges(
|
|
_type
|
|
) | rpl::start_with_next([=] {
|
|
handlePlaylistUpdate();
|
|
});
|
|
// maybe the type change causes a change of the button layout
|
|
QResizeEvent event = { size(), size() };
|
|
resizeEvent(&event);
|
|
}
|
|
}
|
|
|
|
void Widget::handleSongUpdate(const TrackState &state) {
|
|
checkForTypeChange();
|
|
if (state.id.type() != _type || !state.id.audio()) {
|
|
return;
|
|
}
|
|
|
|
if (state.id.audio()->loading()) {
|
|
_playbackProgress->updateLoadingState(state.id.audio()->progress());
|
|
} else {
|
|
_playbackProgress->updateState(state);
|
|
}
|
|
|
|
auto showPause = ShowPauseIcon(state.state);
|
|
if (instance()->isSeeking(_type)) {
|
|
showPause = true;
|
|
}
|
|
_playPause->setIconOverride(state.id.audio()->loading()
|
|
? &st::mediaPlayerCancelIcon
|
|
: showPause
|
|
? &st::mediaPlayerPauseIcon
|
|
: nullptr);
|
|
|
|
updateTimeText(state);
|
|
}
|
|
|
|
void Widget::updateTimeText(const TrackState &state) {
|
|
qint64 display = 0;
|
|
const auto frequency = state.frequency;
|
|
const auto document = state.id.audio();
|
|
if (!IsStoppedOrStopping(state.state)) {
|
|
display = state.position;
|
|
} else if (state.length) {
|
|
display = state.length;
|
|
} else if (const auto song = document->song()) {
|
|
display = (song->duration * frequency);
|
|
}
|
|
|
|
_lastDurationMs = (state.length * 1000LL) / frequency;
|
|
|
|
if (document->loading()) {
|
|
_time = QString::number(qRound(document->progress() * 100)) + '%';
|
|
_playbackSlider->setDisabled(true);
|
|
} else {
|
|
display = display / frequency;
|
|
_time = Ui::FormatDurationText(display);
|
|
_playbackSlider->setDisabled(false);
|
|
}
|
|
if (_seekPositionMs < 0) {
|
|
updateTimeLabel();
|
|
}
|
|
}
|
|
|
|
void Widget::updateTimeLabel() {
|
|
auto timeLabelWidth = _timeLabel->width();
|
|
if (_seekPositionMs >= 0) {
|
|
auto playAlready = _seekPositionMs / 1000LL;
|
|
_timeLabel->setText(Ui::FormatDurationText(playAlready));
|
|
} else {
|
|
_timeLabel->setText(_time);
|
|
}
|
|
if (timeLabelWidth != _timeLabel->width()) {
|
|
updateLabelsGeometry();
|
|
}
|
|
}
|
|
|
|
void Widget::handleSongChange() {
|
|
const auto current = instance()->current(_type);
|
|
const auto document = current.audio();
|
|
if (!current
|
|
|| !document
|
|
|| ((_lastSongId.audio() == document)
|
|
&& (_lastSongId.contextId() == current.contextId()))) {
|
|
return;
|
|
}
|
|
_lastSongId = current;
|
|
|
|
TextWithEntities textWithEntities;
|
|
if (document->isVoiceMessage() || document->isVideoMessage()) {
|
|
if (const auto item = document->owner().message(current.contextId())) {
|
|
const auto name = item->fromOriginal()->name;
|
|
const auto date = [item] {
|
|
const auto parsed = ItemDateTime(item);
|
|
const auto date = parsed.date();
|
|
const auto time = parsed.time().toString(cTimeFormat());
|
|
const auto today = QDateTime::currentDateTime().date();
|
|
if (date == today) {
|
|
return tr::lng_player_message_today(
|
|
tr::now,
|
|
lt_time,
|
|
time);
|
|
} else if (date.addDays(1) == today) {
|
|
return tr::lng_player_message_yesterday(
|
|
tr::now,
|
|
lt_time,
|
|
time);
|
|
}
|
|
return tr::lng_player_message_date(
|
|
tr::now,
|
|
lt_date,
|
|
langDayOfMonthFull(date),
|
|
lt_time,
|
|
time);
|
|
};
|
|
|
|
textWithEntities.text = name + ' ' + date();
|
|
textWithEntities.entities.append(EntityInText(
|
|
EntityType::Semibold,
|
|
0,
|
|
name.size(),
|
|
QString()));
|
|
} else {
|
|
textWithEntities.text = tr::lng_media_audio(tr::now);
|
|
}
|
|
} else {
|
|
textWithEntities = Ui::Text::FormatSongNameFor(document)
|
|
.textWithEntities(true);
|
|
}
|
|
_nameLabel->setMarkedText(textWithEntities);
|
|
|
|
handlePlaylistUpdate();
|
|
}
|
|
|
|
void Widget::handlePlaylistUpdate() {
|
|
const auto previousEnabled = instance()->previousAvailable(_type);
|
|
const auto nextEnabled = instance()->nextAvailable(_type);
|
|
if (!previousEnabled && !nextEnabled) {
|
|
destroyPrevNextButtons();
|
|
} else {
|
|
createPrevNextButtons();
|
|
_previousTrack->setIconOverride(previousEnabled ? nullptr : &st::mediaPlayerPreviousDisabledIcon);
|
|
_previousTrack->setRippleColorOverride(previousEnabled ? nullptr : &st::mediaPlayerBg);
|
|
_previousTrack->setPointerCursor(previousEnabled);
|
|
_nextTrack->setIconOverride(nextEnabled ? nullptr : &st::mediaPlayerNextDisabledIcon);
|
|
_nextTrack->setRippleColorOverride(nextEnabled ? nullptr : &st::mediaPlayerBg);
|
|
_nextTrack->setPointerCursor(nextEnabled);
|
|
}
|
|
}
|
|
|
|
void Widget::createPrevNextButtons() {
|
|
if (!_previousTrack) {
|
|
_previousTrack.create(this, st::mediaPlayerPreviousButton);
|
|
_previousTrack->show();
|
|
_previousTrack->setClickedCallback([=]() {
|
|
instance()->previous(_type);
|
|
});
|
|
_nextTrack.create(this, st::mediaPlayerNextButton);
|
|
_nextTrack->show();
|
|
_nextTrack->setClickedCallback([=]() {
|
|
instance()->next(_type);
|
|
});
|
|
hidePlaylistOn(_previousTrack);
|
|
hidePlaylistOn(_nextTrack);
|
|
updatePlayPrevNextPositions();
|
|
}
|
|
}
|
|
|
|
void Widget::destroyPrevNextButtons() {
|
|
if (_previousTrack) {
|
|
_previousTrack.destroy();
|
|
_nextTrack.destroy();
|
|
updatePlayPrevNextPositions();
|
|
}
|
|
}
|
|
|
|
} // namespace Player
|
|
} // namespace Media
|