#define NOMINMAX #include "brightray/browser/win/win32_desktop_notifications/toast.h" #include <uxtheme.h> #include <windowsx.h> #include <algorithm> #include "brightray/browser/win/win32_desktop_notifications/common.h" #pragma comment(lib, "msimg32.lib") #pragma comment(lib, "uxtheme.lib") using std::min; using std::shared_ptr; namespace brightray { static COLORREF GetAccentColor() { bool success = false; if (IsAppThemed()) { HKEY hkey; if (RegOpenKeyEx(HKEY_CURRENT_USER, TEXT("SOFTWARE\\Microsoft\\Windows\\DWM"), 0, KEY_QUERY_VALUE, &hkey) == ERROR_SUCCESS) { COLORREF color; DWORD type, size; if (RegQueryValueEx(hkey, TEXT("AccentColor"), nullptr, &type, reinterpret_cast<BYTE*>(&color), &(size = sizeof(color))) == ERROR_SUCCESS && type == REG_DWORD) { // convert from RGBA color = RGB(GetRValue(color), GetGValue(color), GetBValue(color)); success = true; } else if ( RegQueryValueEx(hkey, TEXT("ColorizationColor"), nullptr, &type, reinterpret_cast<BYTE*>(&color), &(size = sizeof(color))) == ERROR_SUCCESS && type == REG_DWORD) { // convert from BGRA color = RGB(GetBValue(color), GetGValue(color), GetRValue(color)); success = true; } RegCloseKey(hkey); if (success) return color; } } return GetSysColor(COLOR_ACTIVECAPTION); } // Stretches a bitmap to the specified size, preserves alpha channel static HBITMAP StretchBitmap(HBITMAP bitmap, unsigned width, unsigned height) { // We use StretchBlt for the scaling, but that discards the alpha channel. // So we first create a separate grayscale bitmap from the alpha channel, // scale that separately, and copy it back to the scaled color bitmap. BITMAP bm; if (!GetObject(bitmap, sizeof(bm), &bm)) return NULL; if (width == 0 || height == 0) return NULL; HBITMAP result_bitmap = NULL; HDC hdc_screen = GetDC(NULL); HBITMAP alpha_src_bitmap; { BITMAPINFOHEADER bmi = { sizeof(BITMAPINFOHEADER) }; bmi.biWidth = bm.bmWidth; bmi.biHeight = bm.bmHeight; bmi.biPlanes = bm.bmPlanes; bmi.biBitCount = bm.bmBitsPixel; bmi.biCompression = BI_RGB; void* alpha_src_bits; alpha_src_bitmap = CreateDIBSection(NULL, reinterpret_cast<BITMAPINFO*>(&bmi), DIB_RGB_COLORS, &alpha_src_bits, NULL, 0); if (alpha_src_bitmap) { if (GetDIBits(hdc_screen, bitmap, 0, 0, 0, reinterpret_cast<BITMAPINFO*>(&bmi), DIB_RGB_COLORS) && bmi.biSizeImage > 0 && (bmi.biSizeImage % 4) == 0) { auto buf = reinterpret_cast<BYTE*>( _aligned_malloc(bmi.biSizeImage, sizeof(DWORD))); if (buf) { GetDIBits(hdc_screen, bitmap, 0, bm.bmHeight, buf, reinterpret_cast<BITMAPINFO*>(&bmi), DIB_RGB_COLORS); const DWORD *src = reinterpret_cast<DWORD*>(buf); const DWORD *end = reinterpret_cast<DWORD*>(buf + bmi.biSizeImage); BYTE* dest = reinterpret_cast<BYTE*>(alpha_src_bits); for (; src != end; ++src, ++dest) { BYTE a = *src >> 24; *dest++ = a; *dest++ = a; *dest++ = a; } _aligned_free(buf); } } } } if (alpha_src_bitmap) { BITMAPINFOHEADER bmi = { sizeof(BITMAPINFOHEADER) }; bmi.biWidth = width; bmi.biHeight = height; bmi.biPlanes = 1; bmi.biBitCount = 32; bmi.biCompression = BI_RGB; void* color_bits; auto color_bitmap = CreateDIBSection(NULL, reinterpret_cast<BITMAPINFO*>(&bmi), DIB_RGB_COLORS, &color_bits, NULL, 0); void* alpha_bits; auto alpha_bitmap = CreateDIBSection(NULL, reinterpret_cast<BITMAPINFO*>(&bmi), DIB_RGB_COLORS, &alpha_bits, NULL, 0); HDC hdc = CreateCompatibleDC(NULL); HDC hdc_src = CreateCompatibleDC(NULL); if (color_bitmap && alpha_bitmap && hdc && hdc_src) { SetStretchBltMode(hdc, HALFTONE); // resize color channels SelectObject(hdc, color_bitmap); SelectObject(hdc_src, bitmap); StretchBlt(hdc, 0, 0, width, height, hdc_src, 0, 0, bm.bmWidth, bm.bmHeight, SRCCOPY); // resize alpha channel SelectObject(hdc, alpha_bitmap); SelectObject(hdc_src, alpha_src_bitmap); StretchBlt(hdc, 0, 0, width, height, hdc_src, 0, 0, bm.bmWidth, bm.bmHeight, SRCCOPY); // flush before touching the bits GdiFlush(); // apply the alpha channel auto dest = reinterpret_cast<BYTE*>(color_bits); auto src = reinterpret_cast<const BYTE*>(alpha_bits); auto end = src + (width * height * 4); while (src != end) { dest[3] = src[0]; dest += 4; src += 4; } // create the resulting bitmap result_bitmap = CreateDIBitmap(hdc_screen, &bmi, CBM_INIT, color_bits, reinterpret_cast<BITMAPINFO*>(&bmi), DIB_RGB_COLORS); } if (hdc_src) DeleteDC(hdc_src); if (hdc) DeleteDC(hdc); if (alpha_bitmap) DeleteObject(alpha_bitmap); if (color_bitmap) DeleteObject(color_bitmap); DeleteObject(alpha_src_bitmap); } ReleaseDC(NULL, hdc_screen); return result_bitmap; } DesktopNotificationController::Toast::Toast( HWND hwnd, shared_ptr<NotificationData>* data) : hwnd_(hwnd), data_(*data) { HDC hdc_screen = GetDC(NULL); hdc_ = CreateCompatibleDC(hdc_screen); ReleaseDC(NULL, hdc_screen); } DesktopNotificationController::Toast::~Toast() { DeleteDC(hdc_); if (bitmap_) DeleteBitmap(bitmap_); if (scaled_image_) DeleteBitmap(scaled_image_); } void DesktopNotificationController::Toast::Register(HINSTANCE hinstance) { WNDCLASSEX wc = { sizeof(wc) }; wc.lpfnWndProc = &Toast::WndProc; wc.lpszClassName = class_name_; wc.cbWndExtra = sizeof(Toast*); wc.hInstance = hinstance; wc.hCursor = LoadCursor(NULL, IDC_ARROW); RegisterClassEx(&wc); } LRESULT DesktopNotificationController::Toast::WndProc( HWND hwnd, UINT message, WPARAM wparam, LPARAM lparam) { switch (message) { case WM_CREATE: { auto& cs = reinterpret_cast<const CREATESTRUCT*&>(lparam); auto data = static_cast<shared_ptr<NotificationData>*>(cs->lpCreateParams); auto inst = new Toast(hwnd, data); SetWindowLongPtr(hwnd, 0, (LONG_PTR)inst); } break; case WM_NCDESTROY: delete Get(hwnd); SetWindowLongPtr(hwnd, 0, 0); return 0; case WM_MOUSEACTIVATE: return MA_NOACTIVATE; case WM_TIMER: if (wparam == TimerID_AutoDismiss) { Get(hwnd)->AutoDismiss(); } return 0; case WM_LBUTTONDOWN: { auto inst = Get(hwnd); inst->Dismiss(); Notification notification(inst->data_); if (inst->is_close_hot_) inst->data_->controller->OnNotificationDismissed(notification); else inst->data_->controller->OnNotificationClicked(notification); } return 0; case WM_MOUSEMOVE: { auto inst = Get(hwnd); if (!inst->is_highlighted_) { inst->is_highlighted_ = true; TRACKMOUSEEVENT tme = { sizeof(tme), TME_LEAVE, hwnd }; TrackMouseEvent(&tme); } POINT cursor = { GET_X_LPARAM(lparam), GET_Y_LPARAM(lparam) }; inst->is_close_hot_ = (PtInRect(&inst->close_button_rect_, cursor) != FALSE); if (!inst->is_non_interactive_) inst->CancelDismiss(); inst->UpdateContents(); } return 0; case WM_MOUSELEAVE: { auto inst = Get(hwnd); inst->is_highlighted_ = false; inst->is_close_hot_ = false; inst->UpdateContents(); if (!inst->ease_out_active_ && inst->ease_in_pos_ == 1.0f) inst->ScheduleDismissal(); // Make sure stack collapse happens if needed inst->data_->controller->StartAnimation(); } return 0; case WM_WINDOWPOSCHANGED: { auto& wp = reinterpret_cast<WINDOWPOS*&>(lparam); if (wp->flags & SWP_HIDEWINDOW) { if (!IsWindowVisible(hwnd)) Get(hwnd)->is_highlighted_ = false; } } break; } return DefWindowProc(hwnd, message, wparam, lparam); } HWND DesktopNotificationController::Toast::Create( HINSTANCE hinstance, shared_ptr<NotificationData>& data) { return CreateWindowEx(WS_EX_LAYERED | WS_EX_NOACTIVATE | WS_EX_TOPMOST, class_name_, nullptr, WS_POPUP, 0, 0, 0, 0, NULL, NULL, hinstance, &data); } void DesktopNotificationController::Toast::Draw() { const COLORREF accent = GetAccentColor(); COLORREF back_color; { // base background color is 2/3 of accent // highlighted adds a bit of intensity to every channel int h = is_highlighted_ ? (0xff / 20) : 0; back_color = RGB(min(0xff, (GetRValue(accent) * 2 / 3) + h), min(0xff, (GetGValue(accent) * 2 / 3) + h), min(0xff, (GetBValue(accent) * 2 / 3) + h)); } const float back_luma = (GetRValue(back_color) * 0.299f / 255) + (GetGValue(back_color) * 0.587f / 255) + (GetBValue(back_color) * 0.114f / 255); const struct { float r, g, b; } back_f = { GetRValue(back_color) / 255.0f, GetGValue(back_color) / 255.0f, GetBValue(back_color) / 255.0f, }; COLORREF fore_color, dimmed_color; { // based on the lightness of background, we draw foreground in light // or dark shades of gray blended onto the background with slight // transparency to avoid sharp contrast constexpr float alpha = 0.9f; constexpr float intensity_light[] = { (1.0f * alpha), (0.8f * alpha) }; constexpr float intensity_dark[] = { (0.1f * alpha), (0.3f * alpha) }; // select foreground intensity values (light or dark) auto& i = (back_luma < 0.6f) ? intensity_light : intensity_dark; float r, g, b; r = i[0] + back_f.r * (1 - alpha); g = i[0] + back_f.g * (1 - alpha); b = i[0] + back_f.b * (1 - alpha); fore_color = RGB(r * 0xff, g * 0xff, b * 0xff); r = i[1] + back_f.r * (1 - alpha); g = i[1] + back_f.g * (1 - alpha); b = i[1] + back_f.b * (1 - alpha); dimmed_color = RGB(r * 0xff, g * 0xff, b * 0xff); } // Draw background { auto brush = CreateSolidBrush(back_color); RECT rc = { 0, 0, toast_size_.cx, toast_size_.cy }; FillRect(hdc_, &rc, brush); DeleteBrush(brush); } SetBkMode(hdc_, TRANSPARENT); const auto close = L'\x2715'; auto caption_font = data_->controller->GetCaptionFont(); auto body_font = data_->controller->GetBodyFont(); TEXTMETRIC tm_cap; SelectFont(hdc_, caption_font); GetTextMetrics(hdc_, &tm_cap); auto text_offset_x = margin_.cx; BITMAP image_info = {}; if (scaled_image_) { GetObject(scaled_image_, sizeof(image_info), &image_info); text_offset_x += margin_.cx + image_info.bmWidth; } // calculate close button rect POINT close_pos; { SIZE extent = {}; GetTextExtentPoint32W(hdc_, &close, 1, &extent); close_button_rect_.right = toast_size_.cx; close_button_rect_.top = 0; close_pos.x = close_button_rect_.right - margin_.cy - extent.cx; close_pos.y = close_button_rect_.top + margin_.cy; close_button_rect_.left = close_pos.x - margin_.cy; close_button_rect_.bottom = close_pos.y + extent.cy + margin_.cy; } // image if (scaled_image_) { HDC hdc_image = CreateCompatibleDC(NULL); SelectBitmap(hdc_image, scaled_image_); BLENDFUNCTION blend = { AC_SRC_OVER, 0, 255, AC_SRC_ALPHA }; AlphaBlend(hdc_, margin_.cx, margin_.cy, image_info.bmWidth, image_info.bmHeight, hdc_image, 0, 0, image_info.bmWidth, image_info.bmHeight, blend); DeleteDC(hdc_image); } // caption { RECT rc = { text_offset_x, margin_.cy, close_button_rect_.left, toast_size_.cy }; SelectFont(hdc_, caption_font); SetTextColor(hdc_, fore_color); DrawText(hdc_, data_->caption.data(), (UINT)data_->caption.length(), &rc, DT_SINGLELINE | DT_END_ELLIPSIS | DT_NOPREFIX); } // body text if (!data_->body_text.empty()) { RECT rc = { text_offset_x, 2 * margin_.cy + tm_cap.tmAscent, toast_size_.cx - margin_.cx, toast_size_.cy - margin_.cy }; SelectFont(hdc_, body_font); SetTextColor(hdc_, dimmed_color); DrawText(hdc_, data_->body_text.data(), (UINT)data_->body_text.length(), &rc, DT_LEFT | DT_WORDBREAK | DT_NOPREFIX | DT_END_ELLIPSIS | DT_EDITCONTROL); } // close button { SelectFont(hdc_, caption_font); SetTextColor(hdc_, is_close_hot_ ? fore_color : dimmed_color); ExtTextOut(hdc_, close_pos.x, close_pos.y, 0, nullptr, &close, 1, nullptr); } is_content_updated_ = true; } void DesktopNotificationController::Toast::Invalidate() { is_content_updated_ = false; } bool DesktopNotificationController::Toast::IsRedrawNeeded() const { return !is_content_updated_; } void DesktopNotificationController::Toast::UpdateBufferSize() { if (hdc_) { SIZE new_size; { TEXTMETRIC tm_cap = {}; HFONT font = data_->controller->GetCaptionFont(); if (font) { SelectFont(hdc_, font); if (!GetTextMetrics(hdc_, &tm_cap)) return; } TEXTMETRIC tm_body = {}; font = data_->controller->GetBodyFont(); if (font) { SelectFont(hdc_, font); if (!GetTextMetrics(hdc_, &tm_body)) return; } this->margin_ = { tm_cap.tmAveCharWidth * 2, tm_cap.tmAscent / 2 }; new_size.cx = margin_.cx + (32 * tm_cap.tmAveCharWidth) + margin_.cx; new_size.cy = margin_.cy + (tm_cap.tmHeight) + margin_.cy; if (!data_->body_text.empty()) new_size.cy += margin_.cy + (3 * tm_body.tmHeight); if (data_->image) { BITMAP bm; if (GetObject(data_->image, sizeof(bm), &bm)) { // cap the image size const int max_dim_size = 80; auto width = bm.bmWidth; auto height = bm.bmHeight; if (width < height) { if (height > max_dim_size) { width = width * max_dim_size / height; height = max_dim_size; } } else { if (width > max_dim_size) { height = height * max_dim_size / width; width = max_dim_size; } } ScreenMetrics scr; SIZE image_draw_size = { scr.X(width), scr.Y(height) }; new_size.cx += image_draw_size.cx + margin_.cx; auto height_with_image = margin_.cy + (image_draw_size.cy) + margin_.cy; if (new_size.cy < height_with_image) new_size.cy = height_with_image; UpdateScaledImage(image_draw_size); } } } if (new_size.cx != this->toast_size_.cx || new_size.cy != this->toast_size_.cy) { HDC hdc_screen = GetDC(NULL); auto new_bitmap = CreateCompatibleBitmap(hdc_screen, new_size.cx, new_size.cy); ReleaseDC(NULL, hdc_screen); if (new_bitmap) { if (SelectBitmap(hdc_, new_bitmap)) { RECT dirty1 = {}, dirty2 = {}; if (toast_size_.cx < new_size.cx) { dirty1 = { toast_size_.cx, 0, new_size.cx, toast_size_.cy }; } if (toast_size_.cy < new_size.cy) { dirty2 = { 0, toast_size_.cy, new_size.cx, new_size.cy }; } if (this->bitmap_) DeleteBitmap(this->bitmap_); this->bitmap_ = new_bitmap; this->toast_size_ = new_size; Invalidate(); // Resize also the DWM buffer to prevent flicker during // window resizing. Make sure any existing data is not // overwritten by marking the dirty region. { POINT origin = { 0, 0 }; UPDATELAYEREDWINDOWINFO ulw; ulw.cbSize = sizeof(ulw); ulw.hdcDst = NULL; ulw.pptDst = nullptr; ulw.psize = &toast_size_; ulw.hdcSrc = hdc_; ulw.pptSrc = &origin; ulw.crKey = 0; ulw.pblend = nullptr; ulw.dwFlags = 0; ulw.prcDirty = &dirty1; auto b1 = UpdateLayeredWindowIndirect(hwnd_, &ulw); ulw.prcDirty = &dirty2; auto b2 = UpdateLayeredWindowIndirect(hwnd_, &ulw); _ASSERT(b1 && b2); } return; } DeleteBitmap(new_bitmap); } } } } void DesktopNotificationController::Toast::UpdateScaledImage(const SIZE& size) { BITMAP bm; if (!GetObject(scaled_image_, sizeof(bm), &bm) || bm.bmWidth != size.cx || bm.bmHeight != size.cy) { if (scaled_image_) DeleteBitmap(scaled_image_); scaled_image_ = StretchBitmap(data_->image, size.cx, size.cy); } } void DesktopNotificationController::Toast::UpdateContents() { Draw(); if (IsWindowVisible(hwnd_)) { RECT rc; GetWindowRect(hwnd_, &rc); POINT origin = { 0, 0 }; SIZE size = { rc.right - rc.left, rc.bottom - rc.top }; UpdateLayeredWindow(hwnd_, NULL, nullptr, &size, hdc_, &origin, 0, nullptr, 0); } } void DesktopNotificationController::Toast::Dismiss() { if (!is_non_interactive_) { // Set a flag to prevent further interaction. We don't disable the HWND // because we still want to receive mouse move messages in order to keep // the toast under the cursor and not collapse it while dismissing. is_non_interactive_ = true; AutoDismiss(); } } void DesktopNotificationController::Toast::AutoDismiss() { KillTimer(hwnd_, TimerID_AutoDismiss); StartEaseOut(); } void DesktopNotificationController::Toast::CancelDismiss() { KillTimer(hwnd_, TimerID_AutoDismiss); ease_out_active_ = false; ease_out_pos_ = 0; } void DesktopNotificationController::Toast::ScheduleDismissal() { SetTimer(hwnd_, TimerID_AutoDismiss, 4000, nullptr); } void DesktopNotificationController::Toast::ResetContents() { if (scaled_image_) { DeleteBitmap(scaled_image_); scaled_image_ = NULL; } Invalidate(); } void DesktopNotificationController::Toast::PopUp(int y) { vertical_pos_target_ = vertical_pos_ = y; StartEaseIn(); } void DesktopNotificationController::Toast::SetVerticalPosition(int y) { // Don't restart animation if current target is the same if (y == vertical_pos_target_) return; // Make sure the new animation's origin is at the current position vertical_pos_ += static_cast<int>( (vertical_pos_target_ - vertical_pos_) * stack_collapse_pos_); // Set new target position and start the animation vertical_pos_target_ = y; stack_collapse_start_ = GetTickCount(); data_->controller->StartAnimation(); } HDWP DesktopNotificationController::Toast::Animate( HDWP hdwp, const POINT& origin) { UpdateBufferSize(); if (IsRedrawNeeded()) Draw(); POINT src_origin = { 0, 0 }; UPDATELAYEREDWINDOWINFO ulw; ulw.cbSize = sizeof(ulw); ulw.hdcDst = NULL; ulw.pptDst = nullptr; ulw.psize = nullptr; ulw.hdcSrc = hdc_; ulw.pptSrc = &src_origin; ulw.crKey = 0; ulw.pblend = nullptr; ulw.dwFlags = 0; ulw.prcDirty = nullptr; POINT pt = { 0, 0 }; SIZE size = { 0, 0 }; BLENDFUNCTION blend; UINT dwpFlags = SWP_NOACTIVATE | SWP_SHOWWINDOW | SWP_NOREDRAW | SWP_NOCOPYBITS; auto ease_in_pos = AnimateEaseIn(); auto ease_out_pos = AnimateEaseOut(); auto stack_collapse_pos = AnimateStackCollapse(); auto y_offset = (vertical_pos_target_ - vertical_pos_) * stack_collapse_pos; size.cx = static_cast<int>(toast_size_.cx * ease_in_pos); size.cy = toast_size_.cy; pt.x = origin.x - size.cx; pt.y = static_cast<int>(origin.y - vertical_pos_ - y_offset - size.cy); ulw.pptDst = &pt; ulw.psize = &size; if (ease_in_active_ && ease_in_pos == 1.0f) { ease_in_active_ = false; ScheduleDismissal(); } this->ease_in_pos_ = ease_in_pos; this->stack_collapse_pos_ = stack_collapse_pos; if (ease_out_pos != this->ease_out_pos_) { blend.BlendOp = AC_SRC_OVER; blend.BlendFlags = 0; blend.SourceConstantAlpha = (BYTE)(255 * (1.0f - ease_out_pos)); blend.AlphaFormat = 0; ulw.pblend = &blend; ulw.dwFlags = ULW_ALPHA; this->ease_out_pos_ = ease_out_pos; if (ease_out_pos == 1.0f) { ease_out_active_ = false; dwpFlags &= ~SWP_SHOWWINDOW; dwpFlags |= SWP_HIDEWINDOW; } } if (stack_collapse_pos == 1.0f) { vertical_pos_ = vertical_pos_target_; } // `UpdateLayeredWindowIndirect` updates position, size, and transparency. // `DeferWindowPos` updates z-order, and also position and size in case // ULWI fails, which can happen when one of the dimensions is zero (e.g. // at the beginning of ease-in). auto ulw_result = UpdateLayeredWindowIndirect(hwnd_, &ulw); hdwp = DeferWindowPos(hdwp, hwnd_, HWND_TOPMOST, pt.x, pt.y, size.cx, size.cy, dwpFlags); return hdwp; } void DesktopNotificationController::Toast::StartEaseIn() { _ASSERT(!ease_in_active_); ease_in_start_ = GetTickCount(); ease_in_active_ = true; data_->controller->StartAnimation(); } void DesktopNotificationController::Toast::StartEaseOut() { _ASSERT(!ease_out_active_); ease_out_start_ = GetTickCount(); ease_out_active_ = true; data_->controller->StartAnimation(); } bool DesktopNotificationController::Toast::IsStackCollapseActive() const { return (vertical_pos_ != vertical_pos_target_); } float DesktopNotificationController::Toast::AnimateEaseIn() { if (!ease_in_active_) return ease_in_pos_; constexpr float duration = 500.0f; float elapsed = GetTickCount() - ease_in_start_; float time = std::min(duration, elapsed) / duration; // decelerating exponential ease const float a = -8.0f; auto pos = (std::exp(a * time) - 1.0f) / (std::exp(a) - 1.0f); return pos; } float DesktopNotificationController::Toast::AnimateEaseOut() { if (!ease_out_active_) return ease_out_pos_; constexpr float duration = 120.0f; float elapsed = GetTickCount() - ease_out_start_; float time = std::min(duration, elapsed) / duration; // accelerating circle ease auto pos = 1.0f - std::sqrt(1 - time * time); return pos; } float DesktopNotificationController::Toast::AnimateStackCollapse() { if (!IsStackCollapseActive()) return stack_collapse_pos_; constexpr float duration = 500.0f; float elapsed = GetTickCount() - stack_collapse_start_; float time = std::min(duration, elapsed) / duration; // decelerating exponential ease const float a = -8.0f; auto pos = (std::exp(a * time) - 1.0f) / (std::exp(a) - 1.0f); return pos; } } // namespace brightray