123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164 |
- /*
- * Copyright (c) 2020, Andreas Kling <kling@serenityos.org>
- *
- * SPDX-License-Identifier: BSD-2-Clause
- */
- #include "NotificationWindow.h"
- #include <AK/HashMap.h>
- #include <AK/Optional.h>
- #include <AK/Vector.h>
- #include <LibGUI/BoxLayout.h>
- #include <LibGUI/Desktop.h>
- #include <LibGUI/Label.h>
- #include <LibGUI/Widget.h>
- #include <LibGfx/Bitmap.h>
- #include <LibGfx/Font/FontDatabase.h>
- #include <LibGfx/ShareableBitmap.h>
- namespace NotificationServer {
- static HashMap<u32, RefPtr<NotificationWindow>> s_windows;
- static void update_notification_window_locations(Gfx::IntRect const& screen_rect)
- {
- Optional<Gfx::IntRect> last_window_rect;
- for (auto& window_entry : s_windows) {
- auto& window = window_entry.value;
- Gfx::IntPoint new_window_location;
- if (last_window_rect.has_value())
- new_window_location = last_window_rect.value().bottom_left().translated(0, 10);
- else
- new_window_location = screen_rect.top_right().translated(-window->rect().width() - 24, 7);
- if (window->rect().location() != new_window_location) {
- window->move_to(new_window_location);
- window->set_original_rect(window->rect());
- }
- last_window_rect = window->rect();
- }
- }
- NotificationWindow::NotificationWindow(i32 client_id, DeprecatedString const& text, DeprecatedString const& title, Gfx::ShareableBitmap const& icon)
- {
- m_id = client_id;
- s_windows.set(m_id, this);
- set_window_type(GUI::WindowType::Notification);
- set_resizable(false);
- set_minimizable(false);
- Optional<Gfx::IntRect> lowest_notification_rect_on_screen;
- for (auto& window_entry : s_windows) {
- auto& window = window_entry.value;
- if (!lowest_notification_rect_on_screen.has_value()
- || (window->m_original_rect.y() > lowest_notification_rect_on_screen.value().y()))
- lowest_notification_rect_on_screen = window->m_original_rect;
- }
- Gfx::IntRect rect;
- rect.set_width(220);
- rect.set_height(40);
- rect.set_location(GUI::Desktop::the().rect().top_right().translated(-rect.width() - 24, 7));
- if (lowest_notification_rect_on_screen.has_value())
- rect.set_location(lowest_notification_rect_on_screen.value().bottom_left().translated(0, 10));
- set_rect(rect);
- m_original_rect = rect;
- auto widget = set_main_widget<GUI::Widget>().release_value_but_fixme_should_propagate_errors();
- widget->set_fill_with_background_color(true);
- widget->set_layout<GUI::HorizontalBoxLayout>(8, 6);
- m_image = &widget->add<GUI::ImageWidget>();
- m_image->set_visible(icon.is_valid());
- if (icon.is_valid()) {
- m_image->set_bitmap(icon.bitmap());
- }
- auto& left_container = widget->add<GUI::Widget>();
- left_container.set_layout<GUI::VerticalBoxLayout>();
- m_title_label = &left_container.add<GUI::Label>(title);
- m_title_label->set_font(Gfx::FontDatabase::default_font().bold_variant());
- m_title_label->set_text_alignment(Gfx::TextAlignment::CenterLeft);
- m_text_label = &left_container.add<GUI::Label>(text);
- m_text_label->set_text_alignment(Gfx::TextAlignment::CenterLeft);
- // FIXME: There used to be code for setting the tooltip here, but since we
- // expand the notification now we no longer set the tooltip. Should there be
- // a limit to the lines shown in an expanded notification, at which point a
- // tooltip should be set?
- on_close = [this] {
- s_windows.remove(m_id);
- update_notification_window_locations(GUI::Desktop::the().rect());
- };
- }
- RefPtr<NotificationWindow> NotificationWindow::get_window_by_id(i32 id)
- {
- auto window = s_windows.get(id);
- return window.value_or(nullptr);
- }
- void NotificationWindow::resize_to_fit_text()
- {
- auto line_height = m_text_label->font().glyph_height();
- auto total_height = m_text_label->text_calculated_preferred_height();
- m_text_label->set_fixed_height(total_height);
- set_height(40 - line_height + total_height);
- }
- void NotificationWindow::enter_event(Core::Event&)
- {
- m_hovering = true;
- resize_to_fit_text();
- move_to_front();
- update_notification_window_locations(GUI::Desktop::the().rect());
- }
- void NotificationWindow::leave_event(Core::Event&)
- {
- m_hovering = false;
- m_text_label->set_preferred_height(GUI::SpecialDimension::Grow);
- set_height(40);
- update_notification_window_locations(GUI::Desktop::the().rect());
- }
- void NotificationWindow::set_text(DeprecatedString const& value)
- {
- m_text_label->set_text(value);
- if (m_hovering)
- resize_to_fit_text();
- }
- void NotificationWindow::set_title(DeprecatedString const& value)
- {
- m_title_label->set_text(value);
- }
- void NotificationWindow::set_image(Gfx::ShareableBitmap const& image)
- {
- m_image->set_visible(image.is_valid());
- if (image.is_valid()) {
- m_image->set_bitmap(image.bitmap());
- }
- }
- void NotificationWindow::set_height(int height)
- {
- auto rect = this->rect();
- rect.set_height(height);
- set_rect(rect);
- }
- void NotificationWindow::screen_rects_change_event(GUI::ScreenRectsChangeEvent& event)
- {
- update_notification_window_locations(event.rects()[event.main_screen_index()]);
- }
- }
|