/* * Copyright (c) 2018-2020, Andreas Kling * Copyright (c) 2021, kleines Filmröllchen * Copyright (c) 2021, David Isaksson * Copyright (c) 2022, the SerenityOS developers. * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include class AudioWidget final : public GUI::Widget { C_OBJECT_ABSTRACT(AudioWidget) private: struct VolumeBitmapPair { int volume_threshold { 0 }; NonnullRefPtr bitmap; }; public: static ErrorOr> try_create() { Array volume_level_bitmaps = { { { 66, TRY(Gfx::Bitmap::try_load_from_file("/res/icons/16x16/audio-volume-high.png"sv)) }, { 33, TRY(Gfx::Bitmap::try_load_from_file("/res/icons/16x16/audio-volume-medium.png"sv)) }, { 1, TRY(Gfx::Bitmap::try_load_from_file("/res/icons/16x16/audio-volume-low.png"sv)) }, { 0, TRY(Gfx::Bitmap::try_load_from_file("/res/icons/16x16/audio-volume-zero.png"sv)) }, { 0, TRY(Gfx::Bitmap::try_load_from_file("/res/icons/16x16/audio-volume-muted.png"sv)) } } }; auto audio_client = TRY(Audio::ConnectionToServer::try_create()); NonnullRefPtr audio_widget = TRY(adopt_nonnull_ref_or_enomem(new (nothrow) AudioWidget(move(audio_client), move(volume_level_bitmaps)))); TRY(audio_widget->try_initialize_graphical_elements()); return audio_widget; } private: AudioWidget(NonnullRefPtr audio_client, Array volume_level_bitmaps) : m_audio_client(move(audio_client)) , m_volume_level_bitmaps(move(volume_level_bitmaps)) , m_show_percent(Config::read_bool("AudioApplet"sv, "Applet"sv, "ShowPercent"sv, false)) { m_audio_volume = static_cast(m_audio_client->get_main_mix_volume() * 100); m_audio_muted = m_audio_client->is_main_mix_muted(); m_audio_client->on_main_mix_muted_state_change = [this](bool muted) { if (m_audio_muted == muted) return; m_mute_box->set_checked(!m_audio_muted); m_slider->set_enabled(!muted); m_audio_muted = muted; update(); }; m_audio_client->on_main_mix_volume_change = [this](double volume) { m_audio_volume = static_cast(round(volume * 100)); m_slider->set_value(m_slider->max() - m_audio_volume, GUI::AllowCallback::No); if (!m_audio_muted) update(); }; } ErrorOr try_initialize_graphical_elements() { m_slider_window = add(window()); m_slider_window->set_window_type(GUI::WindowType::Popup); m_root_container = TRY(m_slider_window->try_set_main_widget()); m_root_container->set_fill_with_background_color(true); m_root_container->set_layout(); m_root_container->layout()->set_margins({ 4 }); m_root_container->layout()->set_spacing(0); m_root_container->set_frame_shape(Gfx::FrameShape::Window); m_percent_box = m_root_container->add("\xE2\x84\xB9"); m_percent_box->set_tooltip(m_show_percent ? "Hide percent" : "Show percent"); m_percent_box->set_checked(m_show_percent); m_percent_box->on_checked = [&](bool show_percent) { m_show_percent = show_percent; set_audio_widget_size(m_show_percent); m_percent_box->set_tooltip(m_show_percent ? "Hide percent" : "Show percent"); GUI::Application::the()->hide_tooltip(); Config::write_bool("AudioApplet"sv, "Applet"sv, "ShowPercent"sv, m_show_percent); }; m_slider = m_root_container->add(); m_slider->set_max(100); m_slider->set_page_step(5); m_slider->set_step(5); m_slider->set_value(m_slider->max() - m_audio_volume); m_slider->set_knob_size_mode(GUI::Slider::KnobSizeMode::Proportional); m_slider->on_change = [&](int value) { m_audio_volume = m_slider->max() - value; double volume = clamp(static_cast(m_audio_volume) / m_slider->max(), 0.0, 1.0); m_audio_client->set_main_mix_volume(volume); update(); }; m_mute_box = m_root_container->add("\xE2\x9D\x8C"); m_mute_box->set_checked(m_audio_muted); m_mute_box->set_tooltip(m_audio_muted ? "Unmute" : "Mute"); m_mute_box->on_checked = [&](bool is_muted) { m_mute_box->set_tooltip(is_muted ? "Unmute" : "Mute"); m_audio_client->set_main_mix_muted(is_muted); GUI::Application::the()->hide_tooltip(); }; return {}; }; public: virtual ~AudioWidget() override = default; void set_audio_widget_size(bool show_percent) { if (show_percent) window()->resize(44, 16); else window()->resize(16, 16); } private: virtual void mousedown_event(GUI::MouseEvent& event) override { if (event.button() == GUI::MouseButton::Primary) { if (!m_slider_window->is_visible()) open(); else close(); return; } if (event.button() == GUI::MouseButton::Secondary) { m_audio_client->set_main_mix_muted(!m_audio_muted); update(); } } virtual void mousewheel_event(GUI::MouseEvent& event) override { if (m_audio_muted) return; m_slider->dispatch_event(event); update(); } virtual void paint_event(GUI::PaintEvent& event) override { GUI::Painter painter(*this); painter.add_clip_rect(event.rect()); painter.clear_rect(event.rect(), Color::from_argb(0)); auto& audio_bitmap = choose_bitmap_from_volume(); painter.blit({}, audio_bitmap, audio_bitmap.rect()); if (m_show_percent) { auto volume_text = m_audio_muted ? "mute" : DeprecatedString::formatted("{}%", m_audio_volume); painter.draw_text(Gfx::IntRect { 16, 3, 24, 16 }, volume_text, Gfx::FontDatabase::default_fixed_width_font(), Gfx::TextAlignment::TopLeft, palette().window_text()); } } virtual void applet_area_rect_change_event(GUI::AppletAreaRectChangeEvent&) override { reposition_slider_window(); } void open() { reposition_slider_window(); m_slider_window->show(); } void close() { m_slider_window->hide(); } Gfx::Bitmap& choose_bitmap_from_volume() { if (m_audio_muted) return *m_volume_level_bitmaps.last().bitmap; for (auto& pair : m_volume_level_bitmaps) { if (m_audio_volume >= pair.volume_threshold) return *pair.bitmap; } VERIFY_NOT_REACHED(); } void reposition_slider_window() { constexpr auto width { 50 }; constexpr auto height { 125 }; constexpr auto tray_and_taskbar_padding { 6 }; constexpr auto icon_offset { (width - 16) / 2 }; auto applet_rect = window()->applet_rect_on_screen(); m_slider_window->set_rect( applet_rect.x() - icon_offset, applet_rect.y() - height - tray_and_taskbar_padding, width, height); } NonnullRefPtr m_audio_client; Array m_volume_level_bitmaps; bool m_show_percent { false }; bool m_audio_muted { false }; int m_audio_volume { 100 }; RefPtr m_slider; RefPtr m_slider_window; RefPtr m_mute_box; RefPtr m_percent_box; RefPtr m_root_container; }; ErrorOr serenity_main(Main::Arguments arguments) { TRY(Core::System::pledge("stdio recvfd sendfd rpath wpath cpath unix thread")); auto app = TRY(GUI::Application::try_create(arguments)); Config::pledge_domain("AudioApplet"); TRY(Core::System::unveil("/tmp/session/%sid/portal/audio", "rw")); TRY(Core::System::unveil("/res", "r")); TRY(Core::System::unveil("/sys/kernel/processes", "r")); TRY(Core::System::unveil(nullptr, nullptr)); auto window = TRY(GUI::Window::try_create()); window->set_has_alpha_channel(true); window->set_title("Audio"); window->set_window_type(GUI::WindowType::Applet); auto audio_widget = TRY(window->try_set_main_widget()); window->show(); // This positioning code depends on the window actually existing. static_cast(window->main_widget())->set_audio_widget_size(Config::read_bool("AudioApplet"sv, "Applet"sv, "ShowPercent"sv, false)); TRY(Core::System::pledge("stdio recvfd sendfd rpath")); return app->exec(); }