From de8518a112b3dce44f7294a9e042deb23872ab39 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Magnus=20Gro=C3=9F?= Date: Fri, 26 Oct 2018 17:43:24 +0200 Subject: [PATCH] Implement double playback speed This adds double playback speed for both voice messages and round video messages. The 2x playback speed setting is global and is saved in local storage. Fixes #4907 --- Telegram/Resources/icons/voice2x.png | Bin 0 -> 479 bytes Telegram/Resources/icons/voice2x@2x.png | Bin 0 -> 1015 bytes Telegram/Resources/icons/voice2x@3x.png | Bin 0 -> 1550 bytes Telegram/SourceFiles/facades.cpp | 2 + Telegram/SourceFiles/facades.h | 1 + Telegram/SourceFiles/media/media_audio.cpp | 59 ++++++++++++++++++ Telegram/SourceFiles/media/media_audio.h | 2 + .../media/player/media_player.style | 26 ++++++++ .../media/player/media_player_widget.cpp | 28 +++++++++ .../media/player/media_player_widget.h | 2 + Telegram/SourceFiles/storage/localstorage.cpp | 10 +++ 11 files changed, 130 insertions(+) create mode 100644 Telegram/Resources/icons/voice2x.png create mode 100644 Telegram/Resources/icons/voice2x@2x.png create mode 100644 Telegram/Resources/icons/voice2x@3x.png diff --git a/Telegram/Resources/icons/voice2x.png b/Telegram/Resources/icons/voice2x.png new file mode 100644 index 0000000000000000000000000000000000000000..f01a811168261f586f5dd8083ae14162a9fd0374 GIT binary patch literal 479 zcmV<50U-W~P)%hKGW%RdV4sJ z^O>it*K1M~1>J5JkH-V=psFfTsTA(_J3ue8wA(2SD3`0B{BWg4na5x-@$K!}b zqw)YZEf5GGlgXe`sbDY|e2>X!nDJbpP=L*5gT-RO<#NH{a1io-zelxN#dJCa?}RZO zscf}ch0k6653G*K3@*S>^Z+*nqtWQ&4P#52%?66@k$4gOe!ob12bR2Atxzl$@dU>4 V4Yl55*be{z002ovPDHLkV1irN+x!3k literal 0 HcmV?d00001 diff --git a/Telegram/Resources/icons/voice2x@2x.png b/Telegram/Resources/icons/voice2x@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..a37c17097fa076019cf4c60b8e92283604cdbd00 GIT binary patch literal 1015 zcmVe*#3z-FX7)QqphK7d1$HxanMMc=&-iBdsZ;zs*qj7(KkLv1b zOifMU{r%moz~0{8$j;72XJ;n@0|T+YzYhixj@Q@M74h)sk1^O7(`Yi8gi(&t)23lC z<~OP`XE}|Gj6_^q96mliaCmr#gM$N^&v}cFkC%0)r>D5Oy3)Vf*Vk9p@R&*^9v&X- zF|=A7o1C1KO_!FItO|+eg@=brA!(MA zljE=-SOU8|4Vd=s?r!Nk-01l*vDJKDULFDh0(4sQ^YdMIK37z?x3|lb>+5SuOiZ+k zBrGh9&d$zc4tFN5eC|vt@BdD9b91vu>iqnil9Q7i(x++m`1qLAkxJ9k)AA9v+UAt} zD97dH<)lt4*<^lxo~o*g(&Nva*s?ccP@EBzk^+mN^X#4f0{Bsj0+`hOMTi z=I?bpH#cWhsg^IiaY8}@Ei5dEn@>2_uvU<)b2imRg$2!T8L>khbCoCxvAn#D&(BY* zLg8X#eJXIw%*uJIXQ1V zzZ)@X6tNE7b=`6FNxZkFb;dMpZf+taB?U1tG3e^*f@*hezZCG*Co?lsFxVK=pgIyw zOiV~2u&Er_71VD?<^6bXjZMB#Wd#zI){c?aY)P+)tnD4OsAT(rUWkIz- lk6h?4E-s+Dnp}n0{s2r7lyfdwO~L>G002ovPDHLkV1n(d>UaPE literal 0 HcmV?d00001 diff --git a/Telegram/Resources/icons/voice2x@3x.png b/Telegram/Resources/icons/voice2x@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..6f60123cee0f94c8dd73d121c28aeefe41e3dc3f GIT binary patch literal 1550 zcmV+p2J!icP) zf{Io_p`~D$t=tGML^LW9L_x(C30hf11&Kr@{*Z>5simoDmS(4U-gu{X?w{BD-OuZR zyP27H=9x3kIp>+VZbZc5M{aIzv~%Z9N=Qhc*w|Rwx^*jgdU~SEB{n}lPai*iq}J9} zx_9p$efso?m^G-HvA4H3r=_KFZ*Q+?Uw!ua^=m$J<_rrb0yV?KoSmJ`U%!4`eP|7c zfbZ0)Q-48+goN<$@bIbztb_aT;REm5wTnG6GBRk}wr!Fti;IhN|NebyYipzL-@j{I zUugq(eSCcA*s)`D{P=N+VaJXgl#`P~)Ya9cc)7B&@*j8p2C2(_h_&kKY6ah$H*c6` zW@ePe!-o&M45TFo9X)zf!3VtV>({SWPJeWCbfwu~b3Ic$!0T@Mg}b|1lS?;Y36Dbv z5u97LY#~2CKbLH1S)h?+Zr;3!;H8d^4tn+K6^X=8B3LLiG*oV2&6+iI<;oT6?(QZL zIP~Jh3(CyQG}|D&c=_^W>gnl`<3!C^+7YG+de5Fc76UOlI+_u=zkmO>nA4}Hr#U$} zS$-Q45y9`@zn5d5JbA)m)5_eDk&)cr-!I1qC$Km{Wz7El`_0xvJy*;UXrV(G6Y3^3 z0k^leGk}Odk?jDvefxH~=#3jU_~*}`TK~6i-}u_KYm5r4ONX+uGWjjCLV9|-nZq0g zT~JUUC;$BUGoL$mjy*g)G|iHqpU-;Q1i-AUEV*z%Kmd!Qm*3)~P|*PT(4j;8<;xe@ z_weCExglffa~Sm7w{PX-wY9a%s=+g~! zLO|p9?%g{+aNvN|D(o;ltmzMtr}@sEI}!|ZuhhH-%>wQt=)|cgK0?OH%geJV8=MLn z9v8WofBpKU0hyYb;*^vWs{o)O5X449EbQI8ciEtqRa}guJ9+XX<>uxR3JI*fzMg7o zYOL~dN>m%*qv?)E};^J7`*|bk_<5U4;3&eDT%Q;7=o{^u1*tKMmixLo<4mlr^4nPR+IlIr2ryZ_{%P(Wj`@#$~j%hWSB|PrX!$#A!#A*RCa&2fI{rLf2*Gem1b_xp%S!ZoZNhK-#0YWILt^R@_TL1t607*qoM6N<$g7$S4 A_W%F@ literal 0 HcmV?d00001 diff --git a/Telegram/SourceFiles/facades.cpp b/Telegram/SourceFiles/facades.cpp index 0b83ee97a..749d4098d 100644 --- a/Telegram/SourceFiles/facades.cpp +++ b/Telegram/SourceFiles/facades.cpp @@ -649,6 +649,7 @@ struct Data { bool SuggestEmoji = true; bool SuggestStickersByEmoji = true; base::Observable ReplaceEmojiChanged; + float VoiceMsgPlaybackSpeed = 1.f; bool SoundNotify = true; bool DesktopNotify = true; bool RestoreSoundNotifyFromTray = false; @@ -778,6 +779,7 @@ DefineVar(Global, bool, ReplaceEmoji); DefineVar(Global, bool, SuggestEmoji); DefineVar(Global, bool, SuggestStickersByEmoji); DefineRefVar(Global, base::Observable, ReplaceEmojiChanged); +DefineVar(Global, float, VoiceMsgPlaybackSpeed); DefineVar(Global, bool, SoundNotify); DefineVar(Global, bool, DesktopNotify); DefineVar(Global, bool, RestoreSoundNotifyFromTray); diff --git a/Telegram/SourceFiles/facades.h b/Telegram/SourceFiles/facades.h index e1d2d1f30..b257918b0 100644 --- a/Telegram/SourceFiles/facades.h +++ b/Telegram/SourceFiles/facades.h @@ -304,6 +304,7 @@ DeclareVar(bool, ReplaceEmoji); DeclareVar(bool, SuggestEmoji); DeclareVar(bool, SuggestStickersByEmoji); DeclareRefVar(base::Observable, ReplaceEmojiChanged); +DeclareVar(float, VoiceMsgPlaybackSpeed); DeclareVar(bool, SoundNotify); DeclareVar(bool, DesktopNotify); DeclareVar(bool, RestoreSoundNotifyFromTray); diff --git a/Telegram/SourceFiles/media/media_audio.cpp b/Telegram/SourceFiles/media/media_audio.cpp index fb46a0048..a7496755c 100644 --- a/Telegram/SourceFiles/media/media_audio.cpp +++ b/Telegram/SourceFiles/media/media_audio.cpp @@ -14,6 +14,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "media/media_audio_track.h" #include "platform/platform_audio.h" #include "messenger.h" +#include "facades.h" #include #include @@ -44,6 +45,13 @@ namespace { Player::Mixer *MixerInstance = nullptr; +struct PlaybackSpeedData { + ALuint uiEffectSlot = 0; + ALuint uiEffect = 0; + ALuint uiFilter = 0; +}; +PlaybackSpeedData _playbackSpeedData; + // Thread: Any. bool ContextErrorHappened() { ALenum errCode; @@ -144,6 +152,24 @@ bool CreatePlaybackDevice() { alListener3f(AL_VELOCITY, 0.f, 0.f, 0.f); alListenerfv(AL_ORIENTATION, v); + // playback speed related init + // generate an effect slot and an effect + alGenAuxiliaryEffectSlots(1, &_playbackSpeedData.uiEffectSlot); + alGenEffects(1, &_playbackSpeedData.uiEffect); + // initialize the pitch shifter effect + alEffecti(_playbackSpeedData.uiEffect, AL_EFFECT_TYPE, AL_EFFECT_PITCH_SHIFTER); + // 12 semitones = 1 octave + alEffecti(_playbackSpeedData.uiEffect, AL_PITCH_SHIFTER_COARSE_TUNE, -12); + // connect the effect with the effect slot + alAuxiliaryEffectSloti(_playbackSpeedData.uiEffectSlot, AL_EFFECTSLOT_EFFECT, _playbackSpeedData.uiEffect); + // initialize a filter to disable the direct (dry) path + alGenFilters(1, &_playbackSpeedData.uiFilter); + alFilteri(_playbackSpeedData.uiFilter, AL_FILTER_TYPE, AL_FILTER_LOWPASS); + // disable all frequencies + alFilterf(_playbackSpeedData.uiFilter, AL_LOWPASS_GAIN, 0.f); + // to use the modified playback speed: + // connect both the effect slot and filter with the stream source and set AL_PITCH + alDistanceModel(AL_NONE); return true; @@ -154,6 +180,15 @@ void ClosePlaybackDevice() { if (!AudioDevice) return; LOG(("Audio Info: Closing audio playback device.")); + + // playback speed related + alDeleteFilters(1, &_playbackSpeedData.uiFilter); + alDeleteEffects(1, &_playbackSpeedData.uiEffect); + alDeleteAuxiliaryEffectSlots(1, &_playbackSpeedData.uiEffectSlot); + _playbackSpeedData.uiFilter = 0; + _playbackSpeedData.uiEffect = 0; + _playbackSpeedData.uiEffectSlot = 0; + if (Player::mixer()) { Player::mixer()->detachTracks(); } @@ -290,6 +325,7 @@ void Mixer::Track::createStream() { alSource3f(stream.source, AL_VELOCITY, 0, 0, 0); alSourcei(stream.source, AL_LOOPING, 0); alGenBuffers(3, stream.buffers); + mixer()->updatePlaybackSpeed(); } void Mixer::Track::destroyStream() { @@ -983,6 +1019,29 @@ void Mixer::stop(const AudioMsgId &audio, State state) { if (current) emit updated(current); } +void Mixer::updatePlaybackSpeed() +{ + const auto track = trackForType(AudioMsgId::Type::Voice); + if (!track || track->state.id.type() != AudioMsgId::Type::Voice || !track->isStreamCreated()) { + return; + } + const auto src = track->stream.source; + // Note: This alters the playback speed AND the pitch + alSourcef(src, AL_PITCH, Global::VoiceMsgPlaybackSpeed()); + // fix the pitch using effects and filters + if (Global::VoiceMsgPlaybackSpeed() > 1.f) { + // connect the effect slot with the stream + alSource3i(src, AL_AUXILIARY_SEND_FILTER, Media::Audio::_playbackSpeedData.uiEffectSlot, 0, 0); + // connect the filter with the stream + alSourcei(src, AL_DIRECT_FILTER, Media::Audio::_playbackSpeedData.uiFilter); + } else { + // disconnect the effect slot + alSource3i(src, AL_AUXILIARY_SEND_FILTER, AL_EFFECTSLOT_NULL, 0, 0); + // disconnect the filter + alSourcei(src, AL_DIRECT_FILTER, AL_FILTER_NULL); + } +} + void Mixer::stopAndClear() { Track *current_audio = nullptr, *current_song = nullptr; { diff --git a/Telegram/SourceFiles/media/media_audio.h b/Telegram/SourceFiles/media/media_audio.h index fe1921281..016c26e85 100644 --- a/Telegram/SourceFiles/media/media_audio.h +++ b/Telegram/SourceFiles/media/media_audio.h @@ -117,6 +117,8 @@ public: void stop(const AudioMsgId &audio); void stop(const AudioMsgId &audio, State state); + void updatePlaybackSpeed(); + // Video player audio stream interface. void feedFromVideo(VideoSoundPart &&part); int64 getVideoCorrectedTime(const AudioMsgId &id, TimeMs frameMs, TimeMs systemMs); diff --git a/Telegram/SourceFiles/media/player/media_player.style b/Telegram/SourceFiles/media/player/media_player.style index f012f67b7..f6bc35d9a 100644 --- a/Telegram/SourceFiles/media/player/media_player.style +++ b/Telegram/SourceFiles/media/player/media_player.style @@ -72,6 +72,32 @@ mediaPlayerRepeatInactiveIcon: icon { { "player_repeat", mediaPlayerInactiveFg, point(9px, 11px)} }; +mediaPlayerSpeedButton: IconButton { + width: 31px; + height: 30px; + + icon: icon { + { "voice2x", mediaPlayerActiveFg, point(8px, 11px) } + }; + iconPosition: point(0px, 0px); + + rippleAreaPosition: point(3px, 5px); + rippleAreaSize: 25px; + ripple: RippleAnimation(defaultRippleAnimation) { + color: lightButtonBgOver; + } +} +mediaPlayerSpeedDisabledIcon: icon { + { "voice2x", menuIconFg, point(8px, 11px)} +}; +mediaPlayerSpeedDisabledIconOver: icon { + { "voice2x", menuIconFgOver, point(8px, 11px)} +}; +mediaPlayerSpeedDisabledRippleBg: windowBgOver; +mediaPlayerSpeedInactiveIcon: icon { + { "voice2x", mediaPlayerInactiveFg, point(8px, 11px)} +}; + mediaPlayerVolumeIcon0: icon { { "player_volume0", mediaPlayerActiveFg }, }; diff --git a/Telegram/SourceFiles/media/player/media_player_widget.cpp b/Telegram/SourceFiles/media/player/media_player_widget.cpp index f91d24095..9baa44ee7 100644 --- a/Telegram/SourceFiles/media/player/media_player_widget.cpp +++ b/Telegram/SourceFiles/media/player/media_player_widget.cpp @@ -22,7 +22,9 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "styles/style_media_player.h" #include "styles/style_mediaview.h" #include "history/history_item.h" +#include "storage/localstorage.h" #include "layout.h" +#include "facades.h" namespace Media { namespace Player { @@ -80,6 +82,7 @@ Widget::Widget(QWidget *parent) : RpWidget(parent) , _playPause(this) , _volumeToggle(this, st::mediaPlayerVolumeToggle) , _repeatTrack(this, st::mediaPlayerRepeatButton) +, _playbackSpeed(this, st::mediaPlayerSpeedButton) , _close(this, st::mediaPlayerClose) , _shadow(this) , _playbackSlider(this, st::mediaPlayerPlayback) @@ -128,6 +131,14 @@ Widget::Widget(QWidget *parent) : RpWidget(parent) instance()->toggleRepeat(AudioMsgId::Type::Song); }); + updatePlaybackSpeedIcon(); + _playbackSpeed->setClickedCallback([=] { + Global::SetVoiceMsgPlaybackSpeed(Global::VoiceMsgPlaybackSpeed() == 1.f ? 2.f : 1.f); + mixer()->updatePlaybackSpeed(); + updatePlaybackSpeedIcon(); + Local::writeSettings(); + }); + subscribe(instance()->repeatChangedNotifier(), [this](AudioMsgId::Type type) { if (type == _type) { updateRepeatTrackIcon(); @@ -247,6 +258,9 @@ void Widget::handleSeekFinished(float64 progress) { void Widget::resizeEvent(QResizeEvent *e) { auto right = st::mediaPlayerCloseRight; _close->moveToRight(right, st::mediaPlayerPlayTop); right += _close->width(); + if (_type == AudioMsgId::Type::Voice) { + _playbackSpeed->moveToRight(right, st::mediaPlayerPlayTop); right += _playbackSpeed->width(); + } _repeatTrack->moveToRight(right, st::mediaPlayerPlayTop); right += _repeatTrack->width(); _volumeToggle->moveToRight(right, st::mediaPlayerPlayTop); right += _volumeToggle->width(); @@ -330,6 +344,8 @@ int Widget::getLabelsRight() const { auto result = st::mediaPlayerCloseRight + _close->width(); if (_type == AudioMsgId::Type::Song) { result += _repeatTrack->width() + _volumeToggle->width(); + } else if (_type == AudioMsgId::Type::Voice) { + result += _playbackSpeed->width(); } result += st::mediaPlayerPadding; return result; @@ -353,6 +369,14 @@ void Widget::updateRepeatTrackIcon() { _repeatTrack->setRippleColorOverride(repeating ? nullptr : &st::mediaPlayerRepeatDisabledRippleBg); } +void Widget::updatePlaybackSpeedIcon() +{ + const auto playbackSpeed = Global::VoiceMsgPlaybackSpeed(); + const auto isDefaultSpeed = playbackSpeed == 1.f; + _playbackSpeed->setIconOverride(isDefaultSpeed ? &st::mediaPlayerSpeedDisabledIcon : nullptr, isDefaultSpeed ? &st::mediaPlayerSpeedDisabledIconOver : nullptr); + _playbackSpeed->setRippleColorOverride(isDefaultSpeed ? &st::mediaPlayerSpeedDisabledRippleBg : nullptr); +} + void Widget::checkForTypeChange() { auto hasActiveType = [](AudioMsgId::Type type) { auto current = instance()->current(type); @@ -372,6 +396,7 @@ void Widget::setType(AudioMsgId::Type type) { _type = type; _repeatTrack->setVisible(_type == AudioMsgId::Type::Song); _volumeToggle->setVisible(_type == AudioMsgId::Type::Song); + _playbackSpeed->setVisible(_type == AudioMsgId::Type::Voice); if (!_shadow->isHidden()) { _playbackSlider->setVisible(_type == AudioMsgId::Type::Song); } @@ -384,6 +409,9 @@ void Widget::setType(AudioMsgId::Type type) { ) | rpl::start_with_next([=] { handlePlaylistUpdate(); }); + // maybe the type change causes a change of the button layout + QResizeEvent event = { size(), size() }; + resizeEvent(&event); } } diff --git a/Telegram/SourceFiles/media/player/media_player_widget.h b/Telegram/SourceFiles/media/player/media_player_widget.h index a715a61a1..6e2160f69 100644 --- a/Telegram/SourceFiles/media/player/media_player_widget.h +++ b/Telegram/SourceFiles/media/player/media_player_widget.h @@ -66,6 +66,7 @@ private: void updatePlayPrevNextPositions(); void updateLabelsGeometry(); void updateRepeatTrackIcon(); + void updatePlaybackSpeedIcon(); void createPrevNextButtons(); void destroyPrevNextButtons(); @@ -103,6 +104,7 @@ private: object_ptr _nextTrack = { nullptr }; object_ptr _volumeToggle; object_ptr _repeatTrack; + object_ptr _playbackSpeed; object_ptr _close; object_ptr _shadow = { nullptr }; object_ptr _playbackSlider; diff --git a/Telegram/SourceFiles/storage/localstorage.cpp b/Telegram/SourceFiles/storage/localstorage.cpp index fac293a1e..22f354064 100644 --- a/Telegram/SourceFiles/storage/localstorage.cpp +++ b/Telegram/SourceFiles/storage/localstorage.cpp @@ -598,6 +598,7 @@ enum { dbiCacheSettings = 0x56, dbiAnimationsDisabled = 0x57, dbiScalePercent = 0x58, + dbiPlaybackSpeed = 0x59, dbiEncryptedWithSalt = 333, dbiEncrypted = 444, @@ -1749,6 +1750,14 @@ bool _readSetting(quint32 blockId, QDataStream &stream, int version, ReadSetting Global::SetVideoVolume(snap(v / 1e6, 0., 1.)); } break; + case dbiPlaybackSpeed: { + quint32 v; + stream >> v; + if (!_checkStreamStatus(stream)) return false; + + Global::SetVoiceMsgPlaybackSpeed(v); + } break; + default: LOG(("App Error: unknown blockId in _readSetting: %1").arg(blockId)); return false; @@ -2596,6 +2605,7 @@ void writeSettings() { data.stream << quint32(dbiLoggedPhoneNumber) << cLoggedPhoneNumber(); data.stream << quint32(dbiTxtDomainString) << Global::TxtDomainString(); data.stream << quint32(dbiAnimationsDisabled) << qint32(anim::Disabled() ? 1 : 0); + data.stream << quint32(dbiPlaybackSpeed) << quint32(Global::VoiceMsgPlaybackSpeed()); data.stream << quint32(dbiConnectionType) << qint32(dbictProxiesList); data.stream << qint32(proxies.size());