From 48eb72a9c2439e0ee33f82abc8614fa39a0c9530 Mon Sep 17 00:00:00 2001 From: John Preston Date: Sat, 12 Nov 2016 18:02:19 +0300 Subject: [PATCH] Optimized dropdown animation. Dropdown animation for EmojiPan done. --- Telegram/SourceFiles/app.cpp | 88 +- Telegram/SourceFiles/app.h | 39 +- Telegram/SourceFiles/core/basic_types.h | 1 + Telegram/SourceFiles/core/build_config.h | 82 ++ .../SourceFiles/media/media_clip_reader.cpp | 7 +- Telegram/SourceFiles/stickers/emoji_pan.cpp | 922 ++++++++++++------ Telegram/SourceFiles/stickers/emoji_pan.h | 65 +- Telegram/SourceFiles/stickers/stickers.style | 6 + Telegram/SourceFiles/ui/animation.h | 184 +++- .../ui/effects/panel_animation.cpp | 680 ++++++------- .../SourceFiles/ui/effects/panel_animation.h | 95 +- Telegram/SourceFiles/ui/style/style_core.cpp | 10 +- .../SourceFiles/ui/widgets/inner_dropdown.cpp | 10 +- .../SourceFiles/ui/widgets/popup_menu.cpp | 4 +- Telegram/gyp/Telegram.gyp | 1 + 15 files changed, 1407 insertions(+), 787 deletions(-) create mode 100644 Telegram/SourceFiles/core/build_config.h diff --git a/Telegram/SourceFiles/app.cpp b/Telegram/SourceFiles/app.cpp index c62caffd5..982c51c90 100644 --- a/Telegram/SourceFiles/app.cpp +++ b/Telegram/SourceFiles/app.cpp @@ -2732,40 +2732,74 @@ namespace { } return ::cornersMaskSmall; } - void roundRect(Painter &p, int32 x, int32 y, int32 w, int32 h, const style::color &bg, const CornersPixmaps &c, const style::color *sh) { - int32 cw = c.p[0]->width() / cIntRetinaFactor(), ch = c.p[0]->height() / cIntRetinaFactor(); - if (w < 2 * cw || h < 2 * ch) return; - if (w > 2 * cw) { - p.fillRect(QRect(x + cw, y, w - 2 * cw, ch), bg->b); - p.fillRect(QRect(x + cw, y + h - ch, w - 2 * cw, ch), bg->b); - if (sh) p.fillRect(QRect(x + cw, y + h, w - 2 * cw, st::msgShadow), (*sh)->b); + void roundRect(Painter &p, int32 x, int32 y, int32 w, int32 h, const style::color &bg, const CornersPixmaps &corner, const style::color *shadow, RectParts parts) { + auto cornerWidth = corner.p[0]->width() / cIntRetinaFactor(); + auto cornerHeight = corner.p[0]->height() / cIntRetinaFactor(); + if (w < 2 * cornerWidth || h < 2 * cornerHeight) return; + if (w > 2 * cornerWidth) { + if (parts & RectPart::Top) { + p.fillRect(x + cornerWidth, y, w - 2 * cornerWidth, cornerHeight, bg); + } + if (parts & RectPart::Bottom) { + p.fillRect(x + cornerWidth, y + h - cornerHeight, w - 2 * cornerWidth, cornerHeight, bg); + if (shadow) { + p.fillRect(x + cornerWidth, y + h, w - 2 * cornerWidth, st::msgShadow, *shadow); + } + } } - if (h > 2 * ch) { - p.fillRect(QRect(x, y + ch, w, h - 2 * ch), bg->b); + if (h > 2 * cornerHeight) { + if ((parts & RectPart::NoTopBottom) == qFlags(RectPart::NoTopBottom)) { + p.fillRect(x, y + cornerHeight, w, h - 2 * cornerHeight, bg); + } else { + if (parts & RectPart::Left) { + p.fillRect(x, y + cornerHeight, cornerWidth, h - 2 * cornerHeight, bg); + } + if ((parts & RectPart::Center) && w > 2 * cornerWidth) { + p.fillRect(x + cornerWidth, y + cornerHeight, w - 2 * cornerWidth, h - 2 * cornerHeight, bg); + } + if (parts & RectPart::Right) { + p.fillRect(x + w - cornerWidth, y + cornerHeight, cornerWidth, h - 2 * cornerHeight, bg); + } + } + } + if (parts & RectPart::TopLeft) { + p.drawPixmap(x, y, *corner.p[0]); + } + if (parts & RectPart::TopRight) { + p.drawPixmap(x + w - cornerWidth, y, *corner.p[1]); + } + if (parts & RectPart::BottomLeft) { + p.drawPixmap(x, y + h - cornerHeight, *corner.p[2]); + } + if (parts & RectPart::BottomRight) { + p.drawPixmap(x + w - cornerWidth, y + h - cornerHeight, *corner.p[3]); } - p.drawPixmap(QPoint(x, y), *c.p[0]); - p.drawPixmap(QPoint(x + w - cw, y), *c.p[1]); - p.drawPixmap(QPoint(x, y + h - ch), *c.p[2]); - p.drawPixmap(QPoint(x + w - cw, y + h - ch), *c.p[3]); } - void roundRect(Painter &p, int32 x, int32 y, int32 w, int32 h, const style::color &bg, RoundCorners index, const style::color *sh) { - roundRect(p, x, y, w, h, bg, ::corners[index], sh); + void roundRect(Painter &p, int32 x, int32 y, int32 w, int32 h, const style::color &bg, RoundCorners index, const style::color *shadow, RectParts parts) { + roundRect(p, x, y, w, h, bg, ::corners[index], shadow, parts); } - void roundShadow(Painter &p, int32 x, int32 y, int32 w, int32 h, const style::color &sh, RoundCorners index) { - const CornersPixmaps &c = ::corners[index]; - int32 cw = c.p[0]->width() / cIntRetinaFactor(), ch = c.p[0]->height() / cIntRetinaFactor(); - p.fillRect(x + cw, y + h, w - 2 * cw, st::msgShadow, sh->b); - p.fillRect(x, y + h - ch, cw, st::msgShadow, sh->b); - p.fillRect(x + w - cw, y + h - ch, cw, st::msgShadow, sh->b); - p.drawPixmap(x, y + h - ch + st::msgShadow, *c.p[2]); - p.drawPixmap(x + w - cw, y + h - ch + st::msgShadow, *c.p[3]); + void roundShadow(Painter &p, int32 x, int32 y, int32 w, int32 h, const style::color &shadow, RoundCorners index, RectParts parts) { + auto &corner = ::corners[index]; + auto cornerWidth = corner.p[0]->width() / cIntRetinaFactor(); + auto cornerHeight = corner.p[0]->height() / cIntRetinaFactor(); + if (parts & RectPart::Bottom) { + p.fillRect(x + cornerWidth, y + h, w - 2 * cornerWidth, st::msgShadow, shadow); + } + if (parts & RectPart::BottomLeft) { + p.fillRect(x, y + h - cornerHeight, cornerWidth, st::msgShadow, shadow); + p.drawPixmap(x, y + h - cornerHeight + st::msgShadow, *corner.p[2]); + } + if (parts & RectPart::BottomRight) { + p.fillRect(x + w - cornerWidth, y + h - cornerHeight, cornerWidth, st::msgShadow, shadow); + p.drawPixmap(x + w - cornerWidth, y + h - cornerHeight + st::msgShadow, *corner.p[3]); + } } - void roundRect(Painter &p, int32 x, int32 y, int32 w, int32 h, const style::color &bg, ImageRoundRadius radius) { - uint32 colorKey = ((uint32(bg->c.alpha()) & 0xFF) << 24) | ((uint32(bg->c.red()) & 0xFF) << 16) | ((uint32(bg->c.green()) & 0xFF) << 8) | ((uint32(bg->c.blue()) & 0xFF) << 24); - CornersMap::const_iterator i = cornersMap.find(colorKey); + void roundRect(Painter &p, int32 x, int32 y, int32 w, int32 h, const style::color &bg, ImageRoundRadius radius, RectParts parts) { + auto colorKey = ((uint32(bg->c.alpha()) & 0xFF) << 24) | ((uint32(bg->c.red()) & 0xFF) << 16) | ((uint32(bg->c.green()) & 0xFF) << 8) | ((uint32(bg->c.blue()) & 0xFF) << 24); + auto i = cornersMap.find(colorKey); if (i == cornersMap.cend()) { QImage images[4]; switch (radius) { @@ -2781,7 +2815,7 @@ namespace { } i = cornersMap.insert(colorKey, pixmaps); } - roundRect(p, x, y, w, h, bg, i.value(), 0); + roundRect(p, x, y, w, h, bg, i.value(), nullptr, parts); } WallPapers gServerBackgrounds; diff --git a/Telegram/SourceFiles/app.h b/Telegram/SourceFiles/app.h index 2e14e7d94..f80783141 100644 --- a/Telegram/SourceFiles/app.h +++ b/Telegram/SourceFiles/app.h @@ -279,18 +279,39 @@ namespace App { #endif // !TDESKTOP_DISABLE_NETWORK_PROXY void setProxySettings(QTcpSocket &socket); + enum class RectPart { + TopLeft = 0x001, + Top = 0x002, + TopRight = 0x004, + Left = 0x008, + Center = 0x010, + Right = 0x020, + BottomLeft = 0x040, + Bottom = 0x080, + BottomRight = 0x100, + TopFull = 0x007, + LeftFull = 0x049, + RightFull = 0x124, + BottomFull = 0x1c0, + NoTopBottom = 0x038, + NoLeftRight = 0x092, + Full = 0x1ff, + }; + Q_DECLARE_FLAGS(RectParts, RectPart); + Q_DECLARE_OPERATORS_FOR_FLAGS(RectParts); + QImage **cornersMask(ImageRoundRadius radius); - void roundRect(Painter &p, int32 x, int32 y, int32 w, int32 h, const style::color &bg, RoundCorners index, const style::color *sh = 0); - inline void roundRect(Painter &p, const QRect &rect, const style::color &bg, RoundCorners index, const style::color *sh = 0) { - return roundRect(p, rect.x(), rect.y(), rect.width(), rect.height(), bg, index, sh); + void roundRect(Painter &p, int32 x, int32 y, int32 w, int32 h, const style::color &bg, RoundCorners index, const style::color *shadow = nullptr, RectParts parts = RectPart::Full); + inline void roundRect(Painter &p, const QRect &rect, const style::color &bg, RoundCorners index, const style::color *shadow = nullptr, RectParts parts = RectPart::Full) { + return roundRect(p, rect.x(), rect.y(), rect.width(), rect.height(), bg, index, shadow, parts); } - void roundShadow(Painter &p, int32 x, int32 y, int32 w, int32 h, const style::color &sh, RoundCorners index); - inline void roundShadow(Painter &p, const QRect &rect, const style::color &sh, RoundCorners index) { - return roundShadow(p, rect.x(), rect.y(), rect.width(), rect.height(), sh, index); + void roundShadow(Painter &p, int32 x, int32 y, int32 w, int32 h, const style::color &shadow, RoundCorners index, RectParts parts = RectPart::Full); + inline void roundShadow(Painter &p, const QRect &rect, const style::color &shadow, RoundCorners index, RectParts parts = RectPart::Full) { + return roundShadow(p, rect.x(), rect.y(), rect.width(), rect.height(), shadow, index, parts); } - void roundRect(Painter &p, int32 x, int32 y, int32 w, int32 h, const style::color &bg, ImageRoundRadius radius); - inline void roundRect(Painter &p, const QRect &rect, const style::color &bg, ImageRoundRadius radius) { - return roundRect(p, rect.x(), rect.y(), rect.width(), rect.height(), bg, radius); + void roundRect(Painter &p, int32 x, int32 y, int32 w, int32 h, const style::color &bg, ImageRoundRadius radius, RectParts parts = RectPart::Full); + inline void roundRect(Painter &p, const QRect &rect, const style::color &bg, ImageRoundRadius radius, RectParts parts = RectPart::Full) { + return roundRect(p, rect.x(), rect.y(), rect.width(), rect.height(), bg, radius, parts); } struct WallPaper { diff --git a/Telegram/SourceFiles/core/basic_types.h b/Telegram/SourceFiles/core/basic_types.h index 9487daec9..9c368655b 100644 --- a/Telegram/SourceFiles/core/basic_types.h +++ b/Telegram/SourceFiles/core/basic_types.h @@ -26,6 +26,7 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org #include +#include "core/build_config.h" #include "core/stl_subset.h" #include "core/ordered_set.h" diff --git a/Telegram/SourceFiles/core/build_config.h b/Telegram/SourceFiles/core/build_config.h new file mode 100644 index 000000000..49c89041e --- /dev/null +++ b/Telegram/SourceFiles/core/build_config.h @@ -0,0 +1,82 @@ +/* +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 + +// thanks Chromium + +#if defined(__APPLE__) +#define OS_MAC 1 +#elif defined(__linux__) // __APPLE__ +#define OS_LINUX 1 +#elif defined(_WIN32) // __APPLE__ || __linux__ +#define OS_WIN 1 +#else // __APPLE__ || __linux__ || _WIN32 +#error Please add support for your platform in core/build_config.h +#endif // else for __APPLE__ || __linux__ || _WIN32 + +// For access to standard POSIXish features, use OS_POSIX instead of a +// more specific macro. +#if defined(OS_MAC) || defined(OS_LINUX) +#define OS_POSIX 1 +#endif // OS_MAC || OS_LINUX + +// Compiler detection. +#if defined(__clang__) +#define COMPILER_CLANG 1 +#elif defined(__GNUC__) // __clang__ +#define COMPILER_GCC 1 +#elif defined(_MSC_VER) // __clang__ || __GNUC__ +#define COMPILER_MSVC 1 +#else // _MSC_VER || __clang__ || __GNUC__ +#error Please add support for your compiler in core/build_config.h +#endif // else for _MSC_VER || __clang__ || __GNUC__ + +// Processor architecture detection. +#if defined(_M_X64) || defined(__x86_64__) +#define ARCH_CPU_X86_FAMILY 1 +#define ARCH_CPU_X86_64 1 +#define ARCH_CPU_64_BITS 1 +#elif defined(_M_IX86) || defined(__i386__) +#define ARCH_CPU_X86_FAMILY 1 +#define ARCH_CPU_X86 1 +#define ARCH_CPU_32_BITS 1 +#else +#error Please add support for your architecture in core/build_config.h +#endif + +#if defined(COMPILER_GCC) || defined(COMPILER_CLANG) +#define WARN_UNUSED_RESULT __attribute__((warn_unused_result)) +#else +#define WARN_UNUSED_RESULT +#endif + +#if defined(__GNUC__) +#define FORCE_INLINE inline __attribute__((always_inline)) +#elif defined(_MSC_VER) +#define FORCE_INLINE __forceinline +#else +#define FORCE_INLINE inline +#endif + +#include +static_assert(CHAR_BIT == 8, "Not supported char size."); diff --git a/Telegram/SourceFiles/media/media_clip_reader.cpp b/Telegram/SourceFiles/media/media_clip_reader.cpp index 2c45b2084..7e965c8ec 100644 --- a/Telegram/SourceFiles/media/media_clip_reader.cpp +++ b/Telegram/SourceFiles/media/media_clip_reader.cpp @@ -208,7 +208,8 @@ QPixmap Reader::current(int32 framew, int32 frameh, int32 outerw, int32 outerh, auto frame = frameToShow(); t_assert(frame != nullptr); - if (ms) { + auto shouldBePaused = !ms; + if (!shouldBePaused) { frame->displayed.storeRelease(1); if (_autoPausedGif.loadAcquire()) { _autoPausedGif.storeRelease(0); @@ -218,10 +219,10 @@ QPixmap Reader::current(int32 framew, int32 frameh, int32 outerw, int32 outerh, } } } else { - frame->displayed.storeRelease(-1); // displayed, but should be paused + frame->displayed.storeRelease(-1); } - int32 factor(cIntRetinaFactor()); + auto factor = cIntRetinaFactor(); if (frame->pix.width() == outerw * factor && frame->pix.height() == outerh * factor) { moveToNextShow(); return frame->pix; diff --git a/Telegram/SourceFiles/stickers/emoji_pan.cpp b/Telegram/SourceFiles/stickers/emoji_pan.cpp index 3413ddee3..faa8bf959 100644 --- a/Telegram/SourceFiles/stickers/emoji_pan.cpp +++ b/Telegram/SourceFiles/stickers/emoji_pan.cpp @@ -22,8 +22,8 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org #include "stickers/emoji_pan.h" #include "styles/style_stickers.h" -#include "styles/style_intro.h" #include "ui/widgets/buttons.h" +#include "ui/widgets/shadow.h" #include "boxes/confirmbox.h" #include "boxes/stickersetbox.h" #include "boxes/stickers_box.h" @@ -103,8 +103,8 @@ void EmojiColorPicker::paintEvent(QPaintEvent *e) { } } else { p.drawPixmap(r.left(), r.top(), _cache); + p.setOpacity(1.); } - } void EmojiColorPicker::enterEvent(QEvent *e) { @@ -286,7 +286,7 @@ void EmojiColorPicker::drawVariant(Painter &p, int variant) { QPoint tl(w); if (rtl()) tl.setX(width() - tl.x() - st::emojiPanSize.width()); App::roundRect(p, QRect(tl, st::emojiPanSize), st::emojiPanHover, StickerHoverCorners); - p.setOpacity(1); + p.setOpacity(1.); } int esize = EmojiSizes[EIndex + 1]; p.drawPixmapLeft(w.x() + (st::emojiPanSize.width() - (esize / cIntRetinaFactor())) / 2, w.y() + (st::emojiPanSize.height() - (esize / cIntRetinaFactor())) / 2, width(), App::emojiLarge(), QRect(_variants[variant]->x * esize, _variants[variant]->y * esize, esize, esize)); @@ -295,7 +295,7 @@ void EmojiColorPicker::drawVariant(Painter &p, int variant) { EmojiPanInner::EmojiPanInner() : TWidget() , _maxHeight(int(st::emojiPanMaxHeight) - st::emojiCategory.height) , _a_selected(animation(this, &EmojiPanInner::step_selected)) { - resize(st::emojiPanWidth - st::emojiScroll.width, countHeight()); + resize(st::emojiPanWidth - st::emojiScroll.width - st::buttonRadius, countHeight()); setMouseTracking(true); setAttribute(Qt::WA_OpaquePaintEvent); @@ -394,7 +394,7 @@ void EmojiPanInner::paintEvent(QPaintEvent *e) { QPoint tl(w); if (rtl()) tl.setX(width() - tl.x() - st::emojiPanSize.width()); App::roundRect(p, QRect(tl, st::emojiPanSize), st::emojiPanHover, StickerHoverCorners); - p.setOpacity(1); + p.setOpacity(1.); } p.drawPixmapLeft(w.x() + (st::emojiPanSize.width() - (_esize / cIntRetinaFactor())) / 2, w.y() + (st::emojiPanSize.height() - (_esize / cIntRetinaFactor())) / 2, width(), App::emojiLarge(), QRect(_emojis[c][index]->x * _esize, _emojis[c][index]->y * _esize, _esize, _esize)); } @@ -820,7 +820,7 @@ StickerPanInner::StickerPanInner() : TWidget() void StickerPanInner::setMaxHeight(int32 h) { _maxHeight = h; - resize(st::emojiPanWidth - st::emojiScroll.width, countHeight()); + resize(st::emojiPanWidth - st::emojiScroll.width - st::buttonRadius, countHeight()); _settings->moveToLeft((st::emojiPanWidth - _settings->width()) / 2, height() / 3); } @@ -911,18 +911,22 @@ StickerPanInner::~StickerPanInner() { deleteUnusedInlineLayouts(); } +int StickerPanInner::stickersLeft() const { + return (st::stickerPanPadding - st::buttonRadius); +} + QRect StickerPanInner::stickerRect(int tab, int sel) { int x = 0, y = 0; if (_section == Section::Featured) { y += st::emojiPanHeader + (tab * featuredRowHeight()) + st::stickersTrendingHeader; - x = st::stickerPanPadding + (sel * st::stickerPanSize.width()); + x = stickersLeft() + (sel * st::stickerPanSize.width()); } else { auto &sets = shownSets(); for (int i = 0; i < sets.size(); ++i) { if (i == tab) { int rows = (((sel >= sets[i].pack.size()) ? (sel - sets[i].pack.size()) : sel) / StickerPanPerRow); y += st::emojiPanHeader + rows * st::stickerPanSize.height(); - x = st::stickerPanPadding + ((sel % StickerPanPerRow) * st::stickerPanSize.width()); + x = stickersLeft() + ((sel % StickerPanPerRow) * st::stickerPanSize.width()); break; } else { int cnt = sets[i].pack.size(); @@ -968,7 +972,7 @@ void StickerPanInner::paintInlineItems(Painter &p, const QRect &r) { auto &inlineRow = _inlineRows[row]; if (top >= r.top() + r.height()) break; if (top + inlineRow.height > r.top()) { - int left = st::inlineResultsLeft; + int left = st::inlineResultsLeft - st::buttonRadius; if (row == rows - 1) context.lastRow = true; for (int col = 0, cols = inlineRow.items.size(); col < cols; ++col) { if (left >= tox) break; @@ -991,8 +995,8 @@ void StickerPanInner::paintInlineItems(Painter &p, const QRect &r) { } void StickerPanInner::paintStickers(Painter &p, const QRect &r) { - int32 fromcol = floorclamp(r.x() - st::stickerPanPadding, st::stickerPanSize.width(), 0, StickerPanPerRow); - int32 tocol = ceilclamp(r.x() + r.width() - st::stickerPanPadding, st::stickerPanSize.width(), 0, StickerPanPerRow); + int32 fromcol = floorclamp(r.x() - stickersLeft(), st::stickerPanSize.width(), 0, StickerPanPerRow); + int32 tocol = ceilclamp(r.x() + r.width() - stickersLeft(), st::stickerPanSize.width(), 0, StickerPanPerRow); if (rtl()) { qSwap(fromcol, tocol); fromcol = StickerPanPerRow - fromcol; @@ -1013,7 +1017,7 @@ void StickerPanInner::paintStickers(Painter &p, const QRect &r) { int size = set.pack.size(); - int widthForTitle = featuredContentWidth() - st::emojiPanHeaderLeft; + int widthForTitle = featuredContentWidth() - (st::emojiPanHeaderLeft - st::buttonRadius); if (featuredHasAddButton(c)) { auto add = featuredAddRect(c); auto selected = (_selectedFeaturedSetAdd == c); @@ -1044,20 +1048,20 @@ void StickerPanInner::paintStickers(Painter &p, const QRect &r) { } p.setFont(st::stickersTrendingHeaderFont); p.setPen(st::stickersTrendingHeaderFg); - p.drawTextLeft(st::emojiPanHeaderLeft, y + st::stickersTrendingHeaderTop, width(), titleText, titleWidth); + p.drawTextLeft(st::emojiPanHeaderLeft - st::buttonRadius, y + st::stickersTrendingHeaderTop, width(), titleText, titleWidth); if (set.flags & MTPDstickerSet_ClientFlag::f_unread) { p.setPen(Qt::NoPen); p.setBrush(st::stickersFeaturedUnreadBg); p.setRenderHint(QPainter::HighQualityAntialiasing, true); - p.drawEllipse(rtlrect(st::emojiPanHeaderLeft + titleWidth + st::stickersFeaturedUnreadSkip, y + st::stickersTrendingHeaderTop + st::stickersFeaturedUnreadTop, st::stickersFeaturedUnreadSize, st::stickersFeaturedUnreadSize, width())); + p.drawEllipse(rtlrect(st::emojiPanHeaderLeft - st::buttonRadius + titleWidth + st::stickersFeaturedUnreadSkip, y + st::stickersTrendingHeaderTop + st::stickersFeaturedUnreadTop, st::stickersFeaturedUnreadSize, st::stickersFeaturedUnreadSize, width())); p.setRenderHint(QPainter::HighQualityAntialiasing, false); } p.setFont(st::stickersTrendingSubheaderFont); p.setPen(st::stickersTrendingSubheaderFg); - p.drawTextLeft(st::emojiPanHeaderLeft, y + st::stickersTrendingSubheaderTop, width(), lng_stickers_count(lt_count, size)); + p.drawTextLeft(st::emojiPanHeaderLeft - st::buttonRadius, y + st::stickersTrendingSubheaderTop, width(), lng_stickers_count(lt_count, size)); y += st::stickersTrendingHeader; if (y >= r.y() + r.height()) break; @@ -1104,13 +1108,13 @@ void StickerPanInner::paintSticker(Painter &p, Set &set, int y, int index) { int row = (index / StickerPanPerRow), col = (index % StickerPanPerRow); - QPoint pos(st::stickerPanPadding + col * st::stickerPanSize.width(), y + row * st::stickerPanSize.height()); + QPoint pos(stickersLeft() + col * st::stickerPanSize.width(), y + row * st::stickerPanSize.height()); if (hover > 0) { p.setOpacity(hover); QPoint tl(pos); if (rtl()) tl.setX(width() - tl.x() - st::stickerPanSize.width()); App::roundRect(p, QRect(tl, st::stickerPanSize), st::emojiPanHover, StickerHoverCorners); - p.setOpacity(1); + p.setOpacity(1.); } bool goodThumb = !sticker->thumb->isNull() && ((sticker->thumb->width() >= 128) || (sticker->thumb->height() >= 128)); @@ -1138,7 +1142,7 @@ void StickerPanInner::paintSticker(Painter &p, Set &set, int y, int index) { QPoint xPos = pos + QPoint(st::stickerPanSize.width() - st::stickerPanDelete.width(), 0); p.setOpacity(hover * (xHover + (1 - xHover) * st::stickerPanDeleteOpacity)); st::stickerPanDelete.paint(p, xPos, width()); - p.setOpacity(1); + p.setOpacity(1.); } } @@ -1151,7 +1155,7 @@ bool StickerPanInner::featuredHasAddButton(int index) const { } int StickerPanInner::featuredContentWidth() const { - return st::stickerPanPadding + (StickerPanPerRow * st::stickerPanSize.width()); + return stickersLeft() + (StickerPanPerRow * st::stickerPanSize.width()); } QRect StickerPanInner::featuredAddRect(int index) const { @@ -1634,7 +1638,7 @@ StickerPanInner::InlineRow &StickerPanInner::layoutInlineRow(InlineRow &row, int }); row.height = 0; - int availw = width() - st::inlineResultsLeft; + int availw = width() - (st::inlineResultsLeft - st::buttonRadius); for (int i = 0; i < count; ++i) { int index = indices[i]; int w = sumWidth ? (row.items.at(index)->maxWidth() * availw / sumWidth) : row.items.at(index)->maxWidth(); @@ -1730,7 +1734,7 @@ void StickerPanInner::refreshSwitchPmButton(const InlineCacheEntry *entry) { if (!_switchPmButton) { _switchPmButton = std_::make_unique(this, QString(), st::switchPmButton); _switchPmButton->show(); - _switchPmButton->move(st::inlineResultsLeft, st::emojiPanHeader); + _switchPmButton->move(st::inlineResultsLeft - st::buttonRadius, st::emojiPanHeader); connect(_switchPmButton.get(), SIGNAL(clicked()), this, SLOT(onSwitchPm())); } _switchPmButton->setText(entry->switchPmText); // doesn't perform text.toUpper() @@ -2082,7 +2086,7 @@ void StickerPanInner::updateSelected() { auto p = mapFromGlobal(_lastMousePos); if (showingInlineItems()) { - int sx = (rtl() ? width() - p.x() : p.x()) - st::inlineResultsLeft; + int sx = (rtl() ? width() - p.x() : p.x()) - (st::inlineResultsLeft - st::buttonRadius); int sy = p.y() - st::emojiPanHeader; if (_switchPmButton) { sy -= _switchPmButton->height() + st::inlineResultsSkip; @@ -2157,7 +2161,7 @@ void StickerPanInner::updateSelected() { int selectedFeaturedSetAdd = -1; auto featured = (_section == Section::Featured); auto &sets = shownSets(); - int y, ytill = 0, sx = (rtl() ? width() - p.x() : p.x()) - st::stickerPanPadding; + int y, ytill = 0, sx = (rtl() ? width() - p.x() : p.x()) - stickersLeft(); if (featured) { ytill += st::emojiPanHeader; } @@ -2426,12 +2430,12 @@ EmojiPanel::EmojiPanel(QWidget *parent, const QString &text, uint64 setId, bool , _special(special) , _deleteVisible(false) , _delete(special ? 0 : new Ui::IconButton(this, st::hashtagClose)) { // Stickers::NoneSetId if in emoji - resize(st::emojiPanWidth, st::emojiPanHeader); + resize(st::emojiPanWidth - 2 * st::buttonRadius, st::emojiPanHeader); setMouseTracking(true); setText(text); if (_delete) { _delete->hide(); - _delete->moveToRight(st::emojiPanHeaderLeft - ((_delete->width() - st::hashtagClose.icon.width()) / 2), (st::emojiPanHeader - _delete->height()) / 2, width()); + _delete->moveToRight(st::emojiPanHeaderLeft - ((_delete->width() - st::hashtagClose.icon.width()) / 2) - st::buttonRadius, (st::emojiPanHeader - _delete->height()) / 2, width()); connect(_delete, SIGNAL(clicked()), this, SLOT(onDelete())); } } @@ -2489,7 +2493,7 @@ void EmojiPanel::paintEvent(QPaintEvent *e) { } p.setFont(st::emojiPanHeaderFont); p.setPen(st::emojiPanHeaderFg); - p.drawTextLeft(st::emojiPanHeaderLeft, st::emojiPanHeaderTop, width(), _text); + p.drawTextLeft(st::emojiPanHeaderLeft - st::buttonRadius, st::emojiPanHeaderTop, width(), _text); } EmojiSwitchButton::EmojiSwitchButton(QWidget *parent, bool toStickers) : AbstractButton(parent) @@ -2512,7 +2516,7 @@ void EmojiSwitchButton::updateText(const QString &inlineBotUsername) { if (_toStickers && !inlineBotUsername.isEmpty()) { int32 maxw = 0; for (int c = 0; c < emojiTabCount; ++c) { - maxw = qMax(maxw, st::emojiPanHeaderFont->width(lang(LangKey(lng_emoji_category0 + c)))); + accumulate_max(maxw, st::emojiPanHeaderFont->width(lang(LangKey(lng_emoji_category0 + c)))); } maxw += st::emojiPanHeaderLeft + st::emojiSwitchSkip + (st::emojiSwitchSkip - st::emojiSwitchImgSkip); if (_textWidth > st::emojiPanWidth - maxw) { @@ -2521,7 +2525,7 @@ void EmojiSwitchButton::updateText(const QString &inlineBotUsername) { } } - int32 w = st::emojiSwitchSkip + _textWidth + (st::emojiSwitchSkip - st::emojiSwitchImgSkip); + int32 w = st::emojiSwitchSkip + _textWidth + (st::emojiSwitchSkip - st::emojiSwitchImgSkip) - st::buttonRadius; resize(w, st::emojiPanHeader); } @@ -2541,14 +2545,264 @@ void EmojiSwitchButton::paintEvent(QPaintEvent *e) { } // namespace internal +namespace { + +FORCE_INLINE uint32 twoImagesOnBgWithAlpha( + const anim::Shifted shiftedBg, + const uint32 source1Alpha, + const uint32 source2Alpha, + const uint32 source1, + const uint32 source2, + const uint32 alpha) { + auto source1Pattern = anim::reshifted(anim::shifted(source1) * source1Alpha); + auto bg1Alpha = 256 - anim::getAlpha(source1Pattern); + auto mixed1Pattern = anim::reshifted(shiftedBg * bg1Alpha) + source1Pattern; + auto source2Pattern = anim::reshifted(anim::shifted(source2) * source2Alpha); + auto bg2Alpha = 256 - anim::getAlpha(source2Pattern); + auto mixed2Pattern = anim::reshifted(mixed1Pattern * bg2Alpha) + source2Pattern; + return anim::unshifted(mixed2Pattern * alpha); +} + +FORCE_INLINE uint32 oneImageOnBgWithAlpha( + const anim::Shifted shiftedBg, + const uint32 sourceAlpha, + const uint32 source, + const uint32 alpha) { + auto sourcePattern = anim::reshifted(anim::shifted(source) * sourceAlpha); + auto bgAlpha = 256 - anim::getAlpha(sourcePattern); + auto mixedPattern = anim::reshifted(shiftedBg * bgAlpha) + sourcePattern; + return anim::unshifted(mixedPattern * alpha); +}; + +} // namespace + +class EmojiPan::SlideAnimation : public Ui::RoundShadowAnimation { +public: + enum class Direction { + LeftToRight, + RightToLeft, + }; + void setFinalImages(Direction direction, QImage &&left, QImage &&right, QRect inner); + + void start(); + void paintFrame(QPainter &p, float64 dt, float64 opacity); + +private: + Direction _direction = Direction::LeftToRight; + QPixmap _leftImage, _rightImage; + int _width = 0; + int _height = 0; + int _innerLeft = 0; + int _innerTop = 0; + int _innerRight = 0; + int _innerBottom = 0; + int _innerWidth = 0; + int _innerHeight = 0; + + int _painterInnerLeft = 0; + int _painterInnerTop = 0; + int _painterInnerWidth = 0; + int _painterInnerBottom = 0; + int _painterCategoriesTop = 0; + int _painterInnerHeight = 0; + int _painterInnerRight = 0; + + int _frameIntsPerLineAdd = 0; + +}; + +void EmojiPan::SlideAnimation::setFinalImages(Direction direction, QImage &&left, QImage &&right, QRect inner) { + t_assert(!started()); + _direction = direction; + _leftImage = QPixmap::fromImage(std_::move(left).convertToFormat(QImage::Format_ARGB32_Premultiplied), Qt::ColorOnly); + _rightImage = QPixmap::fromImage(std_::move(right).convertToFormat(QImage::Format_ARGB32_Premultiplied), Qt::ColorOnly); + + t_assert(!_leftImage.isNull()); + t_assert(!_rightImage.isNull()); + _width = _leftImage.width(); + _height = _rightImage.height(); + t_assert(!(_width % cIntRetinaFactor())); + t_assert(!(_height % cIntRetinaFactor())); + t_assert(_leftImage.devicePixelRatio() == _rightImage.devicePixelRatio()); + t_assert(_rightImage.width() == _width); + t_assert(_rightImage.height() == _height); + t_assert(QRect(0, 0, _width, _height).contains(inner)); + _innerLeft = inner.x(); + _innerTop = inner.y(); + _innerWidth = inner.width(); + _innerHeight = inner.height(); + t_assert(!(_innerLeft % cIntRetinaFactor())); + t_assert(!(_innerTop % cIntRetinaFactor())); + t_assert(!(_innerWidth % cIntRetinaFactor())); + t_assert(!(_innerHeight % cIntRetinaFactor())); + _innerRight = _innerLeft + _innerWidth; + _innerBottom = _innerTop + _innerHeight; + + _painterInnerLeft = _innerLeft / cIntRetinaFactor(); + _painterInnerTop = _innerTop / cIntRetinaFactor(); + _painterInnerRight = _innerRight / cIntRetinaFactor(); + _painterInnerBottom = _innerBottom / cIntRetinaFactor(); + _painterInnerWidth = _innerWidth / cIntRetinaFactor(); + _painterInnerHeight = _innerHeight / cIntRetinaFactor(); + _painterCategoriesTop = _painterInnerBottom - st::emojiCategory.height; + + setShadow(st::emojiPanAnimation.shadow); +} + +void EmojiPan::SlideAnimation::start() { + t_assert(!_leftImage.isNull()); + t_assert(!_rightImage.isNull()); + RoundShadowAnimation::start(_width, _height, _leftImage.devicePixelRatio()); + auto checkCorner = [this](const Corner &corner) { + if (!corner.valid()) return; + t_assert(corner.width <= _innerWidth); + t_assert(corner.height <= _innerHeight); + }; + checkCorner(_topLeft); + checkCorner(_topRight); + checkCorner(_bottomLeft); + checkCorner(_bottomRight); + _frameIntsPerLineAdd = (_width - _innerWidth) + _frameIntsPerLineAdded; +} + +void EmojiPan::SlideAnimation::paintFrame(QPainter &p, float64 dt, float64 opacity) { + t_assert(started()); + t_assert(dt >= 0.); + + auto &transition = anim::easeOutCirc; + _frameAlpha = anim::interpolate(1, 256, opacity); + + auto frameInts = _frameInts + _innerLeft + _innerTop * _frameIntsPerLine; + + auto leftToRight = (_direction == Direction::LeftToRight); + + auto easeOut = anim::easeOutCirc(1., dt); + auto easeIn = anim::easeInCirc(1., dt); + + auto arrivingCoord = anim::interpolate(_innerWidth, 0, easeOut); + auto departingCoord = anim::interpolate(0, _innerWidth, easeIn); + if (auto decrease = (arrivingCoord % cIntRetinaFactor())) { + arrivingCoord -= decrease; + } + if (auto decrease = (departingCoord % cIntRetinaFactor())) { + departingCoord -= decrease; + } + auto arrivingAlpha = easeIn; + auto departingAlpha = 1. - easeOut; + auto leftCoord = (leftToRight ? arrivingCoord : departingCoord) * -1; + auto leftAlpha = (leftToRight ? arrivingAlpha : departingAlpha); + auto rightCoord = (leftToRight ? departingCoord : arrivingCoord); + auto rightAlpha = (leftToRight ? departingAlpha : arrivingAlpha); + + // _innerLeft ..(left).. leftTo ..(both).. bothTo ..(none).. noneTo ..(right).. _innerRight + auto leftTo = _innerLeft + snap(_innerWidth + leftCoord, 0, _innerWidth); + auto rightFrom = _innerLeft + snap(rightCoord, 0, _innerWidth); + auto painterRightFrom = rightFrom / cIntRetinaFactor(); + if (opacity < 1.) { + _frame.fill(Qt::transparent); + } + { + Painter p(&_frame); + p.setOpacity(opacity); + p.fillRect(_painterInnerLeft, _painterInnerTop, _painterInnerWidth, _painterCategoriesTop - _painterInnerTop, st::emojiPanBg); + p.fillRect(_painterInnerLeft, _painterCategoriesTop, _painterInnerWidth, _painterInnerBottom - _painterCategoriesTop, st::emojiPanCategories); + p.setCompositionMode(QPainter::CompositionMode_SourceOver); + if (leftTo > _innerLeft) { + p.setOpacity(opacity * leftAlpha); + p.drawPixmap(_painterInnerLeft, _painterInnerTop, _leftImage, _innerLeft - leftCoord, _innerTop, leftTo - _innerLeft, _innerHeight); + } + if (rightFrom < _innerRight) { + p.setOpacity(opacity * rightAlpha); + p.drawPixmap(painterRightFrom, _painterInnerTop, _rightImage, _innerLeft, _innerTop, _innerRight - rightFrom, _innerHeight); + } + } + + // Draw corners + paintCorner(_topLeft, _innerLeft, _innerTop); + paintCorner(_topRight, _innerRight - _topRight.width, _innerTop); + paintCorner(_bottomLeft, _innerLeft, _innerBottom - _bottomLeft.height); + paintCorner(_bottomRight, _innerRight - _bottomRight.width, _innerBottom - _bottomRight.height); + + // Draw shadow upon the transparent + auto outerLeft = _innerLeft; + auto outerTop = _innerTop; + auto outerRight = _innerRight; + auto outerBottom = _innerBottom; + if (_shadow.valid()) { + outerLeft -= _shadow.extend.left(); + outerTop -= _shadow.extend.top(); + outerRight += _shadow.extend.right(); + outerBottom += _shadow.extend.bottom(); + } + if (cIntRetinaFactor() > 1) { + if (auto skipLeft = (outerLeft % cIntRetinaFactor())) { + outerLeft -= skipLeft; + } + if (auto skipTop = (outerTop % cIntRetinaFactor())) { + outerTop -= skipTop; + } + if (auto skipRight = (outerRight % cIntRetinaFactor())) { + outerRight += (cIntRetinaFactor() - skipRight); + } + if (auto skipBottom = (outerBottom % cIntRetinaFactor())) { + outerBottom += (cIntRetinaFactor() - skipBottom); + } + } + + if (opacity == 1.) { + // Fill above the frame top with transparent. + auto fillTopInts = (_frameInts + outerTop * _frameIntsPerLine + outerLeft); + auto fillWidth = (outerRight - outerLeft) * sizeof(uint32); + for (auto fillTop = _innerTop - outerTop; fillTop != 0; --fillTop) { + memset(fillTopInts, 0, fillWidth); + fillTopInts += _frameIntsPerLine; + } + + // Fill to the left and to the right of the frame with transparent. + auto fillLeft = (_innerLeft - outerLeft) * sizeof(uint32); + auto fillRight = (outerRight - _innerRight) * sizeof(uint32); + if (fillLeft || fillRight) { + auto fillInts = _frameInts + _innerTop * _frameIntsPerLine; + for (auto y = _innerTop; y != _innerBottom; ++y) { + memset(fillInts + outerLeft, 0, fillLeft); + memset(fillInts + _innerRight, 0, fillRight); + fillInts += _frameIntsPerLine; + } + } + + // Fill below the frame bottom with transparent. + auto fillBottomInts = (_frameInts + _innerBottom * _frameIntsPerLine + outerLeft); + for (auto fillBottom = outerBottom - _innerBottom; fillBottom != 0; --fillBottom) { + memset(fillBottomInts, 0, fillWidth); + fillBottomInts += _frameIntsPerLine; + } + } + if (_shadow.valid()) { + paintShadow(outerLeft, outerTop, outerRight, outerBottom); + } + + // Debug + //frameInts = _frameInts; + //auto pattern = anim::shifted((static_cast(0xFF) << 24) | (static_cast(0xFF) << 16) | (static_cast(0xFF) << 8) | static_cast(0xFF)); + //for (auto y = 0; y != _finalHeight; ++y) { + // for (auto x = 0; x != _finalWidth; ++x) { + // auto source = *frameInts; + // auto sourceAlpha = (source >> 24); + // *frameInts = anim::unshifted(anim::shifted(source) * 256 + pattern * (256 - sourceAlpha)); + // ++frameInts; + // } + // frameInts += _frameIntsPerLineAdded; + //} + + p.drawImage(outerLeft / cIntRetinaFactor(), outerTop / cIntRetinaFactor(), _frame, outerLeft, outerTop, outerRight - outerLeft, outerBottom - outerTop); +} + EmojiPan::EmojiPan(QWidget *parent) : TWidget(parent) , _maxHeight(st::emojiPanMaxHeight) , _contentMaxHeight(st::emojiPanMaxHeight) , _contentHeight(_contentMaxHeight) , _contentHeightEmoji(_contentHeight - st::emojiCategory.height) , _contentHeightStickers(_contentHeight - st::emojiCategory.height) -, _a_appearance(animation(this, &EmojiPan::step_appearance)) -, _shadow(st::defaultDropdownShadow) , _recent(this, st::emojiCategoryRecent) , _people(this, st::emojiCategoryPeople) , _nature(this, st::emojiCategoryNature) @@ -2558,31 +2812,30 @@ EmojiPan::EmojiPan(QWidget *parent) : TWidget(parent) , _objects(this, st::emojiCategoryObjects) , _symbols(this, st::emojiCategorySymbols) , _a_icons(animation(this, &EmojiPan::step_icons)) -, _a_slide(animation(this, &EmojiPan::step_slide)) , e_scroll(this, st::emojiScroll) , e_inner() , e_switch(&e_scroll, true) , s_scroll(this, st::emojiScroll) , s_inner() , s_switch(&s_scroll, false) { - _width = st::defaultDropdownPadding.left() + st::emojiPanWidth + st::defaultDropdownPadding.right(); - _height = st::defaultDropdownPadding.top() + _contentHeight + st::defaultDropdownPadding.bottom(); + resize(QRect(0, 0, st::emojiPanWidth, _contentHeight).marginsAdded(innerPadding()).size()); + _width = width(); + _height = height(); _bottom = 0; - resize(_width, _height); - e_scroll.resize(st::emojiPanWidth, _contentHeightEmoji); - s_scroll.resize(st::emojiPanWidth, _contentHeightStickers); + e_scroll.resize(st::emojiPanWidth - st::buttonRadius, _contentHeightEmoji); + s_scroll.resize(st::emojiPanWidth - st::buttonRadius, _contentHeightStickers); - e_scroll.move(st::defaultDropdownPadding.left(), st::defaultDropdownPadding.top()); + e_scroll.move(verticalRect().topLeft()); e_scroll.setWidget(&e_inner); - s_scroll.move(st::defaultDropdownPadding.left(), st::defaultDropdownPadding.top()); + s_scroll.move(verticalRect().topLeft()); s_scroll.setWidget(&s_inner); e_inner.moveToLeft(0, 0, e_scroll.width()); s_inner.moveToLeft(0, 0, s_scroll.width()); - int32 left = _iconsLeft = st::defaultDropdownPadding.left() + (st::emojiPanWidth - 8 * st::emojiCategory.width) / 2; - int32 top = _iconsTop = st::defaultDropdownPadding.top() + _contentHeight - st::emojiCategory.height; + int32 left = _iconsLeft = innerRect().x() + (st::emojiPanWidth - 8 * st::emojiCategory.width) / 2; + int32 top = _iconsTop = innerRect().y() + innerRect().height() - st::emojiCategory.height; prepareTab(left, top, _width, _recent, dbietRecent); prepareTab(left, top, _width, _people, dbietPeople); prepareTab(left, top, _width, _nature, dbietNature); @@ -2617,8 +2870,8 @@ EmojiPan::EmojiPan(QWidget *parent) : TWidget(parent) connect(&s_switch, SIGNAL(clicked()), this, SLOT(onSwitch())); connect(&e_switch, SIGNAL(clicked()), this, SLOT(onSwitch())); - s_switch.moveToRight(0, 0, st::emojiPanWidth); - e_switch.moveToRight(0, 0, st::emojiPanWidth); + s_switch.moveToRight(st::buttonRadius, 0, st::emojiPanWidth); + e_switch.moveToRight(st::buttonRadius, 0, st::emojiPanWidth); connect(&s_inner, SIGNAL(displaySet(quint64)), this, SLOT(onDisplaySet(quint64))); connect(&s_inner, SIGNAL(installSet(quint64)), this, SLOT(onInstallSet(quint64))); @@ -2660,24 +2913,23 @@ void EmojiPan::updateContentHeight() { _contentHeightEmoji = he; _contentHeightStickers = hs; - _height = st::defaultDropdownPadding.top() + _contentHeight + st::defaultDropdownPadding.bottom(); - - resize(_width, _height); - move(x(), _bottom - height()); + resize(QRect(0, 0, innerRect().width(), _contentHeight).marginsAdded(innerPadding()).size()); + _height = height(); + move(x(), _bottom - _height); if (was > _contentHeight || (was == _contentHeight && wass > _contentHeightStickers)) { - e_scroll.resize(st::emojiPanWidth, _contentHeightEmoji); - s_scroll.resize(st::emojiPanWidth, _contentHeightStickers); + e_scroll.resize(e_scroll.width(), _contentHeightEmoji); + s_scroll.resize(s_scroll.width(), _contentHeightStickers); s_inner.setMaxHeight(_contentHeightStickers); e_inner.setMaxHeight(_contentHeightEmoji); } else { s_inner.setMaxHeight(_contentHeightStickers); e_inner.setMaxHeight(_contentHeightEmoji); - e_scroll.resize(st::emojiPanWidth, _contentHeightEmoji); - s_scroll.resize(st::emojiPanWidth, _contentHeightStickers); + e_scroll.resize(e_scroll.width(), _contentHeightEmoji); + s_scroll.resize(s_scroll.width(), _contentHeightStickers); } - _iconsTop = st::defaultDropdownPadding.top() + _contentHeight - st::emojiCategory.height; + _iconsTop = innerRect().y() + innerRect().height() - st::emojiCategory.height; _recent->move(_recent->x(), _iconsTop); _people->move(_people->x(), _iconsTop); _nature->move(_nature->x(), _iconsTop); @@ -2730,113 +2982,114 @@ void EmojiPan::paintFeaturedStickerSetsBadge(Painter &p, int iconLeft) const { void EmojiPan::paintEvent(QPaintEvent *e) { Painter p(this); - float64 o = 1; - if (!_cache.isNull()) { - p.setOpacity(o = a_opacity.current()); - } - - QRect r(st::defaultDropdownPadding.left(), st::defaultDropdownPadding.top(), _width - st::defaultDropdownPadding.left() - st::defaultDropdownPadding.right(), _height - st::defaultDropdownPadding.top() - st::defaultDropdownPadding.bottom()); - - _shadow.paint(p, r, st::defaultDropdownShadowShift); - - if (_toCache.isNull()) { - if (_cache.isNull()) { - p.fillRect(myrtlrect(r.x() + r.width() - st::emojiScroll.width, r.y(), st::emojiScroll.width, e_scroll.height()), st::emojiPanBg); - if (_stickersShown && s_inner.showSectionIcons()) { - p.fillRect(r.left(), _iconsTop, r.width(), st::emojiCategory.height, st::emojiPanCategories); - paintStickerSettingsIcon(p); - - if (!_icons.isEmpty()) { - int x = _iconsLeft, selxrel = _iconsLeft + _iconSelX.current(), selx = selxrel - _iconsX.current(); - - QRect clip(x, _iconsTop, _iconsLeft + 7 * st::emojiCategory.width - x, st::emojiCategory.height); - if (rtl()) clip.moveLeft(width() - x - clip.width()); - p.setClipRect(clip); - - auto getSpecialSetIcon = [](uint64 setId, bool active) { - if (setId == Stickers::NoneSetId) { - return active ? &st::emojiSavedGifsActive : &st::emojiSavedGifs; - } else if (setId == Stickers::FeaturedSetId) { - return active ? &st::stickersTrendingActive : &st::stickersTrending; - } - return active ? &st::emojiRecentActive : &st::emojiRecent; - }; - - int i = 0; - i += _iconsX.current() / int(st::emojiCategory.width); - x -= _iconsX.current() % int(st::emojiCategory.width); - selxrel -= _iconsX.current(); - for (int l = qMin(_icons.size(), i + 8); i < l; ++i) { - auto &s = _icons.at(i); - if (s.sticker) { - s.sticker->thumb->load(); - QPixmap pix(s.sticker->thumb->pix(s.pixw, s.pixh)); - - p.drawPixmapLeft(x + (st::emojiCategory.width - s.pixw) / 2, _iconsTop + (st::emojiCategory.height - s.pixh) / 2, width(), pix); - x += st::emojiCategory.width; - } else { - getSpecialSetIcon(s.setId, false)->paint(p, x + st::emojiCategory.iconPosition.x(), _iconsTop + st::emojiCategory.iconPosition.y(), width()); - if (s.setId == Stickers::FeaturedSetId) { - paintFeaturedStickerSetsBadge(p, x); - } - x += st::emojiCategory.width; - } - } - - if (rtl()) selx = width() - selx - st::emojiCategory.width; - p.setOpacity(1.); - p.fillRect(selx, _iconsTop + st::emojiCategory.height - st::stickerIconPadding, st::emojiCategory.width, st::stickerIconSel, st::stickerIconSelColor); - - float64 o_left = snap(float64(_iconsX.current()) / st::stickerIconLeft.width(), 0., 1.); - if (o_left > 0) { - p.setOpacity(o_left); - st::stickerIconLeft.fill(p, rtlrect(_iconsLeft, _iconsTop, st::stickerIconLeft.width(), st::emojiCategory.height, width())); - } - float64 o_right = snap(float64(_iconsMax - _iconsX.current()) / st::stickerIconRight.width(), 0., 1.); - if (o_right > 0) { - p.setOpacity(o_right); - st::stickerIconRight.fill(p, rtlrect(_iconsLeft + 7 * st::emojiCategory.width - st::stickerIconRight.width(), _iconsTop, st::stickerIconRight.width(), st::emojiCategory.height, width())); - } - } - } else if (_stickersShown) { - int32 x = rtl() ? (_recent->x() + _recent->width()) : (_objects->x() + _objects->width()); - p.fillRect(x, _recent->y(), r.left() + r.width() - x, st::emojiCategory.height, st::emojiPanBg); - } else { - p.fillRect(r.left(), _iconsTop, r.width(), st::emojiCategory.height, st::emojiPanCategories); - } - } else { - p.fillRect(r, st::emojiPanBg); - p.drawPixmap(r.left(), r.top(), _cache); + auto ms = getms(); + auto opacityAnimating = _a_opacity.animating(ms); + auto switching = (_slideAnimation != nullptr); + if (_a_show.animating(ms)) { + if (auto opacity = _a_opacity.current(_hiding ? 0. : 1.)) { + _showAnimation->paintFrame(p, 0, 0, width(), _a_show.current(1.), opacity); + } + } else if (!switching && opacityAnimating) { + p.setOpacity(_a_opacity.current(_hiding ? 0. : 1.)); + p.drawPixmap(0, 0, _cache); + } else if ((!switching && _hiding) || isHidden()) { + hideFinished(); + } else if (_showAnimation) { + _showAnimation->paintFrame(p, 0, 0, width(), 1., 1.); + _showAnimation.reset(); + if (!switching && !opacityAnimating) showAll(); + } else if (switching) { + auto slideDt = _a_slide.current(ms, 1.); + _slideAnimation->paintFrame(p, slideDt, _a_opacity.current(_hiding ? 0. : 1.)); + if (!_a_slide.animating()) { + _slideAnimation.reset(); + if (!opacityAnimating) showAll(); } } else { - p.fillRect(QRect(r.left(), r.top(), r.width(), r.height() - st::emojiCategory.height), st::emojiPanBg); - p.fillRect(QRect(r.left(), _iconsTop, r.width(), st::emojiCategory.height), st::emojiPanCategories); - p.setOpacity(o * a_fromAlpha.current()); - QRect fromDst = QRect(r.left() + a_fromCoord.current(), r.top(), _fromCache.width() / cIntRetinaFactor(), _fromCache.height() / cIntRetinaFactor()); - QRect fromSrc = QRect(0, 0, _fromCache.width(), _fromCache.height()); - if (fromDst.x() < r.left() + r.width() && fromDst.x() + fromDst.width() > r.left()) { - if (fromDst.x() < r.left()) { - fromSrc.setX((r.left() - fromDst.x()) * cIntRetinaFactor()); - fromDst.setX(r.left()); - } else if (fromDst.x() + fromDst.width() > r.left() + r.width()) { - fromSrc.setWidth((r.left() + r.width() - fromDst.x()) * cIntRetinaFactor()); - fromDst.setWidth(r.left() + r.width() - fromDst.x()); + if (!_cache.isNull()) _cache = QPixmap(); + if (!_inPanelGrab) Ui::Shadow::paint(p, innerRect(), width(), st::emojiPanAnimation.shadow); + paintContent(p); + } +} + +void EmojiPan::paintContent(Painter &p) { + auto inner = innerRect(); + App::roundRect(p, inner, st::emojiPanBg, ImageRoundRadius::Small, App::RectPart::TopFull); + + auto showSectionIcons = _emojiShown || s_inner.showSectionIcons(); + auto &bottomBg = showSectionIcons ? st::emojiPanCategories : st::emojiPanBg; + auto bottomParts = showSectionIcons ? (App::RectPart::NoTopBottom | App::RectPart::BottomFull) : App::RectPart::BottomFull; + App::roundRect(p, inner.x(), _iconsTop - st::buttonRadius, inner.width(), st::emojiCategory.height + st::buttonRadius, bottomBg, ImageRoundRadius::Small, bottomParts); + + auto horizontal = horizontalRect(); + auto sidesTop = horizontal.y(); + auto sidesHeight = e_scroll.y() + e_scroll.height() - sidesTop; + p.fillRect(myrtlrect(inner.x() + inner.width() - st::emojiScroll.width, sidesTop, st::emojiScroll.width, sidesHeight), st::emojiPanBg); + p.fillRect(myrtlrect(inner.x(), sidesTop, st::buttonRadius, sidesHeight), st::emojiPanBg); + if (_emojiShown) { + auto vertical = verticalRect(); + p.fillRect(vertical.x(), _iconsTop, vertical.width(), st::emojiCategory.height - st::buttonRadius, st::emojiPanCategories); + } else if (showSectionIcons) { + paintStickerSettingsIcon(p); + + if (!_icons.isEmpty()) { + int x = _iconsLeft, selxrel = _iconsLeft + _iconSelX.current(), selx = selxrel - _iconsX.current(); + + QRect clip(x, _iconsTop, _iconsLeft + 7 * st::emojiCategory.width - x, st::emojiCategory.height); + if (rtl()) clip.moveLeft(width() - x - clip.width()); + p.setClipRect(clip); + + auto getSpecialSetIcon = [](uint64 setId, bool active) { + if (setId == Stickers::NoneSetId) { + return active ? &st::emojiSavedGifsActive : &st::emojiSavedGifs; + } else if (setId == Stickers::FeaturedSetId) { + return active ? &st::stickersTrendingActive : &st::stickersTrending; + } + return active ? &st::emojiRecentActive : &st::emojiRecent; + }; + + int i = 0; + i += _iconsX.current() / int(st::emojiCategory.width); + x -= _iconsX.current() % int(st::emojiCategory.width); + selxrel -= _iconsX.current(); + for (int l = qMin(_icons.size(), i + 8); i < l; ++i) { + auto &s = _icons.at(i); + if (s.sticker) { + s.sticker->thumb->load(); + QPixmap pix(s.sticker->thumb->pix(s.pixw, s.pixh)); + + p.drawPixmapLeft(x + (st::emojiCategory.width - s.pixw) / 2, _iconsTop + (st::emojiCategory.height - s.pixh) / 2, width(), pix); + x += st::emojiCategory.width; + } else { + getSpecialSetIcon(s.setId, false)->paint(p, x + st::emojiCategory.iconPosition.x(), _iconsTop + st::emojiCategory.iconPosition.y(), width()); + if (s.setId == Stickers::FeaturedSetId) { + paintFeaturedStickerSetsBadge(p, x); + } + x += st::emojiCategory.width; + } } - p.drawPixmap(fromDst, _fromCache, fromSrc); - } - p.setOpacity(o * a_toAlpha.current()); - QRect toDst = QRect(r.left() + a_toCoord.current(), r.top(), _toCache.width() / cIntRetinaFactor(), _toCache.height() / cIntRetinaFactor()); - QRect toSrc = QRect(0, 0, _toCache.width(), _toCache.height()); - if (toDst.x() < r.left() + r.width() && toDst.x() + toDst.width() > r.left()) { - if (toDst.x() < r.left()) { - toSrc.setX((r.left() - toDst.x()) * cIntRetinaFactor()); - toDst.setX(r.left()); - } else if (toDst.x() + toDst.width() > r.left() + r.width()) { - toSrc.setWidth((r.left() + r.width() - toDst.x()) * cIntRetinaFactor()); - toDst.setWidth(r.left() + r.width() - toDst.x()); + + if (rtl()) selx = width() - selx - st::emojiCategory.width; + p.fillRect(selx, _iconsTop + st::emojiCategory.height - st::stickerIconPadding, st::emojiCategory.width, st::stickerIconSel, st::stickerIconSelColor); + + float64 o_left = snap(float64(_iconsX.current()) / st::stickerIconLeft.width(), 0., 1.); + if (o_left > 0) { + p.setOpacity(o_left); + st::stickerIconLeft.fill(p, rtlrect(_iconsLeft, _iconsTop, st::stickerIconLeft.width(), st::emojiCategory.height, width())); + p.setOpacity(1.); } - p.drawPixmap(toDst, _toCache, toSrc); + float64 o_right = snap(float64(_iconsMax - _iconsX.current()) / st::stickerIconRight.width(), 0., 1.); + if (o_right > 0) { + p.setOpacity(o_right); + st::stickerIconRight.fill(p, rtlrect(_iconsLeft + 7 * st::emojiCategory.width - st::stickerIconRight.width(), _iconsTop, st::stickerIconRight.width(), st::emojiCategory.height, width())); + p.setOpacity(1.); + } + + p.setClipRect(QRect()); } + } else { + p.fillRect(myrtlrect(inner.x() + inner.width() - st::emojiScroll.width, _iconsTop, st::emojiScroll.width, st::emojiCategory.height - st::buttonRadius), st::emojiPanBg); + p.fillRect(myrtlrect(inner.x(), _iconsTop, st::buttonRadius, st::emojiCategory.height - st::buttonRadius), st::emojiPanBg); } } @@ -2846,16 +3099,18 @@ void EmojiPan::moveBottom(int32 bottom, bool force) { move(x(), _bottom - height()); return; } - if (_stickersShown && s_inner.inlineResultsShown()) { + if (!_emojiShown && s_inner.inlineResultsShown()) { + setOrigin(Ui::PanelAnimation::Origin::BottomLeft); moveToLeft(0, _bottom - height()); } else { + setOrigin(Ui::PanelAnimation::Origin::BottomRight); moveToRight(0, _bottom - height()); } } void EmojiPan::enterEvent(QEvent *e) { _hideTimer.stop(); - if (_hiding) showAnimated(); + if (_hiding) showAnimated(_origin); } bool EmojiPan::preventAutoHide() const { @@ -2864,21 +3119,24 @@ bool EmojiPan::preventAutoHide() const { void EmojiPan::leaveEvent(QEvent *e) { if (preventAutoHide() || s_inner.inlineResultsShown()) return; - if (_a_appearance.animating()) { - hideByTimerOrLeave(); + auto ms = getms(); + if (_a_show.animating(ms) || _a_opacity.animating(ms)) { + hideAnimated(); } else { _hideTimer.start(300); } + return TWidget::leaveEvent(e); } void EmojiPan::otherEnter() { _hideTimer.stop(); - showAnimated(); + showAnimated(_origin); } void EmojiPan::otherLeave() { if (preventAutoHide() || s_inner.inlineResultsShown()) return; - if (_a_appearance.animating()) { + auto ms = getms(); + if (_a_opacity.animating(ms)) { hideByTimerOrLeave(); } else { _hideTimer.start(0); @@ -2886,7 +3144,7 @@ void EmojiPan::otherLeave() { } void EmojiPan::mousePressEvent(QMouseEvent *e) { - if (!_stickersShown || e->button() != Qt::LeftButton) return; + if (_emojiShown || e->button() != Qt::LeftButton) return; _iconsMousePos = e ? e->globalPos() : QCursor::pos(); updateSelected(); @@ -2900,7 +3158,7 @@ void EmojiPan::mousePressEvent(QMouseEvent *e) { } void EmojiPan::mouseMoveEvent(QMouseEvent *e) { - if (!_stickersShown) return; + if (_emojiShown) return; _iconsMousePos = e ? e->globalPos() : QCursor::pos(); updateSelected(); @@ -2921,7 +3179,7 @@ void EmojiPan::mouseMoveEvent(QMouseEvent *e) { } void EmojiPan::mouseReleaseEvent(QMouseEvent *e) { - if (!_stickersShown || _icons.isEmpty()) return; + if (_emojiShown || _icons.isEmpty()) return; int32 wasDown = _iconDown; _iconDown = -1; @@ -2975,18 +3233,17 @@ bool EmojiPan::event(QEvent *e) { } void EmojiPan::hideFast() { - if (_a_appearance.animating()) { - _a_appearance.stop(); - } - a_opacity = anim::fvalue(0, 0); + if (isHidden()) return; + _hideTimer.stop(); - hide(); - _cache = QPixmap(); + _hiding = false; + _a_opacity.finish(); + hideFinished(); } void EmojiPan::refreshStickers() { s_inner.refreshStickers(); - if (!_stickersShown) { + if (_emojiShown) { s_inner.preloadImages(); } update(); @@ -2994,9 +3251,9 @@ void EmojiPan::refreshStickers() { void EmojiPan::refreshSavedGifs() { e_switch.updateText(); - e_switch.moveToRight(0, 0, st::emojiPanWidth); + e_switch.moveToRight(st::buttonRadius, 0, st::emojiPanWidth); s_inner.refreshSavedGifs(); - if (!_stickersShown) { + if (_emojiShown) { s_inner.preloadImages(); } } @@ -3022,7 +3279,7 @@ void EmojiPan::onRefreshIcons(bool scrollAnimation) { } updatePanelsPositions(s_panels, s_scroll.scrollTop()); updateSelected(); - if (_stickersShown) { + if (!_emojiShown) { validateSelectedIcon(scrollAnimation ? ValidateIconAnimations::Scroll : ValidateIconAnimations::None); updateContentHeight(); } @@ -3030,17 +3287,17 @@ void EmojiPan::onRefreshIcons(bool scrollAnimation) { } void EmojiPan::onRefreshPanels() { - s_inner.refreshPanels(s_panels); e_inner.refreshPanels(e_panels); - if (_stickersShown) { - updatePanelsPositions(s_panels, s_scroll.scrollTop()); - } else { + s_inner.refreshPanels(s_panels); + if (_emojiShown) { updatePanelsPositions(e_panels, e_scroll.scrollTop()); + } else { + updatePanelsPositions(s_panels, s_scroll.scrollTop()); } } void EmojiPan::leaveToChildEvent(QEvent *e, QWidget *child) { - if (!_stickersShown) return; + if (_emojiShown) return; _iconsMousePos = QCursor::pos(); updateSelected(); } @@ -3089,14 +3346,14 @@ void EmojiPan::updateSelected() { } void EmojiPan::updateIcons() { - if (!_stickersShown || !s_inner.showSectionIcons()) return; + if (_emojiShown || !s_inner.showSectionIcons()) return; QRect r(st::defaultDropdownPadding.left(), st::defaultDropdownPadding.top(), _width - st::defaultDropdownPadding.left() - st::defaultDropdownPadding.right(), _height - st::defaultDropdownPadding.top() - st::defaultDropdownPadding.bottom()); update(r.left(), _iconsTop, r.width(), st::emojiCategory.height); } void EmojiPan::step_icons(uint64 ms, bool timer) { - if (!_stickersShown) { + if (_emojiShown) { _a_icons.stop(); return; } @@ -3135,46 +3392,16 @@ void EmojiPan::step_icons(uint64 ms, bool timer) { } } -void EmojiPan::step_slide(float64 ms, bool timer) { - float64 fullDuration = st::introSlideDelta + st::introSlideDuration, dt = ms / fullDuration; - float64 dt1 = (ms > st::introSlideDuration) ? 1 : (ms / st::introSlideDuration), dt2 = (ms > st::introSlideDelta) ? (ms - st::introSlideDelta) / (st::introSlideDuration) : 0; - if (dt2 >= 1) { - _a_slide.stop(); - a_fromCoord.finish(); - a_fromAlpha.finish(); - a_toCoord.finish(); - a_toAlpha.finish(); - _fromCache = _toCache = QPixmap(); - if (_cache.isNull()) showAll(); - } else { - a_fromCoord.update(dt1, anim::easeInCirc); - a_fromAlpha.update(dt1, anim::easeOutCirc); - a_toCoord.update(dt2, anim::easeOutCirc); - a_toAlpha.update(dt2, anim::easeInCirc); - } - if (timer) update(); -} - -void EmojiPan::step_appearance(float64 ms, bool timer) { - if (_cache.isNull()) { - _a_appearance.stop(); - return; - } - - float64 dt = ms / st::defaultDropdownDuration; - if (dt >= 1) { - _a_appearance.stop(); - a_opacity.finish(); +void EmojiPan::opacityAnimationCallback() { + update(); + if (!_a_opacity.animating()) { if (_hiding) { - hideFinish(); - } else { - _cache = QPixmap(); - if (_toCache.isNull()) showAll(); + _hiding = false; + hideFinished(); + } else if (!_a_show.animating() && !_a_slide.animating()) { + showAll(); } - } else { - a_opacity.update(dt, anim::linear); } - if (timer) update(); } void EmojiPan::hideByTimerOrLeave() { @@ -3183,32 +3410,84 @@ void EmojiPan::hideByTimerOrLeave() { hideAnimated(); } -void EmojiPan::prepareShowHideCache() { - if (_cache.isNull()) { - QPixmap from = _fromCache, to = _toCache; - _fromCache = _toCache = QPixmap(); - showAll(); - _cache = myGrab(this, rect().marginsRemoved(st::defaultDropdownPadding)); - _fromCache = from; _toCache = to; +void EmojiPan::prepareCache() { + if (_a_opacity.animating()) return; + + auto showAnimation = base::take(_a_show); + auto showAnimationData = base::take(_showAnimation); + auto slideAnimation = base::take(_slideAnimation); + showAll(); + _cache = myGrab(this); + _slideAnimation = base::take(slideAnimation); + _showAnimation = base::take(showAnimationData); + _a_show = base::take(showAnimation); + if (_a_show.animating()) { + hideAll(); } } -void EmojiPan::hideAnimated() { - if (_hiding) return; - - prepareShowHideCache(); +void EmojiPan::startOpacityAnimation(bool hiding) { + _hiding = false; + prepareCache(); + _hiding = hiding; hideAll(); - _hiding = true; - a_opacity.start(0); - _a_appearance.start(); + _a_opacity.start([this] { opacityAnimationCallback(); }, _hiding ? 1. : 0., _hiding ? 0. : 1., st::emojiPanDuration); } -void EmojiPan::hideFinish() { +void EmojiPan::startShowAnimation() { + if (!_a_show.animating()) { + auto cache = base::take(_cache); + auto opacityAnimation = base::take(_a_opacity); + auto slideAnimationData = base::take(_slideAnimation); + auto slideAnimation = base::take(_a_slide); + showAll(); + auto image = grabForPanelAnimation(); + _a_slide = base::take(slideAnimation); + _slideAnimation = base::take(slideAnimationData); + _a_opacity = base::take(opacityAnimation); + _cache = base::take(_cache); + + _showAnimation = std_::make_unique(st::emojiPanAnimation, _origin); + auto inner = rect().marginsRemoved(st::defaultDropdownPadding); + _showAnimation->setFinalImage(std_::move(image), QRect(inner.topLeft() * cIntRetinaFactor(), inner.size() * cIntRetinaFactor())); + auto corners = App::cornersMask(ImageRoundRadius::Small); + _showAnimation->setCornerMasks(QImage(*corners[0]), QImage(*corners[1]), QImage(*corners[2]), QImage(*corners[3])); + _showAnimation->start(); + } + hideAll(); + _a_show.start([this] { update(); }, 0., 1., st::emojiPanShowDuration); +} + +QImage EmojiPan::grabForPanelAnimation() { + myEnsureResized(this); + auto result = QImage(size() * cIntRetinaFactor(), QImage::Format_ARGB32_Premultiplied); + result.setDevicePixelRatio(cRetinaFactor()); + result.fill(Qt::transparent); + _inPanelGrab = true; + render(&result); + _inPanelGrab = false; + return std_::move(result); +} + +void EmojiPan::hideAnimated() { + if (isHidden()) return; + if (_hiding) return; + + _hideTimer.stop(); + startOpacityAnimation(true); +} + +EmojiPan::~EmojiPan() = default; + +void EmojiPan::hideFinished() { hide(); e_inner.hideFinish(); s_inner.hideFinish(true); - _cache = _toCache = _fromCache = QPixmap(); - _a_slide.stop(); + _a_show.finish(); + _showAnimation.reset(); + _a_slide.finish(); + _slideAnimation.reset(); + _cache = QPixmap(); _horizontal = false; _hiding = false; @@ -3227,37 +3506,43 @@ void EmojiPan::hideFinish() { Notify::clipStopperHidden(ClipStopperSavedGifsPanel); } -void EmojiPan::showAnimated() { - if (!isHidden() && !_hiding) { - return; - } +void EmojiPan::setOrigin(Ui::PanelAnimation::Origin origin) { + _origin = origin; +} + +void EmojiPan::showAnimated(Ui::PanelAnimation::Origin origin) { + setOrigin(origin); + _hideTimer.stop(); + showStarted(); +} + +void EmojiPan::showStarted() { if (isHidden()) { e_inner.refreshRecent(); if (s_inner.inlineResultsShown() && refreshInlineRows()) { - _stickersShown = true; + _emojiShown = false; _shownFromInlineQuery = true; } else { s_inner.refreshRecent(); - _stickersShown = false; + _emojiShown = true; _shownFromInlineQuery = false; _cache = QPixmap(); // clear after refreshInlineRows() } recountContentMaxHeight(); s_inner.preloadImages(); - _fromCache = _toCache = QPixmap(); - _a_slide.stop(); + _a_slide.finish(); + _slideAnimation.reset(); moveBottom(y() + height(), true); - } else if (_hiding) { - if (s_inner.inlineResultsShown() && refreshInlineRows()) { - onSwitch(); - } + show(); + startShowAnimation(); + return; + } else if (!_hiding) { + return; } - prepareShowHideCache(); - hideAll(); - _hiding = false; - show(); - a_opacity.start(1); - _a_appearance.start(); + if (s_inner.inlineResultsShown() && refreshInlineRows()) { + onSwitch(); + } + startOpacityAnimation(false); emit updateStickers(); } @@ -3277,7 +3562,7 @@ bool EmojiPan::eventFilter(QObject *obj, QEvent *e) { } else if (e->type() == QEvent::MouseButtonPress && static_cast(e)->button() == Qt::LeftButton/* && !dynamic_cast(obj)*/) { if (isHidden() || _hiding) { _hideTimer.stop(); - showAnimated(); + showAnimated(_origin); } else { hideAnimated(); } @@ -3286,59 +3571,46 @@ bool EmojiPan::eventFilter(QObject *obj, QEvent *e) { } void EmojiPan::stickersInstalled(uint64 setId) { - _stickersShown = true; + _emojiShown = false; if (isHidden()) { moveBottom(y() + height(), true); + startShowAnimation(); show(); - a_opacity = anim::fvalue(0, 1); - a_opacity.update(0, anim::linear); - _cache = _fromCache = _toCache = QPixmap(); } showAll(); s_inner.showStickerSet(setId); updateContentHeight(); - showAnimated(); + showAnimated(Ui::PanelAnimation::Origin::BottomRight); } void EmojiPan::notify_inlineItemLayoutChanged(const InlineBots::Layout::ItemBase *layout) { - if (_stickersShown && !isHidden()) { + if (!_emojiShown && !isHidden()) { s_inner.notify_inlineItemLayoutChanged(layout); } } void EmojiPan::ui_repaintInlineItem(const InlineBots::Layout::ItemBase *layout) { - if (_stickersShown && !isHidden()) { + if (!_emojiShown && !isHidden()) { s_inner.ui_repaintInlineItem(layout); } } bool EmojiPan::ui_isInlineItemVisible(const InlineBots::Layout::ItemBase *layout) { - if (_stickersShown && !isHidden()) { + if (!_emojiShown && !isHidden()) { return s_inner.ui_isInlineItemVisible(layout); } return false; } bool EmojiPan::ui_isInlineItemBeingChosen() { - if (_stickersShown && !isHidden()) { + if (!_emojiShown && !isHidden()) { return s_inner.ui_isInlineItemBeingChosen(); } return false; } void EmojiPan::showAll() { - if (_stickersShown) { - s_scroll.show(); - _recent->hide(); - _people->hide(); - _nature->hide(); - _food->hide(); - _activity->hide(); - _travel->hide(); - _objects->hide(); - _symbols->hide(); - e_scroll.hide(); - } else { + if (_emojiShown) { s_scroll.hide(); _recent->show(); _people->show(); @@ -3349,6 +3621,17 @@ void EmojiPan::showAll() { _objects->show(); _symbols->show(); e_scroll.show(); + } else { + s_scroll.show(); + _recent->hide(); + _people->hide(); + _nature->hide(); + _food->hide(); + _activity->hide(); + _travel->hide(); + _objects->hide(); + _symbols->hide(); + e_scroll.hide(); } } @@ -3454,11 +3737,65 @@ void EmojiPan::validateSelectedIcon(ValidateIconAnimations animations) { } } +style::margins EmojiPan::innerPadding() const { + return st::defaultDropdownPadding; +} + +QRect EmojiPan::innerRect() const { + return rect().marginsRemoved(innerPadding()); +} + +QRect EmojiPan::horizontalRect() const { + return innerRect().marginsRemoved(style::margins(0, st::buttonRadius, 0, st::buttonRadius)); +} + +QRect EmojiPan::verticalRect() const { + return innerRect().marginsRemoved(style::margins(st::buttonRadius, 0, st::buttonRadius, 0)); +} + void EmojiPan::onSwitch() { - QPixmap cache = _cache; - _fromCache = myGrab(this, rect().marginsRemoved(st::defaultDropdownPadding)); - _stickersShown = !_stickersShown; - if (!_stickersShown) { + auto cache = base::take(_cache); + auto opacityAnimation = base::take(_a_opacity); + auto showAnimationData = base::take(_showAnimation); + auto showAnimation = base::take(_a_show); + + showAll(); + auto leftImage = grabForPanelAnimation(); + performSwitch(); + showAll(); + auto rightImage = grabForPanelAnimation(); + if (_emojiShown) { + std_::swap_moveable(leftImage, rightImage); + } + + _a_show = base::take(showAnimation); + _showAnimation = base::take(showAnimationData); + _a_opacity = base::take(opacityAnimation); + _cache = base::take(_cache); + + auto direction = _emojiShown ? SlideAnimation::Direction::LeftToRight : SlideAnimation::Direction::RightToLeft; + _slideAnimation = std_::make_unique(); + auto inner = rect().marginsRemoved(st::defaultDropdownPadding); + _slideAnimation->setFinalImages(direction, std_::move(leftImage), std_::move(rightImage), QRect(inner.topLeft() * cIntRetinaFactor(), inner.size() * cIntRetinaFactor())); + auto corners = App::cornersMask(ImageRoundRadius::Small); + _slideAnimation->setCornerMasks(QImage(*corners[0]), QImage(*corners[1]), QImage(*corners[2]), QImage(*corners[3])); + _slideAnimation->start(); + + hideAll(); + + if (_emojiShown) { + s_inner.hideFinish(false); + } else { + e_inner.hideFinish(); + } + + _a_slide.start([this] { update(); }, 0., 1., st::emojiPanSlideDuration, anim::linear); + update(); +} + +void EmojiPan::performSwitch() { + _emojiShown = !_emojiShown; + if (_emojiShown) { Notify::clipStopperHidden(ClipStopperSavedGifsPanel); } else { if (cShowingSavedGifs() && cSavedGifs().isEmpty()) { @@ -3478,27 +3815,6 @@ void EmojiPan::onSwitch() { _iconHovers = _icons.isEmpty() ? QVector() : QVector(_icons.size(), 0); _iconAnimations.clear(); _a_icons.stop(); - - _cache = QPixmap(); - showAll(); - _toCache = myGrab(this, rect().marginsRemoved(st::defaultDropdownPadding)); - _cache = cache; - - hideAll(); - - if (_stickersShown) { - e_inner.hideFinish(); - } else { - s_inner.hideFinish(false); - } - - a_toCoord = (_stickersShown != rtl()) ? anim::ivalue(st::emojiPanWidth, 0) : anim::ivalue(-st::emojiPanWidth, 0); - a_toAlpha = anim::fvalue(0, 1); - a_fromCoord = (_stickersShown != rtl()) ? anim::ivalue(0, -st::emojiPanWidth) : anim::ivalue(0, st::emojiPanWidth); - a_fromAlpha = anim::fvalue(1, 0); - - _a_slide.start(); - update(); } void EmojiPan::onDisplaySet(quint64 setId) { @@ -3593,11 +3909,11 @@ void EmojiPan::onDelayedHide() { void EmojiPan::clearInlineBot() { inlineBotChanged(); e_switch.updateText(); - e_switch.moveToRight(0, 0, st::emojiPanWidth); + e_switch.moveToRight(st::buttonRadius, 0, st::emojiPanWidth); } bool EmojiPan::hideOnNoInlineResults() { - return _inlineBot && _stickersShown && s_inner.inlineResultsShown() && (_shownFromInlineQuery || _inlineBot->username != cInlineGifBotUsername()); + return _inlineBot && !_emojiShown && s_inner.inlineResultsShown() && (_shownFromInlineQuery || _inlineBot->username != cInlineGifBotUsername()); } void EmojiPan::inlineBotChanged() { @@ -3743,7 +4059,7 @@ bool EmojiPan::refreshInlineRows(int32 *added) { } _inlineNextOffset = i.value()->nextOffset; } - if (!entry) prepareShowHideCache(); + if (!entry) prepareCache(); int32 result = s_inner.refreshInlineRows(_inlineBot, entry, false); if (added) *added = result; return (entry != nullptr); @@ -3770,8 +4086,8 @@ int32 EmojiPan::showInlineRows(bool newResults) { } else { _hideTimer.stop(); if (hidden || _hiding) { - showAnimated(); - } else if (!_stickersShown) { + showAnimated(_origin); + } else if (_emojiShown) { onSwitch(); } } diff --git a/Telegram/SourceFiles/stickers/emoji_pan.h b/Telegram/SourceFiles/stickers/emoji_pan.h index 6fed74829..be927c8cc 100644 --- a/Telegram/SourceFiles/stickers/emoji_pan.h +++ b/Telegram/SourceFiles/stickers/emoji_pan.h @@ -23,6 +23,7 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org #include "ui/twidget.h" #include "ui/effects/rect_shadow.h" #include "ui/abstract_button.h" +#include "ui/effects/panel_animation.h" namespace InlineBots { namespace Layout { @@ -60,7 +61,6 @@ class EmojiColorPicker : public TWidget { Q_OBJECT public: - EmojiColorPicker(); void showEmoji(uint32 code); @@ -345,6 +345,7 @@ private: void appendSet(Sets &to, uint64 setId, AppendSkip skip); void selectEmoji(EmojiPtr emoji); + int stickersLeft() const; QRect stickerRect(int tab, int sel); int32 _maxHeight; @@ -510,8 +511,6 @@ public: return _hiding || _hideTimer.isActive(); } - void step_appearance(float64 ms, bool timer); - void step_slide(float64 ms, bool timer); void step_icons(uint64 ms, bool timer); bool eventFilter(QObject *obj, QEvent *e); @@ -535,13 +534,12 @@ public: bool ui_isInlineItemVisible(const InlineBots::Layout::ItemBase *layout); bool ui_isInlineItemBeingChosen(); - bool inlineResultsShown() const { - return s_inner.inlineResultsShown(); - } - - void showAnimated(); + void setOrigin(Ui::PanelAnimation::Origin origin); + void showAnimated(Ui::PanelAnimation::Origin origin); void hideAnimated(); + ~EmojiPan(); + public slots: void refreshStickers(); @@ -549,8 +547,6 @@ private slots: void hideByTimerOrLeave(); void refreshSavedGifs(); - void hideFinish(); - void onWndActiveChanged(); void onScrollEmoji(); @@ -581,6 +577,33 @@ signals: void updateStickers(); private: + void paintContent(Painter &p); + void performSwitch(); + + style::margins innerPadding() const; + + // Rounded rect which has shadow around it. + QRect innerRect() const; + + // Inner rect with removed st::buttonRadius from top and bottom. + // This one is allowed to be not rounded. + QRect horizontalRect() const; + + // Inner rect with removed st::buttonRadius from left and right. + // This one is allowed to be not rounded. + QRect verticalRect() const; + + QImage grabForPanelAnimation(); + void startShowAnimation(); + void startOpacityAnimation(bool hiding); + void prepareCache(); + + class Container; + void opacityAnimationCallback(); + + void hideFinished(); + void showStarted(); + bool preventAutoHide() const; void installSetDone(const MTPmessages_StickerSetInstallResult &result); bool installSetFail(uint64 setId, const RPCError &error); @@ -599,7 +622,6 @@ private: void updateContentHeight(); void leaveToChildEvent(QEvent *e, QWidget *child); - void prepareShowHideCache(); void updateSelected(); void updateIcons(); @@ -614,15 +636,20 @@ private: bool _horizontal = false; int32 _width, _height, _bottom; + + Ui::PanelAnimation::Origin _origin = Ui::PanelAnimation::Origin::BottomRight; + std_::unique_ptr _showAnimation; + FloatAnimation _a_show; + bool _hiding = false; QPixmap _cache; - - anim::fvalue a_opacity = { 0. }; - Animation _a_appearance; - + FloatAnimation _a_opacity; QTimer _hideTimer; + bool _inPanelGrab = false; - Ui::RectShadow _shadow; + class SlideAnimation; + std_::unique_ptr _slideAnimation; + FloatAnimation _a_slide; ChildWidget _recent; ChildWidget _people; @@ -651,12 +678,8 @@ private: anim::ivalue _iconSelX = { 0, 0 }; uint64 _iconsStartAnim = 0; - bool _stickersShown = false; + bool _emojiShown = true; bool _shownFromInlineQuery = false; - QPixmap _fromCache, _toCache; - anim::ivalue a_fromCoord, a_toCoord; - anim::fvalue a_fromAlpha, a_toAlpha; - Animation _a_slide; ScrollArea e_scroll; internal::EmojiPanInner e_inner; diff --git a/Telegram/SourceFiles/stickers/stickers.style b/Telegram/SourceFiles/stickers/stickers.style index d2616bb60..c751dcdc7 100644 --- a/Telegram/SourceFiles/stickers/stickers.style +++ b/Telegram/SourceFiles/stickers/stickers.style @@ -117,12 +117,18 @@ emojiCategoryTravel: IconButton(emojiCategory) { icon: emojiTravel; } emojiCategoryObjects: IconButton(emojiCategory) { icon: emojiObjects; } emojiCategorySymbols: IconButton(emojiCategory) { icon: emojiSymbols; } +emojiPanAnimation: PanelAnimation(defaultPanelAnimation) { + fadeBg: emojiPanBg; +} emojiPanPadding: 12px; emojiPanSize: size(45px, 41px); emojiPanWidth: 345px; emojiPanMaxHeight: 366px; +emojiPanShowDuration: 200; emojiPanDuration: 200; emojiPanHover: #f0f4f7; +emojiPanSlideDuration: 200; +emojiPanSlideDelta: 0; // between hide start and show start emojiPanHeader: 42px; emojiPanHeaderFont: semiboldFont; diff --git a/Telegram/SourceFiles/ui/animation.h b/Telegram/SourceFiles/ui/animation.h index 8b5ff5460..81370ef6d 100644 --- a/Telegram/SourceFiles/ui/animation.h +++ b/Telegram/SourceFiles/ui/animation.h @@ -213,7 +213,110 @@ void startManager(); void stopManager(); void registerClipManager(Media::Clip::Manager *manager); -inline uint64 shifted(uint32 components) { +FORCE_INLINE int interpolate(int a, int b, float64 b_ratio) { + return qRound(a + float64(b - a) * b_ratio); +} + +#ifdef ARCH_CPU_32_BITS +#define SHIFTED_USE_32BIT +#endif // ARCH_CPU_32_BITS + +#ifdef SHIFTED_USE_32BIT + +using ShiftedMultiplier = uint32; + +struct Shifted { + Shifted() = default; + Shifted(uint32 low, uint32 high) : low(low), high(high) { + } + uint32 low = 0; + uint32 high = 0; +}; + +FORCE_INLINE Shifted operator+(Shifted a, Shifted b) { + return Shifted(a.low + b.low, a.high + b.high); +} + +FORCE_INLINE Shifted operator*(Shifted shifted, ShiftedMultiplier multiplier) { + return Shifted(shifted.low * multiplier, shifted.high * multiplier); +} + +FORCE_INLINE Shifted operator*(ShiftedMultiplier multiplier, Shifted shifted) { + return Shifted(shifted.low * multiplier, shifted.high * multiplier); +} + +FORCE_INLINE Shifted shifted(uint32 components) { + return Shifted( + (components & 0x000000FFU) | ((components & 0x0000FF00U) << 8), + ((components & 0x00FF0000U) >> 16) | ((components & 0xFF000000U) >> 8)); +} + +FORCE_INLINE uint32 unshifted(Shifted components) { + return ((components.low & 0x0000FF00U) >> 8) + | ((components.low & 0xFF000000U) >> 16) + | ((components.high & 0x0000FF00U) << 8) + | (components.high & 0xFF000000U); +} + +FORCE_INLINE Shifted reshifted(Shifted components) { + return Shifted((components.low >> 8) & 0x00FF00FFU, (components.high >> 8) & 0x00FF00FFU); +} + +FORCE_INLINE Shifted shifted(QColor color) { + // Make it premultiplied. + auto alpha = static_cast((color.alpha() & 0xFF) + 1); + auto components = Shifted(static_cast(color.blue() & 0xFF) | (static_cast(color.green() & 0xFF) << 16), + static_cast(color.red() & 0xFF) | (static_cast(255) << 16)); + return reshifted(components * alpha); +} + +FORCE_INLINE uint32 getAlpha(Shifted components) { + return (components.high & 0x00FF0000U) >> 16; +} + +FORCE_INLINE Shifted non_premultiplied(QColor color) { + return Shifted(static_cast(color.blue() & 0xFF) | (static_cast(color.green() & 0xFF) << 16), + static_cast(color.red() & 0xFF) | (static_cast(color.alpha() & 0xFF) << 16)); +} + +FORCE_INLINE QColor color(QColor a, QColor b, float64 b_ratio) { + auto bOpacity = snap(interpolate(0, 255, b_ratio), 0, 255) + 1; + auto aOpacity = (256 - bOpacity); + auto components = (non_premultiplied(a) * aOpacity + non_premultiplied(b) * bOpacity); + return { + static_cast((components.high >> 8) & 0xFF), + static_cast((components.low >> 24) & 0xFF), + static_cast((components.low >> 8) & 0xFF), + static_cast((components.high >> 24) & 0xFF), + }; +} + +#else // SHIFTED_USE_32BIT + +using ShiftedMultiplier = uint64; + +struct Shifted { + Shifted() = default; + Shifted(uint32 value) : value(value) { + } + Shifted(uint64 value) : value(value) { + } + uint64 value = 0; +}; + +FORCE_INLINE Shifted operator+(Shifted a, Shifted b) { + return Shifted(a.value + b.value); +} + +FORCE_INLINE Shifted operator*(Shifted shifted, ShiftedMultiplier multiplier) { + return Shifted(shifted.value * multiplier); +} + +FORCE_INLINE Shifted operator*(ShiftedMultiplier multiplier, Shifted shifted) { + return Shifted(shifted.value * multiplier); +} + +FORCE_INLINE Shifted shifted(uint32 components) { auto wide = static_cast(components); return (wide & 0x00000000000000FFULL) | ((wide & 0x000000000000FF00ULL) << 8) @@ -221,82 +324,93 @@ inline uint64 shifted(uint32 components) { | ((wide & 0x00000000FF000000ULL) << 24); } -inline uint32 unshifted(uint64 components) { - return static_cast((components & 0x000000000000FF00ULL) >> 8) - | static_cast((components & 0x00000000FF000000ULL) >> 16) - | static_cast((components & 0x0000FF0000000000ULL) >> 24) - | static_cast((components & 0xFF00000000000000ULL) >> 32); +FORCE_INLINE uint32 unshifted(Shifted components) { + return static_cast((components.value & 0x000000000000FF00ULL) >> 8) + | static_cast((components.value & 0x00000000FF000000ULL) >> 16) + | static_cast((components.value & 0x0000FF0000000000ULL) >> 24) + | static_cast((components.value & 0xFF00000000000000ULL) >> 32); } -inline uint64 reshifted(uint64 components) { - return (components >> 8) & 0x00FF00FF00FF00FFULL; +FORCE_INLINE Shifted reshifted(Shifted components) { + return (components.value >> 8) & 0x00FF00FF00FF00FFULL; } -inline int interpolate(int a, int b, float64 b_ratio) { - return qRound(a + float64(b - a) * b_ratio); +FORCE_INLINE Shifted shifted(QColor color) { + // Make it premultiplied. + auto alpha = static_cast((color.alpha() & 0xFF) + 1); + auto components = static_cast(color.blue() & 0xFF) + | (static_cast(color.green() & 0xFF) << 16) + | (static_cast(color.red() & 0xFF) << 32) + | (static_cast(255) << 48); + return reshifted(components * alpha); } -inline QColor color(QColor a, QColor b, float64 b_ratio) { +FORCE_INLINE uint32 getAlpha(Shifted components) { + return (components.value & 0x00FF000000000000ULL) >> 48; +} + +FORCE_INLINE Shifted non_premultiplied(QColor color) { + return static_cast(color.blue() & 0xFF) + | (static_cast(color.green() & 0xFF) << 16) + | (static_cast(color.red() & 0xFF) << 32) + | (static_cast(color.alpha() & 0xFF) << 48); +} + +FORCE_INLINE QColor color(QColor a, QColor b, float64 b_ratio) { auto bOpacity = snap(interpolate(0, 255, b_ratio), 0, 255) + 1; auto aOpacity = (256 - bOpacity); - auto bBits = static_cast(b.alpha() & 0xFF) - | (static_cast(b.red() & 0xFF) << 16) - | (static_cast(b.green() & 0xFF) << 32) - | (static_cast(b.blue() & 0xFF) << 48); - auto aBits = static_cast(a.alpha() & 0xFF) - | (static_cast(a.red() & 0xFF) << 16) - | (static_cast(a.green() & 0xFF) << 32) - | (static_cast(a.blue() & 0xFF) << 48); - auto resultBits = (aBits * aOpacity + bBits * bOpacity) >> 8; + auto components = (non_premultiplied(a) * aOpacity + non_premultiplied(b) * bOpacity); return { - static_cast((resultBits >> 16) & 0xFF), - static_cast((resultBits >> 32) & 0xFF), - static_cast((resultBits >> 48) & 0xFF), - static_cast(resultBits & 0xFF), + static_cast((components.value >> 40) & 0xFF), + static_cast((components.value >> 24) & 0xFF), + static_cast((components.value >> 8) & 0xFF), + static_cast((components.value >> 56) & 0xFF), }; } -inline QColor color(const style::color &a, QColor b, float64 b_ratio) { +#endif // SHIFTED_USE_32BIT + +FORCE_INLINE QColor color(const style::color &a, QColor b, float64 b_ratio) { return color(a->c, b, b_ratio); } -inline QColor color(QColor a, const style::color &b, float64 b_ratio) { +FORCE_INLINE QColor color(QColor a, const style::color &b, float64 b_ratio) { return color(a, b->c, b_ratio); } -inline QColor color(const style::color &a, const style::color &b, float64 b_ratio) { +FORCE_INLINE QColor color(const style::color &a, const style::color &b, float64 b_ratio) { return color(a->c, b->c, b_ratio); } -inline QPen pen(QColor a, QColor b, float64 b_ratio) { +FORCE_INLINE QPen pen(QColor a, QColor b, float64 b_ratio) { return color(a, b, b_ratio); } -inline QPen pen(const style::color &a, QColor b, float64 b_ratio) { +FORCE_INLINE QPen pen(const style::color &a, QColor b, float64 b_ratio) { return (b_ratio > 0) ? pen(a->c, b, b_ratio) : a; } -inline QPen pen(QColor a, const style::color &b, float64 b_ratio) { +FORCE_INLINE QPen pen(QColor a, const style::color &b, float64 b_ratio) { return (b_ratio < 1) ? pen(a, b->c, b_ratio) : b; } -inline QPen pen(const style::color &a, const style::color &b, float64 b_ratio) { +FORCE_INLINE QPen pen(const style::color &a, const style::color &b, float64 b_ratio) { return (b_ratio > 0) ? ((b_ratio < 1) ? pen(a->c, b->c, b_ratio) : b) : a; } -inline QBrush brush(QColor a, QColor b, float64 b_ratio) { +FORCE_INLINE QBrush brush(QColor a, QColor b, float64 b_ratio) { return color(a, b, b_ratio); } -inline QBrush brush(const style::color &a, QColor b, float64 b_ratio) { +FORCE_INLINE QBrush brush(const style::color &a, QColor b, float64 b_ratio) { return (b_ratio > 0) ? brush(a->c, b, b_ratio) : a; } -inline QBrush brush(QColor a, const style::color &b, float64 b_ratio) { +FORCE_INLINE QBrush brush(QColor a, const style::color &b, float64 b_ratio) { return (b_ratio < 1) ? brush(a, b->c, b_ratio) : b; } -inline QBrush brush(const style::color &a, const style::color &b, float64 b_ratio) { +FORCE_INLINE QBrush brush(const style::color &a, const style::color &b, float64 b_ratio) { return (b_ratio > 0) ? ((b_ratio < 1) ? brush(a->c, b->c, b_ratio) : b) : a; } diff --git a/Telegram/SourceFiles/ui/effects/panel_animation.cpp b/Telegram/SourceFiles/ui/effects/panel_animation.cpp index c41b3bdba..28b3efa84 100644 --- a/Telegram/SourceFiles/ui/effects/panel_animation.cpp +++ b/Telegram/SourceFiles/ui/effects/panel_animation.cpp @@ -23,64 +23,31 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org namespace Ui { -void PanelAnimation::setFinalImage(QImage &&finalImage, QRect inner) { +void RoundShadowAnimation::start(int frameWidth, int frameHeight, float64 devicePixelRatio) { t_assert(!started()); - _finalImage = std_::move(finalImage).convertToFormat(QImage::Format_ARGB32_Premultiplied); - - t_assert(!_finalImage.isNull()); - _finalWidth = _finalImage.width(); - _finalHeight = _finalImage.height(); - _finalInnerLeft = inner.x(); - _finalInnerTop = inner.y(); - _finalInnerWidth = inner.width(); - _finalInnerHeight = inner.height(); - _finalInnerRight = _finalInnerLeft + _finalInnerWidth; - _finalInnerBottom = _finalInnerTop + _finalInnerHeight; - t_assert(QRect(0, 0, _finalWidth, _finalHeight).contains(inner)); - - setStartWidth(); - setStartHeight(); - setStartAlpha(); - setStartFadeTop(); - createFadeMask(); - setWidthDuration(); - setHeightDuration(); - setAlphaDuration(); - setShadow(); - - auto checkCorner = [this, inner](Corner &corner) { - if (!corner.valid()) return; - if ((_startWidth >= 0 && _startWidth < _finalWidth) - || (_startHeight >= 0 && _startHeight < _finalHeight)) { - t_assert(corner.width <= inner.width()); - t_assert(corner.height <= inner.height()); - } - }; - checkCorner(_topLeft); - checkCorner(_topRight); - checkCorner(_bottomLeft); - checkCorner(_bottomRight); - _finalInts = reinterpret_cast(_finalImage.constBits()); - _finalIntsPerLine = (_finalImage.bytesPerLine() >> 2); - _finalIntsPerLineAdded = _finalIntsPerLine - _finalWidth; - t_assert(_finalImage.depth() == static_cast(sizeof(uint32) << 3)); - t_assert(_finalImage.bytesPerLine() == (_finalIntsPerLine << 2)); - t_assert(_finalIntsPerLineAdded >= 0); + _frameWidth = frameWidth; + _frameHeight = frameHeight; + _frame = QImage(_frameWidth, _frameHeight, QImage::Format_ARGB32_Premultiplied); + _frame.setDevicePixelRatio(devicePixelRatio); + _frameIntsPerLine = (_frame.bytesPerLine() >> 2); + _frameInts = reinterpret_cast(_frame.bits()); + _frameIntsPerLineAdded = _frameIntsPerLine - _frameWidth; + t_assert(_frame.depth() == static_cast(sizeof(uint32) << 3)); + t_assert(_frame.bytesPerLine() == (_frameIntsPerLine << 2)); + t_assert(_frameIntsPerLineAdded >= 0); } -void PanelAnimation::setShadow() { - if (_skipShadow) return; - - _shadow.extend = _st.shadow.extend * cIntRetinaFactor(); - _shadow.left = cloneImage(_st.shadow.left); +void RoundShadowAnimation::setShadow(const style::Shadow &st) { + _shadow.extend = st.extend * cIntRetinaFactor(); + _shadow.left = cloneImage(st.left); if (_shadow.valid()) { - _shadow.topLeft = cloneImage(_st.shadow.topLeft); - _shadow.top = cloneImage(_st.shadow.top); - _shadow.topRight = cloneImage(_st.shadow.topRight); - _shadow.right = cloneImage(_st.shadow.right); - _shadow.bottomRight = cloneImage(_st.shadow.bottomRight); - _shadow.bottom = cloneImage(_st.shadow.bottom); - _shadow.bottomLeft = cloneImage(_st.shadow.bottomLeft); + _shadow.topLeft = cloneImage(st.topLeft); + _shadow.top = cloneImage(st.top); + _shadow.topRight = cloneImage(st.topRight); + _shadow.right = cloneImage(st.right); + _shadow.bottomRight = cloneImage(st.bottomRight); + _shadow.bottom = cloneImage(st.bottom); + _shadow.bottomLeft = cloneImage(st.bottomLeft); t_assert(!_shadow.topLeft.isNull() && !_shadow.top.isNull() && !_shadow.topRight.isNull() @@ -99,6 +66,214 @@ void PanelAnimation::setShadow() { } } +void RoundShadowAnimation::setCornerMasks(QImage &&topLeft, QImage &&topRight, QImage &&bottomLeft, QImage &&bottomRight) { + setCornerMask(_topLeft, std_::move(topLeft)); + setCornerMask(_topRight, std_::move(topRight)); + setCornerMask(_bottomLeft, std_::move(bottomLeft)); + setCornerMask(_bottomRight, std_::move(bottomRight)); +} + +void RoundShadowAnimation::setCornerMask(Corner &corner, QImage &&image) { + t_assert(!started()); + corner.image = std_::move(image); + if (corner.valid()) { + corner.width = corner.image.width(); + corner.height = corner.image.height(); + corner.bytes = corner.image.constBits(); + corner.bytesPerPixel = (corner.image.depth() >> 3); + corner.bytesPerLineAdded = corner.image.bytesPerLine() - corner.width * corner.bytesPerPixel; + t_assert(corner.image.depth() == (corner.bytesPerPixel << 3)); + t_assert(corner.bytesPerLineAdded >= 0); + } else { + corner.width = corner.height = 0; + } +} + +QImage RoundShadowAnimation::cloneImage(const style::icon &source) { + if (source.empty()) return QImage(); + + auto result = QImage(source.size() * cIntRetinaFactor(), QImage::Format_ARGB32_Premultiplied); + result.setDevicePixelRatio(cRetinaFactor()); + result.fill(Qt::transparent); + { + Painter p(&result); + source.paint(p, 0, 0, source.width()); + } + return std_::move(result); +} + +void RoundShadowAnimation::paintCorner(Corner &corner, int left, int top) { + auto mask = corner.bytes; + auto bytesPerPixel = corner.bytesPerPixel; + auto bytesPerLineAdded = corner.bytesPerLineAdded; + auto frameInts = _frameInts + top * _frameIntsPerLine + left; + auto frameIntsPerLineAdd = _frameIntsPerLine - corner.width; + for (auto y = 0; y != corner.height; ++y) { + for (auto x = 0; x != corner.width; ++x) { + auto alpha = static_cast(*mask) + 1; + *frameInts = anim::unshifted(anim::shifted(*frameInts) * alpha); + ++frameInts; + mask += bytesPerPixel; + } + frameInts += frameIntsPerLineAdd; + mask += bytesPerLineAdded; + } +} + +void RoundShadowAnimation::paintShadow(int left, int top, int right, int bottom) { + paintShadowCorner(left, top, _shadow.topLeft); + paintShadowCorner(right - _shadow.topRight.width(), top, _shadow.topRight); + paintShadowCorner(right - _shadow.bottomRight.width(), bottom - _shadow.bottomRight.height(), _shadow.bottomRight); + paintShadowCorner(left, bottom - _shadow.bottomLeft.height(), _shadow.bottomLeft); + paintShadowVertical(left, top + _shadow.topLeft.height(), bottom - _shadow.bottomLeft.height(), _shadow.left); + paintShadowVertical(right - _shadow.right.width(), top + _shadow.topRight.height(), bottom - _shadow.bottomRight.height(), _shadow.right); + paintShadowHorizontal(left + _shadow.topLeft.width(), right - _shadow.topRight.width(), top, _shadow.top); + paintShadowHorizontal(left + _shadow.bottomLeft.width(), right - _shadow.bottomRight.width(), bottom - _shadow.bottom.height(), _shadow.bottom); +} + +void RoundShadowAnimation::paintShadowCorner(int left, int top, const QImage &image) { + auto imageWidth = image.width(); + auto imageHeight = image.height(); + auto imageInts = reinterpret_cast(image.constBits()); + auto imageIntsPerLine = (image.bytesPerLine() >> 2); + auto imageIntsPerLineAdded = imageIntsPerLine - imageWidth; + if (left < 0) { + auto shift = -base::take(left); + imageWidth -= shift; + imageInts += shift; + } + if (top < 0) { + auto shift = -base::take(top); + imageHeight -= shift; + imageInts += shift * imageIntsPerLine; + } + if (left + imageWidth > _frameWidth) { + imageWidth = _frameWidth - left; + } + if (top + imageHeight > _frameHeight) { + imageHeight = _frameHeight - top; + } + if (imageWidth < 0 || imageHeight < 0) return; + + auto frameInts = _frameInts + top * _frameIntsPerLine + left; + auto frameIntsPerLineAdd = _frameIntsPerLine - imageWidth; + for (auto y = 0; y != imageHeight; ++y) { + for (auto x = 0; x != imageWidth; ++x) { + auto source = *frameInts; + auto shadowAlpha = qMax(_frameAlpha - int(source >> 24), 0); + *frameInts = anim::unshifted(anim::shifted(source) * 256 + anim::shifted(*imageInts) * shadowAlpha); + ++frameInts; + ++imageInts; + } + frameInts += frameIntsPerLineAdd; + imageInts += imageIntsPerLineAdded; + } +} + +void RoundShadowAnimation::paintShadowVertical(int left, int top, int bottom, const QImage &image) { + auto imageWidth = image.width(); + auto imageInts = reinterpret_cast(image.constBits()); + if (left < 0) { + auto shift = -base::take(left); + imageWidth -= shift; + imageInts += shift; + } + if (top < 0) top = 0; + accumulate_min(bottom, _frameHeight); + accumulate_min(imageWidth, _frameWidth - left); + if (imageWidth < 0 || bottom <= top) return; + + auto frameInts = _frameInts + top * _frameIntsPerLine + left; + auto frameIntsPerLineAdd = _frameIntsPerLine - imageWidth; + for (auto y = top; y != bottom; ++y) { + for (auto x = 0; x != imageWidth; ++x) { + auto source = *frameInts; + auto shadowAlpha = _frameAlpha - (source >> 24); + *frameInts = anim::unshifted(anim::shifted(source) * 256 + anim::shifted(*imageInts) * shadowAlpha); + ++frameInts; + ++imageInts; + } + frameInts += frameIntsPerLineAdd; + imageInts -= imageWidth; + } +} + +void RoundShadowAnimation::paintShadowHorizontal(int left, int right, int top, const QImage &image) { + auto imageHeight = image.height(); + auto imageInts = reinterpret_cast(image.constBits()); + auto imageIntsPerLine = (image.bytesPerLine() >> 2); + if (top < 0) { + auto shift = -base::take(top); + imageHeight -= shift; + imageInts += shift * imageIntsPerLine; + } + if (left < 0) left = 0; + accumulate_min(right, _frameWidth); + accumulate_min(imageHeight, _frameHeight - top); + if (imageHeight < 0 || right <= left) return; + + auto frameInts = _frameInts + top * _frameIntsPerLine + left; + auto frameIntsPerLineAdd = _frameIntsPerLine - (right - left); + for (auto y = 0; y != imageHeight; ++y) { + auto imagePattern = anim::shifted(*imageInts); + for (auto x = left; x != right; ++x) { + auto source = *frameInts; + auto shadowAlpha = _frameAlpha - (source >> 24); + *frameInts = anim::unshifted(anim::shifted(source) * 256 + imagePattern * shadowAlpha); + ++frameInts; + } + frameInts += frameIntsPerLineAdd; + imageInts += imageIntsPerLine; + } +} + +void PanelAnimation::setFinalImage(QImage &&finalImage, QRect inner) { + t_assert(!started()); + _finalImage = QPixmap::fromImage(std_::move(finalImage).convertToFormat(QImage::Format_ARGB32_Premultiplied), Qt::ColorOnly); + + t_assert(!_finalImage.isNull()); + _finalWidth = _finalImage.width(); + _finalHeight = _finalImage.height(); + t_assert(!(_finalWidth % cIntRetinaFactor())); + t_assert(!(_finalHeight % cIntRetinaFactor())); + _finalInnerLeft = inner.x(); + _finalInnerTop = inner.y(); + _finalInnerWidth = inner.width(); + _finalInnerHeight = inner.height(); + t_assert(!(_finalInnerLeft % cIntRetinaFactor())); + t_assert(!(_finalInnerTop % cIntRetinaFactor())); + t_assert(!(_finalInnerWidth % cIntRetinaFactor())); + t_assert(!(_finalInnerHeight % cIntRetinaFactor())); + _finalInnerRight = _finalInnerLeft + _finalInnerWidth; + _finalInnerBottom = _finalInnerTop + _finalInnerHeight; + t_assert(QRect(0, 0, _finalWidth, _finalHeight).contains(inner)); + + setStartWidth(); + setStartHeight(); + setStartAlpha(); + setStartFadeTop(); + createFadeMask(); + setWidthDuration(); + setHeightDuration(); + setAlphaDuration(); + if (!_skipShadow) { + setShadow(_st.shadow); + } + + auto checkCorner = [this, inner](Corner &corner) { + if (!corner.valid()) return; + if ((_startWidth >= 0 && _startWidth < _finalWidth) + || (_startHeight >= 0 && _startHeight < _finalHeight)) { + t_assert(corner.width <= inner.width()); + t_assert(corner.height <= inner.height()); + } + }; + checkCorner(_topLeft); + checkCorner(_topRight); + checkCorner(_bottomLeft); + checkCorner(_bottomRight); +} + void PanelAnimation::setStartWidth() { _startWidth = qRound(_st.startWidth * _finalInnerWidth); if (_startWidth >= 0) t_assert(_startWidth <= _finalInnerWidth); @@ -120,30 +295,30 @@ void PanelAnimation::setStartFadeTop() { void PanelAnimation::createFadeMask() { auto resultHeight = qRound(_finalImage.height() * _st.fadeHeight); + if (auto remove = (resultHeight % cIntRetinaFactor())) { + resultHeight -= remove; + } auto finalAlpha = qRound(_st.fadeOpacity * 255); t_assert(finalAlpha >= 0 && finalAlpha < 256); - auto result = QImage(1, resultHeight, QImage::Format_ARGB32_Premultiplied); + auto result = QImage(cIntRetinaFactor(), resultHeight, QImage::Format_ARGB32_Premultiplied); auto ints = reinterpret_cast(result.bits()); - auto intsPerLine = (result.bytesPerLine() >> 2); + auto intsPerLineAdded = (result.bytesPerLine() >> 2) - cIntRetinaFactor(); auto up = (_origin == PanelAnimation::Origin::BottomLeft || _origin == PanelAnimation::Origin::BottomRight); auto from = up ? resultHeight : 0, to = resultHeight - from, delta = up ? -1 : 1; + auto fadeFirstAlpha = up ? (finalAlpha + 1) : 1; + auto fadeLastAlpha = up ? 1 : (finalAlpha + 1); + _fadeFirst = QBrush(QColor(_st.fadeBg->c.red(), _st.fadeBg->c.green(), _st.fadeBg->c.blue(), (_st.fadeBg->c.alpha() * fadeFirstAlpha) >> 8)); + _fadeLast = QBrush(QColor(_st.fadeBg->c.red(), _st.fadeBg->c.green(), _st.fadeBg->c.blue(), (_st.fadeBg->c.alpha() * fadeLastAlpha) >> 8)); for (auto y = from; y != to; y += delta) { auto alpha = static_cast(finalAlpha * y) / resultHeight; - *ints = (0xFFU << 24) | (alpha << 16) | (alpha << 8) | alpha; - ints += intsPerLine; + auto value = (0xFFU << 24) | (alpha << 16) | (alpha << 8) | alpha; + for (auto x = 0; x != cIntRetinaFactor(); ++x) { + *ints++ = value; + } + ints += intsPerLineAdded; } - _fadeMask = style::colorizeImage(result, _st.fadeBg); + _fadeMask = QPixmap::fromImage(style::colorizeImage(result, _st.fadeBg), Qt::ColorOnly); _fadeHeight = _fadeMask.height(); - _fadeInts = reinterpret_cast(_fadeMask.constBits()); - _fadeIntsPerLine = (_fadeMask.bytesPerLine() >> 2); - t_assert(_fadeMask.bytesPerLine() == (_fadeIntsPerLine << 2)); -} - -void PanelAnimation::setCornerMasks(QImage &&topLeft, QImage &&topRight, QImage &&bottomLeft, QImage &&bottomRight) { - setCornerMask(_topLeft, std_::move(topLeft)); - setCornerMask(_topRight, std_::move(topRight)); - setCornerMask(_bottomLeft, std_::move(bottomLeft)); - setCornerMask(_bottomRight, std_::move(bottomRight)); } void PanelAnimation::setSkipShadow(bool skipShadow) { @@ -151,41 +326,6 @@ void PanelAnimation::setSkipShadow(bool skipShadow) { _skipShadow = skipShadow; } -void PanelAnimation::setCornerMask(Corner &corner, QImage &&image) { - t_assert(!started()); - corner.image = std_::move(image); - if (corner.valid()) { - corner.width = corner.image.width(); - corner.height = corner.image.height(); - corner.bytes = corner.image.constBits(); - corner.bytesPerPixel = (corner.image.depth() >> 3); - corner.bytesPerLineAdded = corner.image.bytesPerLine() - corner.width * corner.bytesPerPixel; - t_assert(corner.image.depth() == (corner.bytesPerPixel << 3)); - t_assert(corner.bytesPerLineAdded >= 0); - if (_startWidth >= 0) t_assert(corner.width <= _startWidth); - if (_startHeight >= 0) t_assert(corner.height <= _startHeight); - if (!_finalImage.isNull()) { - t_assert(corner.width <= _finalInnerWidth); - t_assert(corner.height <= _finalInnerHeight); - } - } else { - corner.width = corner.height = 0; - } -} - -QImage PanelAnimation::cloneImage(const style::icon &source) { - if (source.empty()) return QImage(); - - auto result = QImage(source.size() * cIntRetinaFactor(), QImage::Format_ARGB32_Premultiplied); - result.setDevicePixelRatio(cRetinaFactor()); - result.fill(Qt::transparent); - { - Painter p(&result); - source.paint(p, 0, 0, source.width()); - } - return std_::move(result); -} - void PanelAnimation::setWidthDuration() { _widthDuration = _st.widthDuration; t_assert(_widthDuration >= 0.); @@ -207,35 +347,46 @@ void PanelAnimation::setAlphaDuration() { } void PanelAnimation::start() { - t_assert(!started()); t_assert(!_finalImage.isNull()); - _frame = QImage(_finalWidth, _finalHeight, QImage::Format_ARGB32_Premultiplied); - _frame.setDevicePixelRatio(_finalImage.devicePixelRatio()); - _frameIntsPerLine = (_frame.bytesPerLine() >> 2); - _frameInts = reinterpret_cast(_frame.bits()); - _frameIntsPerLineAdded = _frameIntsPerLine - _finalWidth; - t_assert(_frame.depth() == static_cast(sizeof(uint32) << 3)); - t_assert(_frame.bytesPerLine() == (_frameIntsPerLine << 2)); - t_assert(_frameIntsPerLineAdded >= 0); + RoundShadowAnimation::start(_finalWidth, _finalHeight, _finalImage.devicePixelRatio()); + auto checkCorner = [this](const Corner &corner) { + if (!corner.valid()) return; + if (_startWidth >= 0) t_assert(corner.width <= _startWidth); + if (_startHeight >= 0) t_assert(corner.height <= _startHeight); + t_assert(corner.width <= _finalInnerWidth); + t_assert(corner.height <= _finalInnerHeight); + }; + checkCorner(_topLeft); + checkCorner(_topRight); + checkCorner(_bottomLeft); + checkCorner(_bottomRight); } -const QImage &PanelAnimation::getFrame(float64 dt, float64 opacity) { +void PanelAnimation::paintFrame(QPainter &p, int x, int y, int outerWidth, float64 dt, float64 opacity) { t_assert(started()); t_assert(dt >= 0.); auto &transition = anim::easeOutCirc; - constexpr auto finalAlpha = 256; - auto alpha = (dt >= _alphaDuration) ? finalAlpha : anim::interpolate(_startAlpha + 1, finalAlpha, transition(1., dt / _alphaDuration)); - _frameAlpha = anim::interpolate(0, alpha, opacity); + if (dt < _alphaDuration) opacity *= transition(1., dt / _alphaDuration); + _frameAlpha = anim::interpolate(1, 256, opacity); auto frameWidth = (_startWidth < 0 || dt >= _widthDuration) ? _finalInnerWidth : anim::interpolate(_startWidth, _finalInnerWidth, transition(1., dt / _widthDuration)); auto frameHeight = (_startHeight < 0 || dt >= _heightDuration) ? _finalInnerHeight : anim::interpolate(_startHeight, _finalInnerHeight, transition(1., dt / _heightDuration)); + if (auto decrease = (frameWidth % cIntRetinaFactor())) { + frameWidth -= decrease; + } + if (auto decrease = (frameHeight % cIntRetinaFactor())) { + frameHeight -= decrease; + } auto frameLeft = (_origin == Origin::TopLeft || _origin == Origin::BottomLeft) ? _finalInnerLeft : (_finalInnerRight - frameWidth); auto frameTop = (_origin == Origin::TopLeft || _origin == Origin::TopRight) ? _finalInnerTop : (_finalInnerBottom - frameHeight); auto frameRight = frameLeft + frameWidth; auto frameBottom = frameTop + frameHeight; auto fadeTop = (_fadeHeight > 0) ? snap(anim::interpolate(_startFadeTop, _finalInnerHeight, transition(1., dt)), 0, frameHeight) : frameHeight; + if (auto decrease = (fadeTop % cIntRetinaFactor())) { + fadeTop -= decrease; + } auto fadeBottom = (fadeTop < frameHeight) ? qMin(fadeTop + _fadeHeight, frameHeight) : frameHeight; auto fadeSkipLines = 0; if (_origin == Origin::BottomLeft || _origin == Origin::BottomRight) { @@ -247,97 +398,97 @@ const QImage &PanelAnimation::getFrame(float64 dt, float64 opacity) { fadeTop += frameTop; fadeBottom += frameTop; - auto finalInts = _finalInts + frameLeft + frameTop * _finalIntsPerLine; + if (opacity < 1.) { + _frame.fill(Qt::transparent); + } + { + Painter p(&_frame); + p.setOpacity(opacity); + auto painterFrameLeft = frameLeft / cIntRetinaFactor(); + auto painterFrameTop = frameTop / cIntRetinaFactor(); + auto painterFadeBottom = fadeBottom / cIntRetinaFactor(); + p.drawPixmap(painterFrameLeft, painterFrameTop, _finalImage, frameLeft, frameTop, frameWidth, frameHeight); + if (_fadeHeight) { + if (frameTop != fadeTop) { + p.fillRect(painterFrameLeft, painterFrameTop, frameWidth, fadeTop - frameTop, _fadeFirst); + } + if (fadeTop != fadeBottom) { + auto painterFadeTop = fadeTop / cIntRetinaFactor(); + auto painterFrameWidth = frameWidth / cIntRetinaFactor(); + auto painterFrameHeight = frameHeight / cIntRetinaFactor(); + p.drawPixmap(painterFrameLeft, painterFadeTop, painterFrameWidth, painterFadeBottom - painterFadeTop, _fadeMask, 0, fadeSkipLines, cIntRetinaFactor(), fadeBottom - fadeTop); + } + if (fadeBottom != frameBottom) { + p.fillRect(painterFrameLeft, painterFadeBottom, frameWidth, frameBottom - fadeBottom, _fadeLast); + } + } + } auto frameInts = _frameInts + frameLeft + frameTop * _frameIntsPerLine; - auto finalIntsPerLineAdd = (_finalWidth - frameWidth) + _finalIntsPerLineAdded; auto frameIntsPerLineAdd = (_finalWidth - frameWidth) + _frameIntsPerLineAdded; - // Draw frameWidth x fadeTop with fade first color. - auto fadeInts = _fadeInts + fadeSkipLines * _fadeIntsPerLine; - auto fadeWithMasterAlpha = [this](uint32 fade) { - auto fadeAlphaAddition = (256 - (fade >> 24)); - auto fadePattern = anim::shifted(fade); - return [this, fadeAlphaAddition, fadePattern](uint32 source) { - auto sourceAlpha = (source >> 24) + 1; - auto sourcePattern = anim::shifted(source); - auto mixedPattern = anim::reshifted(fadePattern * sourceAlpha + sourcePattern * fadeAlphaAddition); - return anim::unshifted(mixedPattern * _frameAlpha); - }; - }; - if (frameTop != fadeTop) { - // Take the fade components from the first line of the fade mask. - auto withMasterAlpha = fadeWithMasterAlpha(_fadeInts ? *_fadeInts : 0); - for (auto y = frameTop; y != fadeTop; ++y) { - for (auto x = frameLeft; x != frameRight; ++x) { - *frameInts++ = withMasterAlpha(*finalInts++); - } - finalInts += finalIntsPerLineAdd; - frameInts += frameIntsPerLineAdd; - } - } - - // Draw frameWidth x (fadeBottom - fadeTop) with fade gradient. - for (auto y = fadeTop; y != fadeBottom; ++y) { - auto withMasterAlpha = fadeWithMasterAlpha(*fadeInts); - for (auto x = frameLeft; x != frameRight; ++x) { - *frameInts++ = withMasterAlpha(*finalInts++); - } - finalInts += finalIntsPerLineAdd; - frameInts += frameIntsPerLineAdd; - fadeInts += _fadeIntsPerLine; - } - - // Draw frameWidth x (frameBottom - fadeBottom) with fade final color. - if (fadeBottom != frameBottom) { - // Take the fade components from the last line of the fade mask. - auto withMasterAlpha = fadeWithMasterAlpha(*(fadeInts - _fadeIntsPerLine)); - for (auto y = fadeBottom; y != frameBottom; ++y) { - for (auto x = frameLeft; x != frameRight; ++x) { - *frameInts++ = withMasterAlpha(*finalInts++); - } - finalInts += finalIntsPerLineAdd; - frameInts += frameIntsPerLineAdd; - } - } - // Draw corners - auto innerLeft = qMax(_finalInnerLeft, frameLeft); - auto innerTop = qMax(_finalInnerTop, frameTop); - auto innerRight = qMin(_finalInnerRight, frameRight); - auto innerBottom = qMin(_finalInnerBottom, frameBottom); - if (innerLeft != _finalInnerLeft || innerTop != _finalInnerTop) { - paintCorner(_topLeft, innerLeft, innerTop); - } - if (innerRight != _finalInnerRight || innerTop != _finalInnerTop) { - paintCorner(_topRight, innerRight - _topRight.width, innerTop); - } - if (innerLeft != _finalInnerLeft || innerBottom != _finalInnerBottom) { - paintCorner(_bottomLeft, innerLeft, innerBottom - _bottomLeft.height); - } - if (innerRight != _finalInnerRight || innerBottom != _finalInnerBottom) { - paintCorner(_bottomRight, innerRight - _bottomRight.width, innerBottom - _bottomRight.height); - } + paintCorner(_topLeft, frameLeft, frameTop); + paintCorner(_topRight, frameRight - _topRight.width, frameTop); + paintCorner(_bottomLeft, frameLeft, frameBottom - _bottomLeft.height); + paintCorner(_bottomRight, frameRight - _bottomRight.width, frameBottom - _bottomRight.height); - // Fill the rest with transparent - if (frameTop) { - memset(_frameInts, 0, _frameIntsPerLine * frameTop * sizeof(uint32)); + // Draw shadow upon the transparent + auto outerLeft = frameLeft; + auto outerTop = frameTop; + auto outerRight = frameRight; + auto outerBottom = frameBottom; + if (_shadow.valid()) { + outerLeft -= _shadow.extend.left(); + outerTop -= _shadow.extend.top(); + outerRight += _shadow.extend.right(); + outerBottom += _shadow.extend.bottom(); } - auto widthLeft = (_finalWidth - frameRight); - if (frameLeft || widthLeft) { - auto frameInts = _frameInts + frameTop * _frameIntsPerLine; - for (auto y = frameTop; y != frameBottom; ++y) { - memset(frameInts, 0, frameLeft * sizeof(uint32)); - memset(frameInts + frameLeft + frameWidth, 0, widthLeft * sizeof(uint32)); - frameInts += _frameIntsPerLine; + if (cIntRetinaFactor() > 1) { + if (auto skipLeft = (outerLeft % cIntRetinaFactor())) { + outerLeft -= skipLeft; + } + if (auto skipTop = (outerTop % cIntRetinaFactor())) { + outerTop -= skipTop; + } + if (auto skipRight = (outerRight % cIntRetinaFactor())) { + outerRight += (cIntRetinaFactor() - skipRight); + } + if (auto skipBottom = (outerBottom % cIntRetinaFactor())) { + outerBottom += (cIntRetinaFactor() - skipBottom); } } - if (auto heightLeft = (_finalHeight - frameBottom)) { - memset(_frameInts + frameBottom * _frameIntsPerLine, 0, _frameIntsPerLine * heightLeft * sizeof(uint32)); + + if (opacity == 1.) { + // Fill above the frame top with transparent. + auto fillTopInts = (_frameInts + outerTop * _frameIntsPerLine + outerLeft); + auto fillWidth = (outerRight - outerLeft) * sizeof(uint32); + for (auto fillTop = frameTop - outerTop; fillTop != 0; --fillTop) { + memset(fillTopInts, 0, fillWidth); + fillTopInts += _frameIntsPerLine; + } + + // Fill to the left and to the right of the frame with transparent. + auto fillLeft = (frameLeft - outerLeft) * sizeof(uint32); + auto fillRight = (outerRight - frameRight) * sizeof(uint32); + if (fillLeft || fillRight) { + auto fillInts = _frameInts + frameTop * _frameIntsPerLine; + for (auto y = frameTop; y != frameBottom; ++y) { + memset(fillInts + outerLeft, 0, fillLeft); + memset(fillInts + frameRight, 0, fillRight); + fillInts += _frameIntsPerLine; + } + } + + // Fill below the frame bottom with transparent. + auto fillBottomInts = (_frameInts + frameBottom * _frameIntsPerLine + outerLeft); + for (auto fillBottom = outerBottom - frameBottom; fillBottom != 0; --fillBottom) { + memset(fillBottomInts, 0, fillWidth); + fillBottomInts += _frameIntsPerLine; + } } - // Draw shadow if (_shadow.valid()) { - paintShadow(innerLeft, innerTop, innerRight, innerBottom); + paintShadow(outerLeft, outerTop, outerRight, outerBottom); } // Debug @@ -353,140 +504,7 @@ const QImage &PanelAnimation::getFrame(float64 dt, float64 opacity) { // frameInts += _frameIntsPerLineAdded; //} - return _frame; -} - -void PanelAnimation::paintCorner(Corner &corner, int left, int top) { - auto mask = corner.bytes; - auto bytesPerPixel = corner.bytesPerPixel; - auto bytesPerLineAdded = corner.bytesPerLineAdded; - auto frameInts = _frameInts + top * _frameIntsPerLine + left; - auto frameIntsPerLineAdd = _frameIntsPerLine - corner.width; - for (auto y = 0; y != corner.height; ++y) { - for (auto x = 0; x != corner.width; ++x) { - auto alpha = static_cast(*mask) + 1; - *frameInts = anim::unshifted(anim::shifted(*frameInts) * alpha); - ++frameInts; - mask += bytesPerPixel; - } - frameInts += frameIntsPerLineAdd; - mask += bytesPerLineAdded; - } -} - -void PanelAnimation::paintShadow(int left, int top, int right, int bottom) { - left -= _shadow.extend.left(); - top -= _shadow.extend.top(); - right += _shadow.extend.right(); - bottom += _shadow.extend.bottom(); - paintShadowCorner(left, top, _shadow.topLeft); - paintShadowCorner(right - _shadow.topRight.width(), top, _shadow.topRight); - paintShadowCorner(right - _shadow.bottomRight.width(), bottom - _shadow.bottomRight.height(), _shadow.bottomRight); - paintShadowCorner(left, bottom - _shadow.bottomLeft.height(), _shadow.bottomLeft); - paintShadowVertical(left, top + _shadow.topLeft.height(), bottom - _shadow.bottomLeft.height(), _shadow.left); - paintShadowVertical(right - _shadow.right.width(), top + _shadow.topRight.height(), bottom - _shadow.bottomRight.height(), _shadow.right); - paintShadowHorizontal(left + _shadow.topLeft.width(), right - _shadow.topRight.width(), top, _shadow.top); - paintShadowHorizontal(left + _shadow.bottomLeft.width(), right - _shadow.bottomRight.width(), bottom - _shadow.bottom.height(), _shadow.bottom); -} - -void PanelAnimation::paintShadowCorner(int left, int top, const QImage &image) { - auto imageWidth = image.width(); - auto imageHeight = image.height(); - auto imageInts = reinterpret_cast(image.constBits()); - auto imageIntsPerLine = (image.bytesPerLine() >> 2); - auto imageIntsPerLineAdded = imageIntsPerLine - imageWidth; - if (left < 0) { - auto shift = -base::take(left); - imageWidth -= shift; - imageInts += shift; - } - if (top < 0) { - auto shift = -base::take(top); - imageHeight -= shift; - imageInts += shift * imageIntsPerLine; - } - if (left + imageWidth > _finalWidth) { - imageWidth = _finalWidth - left; - } - if (top + imageHeight > _finalHeight) { - imageHeight = _finalHeight - top; - } - if (imageWidth < 0 || imageHeight < 0) return; - - auto frameInts = _frameInts + top * _frameIntsPerLine + left; - auto frameIntsPerLineAdd = _frameIntsPerLine - imageWidth; - for (auto y = 0; y != imageHeight; ++y) { - for (auto x = 0; x != imageWidth; ++x) { - auto source = *frameInts; - auto shadowAlpha = _frameAlpha - (source >> 24); - *frameInts = anim::unshifted(anim::shifted(source) * 256 + anim::shifted(*imageInts) * shadowAlpha); - ++frameInts; - ++imageInts; - } - frameInts += frameIntsPerLineAdd; - imageInts += imageIntsPerLineAdded; - } -} - -void PanelAnimation::paintShadowVertical(int left, int top, int bottom, const QImage &image) { - auto imageWidth = image.width(); - auto imageInts = reinterpret_cast(image.constBits()); - if (left < 0) { - auto shift = -base::take(left); - imageWidth -= shift; - imageInts += shift; - } - if (top < 0) top = 0; - if (bottom > _finalHeight) bottom = _finalHeight; - if (left + imageWidth > _finalWidth) { - imageWidth = _finalWidth - left; - } - if (imageWidth < 0 || bottom <= top) return; - - auto frameInts = _frameInts + top * _frameIntsPerLine + left; - auto frameIntsPerLineAdd = _frameIntsPerLine - imageWidth; - for (auto y = top; y != bottom; ++y) { - for (auto x = 0; x != imageWidth; ++x) { - auto source = *frameInts; - auto shadowAlpha = _frameAlpha - (source >> 24); - *frameInts = anim::unshifted(anim::shifted(source) * 256 + anim::shifted(*imageInts) * shadowAlpha); - ++frameInts; - ++imageInts; - } - frameInts += frameIntsPerLineAdd; - imageInts -= imageWidth; - } -} - -void PanelAnimation::paintShadowHorizontal(int left, int right, int top, const QImage &image) { - auto imageHeight = image.height(); - auto imageInts = reinterpret_cast(image.constBits()); - auto imageIntsPerLine = (image.bytesPerLine() >> 2); - if (top < 0) { - auto shift = -base::take(top); - imageHeight -= shift; - imageInts += shift * imageIntsPerLine; - } - if (left < 0) left = 0; - if (right > _finalWidth) right = _finalWidth; - if (top + imageHeight > _finalHeight) { - imageHeight = _finalHeight - top; - } - if (imageHeight < 0 || right <= left) return; - - auto frameInts = _frameInts + top * _frameIntsPerLine + left; - auto frameIntsPerLineAdd = _frameIntsPerLine - (right - left); - for (auto y = 0; y != imageHeight; ++y) { - auto imagePattern = anim::shifted(*imageInts); - for (auto x = left; x != right; ++x) { - auto source = *frameInts; - auto shadowAlpha = _frameAlpha - (source >> 24); - *frameInts = anim::unshifted(anim::shifted(source) * 256 + imagePattern * shadowAlpha); - ++frameInts; - } - frameInts += frameIntsPerLineAdd; - imageInts += imageIntsPerLine; - } + p.drawImage(rtlpoint(x + (outerLeft / cIntRetinaFactor()), y + (outerTop / cIntRetinaFactor()), outerWidth), _frame, QRect(outerLeft, outerTop, outerRight - outerLeft, outerBottom - outerTop)); } } // namespace Ui diff --git a/Telegram/SourceFiles/ui/effects/panel_animation.h b/Telegram/SourceFiles/ui/effects/panel_animation.h index 988315483..1e7af2caa 100644 --- a/Telegram/SourceFiles/ui/effects/panel_animation.h +++ b/Telegram/SourceFiles/ui/effects/panel_animation.h @@ -24,34 +24,13 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org namespace Ui { -class PanelAnimation { +class RoundShadowAnimation { public: - enum class Origin { - TopLeft, - TopRight, - BottomLeft, - BottomRight, - }; - PanelAnimation(const style::PanelAnimation &st, Origin origin) : _st(st), _origin(origin) { - } - - void setFinalImage(QImage &&finalImage, QRect inner); void setCornerMasks(QImage &&topLeft, QImage &&topRight, QImage &&bottomLeft, QImage &&bottomRight); - void setSkipShadow(bool skipShadow); - void start(); - const QImage &getFrame(float64 dt, float64 opacity); - -private: - void setStartWidth(); - void setStartHeight(); - void setStartAlpha(); - void setStartFadeTop(); - void createFadeMask(); - void setWidthDuration(); - void setHeightDuration(); - void setAlphaDuration(); - void setShadow(); +protected: + void start(int frameWidth, int frameHeight, float64 devicePixelRatio); + void setShadow(const style::Shadow &st); bool started() const { return !_frame.isNull(); @@ -86,13 +65,54 @@ private: void paintShadowVertical(int left, int top, int bottom, const QImage &image); void paintShadowHorizontal(int left, int right, int top, const QImage &image); + Shadow _shadow; + + Corner _topLeft; + Corner _topRight; + Corner _bottomLeft; + Corner _bottomRight; + + QImage _frame; + uint32 *_frameInts = nullptr; + int _frameWidth = 0; + int _frameHeight = 0; + int _frameAlpha = 0; // recounted each getFrame() + int _frameIntsPerLine = 0; + int _frameIntsPerLineAdded = 0; + +}; + +class PanelAnimation : public RoundShadowAnimation { +public: + enum class Origin { + TopLeft, + TopRight, + BottomLeft, + BottomRight, + }; + PanelAnimation(const style::PanelAnimation &st, Origin origin) : _st(st), _origin(origin) { + } + + void setFinalImage(QImage &&finalImage, QRect inner); + void setSkipShadow(bool skipShadow); + + void start(); + void paintFrame(QPainter &p, int x, int y, int outerWidth, float64 dt, float64 opacity); + +private: + void setStartWidth(); + void setStartHeight(); + void setStartAlpha(); + void setStartFadeTop(); + void createFadeMask(); + void setWidthDuration(); + void setHeightDuration(); + void setAlphaDuration(); + const style::PanelAnimation &_st; Origin _origin = Origin::TopLeft; - QImage _finalImage; - const uint32 *_finalInts = nullptr; - int _finalIntsPerLine = 0; - int _finalIntsPerLineAdded = 0; + QPixmap _finalImage; int _finalWidth = 0; int _finalHeight = 0; int _finalInnerLeft = 0; @@ -102,33 +122,20 @@ private: int _finalInnerWidth = 0; int _finalInnerHeight = 0; - Shadow _shadow; bool _skipShadow = false; int _startWidth = -1; int _startHeight = -1; int _startAlpha = 0; int _startFadeTop = 0; - QImage _fadeMask; + QPixmap _fadeMask; int _fadeHeight = 0; - const uint32 *_fadeInts = nullptr; - int _fadeIntsPerLine = 0; - - Corner _topLeft; - Corner _topRight; - Corner _bottomLeft; - Corner _bottomRight; + QBrush _fadeFirst, _fadeLast; float64 _widthDuration = 1.; float64 _heightDuration = 1.; float64 _alphaDuration = 1.; - QImage _frame; - uint32 *_frameInts = nullptr; - int _frameAlpha = 0; // recounted each getFrame() - int _frameIntsPerLine = 0; - int _frameIntsPerLineAdded = 0; - }; } // namespace Ui diff --git a/Telegram/SourceFiles/ui/style/style_core.cpp b/Telegram/SourceFiles/ui/style/style_core.cpp index 98ff07b11..f375af726 100644 --- a/Telegram/SourceFiles/ui/style/style_core.cpp +++ b/Telegram/SourceFiles/ui/style/style_core.cpp @@ -85,15 +85,7 @@ void colorizeImage(const QImage &src, QColor c, QImage *outResult, QRect srcRect auto height = srcRect.height(); t_assert(outResult && outResult->rect().contains(QRect(dstPoint, srcRect.size()))); - auto initialAlpha = c.alpha() + 1; - auto red = (c.red() * initialAlpha) >> 8; - auto green = (c.green() * initialAlpha) >> 8; - auto blue = (c.blue() * initialAlpha) >> 8; - auto alpha = (255 * initialAlpha) >> 8; - auto pattern = static_cast(blue) - | (static_cast(green) << 16) - | (static_cast(red) << 32) - | (static_cast(alpha) << 48); + auto pattern = anim::shifted(c); auto resultBytesPerPixel = (src.depth() >> 3); auto resultIntsPerPixel = 1; diff --git a/Telegram/SourceFiles/ui/widgets/inner_dropdown.cpp b/Telegram/SourceFiles/ui/widgets/inner_dropdown.cpp index bc0c2d450..7f0a4f37f 100644 --- a/Telegram/SourceFiles/ui/widgets/inner_dropdown.cpp +++ b/Telegram/SourceFiles/ui/widgets/inner_dropdown.cpp @@ -110,7 +110,7 @@ void InnerDropdown::paintEvent(QPaintEvent *e) { auto ms = getms(); if (_a_show.animating(ms)) { if (auto opacity = _a_opacity.current(ms, _hiding ? 0. : 1.)) { - p.drawImage(0, 0, _showAnimation->getFrame(_a_show.current(1.), opacity)); + _showAnimation->paintFrame(p, 0, 0, width(), _a_show.current(1.), opacity); } } else if (_a_opacity.animating(ms)) { p.setOpacity(_a_opacity.current(0.)); @@ -118,10 +118,11 @@ void InnerDropdown::paintEvent(QPaintEvent *e) { } else if (_hiding || isHidden()) { hideFinished(); } else if (_showAnimation) { - p.drawImage(0, 0, _showAnimation->getFrame(1., 1.)); + _showAnimation->paintFrame(p, 0, 0, width(), 1., 1.); _showAnimation.reset(); showChildren(); } else { + if (!_cache.isNull()) _cache = QPixmap(); auto inner = rect().marginsRemoved(_st.padding); Shadow::paint(p, inner, width(), _st.shadow); App::roundRect(p, inner, _st.bg, ImageRoundRadius::Small); @@ -207,6 +208,9 @@ void InnerDropdown::prepareCache() { _cache = myGrab(this); _showAnimation = base::take(showAnimationData); _a_show = base::take(showAnimation); + if (_a_show.animating()) { + hideChildren(); + } } void InnerDropdown::startOpacityAnimation(bool hiding) { @@ -270,7 +274,7 @@ void InnerDropdown::opacityAnimationCallback() { if (_hiding) { _hiding = false; hideFinished(); - } else { + } else if (!_a_show.animating()) { showChildren(); } } diff --git a/Telegram/SourceFiles/ui/widgets/popup_menu.cpp b/Telegram/SourceFiles/ui/widgets/popup_menu.cpp index 91772619f..d83687937 100644 --- a/Telegram/SourceFiles/ui/widgets/popup_menu.cpp +++ b/Telegram/SourceFiles/ui/widgets/popup_menu.cpp @@ -109,7 +109,7 @@ void PopupMenu::paintEvent(QPaintEvent *e) { auto ms = getms(); if (_a_show.animating(ms)) { if (auto opacity = _a_opacity.current(ms, _hiding ? 0. : 1.)) { - p.drawImage(0, 0, _showAnimation->getFrame(_a_show.current(1.), opacity)); + _showAnimation->paintFrame(p, 0, 0, width(), _a_show.current(1.), opacity); } } else if (_a_opacity.animating(ms)) { p.setOpacity(_a_opacity.current(0.)); @@ -117,7 +117,7 @@ void PopupMenu::paintEvent(QPaintEvent *e) { } else if (_hiding || isHidden()) { hideFinished(); } else if (_showAnimation) { - p.drawImage(0, 0, _showAnimation->getFrame(1., 1.)); + _showAnimation->paintFrame(p, 0, 0, width(), 1., 1.); _showAnimation.reset(); showChildren(); } else { diff --git a/Telegram/gyp/Telegram.gyp b/Telegram/gyp/Telegram.gyp index dc7818dfd..d19b437ca 100644 --- a/Telegram/gyp/Telegram.gyp +++ b/Telegram/gyp/Telegram.gyp @@ -201,6 +201,7 @@ '<(src_loc)/boxes/stickers_box.h', '<(src_loc)/boxes/usernamebox.cpp', '<(src_loc)/boxes/usernamebox.h', + '<(src_loc)/core/build_config.h', '<(src_loc)/core/basic_types.h', '<(src_loc)/core/click_handler.cpp', '<(src_loc)/core/click_handler.h',