Files
noctalia-shell/src/app/application.cpp
T
2026-05-08 08:28:46 -04:00

1353 lines
52 KiB
C++

#include "application.h"
#include "app/poll_source.h"
#include "compositors/compositor_detect.h"
#include "compositors/output_backend.h"
#include "config/config_types.h"
#include "core/build_info.h"
#include "core/deferred_call.h"
#include "core/log.h"
#include "core/process.h"
#include "core/resource_paths.h"
#include "i18n/i18n.h"
#include "i18n/i18n_service.h"
#include "ipc/ipc_arg_parse.h"
#include "launcher/app_provider.h"
#include "launcher/emoji_provider.h"
#include "launcher/math_provider.h"
#include "launcher/wallpaper_provider.h"
#include "notification/notifications.h"
#include "render/animation/motion_service.h"
#include "shell/clipboard/clipboard_panel.h"
#include "shell/clipboard/clipboard_paste.h"
#include "shell/control_center/control_center_panel.h"
#include "shell/launcher/launcher_panel.h"
#include "shell/session/session_panel.h"
#include "shell/setup_wizard/setup_wizard_panel.h"
#include "shell/test/test_panel.h"
#include "shell/tray/tray_drawer_panel.h"
#include "shell/wallpaper/panel/wallpaper_panel.h"
#include "system/distro_info.h"
#include "time/time_format.h"
#include "ui/controls/input.h"
#include "ui/dialogs/color_picker_dialog.h"
#include "ui/dialogs/file_dialog.h"
#include "ui/dialogs/glyph_picker_dialog.h"
#include "ui/style.h"
#include "util/file_utils.h"
#include <algorithm>
#include <chrono>
#include <csignal>
#include <cstdint>
#include <filesystem>
#include <malloc.h>
#include <optional>
#include <stdexcept>
#include <string_view>
#include <thread>
std::atomic<bool> Application::s_shutdownRequested{false};
namespace {
constexpr Logger kLog("app");
constexpr bool kLockKeysEnabled = true;
template <typename Factory>
auto makeWithStartupBackoff(std::string_view label, Factory&& factory) -> decltype(factory()) {
using namespace std::chrono_literals;
constexpr int kAttempts = 7;
auto delay = 50ms;
int failedAttempts = 0;
for (int attempt = 1; attempt <= kAttempts; ++attempt) {
try {
auto value = factory();
if (failedAttempts > 0) {
kLog.info("{} init succeeded after {} retr{}", label, failedAttempts, failedAttempts == 1 ? "y" : "ies");
}
return value;
} catch (const std::exception& e) {
if (attempt == kAttempts) {
throw;
}
failedAttempts = attempt;
kLog.warn("{} init attempt {}/{} failed: {}; retrying in {}ms", label, attempt, kAttempts, e.what(),
delay.count());
std::this_thread::sleep_for(delay);
delay *= 2;
}
}
throw std::runtime_error(std::string(label) + " init failed");
}
float elapsedSince(std::chrono::steady_clock::time_point start) {
return std::chrono::duration<float, std::milli>(std::chrono::steady_clock::now() - start).count();
}
template <typename Fn> void runStartupPhase(std::string_view label, Fn&& fn) {
constexpr float kSlowStartupPhaseDebugMs = 50.0f;
constexpr float kSlowStartupPhaseWarnMs = 1000.0f;
const auto start = std::chrono::steady_clock::now();
try {
fn();
} catch (...) {
kLog.warn("startup phase {} failed after {:.1f}ms", label, elapsedSince(start));
throw;
}
const float ms = elapsedSince(start);
if (ms >= kSlowStartupPhaseWarnMs) {
kLog.warn("startup phase {} took {:.1f}ms", label, ms);
} else if (ms >= kSlowStartupPhaseDebugMs) {
kLog.debug("startup phase {} took {:.1f}ms", label, ms);
}
}
void signal_handler(int signum) {
if (signum == SIGTERM || signum == SIGINT) {
Application::s_shutdownRequested = true;
}
}
} // namespace
Application::Application() : m_lockKeysService(m_wayland), m_weatherService(m_configService, m_httpClient) {
m_notificationManager.loadPersistedHistory();
notify::setInstance(&m_notificationManager);
LockScreen::setInstance(&m_lockScreen);
auto shouldRefreshControlCenter = [this]() { return m_panelManager.isOpenPanel("control-center"); };
m_notificationManager.addEventCallback(
[this, shouldRefreshControlCenter](const Notification& n, NotificationEvent event) {
const char* kind = "updated";
if (event == NotificationEvent::Added) {
kind = "added";
} else if (event == NotificationEvent::Closed) {
kind = "closed";
}
const char* origin = (n.origin == NotificationOrigin::Internal) ? "internal" : "external";
kLog.debug("notification {} id={} origin={}", kind, n.id, origin);
if (event == NotificationEvent::Added && m_panelManager.isActivePanelContext("notifications")) {
m_notificationManager.markNotificationHistorySeen();
}
// Keep bar widgets in sync with notification state changes.
m_bar.refresh();
if (shouldRefreshControlCenter()) {
m_panelManager.refresh();
}
});
m_notificationManager.setStateCallback([this, shouldRefreshControlCenter]() {
m_bar.refresh();
if (shouldRefreshControlCenter()) {
m_panelManager.refresh();
}
});
}
Application::~Application() {
m_notificationManager.flushPersistedHistory();
m_wayland.setClipboardService(nullptr);
m_wayland.setVirtualKeyboardService(nullptr);
LockScreen::setInstance(nullptr);
notify::setInstance(nullptr);
}
void Application::syncNotificationDaemon() {
if (m_bus == nullptr) {
m_notificationPollSource.setDbusService(nullptr);
m_notificationDbus.reset();
return;
}
if (!m_configService.config().notification.enableDaemon) {
if (m_notificationDbus != nullptr) {
kLog.info("notification daemon disabled by config");
}
m_notificationPollSource.setDbusService(nullptr);
m_notificationDbus.reset();
return;
}
if (m_notificationDbus != nullptr) {
return;
}
try {
m_notificationDbus = makeWithStartupBackoff("notification service", [this]() {
return std::make_unique<NotificationService>(*m_bus, m_notificationManager);
});
m_notificationPollSource.setDbusService(m_notificationDbus.get());
kLog.info("listening on org.freedesktop.Notifications");
} catch (const std::exception& e) {
kLog.warn("notifications disabled: {}", e.what());
m_notificationDbus.reset();
m_notificationPollSource.setDbusService(nullptr);
m_notificationManager.addInternal("Noctalia", i18n::tr("notifications.internal.dbus-disabled"), e.what(),
Urgency::Low);
}
}
void Application::syncPolkitAgent() {
if (m_systemBus == nullptr) {
m_polkitPollSource.reset();
m_polkitAgent.reset();
return;
}
if (!m_configService.config().shell.polkitAgent) {
if (m_polkitAgent != nullptr) {
kLog.info("polkit agent disabled by config");
}
m_polkitPollSource.reset();
m_polkitAgent.reset();
return;
}
if (m_polkitAgent != nullptr) {
return;
}
m_polkitAgent = std::make_unique<PolkitAgent>(*m_systemBus);
m_polkitAgent->setReadyCallback([this](bool ok, const std::string& error) {
if (!ok) {
kLog.warn("polkit agent disabled: {}", error);
m_polkitPollSource.reset();
m_polkitAgent.reset();
return;
}
kLog.info("polkit authentication agent active");
});
m_polkitAgent->setStateCallback([this]() {
if (m_polkitAgent == nullptr) {
return;
}
const bool hasPending = m_polkitAgent->hasPendingRequest();
const bool needsInput = m_polkitAgent->isResponseRequired();
if (!hasPending) {
if (m_panelManager.isOpenPanel("polkit")) {
m_panelManager.close();
}
return;
}
if (needsInput) {
if (!m_panelManager.isOpenPanel("polkit")) {
wl_output* output = m_wayland.preferredPanelOutput(std::chrono::milliseconds(1200));
m_panelManager.openPanel("polkit", PanelOpenRequest{.output = output});
} else {
m_panelManager.refresh();
}
} else if (m_panelManager.isOpenPanel("polkit")) {
m_panelManager.refresh();
}
});
m_polkitPollSource = std::make_unique<PolkitPollSource>(*m_polkitAgent);
m_polkitAgent->start();
}
void Application::run() {
initLogFile();
kLog.info("noctalia {}", noctalia::build_info::displayVersion());
runStartupPhase("initServices", [this]() { initServices(); });
runStartupPhase("initUi", [this]() { initUi(); });
runStartupPhase("initIpc", [this]() { initIpc(); });
runStartupPhase("buildPollSources", [this]() { (void)buildPollSources(); });
runStartupPhase("startup hooks", [this]() {
m_hookManager.reload(m_configService.config().hooks);
m_hookManager.fire(HookKind::Started);
});
runStartupPhase("telemetry enqueue",
[this]() { m_telemetryService.maybeSend(m_configService, m_httpClient, m_wayland); });
runStartupPhase("malloc_trim", []() { malloc_trim(0); });
m_trayInitTimer.start(std::chrono::milliseconds(500), [this]() { startTrayService(); });
m_polkitInitTimer.start(std::chrono::milliseconds(0), [this]() { syncPolkitAgent(); });
m_mainLoop = std::make_unique<MainLoop>(m_wayland, m_bar, [this]() { return currentPollSources(); });
m_mainLoop->run();
kLog.info("shutdown");
}
void Application::initServices() {
std::signal(SIGTERM, signal_handler);
std::signal(SIGINT, signal_handler);
auto shouldRefreshControlCenter = [this]() { return m_panelManager.isOpenPanel("control-center"); };
auto applyMotionConfig = [this]() {
auto& motion = MotionService::instance();
motion.setSpeed(m_configService.config().shell.animation.speed);
motion.setEnabled(m_configService.config().shell.animation.enabled);
};
auto applyPasswordMaskStyle = [this]() {
const auto style = m_configService.config().shell.passwordMaskStyle == PasswordMaskStyle::RandomIcons
? Input::PasswordMaskStyle::RandomIcons
: Input::PasswordMaskStyle::CircleFilled;
Input::setPasswordMaskStyle(style);
};
applyMotionConfig();
applyPasswordMaskStyle();
m_httpClient.setOfflineMode(m_configService.config().shell.offlineMode);
m_configService.addReloadCallback(applyMotionConfig);
m_configService.addReloadCallback(applyPasswordMaskStyle);
m_configService.addReloadCallback(
[this]() { m_httpClient.setOfflineMode(m_configService.config().shell.offlineMode); });
m_communityPaletteService.setReadyCallback([this]() { m_settingsWindow.onExternalOptionsChanged(); });
m_communityPaletteService.sync();
m_configService.addReloadCallback([this]() { m_communityPaletteService.sync(); });
m_communityTemplateService.setReadyCallback([this]() {
if (m_configService.config().theme.templates.enableCommunityTemplates) {
m_themeService.onConfigReload();
m_settingsWindow.onExternalOptionsChanged();
}
});
m_communityTemplateService.sync(m_configService.config().theme.templates);
m_configService.addReloadCallback(
[this]() { m_communityTemplateService.sync(m_configService.config().theme.templates); });
// i18n has no dependencies on other services and must be ready before any
// UI construction reads a translated string.
i18n::Service::instance().init(m_configService.config().shell.lang);
m_configService.addReloadCallback(
[this]() { i18n::Service::instance().setLanguage(m_configService.config().shell.lang); });
// Apply theme before any UI constructs palette-dependent scene nodes.
m_themeService.setResolvedCallback([this](const noctalia::theme::GeneratedPalette& generated, std::string_view mode) {
m_templateApplyService.apply(generated, mode);
m_hookManager.fire(HookKind::ColorsChanged);
});
m_themeService.apply();
m_configService.addReloadCallback([this]() { m_themeService.onConfigReload(); });
// Watch the dconf user database so Auto mode reacts immediately to system
// color-scheme changes (org.gnome.desktop.interface color-scheme).
{
const char* xdg = std::getenv("XDG_CONFIG_HOME");
const char* home = std::getenv("HOME");
std::filesystem::path dconfDb;
if (xdg != nullptr && xdg[0] != '\0') {
dconfDb = std::filesystem::path(xdg) / "dconf" / "user";
} else if (home != nullptr && home[0] != '\0') {
dconfDb = std::filesystem::path(home) / ".config" / "dconf" / "user";
}
if (!dconfDb.empty()) {
m_fileWatcher.watch(dconfDb, [this]() { m_themeService.onAutoSchemeChanged(); });
}
}
if (!m_wayland.connect()) {
throw std::runtime_error("failed to connect to Wayland display");
}
m_glShared.initialize(m_wayland.display());
m_sharedTextureCache.initialize(&m_glShared);
m_asyncTextureCache.initialize(&m_glShared);
m_wayland.setClipboardService(&m_clipboardService);
m_wayland.setVirtualKeyboardService(&m_virtualKeyboardService);
Input::setClipboardService(&m_clipboardService);
Input::setValidateKeyMatcher([this](std::uint32_t sym, std::uint32_t modifiers) {
return m_configService.matchesKeybind(KeybindAction::Validate, sym, modifiers);
});
m_wayland.setOutputChangeCallback([this]() {
if (m_brightnessService != nullptr) {
m_brightnessService->onOutputsChanged();
}
m_wallpaper.onOutputChange();
m_backdrop.onOutputChange();
m_bar.onOutputChange();
m_dock.onOutputChange();
m_desktopWidgetsController.onOutputChange();
m_screenCorners.onOutputChange();
m_lockScreen.onOutputChange();
});
m_clipboardService.setChangeCallback([this]() {
if (m_panelManager.isOpenPanel("clipboard")) {
m_panelManager.refresh();
}
});
m_wayland.setWorkspaceChangeCallback([this]() { m_bar.refresh(); });
m_wayland.setToplevelChangeCallback([this]() {
m_bar.refresh();
m_dock.refresh();
});
if constexpr (kLockKeysEnabled) {
m_lockKeysService.refreshNow();
m_lockKeysService.setChangeCallback(
[this](const WaylandSeat::LockKeysState& previous, const WaylandSeat::LockKeysState& current) {
m_lockKeysOsd.onLockKeysChanged(previous, current);
m_bar.refresh();
});
}
m_idleInhibitor.initialize(m_wayland, &m_renderContext);
m_idleInhibitor.setChangeCallback([this, shouldRefreshControlCenter]() {
m_bar.refresh();
if (shouldRefreshControlCenter()) {
m_panelManager.refresh();
}
});
m_idleManager.initialize(m_wayland);
m_idleManager.setCommandRunner([this](const std::string& command) { return runUserCommand(command); });
m_idleManager.reload(m_configService.config().idle);
m_configService.addReloadCallback([this]() { m_idleManager.reload(m_configService.config().idle); });
m_hookManager.setCommandRunner([this](const std::string& command) { return runUserCommand(command); });
m_hookManager.setBlockingCommandRunner(
[this](const std::string& command) { return runUserCommandBlocking(command); });
m_hookManager.reload(m_configService.config().hooks);
m_configService.addReloadCallback([this]() { m_hookManager.reload(m_configService.config().hooks); });
m_nightLightManager.reload(m_configService.config().nightlight);
m_nightLightManager.setChangeCallback([this, shouldRefreshControlCenter]() {
m_bar.refresh();
if (shouldRefreshControlCenter()) {
m_panelManager.refresh();
}
});
m_configService.addReloadCallback([this]() { m_nightLightManager.reload(m_configService.config().nightlight); });
// Register all wallpaper consumers in the single-callback slot.
m_configService.setWallpaperChangeCallback([this]() {
m_wallpaper.onStateChange();
m_backdrop.onStateChange();
m_lockScreen.onWallpaperChanged();
m_themeService.onWallpaperChange();
if (m_panelManager.isOpenPanel("control-center")) {
m_panelManager.refresh();
}
m_hookManager.fire(HookKind::WallpaperChanged);
});
m_themeService.setChangeCallback([this]() {
m_bar.requestRedraw();
m_dock.requestRedraw();
m_desktopWidgetsController.requestRedraw();
m_panelManager.requestRedraw();
m_notificationToast.requestRedraw();
m_lockScreen.onThemeChanged();
m_osdOverlay.requestRedraw();
m_trayMenu.onThemeChanged();
m_backdrop.onThemeChanged();
m_settingsWindow.onThemeChanged();
m_colorPickerDialogPopup.requestRedraw();
m_glyphPickerDialogPopup.requestRedraw();
m_fileDialogPopup.requestRedraw();
});
if (const auto distro = DistroDetector::detect(); distro.has_value()) {
const auto& label = !distro->prettyName.empty() ? distro->prettyName
: !distro->name.empty() ? distro->name
: distro->id;
kLog.info("distro: {}", label);
} else {
kLog.info("distro: unknown");
}
try {
m_systemMonitor = std::make_unique<SystemMonitorService>(m_configService.config().system.monitor.enabled);
if (m_systemMonitor->isRunning()) {
kLog.info("system monitor service active");
} else {
kLog.info("system monitor service disabled by config");
}
m_configService.addReloadCallback([this, shouldRefreshControlCenter]() {
if (m_systemMonitor == nullptr) {
return;
}
const bool enabled = m_configService.config().system.monitor.enabled;
const bool wasRunning = m_systemMonitor->isRunning();
if (enabled == wasRunning) {
return;
}
try {
m_systemMonitor->setEnabled(enabled);
} catch (const std::exception& e) {
kLog.warn("system monitor service failed to start: {}", e.what());
return;
}
kLog.info("system monitor service {}", m_systemMonitor->isRunning() ? "active" : "disabled by config");
m_bar.refresh();
m_desktopWidgetsController.requestLayout();
if (shouldRefreshControlCenter()) {
m_panelManager.refresh();
}
});
} catch (const std::exception& e) {
kLog.warn("system monitor service disabled: {}", e.what());
m_systemMonitor.reset();
}
try {
m_systemBus = makeWithStartupBackoff("system dbus", []() { return std::make_unique<SystemBus>(); });
kLog.info("connected to system bus");
} catch (const std::exception& e) {
kLog.warn("system dbus disabled: {}", e.what());
m_systemBus.reset();
}
if (m_systemBus != nullptr) {
try {
m_powerProfilesService = std::make_unique<PowerProfilesService>(*m_systemBus);
m_powerProfilesService->setChangeCallback(
[this, shouldRefreshControlCenter](const PowerProfilesState& /*state*/) {
m_bar.refresh();
if (shouldRefreshControlCenter()) {
m_panelManager.refresh();
}
});
if (!m_powerProfilesService->activeProfile().empty()) {
kLog.info("power profiles active profile: {}", m_powerProfilesService->activeProfile());
} else {
kLog.info("power profiles service active");
}
} catch (const std::exception& e) {
kLog.warn("power profiles disabled: {}", e.what());
m_powerProfilesService.reset();
}
try {
m_upowerService = std::make_unique<UPowerService>(*m_systemBus);
m_prevUpowerForHooks = m_upowerService->state();
m_upowerService->setChangeCallback([this]() {
onUpowerStateChangedForHooks();
m_bar.refresh();
});
} catch (const std::exception& e) {
kLog.warn("upower disabled: {}", e.what());
m_upowerService.reset();
}
try {
m_networkService = std::make_unique<NetworkService>(*m_systemBus);
m_prevWirelessEnabledForEvents = m_networkService->state().wirelessEnabled;
m_networkService->setChangeCallback(
[this, shouldRefreshControlCenter](const NetworkState& state, NetworkChangeOrigin origin) {
onNetworkStateChangedForEvents(state, origin);
m_bar.refresh();
if (shouldRefreshControlCenter()) {
m_panelManager.refresh();
}
});
kLog.info("network service active");
} catch (const std::exception& e) {
kLog.warn("network service disabled: {}", e.what());
m_networkService.reset();
}
if (m_networkService != nullptr) {
try {
m_networkSecretAgent = std::make_unique<NetworkSecretAgent>(*m_systemBus);
} catch (const std::exception& e) {
kLog.warn("network secret agent disabled: {}", e.what());
m_networkSecretAgent.reset();
}
}
try {
m_bluetoothService = std::make_unique<BluetoothService>(*m_systemBus);
m_prevBluetoothPoweredForEvents = m_bluetoothService->state().powered;
auto refreshBluetoothUi = [this, shouldRefreshControlCenter]() {
m_bar.refresh();
if (shouldRefreshControlCenter()) {
m_panelManager.refresh();
}
};
m_bluetoothService->setStateCallback(
[this, refreshBluetoothUi](const BluetoothState& state, BluetoothStateChangeOrigin origin) {
onBluetoothStateChangedForEvents(state, origin);
refreshBluetoothUi();
});
m_bluetoothService->setDevicesCallback(
[refreshBluetoothUi](const std::vector<BluetoothDeviceInfo>& /*devices*/) { refreshBluetoothUi(); });
kLog.info("bluetooth service active");
} catch (const std::exception& e) {
kLog.warn("bluetooth service disabled: {}", e.what());
m_bluetoothService.reset();
}
if (m_bluetoothService != nullptr) {
try {
m_bluetoothAgent = std::make_unique<BluetoothAgent>(*m_systemBus);
m_bluetoothAgent->setRequestCallback(
[this, shouldRefreshControlCenter](const BluetoothPairingRequest& /*request*/) {
if (shouldRefreshControlCenter()) {
m_panelManager.refresh();
}
});
} catch (const std::exception& e) {
kLog.warn("bluetooth agent disabled: {}", e.what());
m_bluetoothAgent.reset();
}
}
m_configService.addReloadCallback([this]() { syncPolkitAgent(); });
}
try {
m_brightnessService =
std::make_unique<BrightnessService>(m_systemBus.get(), m_wayland, m_configService.config().brightness);
m_brightnessService->setChangeCallback([this, shouldRefreshControlCenter]() {
m_brightnessOsd.onBrightnessChanged(*m_brightnessService);
m_bar.refresh();
if (shouldRefreshControlCenter()) {
m_panelManager.refresh();
}
});
m_configService.addReloadCallback([this, shouldRefreshControlCenter]() {
if (m_brightnessService == nullptr) {
return;
}
m_brightnessService->reload(m_configService.config().brightness);
m_bar.refresh();
if (shouldRefreshControlCenter()) {
m_panelManager.refresh();
}
});
} catch (const std::exception& e) {
kLog.warn("brightness service disabled: {}", e.what());
m_brightnessService.reset();
}
try {
m_pipewireService = std::make_unique<PipeWireService>();
m_pipewireSpectrum = std::make_unique<PipeWireSpectrum>(*m_pipewireService);
m_soundPlayer = std::make_unique<SoundPlayer>(m_pipewireService->loop());
auto applySoundConfig = [this]() {
if (m_soundPlayer == nullptr) {
return;
}
const auto& audio = m_configService.config().audio;
m_soundPlayer->setVolume(audio.enableSounds ? audio.soundVolume : 0.0f);
auto resolveSoundPath = [](const std::string& configured, std::string_view bundledRelative) {
if (configured.empty()) {
return paths::assetPath(bundledRelative);
}
const std::filesystem::path expanded = FileUtils::expandUserPath(configured);
if (expanded.is_absolute()) {
return expanded;
}
return paths::assetPath(expanded.string());
};
(void)m_soundPlayer->load("volume-change", resolveSoundPath(audio.volumeChangeSound, "sounds/volume-change.wav"));
(void)m_soundPlayer->load("notification", resolveSoundPath(audio.notificationSound, "sounds/notification.wav"));
};
applySoundConfig();
m_configService.addReloadCallback(applySoundConfig);
} catch (const std::exception& e) {
kLog.warn("pipewire disabled: {}", e.what());
m_soundPlayer.reset();
m_pipewireSpectrum.reset();
m_pipewireService.reset();
}
try {
m_bus = makeWithStartupBackoff("session dbus", []() { return std::make_unique<SessionBus>(); });
kLog.info("connected to session bus");
} catch (const std::exception& e) {
kLog.warn("dbus disabled: {}", e.what());
m_notificationManager.addInternal("Noctalia", i18n::tr("notifications.internal.session-bus-unavailable"), e.what(),
Urgency::Low);
}
if (m_bus != nullptr) {
try {
m_debugService = makeWithStartupBackoff(
"debug service", [this]() { return std::make_unique<DebugService>(*m_bus, m_notificationManager); });
kLog.info("debug service active on dev.noctalia.Debug");
} catch (const std::exception& e) {
kLog.warn("debug service disabled: {}", e.what());
m_debugService.reset();
}
try {
m_mprisService =
makeWithStartupBackoff("mpris service", [this]() { return std::make_unique<MprisService>(*m_bus); });
auto applyMprisConfig = [this]() {
if (m_mprisService == nullptr) {
return;
}
m_mprisService->setBlacklist(m_configService.config().shell.mpris.blacklist);
};
applyMprisConfig();
m_configService.addReloadCallback(applyMprisConfig);
m_mprisService->setChangeCallback([this, shouldRefreshControlCenter]() {
m_bar.refresh();
if (shouldRefreshControlCenter()) {
m_panelManager.refresh();
}
});
kLog.info("mpris discovery active");
} catch (const std::exception& e) {
kLog.warn("mpris disabled: {}", e.what());
m_mprisService.reset();
m_notificationManager.addInternal("Noctalia", i18n::tr("notifications.internal.mpris-disabled"), e.what(),
Urgency::Low);
}
syncNotificationDaemon();
m_configService.addReloadCallback([this]() { syncNotificationDaemon(); });
m_trayService = std::make_unique<TrayService>(*m_bus);
m_trayService->setChangeCallback([this]() {
m_bar.refresh();
m_trayMenu.onTrayChanged();
});
m_trayService->setMenuToggleCallback([this](const std::string& itemId) { m_trayMenu.toggleForItem(itemId); });
}
m_weatherService.initialize();
if (m_weatherService.hasData()) {
const WeatherSnapshot& snapshot = m_weatherService.snapshot();
m_nightLightManager.setWeatherCoordinates(snapshot.latitude, snapshot.longitude);
} else {
m_nightLightManager.setWeatherCoordinates(std::nullopt, std::nullopt);
}
m_weatherService.addChangeCallback([this, shouldRefreshControlCenter]() {
if (m_weatherService.hasData()) {
const WeatherSnapshot& snapshot = m_weatherService.snapshot();
m_nightLightManager.setWeatherCoordinates(snapshot.latitude, snapshot.longitude);
} else {
m_nightLightManager.setWeatherCoordinates(std::nullopt, std::nullopt);
}
m_bar.refresh();
m_desktopWidgetsController.requestLayout();
if (shouldRefreshControlCenter()) {
m_panelManager.refresh();
}
});
}
void Application::startTrayService() {
if (m_bus == nullptr || m_trayService == nullptr) {
return;
}
try {
m_trayService->start();
} catch (const std::exception& e) {
kLog.warn("tray watcher disabled: {}", e.what());
}
}
void Application::initUi() {
auto shouldRefreshControlCenter = [this]() { return m_panelManager.isOpenPanel("control-center"); };
m_renderContext.initialize(m_glShared);
m_renderContext.setTextFontFamily(m_configService.config().shell.fontFamily);
m_wallpaper.initialize(m_wayland, &m_configService, &m_renderContext, &m_sharedTextureCache);
m_backdrop.initialize(m_wayland, &m_configService, &m_sharedTextureCache, &m_glShared);
m_settingsWindow.initialize(m_wayland, &m_configService, &m_renderContext, &m_dependencyService);
m_settingsWindow.setOpenDesktopWidgetEditor([this]() { m_desktopWidgetsController.toggleEdit(); });
m_lockScreen.initialize(m_wayland, &m_renderContext, &m_configService, &m_sharedTextureCache);
m_lockScreen.setSessionHooks([this]() { m_hookManager.fire(HookKind::SessionLocked); },
[this]() { m_hookManager.fire(HookKind::SessionUnlocked); });
m_sessionActionHooks.onLogout = [this]() { return m_hookManager.fireBlocking(HookKind::LoggingOut); };
m_sessionActionHooks.onReboot = [this]() { return m_hookManager.fireBlocking(HookKind::Rebooting); };
m_sessionActionHooks.onShutdown = [this]() { return m_hookManager.fireBlocking(HookKind::ShuttingDown); };
m_wayland.setPointerEventCallback([this](const PointerEvent& event) {
if (m_lockScreen.isActive()) {
m_lockScreen.onPointerEvent(event);
return;
}
if (m_desktopWidgetsController.onPointerEvent(event)) {
return;
}
if (m_trayMenu.onPointerEvent(event))
return;
if (m_settingsWindow.onPointerEvent(event))
return;
if (m_colorPickerDialogPopup.onPointerEvent(event))
return;
if (m_glyphPickerDialogPopup.onPointerEvent(event))
return;
if (m_fileDialogPopup.onPointerEvent(event))
return;
if (m_bar.onPointerEvent(event))
return;
if (m_dock.onPointerEvent(event))
return;
if (m_panelManager.onPointerEvent(event))
return;
m_notificationToast.onPointerEvent(event);
});
m_wayland.setKeyboardEventCallback([this](const KeyboardEvent& event) {
if (m_lockScreen.isActive()) {
m_lockScreen.onKeyboardEvent(event);
return;
}
if (m_desktopWidgetsController.isEditing()) {
m_desktopWidgetsController.onKeyboardEvent(event);
return;
}
if (m_colorPickerDialogPopup.isOpen()) {
m_colorPickerDialogPopup.onKeyboardEvent(event);
return;
}
if (m_glyphPickerDialogPopup.isOpen()) {
m_glyphPickerDialogPopup.onKeyboardEvent(event);
return;
}
if (m_fileDialogPopup.isOpen()) {
m_fileDialogPopup.onKeyboardEvent(event);
return;
}
if (m_settingsWindow.ownsKeyboardSurface(m_wayland.lastKeyboardSurface())) {
m_settingsWindow.onKeyboardEvent(event);
return;
}
m_panelManager.onKeyboardEvent(event);
});
// Panel manager must be before bar so widgets can access PanelManager::instance()
m_panelManager.initialize(m_wayland, &m_configService, &m_renderContext);
m_panelManager.setOpenSettingsWindowCallback([this]() { m_settingsWindow.open(); });
m_panelManager.setToggleSettingsWindowCallback([this]() {
if (m_settingsWindow.isOpen()) {
m_settingsWindow.close();
return;
}
m_settingsWindow.open();
});
auto clipboardPanel = std::make_unique<ClipboardPanel>(&m_clipboardService, &m_configService, &m_thumbnailService,
&m_asyncTextureCache);
clipboardPanel->setActivateCallback([this](const ClipboardEntry& entry) {
m_panelManager.close();
const ClipboardAutoPasteMode mode = m_configService.config().shell.clipboardAutoPaste;
if (mode == ClipboardAutoPasteMode::Off) {
return;
}
const bool isImage = entry.isImage();
m_clipboardAutoPasteTimer.stop();
m_clipboardAutoPasteTimer.start(std::chrono::milliseconds(Style::animFast + 30), [this, isImage]() {
DeferredCall::callLater([this, isImage]() {
const ClipboardAutoPasteMode activeMode = m_configService.config().shell.clipboardAutoPaste;
(void)clipboard_paste::pasteEntry(isImage, activeMode, m_virtualKeyboardService);
});
});
});
m_panelManager.registerPanel("clipboard", std::move(clipboardPanel));
m_panelManager.registerPanel("session", std::make_unique<SessionPanel>(&m_configService, m_sessionActionHooks));
m_panelManager.registerPanel("test", std::make_unique<TestPanel>());
m_panelManager.registerPanel("control-center",
std::make_unique<ControlCenterPanel>(
&m_notificationManager, m_pipewireService.get(), m_mprisService.get(),
&m_configService, &m_httpClient, &m_weatherService, m_pipewireSpectrum.get(),
m_upowerService.get(), m_powerProfilesService.get(), m_networkService.get(),
m_networkSecretAgent.get(), m_bluetoothService.get(), m_bluetoothAgent.get(),
m_brightnessService.get(), m_systemMonitor.get(), &m_nightLightManager,
&m_themeService, &m_idleInhibitor, &m_dependencyService, &m_wayland, &m_wallpaper));
{
auto launcherPanel = std::make_unique<LauncherPanel>(&m_configService, &m_asyncTextureCache);
launcherPanel->addProvider(std::make_unique<AppProvider>(&m_wayland));
launcherPanel->addProvider(std::make_unique<WallpaperProvider>(&m_configService, &m_wayland));
launcherPanel->addProvider(std::make_unique<MathProvider>(&m_clipboardService));
launcherPanel->addProvider(std::make_unique<EmojiProvider>(&m_clipboardService));
m_panelManager.registerPanel("launcher", std::move(launcherPanel));
}
m_panelManager.registerPanel("wallpaper",
std::make_unique<WallpaperPanel>(&m_wayland, &m_configService, &m_thumbnailService));
std::size_t trayDrawerColumns = 3;
if (const auto it = m_configService.config().widgets.find("tray"); it != m_configService.config().widgets.end()) {
trayDrawerColumns =
static_cast<std::size_t>(std::clamp<std::int64_t>(it->second.getInt("drawer_columns", 3), 1, 5));
}
m_panelManager.registerPanel(
"tray-drawer", std::make_unique<TrayDrawerPanel>(m_trayService.get(), &m_configService, trayDrawerColumns));
m_panelManager.registerPanel(
"polkit", std::make_unique<PolkitPanel>(&m_configService, [this]() { return m_polkitAgent.get(); }));
m_panelManager.registerPanel("setup-wizard", std::make_unique<SetupWizardPanel>(&m_configService, &m_wayland));
if (SetupWizardPanel::isFirstRun(m_configService)) {
DeferredCall::callLater([]() { PanelManager::instance().togglePanel("setup-wizard"); });
}
m_notificationToast.initialize(m_wayland, &m_configService, &m_notificationManager, &m_renderContext, &m_httpClient);
m_configService.setNotificationManager(&m_notificationManager);
m_notificationManager.setSoundPlayer(m_soundPlayer.get());
m_osdOverlay.initialize(m_wayland, &m_configService, &m_renderContext);
m_audioOsd.bindOverlay(m_osdOverlay);
m_audioOsd.setSoundPlayer(m_soundPlayer.get());
if (m_pipewireService != nullptr) {
m_audioOsd.primeFromService(*m_pipewireService);
}
m_brightnessOsd.bindOverlay(m_osdOverlay);
if (m_brightnessService != nullptr) {
m_brightnessOsd.primeFromService(*m_brightnessService);
}
if constexpr (kLockKeysEnabled) {
m_lockKeysOsd.bindOverlay(m_osdOverlay);
m_lockKeysOsd.primeFromService(m_lockKeysService);
}
m_screenCorners.initialize(m_wayland, &m_configService, &m_renderContext);
m_screenCorners.onConfigReload();
m_trayMenu.initialize(m_wayland, &m_configService, m_trayService.get(), &m_renderContext);
m_bar.initialize(m_wayland, &m_configService, &m_timeService, &m_notificationManager, m_trayService.get(),
m_pipewireService.get(), m_upowerService.get(), m_systemMonitor.get(), m_powerProfilesService.get(),
m_networkService.get(), &m_idleInhibitor, m_mprisService.get(), m_pipewireSpectrum.get(),
&m_httpClient, &m_weatherService, &m_renderContext, &m_nightLightManager, &m_themeService,
m_bluetoothService.get(), m_brightnessService.get(), kLockKeysEnabled ? &m_lockKeysService : nullptr,
&m_fileWatcher);
m_panelManager.setAttachedPanelGeometryCallback(
[this](wl_output* output, std::optional<AttachedPanelGeometry> geometry) {
m_bar.setAttachedPanelGeometry(output, geometry);
});
m_panelManager.setClickShieldExcludeRectsProvider(
[this](wl_output* output) { return m_bar.surfaceRectsForOutput(output); });
m_panelManager.setFocusGrabBarSurfacesProvider([this]() { return m_bar.allBarSurfaces(); });
m_bar.setAutoHideSuppressionCallback([this]() { return m_trayMenu.isOpen() || m_panelManager.isAttachedOpen(); });
// When config reloads, refresh any open panel: bar-driven attached decoration restyle and
// shell-driven compositor blur.
m_configService.addReloadCallback([this]() { m_panelManager.onConfigReloaded(); });
m_configService.addReloadCallback([this]() { m_screenCorners.onConfigReload(); });
m_layerPopupHosts.registerHost(
[this](wl_surface* surface) { return m_panelManager.popupParentContextForSurface(surface); },
[this](wl_surface* surface) { m_panelManager.beginAttachedPopup(surface); },
[this](wl_surface* surface) { m_panelManager.endAttachedPopup(surface); },
[this]() { return m_panelManager.fallbackPopupParentContext(); });
m_layerPopupHosts.registerHost([this](wl_surface* surface) { return m_bar.popupParentContextForSurface(surface); },
[this](wl_surface* surface) { m_bar.beginAttachedPopup(surface); },
[this](wl_surface* surface) { m_bar.endAttachedPopup(surface); },
[this]() {
return m_bar.preferredPopupParentContext(
m_wayland.preferredPanelOutput(std::chrono::milliseconds(1200)));
});
m_colorPickerDialogPopup.initialize(m_wayland, m_configService, m_renderContext, m_layerPopupHosts);
ColorPickerDialog::setPresenter(&m_colorPickerDialogPopup);
m_glyphPickerDialogPopup.initialize(m_wayland, m_configService, m_renderContext, m_layerPopupHosts);
GlyphPickerDialog::setPresenter(&m_glyphPickerDialogPopup);
m_fileDialogPopup.initialize(m_wayland, m_configService, m_renderContext, m_layerPopupHosts, m_thumbnailService);
FileDialog::setPresenter(&m_fileDialogPopup);
m_dock.initialize(m_wayland, &m_configService, &m_renderContext);
m_desktopWidgetsController.initialize(m_wayland, &m_configService, m_pipewireSpectrum.get(), &m_weatherService,
&m_renderContext, m_mprisService.get(), &m_httpClient, m_systemMonitor.get());
m_iconThemePollSource.setChangeCallback([this]() { onIconThemeChanged(); });
std::string lastShellFontFamily = m_configService.config().shell.fontFamily;
m_configService.addReloadCallback([this, lastShellFontFamily]() mutable {
const std::string& newShellFontFamily = m_configService.config().shell.fontFamily;
if (newShellFontFamily == lastShellFontFamily) {
return;
}
lastShellFontFamily = newShellFontFamily;
m_renderContext.setTextFontFamily(newShellFontFamily);
m_bar.requestLayout();
m_dock.requestLayout();
m_desktopWidgetsController.requestLayout();
m_panelManager.requestLayout();
m_notificationToast.requestLayout();
m_lockScreen.onFontChanged();
m_osdOverlay.requestLayout();
m_trayMenu.onFontChanged();
m_backdrop.onFontChanged();
m_settingsWindow.onFontChanged();
m_colorPickerDialogPopup.requestLayout();
m_glyphPickerDialogPopup.requestLayout();
m_fileDialogPopup.requestLayout();
});
m_configService.addReloadCallback([]() { malloc_trim(0); });
m_timeService.setTickSecondCallback([this]() {
m_wallpaper.onSecondTick();
if (m_lockScreen.isActive()) {
if (formatLocalTime("{:%S}") == "00") {
m_lockScreen.onSecondTick();
}
} else {
m_bar.onSecondTick();
m_desktopWidgetsController.onSecondTick();
}
});
if (m_pipewireService != nullptr) {
m_audioOsd.suppressFor(std::chrono::milliseconds(2000));
m_pipewireService->setChangeCallback([this, shouldRefreshControlCenter]() {
if (m_pipewireSpectrum != nullptr) {
m_pipewireSpectrum->handleAudioStateChanged();
}
m_bar.refresh();
if (shouldRefreshControlCenter()) {
m_panelManager.refresh();
}
if (m_pipewireService != nullptr) {
m_audioOsd.onAudioStateChanged(*m_pipewireService);
}
});
m_pipewireService->setVolumePreviewCallback([this](bool isInput, std::uint32_t id, float volume, bool muted) {
if (isInput) {
m_audioOsd.showInput(id, volume, muted);
} else {
m_audioOsd.showOutput(id, volume, muted);
}
});
}
}
void Application::initIpc() {
if (m_ipcService.start()) {
kLog.info("IPC socket at {}", m_ipcService.socketPath());
} else {
kLog.warn("IPC disabled: could not bind socket");
}
m_ipcService.registerHandler(
"status",
[this](const std::string&) -> std::string {
const bool panelOpen = m_panelManager.isOpen();
std::string json = "{\n";
json += " \"barVisible\": ";
json += m_bar.isVisible() ? "true" : "false";
json += ",\n \"panelOpen\": ";
json += panelOpen ? "true" : "false";
json += ",\n \"activePanelId\": ";
json += panelOpen ? ("\"" + m_panelManager.activePanelId() + "\"") : "null";
json += "\n}\n";
return json;
},
"status", "Print current state as JSON");
auto applyNotificationDnd = [this](bool enabled) {
m_notificationManager.setDoNotDisturb(enabled);
m_bar.refresh();
if (m_panelManager.isOpenPanel("control-center")) {
m_panelManager.refresh();
}
};
m_ipcService.registerHandler(
"notification-dnd-set",
[applyNotificationDnd](const std::string& args) -> std::string {
const auto parts = noctalia::ipc::splitWords(args);
if (parts.size() != 1) {
return "error: notification-dnd-set requires <on|off|true|false|1|0>\n";
}
const std::string value = parts[0];
if (value == "on" || value == "true" || value == "1") {
applyNotificationDnd(true);
return "ok\n";
}
if (value == "off" || value == "false" || value == "0") {
applyNotificationDnd(false);
return "ok\n";
}
return "error: invalid value (use on/off, true/false, 1/0)\n";
},
"notification-dnd-set <on|off|true|false|1|0>", "Set notification Do Not Disturb state");
m_ipcService.registerHandler(
"notification-dnd-toggle",
[this, applyNotificationDnd](const std::string&) -> std::string {
applyNotificationDnd(!m_notificationManager.doNotDisturb());
return "ok\n";
},
"notification-dnd-toggle", "Toggle notification Do Not Disturb state");
m_ipcService.registerHandler(
"notification-dnd-status",
[this](const std::string&) -> std::string { return m_notificationManager.doNotDisturb() ? "on\n" : "off\n"; },
"notification-dnd-status", "Print notification Do Not Disturb state");
m_ipcService.registerHandler(
"notification-clear-active",
[this](const std::string&) -> std::string {
std::vector<uint32_t> activeIds;
activeIds.reserve(m_notificationManager.all().size());
for (const auto& notification : m_notificationManager.all()) {
activeIds.push_back(notification.id);
}
for (const uint32_t id : activeIds) {
(void)m_notificationManager.close(id, CloseReason::Dismissed);
}
if (m_panelManager.isOpenPanel("control-center")) {
m_panelManager.refresh();
}
return "ok\n";
},
"notification-clear-active", "Dismiss all currently active notifications");
m_ipcService.registerHandler(
"notification-clear-history",
[this](const std::string&) -> std::string {
m_notificationManager.clearHistory();
if (m_panelManager.isOpenPanel("control-center")) {
m_panelManager.refresh();
}
return "ok\n";
},
"notification-clear-history", "Clear notification history");
m_ipcService.registerHandler(
"dpms-on",
[this](const std::string&) -> std::string {
if (!compositors::setOutputPower(m_wayland, true)) {
return "error: failed to execute dpms-on command\n";
}
return "ok\n";
},
"dpms-on", "Turn monitors on");
m_ipcService.registerHandler(
"dpms-off",
[this](const std::string&) -> std::string {
if (!compositors::setOutputPower(m_wayland, false)) {
return "error: failed to execute dpms-off command\n";
}
return "ok\n";
},
"dpms-off", "Turn monitors off");
if (m_brightnessService != nullptr) {
m_brightnessService->registerIpc(m_ipcService,
[this]() { m_brightnessOsd.suppressFor(std::chrono::milliseconds(250)); });
}
m_configService.registerIpc(m_ipcService);
m_bar.registerIpc(m_ipcService);
m_desktopWidgetsController.registerIpc(m_ipcService);
m_lockScreen.registerIpc(m_ipcService);
m_panelManager.registerIpc(m_ipcService);
m_idleInhibitor.registerIpc(m_ipcService);
m_nightLightManager.registerIpc(m_ipcService);
m_themeService.registerIpc(m_ipcService);
m_dock.registerIpc(m_ipcService);
m_wallpaper.registerIpc(m_ipcService);
if (m_mprisService) {
m_mprisService->registerIpc(m_ipcService);
}
if (m_pipewireService) {
m_pipewireService->registerIpc(m_ipcService, m_configService);
}
}
bool Application::runUserCommand(const std::string& command) {
constexpr std::string_view prefix = "noctalia:";
if (command.rfind(prefix, 0) == 0) {
const std::string response = m_ipcService.execute(command.substr(prefix.size()));
if (response.rfind("error:", 0) == 0) {
kLog.warn("IPC command '{}' failed: {}", command, response.substr(0, response.find('\n')));
return false;
}
return true;
}
if (!process::runAsync(command)) {
kLog.warn("command failed to launch: {}", command);
return false;
}
return true;
}
bool Application::runUserCommandBlocking(const std::string& command) {
constexpr std::string_view prefix = "noctalia:";
if (command.rfind(prefix, 0) == 0) {
const std::string response = m_ipcService.execute(command.substr(prefix.size()));
if (response.rfind("error:", 0) == 0) {
kLog.warn("IPC command '{}' failed: {}", command, response.substr(0, response.find('\n')));
return false;
}
return true;
}
const auto result = process::runSync(command);
if (!result) {
kLog.warn("command failed: {} exit_code={} stderr={}", command, result.exitCode, result.err);
return false;
}
return true;
}
bool Application::runIdleCommand(const std::string& command) { return runUserCommand(command); }
void Application::onIconThemeChanged() {
kLog.info("system icon theme changed; refreshing icon consumers");
m_bar.reload();
m_dock.reload();
m_panelManager.onIconThemeChanged();
m_notificationToast.requestLayout();
}
void Application::onUpowerStateChangedForHooks() {
if (m_upowerService == nullptr) {
return;
}
const UPowerState next = m_upowerService->state();
if (!m_prevUpowerForHooks.has_value()) {
m_prevUpowerForHooks = next;
return;
}
const UPowerState prev = *m_prevUpowerForHooks;
if (prev.state != next.state) {
m_hookManager.fire(HookKind::BatteryStateChanged, {{"NOCTALIA_BATTERY_STATE", batteryStateLabel(next.state)}});
}
const std::int32_t thr = m_configService.config().hooks.batteryLowPercentThreshold;
if (thr > 0 && next.isPresent) {
const bool wasAbove = !prev.isPresent || prev.percentage > static_cast<double>(thr);
const bool isAtOrBelow = next.percentage <= static_cast<double>(thr);
if (wasAbove && isAtOrBelow) {
m_hookManager.fire(HookKind::BatteryUnderThreshold,
{{"NOCTALIA_BATTERY_PERCENT", std::to_string(static_cast<int>(next.percentage))}});
}
}
m_prevUpowerForHooks = next;
}
void Application::onNetworkStateChangedForEvents(const NetworkState& state, NetworkChangeOrigin origin) {
if (!m_prevWirelessEnabledForEvents.has_value()) {
m_prevWirelessEnabledForEvents = state.wirelessEnabled;
return;
}
const bool prev = *m_prevWirelessEnabledForEvents;
if (prev != state.wirelessEnabled) {
if (origin != NetworkChangeOrigin::Noctalia) {
if (state.wirelessEnabled) {
m_notificationManager.addInternal(i18n::tr("notifications.internal.network"),
i18n::tr("notifications.internal.wifi-enabled"), "");
} else {
m_notificationManager.addInternal(i18n::tr("notifications.internal.network"),
i18n::tr("notifications.internal.wifi-disabled"), "");
}
}
if (state.wirelessEnabled) {
m_hookManager.fire(HookKind::WifiEnabled);
} else {
m_hookManager.fire(HookKind::WifiDisabled);
}
}
m_prevWirelessEnabledForEvents = state.wirelessEnabled;
}
void Application::onBluetoothStateChangedForEvents(const BluetoothState& state, BluetoothStateChangeOrigin origin) {
if (!m_prevBluetoothPoweredForEvents.has_value()) {
m_prevBluetoothPoweredForEvents = state.powered;
return;
}
const bool prev = *m_prevBluetoothPoweredForEvents;
if (prev != state.powered) {
if (origin != BluetoothStateChangeOrigin::Noctalia) {
if (state.powered) {
m_notificationManager.addInternal(i18n::tr("notifications.internal.bluetooth"),
i18n::tr("notifications.internal.bluetooth-enabled"), "");
} else {
m_notificationManager.addInternal(i18n::tr("notifications.internal.bluetooth"),
i18n::tr("notifications.internal.bluetooth-disabled"), "");
}
}
if (state.powered) {
m_hookManager.fire(HookKind::BluetoothEnabled);
} else {
m_hookManager.fire(HookKind::BluetoothDisabled);
}
}
m_prevBluetoothPoweredForEvents = state.powered;
}
std::vector<PollSource*> Application::currentPollSources() {
std::vector<PollSource*> sources;
if (m_busPollSource != nullptr) {
sources.push_back(m_busPollSource.get());
}
if (m_systemBusPollSource != nullptr) {
sources.push_back(m_systemBusPollSource.get());
}
sources.push_back(&m_notificationPollSource);
sources.push_back(&m_timePollSource);
sources.push_back(&m_configPollSource);
sources.push_back(&m_desktopWidgetsPollSource);
sources.push_back(&m_desktopEntryPollSource);
sources.push_back(&m_iconThemePollSource);
sources.push_back(&m_clipboardPollSource);
sources.push_back(&m_timerPollSource);
sources.push_back(&m_keyRepeatPollSource);
sources.push_back(&m_workspacePollSource);
if constexpr (kLockKeysEnabled) {
sources.push_back(&m_lockKeysPollSource);
}
if (m_pipewirePollSource != nullptr) {
sources.push_back(m_pipewirePollSource.get());
}
if (m_pipewireSpectrumPollSource != nullptr) {
sources.push_back(m_pipewireSpectrumPollSource.get());
}
if (m_polkitPollSource != nullptr) {
sources.push_back(m_polkitPollSource.get());
}
if (m_brightnessPollSource != nullptr) {
sources.push_back(m_brightnessPollSource.get());
}
sources.push_back(&m_fileWatchPollSource);
sources.push_back(&m_ipcPollSource);
sources.push_back(&m_httpClientPollSource);
sources.push_back(&m_weatherPollSource);
sources.push_back(&m_thumbnailService);
sources.push_back(&m_asyncTextureCache);
return sources;
}
std::vector<PollSource*> Application::buildPollSources() {
if (m_bus != nullptr) {
if (m_busPollSource == nullptr) {
m_busPollSource = std::make_unique<SessionBusPollSource>(*m_bus);
}
} else {
m_busPollSource.reset();
}
if (m_systemBus != nullptr) {
if (m_systemBusPollSource == nullptr) {
m_systemBusPollSource = std::make_unique<SystemBusPollSource>(*m_systemBus);
}
} else {
m_systemBusPollSource.reset();
}
if (m_pipewireService != nullptr) {
if (m_pipewirePollSource == nullptr) {
m_pipewirePollSource = std::make_unique<PipeWirePollSource>(*m_pipewireService);
}
} else {
m_pipewirePollSource.reset();
}
if (m_pipewireSpectrum != nullptr) {
if (m_pipewireSpectrumPollSource == nullptr) {
m_pipewireSpectrumPollSource = std::make_unique<PipeWireSpectrumPollSource>(*m_pipewireSpectrum);
}
} else {
m_pipewireSpectrumPollSource.reset();
}
if (m_brightnessService != nullptr) {
if (m_brightnessPollSource == nullptr) {
m_brightnessPollSource = std::make_unique<BrightnessPollSource>(*m_brightnessService);
}
} else {
m_brightnessPollSource.reset();
}
return currentPollSources();
}