From 3c8fb5f1f60abfec5fa5238347cca7ae24fd48c4 Mon Sep 17 00:00:00 2001 From: John Preston Date: Sat, 8 Oct 2016 21:28:43 +0300 Subject: [PATCH] Media player vector button added to new audio player cover. --- Telegram/Resources/icons/player_pause.png | Bin 118 -> 0 bytes Telegram/Resources/icons/player_pause@2x.png | Bin 156 -> 0 bytes Telegram/Resources/icons/player_play.png | Bin 14676 -> 0 bytes Telegram/Resources/icons/player_play@2x.png | Bin 14817 -> 0 bytes .../media/player/media_player.style | 59 ++-- .../media/player/media_player_button.cpp | 266 +++++++++--------- .../media/player/media_player_button.h | 38 ++- .../media/player/media_player_cover.cpp | 40 ++- .../media/player/media_player_cover.h | 3 +- .../player/media_player_title_button.cpp | 111 ++++++++ .../media/player/media_player_title_button.h | 54 ++++ Telegram/SourceFiles/title.cpp | 2 +- Telegram/gyp/Telegram.gyp | 2 + 13 files changed, 404 insertions(+), 171 deletions(-) delete mode 100644 Telegram/Resources/icons/player_pause.png delete mode 100644 Telegram/Resources/icons/player_pause@2x.png delete mode 100644 Telegram/Resources/icons/player_play.png delete mode 100644 Telegram/Resources/icons/player_play@2x.png create mode 100644 Telegram/SourceFiles/media/player/media_player_title_button.cpp create mode 100644 Telegram/SourceFiles/media/player/media_player_title_button.h diff --git a/Telegram/Resources/icons/player_pause.png b/Telegram/Resources/icons/player_pause.png deleted file mode 100644 index 85ea133ff0c15b6ded17aefe5d72038f7fc677dc..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 118 zcmeAS@N?(olHy`uVBq!ia0vp^d_XL~!3HGNrubO_DO*n$#}JK)Z@Ufo7!-JzR{pOS z*uxhdZM|dG7a=x1hDmmcnGBkD)pDd;J3RKzOo)9p(em8K*^8H`WWArGl5=5=g`4gD S-xffF7(8A5T-G@yGywpsj(d666g7v&pqrP%^M7XXFc?pcKk37?VCK~PJ;77&J>{G8&cgGS!n?SC{ONC` zwu&hSEO2OKV&xLi2v~3+{y?!`bky6D^e~q-bF$B$<1PNra`(*RIp=}aF?hQAxvXg%^#2ZhR76z%w)5_G`E0fZ_2p zy?lGl|Nnhwe&@3DGKa>tKe{GzUj!kvW@I=&PJZtupF3BrAm6*sU0q9lR$Ifn9E5Ja z%l|AxM~>Z#Q0f7#IO$FnMrGC1<4V~q;kc(;M2%2pvu7#lEOxmPp4NKw&P6hwOcLnWRX9(t7>?{^qd{Jo#C|g$cVbF(<_bVyf7e;F9UMD5Vpr{ztfkAP(?CKQE?ZVxLUj{Q_5iMRO4n zW$f~lJf7cHFGv2$MrT~tl6k&XtHo=5anqhA_H>#T5_}>NBZ`KAIf>er=PjY5V~XGbXVo?`K0T;MyI%4)-7RqaYJw6eaY-^jR~eYdSM#my(U{u7$~*6bg;n9mlKd9ZNgz&|ZR|jjHR9Xs~4E;hd>v;|8coS_W4R(a6Iifh_M)2!seRW`PW zkQ*HISCtf(xCaZ0su@1nhcbM-tvTn-j5x2jI47os{+J+=cTwn*1>xb~8)OJ&Z3zly zS(};*68MWcCXf{^6~&Y+BxNZHa{y&~4zhvDimTjE7N`Yz%BnnN+PdQAG+mj-yk$&h z_+}$eHkOq38~{&uOSF5hHoU|*7YhpmyLPN@KmKl+<3 z(IL%}Qp1`<7U&#UxMcYThuH*vy0vw&qoM_elBL)V_P1O%y13^Wr&!~V@;$-Ul@--L z&T^{1BQOEn(%KwW!&$d!|91kz{J`G5Mr{TgU3XKMCy;c{;FwcxO|kKoX|f=0bf<@z zTdnOSGQ5BOSG3ld2wPhxXmQCdozRSbtjT0Ckg+MjB#0S)Nn5jWsGsyFwNefA$%H^& zyy$<|t+2b@y6gH@*xhbzI(HhHn@iNIwKW5mfA)sw!UVZZz_|fQ>XR~jdvm*EO!IJM zGOv*h?D)$vL9Q$Cw0Cqg?rIvv6f}a4x-pz8@DDwh$*`5&E|5DFzHzJ4I?0WlaEpWH z72V>*FdZubX;5Q^i;4&e7%oVI8Z%r}L{PwRK^oMU;i4jf0)`9HpvDXr6%iCLT#yDe zX1J(`pn&0mG^jDdMMVS!3>Tz9jTtT~A}C8Ps4>GuMFa&57oj3(}y* z*TofSKKzFb^2A?_JldCg{kLKASRkhkj~5V{zaOC&_9JxV8u|SLp{jt;pL-CJ-$rP? z`NmUcw-N0JBl#^w@28IrpI*Iu!`h)QdV17}O{ey0XM0baiHv@7+r;i4wvN8KX7uXN zSN1b2zxn&p{*i6o`PB0p4xFj2TX*8)%hkQdU;FNtO^ADcx%%yM7v2$9Y+W_>*gvb* zpOjL+Uq0~ic;w*09iM)kzIJp$x^TpNW!)3Ay%&!Ed&ffdy~AGYV(iUoO9R2I!>6dzTKRdU;ue>yk4*&YryC;?@5*afZ8QPxzaO<9Z F{{fGF?&Sag diff --git a/Telegram/Resources/icons/player_play@2x.png b/Telegram/Resources/icons/player_play@2x.png deleted file mode 100644 index efdd63789bc0ab325dcb3f990bbe858b65a5bf56..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 14817 zcmeI3ZEO@p7{|AYZ8-$0B37biSrB>Y-R`|?+grA$rB`eTS4z2R8&uft&b6D~-JZ8w zdTo@Kf(lgu(I9CAjfs5l15%?TN|f*czmQl68Y3}50)*J|W(+Y<8l2s`-c7Gp1`HqP zaxYJ3{{LrYfAh?AzwEa9x)-O~X4xo;np#`qZ-Bq^;b(4PKKvdQ6Eoq@w%fBplU+Lu~a3SGlKzc5%*9mh-Squifbu;M>E|sTbq3 z!AOiNA)&#)G&LOjtAc4WjF`Z(ot>SIPPapgw?ezu%W^K(<+4MCUGIt-Qo2vSIMI`v>z z46qGA*E-@ds9pyzWlpZEH`GvOamHjadnTMaNT?Io%$}spDuv_8%`+K+UowEt>E%3j&I#W^&Mk1ar4;MQ!I`Me{*txxkgdqDFHd%Iq9)11XXr-BlJ;i{w#cA*974lHpS$ zQY&C%(N>YoG$P7$SqW8ISc}7H0iRoBb2TF_tJH+rsBTD68PxiHu)(3Kiol1wUZA)= z_R?~t-0t!4irpJ(@!I8b5aKvV=1VW0w^LorZqFIv@>N>|L+1O^CR|&$LMSW(Fu1`GLI#hID@W*j7}*I zDqCSe-0eEPa`agD=);ey6Nxceb_p&Q`QL+nRAUMpeV-N{yD)jJV9RH+e3sg0~64SIYBl zUSzYIvmNzn0)(6VD%`-jS(YAnU6v=iBd4)BV|16IMx-NUOfD7j58YbHWGlQ~fOjfv z`c`G^BByu4eGW2Lbe|KW>Bt}`4QxzsVG*GMf(xaAjR`I+B2++dp){~D!G%SH3J5Ng z1~w+Ru!v9r!G+Sm#sn7@5h@_KP#V~n;KCw81q2sL0~-@uSVX9R;6iC&V}c8d2o(@q zC=F~(aA6Ul0)h*rfsF|+EFx4uaG^A?F~NmJgbD~Qlm<2?xUh&&0l|gRz{Ug@77;2S zxKJ9{nBc-9LIngDN&_1cTv$Y?fZ#%DU}J&{iwG4ETqq4}OmJZlp#p*nrGbqJE-WHc zKyaZnura}fMT80nE|dl~z9TMM=HWjOg-`r-!bkhAtbg({d@PWbYZ?L+)%`3*ZQM*z zH*Uk8o6m0>99%O(Gdt%k zDLj4X{h!ZYeEK`7xMo@HZm@UZ zz|fz=zBQHAjg7IV-ua5AE)8!N2c*7vD=LqP16$7(zb##x5q$65R_@^&wuOEBHw}IN z#zT!8mMxH)XRbIpykCGz%BDY1&bQc) zFR9&Cw7Qb{{7~fjgF}r6AAe={Tl)r{>t8&#K;8P=p8m-2GryeK@N56HDdBT3o$_20 zuWr9}=7^{NwZ*?XuDp3>PGQgR-!-duUifGA${C+t`SxPrqbE1)TYkBFDpkhYt`{vD RX*3_gtgWu|A6T-s=Rfh$H2DAk diff --git a/Telegram/SourceFiles/media/player/media_player.style b/Telegram/SourceFiles/media/player/media_player.style index 7d2a72458..6ae882faf 100644 --- a/Telegram/SourceFiles/media/player/media_player.style +++ b/Telegram/SourceFiles/media/player/media_player.style @@ -22,16 +22,42 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org using "basic.style"; using "ui/widgets/widgets.style"; +MediaPlayerButton { + playPosition: point; + playOuter: size; + pausePosition: point; + pauseOuter: size; + pauseStroke: pixels; + cancelPosition: point; + cancelOuter: size; + cancelStroke: pixels; +} + mediaPlayerTitleButtonSize: size(titleHeight, titleHeight); mediaPlayerTitleButtonInner: size(25px, 25px); mediaPlayerTitleButtonInnerBg: #49708f; -mediaPlayerTitleButtonTransformDuration: 200; +mediaPlayerButtonTransformDuration: 200; -mediaPlayerTitleButtonPauseLeft: 8px; -mediaPlayerTitleButtonPauseTop: 8px; -mediaPlayerTitleButtonPauseStroke: 3px; -mediaPlayerTitleButtonPlayLeft: 10px; -mediaPlayerTitleButtonPlayTop: 7px; +mediaPlayerTitleButton: MediaPlayerButton { + playPosition: point(10px, 7px); + playOuter: size(29px, 25px); + pausePosition: point(8px, 8px); + pauseOuter: size(25px, 25px); + pauseStroke: 3px; + cancelPosition: point(8px, 8px); + cancelOuter: size(25px, 25px); + cancelStroke: 2px; +} +mediaPlayerButton: MediaPlayerButton { + playPosition: point(3px, 0px); + playOuter: size(22px, 18px); + pausePosition: point(1px, 1px); + pauseOuter: size(16px, 18px); + pauseStroke: 5px; + cancelPosition: point(0px, 1px); + cancelOuter: size(16px, 18px); + cancelStroke: 3px; +} mediaPlayerMarginLeft: 10px; mediaPlayerMarginBottom: 10px; @@ -41,35 +67,28 @@ mediaPlayerCoverHeight: 102px; mediaPlayerActiveFg: #54b5ed; mediaPlayerInactiveFg: #dfebf2; -mediaPlayerPlayButton: IconButton { - width: 32px; +mediaPlayerButtonSize: size(32px, 32px); +mediaPlayerButtonPosition: point(8px, 7px); + +mediaPlayerRepeatButton: IconButton { + width: 31px; height: 32px; opacity: 1.; overOpacity: 1.; icon: icon { - { "player_play", mediaPlayerActiveFg, point(6px, 7px) }, + { "player_repeat", mediaPlayerActiveFg, point(9px, 9px)} }; iconPosition: point(0px, 0px); downIconPosition: point(0px, 0px); duration: 0; } -mediaPlayerPauseIcon: icon { - { "player_pause", mediaPlayerActiveFg, point(9px, 8px) } -}; - -mediaPlayerRepeatButton: IconButton(mediaPlayerPlayButton) { - width: 31px; - icon: icon { - { "player_repeat", mediaPlayerActiveFg, point(9px, 9px)} - }; -} mediaPlayerRepeatDisabledIcon: icon { { "player_repeat", mediaPlayerInactiveFg, point(9px, 9px)} }; -mediaPlayerPreviousButton: IconButton(mediaPlayerPlayButton) { +mediaPlayerPreviousButton: IconButton(mediaPlayerRepeatButton) { width: 37px; icon: icon { { "player_previous", mediaPlayerActiveFg, point(10px, 10px) }, diff --git a/Telegram/SourceFiles/media/player/media_player_button.cpp b/Telegram/SourceFiles/media/player/media_player_button.cpp index 43abc77a6..e56f39060 100644 --- a/Telegram/SourceFiles/media/player/media_player_button.cpp +++ b/Telegram/SourceFiles/media/player/media_player_button.cpp @@ -21,84 +21,8 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org #include "stdafx.h" #include "media/player/media_player_button.h" -#include "styles/style_media_player.h" -#include "media/media_audio.h" -#include "media/player/media_player_instance.h" -#include "shortcuts.h" - namespace Media { namespace Player { - -TitleButton::TitleButton(QWidget *parent) : Button(parent) { - setAttribute(Qt::WA_OpaquePaintEvent); - resize(st::mediaPlayerTitleButtonSize); - - setClickedCallback([this]() { - if (exists()) { - if (_showPause) { - instance()->pause(); - } else { - instance()->play(); - } - } - }); - - if (exists()) { - subscribe(instance()->updatedNotifier(), [this](const UpdatedEvent &e) { - updatePauseState(); - }); - updatePauseState(); - finishIconTransform(); - } -} - -void TitleButton::updatePauseState() { - AudioMsgId playing; - auto playbackState = audioPlayer()->currentState(&playing, AudioMsgId::Type::Song); - auto stopped = ((playbackState.state & AudioPlayerStoppedMask) || playbackState.state == AudioPlayerFinishing); - auto showPause = !stopped && (playbackState.state == AudioPlayerPlaying || playbackState.state == AudioPlayerResuming || playbackState.state == AudioPlayerStarting); - if (exists() && instance()->isSeeking()) { - showPause = true; - } - setShowPause(showPause); -} - -void TitleButton::setShowPause(bool showPause) { - if (_showPause != showPause) { - _showPause = showPause; - _iconTransformToPause.start([this] { update(); }, _showPause ? 0. : 1., _showPause ? 1. : 0., st::mediaPlayerTitleButtonTransformDuration); - update(); - } -} - -void TitleButton::finishIconTransform() { - if (_iconTransformToPause.animating(getms())) { - _iconTransformToPause.finish(); - update(); - } -} - -void TitleButton::paintEvent(QPaintEvent *e) { - Painter p(this); - p.fillRect(rect(), st::titleBg); - - p.setBrush(st::mediaPlayerTitleButtonInnerBg); - p.setPen(Qt::NoPen); - - p.setRenderHint(QPainter::HighQualityAntialiasing, true); - p.drawEllipse((width() - st::mediaPlayerTitleButtonInner.width()) / 2, (height() - st::mediaPlayerTitleButtonInner.height()) / 2, st::mediaPlayerTitleButtonInner.width(), st::mediaPlayerTitleButtonInner.height()); - p.setRenderHint(QPainter::HighQualityAntialiasing, false); - - paintIcon(p); -} - -void TitleButton::onStateChanged(int oldState, ButtonStateChangeSource source) { - if ((oldState & StateOver) != (_state & StateOver)) { - auto over = (_state & StateOver); - _iconFg.start([this] { update(); }, over ? st::titleButtonFg->c : st::titleButtonActiveFg->c, over ? st::titleButtonActiveFg->c : st::titleButtonFg->c, st::titleButtonDuration); - } -} - namespace { template @@ -119,65 +43,147 @@ QPainterPath interpolatePaths(QPointF (&from)[N], QPointF (&to)[N], float64 k) { } // namespace -void TitleButton::paintIcon(Painter &p) { - auto over = (_state & StateOver); - auto icon = _iconFg.current(getms(), over ? st::titleButtonActiveFg->c : st::titleButtonFg->c); - auto showPause = _iconTransformToPause.current(getms(), _showPause ? 1. : 0.); - auto pauseWidth = st::mediaPlayerTitleButtonInner.width() - 2 * st::mediaPlayerTitleButtonPauseLeft; - auto playWidth = pauseWidth; - auto pauseHeight = st::mediaPlayerTitleButtonInner.height() - 2 * st::mediaPlayerTitleButtonPauseTop; - auto playHeight = st::mediaPlayerTitleButtonInner.height() - 2 * st::mediaPlayerTitleButtonPlayTop; - auto pauseStroke = st::mediaPlayerTitleButtonPauseStroke; +PlayButtonLayout::PlayButtonLayout(const style::MediaPlayerButton &st, State state, UpdateCallback &&callback) +: _st(st) +, _state(state) +, _oldState(state) +, _nextState(state) +, _callback(std_::move(callback)) { +} - qreal left = (width() - st::mediaPlayerTitleButtonInner.width()) / 2; - qreal top = (height() - st::mediaPlayerTitleButtonInner.height()) / 2; +void PlayButtonLayout::setState(State state) { + if (_nextState == state) return; - auto pauseLeft = left + st::mediaPlayerTitleButtonPauseLeft; - auto playLeft = left + st::mediaPlayerTitleButtonPlayLeft; - auto pauseTop = top + st::mediaPlayerTitleButtonPauseTop; - auto playTop = top + st::mediaPlayerTitleButtonPlayTop; - - p.setRenderHint(QPainter::HighQualityAntialiasing, true); - p.setPen(Qt::NoPen); - - if (showPause == 0.) { - QPainterPath pathPlay; - pathPlay.moveTo(playLeft, playTop); - pathPlay.lineTo(playLeft + playWidth, playTop + (playHeight / 2.)); - pathPlay.lineTo(playLeft, playTop + playHeight); - pathPlay.lineTo(playLeft, playTop); - p.fillPath(pathPlay, icon); - } else { - QPointF pathLeftPause[] = { - { pauseLeft, pauseTop }, - { pauseLeft + pauseStroke, pauseTop }, - { pauseLeft + pauseStroke, pauseTop + pauseHeight }, - { pauseLeft, pauseTop + pauseHeight }, - }; - QPointF pathLeftPlay[] = { - { playLeft, playTop }, - { playLeft + (playWidth / 2.), playTop + (playHeight / 4.) }, - { playLeft + (playWidth / 2.), playTop + (3 * playHeight / 4.) }, - { playLeft, playTop + playHeight }, - }; - p.fillPath(interpolatePaths(pathLeftPause, pathLeftPlay, showPause), icon); - - QPointF pathRightPause[] = { - { pauseLeft + pauseWidth - pauseStroke, pauseTop }, - { pauseLeft + pauseWidth, pauseTop }, - { pauseLeft + pauseWidth, pauseTop + pauseHeight }, - { pauseLeft + pauseWidth - pauseStroke, pauseTop + pauseHeight }, - }; - QPointF pathRightPlay[] = { - { playLeft + (playWidth / 2.), playTop + (playHeight / 4.) }, - { playLeft + playWidth, playTop + (playHeight / 2.) }, - { playLeft + playWidth, playTop + (playHeight / 2.) }, - { playLeft + (playWidth / 2.), playTop + (3 * playHeight / 4.) }, - }; - p.fillPath(interpolatePaths(pathRightPause, pathRightPlay, showPause), icon); + _nextState = state; + if (!_transformProgress.animating(getms())) { + _oldState = _state; + _state = _nextState; + _transformBackward = false; + if (_state != _oldState) startTransform(0., 1.); + } else if (_oldState == _nextState) { + qSwap(_oldState, _state); + startTransform(_transformBackward ? 0. : 1., _transformBackward ? 1. : 0.); + _transformBackward = !_transformBackward; } +} + +void PlayButtonLayout::paint(Painter &p, const QBrush &brush) { + if (_transformProgress.animating(getms())) { + auto from = _oldState, to = _state; + auto backward = _transformBackward; + auto progress = _transformProgress.current(1.); + if (from == State::Cancel || (from == State::Pause && to == State::Play)) { + qSwap(from, to); + backward = !backward; + } + if (backward) progress = 1. - progress; + + t_assert(from != to); + if (from == State::Play) { + if (to == State::Pause) { + paintPlayToPause(p, brush, progress); + } else { + t_assert(to == State::Cancel); + paintPlayToCancel(p, brush, progress); + } + } else { + t_assert(from == State::Pause && to == State::Cancel); + paintPauseToCancel(p, brush, progress); + } + } else { + switch (_state) { + case State::Play: paintPlay(p, brush); break; + case State::Pause: paintPlayToPause(p, brush, 1.); break; + case State::Cancel: paintPlayToCancel(p, brush, 1.); break; + } + } +} + +void PlayButtonLayout::paintPlay(Painter &p, const QBrush &brush) { + auto playLeft = 0. + _st.playPosition.x(); + auto playTop = 0. + _st.playPosition.y(); + auto playWidth = _st.playOuter.width() - 2 * playLeft; + auto playHeight = _st.playOuter.height() - 2 * playTop; + + p.setPen(Qt::NoPen); + p.setRenderHint(QPainter::HighQualityAntialiasing, true); + + QPainterPath pathPlay; + pathPlay.moveTo(playLeft, playTop); + pathPlay.lineTo(playLeft + playWidth, playTop + (playHeight / 2.)); + pathPlay.lineTo(playLeft, playTop + playHeight); + pathPlay.lineTo(playLeft, playTop); + p.fillPath(pathPlay, brush); + p.setRenderHint(QPainter::HighQualityAntialiasing, false); } +void PlayButtonLayout::paintPlayToPause(Painter &p, const QBrush &brush, float64 progress) { + auto playLeft = 0. + _st.playPosition.x(); + auto playTop = 0. + _st.playPosition.y(); + auto playWidth = _st.playOuter.width() - 2 * playLeft; + auto playHeight = _st.playOuter.height() - 2 * playTop; + + auto pauseLeft = 0. + _st.pausePosition.x(); + auto pauseTop = 0. + _st.pausePosition.y(); + auto pauseWidth = _st.pauseOuter.width() - 2 * pauseLeft; + auto pauseHeight = _st.pauseOuter.height() - 2 * pauseTop; + auto pauseStroke = 0. + _st.pauseStroke; + + p.setPen(Qt::NoPen); + p.setRenderHint(QPainter::HighQualityAntialiasing, true); + + QPointF pathLeftPause[] = { + { pauseLeft, pauseTop }, + { pauseLeft + pauseStroke, pauseTop }, + { pauseLeft + pauseStroke, pauseTop + pauseHeight }, + { pauseLeft, pauseTop + pauseHeight }, + }; + QPointF pathLeftPlay[] = { + { playLeft, playTop }, + { playLeft + (playWidth / 2.), playTop + (playHeight / 4.) }, + { playLeft + (playWidth / 2.), playTop + (3 * playHeight / 4.) }, + { playLeft, playTop + playHeight }, + }; + p.fillPath(interpolatePaths(pathLeftPause, pathLeftPlay, progress), brush); + + QPointF pathRightPause[] = { + { pauseLeft + pauseWidth - pauseStroke, pauseTop }, + { pauseLeft + pauseWidth, pauseTop }, + { pauseLeft + pauseWidth, pauseTop + pauseHeight }, + { pauseLeft + pauseWidth - pauseStroke, pauseTop + pauseHeight }, + }; + QPointF pathRightPlay[] = { + { playLeft + (playWidth / 2.), playTop + (playHeight / 4.) }, + { playLeft + playWidth, playTop + (playHeight / 2.) }, + { playLeft + playWidth, playTop + (playHeight / 2.) }, + { playLeft + (playWidth / 2.), playTop + (3 * playHeight / 4.) }, + }; + p.fillPath(interpolatePaths(pathRightPause, pathRightPlay, progress), brush); + + p.setRenderHint(QPainter::HighQualityAntialiasing, false); +} + +void PlayButtonLayout::paintPlayToCancel(Painter &p, const QBrush &brush, float64 progress) { + +} + +void PlayButtonLayout::paintPauseToCancel(Painter &p, const QBrush &brush, float64 progress) { + +} + +void PlayButtonLayout::animationCallback() { + if (!_transformProgress.animating()) { + auto finalState = _nextState; + _nextState = _state; + setState(finalState); + } + _callback(); +} + +void PlayButtonLayout::startTransform(float64 from, float64 to) { + _transformProgress.start([this] { animationCallback(); }, from, to, st::mediaPlayerButtonTransformDuration); +} + } // namespace Player } // namespace Media diff --git a/Telegram/SourceFiles/media/player/media_player_button.h b/Telegram/SourceFiles/media/player/media_player_button.h index 4223a2d51..f91f680a3 100644 --- a/Telegram/SourceFiles/media/player/media_player_button.h +++ b/Telegram/SourceFiles/media/player/media_player_button.h @@ -21,30 +21,40 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org #pragma once #include "ui/button.h" +#include "styles/style_media_player.h" namespace Media { namespace Player { -class TitleButton : public Button, private base::Subscriber { +class PlayButtonLayout { public: - TitleButton(QWidget *parent); + enum class State { + Play, + Pause, + Cancel, + }; + using UpdateCallback = FloatAnimation::Callback; + PlayButtonLayout(const style::MediaPlayerButton &st, State state, UpdateCallback &&callback); - void updatePauseState(); - -protected: - void paintEvent(QPaintEvent *e) override; - - void onStateChanged(int oldState, ButtonStateChangeSource source) override; + void setState(State state); + void paint(Painter &p, const QBrush &brush); private: - void paintIcon(Painter &p); + void animationCallback(); + void startTransform(float64 from, float64 to); - void setShowPause(bool showPause); - void finishIconTransform(); + void paintPlay(Painter &p, const QBrush &brush); + void paintPlayToPause(Painter &p, const QBrush &brush, float64 progress); + void paintPlayToCancel(Painter &p, const QBrush &brush, float64 progress); + void paintPauseToCancel(Painter &p, const QBrush &brush, float64 progress); - bool _showPause = true; - FloatAnimation _iconTransformToPause; - ColorAnimation _iconFg; + const style::MediaPlayerButton &_st; + + State _state, _oldState, _nextState; + FloatAnimation _transformProgress; + bool _transformBackward = false; + + UpdateCallback _callback; }; diff --git a/Telegram/SourceFiles/media/player/media_player_cover.cpp b/Telegram/SourceFiles/media/player/media_player_cover.cpp index a2c6da7bd..62e02dcf4 100644 --- a/Telegram/SourceFiles/media/player/media_player_cover.cpp +++ b/Telegram/SourceFiles/media/player/media_player_cover.cpp @@ -26,6 +26,7 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org #include "ui/buttons/icon_button.h" #include "media/media_audio.h" #include "media/view/media_clip_playback.h" +#include "media/player/media_player_button.h" #include "media/player/media_player_instance.h" #include "media/player/media_player_volume_controller.h" #include "styles/style_media_player.h" @@ -35,11 +36,42 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org namespace Media { namespace Player { +class CoverWidget::PlayButton : public Button { +public: + PlayButton(QWidget *parent); + + using State = PlayButtonLayout::State; + void setState(State state); + +protected: + void paintEvent(QPaintEvent *e) override; + +private: + PlayButtonLayout _layout; + +}; + +CoverWidget::PlayButton::PlayButton(QWidget *parent) : Button(parent) +, _layout(st::mediaPlayerButton, State::Pause, [this] { update(); }) { + resize(st::mediaPlayerButtonSize); +} + +void CoverWidget::PlayButton::setState(State state) { + _layout.setState(state); +} + +void CoverWidget::PlayButton::paintEvent(QPaintEvent *e) { + Painter p(this); + + p.translate(st::mediaPlayerButtonPosition.x(), st::mediaPlayerButtonPosition.y()); + _layout.paint(p, st::mediaPlayerActiveFg); +} + CoverWidget::CoverWidget(QWidget *parent) : TWidget(parent) , _nameLabel(this, st::mediaPlayerName) , _timeLabel(this, st::mediaPlayerTime) , _playback(this, st::mediaPlayerPlayback) -, _playPause(this, st::mediaPlayerPlayButton) +, _playPause(this) , _volumeController(this) , _repeatTrack(this, st::mediaPlayerRepeatButton) { setAttribute(Qt::WA_OpaquePaintEvent); @@ -51,9 +83,7 @@ CoverWidget::CoverWidget(QWidget *parent) : TWidget(parent) handleSeekFinished(value); }); - if (_showPause) { - _playPause->setIcon(&st::mediaPlayerPauseIcon); - } + _playPause->setState(_showPause ? PlayButton::State::Pause : PlayButton::State::Play); _playPause->setClickedCallback([this]() { if (exists()) { if (_showPause) { @@ -173,7 +203,7 @@ void CoverWidget::handleSongUpdate(const UpdatedEvent &e) { } if (_showPause != showPause) { _showPause = showPause; - _playPause->setIcon(_showPause ? &st::mediaPlayerPauseIcon : nullptr); + _playPause->setState(showPause ? PlayButton::State::Pause : PlayButton::State::Play); } updateTimeText(audioId, playbackState); diff --git a/Telegram/SourceFiles/media/player/media_player_cover.h b/Telegram/SourceFiles/media/player/media_player_cover.h index 1290b8148..9da7bd306 100644 --- a/Telegram/SourceFiles/media/player/media_player_cover.h +++ b/Telegram/SourceFiles/media/player/media_player_cover.h @@ -69,11 +69,12 @@ private: int64 _lastDurationMs = 0; QString _time; + class PlayButton; ChildWidget _nameLabel; ChildWidget _timeLabel; ChildWidget _playback; ChildWidget _previousTrack = { nullptr }; - ChildWidget _playPause; + ChildWidget _playPause; ChildWidget _nextTrack = { nullptr }; ChildWidget _volumeController; ChildWidget _repeatTrack; diff --git a/Telegram/SourceFiles/media/player/media_player_title_button.cpp b/Telegram/SourceFiles/media/player/media_player_title_button.cpp new file mode 100644 index 000000000..65531ec05 --- /dev/null +++ b/Telegram/SourceFiles/media/player/media_player_title_button.cpp @@ -0,0 +1,111 @@ +/* +This file is part of Telegram Desktop, +the official desktop version of Telegram messaging app, see https://telegram.org + +Telegram Desktop is free software: you can redistribute it and/or modify +it under the terms of the GNU General Public License as published by +the Free Software Foundation, either version 3 of the License, or +(at your option) any later version. + +It is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. + +In addition, as a special exception, the copyright holders give permission +to link the code of portions of this program with the OpenSSL library. + +Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE +Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org +*/ +#include "stdafx.h" +#include "media/player/media_player_title_button.h" + +#include "media/player/media_player_button.h" +#include "media/media_audio.h" +#include "media/player/media_player_instance.h" +#include "shortcuts.h" + +namespace Media { +namespace Player { + +using State = PlayButtonLayout::State; + +TitleButton::TitleButton(QWidget *parent) : Button(parent) +, _layout(std_::make_unique(st::mediaPlayerTitleButton, State::Pause, [this] { update(); })) { + setAttribute(Qt::WA_OpaquePaintEvent); + resize(st::mediaPlayerTitleButtonSize); + + setClickedCallback([this]() { + if (exists()) { + if (_showPause) { + instance()->pause(); + } else { + instance()->play(); + } + } + }); + + if (exists()) { + subscribe(instance()->updatedNotifier(), [this](const UpdatedEvent &e) { + updatePauseState(); + }); + updatePauseState(); + } +} + +void TitleButton::updatePauseState() { + AudioMsgId playing; + auto playbackState = audioPlayer()->currentState(&playing, AudioMsgId::Type::Song); + auto stopped = ((playbackState.state & AudioPlayerStoppedMask) || playbackState.state == AudioPlayerFinishing); + auto showPause = !stopped && (playbackState.state == AudioPlayerPlaying || playbackState.state == AudioPlayerResuming || playbackState.state == AudioPlayerStarting); + if (exists() && instance()->isSeeking()) { + showPause = true; + } + setShowPause(showPause); +} + +void TitleButton::setShowPause(bool showPause) { + if (_showPause != showPause) { + _showPause = showPause; + _layout->setState(_showPause ? PlayButtonLayout::State::Pause : PlayButtonLayout::State::Play); + update(); + } +} + +void TitleButton::paintEvent(QPaintEvent *e) { + Painter p(this); + p.fillRect(rect(), st::titleBg); + + p.setBrush(st::mediaPlayerTitleButtonInnerBg); + p.setPen(Qt::NoPen); + + p.setRenderHint(QPainter::HighQualityAntialiasing, true); + p.drawEllipse((width() - st::mediaPlayerTitleButtonInner.width()) / 2, (height() - st::mediaPlayerTitleButtonInner.height()) / 2, st::mediaPlayerTitleButtonInner.width(), st::mediaPlayerTitleButtonInner.height()); + p.setRenderHint(QPainter::HighQualityAntialiasing, false); + + paintIcon(p); +} + +void TitleButton::onStateChanged(int oldState, ButtonStateChangeSource source) { + if ((oldState & StateOver) != (_state & StateOver)) { + auto over = (_state & StateOver); + _iconFg.start([this] { update(); }, over ? st::titleButtonFg->c : st::titleButtonActiveFg->c, over ? st::titleButtonActiveFg->c : st::titleButtonFg->c, st::titleButtonDuration); + } +} + +void TitleButton::paintIcon(Painter &p) { + auto over = (_state & StateOver); + auto icon = _iconFg.current(getms(), over ? st::titleButtonActiveFg->c : st::titleButtonFg->c); + + auto left = (width() - st::mediaPlayerTitleButtonInner.width()) / 2; + auto top = (height() - st::mediaPlayerTitleButtonInner.height()) / 2; + p.translate(left, top); + + _layout->paint(p, icon); +} + +TitleButton::~TitleButton() = default; + +} // namespace Player +} // namespace Media diff --git a/Telegram/SourceFiles/media/player/media_player_title_button.h b/Telegram/SourceFiles/media/player/media_player_title_button.h new file mode 100644 index 000000000..5ed69e6f0 --- /dev/null +++ b/Telegram/SourceFiles/media/player/media_player_title_button.h @@ -0,0 +1,54 @@ +/* +This file is part of Telegram Desktop, +the official desktop version of Telegram messaging app, see https://telegram.org + +Telegram Desktop is free software: you can redistribute it and/or modify +it under the terms of the GNU General Public License as published by +the Free Software Foundation, either version 3 of the License, or +(at your option) any later version. + +It is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. + +In addition, as a special exception, the copyright holders give permission +to link the code of portions of this program with the OpenSSL library. + +Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE +Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org +*/ +#pragma once + +#include "ui/button.h" + +namespace Media { +namespace Player { + +class PlayButtonLayout; + +class TitleButton : public Button, private base::Subscriber { +public: + TitleButton(QWidget *parent); + + void updatePauseState(); + + ~TitleButton(); + +protected: + void paintEvent(QPaintEvent *e) override; + + void onStateChanged(int oldState, ButtonStateChangeSource source) override; + +private: + void paintIcon(Painter &p); + void setShowPause(bool showPause); + + bool _showPause = true; + std_::unique_ptr _layout; + ColorAnimation _iconFg; + +}; + +} // namespace Clip +} // namespace Media diff --git a/Telegram/SourceFiles/title.cpp b/Telegram/SourceFiles/title.cpp index f28c19004..61d83f233 100644 --- a/Telegram/SourceFiles/title.cpp +++ b/Telegram/SourceFiles/title.cpp @@ -28,7 +28,7 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org #include "boxes/contactsbox.h" #include "boxes/aboutbox.h" #include "media/media_audio.h" -#include "media/player/media_player_button.h" +#include "media/player/media_player_title_button.h" #include "media/player/media_player_widget.h" #include "media/player/media_player_instance.h" diff --git a/Telegram/gyp/Telegram.gyp b/Telegram/gyp/Telegram.gyp index a3322c09d..da6c92f3e 100644 --- a/Telegram/gyp/Telegram.gyp +++ b/Telegram/gyp/Telegram.gyp @@ -268,6 +268,8 @@ '<(src_loc)/media/player/media_player_instance.h', '<(src_loc)/media/player/media_player_list.cpp', '<(src_loc)/media/player/media_player_list.h', + '<(src_loc)/media/player/media_player_title_button.cpp', + '<(src_loc)/media/player/media_player_title_button.h', '<(src_loc)/media/player/media_player_volume_controller.cpp', '<(src_loc)/media/player/media_player_volume_controller.h', '<(src_loc)/media/player/media_player_widget.cpp',