2024-12-16 19:21:44 +01:00
# include <re2/re2.h>
2022-03-16 21:37:21 +01:00
# include "Compositor.hpp"
2024-10-12 02:29:51 +02:00
# include "debug/Log.hpp"
2022-07-10 15:41:26 +02:00
# include "helpers/Splashes.hpp"
2024-03-03 19:39:20 +01:00
# include "config/ConfigValue.hpp"
2024-03-09 17:52:59 +01:00
# include "managers/CursorManager.hpp"
2024-04-23 02:27:08 +02:00
# include "managers/TokenManager.hpp"
2024-05-05 23:18:10 +02:00
# include "managers/PointerManager.hpp"
2024-05-10 19:27:57 +02:00
# include "managers/SeatManager.hpp"
2024-12-06 15:45:02 +01:00
# include "managers/VersionKeeperManager.hpp"
2024-04-07 04:31:51 +02:00
# include "managers/eventLoop/EventLoopManager.hpp"
2024-07-21 13:09:54 +02:00
# include <aquamarine/output/Output.hpp>
2024-08-03 14:02:10 +02:00
# include <bit>
2024-10-13 14:24:10 +02:00
# include <ctime>
2022-07-10 15:41:26 +02:00
# include <random>
2024-10-13 14:24:10 +02:00
# include <print>
2024-07-21 13:09:54 +02:00
# include <cstring>
# include <filesystem>
2023-07-09 23:08:40 +02:00
# include <unordered_set>
2022-07-31 00:27:32 +02:00
# include "debug/HyprCtl.hpp"
2023-02-19 14:45:56 +01:00
# include "debug/CrashReporter.hpp"
2023-01-05 20:17:55 +01:00
# ifdef USES_SYSTEMD
2024-04-20 19:50:07 +02:00
# include <helpers/SdDaemon.hpp> // for SdNotify
2023-01-05 20:17:55 +01:00
# endif
2023-03-01 15:06:52 +01:00
# include <ranges>
2024-06-11 17:17:45 +02:00
# include "helpers/varlist/VarList.hpp"
2024-04-20 15:14:54 +02:00
# include "protocols/FractionalScale.hpp"
2024-04-27 00:55:41 +02:00
# include "protocols/PointerConstraints.hpp"
2024-05-09 22:47:21 +02:00
# include "protocols/LayerShell.hpp"
2024-05-11 00:28:33 +02:00
# include "protocols/XDGShell.hpp"
2024-10-05 01:44:16 +02:00
# include "protocols/XDGOutput.hpp"
2024-10-06 15:07:07 +02:00
# include "protocols/SecurityContext.hpp"
2024-06-08 10:07:59 +02:00
# include "protocols/core/Compositor.hpp"
# include "protocols/core/Subcompositor.hpp"
2024-04-30 03:41:27 +02:00
# include "desktop/LayerSurface.hpp"
2024-07-21 13:09:54 +02:00
# include "render/Renderer.hpp"
2024-05-25 22:43:51 +02:00
# include "xwayland/XWayland.hpp"
2024-09-30 18:25:39 +02:00
# include "helpers/ByteOperations.hpp"
2024-10-23 01:51:25 +02:00
# include "render/decorations/CHyprGroupBarDecoration.hpp"
2022-03-16 21:37:21 +01:00
2024-06-11 17:17:45 +02:00
# include <hyprutils/string/String.hpp>
2024-07-21 13:09:54 +02:00
# include <aquamarine/input/Input.hpp>
2024-06-11 17:17:45 +02:00
2024-07-20 00:37:20 +02:00
# include <fcntl.h>
2024-04-28 19:58:31 +02:00
# include <sys/types.h>
2024-04-28 23:25:24 +02:00
# include <sys/stat.h>
2024-06-18 21:52:55 +02:00
# include <sys/resource.h>
2024-04-28 19:58:31 +02:00
2024-07-21 13:09:54 +02:00
using namespace Hyprutils : : String ;
using namespace Aquamarine ;
2024-12-19 02:56:01 +01:00
static int handleCritSignal ( int signo , void * data ) {
2023-09-06 12:51:36 +02:00
Debug : : log ( LOG , " Hyprland received signal {} " , signo ) ;
2022-07-13 18:18:23 +02:00
2023-08-19 19:24:48 +02:00
if ( signo = = SIGTERM | | signo = = SIGINT | | signo = = SIGKILL )
2024-07-24 19:07:36 +02:00
g_pCompositor - > stopCompositor ( ) ;
2022-07-13 18:18:23 +02:00
2023-08-19 19:24:48 +02:00
return 0 ;
2022-07-13 18:18:23 +02:00
}
2024-12-19 02:56:01 +01:00
static void handleUnrecoverableSignal ( int sig ) {
2023-02-27 13:32:38 +01:00
2023-03-01 22:55:30 +01:00
// remove our handlers
signal ( SIGABRT , SIG_DFL ) ;
signal ( SIGSEGV , SIG_DFL ) ;
2023-12-20 23:54:52 +01:00
if ( g_pHookSystem & & g_pHookSystem - > m_bCurrentEventPlugin ) {
2023-02-27 13:32:38 +01:00
longjmp ( g_pHookSystem - > m_jbHookFaultJumpBuf , 1 ) ;
return ;
}
2024-04-27 18:38:48 +02:00
// Kill the program if the crash-reporter is caught in a deadlock.
signal ( SIGALRM , [ ] ( int _ ) {
char const * msg = " \n CrashReporter exceeded timeout, forcefully exiting \n " ;
write ( 2 , msg , strlen ( msg ) ) ;
abort ( ) ;
} ) ;
alarm ( 15 ) ;
2024-12-07 18:51:18 +01:00
NCrashReporter : : createAndSaveCrash ( sig ) ;
2023-03-01 22:55:30 +01:00
2023-02-20 12:02:44 +01:00
abort ( ) ;
2023-02-19 14:45:56 +01:00
}
2024-12-19 02:56:01 +01:00
static void handleUserSignal ( int sig ) {
2023-09-29 17:38:13 +02:00
if ( sig = = SIGUSR1 ) {
// means we have to unwind a timed out event
throw std : : exception ( ) ;
}
}
2024-12-07 18:51:18 +01:00
static eLogLevel aqLevelToHl ( Aquamarine : : eBackendLogLevel level ) {
2024-07-21 13:09:54 +02:00
switch ( level ) {
case Aquamarine : : eBackendLogLevel : : AQ_LOG_TRACE : return TRACE ;
case Aquamarine : : eBackendLogLevel : : AQ_LOG_DEBUG : return LOG ;
case Aquamarine : : eBackendLogLevel : : AQ_LOG_ERROR : return ERR ;
case Aquamarine : : eBackendLogLevel : : AQ_LOG_WARNING : return WARN ;
case Aquamarine : : eBackendLogLevel : : AQ_LOG_CRITICAL : return CRIT ;
default : break ;
}
return NONE ;
}
2024-12-19 02:56:01 +01:00
static void aqLog ( Aquamarine : : eBackendLogLevel level , std : : string msg ) {
2024-07-21 13:09:54 +02:00
Debug : : log ( aqLevelToHl ( level ) , " [AQ] {} " , msg ) ;
}
2024-06-19 18:36:40 +02:00
void CCompositor : : bumpNofile ( ) {
if ( ! getrlimit ( RLIMIT_NOFILE , & m_sOriginalNofile ) )
Debug : : log ( LOG , " Old rlimit: soft -> {}, hard -> {} " , m_sOriginalNofile . rlim_cur , m_sOriginalNofile . rlim_max ) ;
else {
Debug : : log ( ERR , " Failed to get NOFILE rlimits " ) ;
m_sOriginalNofile . rlim_max = 0 ;
return ;
}
2024-06-18 21:52:55 +02:00
2024-06-19 18:36:40 +02:00
rlimit newLimit = m_sOriginalNofile ;
2024-06-18 21:52:55 +02:00
2024-06-19 18:36:40 +02:00
newLimit . rlim_cur = newLimit . rlim_max ;
2024-06-18 21:52:55 +02:00
2024-06-19 18:36:40 +02:00
if ( setrlimit ( RLIMIT_NOFILE , & newLimit ) < 0 ) {
Debug : : log ( ERR , " Failed bumping NOFILE limits higher " ) ;
m_sOriginalNofile . rlim_max = 0 ;
return ;
}
2024-06-18 21:52:55 +02:00
2024-06-19 18:36:40 +02:00
if ( ! getrlimit ( RLIMIT_NOFILE , & newLimit ) )
Debug : : log ( LOG , " New rlimit: soft -> {}, hard -> {} " , newLimit . rlim_cur , newLimit . rlim_max ) ;
}
2024-06-18 21:52:55 +02:00
2024-06-19 18:36:40 +02:00
void CCompositor : : restoreNofile ( ) {
if ( m_sOriginalNofile . rlim_max < = 0 )
return ;
2024-06-18 21:52:55 +02:00
2024-06-19 18:36:40 +02:00
if ( setrlimit ( RLIMIT_NOFILE , & m_sOriginalNofile ) < 0 )
Debug : : log ( ERR , " Failed restoring NOFILE limits " ) ;
2024-06-18 21:52:55 +02:00
}
2024-12-07 18:51:18 +01:00
CCompositor : : CCompositor ( ) : m_iHyprlandPID ( getpid ( ) ) {
2024-04-28 23:25:24 +02:00
m_szHyprTempDataRoot = std : : string { getenv ( " XDG_RUNTIME_DIR " ) } + " /hypr " ;
if ( m_szHyprTempDataRoot . starts_with ( " /hypr " ) ) {
2024-10-13 14:24:10 +02:00
std : : println ( " Bailing out, $XDG_RUNTIME_DIR is invalid " ) ;
2024-04-28 23:25:24 +02:00
throw std : : runtime_error ( " CCompositor() failed " ) ;
}
if ( ! m_szHyprTempDataRoot . starts_with ( " /run/user " ) )
2024-10-13 14:24:10 +02:00
std : : println ( " [!!WARNING!!] XDG_RUNTIME_DIR looks non-standard. Proceeding anyways... " ) ;
2024-04-28 23:25:24 +02:00
2024-04-28 21:06:40 +02:00
std : : random_device dev ;
std : : mt19937 engine ( dev ( ) ) ;
std : : uniform_int_distribution < > distribution ( 0 , INT32_MAX ) ;
2024-12-07 18:51:18 +01:00
m_szInstanceSignature = std : : format ( " {}_{}_{} " , GIT_COMMIT_HASH , std : : time ( nullptr ) , distribution ( engine ) ) ;
2022-03-18 23:52:36 +01:00
2022-06-29 12:58:49 +02:00
setenv ( " HYPRLAND_INSTANCE_SIGNATURE " , m_szInstanceSignature . c_str ( ) , true ) ;
2024-04-28 23:25:24 +02:00
if ( ! std : : filesystem : : exists ( m_szHyprTempDataRoot ) )
mkdir ( m_szHyprTempDataRoot . c_str ( ) , S_IRWXU ) ;
else if ( ! std : : filesystem : : is_directory ( m_szHyprTempDataRoot ) ) {
2024-10-13 14:24:10 +02:00
std : : println ( " Bailing out, {} is not a directory " , m_szHyprTempDataRoot ) ;
2024-04-28 23:25:24 +02:00
throw std : : runtime_error ( " CCompositor() failed " ) ;
2024-04-28 21:06:40 +02:00
}
2022-12-21 16:41:02 +01:00
2024-04-28 23:25:24 +02:00
m_szInstancePath = m_szHyprTempDataRoot + " / " + m_szInstanceSignature ;
2024-04-28 21:06:40 +02:00
2024-04-28 23:25:24 +02:00
if ( std : : filesystem : : exists ( m_szInstancePath ) ) {
2024-10-13 14:24:10 +02:00
std : : println ( " Bailing out, {} exists?? " , m_szInstancePath ) ;
2024-04-28 23:25:24 +02:00
throw std : : runtime_error ( " CCompositor() failed " ) ;
2024-04-28 21:06:40 +02:00
}
2024-04-28 23:25:24 +02:00
if ( mkdir ( m_szInstancePath . c_str ( ) , S_IRWXU ) < 0 ) {
2024-10-13 14:24:10 +02:00
std : : println ( " Bailing out, couldn't create {} " , m_szInstancePath ) ;
2024-04-28 23:25:24 +02:00
throw std : : runtime_error ( " CCompositor() failed " ) ;
2024-04-28 21:06:40 +02:00
}
2022-06-29 12:58:49 +02:00
2024-04-28 23:25:24 +02:00
Debug : : init ( m_szInstancePath ) ;
2022-06-03 17:48:07 +02:00
2023-09-06 21:45:37 +02:00
Debug : : log ( LOG , " Instance Signature: {} " , m_szInstanceSignature ) ;
2022-06-03 17:41:57 +02:00
2024-04-28 23:25:24 +02:00
Debug : : log ( LOG , " Runtime directory: {} " , m_szInstancePath ) ;
2023-09-06 12:51:36 +02:00
Debug : : log ( LOG , " Hyprland PID: {} " , m_iHyprlandPID ) ;
2022-08-28 11:19:08 +02:00
2022-06-25 20:49:06 +02:00
Debug : : log ( LOG , " ===== SYSTEM INFO: ===== " ) ;
logSystemInfo ( ) ;
Debug : : log ( LOG , " ======================== " ) ;
2022-06-25 21:16:52 +02:00
Debug : : log ( NONE , " \n \n " ) ; // pad
2023-01-07 12:33:36 +01:00
Debug : : log ( INFO , " If you are crashing, or encounter any bugs, please consult https://wiki.hyprland.org/Crashes-and-Bugs/ \n \n " ) ;
2022-06-25 21:16:52 +02:00
2022-07-10 15:41:26 +02:00
setRandomSplash ( ) ;
2023-09-06 21:45:37 +02:00
Debug : : log ( LOG , " \n Current splash: {} \n \n " , m_szCurrentSplash ) ;
2024-06-18 21:52:55 +02:00
bumpNofile ( ) ;
2023-03-05 14:37:21 +01:00
}
CCompositor : : ~ CCompositor ( ) {
2024-07-24 19:07:36 +02:00
if ( ! m_bIsShuttingDown )
cleanup ( ) ;
2023-03-05 14:37:21 +01:00
}
2022-07-10 15:41:26 +02:00
2023-03-05 14:37:21 +01:00
void CCompositor : : setRandomSplash ( ) {
2024-12-05 02:59:29 +01:00
auto tt = std : : chrono : : system_clock : : to_time_t ( std : : chrono : : system_clock : : now ( ) ) ;
auto local = * localtime ( & tt ) ;
const auto * SPLASHES = & NSplashes : : SPLASHES ;
if ( local . tm_mon + 1 = = 12 & & local . tm_mday > = 23 & & local . tm_mday < = 27 ) // dec 23-27
SPLASHES = & NSplashes : : SPLASHES_CHRISTMAS ;
if ( ( local . tm_mon + 1 = = 12 & & local . tm_mday > = 29 ) | | ( local . tm_mon + 1 = = 1 & & local . tm_mday < = 3 ) )
SPLASHES = & NSplashes : : SPLASHES_NEWYEAR ;
2023-03-05 14:37:21 +01:00
std : : random_device dev ;
std : : mt19937 engine ( dev ( ) ) ;
2024-12-05 02:59:29 +01:00
std : : uniform_int_distribution < > distribution ( 0 , SPLASHES - > size ( ) - 1 ) ;
2023-03-05 14:37:21 +01:00
2024-12-05 02:59:29 +01:00
m_szCurrentSplash = SPLASHES - > at ( distribution ( engine ) ) ;
2023-03-05 14:37:21 +01:00
}
2024-07-21 13:09:54 +02:00
static std : : vector < SP < Aquamarine : : IOutput > > pendingOutputs ;
2024-10-06 15:07:07 +02:00
//
static bool filterGlobals ( const wl_client * client , const wl_global * global , void * data ) {
if ( ! PROTO : : securityContext - > isClientSandboxed ( client ) )
return true ;
return ! g_pProtocolManager | | ! g_pProtocolManager - > isGlobalPrivileged ( global ) ;
}
2024-07-21 13:09:54 +02:00
//
2024-07-22 13:16:25 +02:00
void CCompositor : : initServer ( std : : string socketName , int socketFd ) {
2023-03-24 14:00:54 +01:00
2022-03-16 21:37:21 +01:00
m_sWLDisplay = wl_display_create ( ) ;
2024-10-06 15:07:07 +02:00
wl_display_set_global_filter ( m_sWLDisplay , : : filterGlobals , nullptr ) ;
2022-07-13 18:18:23 +02:00
m_sWLEventLoop = wl_display_get_event_loop ( m_sWLDisplay ) ;
// register crit signal handler
2024-06-06 20:27:09 +02:00
m_critSigSource = wl_event_loop_add_signal ( m_sWLEventLoop , SIGTERM , handleCritSignal , nullptr ) ;
2024-02-23 00:10:59 +01:00
if ( ! envEnabled ( " HYPRLAND_NO_CRASHREPORTER " ) ) {
signal ( SIGSEGV , handleUnrecoverableSignal ) ;
signal ( SIGABRT , handleUnrecoverableSignal ) ;
}
2023-09-29 17:38:13 +02:00
signal ( SIGUSR1 , handleUserSignal ) ;
2022-07-13 18:18:23 +02:00
2023-03-05 14:37:21 +01:00
initManagers ( STAGE_PRIORITY ) ;
2023-12-21 22:27:12 +01:00
if ( envEnabled ( " HYPRLAND_TRACE " ) )
2023-08-21 19:36:09 +02:00
Debug : : trace = true ;
2024-09-30 18:25:39 +02:00
// set the buffer size to 1MB to avoid disconnects due to an app hanging for a short while
wl_display_set_default_max_buffer_size ( m_sWLDisplay , 1 _MB ) ;
2024-12-07 18:51:18 +01:00
Aquamarine : : SBackendOptions options { } ;
2024-07-21 13:09:54 +02:00
options . logFunction = aqLog ;
2023-03-05 14:37:21 +01:00
2024-07-21 13:09:54 +02:00
std : : vector < Aquamarine : : SBackendImplementationOptions > implementations ;
Aquamarine : : SBackendImplementationOptions option ;
option . backendType = Aquamarine : : eBackendType : : AQ_BACKEND_HEADLESS ;
option . backendRequestMode = Aquamarine : : eBackendRequestMode : : AQ_BACKEND_REQUEST_MANDATORY ;
implementations . emplace_back ( option ) ;
option . backendType = Aquamarine : : eBackendType : : AQ_BACKEND_DRM ;
option . backendRequestMode = Aquamarine : : eBackendRequestMode : : AQ_BACKEND_REQUEST_IF_AVAILABLE ;
implementations . emplace_back ( option ) ;
option . backendType = Aquamarine : : eBackendType : : AQ_BACKEND_WAYLAND ;
option . backendRequestMode = Aquamarine : : eBackendRequestMode : : AQ_BACKEND_REQUEST_FALLBACK ;
implementations . emplace_back ( option ) ;
2023-03-05 14:37:21 +01:00
2024-07-21 13:09:54 +02:00
m_pAqBackend = CBackend : : create ( implementations , options ) ;
2022-03-16 21:37:21 +01:00
2024-07-21 13:09:54 +02:00
if ( ! m_pAqBackend ) {
Debug : : log ( CRIT ,
2024-07-30 13:50:13 +02:00
" m_pAqBackend was null! This usually means aquamarine could not find a GPU or encountered some issues. Make sure you're running either on a tty or on a Wayland "
2024-07-21 13:09:54 +02:00
" session, NOT an X11 one. " ) ;
throwError ( " CBackend::create() failed! " ) ;
2022-03-16 21:37:21 +01:00
}
2024-07-21 13:09:54 +02:00
// TODO: headless only
2024-03-15 15:28:14 +01:00
2024-07-21 13:09:54 +02:00
initAllSignals ( ) ;
2024-03-15 15:28:14 +01:00
2024-07-21 13:09:54 +02:00
if ( ! m_pAqBackend - > start ( ) ) {
Debug : : log ( CRIT ,
2024-07-30 13:50:13 +02:00
" m_pAqBackend couldn't start! This usually means aquamarine could not find a GPU or encountered some issues. Make sure you're running either on a tty or on a "
2024-07-21 13:09:54 +02:00
" Wayland session, NOT an X11 one. " ) ;
throwError ( " CBackend::create() failed! " ) ;
2022-03-24 15:57:46 +01:00
}
2024-07-21 13:09:54 +02:00
m_bInitialized = true ;
2022-03-16 21:37:21 +01:00
2024-07-21 13:09:54 +02:00
m_iDRMFD = m_pAqBackend - > drmFD ( ) ;
Debug : : log ( LOG , " Running on DRMFD: {} " , m_iDRMFD ) ;
2022-03-16 21:37:21 +01:00
2024-07-22 13:16:25 +02:00
if ( ! socketName . empty ( ) & & socketFd ! = - 1 ) {
fcntl ( socketFd , F_SETFD , FD_CLOEXEC ) ;
const auto RETVAL = wl_display_add_socket_fd ( m_sWLDisplay , socketFd ) ;
2024-07-21 13:09:54 +02:00
if ( RETVAL > = 0 ) {
2024-07-22 13:16:25 +02:00
m_szWLDisplaySocket = socketName ;
Debug : : log ( LOG , " wl_display_add_socket_fd for {} succeeded with {} " , socketName , RETVAL ) ;
} else
Debug : : log ( WARN , " wl_display_add_socket_fd for {} returned {}: skipping " , socketName , RETVAL ) ;
} else {
// get socket, avoid using 0
for ( int candidate = 1 ; candidate < = 32 ; candidate + + ) {
const auto CANDIDATESTR = ( " wayland- " + std : : to_string ( candidate ) ) ;
const auto RETVAL = wl_display_add_socket ( m_sWLDisplay , CANDIDATESTR . c_str ( ) ) ;
if ( RETVAL > = 0 ) {
m_szWLDisplaySocket = CANDIDATESTR ;
Debug : : log ( LOG , " wl_display_add_socket for {} succeeded with {} " , CANDIDATESTR , RETVAL ) ;
break ;
} else
Debug : : log ( WARN , " wl_display_add_socket for {} returned {}: skipping candidate {} " , CANDIDATESTR , RETVAL , candidate ) ;
2024-07-21 13:09:54 +02:00
}
2022-03-16 21:37:21 +01:00
}
2024-07-21 13:09:54 +02:00
if ( m_szWLDisplaySocket . empty ( ) ) {
Debug : : log ( WARN , " All candidates failed, trying wl_display_add_socket_auto " ) ;
const auto SOCKETSTR = wl_display_add_socket_auto ( m_sWLDisplay ) ;
if ( SOCKETSTR )
m_szWLDisplaySocket = SOCKETSTR ;
}
2022-03-24 17:17:08 +01:00
2024-07-21 13:09:54 +02:00
if ( m_szWLDisplaySocket . empty ( ) ) {
Debug : : log ( CRIT , " m_szWLDisplaySocket NULL! " ) ;
throwError ( " m_szWLDisplaySocket was null! (wl_display_add_socket and wl_display_add_socket_auto failed) " ) ;
2022-03-24 17:17:08 +01:00
}
2022-03-16 21:37:21 +01:00
2024-07-21 13:09:54 +02:00
setenv ( " WAYLAND_DISPLAY " , m_szWLDisplaySocket . c_str ( ) , 1 ) ;
setenv ( " XDG_SESSION_TYPE " , " wayland " , 1 ) ;
2024-08-15 18:14:48 +02:00
if ( ! getenv ( " XDG_CURRENT_DESKTOP " ) ) {
setenv ( " XDG_CURRENT_DESKTOP " , " Hyprland " , 1 ) ;
m_bDesktopEnvSet = true ;
}
2024-07-21 13:09:54 +02:00
2024-06-08 10:07:59 +02:00
initManagers ( STAGE_BASICINIT ) ;
2022-03-19 14:09:11 +01:00
2024-07-21 13:09:54 +02:00
initManagers ( STAGE_LATE ) ;
2024-08-26 17:25:39 +02:00
for ( auto const & o : pendingOutputs ) {
2024-07-21 13:09:54 +02:00
onNewMonitor ( o ) ;
2022-08-27 23:24:36 +02:00
}
2024-07-21 13:09:54 +02:00
pendingOutputs . clear ( ) ;
}
2022-08-27 23:10:13 +02:00
2024-07-21 13:09:54 +02:00
void CCompositor : : initAllSignals ( ) {
m_pAqBackend - > events . newOutput . registerStaticListener (
[ this ] ( void * p , std : : any data ) {
auto output = std : : any_cast < SP < Aquamarine : : IOutput > > ( data ) ;
Debug : : log ( LOG , " New aquamarine output with name {} " , output - > name ) ;
if ( m_bInitialized )
onNewMonitor ( output ) ;
else
pendingOutputs . emplace_back ( output ) ;
} ,
nullptr ) ;
m_pAqBackend - > events . newPointer . registerStaticListener (
[ ] ( void * data , std : : any d ) {
auto dev = std : : any_cast < SP < Aquamarine : : IPointer > > ( d ) ;
Debug : : log ( LOG , " New aquamarine pointer with name {} " , dev - > getName ( ) ) ;
g_pInputManager - > newMouse ( dev ) ;
g_pInputManager - > updateCapabilities ( ) ;
} ,
nullptr ) ;
m_pAqBackend - > events . newKeyboard . registerStaticListener (
[ ] ( void * data , std : : any d ) {
auto dev = std : : any_cast < SP < Aquamarine : : IKeyboard > > ( d ) ;
Debug : : log ( LOG , " New aquamarine keyboard with name {} " , dev - > getName ( ) ) ;
g_pInputManager - > newKeyboard ( dev ) ;
g_pInputManager - > updateCapabilities ( ) ;
} ,
nullptr ) ;
m_pAqBackend - > events . newTouch . registerStaticListener (
[ ] ( void * data , std : : any d ) {
auto dev = std : : any_cast < SP < Aquamarine : : ITouch > > ( d ) ;
Debug : : log ( LOG , " New aquamarine touch with name {} " , dev - > getName ( ) ) ;
g_pInputManager - > newTouchDevice ( dev ) ;
g_pInputManager - > updateCapabilities ( ) ;
} ,
nullptr ) ;
2022-11-05 19:04:44 +01:00
2024-07-21 13:09:54 +02:00
m_pAqBackend - > events . newSwitch . registerStaticListener (
[ ] ( void * data , std : : any d ) {
auto dev = std : : any_cast < SP < Aquamarine : : ISwitch > > ( d ) ;
Debug : : log ( LOG , " New aquamarine switch with name {} " , dev - > getName ( ) ) ;
g_pInputManager - > newSwitch ( dev ) ;
} ,
nullptr ) ;
2022-11-05 19:04:44 +01:00
2024-07-21 13:09:54 +02:00
m_pAqBackend - > events . newTablet . registerStaticListener (
[ ] ( void * data , std : : any d ) {
auto dev = std : : any_cast < SP < Aquamarine : : ITablet > > ( d ) ;
Debug : : log ( LOG , " New aquamarine tablet with name {} " , dev - > getName ( ) ) ;
g_pInputManager - > newTablet ( dev ) ;
} ,
nullptr ) ;
2022-07-10 15:41:26 +02:00
2024-07-21 13:09:54 +02:00
m_pAqBackend - > events . newTabletPad . registerStaticListener (
[ ] ( void * data , std : : any d ) {
auto dev = std : : any_cast < SP < Aquamarine : : ITabletPad > > ( d ) ;
Debug : : log ( LOG , " New aquamarine tablet pad with name {} " , dev - > getName ( ) ) ;
g_pInputManager - > newTabletPad ( dev ) ;
} ,
nullptr ) ;
2022-07-10 15:41:26 +02:00
2024-07-21 13:09:54 +02:00
if ( m_pAqBackend - > hasSession ( ) ) {
m_pAqBackend - > session - > events . changeActive . registerStaticListener (
[ this ] ( void * , std : : any ) {
if ( m_pAqBackend - > session - > active ) {
Debug : : log ( LOG , " Session got activated! " ) ;
2022-08-05 13:03:37 +02:00
2024-07-21 13:09:54 +02:00
m_bSessionActive = true ;
2022-08-27 23:10:13 +02:00
2024-08-26 20:24:30 +02:00
for ( auto const & m : m_vMonitors ) {
2024-10-20 00:03:29 +02:00
scheduleFrameForMonitor ( m ) ;
g_pHyprRenderer - > applyMonitorRule ( m , & m - > activeMonitorRule , true ) ;
2024-07-21 13:09:54 +02:00
}
2022-04-21 15:59:28 +02:00
2024-07-21 13:09:54 +02:00
g_pConfigManager - > m_bWantsMonitorReload = true ;
2024-08-13 19:27:00 +02:00
g_pCursorManager - > syncGsettings ( ) ;
2024-07-21 13:09:54 +02:00
} else {
Debug : : log ( LOG , " Session got deactivated! " ) ;
2024-01-19 16:20:22 +01:00
2024-07-21 13:09:54 +02:00
m_bSessionActive = false ;
2024-08-26 17:25:39 +02:00
for ( auto const & m : m_vMonitors ) {
2024-07-21 13:09:54 +02:00
m - > noFrameSchedule = true ;
m - > framesToSkip = 1 ;
}
}
} ,
nullptr ) ;
}
}
2024-01-19 16:20:22 +01:00
2024-07-21 13:09:54 +02:00
void CCompositor : : removeAllSignals ( ) {
;
2024-01-19 16:20:22 +01:00
}
2024-05-08 19:11:08 +02:00
void CCompositor : : cleanEnvironment ( ) {
// in compositor constructor
unsetenv ( " WAYLAND_DISPLAY " ) ;
// in startCompositor
unsetenv ( " HYPRLAND_INSTANCE_SIGNATURE " ) ;
// in main
unsetenv ( " HYPRLAND_CMD " ) ;
unsetenv ( " XDG_BACKEND " ) ;
2024-08-15 18:14:48 +02:00
if ( m_bDesktopEnvSet )
unsetenv ( " XDG_CURRENT_DESKTOP " ) ;
2024-05-08 19:11:08 +02:00
2024-08-16 09:19:08 +02:00
if ( m_pAqBackend - > hasSession ( ) & & ! envEnabled ( " HYPRLAND_NO_SD_VARS " ) ) {
2024-05-08 19:11:08 +02:00
const auto CMD =
# ifdef USES_SYSTEMD
2024-06-23 09:13:17 +02:00
" systemctl --user unset-environment DISPLAY WAYLAND_DISPLAY HYPRLAND_INSTANCE_SIGNATURE XDG_CURRENT_DESKTOP QT_QPA_PLATFORMTHEME PATH XDG_DATA_DIRS && hash "
2024-05-08 19:11:08 +02:00
" dbus-update-activation-environment 2>/dev/null && "
# endif
2024-06-23 09:13:17 +02:00
" dbus-update-activation-environment --systemd WAYLAND_DISPLAY XDG_CURRENT_DESKTOP HYPRLAND_INSTANCE_SIGNATURE QT_QPA_PLATFORMTHEME PATH XDG_DATA_DIRS " ;
2024-05-08 19:11:08 +02:00
g_pKeybindManager - > spawn ( CMD ) ;
}
}
2024-07-24 19:07:36 +02:00
void CCompositor : : stopCompositor ( ) {
Debug : : log ( LOG , " Hyprland is stopping! " ) ;
// this stops the wayland loop, wl_display_run
wl_display_terminate ( m_sWLDisplay ) ;
m_bIsShuttingDown = true ;
}
2022-07-13 18:18:23 +02:00
void CCompositor : : cleanup ( ) {
2024-07-24 19:07:36 +02:00
if ( ! m_sWLDisplay )
2022-04-21 15:59:28 +02:00
return ;
2024-02-23 17:48:27 +01:00
signal ( SIGABRT , SIG_DFL ) ;
signal ( SIGSEGV , SIG_DFL ) ;
2023-08-08 16:16:34 +02:00
removeLockFile ( ) ;
2023-12-06 15:46:18 +01:00
m_bIsShuttingDown = true ;
Debug : : shuttingDown = true ;
2022-10-24 01:14:42 +02:00
2023-07-06 15:24:49 +02:00
# ifdef USES_SYSTEMD
2024-12-07 18:51:18 +01:00
if ( NSystemd : : sdBooted ( ) > 0 & & ! envEnabled ( " HYPRLAND_NO_SD_NOTIFY " ) )
NSystemd : : sdNotify ( 0 , " STOPPING=1 " ) ;
2023-07-06 15:24:49 +02:00
# endif
2024-05-08 19:11:08 +02:00
cleanEnvironment ( ) ;
2023-03-01 00:11:49 +01:00
// unload all remaining plugins while the compositor is
// still in a normal working state.
g_pPluginSystem - > unloadAllPlugins ( ) ;
2024-06-08 10:07:59 +02:00
m_pLastFocus . reset ( ) ;
2024-04-27 13:43:12 +02:00
m_pLastWindow . reset ( ) ;
2022-05-29 00:07:31 +02:00
2022-06-30 15:44:26 +02:00
m_vWorkspaces . clear ( ) ;
m_vWindows . clear ( ) ;
2022-05-29 00:00:47 +02:00
2024-08-26 20:24:30 +02:00
for ( auto const & m : m_vMonitors ) {
2024-10-20 00:03:29 +02:00
g_pHyprOpenGL - > destroyMonitorResources ( m ) ;
2022-08-22 14:17:25 +02:00
2024-07-21 13:09:54 +02:00
m - > output - > state - > setEnabled ( false ) ;
2024-01-28 02:57:13 +01:00
m - > state . commit ( ) ;
2022-08-25 21:35:47 +02:00
}
2024-05-25 22:43:51 +02:00
g_pXWayland . reset ( ) ;
2022-10-06 19:43:50 +02:00
2024-05-25 22:43:51 +02:00
m_vMonitors . clear ( ) ;
2022-04-21 15:59:28 +02:00
2024-04-08 16:32:31 +02:00
wl_display_destroy_clients ( g_pCompositor - > m_sWLDisplay ) ;
2024-01-19 16:20:22 +01:00
removeAllSignals ( ) ;
2024-02-24 03:50:54 +01:00
g_pInputManager . reset ( ) ;
2024-01-19 16:20:22 +01:00
g_pDecorationPositioner . reset ( ) ;
2024-03-09 17:52:59 +01:00
g_pCursorManager . reset ( ) ;
2024-01-19 16:20:22 +01:00
g_pPluginSystem . reset ( ) ;
g_pHyprNotificationOverlay . reset ( ) ;
g_pDebugOverlay . reset ( ) ;
g_pEventManager . reset ( ) ;
g_pSessionLockManager . reset ( ) ;
g_pProtocolManager . reset ( ) ;
g_pHyprRenderer . reset ( ) ;
g_pHyprOpenGL . reset ( ) ;
g_pThreadManager . reset ( ) ;
g_pConfigManager . reset ( ) ;
g_pLayoutManager . reset ( ) ;
g_pHyprError . reset ( ) ;
g_pConfigManager . reset ( ) ;
g_pAnimationManager . reset ( ) ;
g_pKeybindManager . reset ( ) ;
g_pHookSystem . reset ( ) ;
g_pWatchdog . reset ( ) ;
g_pXWaylandManager . reset ( ) ;
2024-05-10 19:27:57 +02:00
g_pPointerManager . reset ( ) ;
g_pSeatManager . reset ( ) ;
2024-06-13 12:08:02 +02:00
g_pHyprCtl . reset ( ) ;
g_pEventLoopManager . reset ( ) ;
2024-05-03 15:42:08 +02:00
2024-07-21 13:09:54 +02:00
if ( m_pAqBackend )
m_pAqBackend . reset ( ) ;
2024-05-03 15:42:08 +02:00
2024-06-06 20:27:09 +02:00
if ( m_critSigSource )
wl_event_source_remove ( m_critSigSource ) ;
2024-07-24 19:07:36 +02:00
// this frees all wayland resources, including sockets
wl_display_destroy ( m_sWLDisplay ) ;
2024-07-31 21:00:14 +02:00
Debug : : close ( ) ;
2022-03-18 23:52:36 +01:00
}
2022-03-17 17:08:54 +01:00
2023-03-05 14:37:21 +01:00
void CCompositor : : initManagers ( eManagersInitStage stage ) {
switch ( stage ) {
case STAGE_PRIORITY : {
2024-04-07 04:31:51 +02:00
Debug : : log ( LOG , " Creating the EventLoopManager! " ) ;
2024-07-05 23:05:03 +02:00
g_pEventLoopManager = std : : make_unique < CEventLoopManager > ( m_sWLDisplay , m_sWLEventLoop ) ;
2024-04-07 04:31:51 +02:00
2023-03-05 14:37:21 +01:00
Debug : : log ( LOG , " Creating the HookSystem! " ) ;
g_pHookSystem = std : : make_unique < CHookSystemManager > ( ) ;
2022-03-19 17:48:18 +01:00
2023-03-05 14:37:21 +01:00
Debug : : log ( LOG , " Creating the KeybindManager! " ) ;
g_pKeybindManager = std : : make_unique < CKeybindManager > ( ) ;
2022-04-23 21:47:16 +02:00
2023-03-05 14:37:21 +01:00
Debug : : log ( LOG , " Creating the AnimationManager! " ) ;
g_pAnimationManager = std : : make_unique < CAnimationManager > ( ) ;
2022-07-16 15:57:31 +02:00
2023-03-05 14:37:21 +01:00
Debug : : log ( LOG , " Creating the ConfigManager! " ) ;
g_pConfigManager = std : : make_unique < CConfigManager > ( ) ;
2022-03-17 17:08:54 +01:00
2023-03-05 14:37:21 +01:00
Debug : : log ( LOG , " Creating the CHyprError! " ) ;
g_pHyprError = std : : make_unique < CHyprError > ( ) ;
2023-01-20 20:48:07 +01:00
2023-03-05 14:37:21 +01:00
Debug : : log ( LOG , " Creating the LayoutManager! " ) ;
g_pLayoutManager = std : : make_unique < CLayoutManager > ( ) ;
2022-03-17 17:08:54 +01:00
2024-04-23 02:27:08 +02:00
Debug : : log ( LOG , " Creating the TokenManager! " ) ;
g_pTokenManager = std : : make_unique < CTokenManager > ( ) ;
2023-03-05 14:37:21 +01:00
g_pConfigManager - > init ( ) ;
2023-09-29 18:04:20 +02:00
g_pWatchdog = std : : make_unique < CWatchdog > ( ) ; // requires config
2024-08-12 19:19:03 +02:00
// wait for watchdog to initialize to not hit data races in reading config values.
while ( ! g_pWatchdog - > m_bWatchdogInitialized ) {
std : : this_thread : : yield ( ) ;
}
2024-05-05 23:18:10 +02:00
Debug : : log ( LOG , " Creating the PointerManager! " ) ;
g_pPointerManager = std : : make_unique < CPointerManager > ( ) ;
2024-07-21 13:09:54 +02:00
Debug : : log ( LOG , " Creating the EventManager! " ) ;
g_pEventManager = std : : make_unique < CEventManager > ( ) ;
2023-03-05 14:37:21 +01:00
} break ;
2024-06-08 10:07:59 +02:00
case STAGE_BASICINIT : {
2024-06-09 21:10:46 +02:00
Debug : : log ( LOG , " Creating the CHyprOpenGLImpl! " ) ;
g_pHyprOpenGL = std : : make_unique < CHyprOpenGLImpl > ( ) ;
2024-06-08 10:07:59 +02:00
Debug : : log ( LOG , " Creating the ProtocolManager! " ) ;
g_pProtocolManager = std : : make_unique < CProtocolManager > ( ) ;
Debug : : log ( LOG , " Creating the SeatManager! " ) ;
g_pSeatManager = std : : make_unique < CSeatManager > ( ) ;
} break ;
2023-03-05 14:37:21 +01:00
case STAGE_LATE : {
Debug : : log ( LOG , " Creating the ThreadManager! " ) ;
g_pThreadManager = std : : make_unique < CThreadManager > ( ) ;
2022-03-17 20:22:29 +01:00
2024-02-05 02:43:45 +01:00
Debug : : log ( LOG , " Creating CHyprCtl " ) ;
g_pHyprCtl = std : : make_unique < CHyprCtl > ( ) ;
2023-03-05 14:37:21 +01:00
Debug : : log ( LOG , " Creating the InputManager! " ) ;
g_pInputManager = std : : make_unique < CInputManager > ( ) ;
2022-04-04 19:44:25 +02:00
2023-03-05 14:37:21 +01:00
Debug : : log ( LOG , " Creating the HyprRenderer! " ) ;
g_pHyprRenderer = std : : make_unique < CHyprRenderer > ( ) ;
2022-03-19 15:59:53 +01:00
2023-03-05 14:37:21 +01:00
Debug : : log ( LOG , " Creating the XWaylandManager! " ) ;
g_pXWaylandManager = std : : make_unique < CHyprXWaylandManager > ( ) ;
2022-12-05 18:05:15 +01:00
2023-03-05 14:37:21 +01:00
Debug : : log ( LOG , " Creating the SessionLockManager! " ) ;
g_pSessionLockManager = std : : make_unique < CSessionLockManager > ( ) ;
2022-05-28 17:32:19 +02:00
2023-03-05 14:37:21 +01:00
Debug : : log ( LOG , " Creating the HyprDebugOverlay! " ) ;
g_pDebugOverlay = std : : make_unique < CHyprDebugOverlay > ( ) ;
2023-02-27 13:32:38 +01:00
2023-03-05 14:37:21 +01:00
Debug : : log ( LOG , " Creating the HyprNotificationOverlay! " ) ;
g_pHyprNotificationOverlay = std : : make_unique < CHyprNotificationOverlay > ( ) ;
2022-03-17 17:08:54 +01:00
2023-03-05 14:37:21 +01:00
Debug : : log ( LOG , " Creating the PluginSystem! " ) ;
g_pPluginSystem = std : : make_unique < CPluginSystem > ( ) ;
2023-05-01 16:10:53 +02:00
g_pConfigManager - > handlePluginLoads ( ) ;
2023-11-11 15:37:17 +01:00
Debug : : log ( LOG , " Creating the DecorationPositioner! " ) ;
g_pDecorationPositioner = std : : make_unique < CDecorationPositioner > ( ) ;
2024-03-09 17:52:59 +01:00
Debug : : log ( LOG , " Creating the CursorManager! " ) ;
g_pCursorManager = std : : make_unique < CCursorManager > ( ) ;
2024-05-25 22:43:51 +02:00
2024-12-06 15:45:02 +01:00
Debug : : log ( LOG , " Creating the VersionKeeper! " ) ;
g_pVersionKeeperMgr = std : : make_unique < CVersionKeeperManager > ( ) ;
2024-05-25 22:43:51 +02:00
Debug : : log ( LOG , " Starting XWayland " ) ;
2024-09-05 18:26:46 +02:00
g_pXWayland = std : : make_unique < CXWayland > ( g_pCompositor - > m_bEnableXwayland ) ;
2023-03-05 14:37:21 +01:00
} break ;
default : UNREACHABLE ( ) ;
}
}
2022-08-01 23:31:25 +02:00
2023-08-08 16:16:34 +02:00
void CCompositor : : createLockFile ( ) {
2024-04-28 23:25:24 +02:00
const auto PATH = m_szInstancePath + " /hyprland.lock " ;
2023-08-08 16:16:34 +02:00
std : : ofstream ofs ( PATH , std : : ios : : trunc ) ;
ofs < < m_iHyprlandPID < < " \n " < < m_szWLDisplaySocket < < " \n " ;
ofs . close ( ) ;
}
void CCompositor : : removeLockFile ( ) {
2024-04-28 23:25:24 +02:00
const auto PATH = m_szInstancePath + " /hyprland.lock " ;
2023-08-08 16:16:34 +02:00
if ( std : : filesystem : : exists ( PATH ) )
std : : filesystem : : remove ( PATH ) ;
}
2023-11-01 19:53:36 +01:00
void CCompositor : : prepareFallbackOutput ( ) {
// create a backup monitor
2024-07-21 13:09:54 +02:00
SP < Aquamarine : : IBackendImplementation > headless ;
2024-08-26 20:24:30 +02:00
for ( auto const & impl : m_pAqBackend - > getImplementations ( ) ) {
2024-07-21 13:09:54 +02:00
if ( impl - > type ( ) = = Aquamarine : : AQ_BACKEND_HEADLESS ) {
headless = impl ;
break ;
}
}
2023-11-01 19:53:36 +01:00
if ( ! headless ) {
2024-07-21 13:09:54 +02:00
Debug : : log ( WARN , " No headless in prepareFallbackOutput?! " ) ;
2023-11-01 19:53:36 +01:00
return ;
}
2024-07-21 13:09:54 +02:00
headless - > createOutput ( ) ;
2023-11-01 19:53:36 +01:00
}
2024-07-22 13:16:25 +02:00
void CCompositor : : startCompositor ( ) {
2022-03-16 21:37:21 +01:00
signal ( SIGPIPE , SIG_IGN ) ;
2024-08-16 09:19:08 +02:00
if (
/* Session-less Hyprland usually means a nest, don't update the env in that case */
m_pAqBackend - > hasSession ( ) & &
/* Activation environment management is not disabled */
! envEnabled ( " HYPRLAND_NO_SD_VARS " ) ) {
2023-12-26 18:16:59 +01:00
const auto CMD =
2023-04-15 21:03:09 +02:00
# ifdef USES_SYSTEMD
2024-06-23 09:13:17 +02:00
" systemctl --user import-environment DISPLAY WAYLAND_DISPLAY HYPRLAND_INSTANCE_SIGNATURE XDG_CURRENT_DESKTOP QT_QPA_PLATFORMTHEME PATH XDG_DATA_DIRS && hash "
2023-12-26 18:16:59 +01:00
" dbus-update-activation-environment 2>/dev/null && "
2023-04-15 21:03:09 +02:00
# endif
2024-06-23 09:13:17 +02:00
" dbus-update-activation-environment --systemd WAYLAND_DISPLAY XDG_CURRENT_DESKTOP HYPRLAND_INSTANCE_SIGNATURE QT_QPA_PLATFORMTHEME PATH XDG_DATA_DIRS " ;
2023-12-26 18:16:59 +01:00
g_pKeybindManager - > spawn ( CMD ) ;
}
2022-11-04 11:37:14 +01:00
2023-09-06 21:45:37 +02:00
Debug : : log ( LOG , " Running on WAYLAND_DISPLAY: {} " , m_szWLDisplaySocket ) ;
2022-03-16 21:37:21 +01:00
2023-11-01 19:53:36 +01:00
prepareFallbackOutput ( ) ;
2023-09-29 18:51:07 +02:00
g_pHyprRenderer - > setCursorFromName ( " left_ptr " ) ;
2022-03-16 21:37:21 +01:00
2023-01-05 20:17:55 +01:00
# ifdef USES_SYSTEMD
2024-12-07 18:51:18 +01:00
if ( NSystemd : : sdBooted ( ) > 0 ) {
2023-01-05 20:17:55 +01:00
// tell systemd that we are ready so it can start other bond, following, related units
2023-12-21 22:27:12 +01:00
if ( ! envEnabled ( " HYPRLAND_NO_SD_NOTIFY " ) )
2024-12-07 18:51:18 +01:00
NSystemd : : sdNotify ( 0 , " READY=1 " ) ;
2023-12-21 22:27:12 +01:00
} else
2023-01-05 20:17:55 +01:00
Debug : : log ( LOG , " systemd integration is baked in but system itself is not booted à la systemd! " ) ;
# endif
2023-08-08 16:16:34 +02:00
createLockFile ( ) ;
2024-06-08 10:07:59 +02:00
EMIT_HOOK_EVENT ( " ready " , nullptr ) ;
2022-03-16 21:37:21 +01:00
// This blocks until we are done.
Debug : : log ( LOG , " Hyprland is ready, running the event loop! " ) ;
2024-07-05 23:05:03 +02:00
g_pEventLoopManager - > enterLoop ( ) ;
2022-03-17 20:22:29 +01:00
}
2024-10-20 00:03:29 +02:00
PHLMONITOR CCompositor : : getMonitorFromID ( const MONITORID & id ) {
2024-08-26 20:24:30 +02:00
for ( auto const & m : m_vMonitors ) {
2024-08-08 21:01:50 +02:00
if ( m - > ID = = id ) {
2024-10-20 00:03:29 +02:00
return m ;
2022-03-17 20:22:29 +01:00
}
}
return nullptr ;
}
2024-10-20 00:03:29 +02:00
PHLMONITOR CCompositor : : getMonitorFromName ( const std : : string & name ) {
2024-08-26 20:24:30 +02:00
for ( auto const & m : m_vMonitors ) {
2022-06-30 15:44:26 +02:00
if ( m - > szName = = name ) {
2024-10-20 00:03:29 +02:00
return m ;
2023-07-09 23:10:35 +02:00
}
}
return nullptr ;
}
2024-10-20 00:03:29 +02:00
PHLMONITOR CCompositor : : getMonitorFromDesc ( const std : : string & desc ) {
2024-08-26 20:24:30 +02:00
for ( auto const & m : m_vMonitors ) {
2023-11-30 02:48:10 +01:00
if ( m - > szDescription . starts_with ( desc ) )
2024-10-20 00:03:29 +02:00
return m ;
2022-05-29 20:15:34 +02:00
}
return nullptr ;
}
2024-10-20 00:03:29 +02:00
PHLMONITOR CCompositor : : getMonitorFromCursor ( ) {
2024-05-05 23:18:10 +02:00
return getMonitorFromVector ( g_pPointerManager - > position ( ) ) ;
2022-03-18 22:35:51 +01:00
}
2024-10-20 00:03:29 +02:00
PHLMONITOR CCompositor : : getMonitorFromVector ( const Vector2D & point ) {
2024-10-26 03:06:13 +02:00
PHLMONITOR mon ;
2024-08-26 20:24:30 +02:00
for ( auto const & m : m_vMonitors ) {
2024-05-05 23:18:10 +02:00
if ( CBox { m - > vecPosition , m - > vecSize } . containsPoint ( point ) ) {
mon = m ;
break ;
}
}
2022-03-20 11:22:55 +01:00
2024-05-05 23:18:10 +02:00
if ( ! mon ) {
2024-10-26 03:06:13 +02:00
float bestDistance = 0.f ;
PHLMONITOR pBestMon ;
2022-05-25 18:40:03 +02:00
2024-08-26 17:25:39 +02:00
for ( auto const & m : m_vMonitors ) {
2022-06-30 15:44:26 +02:00
float dist = vecToRectDistanceSquared ( point , m - > vecPosition , m - > vecPosition + m - > vecSize ) ;
2022-05-25 18:40:03 +02:00
if ( dist < bestDistance | | ! pBestMon ) {
bestDistance = dist ;
2024-05-05 23:18:10 +02:00
pBestMon = m ;
2022-05-25 18:40:03 +02:00
}
}
if ( ! pBestMon ) { // ?????
Debug : : log ( WARN , " getMonitorFromVector no close mon??? " ) ;
2024-10-20 00:03:29 +02:00
return m_vMonitors . front ( ) ;
2022-05-25 18:40:03 +02:00
}
2024-10-20 00:03:29 +02:00
return pBestMon ;
2022-03-20 11:22:55 +01:00
}
2024-10-20 00:03:29 +02:00
return mon ;
2022-03-20 11:22:55 +01:00
}
2024-04-27 13:43:12 +02:00
void CCompositor : : removeWindowFromVectorSafe ( PHLWINDOW pWindow ) {
if ( ! pWindow - > m_bFadingOut ) {
2024-04-19 23:16:35 +02:00
EMIT_HOOK_EVENT ( " destroyWindow " , pWindow ) ;
2024-04-27 13:43:12 +02:00
std : : erase_if ( m_vWindows , [ & ] ( SP < CWindow > & el ) { return el = = pWindow ; } ) ;
std : : erase_if ( m_vWindowsFadingOut , [ & ] ( PHLWINDOWREF el ) { return el . lock ( ) = = pWindow ; } ) ;
2024-04-06 15:51:35 +02:00
}
2022-03-18 22:53:27 +01:00
}
2024-10-20 00:03:29 +02:00
bool CCompositor : : monitorExists ( PHLMONITOR pMonitor ) {
2024-08-26 20:24:30 +02:00
for ( auto const & m : m_vRealMonitors ) {
2024-10-20 00:03:29 +02:00
if ( m = = pMonitor )
2024-02-07 01:18:47 +01:00
return true ;
}
return false ;
}
2024-04-27 13:43:12 +02:00
PHLWINDOW CCompositor : : vectorToWindowUnified ( const Vector2D & pos , uint8_t properties , PHLWINDOW pIgnoreWindow ) {
2024-03-03 19:39:20 +01:00
const auto PMONITOR = getMonitorFromVector ( pos ) ;
static auto PRESIZEONBORDER = CConfigValue < Hyprlang : : INT > ( " general:resize_on_border " ) ;
static auto PBORDERSIZE = CConfigValue < Hyprlang : : INT > ( " general:border_size " ) ;
static auto PBORDERGRABEXTEND = CConfigValue < Hyprlang : : INT > ( " general:extend_border_grab_area " ) ;
static auto PSPECIALFALLTHRU = CConfigValue < Hyprlang : : INT > ( " input:special_fallthrough " ) ;
const auto BORDER_GRAB_AREA = * PRESIZEONBORDER ? * PBORDERSIZE + * PBORDERGRABEXTEND : 0 ;
2022-05-31 14:01:00 +02:00
2022-09-10 13:11:02 +02:00
// pinned windows on top of floating regardless
2024-02-04 16:40:20 +01:00
if ( properties & ALLOW_FLOATING ) {
2024-08-26 20:24:30 +02:00
for ( auto const & w : m_vWindows | std : : views : : reverse ) {
2024-07-13 12:36:29 +02:00
if ( w - > m_bIsFloating & & w - > m_bIsMapped & & ! w - > isHidden ( ) & & ! w - > m_bX11ShouldntFocus & & w - > m_bPinned & & ! w - > m_sWindowData . noFocus . valueOrDefault ( ) & &
w ! = pIgnoreWindow ) {
2024-10-31 00:20:32 +01:00
const auto BB = w - > getWindowBoxUnified ( properties ) ;
CBox box = BB . copy ( ) . expand ( ! w - > isX11OverrideRedirect ( ) ? BORDER_GRAB_AREA : 0 ) ;
2024-05-05 23:18:10 +02:00
if ( box . containsPoint ( g_pPointerManager - > position ( ) ) )
2024-04-27 13:43:12 +02:00
return w ;
2024-02-04 16:40:20 +01:00
if ( ! w - > m_bIsX11 ) {
if ( w - > hasPopupAt ( pos ) )
2024-04-27 13:43:12 +02:00
return w ;
2024-02-04 16:40:20 +01:00
}
2022-09-10 13:11:02 +02:00
}
2022-09-25 20:07:48 +02:00
}
2022-09-10 13:11:02 +02:00
}
2024-04-27 13:43:12 +02:00
auto windowForWorkspace = [ & ] ( bool special ) - > PHLWINDOW {
auto floating = [ & ] ( bool aboveFullscreen ) - > PHLWINDOW {
2024-08-26 20:24:30 +02:00
for ( auto const & w : m_vWindows | std : : views : : reverse ) {
2023-11-18 18:00:24 +01:00
2024-04-02 21:32:39 +02:00
if ( special & & ! w - > onSpecialWorkspace ( ) ) // because special floating may creep up into regular
2023-11-18 18:00:24 +01:00
continue ;
2024-11-22 17:01:02 +01:00
if ( ! w - > m_pWorkspace )
continue ;
2024-10-27 19:45:38 +01:00
const auto PWINDOWMONITOR = w - > m_pMonitor . lock ( ) ;
2024-03-03 18:04:39 +01:00
// to avoid focusing windows behind special workspaces from other monitors
2024-10-31 00:20:32 +01:00
if ( ! * PSPECIALFALLTHRU & & PWINDOWMONITOR & & PWINDOWMONITOR - > activeSpecialWorkspace & & w - > m_pWorkspace ! = PWINDOWMONITOR - > activeSpecialWorkspace ) {
const auto BB = w - > getWindowBoxUnified ( properties ) ;
if ( BB . x > = PWINDOWMONITOR - > vecPosition . x & & BB . y > = PWINDOWMONITOR - > vecPosition . y & &
BB . x + BB . width < = PWINDOWMONITOR - > vecPosition . x + PWINDOWMONITOR - > vecSize . x & &
BB . y + BB . height < = PWINDOWMONITOR - > vecPosition . y + PWINDOWMONITOR - > vecSize . y )
continue ;
}
2024-03-03 18:04:39 +01:00
2024-11-22 17:01:02 +01:00
if ( w - > m_bIsFloating & & w - > m_bIsMapped & & w - > m_pWorkspace - > isVisible ( ) & & ! w - > isHidden ( ) & & ! w - > m_bPinned & & ! w - > m_sWindowData . noFocus . valueOrDefault ( ) & &
2024-04-27 13:43:12 +02:00
w ! = pIgnoreWindow & & ( ! aboveFullscreen | | w - > m_bCreatedOverFullscreen ) ) {
2024-02-04 16:40:20 +01:00
// OR windows should add focus to parent
2024-08-30 14:12:23 +02:00
if ( w - > m_bX11ShouldntFocus & & ! w - > isX11OverrideRedirect ( ) )
2024-02-04 16:40:20 +01:00
continue ;
2022-10-01 19:25:02 +02:00
2024-10-31 00:20:32 +01:00
const auto BB = w - > getWindowBoxUnified ( properties ) ;
CBox box = BB . copy ( ) . expand ( ! w - > isX11OverrideRedirect ( ) ? BORDER_GRAB_AREA : 0 ) ;
2024-05-05 23:18:10 +02:00
if ( box . containsPoint ( g_pPointerManager - > position ( ) ) ) {
2023-11-18 18:00:24 +01:00
2024-08-30 14:12:23 +02:00
if ( w - > m_bIsX11 & & w - > isX11OverrideRedirect ( ) & & ! w - > m_pXWaylandSurface - > wantsFocus ( ) ) {
2024-02-04 16:40:20 +01:00
// Override Redirect
2024-04-27 13:43:12 +02:00
return g_pCompositor - > m_pLastWindow . lock ( ) ; // we kinda trick everything here.
// TODO: this is wrong, we should focus the parent, but idk how to get it considering it's nullptr in most cases.
2024-02-04 16:40:20 +01:00
}
2022-10-01 19:25:02 +02:00
2024-04-27 13:43:12 +02:00
return w ;
2024-02-04 16:40:20 +01:00
}
if ( ! w - > m_bIsX11 ) {
if ( w - > hasPopupAt ( pos ) )
2024-04-27 13:43:12 +02:00
return w ;
2024-02-04 16:40:20 +01:00
}
2023-11-18 18:00:24 +01:00
}
2022-10-01 19:25:02 +02:00
}
2024-02-22 16:42:17 +01:00
return nullptr ;
} ;
if ( properties & ALLOW_FLOATING ) {
// first loop over floating cuz they're above, m_lWindows should be sorted bottom->top, for tiled it doesn't matter.
auto found = floating ( true ) ;
if ( found )
return found ;
2023-11-18 18:00:24 +01:00
}
2022-07-12 09:49:56 +02:00
2024-02-04 16:40:20 +01:00
if ( properties & FLOATING_ONLY )
2024-02-22 16:42:17 +01:00
return floating ( false ) ;
2024-02-04 16:40:20 +01:00
2024-08-08 21:01:50 +02:00
const WORKSPACEID WSPID = special ? PMONITOR - > activeSpecialWorkspaceID ( ) : PMONITOR - > activeWorkspaceID ( ) ;
const auto PWORKSPACE = getWorkspaceByID ( WSPID ) ;
2023-12-03 13:53:12 +01:00
if ( PWORKSPACE - > m_bHasFullscreenWindow )
2024-11-22 17:01:02 +01:00
return PWORKSPACE - > getFullscreenWindow ( ) ;
2023-12-03 13:53:12 +01:00
2024-02-22 16:42:17 +01:00
auto found = floating ( false ) ;
if ( found )
return found ;
2023-11-18 18:00:24 +01:00
// for windows, we need to check their extensions too, first.
2024-08-26 20:24:30 +02:00
for ( auto const & w : m_vWindows ) {
2024-04-02 21:32:39 +02:00
if ( special ! = w - > onSpecialWorkspace ( ) )
2023-11-18 18:00:24 +01:00
continue ;
2024-11-22 17:01:02 +01:00
if ( ! w - > m_pWorkspace )
continue ;
2024-08-08 21:01:50 +02:00
if ( ! w - > m_bIsX11 & & ! w - > m_bIsFloating & & w - > m_bIsMapped & & w - > workspaceID ( ) = = WSPID & & ! w - > isHidden ( ) & & ! w - > m_bX11ShouldntFocus & &
2024-07-11 16:10:42 +02:00
! w - > m_sWindowData . noFocus . valueOrDefault ( ) & & w ! = pIgnoreWindow ) {
2023-12-03 13:53:12 +01:00
if ( w - > hasPopupAt ( pos ) )
2024-04-27 13:43:12 +02:00
return w ;
2022-07-12 09:49:56 +02:00
}
2022-09-25 20:07:48 +02:00
}
2023-12-03 13:53:12 +01:00
2024-08-26 20:24:30 +02:00
for ( auto const & w : m_vWindows ) {
2024-04-02 21:32:39 +02:00
if ( special ! = w - > onSpecialWorkspace ( ) )
2023-11-18 18:00:24 +01:00
continue ;
2022-03-20 11:14:24 +01:00
2024-11-22 17:01:02 +01:00
if ( ! w - > m_pWorkspace )
continue ;
2024-10-31 00:20:32 +01:00
if ( ! w - > m_bIsFloating & & w - > m_bIsMapped & & w - > workspaceID ( ) = = WSPID & & ! w - > isHidden ( ) & & ! w - > m_bX11ShouldntFocus & & ! w - > m_sWindowData . noFocus . valueOrDefault ( ) & &
w ! = pIgnoreWindow ) {
CBox box = ( properties & USE_PROP_TILED ) ? w - > getWindowBoxUnified ( properties ) : CBox { w - > m_vPosition , w - > m_vSize } ;
if ( box . containsPoint ( pos ) )
return w ;
}
2022-07-12 09:49:56 +02:00
}
2022-03-19 20:30:21 +01:00
2023-11-18 18:00:24 +01:00
return nullptr ;
} ;
// special workspace
2024-04-02 21:32:39 +02:00
if ( PMONITOR - > activeSpecialWorkspace & & ! * PSPECIALFALLTHRU )
2023-11-18 18:00:24 +01:00
return windowForWorkspace ( true ) ;
2024-04-02 21:32:39 +02:00
if ( PMONITOR - > activeSpecialWorkspace ) {
2024-01-09 13:17:55 +01:00
const auto PWINDOW = windowForWorkspace ( true ) ;
if ( PWINDOW )
return PWINDOW ;
}
2023-11-18 18:00:24 +01:00
return windowForWorkspace ( false ) ;
2022-03-19 20:30:21 +01:00
}
2024-06-08 10:07:59 +02:00
SP < CWLSurfaceResource > CCompositor : : vectorWindowToSurface ( const Vector2D & pos , PHLWINDOW pWindow , Vector2D & sl ) {
2022-04-02 13:02:16 +02:00
2024-04-27 13:43:12 +02:00
if ( ! validMapped ( pWindow ) )
2022-04-02 13:02:16 +02:00
return nullptr ;
RASSERT ( ! pWindow - > m_bIsX11 , " Cannot call vectorWindowToSurface on an X11 window! " ) ;
2024-05-11 00:28:33 +02:00
// try popups first
2024-06-08 10:07:59 +02:00
const auto PPOPUP = pWindow - > m_pPopupHead - > at ( pos ) ;
2022-04-02 13:02:16 +02:00
2024-06-08 10:07:59 +02:00
if ( PPOPUP ) {
2024-05-11 00:28:33 +02:00
const auto OFF = PPOPUP - > coordsRelativeToParent ( ) ;
2024-06-08 10:07:59 +02:00
sl = pos - pWindow - > m_vRealPosition . goal ( ) - OFF ;
return PPOPUP - > m_pWLSurface - > resource ( ) ;
2024-05-11 00:28:33 +02:00
}
2022-04-02 13:02:16 +02:00
2024-06-08 10:07:59 +02:00
auto [ surf , local ] = pWindow - > m_pWLSurface - > resource ( ) - > at ( pos - pWindow - > m_vRealPosition . goal ( ) , true ) ;
if ( surf ) {
sl = local ;
return surf ;
2022-04-02 13:02:16 +02:00
}
2024-06-08 10:07:59 +02:00
return nullptr ;
2022-04-02 13:02:16 +02:00
}
2024-06-08 10:07:59 +02:00
Vector2D CCompositor : : vectorToSurfaceLocal ( const Vector2D & vec , PHLWINDOW pWindow , SP < CWLSurfaceResource > pSurface ) {
2024-04-27 13:43:12 +02:00
if ( ! validMapped ( pWindow ) )
2023-09-23 02:21:59 +02:00
return { } ;
if ( pWindow - > m_bIsX11 )
2024-03-02 01:35:17 +01:00
return vec - pWindow - > m_vRealPosition . goal ( ) ;
2023-09-23 02:21:59 +02:00
2024-05-11 00:28:33 +02:00
const auto PPOPUP = pWindow - > m_pPopupHead - > at ( vec ) ;
if ( PPOPUP )
return vec - PPOPUP - > coordsGlobal ( ) ;
2023-09-23 02:21:59 +02:00
2024-06-08 10:07:59 +02:00
std : : tuple < SP < CWLSurfaceResource > , Vector2D > iterData = { pSurface , { - 1337 , - 1337 } } ;
2023-09-23 02:21:59 +02:00
2024-06-08 10:07:59 +02:00
pWindow - > m_pWLSurface - > resource ( ) - > breadthfirst (
[ ] ( SP < CWLSurfaceResource > surf , const Vector2D & offset , void * data ) {
const auto PDATA = ( std : : tuple < SP < CWLSurfaceResource > , Vector2D > * ) data ;
if ( surf = = std : : get < 0 > ( * PDATA ) )
std : : get < 1 > ( * PDATA ) = offset ;
2023-09-23 02:21:59 +02:00
} ,
& iterData ) ;
2024-05-11 00:28:33 +02:00
CBox geom = pWindow - > m_pXDGSurface - > current . geometry ;
2023-10-30 20:36:34 +01:00
2024-06-08 10:07:59 +02:00
if ( std : : get < 1 > ( iterData ) = = Vector2D { - 1337 , - 1337 } )
2024-03-02 01:35:17 +01:00
return vec - pWindow - > m_vRealPosition . goal ( ) ;
2023-09-23 02:21:59 +02:00
2024-06-08 10:07:59 +02:00
return vec - pWindow - > m_vRealPosition . goal ( ) - std : : get < 1 > ( iterData ) + Vector2D { geom . x , geom . y } ;
2023-09-23 02:21:59 +02:00
}
2024-10-20 00:03:29 +02:00
PHLMONITOR CCompositor : : getMonitorFromOutput ( SP < Aquamarine : : IOutput > out ) {
2024-08-26 20:24:30 +02:00
for ( auto const & m : m_vMonitors ) {
2022-06-30 15:44:26 +02:00
if ( m - > output = = out ) {
2024-10-20 00:03:29 +02:00
return m ;
2022-03-19 20:56:19 +01:00
}
}
return nullptr ;
}
2024-10-20 00:03:29 +02:00
PHLMONITOR CCompositor : : getRealMonitorFromOutput ( SP < Aquamarine : : IOutput > out ) {
2024-08-26 20:24:30 +02:00
for ( auto const & m : m_vRealMonitors ) {
2024-02-18 03:24:01 +01:00
if ( m - > output = = out ) {
2024-10-20 00:03:29 +02:00
return m ;
2024-02-18 03:24:01 +01:00
}
}
return nullptr ;
}
2024-06-08 10:07:59 +02:00
void CCompositor : : focusWindow ( PHLWINDOW pWindow , SP < CWLSurfaceResource > pSurface ) {
2022-03-18 22:53:27 +01:00
2024-03-03 19:39:20 +01:00
static auto PFOLLOWMOUSE = CConfigValue < Hyprlang : : INT > ( " input:follow_mouse " ) ;
static auto PSPECIALFALLTHROUGH = CConfigValue < Hyprlang : : INT > ( " input:special_fallthrough " ) ;
2023-10-26 23:17:49 +02:00
2024-05-10 19:27:57 +02:00
if ( g_pSessionLockManager - > isSessionLocked ( ) ) {
Debug : : log ( LOG , " Refusing a keyboard focus to a window because of a sessionlock " ) ;
2022-04-18 17:16:01 +02:00
return ;
}
2024-01-19 16:45:34 +01:00
if ( ! g_pInputManager - > m_dExclusiveLSes . empty ( ) ) {
Debug : : log ( LOG , " Refusing a keyboard focus to a window because of an exclusive ls " ) ;
return ;
}
2024-08-30 14:12:23 +02:00
if ( pWindow & & pWindow - > m_bIsX11 & & pWindow - > isX11OverrideRedirect ( ) & & ! pWindow - > m_pXWaylandSurface - > wantsFocus ( ) )
2024-03-09 23:39:23 +01:00
return ;
2023-09-12 23:37:08 +02:00
g_pLayoutManager - > getCurrentLayout ( ) - > bringWindowToTop ( pWindow ) ;
2023-08-11 17:27:09 +02:00
2024-04-27 13:43:12 +02:00
if ( ! pWindow | | ! validMapped ( pWindow ) ) {
2023-10-06 02:11:47 +02:00
2024-04-27 13:43:12 +02:00
if ( m_pLastWindow . expired ( ) & & ! pWindow )
2023-10-06 02:11:47 +02:00
return ;
2024-04-27 13:43:12 +02:00
const auto PLASTWINDOW = m_pLastWindow . lock ( ) ;
m_pLastWindow . reset ( ) ;
2022-08-08 20:42:14 +02:00
2024-04-27 13:43:12 +02:00
if ( PLASTWINDOW & & PLASTWINDOW - > m_bIsMapped ) {
2022-08-08 20:42:14 +02:00
updateWindowAnimatedDecorationValues ( PLASTWINDOW ) ;
2022-08-08 20:21:11 +02:00
2022-10-28 20:18:10 +02:00
g_pXWaylandManager - > activateWindow ( PLASTWINDOW , false ) ;
2022-08-08 20:21:11 +02:00
}
2024-05-10 19:27:57 +02:00
g_pSeatManager - > setKeyboardFocus ( nullptr ) ;
2022-08-08 20:21:11 +02:00
2022-08-21 18:34:38 +02:00
g_pEventManager - > postEvent ( SHyprIPCEvent { " activewindow " , " , " } ) ;
2024-05-01 14:57:27 +02:00
g_pEventManager - > postEvent ( SHyprIPCEvent { " activewindowv2 " , " " } ) ;
2022-08-21 18:34:38 +02:00
2024-04-27 13:43:12 +02:00
EMIT_HOOK_EVENT ( " activeWindow " , ( PHLWINDOW ) nullptr ) ;
2023-02-19 21:54:53 +01:00
2022-10-24 13:25:36 +02:00
g_pLayoutManager - > getCurrentLayout ( ) - > onWindowFocusChange ( nullptr ) ;
2024-06-08 10:07:59 +02:00
m_pLastFocus . reset ( ) ;
2022-10-31 13:26:07 +01:00
g_pInputManager - > recheckIdleInhibitorStatus ( ) ;
2022-05-14 14:37:57 +02:00
return ;
}
2024-07-11 16:10:42 +02:00
if ( pWindow - > m_sWindowData . noFocus . valueOrDefault ( ) ) {
2022-05-14 20:56:21 +02:00
Debug : : log ( LOG , " Ignoring focus to nofocus window! " ) ;
2022-03-18 22:53:27 +01:00
return ;
}
2024-11-09 03:27:01 +01:00
if ( m_pLastWindow . lock ( ) = = pWindow & & g_pSeatManager - > state . keyboardFocus = = pSurface & & g_pSeatManager - > state . keyboardFocus )
2022-04-05 18:29:58 +02:00
return ;
2022-09-10 13:11:02 +02:00
if ( pWindow - > m_bPinned )
2024-04-02 21:32:39 +02:00
pWindow - > m_pWorkspace = m_pLastMonitor - > activeWorkspace ;
2022-09-10 13:11:02 +02:00
2024-10-27 19:45:38 +01:00
const auto PMONITOR = pWindow - > m_pMonitor . lock ( ) ;
2024-01-07 12:06:33 +01:00
2024-11-22 17:01:02 +01:00
if ( ! pWindow - > m_pWorkspace | | ! pWindow - > m_pWorkspace - > isVisible ( ) ) {
2024-04-02 21:32:39 +02:00
const auto PWORKSPACE = pWindow - > m_pWorkspace ;
2023-01-20 20:57:35 +01:00
// This is to fix incorrect feedback on the focus history.
PWORKSPACE - > m_pLastFocusedWindow = pWindow ;
2024-12-19 20:11:07 +01:00
if ( m_pLastMonitor - > activeWorkspace )
PWORKSPACE - > rememberPrevWorkspace ( m_pLastMonitor - > activeWorkspace ) ;
2024-04-13 15:39:20 +02:00
if ( PWORKSPACE - > m_bIsSpecialWorkspace )
m_pLastMonitor - > changeWorkspace ( PWORKSPACE , false , true ) ; // if special ws, open on current monitor
2024-11-17 00:04:57 +01:00
else if ( PMONITOR )
2024-04-13 15:39:20 +02:00
PMONITOR - > changeWorkspace ( PWORKSPACE , false , true ) ;
2023-01-20 20:57:35 +01:00
// changeworkspace already calls focusWindow
return ;
}
2022-08-06 21:05:19 +02:00
2024-04-27 13:43:12 +02:00
const auto PLASTWINDOW = m_pLastWindow . lock ( ) ;
2022-12-16 18:17:31 +01:00
m_pLastWindow = pWindow ;
2022-04-23 14:16:02 +02:00
2024-01-09 13:17:55 +01:00
/* If special fallthrough is enabled, this behavior will be disabled, as I have no better idea of nicely tracking which
window focuses are " via keybinds " and which ones aren ' t . */
2024-11-17 00:04:57 +01:00
if ( PMONITOR & & PMONITOR - > activeSpecialWorkspace & & PMONITOR - > activeSpecialWorkspace ! = pWindow - > m_pWorkspace & & ! pWindow - > m_bPinned & & ! * PSPECIALFALLTHROUGH )
2024-01-07 12:06:33 +01:00
PMONITOR - > setSpecialWorkspace ( nullptr ) ;
2022-04-23 14:16:02 +02:00
// we need to make the PLASTWINDOW not equal to m_pLastWindow so that RENDERDATA is correct for an unfocused window
2024-04-27 13:43:12 +02:00
if ( PLASTWINDOW & & PLASTWINDOW - > m_bIsMapped ) {
2023-12-08 17:02:16 +01:00
PLASTWINDOW - > updateDynamicRules ( ) ;
2022-07-12 13:40:55 +02:00
updateWindowAnimatedDecorationValues ( PLASTWINDOW ) ;
2022-04-23 14:16:02 +02:00
2024-08-30 14:12:23 +02:00
if ( ! pWindow - > m_bIsX11 | | ! pWindow - > isX11OverrideRedirect ( ) )
2022-10-28 21:12:17 +02:00
g_pXWaylandManager - > activateWindow ( PLASTWINDOW , false ) ;
2022-04-05 18:29:58 +02:00
}
2022-04-23 14:16:02 +02:00
m_pLastWindow = PLASTWINDOW ;
2024-06-08 10:07:59 +02:00
const auto PWINDOWSURFACE = pSurface ? pSurface : pWindow - > m_pWLSurface - > resource ( ) ;
2022-03-18 22:53:27 +01:00
2022-04-02 18:57:09 +02:00
focusSurface ( PWINDOWSURFACE , pWindow ) ;
2022-04-23 14:16:02 +02:00
g_pXWaylandManager - > activateWindow ( pWindow , true ) ; // sets the m_pLastWindow
2022-04-02 18:57:09 +02:00
2023-12-08 17:02:16 +01:00
pWindow - > updateDynamicRules ( ) ;
2022-07-12 13:40:55 +02:00
updateWindowAnimatedDecorationValues ( pWindow ) ;
2022-05-24 22:21:31 +02:00
2023-09-01 17:10:03 +02:00
if ( pWindow - > m_bIsUrgent )
2023-01-14 20:31:11 +01:00
pWindow - > m_bIsUrgent = false ;
2022-05-24 22:21:31 +02:00
// Send an event
2024-05-11 00:28:33 +02:00
g_pEventManager - > postEvent ( SHyprIPCEvent { " activewindow " , pWindow - > m_szClass + " , " + pWindow - > m_szTitle } ) ;
2024-04-27 13:43:12 +02:00
g_pEventManager - > postEvent ( SHyprIPCEvent { " activewindowv2 " , std : : format ( " {:x} " , ( uintptr_t ) pWindow . get ( ) ) } ) ;
2022-05-29 11:24:42 +02:00
2023-02-19 21:54:53 +01:00
EMIT_HOOK_EVENT ( " activeWindow " , pWindow ) ;
2022-10-24 13:25:36 +02:00
g_pLayoutManager - > getCurrentLayout ( ) - > onWindowFocusChange ( pWindow ) ;
2022-10-31 13:26:07 +01:00
g_pInputManager - > recheckIdleInhibitorStatus ( ) ;
2023-01-20 19:15:15 +01:00
// move to front of the window history
2024-04-27 13:43:12 +02:00
const auto HISTORYPIVOT = std : : find_if ( m_vWindowFocusHistory . begin ( ) , m_vWindowFocusHistory . end ( ) , [ & ] ( const auto & other ) { return other . lock ( ) = = pWindow ; } ) ;
2023-01-20 19:15:15 +01:00
if ( HISTORYPIVOT = = m_vWindowFocusHistory . end ( ) ) {
2023-09-20 17:25:03 +02:00
Debug : : log ( ERR , " BUG THIS: {} has no pivot in history " , pWindow ) ;
2023-01-20 19:15:15 +01:00
} else {
std : : rotate ( m_vWindowFocusHistory . begin ( ) , HISTORYPIVOT , HISTORYPIVOT + 1 ) ;
}
2023-10-26 23:17:49 +02:00
2024-03-03 19:39:20 +01:00
if ( * PFOLLOWMOUSE = = 0 )
2023-10-26 23:17:49 +02:00
g_pInputManager - > sendMotionEventsToFocused ( ) ;
2022-03-20 14:36:55 +01:00
}
2024-06-08 10:07:59 +02:00
void CCompositor : : focusSurface ( SP < CWLSurfaceResource > pSurface , PHLWINDOW pWindowOwner ) {
2022-04-18 17:16:01 +02:00
2024-06-08 10:07:59 +02:00
if ( g_pSeatManager - > state . keyboardFocus = = pSurface | | ( pWindowOwner & & g_pSeatManager - > state . keyboardFocus = = pWindowOwner - > m_pWLSurface - > resource ( ) ) )
2022-12-16 18:17:31 +01:00
return ; // Don't focus when already focused on this.
2022-03-20 14:36:55 +01:00
2024-09-04 17:59:00 +02:00
if ( g_pSessionLockManager - > isSessionLocked ( ) & & pSurface & & ! g_pSessionLockManager - > isSurfaceSessionLock ( pSurface ) )
2024-01-07 14:04:25 +01:00
return ;
2023-02-12 20:20:13 +01:00
2024-05-11 02:02:57 +02:00
if ( g_pSeatManager - > seatGrab & & ! g_pSeatManager - > seatGrab - > accepts ( pSurface ) ) {
2024-07-22 18:06:11 +02:00
Debug : : log ( LOG , " surface {:x} won't receive kb focus becuase grab rejected it " , ( uintptr_t ) pSurface . get ( ) ) ;
2024-05-11 02:02:57 +02:00
return ;
}
2024-06-08 10:07:59 +02:00
const auto PLASTSURF = m_pLastFocus . lock ( ) ;
2024-03-02 22:04:55 +01:00
2022-07-07 21:47:59 +02:00
// Unfocus last surface if should
2022-10-28 20:18:10 +02:00
if ( m_pLastFocus & & ! pWindowOwner )
2024-06-08 10:07:59 +02:00
g_pXWaylandManager - > activateSurface ( m_pLastFocus . lock ( ) , false ) ;
2022-07-07 21:47:59 +02:00
2022-07-11 12:29:50 +02:00
if ( ! pSurface ) {
2024-05-10 19:27:57 +02:00
g_pSeatManager - > setKeyboardFocus ( nullptr ) ;
2022-07-20 18:39:08 +02:00
g_pEventManager - > postEvent ( SHyprIPCEvent { " activewindow " , " , " } ) ; // unfocused
2024-05-01 14:57:27 +02:00
g_pEventManager - > postEvent ( SHyprIPCEvent { " activewindowv2 " , " " } ) ;
2024-06-08 10:07:59 +02:00
EMIT_HOOK_EVENT ( " keyboardFocus " , ( SP < CWLSurfaceResource > ) nullptr ) ;
m_pLastFocus . reset ( ) ;
2022-03-27 19:16:33 +02:00
return ;
2022-07-11 12:29:50 +02:00
}
2022-09-25 20:07:48 +02:00
2024-05-10 19:27:57 +02:00
if ( g_pSeatManager - > keyboard )
g_pSeatManager - > setKeyboardFocus ( pSurface ) ;
2022-04-02 19:09:27 +02:00
2022-04-10 11:17:06 +02:00
if ( pWindowOwner )
2024-07-22 18:06:11 +02:00
Debug : : log ( LOG , " Set keyboard focus to surface {:x}, with {} " , ( uintptr_t ) pSurface . get ( ) , pWindowOwner ) ;
2022-03-31 19:41:55 +02:00
else
2024-07-22 18:06:11 +02:00
Debug : : log ( LOG , " Set keyboard focus to surface {:x} " , ( uintptr_t ) pSurface . get ( ) ) ;
2022-07-08 13:19:57 +02:00
2022-11-02 19:54:41 +01:00
g_pXWaylandManager - > activateSurface ( pSurface , true ) ;
2022-07-08 13:19:57 +02:00
m_pLastFocus = pSurface ;
2023-02-19 21:54:53 +01:00
EMIT_HOOK_EVENT ( " keyboardFocus " , pSurface ) ;
2024-03-02 22:04:55 +01:00
2024-06-08 10:07:59 +02:00
const auto SURF = CWLSurface : : fromResource ( pSurface ) ;
const auto OLDSURF = CWLSurface : : fromResource ( PLASTSURF ) ;
2024-03-02 22:04:55 +01:00
if ( OLDSURF & & OLDSURF - > constraint ( ) )
OLDSURF - > constraint ( ) - > deactivate ( ) ;
if ( SURF & & SURF - > constraint ( ) )
SURF - > constraint ( ) - > activate ( ) ;
2022-03-18 23:16:15 +01:00
}
2024-10-20 00:03:29 +02:00
SP < CWLSurfaceResource > CCompositor : : vectorToLayerPopupSurface ( const Vector2D & pos , PHLMONITOR monitor , Vector2D * sCoords , PHLLS * ppLayerSurfaceFound ) {
2024-08-26 17:25:39 +02:00
for ( auto const & lsl : monitor - > m_aLayerSurfaceLayers | std : : views : : reverse ) {
for ( auto const & ls : lsl | std : : views : : reverse ) {
2024-05-09 22:47:21 +02:00
if ( ls - > fadingOut | | ! ls - > layerSurface | | ( ls - > layerSurface & & ! ls - > layerSurface - > mapped ) | | ls - > alpha . value ( ) = = 0.f )
2024-03-25 17:20:30 +01:00
continue ;
2024-05-09 22:47:21 +02:00
auto SURFACEAT = ls - > popupHead - > at ( pos , true ) ;
2024-03-25 17:20:30 +01:00
if ( SURFACEAT ) {
2024-05-09 22:47:21 +02:00
* ppLayerSurfaceFound = ls . lock ( ) ;
* sCoords = pos - SURFACEAT - > coordsGlobal ( ) ;
2024-06-08 10:07:59 +02:00
return SURFACEAT - > m_pWLSurface - > resource ( ) ;
2024-03-25 17:20:30 +01:00
}
}
}
return nullptr ;
}
2024-06-08 10:07:59 +02:00
SP < CWLSurfaceResource > CCompositor : : vectorToLayerSurface ( const Vector2D & pos , std : : vector < PHLLSREF > * layerSurfaces , Vector2D * sCoords , PHLLS * ppLayerSurfaceFound ) {
2024-08-26 17:25:39 +02:00
for ( auto const & ls : * layerSurfaces | std : : views : : reverse ) {
2024-03-02 01:35:17 +01:00
if ( ls - > fadingOut | | ! ls - > layerSurface | | ( ls - > layerSurface & & ! ls - > layerSurface - > surface - > mapped ) | | ls - > alpha . value ( ) = = 0.f )
2022-03-20 14:36:55 +01:00
continue ;
2024-06-11 17:01:25 +02:00
auto [ surf , local ] = ls - > layerSurface - > surface - > at ( pos - ls - > geometry . pos ( ) , true ) ;
2022-11-18 21:35:15 +01:00
2024-06-08 10:07:59 +02:00
if ( surf ) {
if ( surf - > current . input . empty ( ) )
2022-12-22 16:03:32 +01:00
continue ;
2024-05-09 22:47:21 +02:00
* ppLayerSurfaceFound = ls . lock ( ) ;
2024-06-08 10:07:59 +02:00
* sCoords = local ;
return surf ;
2022-07-01 20:14:33 +02:00
}
2022-03-20 14:36:55 +01:00
}
return nullptr ;
}
2024-06-08 10:07:59 +02:00
PHLWINDOW CCompositor : : getWindowFromSurface ( SP < CWLSurfaceResource > pSurface ) {
2024-07-24 11:07:22 +02:00
if ( ! pSurface | | ! pSurface - > hlSurface )
return nullptr ;
2022-03-20 14:36:55 +01:00
2024-07-24 11:07:22 +02:00
return pSurface - > hlSurface - > getWindow ( ) ;
2022-03-20 15:55:47 +01:00
}
2024-04-27 13:43:12 +02:00
PHLWINDOW CCompositor : : getWindowFromHandle ( uint32_t handle ) {
2024-08-26 20:24:30 +02:00
for ( auto const & w : m_vWindows ) {
2022-12-05 18:57:59 +01:00
if ( ( uint32_t ) ( ( ( uint64_t ) w . get ( ) ) & 0xFFFFFFFF ) = = handle ) {
2024-04-27 13:43:12 +02:00
return w ;
2022-12-05 18:57:59 +01:00
}
2022-12-05 18:05:15 +01:00
}
return nullptr ;
}
2024-08-08 21:01:50 +02:00
PHLWORKSPACE CCompositor : : getWorkspaceByID ( const WORKSPACEID & id ) {
2024-08-26 20:24:30 +02:00
for ( auto const & w : m_vWorkspaces ) {
2024-04-02 21:32:39 +02:00
if ( w - > m_iID = = id & & ! w - > inert ( ) )
return w ;
2022-03-20 15:55:47 +01:00
}
return nullptr ;
}
void CCompositor : : sanityCheckWorkspaces ( ) {
2022-12-29 17:06:30 +01:00
auto it = m_vWorkspaces . begin ( ) ;
while ( it ! = m_vWorkspaces . end ( ) ) {
2024-04-06 16:53:32 +02:00
const auto & WORKSPACE = * it ;
2024-04-05 20:32:05 +02:00
// If ref == 1, only the compositor holds a ref, which means it's inactive and has no mapped windows.
2024-05-05 18:16:00 +02:00
if ( ! WORKSPACE - > m_bPersistent & & WORKSPACE . strongRef ( ) = = 1 ) {
2024-04-05 20:32:05 +02:00
it = m_vWorkspaces . erase ( it ) ;
2023-04-13 22:09:50 +02:00
continue ;
2023-10-09 21:28:22 +02:00
}
2023-04-13 22:09:50 +02:00
2022-12-29 17:06:30 +01:00
+ + it ;
2022-03-20 15:55:47 +01:00
}
}
2024-04-27 13:43:12 +02:00
PHLWINDOW CCompositor : : getUrgentWindow ( ) {
2024-08-26 20:24:30 +02:00
for ( auto const & w : m_vWindows ) {
2023-01-21 11:18:55 +01:00
if ( w - > m_bIsMapped & & w - > m_bIsUrgent )
2024-04-27 13:43:12 +02:00
return w ;
2023-01-21 11:18:55 +01:00
}
return nullptr ;
}
2024-04-27 13:43:12 +02:00
bool CCompositor : : isWindowActive ( PHLWINDOW pWindow ) {
if ( m_pLastWindow . expired ( ) & & ! m_pLastFocus )
2022-05-29 00:07:31 +02:00
return false ;
2024-04-02 02:15:58 +02:00
if ( ! pWindow - > m_bIsMapped )
2022-04-02 18:57:09 +02:00
return false ;
2024-06-08 10:07:59 +02:00
const auto PSURFACE = pWindow - > m_pWLSurface - > resource ( ) ;
2022-04-02 13:02:16 +02:00
2024-04-27 13:43:12 +02:00
return PSURFACE = = m_pLastFocus | | pWindow = = m_pLastWindow . lock ( ) ;
2022-04-04 16:25:30 +02:00
}
2024-04-27 13:43:12 +02:00
void CCompositor : : changeWindowZOrder ( PHLWINDOW pWindow , bool top ) {
if ( ! validMapped ( pWindow ) )
2022-04-04 16:25:30 +02:00
return ;
2024-11-18 15:44:15 +01:00
if ( top )
pWindow - > m_bCreatedOverFullscreen = true ;
2024-10-12 02:29:51 +02:00
if ( pWindow = = ( top ? m_vWindows . back ( ) : m_vWindows . front ( ) ) )
return ;
2024-04-27 13:43:12 +02:00
auto moveToZ = [ & ] ( PHLWINDOW pw , bool top ) - > void {
2023-09-21 23:18:26 +02:00
if ( top ) {
for ( auto it = m_vWindows . begin ( ) ; it ! = m_vWindows . end ( ) ; + + it ) {
2024-04-27 13:43:12 +02:00
if ( * it = = pw ) {
2023-09-21 23:18:26 +02:00
std : : rotate ( it , it + 1 , m_vWindows . end ( ) ) ;
break ;
}
}
} else {
for ( auto it = m_vWindows . rbegin ( ) ; it ! = m_vWindows . rend ( ) ; + + it ) {
2024-04-27 13:43:12 +02:00
if ( * it = = pw ) {
2023-09-21 23:18:26 +02:00
std : : rotate ( it , it + 1 , m_vWindows . rend ( ) ) ;
break ;
}
2022-08-28 19:47:06 +02:00
}
}
2023-01-11 17:57:54 +01:00
if ( pw - > m_bIsMapped )
2024-10-27 19:45:38 +01:00
g_pHyprRenderer - > damageMonitor ( pw - > m_pMonitor . lock ( ) ) ;
2022-08-28 19:47:06 +02:00
} ;
2024-05-25 22:43:51 +02:00
if ( ! pWindow - > m_bIsX11 )
2023-09-21 23:18:26 +02:00
moveToZ ( pWindow , top ) ;
2024-05-25 22:43:51 +02:00
else {
2023-09-21 23:18:26 +02:00
// move X11 window stack
2022-08-28 19:47:06 +02:00
2024-12-16 16:58:19 +01:00
std : : vector < PHLWINDOW > toMove ;
2022-08-28 19:47:06 +02:00
2024-12-16 16:58:19 +01:00
auto x11Stack = [ & ] ( PHLWINDOW pw , bool top , auto & & x11Stack ) - > void {
2023-09-21 23:18:26 +02:00
if ( top )
toMove . emplace_back ( pw ) ;
else
2024-12-16 16:58:19 +01:00
toMove . insert ( toMove . begin ( ) , pw ) ;
2022-08-28 19:47:06 +02:00
2024-08-26 20:24:30 +02:00
for ( auto const & w : m_vWindows ) {
2024-12-07 18:51:18 +01:00
if ( w - > m_bIsMapped & & ! w - > isHidden ( ) & & w - > m_bIsX11 & & w - > x11TransientFor ( ) = = pw & & w ! = pw & & std : : find ( toMove . begin ( ) , toMove . end ( ) , w ) = = toMove . end ( ) ) {
2024-04-27 13:43:12 +02:00
x11Stack ( w , top , x11Stack ) ;
2023-09-21 23:18:26 +02:00
}
}
} ;
2022-08-28 19:47:06 +02:00
2023-09-21 23:18:26 +02:00
x11Stack ( pWindow , top , x11Stack ) ;
2022-08-28 19:47:06 +02:00
2024-12-07 18:51:18 +01:00
for ( const auto & it : toMove ) {
2023-09-21 23:18:26 +02:00
moveToZ ( it , top ) ;
}
2022-08-28 19:47:06 +02:00
}
2022-04-05 19:28:10 +02:00
}
2024-08-08 21:01:50 +02:00
void CCompositor : : cleanupFadingOut ( const MONITORID & monid ) {
2024-08-26 20:24:30 +02:00
for ( auto const & ww : m_vWindowsFadingOut ) {
2024-04-27 13:43:12 +02:00
2024-05-09 22:47:21 +02:00
auto w = ww . lock ( ) ;
2022-04-26 17:51:00 +02:00
2024-10-27 19:45:38 +01:00
if ( w - > monitorID ( ) ! = monid & & w - > m_pMonitor )
2022-07-12 23:11:34 +02:00
continue ;
2024-04-06 15:59:30 +02:00
if ( ! w - > m_bFadingOut | | w - > m_fAlpha . value ( ) = = 0.f ) {
2022-06-30 15:44:26 +02:00
2024-04-06 15:59:30 +02:00
w - > m_bFadingOut = false ;
2024-03-14 19:25:28 +01:00
2024-04-06 15:59:30 +02:00
if ( ! w - > m_bReadyToDelete )
continue ;
2023-03-01 14:15:51 +01:00
2024-04-06 15:59:30 +02:00
removeWindowFromVectorSafe ( w ) ;
2022-04-10 11:17:06 +02:00
2024-05-09 22:47:21 +02:00
w . reset ( ) ;
2022-04-10 11:17:06 +02:00
Debug : : log ( LOG , " Cleanup: destroyed a window " ) ;
2022-04-05 19:28:10 +02:00
return ;
}
}
2022-05-14 17:23:46 +02:00
2024-08-07 13:35:02 +02:00
bool layersDirty = false ;
2024-08-26 20:24:30 +02:00
for ( auto const & lsr : m_vSurfacesFadingOut ) {
2022-07-25 22:40:34 +02:00
2024-04-30 03:41:27 +02:00
auto ls = lsr . lock ( ) ;
2022-07-25 22:40:34 +02:00
2024-08-07 13:35:02 +02:00
if ( ! ls ) {
layersDirty = true ;
2024-04-30 03:41:27 +02:00
continue ;
2024-08-07 13:35:02 +02:00
}
2022-07-25 22:40:34 +02:00
2024-10-27 19:45:38 +01:00
if ( ls - > monitorID ( ) ! = monid & & ls - > monitor )
2022-07-12 23:11:34 +02:00
continue ;
2022-08-01 12:16:33 +02:00
// mark blur for recalc
if ( ls - > layer = = ZWLR_LAYER_SHELL_V1_LAYER_BACKGROUND | | ls - > layer = = ZWLR_LAYER_SHELL_V1_LAYER_BOTTOM )
g_pHyprOpenGL - > markBlurDirtyForMonitor ( getMonitorFromID ( monid ) ) ;
2024-02-28 16:00:34 +01:00
if ( ls - > fadingOut & & ls - > readyToDelete & & ls - > isFadedOut ( ) ) {
2024-08-26 17:25:39 +02:00
for ( auto const & m : m_vMonitors ) {
2023-01-22 17:03:25 +01:00
for ( auto & lsl : m - > m_aLayerSurfaceLayers ) {
2024-04-30 03:41:27 +02:00
if ( ! lsl . empty ( ) & & std : : find_if ( lsl . begin ( ) , lsl . end ( ) , [ & ] ( auto & other ) { return other = = ls ; } ) ! = lsl . end ( ) ) {
2024-08-07 13:35:02 +02:00
std : : erase_if ( lsl , [ & ] ( auto & other ) { return other = = ls | | ! other ; } ) ;
2022-07-25 21:08:54 +02:00
}
2022-07-23 15:48:08 +02:00
}
}
2024-04-30 03:41:27 +02:00
std : : erase_if ( m_vSurfacesFadingOut , [ ls ] ( const auto & el ) { return el . lock ( ) = = ls ; } ) ;
2024-05-09 22:47:21 +02:00
std : : erase_if ( m_vLayers , [ ls ] ( const auto & el ) { return el = = ls ; } ) ;
ls . reset ( ) ;
2022-07-25 22:00:46 +02:00
2022-05-14 17:28:55 +02:00
Debug : : log ( LOG , " Cleanup: destroyed a layersurface " ) ;
2022-12-16 18:17:31 +01:00
glFlush ( ) ; // to free mem NOW.
2022-05-14 17:23:46 +02:00
return ;
}
}
2024-08-07 13:35:02 +02:00
if ( layersDirty )
std : : erase_if ( m_vSurfacesFadingOut , [ ] ( const auto & el ) { return el . expired ( ) ; } ) ;
2022-04-09 13:26:55 +02:00
}
2024-04-30 03:41:27 +02:00
void CCompositor : : addToFadingOutSafe ( PHLLS pLS ) {
const auto FOUND = std : : find_if ( m_vSurfacesFadingOut . begin ( ) , m_vSurfacesFadingOut . end ( ) , [ & ] ( auto & other ) { return other . lock ( ) = = pLS ; } ) ;
2022-07-25 21:08:54 +02:00
if ( FOUND ! = m_vSurfacesFadingOut . end ( ) )
2022-12-16 18:17:31 +01:00
return ; // if it's already added, don't add it.
2022-07-25 21:08:54 +02:00
m_vSurfacesFadingOut . emplace_back ( pLS ) ;
}
2024-06-12 22:57:06 +02:00
void CCompositor : : removeFromFadingOutSafe ( PHLLS ls ) {
std : : erase ( m_vSurfacesFadingOut , ls ) ;
}
2024-04-27 13:43:12 +02:00
void CCompositor : : addToFadingOutSafe ( PHLWINDOW pWindow ) {
const auto FOUND = std : : find_if ( m_vWindowsFadingOut . begin ( ) , m_vWindowsFadingOut . end ( ) , [ & ] ( PHLWINDOWREF & other ) { return other . lock ( ) = = pWindow ; } ) ;
2022-07-25 21:08:54 +02:00
if ( FOUND ! = m_vWindowsFadingOut . end ( ) )
2022-12-16 18:17:31 +01:00
return ; // if it's already added, don't add it.
2022-07-25 21:08:54 +02:00
m_vWindowsFadingOut . emplace_back ( pWindow ) ;
}
2024-04-27 13:43:12 +02:00
PHLWINDOW CCompositor : : getWindowInDirection ( PHLWINDOW pWindow , char dir ) {
2022-06-23 20:39:48 +02:00
2023-11-05 17:18:41 +01:00
if ( ! isDirection ( dir ) )
return nullptr ;
2023-01-26 15:36:22 +01:00
// 0 -> history, 1 -> shared length
2024-05-22 21:51:46 +02:00
static auto PMETHOD = CConfigValue < Hyprlang : : INT > ( " binds:focus_preferred_method " ) ;
static auto PMONITORFALLBACK = CConfigValue < Hyprlang : : INT > ( " binds:window_direction_monitor_fallback " ) ;
2022-06-23 20:39:48 +02:00
2024-10-27 19:45:38 +01:00
const auto PMONITOR = pWindow - > m_pMonitor . lock ( ) ;
2022-04-09 13:26:55 +02:00
2023-11-04 22:03:08 +01:00
if ( ! PMONITOR )
return nullptr ; // ??
2023-01-26 15:36:22 +01:00
2024-07-31 19:55:52 +02:00
const auto WINDOWIDEALBB = pWindow - > isFullscreen ( ) ? CBox { PMONITOR - > vecPosition , PMONITOR - > vecSize } : pWindow - > getWindowIdealBoundingBoxIgnoreReserved ( ) ;
2023-11-04 22:03:08 +01:00
const auto POSA = Vector2D ( WINDOWIDEALBB . x , WINDOWIDEALBB . y ) ;
const auto SIZEA = Vector2D ( WINDOWIDEALBB . width , WINDOWIDEALBB . height ) ;
2024-04-02 21:32:39 +02:00
const auto PWORKSPACE = pWindow - > m_pWorkspace ;
2023-11-04 22:03:08 +01:00
auto leaderValue = - 1 ;
2024-04-27 13:43:12 +02:00
PHLWINDOW leaderWindow = nullptr ;
2022-04-09 13:26:55 +02:00
2023-11-05 17:18:41 +01:00
if ( ! pWindow - > m_bIsFloating ) {
2022-04-09 13:26:55 +02:00
2023-11-05 17:18:41 +01:00
// for tiled windows, we calc edges
2024-08-26 20:24:30 +02:00
for ( auto const & w : m_vWindows ) {
2024-11-22 17:01:02 +01:00
if ( w = = pWindow | | ! w - > m_pWorkspace | | ! w - > m_bIsMapped | | w - > isHidden ( ) | | ( ! w - > isFullscreen ( ) & & w - > m_bIsFloating ) | | ! w - > m_pWorkspace - > isVisible ( ) )
2023-11-05 17:18:41 +01:00
continue ;
2023-10-20 11:53:37 +02:00
2024-10-27 19:45:38 +01:00
if ( pWindow - > m_pMonitor = = w - > m_pMonitor & & pWindow - > m_pWorkspace ! = w - > m_pWorkspace )
2023-11-05 17:18:41 +01:00
continue ;
2022-11-24 00:40:05 +01:00
2024-07-31 19:55:52 +02:00
if ( PWORKSPACE - > m_bHasFullscreenWindow & & ! w - > isFullscreen ( ) & & ! w - > m_bCreatedOverFullscreen )
2023-11-05 17:18:41 +01:00
continue ;
2022-06-23 20:39:48 +02:00
2024-10-27 19:45:38 +01:00
if ( ! * PMONITORFALLBACK & & pWindow - > m_pMonitor ! = w - > m_pMonitor )
2024-05-22 21:51:46 +02:00
continue ;
2023-11-05 17:18:41 +01:00
const auto BWINDOWIDEALBB = w - > getWindowIdealBoundingBoxIgnoreReserved ( ) ;
2022-06-23 20:39:48 +02:00
2023-11-05 17:18:41 +01:00
const auto POSB = Vector2D ( BWINDOWIDEALBB . x , BWINDOWIDEALBB . y ) ;
const auto SIZEB = Vector2D ( BWINDOWIDEALBB . width , BWINDOWIDEALBB . height ) ;
2023-01-26 15:36:22 +01:00
2023-11-05 17:18:41 +01:00
double intersectLength = - 1 ;
2023-01-26 15:36:22 +01:00
2023-11-05 17:18:41 +01:00
switch ( dir ) {
case ' l ' :
if ( STICKS ( POSA . x , POSB . x + SIZEB . x ) ) {
intersectLength = std : : max ( 0.0 , std : : min ( POSA . y + SIZEA . y , POSB . y + SIZEB . y ) - std : : max ( POSA . y , POSB . y ) ) ;
}
break ;
case ' r ' :
if ( STICKS ( POSA . x + SIZEA . x , POSB . x ) ) {
intersectLength = std : : max ( 0.0 , std : : min ( POSA . y + SIZEA . y , POSB . y + SIZEB . y ) - std : : max ( POSA . y , POSB . y ) ) ;
}
break ;
case ' t ' :
case ' u ' :
if ( STICKS ( POSA . y , POSB . y + SIZEB . y ) ) {
intersectLength = std : : max ( 0.0 , std : : min ( POSA . x + SIZEA . x , POSB . x + SIZEB . x ) - std : : max ( POSA . x , POSB . x ) ) ;
}
break ;
case ' b ' :
case ' d ' :
if ( STICKS ( POSA . y + SIZEA . y , POSB . y ) ) {
intersectLength = std : : max ( 0.0 , std : : min ( POSA . x + SIZEA . x , POSB . x + SIZEB . x ) - std : : max ( POSA . x , POSB . x ) ) ;
}
break ;
}
2023-01-26 15:36:22 +01:00
2024-03-03 19:39:20 +01:00
if ( * PMETHOD = = 0 /* history */ ) {
2023-11-05 17:18:41 +01:00
if ( intersectLength > 0 ) {
// get idx
int windowIDX = - 1 ;
for ( size_t i = 0 ; i < g_pCompositor - > m_vWindowFocusHistory . size ( ) ; + + i ) {
2024-04-27 13:43:12 +02:00
if ( g_pCompositor - > m_vWindowFocusHistory [ i ] . lock ( ) = = w ) {
2023-11-05 17:18:41 +01:00
windowIDX = i ;
break ;
}
2023-01-26 15:36:22 +01:00
}
2023-11-05 17:18:41 +01:00
windowIDX = g_pCompositor - > m_vWindowFocusHistory . size ( ) - windowIDX ;
2023-01-26 15:36:22 +01:00
2023-11-05 17:18:41 +01:00
if ( windowIDX > leaderValue ) {
leaderValue = windowIDX ;
2024-04-27 13:43:12 +02:00
leaderWindow = w ;
2023-11-05 17:18:41 +01:00
}
}
} else /* length */ {
if ( intersectLength > leaderValue ) {
leaderValue = intersectLength ;
2024-04-27 13:43:12 +02:00
leaderWindow = w ;
2023-01-26 15:36:22 +01:00
}
}
2023-11-05 17:18:41 +01:00
}
} else {
// for floating windows, we calculate best distance and angle.
// if there is a window with angle better than THRESHOLD, only distance counts
if ( dir = = ' u ' )
dir = ' t ' ;
if ( dir = = ' d ' )
dir = ' b ' ;
static const std : : unordered_map < char , Vector2D > VECTORS = { { ' r ' , { 1 , 0 } } , { ' t ' , { 0 , - 1 } } , { ' b ' , { 0 , 1 } } , { ' l ' , { - 1 , 0 } } } ;
//
2024-12-07 18:51:18 +01:00
auto vectorAngles = [ ] ( const Vector2D & a , const Vector2D & b ) - > double {
2023-11-05 17:18:41 +01:00
double dot = a . x * b . x + a . y * b . y ;
double ang = std : : acos ( dot / ( a . size ( ) * b . size ( ) ) ) ;
return ang ;
} ;
float bestAngleAbs = 2.0 * M_PI ;
constexpr float THRESHOLD = 0.3 * M_PI ;
2024-08-26 20:24:30 +02:00
for ( auto const & w : m_vWindows ) {
2024-11-22 17:01:02 +01:00
if ( w = = pWindow | | ! w - > m_bIsMapped | | ! w - > m_pWorkspace | | w - > isHidden ( ) | | ( ! w - > isFullscreen ( ) & & ! w - > m_bIsFloating ) | | ! w - > m_pWorkspace - > isVisible ( ) )
2023-11-05 17:18:41 +01:00
continue ;
2024-10-27 19:45:38 +01:00
if ( pWindow - > m_pMonitor = = w - > m_pMonitor & & pWindow - > m_pWorkspace ! = w - > m_pWorkspace )
2023-11-05 17:18:41 +01:00
continue ;
2024-07-31 19:55:52 +02:00
if ( PWORKSPACE - > m_bHasFullscreenWindow & & ! w - > isFullscreen ( ) & & ! w - > m_bCreatedOverFullscreen )
2023-11-05 17:18:41 +01:00
continue ;
2024-10-27 19:45:38 +01:00
if ( ! * PMONITORFALLBACK & & pWindow - > m_pMonitor ! = w - > m_pMonitor )
2024-05-22 21:51:46 +02:00
continue ;
2023-11-05 17:18:41 +01:00
const auto DIST = w - > middle ( ) . distance ( pWindow - > middle ( ) ) ;
const auto ANGLE = vectorAngles ( Vector2D { w - > middle ( ) - pWindow - > middle ( ) } , VECTORS . at ( dir ) ) ;
if ( ANGLE > M_PI_2 )
continue ; // if the angle is over 90 degrees, ignore. Wrong direction entirely.
if ( ( bestAngleAbs < THRESHOLD & & DIST < leaderValue & & ANGLE < THRESHOLD ) | | ( ANGLE < bestAngleAbs & & bestAngleAbs > THRESHOLD ) | | leaderValue = = - 1 ) {
leaderValue = DIST ;
bestAngleAbs = ANGLE ;
2024-04-27 13:43:12 +02:00
leaderWindow = w ;
2023-01-26 15:36:22 +01:00
}
}
2023-11-05 17:18:41 +01:00
if ( ! leaderWindow & & PWORKSPACE - > m_bHasFullscreenWindow )
2024-11-22 17:01:02 +01:00
leaderWindow = PWORKSPACE - > getFullscreenWindow ( ) ;
2022-04-09 13:26:55 +02:00
}
2023-01-26 15:36:22 +01:00
if ( leaderValue ! = - 1 )
return leaderWindow ;
2022-04-09 13:26:55 +02:00
return nullptr ;
2022-04-11 19:51:37 +02:00
}
2024-04-27 13:43:12 +02:00
PHLWINDOW CCompositor : : getNextWindowOnWorkspace ( PHLWINDOW pWindow , bool focusableOnly , std : : optional < bool > floating ) {
2022-04-13 20:45:06 +02:00
bool gotToWindow = false ;
2024-08-26 20:24:30 +02:00
for ( auto const & w : m_vWindows ) {
2024-04-27 13:43:12 +02:00
if ( w ! = pWindow & & ! gotToWindow )
2022-04-13 20:45:06 +02:00
continue ;
2024-04-27 13:43:12 +02:00
if ( w = = pWindow ) {
2022-04-13 20:45:06 +02:00
gotToWindow = true ;
continue ;
}
2023-12-24 15:08:48 +01:00
if ( floating . has_value ( ) & & w - > m_bIsFloating ! = floating . value ( ) )
continue ;
2024-07-11 16:10:42 +02:00
if ( w - > m_pWorkspace = = pWindow - > m_pWorkspace & & w - > m_bIsMapped & & ! w - > isHidden ( ) & & ( ! focusableOnly | | ! w - > m_sWindowData . noFocus . valueOrDefault ( ) ) )
2024-04-27 13:43:12 +02:00
return w ;
2022-04-13 20:45:06 +02:00
}
2024-08-26 20:24:30 +02:00
for ( auto const & w : m_vWindows ) {
2023-12-24 15:08:48 +01:00
if ( floating . has_value ( ) & & w - > m_bIsFloating ! = floating . value ( ) )
continue ;
2024-07-11 16:10:42 +02:00
if ( w ! = pWindow & & w - > m_pWorkspace = = pWindow - > m_pWorkspace & & w - > m_bIsMapped & & ! w - > isHidden ( ) & & ( ! focusableOnly | | ! w - > m_sWindowData . noFocus . valueOrDefault ( ) ) )
2024-04-27 13:43:12 +02:00
return w ;
2022-04-13 20:45:06 +02:00
}
2022-04-21 16:38:48 +02:00
return nullptr ;
}
2024-04-27 13:43:12 +02:00
PHLWINDOW CCompositor : : getPrevWindowOnWorkspace ( PHLWINDOW pWindow , bool focusableOnly , std : : optional < bool > floating ) {
2022-07-09 18:39:41 +02:00
bool gotToWindow = false ;
2024-08-26 20:24:30 +02:00
for ( auto const & w : m_vWindows | std : : views : : reverse ) {
2024-04-27 13:43:12 +02:00
if ( w ! = pWindow & & ! gotToWindow )
2022-07-09 18:39:41 +02:00
continue ;
2024-04-27 13:43:12 +02:00
if ( w = = pWindow ) {
2022-07-09 18:39:41 +02:00
gotToWindow = true ;
continue ;
}
2023-12-24 15:08:48 +01:00
if ( floating . has_value ( ) & & w - > m_bIsFloating ! = floating . value ( ) )
continue ;
2024-07-11 16:10:42 +02:00
if ( w - > m_pWorkspace = = pWindow - > m_pWorkspace & & w - > m_bIsMapped & & ! w - > isHidden ( ) & & ( ! focusableOnly | | ! w - > m_sWindowData . noFocus . valueOrDefault ( ) ) )
2024-04-27 13:43:12 +02:00
return w ;
2022-07-09 18:39:41 +02:00
}
2024-08-26 20:24:30 +02:00
for ( auto const & w : m_vWindows | std : : views : : reverse ) {
2023-12-24 15:08:48 +01:00
if ( floating . has_value ( ) & & w - > m_bIsFloating ! = floating . value ( ) )
continue ;
2024-07-11 16:10:42 +02:00
if ( w ! = pWindow & & w - > m_pWorkspace = = pWindow - > m_pWorkspace & & w - > m_bIsMapped & & ! w - > isHidden ( ) & & ( ! focusableOnly | | ! w - > m_sWindowData . noFocus . valueOrDefault ( ) ) )
2024-04-27 13:43:12 +02:00
return w ;
2022-07-09 18:39:41 +02:00
}
return nullptr ;
}
2024-08-08 21:01:50 +02:00
WORKSPACEID CCompositor : : getNextAvailableNamedWorkspace ( ) {
WORKSPACEID lowest = - 1337 + 1 ;
2024-08-26 17:25:39 +02:00
for ( auto const & w : m_vWorkspaces ) {
2022-06-30 15:44:26 +02:00
if ( w - > m_iID < - 1 & & w - > m_iID < lowest )
lowest = w - > m_iID ;
2022-04-21 16:38:48 +02:00
}
2022-04-21 21:35:08 +02:00
return lowest - 1 ;
2022-04-21 16:38:48 +02:00
}
2024-04-02 21:32:39 +02:00
PHLWORKSPACE CCompositor : : getWorkspaceByName ( const std : : string & name ) {
2024-08-26 20:24:30 +02:00
for ( auto const & w : m_vWorkspaces ) {
2024-04-02 21:32:39 +02:00
if ( w - > m_szName = = name & & ! w - > inert ( ) )
return w ;
2022-04-21 16:38:48 +02:00
}
return nullptr ;
}
2024-04-02 21:32:39 +02:00
PHLWORKSPACE CCompositor : : getWorkspaceByString ( const std : : string & str ) {
2023-10-15 20:07:23 +02:00
if ( str . starts_with ( " name: " ) ) {
2022-04-21 16:38:48 +02:00
return getWorkspaceByName ( str . substr ( str . find_first_of ( ' : ' ) + 1 ) ) ;
}
try {
2024-06-22 23:52:42 +02:00
return getWorkspaceByID ( getWorkspaceIDNameFromString ( str ) . id ) ;
2022-12-16 18:17:31 +01:00
} catch ( std : : exception & e ) { Debug : : log ( ERR , " Error in getWorkspaceByString, invalid id " ) ; }
2022-04-21 16:38:48 +02:00
2022-04-13 20:45:06 +02:00
return nullptr ;
2022-04-23 14:16:02 +02:00
}
2022-04-24 17:53:50 +02:00
bool CCompositor : : isPointOnAnyMonitor ( const Vector2D & point ) {
2024-08-26 17:25:39 +02:00
for ( auto const & m : m_vMonitors ) {
2022-06-30 15:44:26 +02:00
if ( VECINRECT ( point , m - > vecPosition . x , m - > vecPosition . y , m - > vecSize . x + m - > vecPosition . x , m - > vecSize . y + m - > vecPosition . y ) )
2022-04-24 17:53:50 +02:00
return true ;
}
return false ;
2022-04-25 13:40:46 +02:00
}
2024-10-20 00:03:29 +02:00
bool CCompositor : : isPointOnReservedArea ( const Vector2D & point , const PHLMONITOR pMonitor ) {
2023-12-26 17:24:31 +01:00
const auto PMONITOR = pMonitor ? pMonitor : getMonitorFromVector ( point ) ;
const auto XY1 = PMONITOR - > vecPosition + PMONITOR - > vecReservedTopLeft ;
const auto XY2 = PMONITOR - > vecPosition + PMONITOR - > vecSize - PMONITOR - > vecReservedBottomRight ;
return ! VECINRECT ( point , XY1 . x , XY1 . y , XY2 . x , XY2 . y ) ;
}
2024-10-20 00:03:29 +02:00
PHLMONITOR CCompositor : : getMonitorInDirection ( const char & dir ) {
return getMonitorInDirection ( m_pLastMonitor . lock ( ) , dir ) ;
2024-02-25 15:03:00 +01:00
}
2024-10-20 00:03:29 +02:00
PHLMONITOR CCompositor : : getMonitorInDirection ( PHLMONITOR pSourceMonitor , const char & dir ) {
2024-02-26 01:05:20 +01:00
if ( ! pSourceMonitor )
return nullptr ;
2024-02-25 15:03:00 +01:00
const auto POSA = pSourceMonitor - > vecPosition ;
const auto SIZEA = pSourceMonitor - > vecSize ;
2022-05-05 12:50:25 +02:00
2022-12-16 18:17:31 +01:00
auto longestIntersect = - 1 ;
2024-10-20 00:03:29 +02:00
PHLMONITOR longestIntersectMonitor = nullptr ;
2022-05-05 12:50:25 +02:00
2024-08-26 20:24:30 +02:00
for ( auto const & m : m_vMonitors ) {
2024-05-05 23:18:10 +02:00
if ( m = = m_pLastMonitor )
2022-05-05 12:50:25 +02:00
continue ;
2022-12-16 18:17:31 +01:00
const auto POSB = m - > vecPosition ;
2022-06-30 15:44:26 +02:00
const auto SIZEB = m - > vecSize ;
2022-05-05 12:50:25 +02:00
switch ( dir ) {
case ' l ' :
if ( STICKS ( POSA . x , POSB . x + SIZEB . x ) ) {
2022-09-28 19:43:35 +02:00
const auto INTERSECTLEN = std : : max ( 0.0 , std : : min ( POSA . y + SIZEA . y , POSB . y + SIZEB . y ) - std : : max ( POSA . y , POSB . y ) ) ;
2022-05-05 12:50:25 +02:00
if ( INTERSECTLEN > longestIntersect ) {
2022-12-16 18:17:31 +01:00
longestIntersect = INTERSECTLEN ;
2024-10-20 00:03:29 +02:00
longestIntersectMonitor = m ;
2022-05-05 12:50:25 +02:00
}
}
break ;
case ' r ' :
if ( STICKS ( POSA . x + SIZEA . x , POSB . x ) ) {
2022-09-28 19:43:35 +02:00
const auto INTERSECTLEN = std : : max ( 0.0 , std : : min ( POSA . y + SIZEA . y , POSB . y + SIZEB . y ) - std : : max ( POSA . y , POSB . y ) ) ;
2022-05-05 12:50:25 +02:00
if ( INTERSECTLEN > longestIntersect ) {
2022-12-16 18:17:31 +01:00
longestIntersect = INTERSECTLEN ;
2024-10-20 00:03:29 +02:00
longestIntersectMonitor = m ;
2022-05-05 12:50:25 +02:00
}
}
break ;
case ' t ' :
case ' u ' :
if ( STICKS ( POSA . y , POSB . y + SIZEB . y ) ) {
2022-09-28 19:43:35 +02:00
const auto INTERSECTLEN = std : : max ( 0.0 , std : : min ( POSA . x + SIZEA . x , POSB . x + SIZEB . x ) - std : : max ( POSA . x , POSB . x ) ) ;
2022-05-05 12:50:25 +02:00
if ( INTERSECTLEN > longestIntersect ) {
2022-12-16 18:17:31 +01:00
longestIntersect = INTERSECTLEN ;
2024-10-20 00:03:29 +02:00
longestIntersectMonitor = m ;
2022-05-05 12:50:25 +02:00
}
}
break ;
case ' b ' :
case ' d ' :
if ( STICKS ( POSA . y + SIZEA . y , POSB . y ) ) {
2022-09-28 19:43:35 +02:00
const auto INTERSECTLEN = std : : max ( 0.0 , std : : min ( POSA . x + SIZEA . x , POSB . x + SIZEB . x ) - std : : max ( POSA . x , POSB . x ) ) ;
2022-05-05 12:50:25 +02:00
if ( INTERSECTLEN > longestIntersect ) {
2022-12-16 18:17:31 +01:00
longestIntersect = INTERSECTLEN ;
2024-10-20 00:03:29 +02:00
longestIntersectMonitor = m ;
2022-05-05 12:50:25 +02:00
}
}
break ;
}
}
if ( longestIntersect ! = - 1 )
return longestIntersectMonitor ;
2022-04-25 13:40:46 +02:00
return nullptr ;
2022-05-14 17:23:46 +02:00
}
2022-05-26 21:23:13 +02:00
2022-07-12 13:40:55 +02:00
void CCompositor : : updateAllWindowsAnimatedDecorationValues ( ) {
2024-08-26 17:25:39 +02:00
for ( auto const & w : m_vWindows ) {
2022-06-30 15:44:26 +02:00
if ( ! w - > m_bIsMapped )
2022-05-26 21:23:13 +02:00
continue ;
2024-04-27 13:43:12 +02:00
updateWindowAnimatedDecorationValues ( w ) ;
2022-05-26 21:23:13 +02:00
}
}
2024-04-27 13:43:12 +02:00
void CCompositor : : updateWindowAnimatedDecorationValues ( PHLWINDOW pWindow ) {
2022-05-26 21:23:13 +02:00
// optimization
2024-03-03 19:39:20 +01:00
static auto PACTIVECOL = CConfigValue < Hyprlang : : CUSTOMTYPE > ( " general:col.active_border " ) ;
static auto PINACTIVECOL = CConfigValue < Hyprlang : : CUSTOMTYPE > ( " general:col.inactive_border " ) ;
static auto PNOGROUPACTIVECOL = CConfigValue < Hyprlang : : CUSTOMTYPE > ( " general:col.nogroup_border_active " ) ;
static auto PNOGROUPINACTIVECOL = CConfigValue < Hyprlang : : CUSTOMTYPE > ( " general:col.nogroup_border " ) ;
static auto PGROUPACTIVECOL = CConfigValue < Hyprlang : : CUSTOMTYPE > ( " group:col.border_active " ) ;
static auto PGROUPINACTIVECOL = CConfigValue < Hyprlang : : CUSTOMTYPE > ( " group:col.border_inactive " ) ;
static auto PGROUPACTIVELOCKEDCOL = CConfigValue < Hyprlang : : CUSTOMTYPE > ( " group:col.border_locked_active " ) ;
static auto PGROUPINACTIVELOCKEDCOL = CConfigValue < Hyprlang : : CUSTOMTYPE > ( " group:col.border_locked_inactive " ) ;
static auto PINACTIVEALPHA = CConfigValue < Hyprlang : : FLOAT > ( " decoration:inactive_opacity " ) ;
static auto PACTIVEALPHA = CConfigValue < Hyprlang : : FLOAT > ( " decoration:active_opacity " ) ;
static auto PFULLSCREENALPHA = CConfigValue < Hyprlang : : FLOAT > ( " decoration:fullscreen_opacity " ) ;
2024-11-05 16:44:40 +01:00
static auto PSHADOWCOL = CConfigValue < Hyprlang : : INT > ( " decoration:shadow:color " ) ;
static auto PSHADOWCOLINACTIVE = CConfigValue < Hyprlang : : INT > ( " decoration:shadow:color_inactive " ) ;
2024-03-03 19:39:20 +01:00
static auto PDIMSTRENGTH = CConfigValue < Hyprlang : : FLOAT > ( " decoration:dim_strength " ) ;
static auto PDIMENABLED = CConfigValue < Hyprlang : : INT > ( " decoration:dim_inactive " ) ;
auto * const ACTIVECOL = ( CGradientValueData * ) ( PACTIVECOL . ptr ( ) ) - > getData ( ) ;
auto * const INACTIVECOL = ( CGradientValueData * ) ( PINACTIVECOL . ptr ( ) ) - > getData ( ) ;
auto * const NOGROUPACTIVECOL = ( CGradientValueData * ) ( PNOGROUPACTIVECOL . ptr ( ) ) - > getData ( ) ;
auto * const NOGROUPINACTIVECOL = ( CGradientValueData * ) ( PNOGROUPINACTIVECOL . ptr ( ) ) - > getData ( ) ;
auto * const GROUPACTIVECOL = ( CGradientValueData * ) ( PGROUPACTIVECOL . ptr ( ) ) - > getData ( ) ;
auto * const GROUPINACTIVECOL = ( CGradientValueData * ) ( PGROUPINACTIVECOL . ptr ( ) ) - > getData ( ) ;
auto * const GROUPACTIVELOCKEDCOL = ( CGradientValueData * ) ( PGROUPACTIVELOCKEDCOL . ptr ( ) ) - > getData ( ) ;
auto * const GROUPINACTIVELOCKEDCOL = ( CGradientValueData * ) ( PGROUPINACTIVELOCKEDCOL . ptr ( ) ) - > getData ( ) ;
auto setBorderColor = [ & ] ( CGradientValueData grad ) - > void {
2022-11-26 18:56:43 +01:00
if ( grad = = pWindow - > m_cRealBorderColor )
return ;
pWindow - > m_cRealBorderColorPrevious = pWindow - > m_cRealBorderColor ;
2022-12-16 18:17:31 +01:00
pWindow - > m_cRealBorderColor = grad ;
2023-02-01 22:06:01 +01:00
pWindow - > m_fBorderFadeAnimationProgress . setValueAndWarp ( 0.f ) ;
pWindow - > m_fBorderFadeAnimationProgress = 1.f ;
2022-11-26 18:56:43 +01:00
} ;
2024-08-30 15:18:12 +02:00
const bool IS_SHADOWED_BY_MODAL = pWindow - > m_pXDGSurface & & pWindow - > m_pXDGSurface - > toplevel & & pWindow - > m_pXDGSurface - > toplevel - > anyChildModal ( ) ;
2022-07-12 13:40:55 +02:00
// border
2022-05-26 21:23:13 +02:00
const auto RENDERDATA = g_pLayoutManager - > getCurrentLayout ( ) - > requestRenderHints ( pWindow ) ;
2022-12-31 16:23:56 +01:00
if ( RENDERDATA . isBorderGradient )
setBorderColor ( * RENDERDATA . borderGradient ) ;
2023-02-19 22:07:32 +01:00
else {
2024-04-27 13:43:12 +02:00
const bool GROUPLOCKED = pWindow - > m_sGroupData . pNextWindow . lock ( ) ? pWindow - > getGroupHead ( ) - > m_sGroupData . locked : false ;
2024-05-05 18:16:00 +02:00
if ( pWindow = = m_pLastWindow ) {
2023-09-22 01:42:00 +02:00
const auto * const ACTIVECOLOR =
2024-04-27 13:43:12 +02:00
! pWindow - > m_sGroupData . pNextWindow . lock ( ) ? ( ! pWindow - > m_sGroupData . deny ? ACTIVECOL : NOGROUPACTIVECOL ) : ( GROUPLOCKED ? GROUPACTIVELOCKEDCOL : GROUPACTIVECOL ) ;
2024-07-11 16:10:42 +02:00
setBorderColor ( pWindow - > m_sWindowData . activeBorderColor . valueOr ( * ACTIVECOLOR ) ) ;
2023-02-19 22:07:32 +01:00
} else {
2024-04-27 13:43:12 +02:00
const auto * const INACTIVECOLOR = ! pWindow - > m_sGroupData . pNextWindow . lock ( ) ? ( ! pWindow - > m_sGroupData . deny ? INACTIVECOL : NOGROUPINACTIVECOL ) :
( GROUPLOCKED ? GROUPINACTIVELOCKEDCOL : GROUPINACTIVECOL ) ;
2024-07-11 16:10:42 +02:00
setBorderColor ( pWindow - > m_sWindowData . inactiveBorderColor . valueOr ( * INACTIVECOLOR ) ) ;
2023-02-19 22:07:32 +01:00
}
}
2022-07-12 13:40:55 +02:00
2023-02-01 22:06:01 +01:00
// tick angle if it's not running (aka dead)
if ( ! pWindow - > m_fBorderAngleAnimationProgress . isBeingAnimated ( ) )
pWindow - > m_fBorderAngleAnimationProgress . setValueAndWarp ( 0.f ) ;
2022-07-12 13:40:55 +02:00
// opacity
2024-04-02 21:32:39 +02:00
const auto PWORKSPACE = pWindow - > m_pWorkspace ;
2024-07-31 19:55:52 +02:00
if ( pWindow - > isEffectiveInternalFSMode ( FSMODE_FULLSCREEN ) ) {
2024-07-11 16:10:42 +02:00
pWindow - > m_fActiveInactiveAlpha = pWindow - > m_sWindowData . alphaFullscreen . valueOrDefault ( ) . applyAlpha ( * PFULLSCREENALPHA ) ;
2022-07-12 13:40:55 +02:00
} else {
2024-05-05 18:16:00 +02:00
if ( pWindow = = m_pLastWindow )
2024-07-11 16:10:42 +02:00
pWindow - > m_fActiveInactiveAlpha = pWindow - > m_sWindowData . alpha . valueOrDefault ( ) . applyAlpha ( * PACTIVEALPHA ) ;
2022-07-12 13:40:55 +02:00
else
2024-07-11 16:10:42 +02:00
pWindow - > m_fActiveInactiveAlpha = pWindow - > m_sWindowData . alphaInactive . valueOrDefault ( ) . applyAlpha ( * PINACTIVEALPHA ) ;
2022-07-12 13:40:55 +02:00
}
2022-07-16 12:44:45 +02:00
2022-08-30 12:46:17 +02:00
// dim
2024-08-30 15:18:12 +02:00
float goalDim = 1.F ;
if ( pWindow = = m_pLastWindow . lock ( ) | | pWindow - > m_sWindowData . noDim . valueOrDefault ( ) | | ! * PDIMENABLED )
goalDim = 0 ;
else
goalDim = * PDIMSTRENGTH ;
if ( IS_SHADOWED_BY_MODAL )
goalDim + = ( 1.F - goalDim ) / 2.F ;
pWindow - > m_fDimPercent = goalDim ;
2022-08-30 12:46:17 +02:00
2022-07-16 12:44:45 +02:00
// shadow
2024-08-30 14:12:23 +02:00
if ( ! pWindow - > isX11OverrideRedirect ( ) & & ! pWindow - > m_bX11DoesntWantBorders ) {
2024-11-05 16:44:40 +01:00
if ( pWindow = = m_pLastWindow )
2024-12-03 19:58:24 +01:00
pWindow - > m_cRealShadowColor = CHyprColor ( * PSHADOWCOL ) ;
2024-11-05 16:44:40 +01:00
else
2024-12-03 19:58:24 +01:00
pWindow - > m_cRealShadowColor = CHyprColor ( * PSHADOWCOLINACTIVE ! = INT64_MAX ? * PSHADOWCOLINACTIVE : * PSHADOWCOL ) ;
2022-07-16 12:44:45 +02:00
} else {
2024-12-03 19:58:24 +01:00
pWindow - > m_cRealShadowColor . setValueAndWarp ( CHyprColor ( 0 , 0 , 0 , 0 ) ) ; // no shadow
2022-07-16 12:44:45 +02:00
}
2023-02-28 00:34:41 +01:00
2023-11-11 15:37:17 +01:00
pWindow - > updateWindowDecos ( ) ;
2022-05-30 17:11:35 +02:00
}
2024-08-08 21:01:50 +02:00
MONITORID CCompositor : : getNextAvailableMonitorID ( std : : string const & name ) {
2023-07-18 12:12:05 +02:00
// reuse ID if it's already in the map, and the monitor with that ID is not being used by another monitor
if ( m_mMonitorIDMap . contains ( name ) & & ! std : : any_of ( m_vRealMonitors . begin ( ) , m_vRealMonitors . end ( ) , [ & ] ( auto m ) { return m - > ID = = m_mMonitorIDMap [ name ] ; } ) )
2023-07-09 23:08:40 +02:00
return m_mMonitorIDMap [ name ] ;
// otherwise, find minimum available ID that is not in the map
2024-08-08 21:01:50 +02:00
std : : unordered_set < MONITORID > usedIDs ;
2023-07-09 23:08:40 +02:00
for ( auto const & monitor : m_vRealMonitors ) {
usedIDs . insert ( monitor - > ID ) ;
2022-05-30 17:11:35 +02:00
}
2024-08-08 21:01:50 +02:00
MONITORID nextID = 0 ;
2023-07-09 23:08:40 +02:00
while ( usedIDs . count ( nextID ) > 0 ) {
nextID + + ;
}
m_mMonitorIDMap [ name ] = nextID ;
return nextID ;
2022-05-30 17:11:35 +02:00
}
2022-05-30 20:05:38 +02:00
2024-10-20 00:03:29 +02:00
void CCompositor : : swapActiveWorkspaces ( PHLMONITOR pMonitorA , PHLMONITOR pMonitorB ) {
2022-08-25 21:25:28 +02:00
2024-04-02 21:32:39 +02:00
const auto PWORKSPACEA = pMonitorA - > activeWorkspace ;
const auto PWORKSPACEB = pMonitorB - > activeWorkspace ;
2022-08-25 21:25:28 +02:00
2024-10-27 19:45:38 +01:00
PWORKSPACEA - > m_pMonitor = pMonitorB ;
2022-08-25 21:25:28 +02:00
PWORKSPACEA - > moveToMonitor ( pMonitorB - > ID ) ;
2024-08-26 20:24:30 +02:00
for ( auto const & w : m_vWindows ) {
2024-04-02 21:32:39 +02:00
if ( w - > m_pWorkspace = = PWORKSPACEA ) {
2023-03-20 02:50:46 +01:00
if ( w - > m_bPinned ) {
2024-04-02 21:32:39 +02:00
w - > m_pWorkspace = PWORKSPACEB ;
2023-03-20 02:50:46 +01:00
continue ;
}
2024-10-27 19:45:38 +01:00
w - > m_pMonitor = pMonitorB ;
2022-08-25 21:25:28 +02:00
2022-08-27 19:11:21 +02:00
// additionally, move floating and fs windows manually
if ( w - > m_bIsFloating )
2024-03-30 17:57:43 +01:00
w - > m_vRealPosition = w - > m_vRealPosition . goal ( ) - pMonitorA - > vecPosition + pMonitorB - > vecPosition ;
2022-08-27 19:11:21 +02:00
2024-07-31 19:55:52 +02:00
if ( w - > isFullscreen ( ) ) {
2022-08-27 19:11:21 +02:00
w - > m_vRealPosition = pMonitorB - > vecPosition ;
2022-12-16 18:17:31 +01:00
w - > m_vRealSize = pMonitorB - > vecSize ;
2022-08-25 21:25:28 +02:00
}
w - > updateToplevel ( ) ;
}
}
2024-10-27 19:45:38 +01:00
PWORKSPACEB - > m_pMonitor = pMonitorA ;
2022-08-25 21:25:28 +02:00
PWORKSPACEB - > moveToMonitor ( pMonitorA - > ID ) ;
2024-08-26 20:24:30 +02:00
for ( auto const & w : m_vWindows ) {
2024-04-02 21:32:39 +02:00
if ( w - > m_pWorkspace = = PWORKSPACEB ) {
2023-03-20 02:50:46 +01:00
if ( w - > m_bPinned ) {
2024-04-02 21:32:39 +02:00
w - > m_pWorkspace = PWORKSPACEA ;
2023-03-20 02:50:46 +01:00
continue ;
}
2024-10-27 19:45:38 +01:00
w - > m_pMonitor = pMonitorA ;
2022-08-25 21:25:28 +02:00
2022-08-27 19:11:21 +02:00
// additionally, move floating and fs windows manually
if ( w - > m_bIsFloating )
2024-03-30 17:57:43 +01:00
w - > m_vRealPosition = w - > m_vRealPosition . goal ( ) - pMonitorB - > vecPosition + pMonitorA - > vecPosition ;
2022-08-27 19:11:21 +02:00
2024-07-31 19:55:52 +02:00
if ( w - > isFullscreen ( ) ) {
2022-08-27 19:11:21 +02:00
w - > m_vRealPosition = pMonitorA - > vecPosition ;
2022-12-16 18:17:31 +01:00
w - > m_vRealSize = pMonitorA - > vecSize ;
2022-08-25 21:25:28 +02:00
}
w - > updateToplevel ( ) ;
}
}
2024-04-02 21:32:39 +02:00
pMonitorA - > activeWorkspace = PWORKSPACEB ;
pMonitorB - > activeWorkspace = PWORKSPACEA ;
2022-08-25 21:25:28 +02:00
2024-03-04 18:05:20 +01:00
PWORKSPACEA - > rememberPrevWorkspace ( PWORKSPACEB ) ;
PWORKSPACEB - > rememberPrevWorkspace ( PWORKSPACEA ) ;
2022-08-25 21:25:28 +02:00
g_pLayoutManager - > getCurrentLayout ( ) - > recalculateMonitor ( pMonitorA - > ID ) ;
g_pLayoutManager - > getCurrentLayout ( ) - > recalculateMonitor ( pMonitorB - > ID ) ;
2023-04-25 00:21:51 +02:00
updateFullscreenFadeOnWorkspace ( PWORKSPACEB ) ;
updateFullscreenFadeOnWorkspace ( PWORKSPACEA ) ;
2024-02-13 18:53:50 +01:00
if ( pMonitorA - > ID = = g_pCompositor - > m_pLastMonitor - > ID | | pMonitorB - > ID = = g_pCompositor - > m_pLastMonitor - > ID ) {
const auto LASTWIN = pMonitorA - > ID = = g_pCompositor - > m_pLastMonitor - > ID ? PWORKSPACEB - > getLastFocusedWindow ( ) : PWORKSPACEA - > getLastFocusedWindow ( ) ;
g_pCompositor - > focusWindow ( LASTWIN ? LASTWIN :
( g_pCompositor - > vectorToWindowUnified ( g_pInputManager - > getMouseCoordsInternal ( ) , RESERVED_EXTENTS | INPUT_EXTENTS | ALLOW_FLOATING ) ) ) ;
2024-03-25 02:50:41 +01:00
const auto PNEWWORKSPACE = pMonitorA - > ID = = g_pCompositor - > m_pLastMonitor - > ID ? PWORKSPACEB : PWORKSPACEA ;
g_pEventManager - > postEvent ( SHyprIPCEvent { " workspace " , PNEWWORKSPACE - > m_szName } ) ;
g_pEventManager - > postEvent ( SHyprIPCEvent { " workspacev2 " , std : : format ( " {},{} " , PNEWWORKSPACE - > m_iID , PNEWWORKSPACE - > m_szName ) } ) ;
EMIT_HOOK_EVENT ( " workspace " , PNEWWORKSPACE ) ;
2024-02-13 18:53:50 +01:00
}
2022-08-26 16:05:02 +02:00
// event
g_pEventManager - > postEvent ( SHyprIPCEvent { " moveworkspace " , PWORKSPACEA - > m_szName + " , " + pMonitorB - > szName } ) ;
2024-03-08 18:39:53 +01:00
g_pEventManager - > postEvent ( SHyprIPCEvent { " moveworkspacev2 " , std : : format ( " {},{},{} " , PWORKSPACEA - > m_iID , PWORKSPACEA - > m_szName , pMonitorB - > szName ) } ) ;
2024-04-02 21:32:39 +02:00
EMIT_HOOK_EVENT ( " moveWorkspace " , ( std : : vector < std : : any > { PWORKSPACEA , pMonitorB } ) ) ;
2022-08-26 16:05:02 +02:00
g_pEventManager - > postEvent ( SHyprIPCEvent { " moveworkspace " , PWORKSPACEB - > m_szName + " , " + pMonitorA - > szName } ) ;
2024-03-08 18:39:53 +01:00
g_pEventManager - > postEvent ( SHyprIPCEvent { " moveworkspacev2 " , std : : format ( " {},{},{} " , PWORKSPACEB - > m_iID , PWORKSPACEB - > m_szName , pMonitorA - > szName ) } ) ;
2024-04-02 21:32:39 +02:00
EMIT_HOOK_EVENT ( " moveWorkspace " , ( std : : vector < std : : any > { PWORKSPACEB , pMonitorA } ) ) ;
2022-08-25 21:25:28 +02:00
}
2024-10-20 00:03:29 +02:00
PHLMONITOR CCompositor : : getMonitorFromString ( const std : : string & name ) {
2024-02-27 23:11:59 +01:00
if ( name = = " current " )
2024-10-20 00:03:29 +02:00
return g_pCompositor - > m_pLastMonitor . lock ( ) ;
2024-02-27 23:11:59 +01:00
else if ( isDirection ( name ) )
return getMonitorInDirection ( name [ 0 ] ) ;
else if ( name [ 0 ] = = ' + ' | | name [ 0 ] = = ' - ' ) {
2022-11-10 13:22:19 +01:00
// relative
2023-01-14 20:45:24 +01:00
if ( m_vMonitors . size ( ) = = 1 )
2024-10-20 00:03:29 +02:00
return * m_vMonitors . begin ( ) ;
2023-01-14 20:45:24 +01:00
2022-11-10 13:22:19 +01:00
const auto OFFSET = name [ 0 ] = = ' - ' ? name : name . substr ( 1 ) ;
if ( ! isNumber ( OFFSET ) ) {
Debug : : log ( ERR , " Error in getMonitorFromString: Not a number in relative. " ) ;
return nullptr ;
}
2022-12-08 13:09:48 +01:00
int offsetLeft = std : : stoi ( OFFSET ) ;
2022-12-16 18:17:31 +01:00
offsetLeft = offsetLeft < 0 ? - ( ( - offsetLeft ) % m_vMonitors . size ( ) ) : offsetLeft % m_vMonitors . size ( ) ;
2022-11-10 13:22:19 +01:00
int currentPlace = 0 ;
for ( int i = 0 ; i < ( int ) m_vMonitors . size ( ) ; i + + ) {
2024-05-05 23:18:10 +02:00
if ( m_vMonitors [ i ] = = m_pLastMonitor ) {
2022-11-10 13:22:19 +01:00
currentPlace = i ;
break ;
}
}
currentPlace + = offsetLeft ;
if ( currentPlace < 0 ) {
2023-01-23 21:56:05 +01:00
currentPlace = m_vMonitors . size ( ) + currentPlace ;
2022-11-10 13:22:19 +01:00
} else {
currentPlace = currentPlace % m_vMonitors . size ( ) ;
}
2023-01-23 21:56:43 +01:00
if ( currentPlace ! = std : : clamp ( currentPlace , 0 , ( int ) m_vMonitors . size ( ) - 1 ) ) {
2022-11-10 13:22:19 +01:00
Debug : : log ( WARN , " Error in getMonitorFromString: Vaxry's code sucks. " ) ;
2023-01-14 20:45:24 +01:00
currentPlace = std : : clamp ( currentPlace , 0 , ( int ) m_vMonitors . size ( ) - 1 ) ;
2022-11-10 13:22:19 +01:00
}
2024-10-20 00:03:29 +02:00
return m_vMonitors [ currentPlace ] ;
2022-11-10 13:22:19 +01:00
} else if ( isNumber ( name ) ) {
2022-08-25 21:25:28 +02:00
// change by ID
2024-08-08 21:01:50 +02:00
MONITORID monID = MONITOR_INVALID ;
2022-08-25 21:25:28 +02:00
try {
monID = std : : stoi ( name ) ;
} catch ( std : : exception & e ) {
// shouldn't happen but jic
Debug : : log ( ERR , " Error in getMonitorFromString: invalid num " ) ;
return nullptr ;
}
2024-08-08 21:01:50 +02:00
if ( monID > - 1 & & monID < ( MONITORID ) m_vMonitors . size ( ) ) {
2023-03-02 13:04:41 +01:00
return getMonitorFromID ( monID ) ;
2022-08-25 21:25:28 +02:00
} else {
Debug : : log ( ERR , " Error in getMonitorFromString: invalid arg 1 " ) ;
return nullptr ;
}
2024-02-27 23:11:59 +01:00
} else {
2024-08-26 20:24:30 +02:00
for ( auto const & m : m_vMonitors ) {
2023-03-16 15:03:40 +01:00
if ( ! m - > output )
continue ;
2024-02-27 23:11:59 +01:00
if ( m - > matchesStaticSelector ( name ) ) {
2024-10-20 00:03:29 +02:00
return m ;
2023-03-02 13:04:41 +01:00
}
}
2022-08-25 21:25:28 +02:00
}
return nullptr ;
}
2024-10-20 00:03:29 +02:00
void CCompositor : : moveWorkspaceToMonitor ( PHLWORKSPACE pWorkspace , PHLMONITOR pMonitor , bool noWarpCursor ) {
2022-05-30 20:05:38 +02:00
2024-04-02 21:32:39 +02:00
// We trust the monitor to be correct.
2022-05-30 20:05:38 +02:00
2024-10-27 19:45:38 +01:00
if ( pWorkspace - > m_pMonitor = = pMonitor )
2022-05-30 20:05:38 +02:00
return ;
2023-09-06 12:51:36 +02:00
Debug : : log ( LOG , " moveWorkspaceToMonitor: Moving {} to monitor {} " , pWorkspace - > m_iID , pMonitor - > ID ) ;
2022-05-30 20:05:38 +02:00
2024-10-27 19:45:38 +01:00
const auto POLDMON = pWorkspace - > m_pMonitor . lock ( ) ;
2022-05-30 20:05:38 +02:00
2024-04-02 21:32:39 +02:00
const bool SWITCHINGISACTIVE = POLDMON ? POLDMON - > activeWorkspace = = pWorkspace : false ;
2022-05-30 20:05:38 +02:00
// fix old mon
2024-08-08 21:01:50 +02:00
WORKSPACEID nextWorkspaceOnMonitorID = WORKSPACE_INVALID ;
2024-03-04 18:05:20 +01:00
if ( ! SWITCHINGISACTIVE )
2023-04-28 22:40:40 +02:00
nextWorkspaceOnMonitorID = pWorkspace - > m_iID ;
else {
2024-08-26 17:25:39 +02:00
for ( auto const & w : m_vWorkspaces ) {
2024-10-27 19:45:38 +01:00
if ( w - > m_pMonitor = = POLDMON & & w - > m_iID ! = pWorkspace - > m_iID & & ! w - > m_bIsSpecialWorkspace ) {
2023-04-28 22:40:40 +02:00
nextWorkspaceOnMonitorID = w - > m_iID ;
break ;
}
2022-05-30 20:05:38 +02:00
}
2024-08-08 21:01:50 +02:00
if ( nextWorkspaceOnMonitorID = = WORKSPACE_INVALID ) {
2023-04-28 22:40:40 +02:00
nextWorkspaceOnMonitorID = 1 ;
2022-05-30 20:05:38 +02:00
2023-04-28 22:40:40 +02:00
while ( getWorkspaceByID ( nextWorkspaceOnMonitorID ) | | [ & ] ( ) - > bool {
const auto B = g_pConfigManager - > getBoundMonitorForWS ( std : : to_string ( nextWorkspaceOnMonitorID ) ) ;
return B & & B ! = POLDMON ;
} ( ) )
nextWorkspaceOnMonitorID + + ;
2022-05-30 20:05:38 +02:00
2023-09-06 12:51:36 +02:00
Debug : : log ( LOG , " moveWorkspaceToMonitor: Plugging gap with new {} " , nextWorkspaceOnMonitorID ) ;
2023-04-14 16:03:53 +02:00
2023-04-28 22:40:40 +02:00
g_pCompositor - > createNewWorkspace ( nextWorkspaceOnMonitorID , POLDMON - > ID ) ;
}
2022-05-30 20:05:38 +02:00
2023-09-06 12:51:36 +02:00
Debug : : log ( LOG , " moveWorkspaceToMonitor: Plugging gap with existing {} " , nextWorkspaceOnMonitorID ) ;
2024-02-10 00:47:00 +01:00
POLDMON - > changeWorkspace ( nextWorkspaceOnMonitorID , false , true , true ) ;
2023-04-28 22:40:40 +02:00
}
2023-04-13 22:09:50 +02:00
2022-05-30 20:05:38 +02:00
// move the workspace
2024-10-27 19:45:38 +01:00
pWorkspace - > m_pMonitor = pMonitor ;
2022-05-30 20:51:45 +02:00
pWorkspace - > moveToMonitor ( pMonitor - > ID ) ;
2022-05-30 20:05:38 +02:00
2024-08-26 20:24:30 +02:00
for ( auto const & w : m_vWindows ) {
2024-04-02 21:32:39 +02:00
if ( w - > m_pWorkspace = = pWorkspace ) {
2023-03-20 02:50:46 +01:00
if ( w - > m_bPinned ) {
2024-04-02 21:32:39 +02:00
w - > m_pWorkspace = g_pCompositor - > getWorkspaceByID ( nextWorkspaceOnMonitorID ) ;
2023-03-20 02:50:46 +01:00
continue ;
}
2024-10-27 19:45:38 +01:00
w - > m_pMonitor = pMonitor ;
2022-07-25 14:22:32 +02:00
2022-08-27 19:11:21 +02:00
// additionally, move floating and fs windows manually
2022-10-14 21:46:32 +02:00
if ( w - > m_bIsMapped & & ! w - > isHidden ( ) ) {
2023-05-06 17:49:46 +02:00
if ( POLDMON ) {
if ( w - > m_bIsFloating )
2024-03-30 17:57:43 +01:00
w - > m_vRealPosition = w - > m_vRealPosition . goal ( ) - POLDMON - > vecPosition + pMonitor - > vecPosition ;
2022-08-27 19:11:21 +02:00
2024-07-31 19:55:52 +02:00
if ( w - > isFullscreen ( ) ) {
2023-05-06 17:49:46 +02:00
w - > m_vRealPosition = pMonitor - > vecPosition ;
w - > m_vRealSize = pMonitor - > vecSize ;
}
} else {
2024-03-02 01:35:17 +01:00
w - > m_vRealPosition = Vector2D { ( int ) w - > m_vRealPosition . goal ( ) . x % ( int ) pMonitor - > vecSize . x , ( int ) w - > m_vRealPosition . goal ( ) . y % ( int ) pMonitor - > vecSize . y } ;
2022-08-27 19:11:21 +02:00
}
2022-07-25 14:22:32 +02:00
}
2022-08-06 20:57:38 +02:00
w - > updateToplevel ( ) ;
2022-07-25 14:22:32 +02:00
}
2022-05-30 20:05:38 +02:00
}
2024-10-20 00:03:29 +02:00
if ( SWITCHINGISACTIVE & & POLDMON = = g_pCompositor - > m_pLastMonitor ) { // if it was active, preserve its' status. If it wasn't, don't.
2024-04-02 21:32:39 +02:00
Debug : : log ( LOG , " moveWorkspaceToMonitor: SWITCHINGISACTIVE, active {} -> {} " , pMonitor - > activeWorkspaceID ( ) , pWorkspace - > m_iID ) ;
2022-05-31 17:17:44 +02:00
2024-04-04 19:30:32 +02:00
if ( valid ( pMonitor - > activeWorkspace ) ) {
pMonitor - > activeWorkspace - > m_bVisible = false ;
2024-04-02 21:32:39 +02:00
pMonitor - > activeWorkspace - > startAnim ( false , false ) ;
2024-04-04 19:30:32 +02:00
}
2022-05-30 20:05:38 +02:00
2024-02-10 00:47:00 +01:00
setActiveMonitor ( pMonitor ) ;
2024-04-02 21:32:39 +02:00
pMonitor - > activeWorkspace = pWorkspace ;
2022-05-30 20:05:38 +02:00
g_pLayoutManager - > getCurrentLayout ( ) - > recalculateMonitor ( pMonitor - > ID ) ;
pWorkspace - > startAnim ( true , true , true ) ;
2024-04-04 19:30:32 +02:00
pWorkspace - > m_bVisible = true ;
2022-05-30 20:05:38 +02:00
2024-01-15 16:30:46 +01:00
if ( ! noWarpCursor )
2024-05-05 23:18:10 +02:00
g_pPointerManager - > warpTo ( pMonitor - > vecPosition + pMonitor - > vecTransformedSize / 2.F ) ;
2023-05-06 17:13:26 +02:00
2023-12-12 15:58:43 +01:00
g_pInputManager - > sendMotionEventsToFocused ( ) ;
2022-05-30 20:05:38 +02:00
}
// finalize
2023-05-06 17:49:46 +02:00
if ( POLDMON ) {
g_pLayoutManager - > getCurrentLayout ( ) - > recalculateMonitor ( POLDMON - > ID ) ;
2024-04-02 21:32:39 +02:00
updateFullscreenFadeOnWorkspace ( POLDMON - > activeWorkspace ) ;
2024-06-12 16:02:19 +02:00
updateSuspendedStates ( ) ;
2023-05-06 17:49:46 +02:00
}
2022-05-30 20:05:38 +02:00
2023-04-25 00:23:12 +02:00
updateFullscreenFadeOnWorkspace ( pWorkspace ) ;
2024-06-12 16:02:19 +02:00
updateSuspendedStates ( ) ;
2023-04-25 00:23:12 +02:00
2022-08-26 16:05:02 +02:00
// event
g_pEventManager - > postEvent ( SHyprIPCEvent { " moveworkspace " , pWorkspace - > m_szName + " , " + pMonitor - > szName } ) ;
2024-03-08 18:39:53 +01:00
g_pEventManager - > postEvent ( SHyprIPCEvent { " moveworkspacev2 " , std : : format ( " {},{},{} " , pWorkspace - > m_iID , pWorkspace - > m_szName , pMonitor - > szName ) } ) ;
2024-04-02 21:32:39 +02:00
EMIT_HOOK_EVENT ( " moveWorkspace " , ( std : : vector < std : : any > { pWorkspace , pMonitor } ) ) ;
2022-05-31 12:33:08 +02:00
}
2024-08-08 21:01:50 +02:00
bool CCompositor : : workspaceIDOutOfBounds ( const WORKSPACEID & id ) {
WORKSPACEID lowestID = INT64_MAX ;
WORKSPACEID highestID = INT64_MIN ;
2022-05-31 12:33:08 +02:00
2024-08-26 17:25:39 +02:00
for ( auto const & w : m_vWorkspaces ) {
2022-11-27 23:42:22 +01:00
if ( w - > m_bIsSpecialWorkspace )
2022-09-17 16:05:12 +02:00
continue ;
2022-06-30 15:44:26 +02:00
if ( w - > m_iID < lowestID )
lowestID = w - > m_iID ;
2022-09-25 20:07:48 +02:00
2022-06-30 15:44:26 +02:00
if ( w - > m_iID > highestID )
highestID = w - > m_iID ;
2022-05-31 12:33:08 +02:00
}
return std : : clamp ( id , lowestID , highestID ) ! = id ;
2022-06-26 12:12:29 +02:00
}
2024-04-02 21:32:39 +02:00
void CCompositor : : updateFullscreenFadeOnWorkspace ( PHLWORKSPACE pWorkspace ) {
2023-04-25 00:21:51 +02:00
const auto FULLSCREEN = pWorkspace - > m_bHasFullscreenWindow ;
2024-08-26 20:24:30 +02:00
for ( auto const & w : g_pCompositor - > m_vWindows ) {
2024-04-02 21:32:39 +02:00
if ( w - > m_pWorkspace = = pWorkspace ) {
2023-04-25 00:21:51 +02:00
2024-07-31 19:55:52 +02:00
if ( w - > m_bFadingOut | | w - > m_bPinned | | w - > isFullscreen ( ) )
2023-04-25 00:21:51 +02:00
continue ;
if ( ! FULLSCREEN )
w - > m_fAlpha = 1.f ;
2024-07-31 19:55:52 +02:00
else if ( ! w - > isFullscreen ( ) )
2023-04-25 00:21:51 +02:00
w - > m_fAlpha = ! w - > m_bCreatedOverFullscreen ? 0.f : 1.f ;
}
}
2024-10-27 19:45:38 +01:00
const auto PMONITOR = pWorkspace - > m_pMonitor . lock ( ) ;
2023-04-25 00:21:51 +02:00
2024-04-02 21:32:39 +02:00
if ( pWorkspace - > m_iID = = PMONITOR - > activeWorkspaceID ( ) | | pWorkspace - > m_iID = = PMONITOR - > activeSpecialWorkspaceID ( ) ) {
2024-08-26 17:25:39 +02:00
for ( auto const & ls : PMONITOR - > m_aLayerSurfaceLayers [ ZWLR_LAYER_SHELL_V1_LAYER_TOP ] ) {
2023-08-21 20:55:57 +02:00
if ( ! ls - > fadingOut )
2024-07-31 19:55:52 +02:00
ls - > alpha = FULLSCREEN & & pWorkspace - > m_efFullscreenMode = = FSMODE_FULLSCREEN ? 0.f : 1.f ;
2023-08-21 20:55:57 +02:00
}
2023-04-25 00:21:51 +02:00
}
}
2024-07-31 19:55:52 +02:00
void CCompositor : : changeWindowFullscreenModeClient ( const PHLWINDOW PWINDOW , const eFullscreenMode MODE , const bool ON ) {
setWindowFullscreenClient ( PWINDOW ,
( eFullscreenMode ) ( ON ? ( uint8_t ) PWINDOW - > m_sFullscreenState . client | ( uint8_t ) MODE : ( ( uint8_t ) PWINDOW - > m_sFullscreenState . client & ( uint8_t ) ~ MODE ) ) ) ;
}
void CCompositor : : setWindowFullscreenInternal ( const PHLWINDOW PWINDOW , const eFullscreenMode MODE ) {
if ( PWINDOW - > m_sWindowData . syncFullscreen . valueOrDefault ( ) )
2024-12-07 18:51:18 +01:00
setWindowFullscreenState ( PWINDOW , SFullscreenState { . internal = MODE , . client = MODE } ) ;
2024-07-31 19:55:52 +02:00
else
2024-12-07 18:51:18 +01:00
setWindowFullscreenState ( PWINDOW , SFullscreenState { . internal = MODE , . client = PWINDOW - > m_sFullscreenState . client } ) ;
2024-07-31 19:55:52 +02:00
}
void CCompositor : : setWindowFullscreenClient ( const PHLWINDOW PWINDOW , const eFullscreenMode MODE ) {
if ( PWINDOW - > m_sWindowData . syncFullscreen . valueOrDefault ( ) )
2024-12-07 18:51:18 +01:00
setWindowFullscreenState ( PWINDOW , SFullscreenState { . internal = MODE , . client = MODE } ) ;
2024-07-31 19:55:52 +02:00
else
2024-12-07 18:51:18 +01:00
setWindowFullscreenState ( PWINDOW , SFullscreenState { . internal = PWINDOW - > m_sFullscreenState . internal , . client = MODE } ) ;
2024-07-31 19:55:52 +02:00
}
2024-12-07 18:51:18 +01:00
void CCompositor : : setWindowFullscreenState ( const PHLWINDOW PWINDOW , SFullscreenState state ) {
2024-11-23 15:32:13 +01:00
static auto PDIRECTSCANOUT = CConfigValue < Hyprlang : : INT > ( " render:direct_scanout " ) ;
static auto PALLOWPINFULLSCREEN = CConfigValue < Hyprlang : : INT > ( " binds:allow_pin_fullscreen " ) ;
2024-07-21 23:25:20 +02:00
2024-07-31 19:55:52 +02:00
if ( ! validMapped ( PWINDOW ) | | g_pCompositor - > m_bUnsafeState )
2022-06-26 12:12:29 +02:00
return ;
2024-07-31 19:55:52 +02:00
state . internal = std : : clamp ( state . internal , ( eFullscreenMode ) 0 , FSMODE_MAX ) ;
state . client = std : : clamp ( state . client , ( eFullscreenMode ) 0 , FSMODE_MAX ) ;
2022-09-21 15:09:26 +02:00
2024-10-27 19:45:38 +01:00
const auto PMONITOR = PWINDOW - > m_pMonitor . lock ( ) ;
2024-07-31 19:55:52 +02:00
const auto PWORKSPACE = PWINDOW - > m_pWorkspace ;
2024-03-13 03:09:20 +01:00
2024-07-31 19:55:52 +02:00
const eFullscreenMode CURRENT_EFFECTIVE_MODE = ( eFullscreenMode ) std : : bit_floor ( ( uint8_t ) PWINDOW - > m_sFullscreenState . internal ) ;
const eFullscreenMode EFFECTIVE_MODE = ( eFullscreenMode ) std : : bit_floor ( ( uint8_t ) state . internal ) ;
2022-09-29 11:20:12 +02:00
2024-11-23 15:32:13 +01:00
if ( * PALLOWPINFULLSCREEN & & ! PWINDOW - > m_bPinFullscreened & & ! PWINDOW - > isFullscreen ( ) & & PWINDOW - > m_bPinned ) {
PWINDOW - > m_bPinned = false ;
PWINDOW - > m_bPinFullscreened = true ;
}
2024-12-10 21:09:47 +01:00
if ( PWORKSPACE - > m_bHasFullscreenWindow & & ! PWINDOW - > isFullscreen ( ) )
setWindowFullscreenInternal ( PWORKSPACE - > getFullscreenWindow ( ) , FSMODE_NONE ) ;
const bool CHANGEINTERNAL = ! ( PWINDOW - > m_bPinned | | CURRENT_EFFECTIVE_MODE = = EFFECTIVE_MODE ) ;
2024-11-23 15:32:13 +01:00
if ( * PALLOWPINFULLSCREEN & & PWINDOW - > m_bPinFullscreened & & PWINDOW - > isFullscreen ( ) & & ! PWINDOW - > m_bPinned & & state . internal = = FSMODE_NONE ) {
PWINDOW - > m_bPinned = true ;
PWINDOW - > m_bPinFullscreened = false ;
}
2022-09-29 11:20:12 +02:00
2024-07-31 19:55:52 +02:00
// TODO: update the state on syncFullscreen changes
if ( ! CHANGEINTERNAL & & PWINDOW - > m_sWindowData . syncFullscreen . valueOrDefault ( ) )
return ;
2023-09-28 18:49:33 +02:00
2024-07-31 19:55:52 +02:00
PWINDOW - > m_sFullscreenState . client = state . client ;
g_pXWaylandManager - > setWindowFullscreen ( PWINDOW , state . client & FSMODE_FULLSCREEN ) ;
2024-08-23 21:42:14 +02:00
if ( ! CHANGEINTERNAL ) {
PWINDOW - > updateDynamicRules ( ) ;
updateWindowAnimatedDecorationValues ( PWINDOW ) ;
2024-10-27 19:45:38 +01:00
g_pLayoutManager - > getCurrentLayout ( ) - > recalculateMonitor ( PWINDOW - > monitorID ( ) ) ;
2022-09-29 11:20:12 +02:00
return ;
2024-08-23 21:42:14 +02:00
}
2022-09-29 11:20:12 +02:00
2024-07-31 19:55:52 +02:00
g_pLayoutManager - > getCurrentLayout ( ) - > fullscreenRequestForWindow ( PWINDOW , CURRENT_EFFECTIVE_MODE , EFFECTIVE_MODE ) ;
2022-06-26 12:12:29 +02:00
2024-07-31 19:55:52 +02:00
PWINDOW - > m_sFullscreenState . internal = state . internal ;
PWORKSPACE - > m_efFullscreenMode = EFFECTIVE_MODE ;
PWORKSPACE - > m_bHasFullscreenWindow = EFFECTIVE_MODE ! = FSMODE_NONE ;
2022-09-25 20:07:48 +02:00
2024-07-31 19:55:52 +02:00
g_pEventManager - > postEvent ( SHyprIPCEvent { " fullscreen " , std : : to_string ( ( int ) EFFECTIVE_MODE ! = FSMODE_NONE ) } ) ;
EMIT_HOOK_EVENT ( " fullscreen " , PWINDOW ) ;
2024-08-01 11:43:02 +02:00
PWINDOW - > updateDynamicRules ( ) ;
2024-07-31 19:55:52 +02:00
updateWindowAnimatedDecorationValues ( PWINDOW ) ;
2024-10-27 19:45:38 +01:00
g_pLayoutManager - > getCurrentLayout ( ) - > recalculateMonitor ( PWINDOW - > monitorID ( ) ) ;
2022-11-15 11:21:26 +01:00
2022-06-26 12:12:29 +02:00
// make all windows on the same workspace under the fullscreen window
2024-08-26 20:24:30 +02:00
for ( auto const & w : m_vWindows ) {
2024-07-31 19:55:52 +02:00
if ( w - > m_pWorkspace = = PWORKSPACE & & ! w - > isFullscreen ( ) & & ! w - > m_bFadingOut & & ! w - > m_bPinned )
2022-06-30 15:44:26 +02:00
w - > m_bCreatedOverFullscreen = false ;
2022-09-19 11:23:13 +02:00
}
2024-07-31 19:55:52 +02:00
2023-04-25 00:21:51 +02:00
updateFullscreenFadeOnWorkspace ( PWORKSPACE ) ;
2022-09-25 20:07:48 +02:00
2024-07-31 19:55:52 +02:00
g_pXWaylandManager - > setWindowSize ( PWINDOW , PWINDOW - > m_vRealSize . goal ( ) , true ) ;
2022-08-22 14:22:21 +02:00
2024-11-22 17:01:02 +01:00
PWORKSPACE - > forceReportSizesToWindows ( ) ;
2022-10-31 13:26:07 +01:00
g_pInputManager - > recheckIdleInhibitorStatus ( ) ;
2022-11-05 13:50:47 +01:00
2024-07-21 13:09:54 +02:00
// further updates require a monitor
if ( ! PMONITOR )
return ;
// send a scanout tranche if we are entering fullscreen, and send a regular one if we aren't.
2024-07-21 23:25:20 +02:00
// ignore if DS is disabled.
2024-08-06 15:52:19 +02:00
if ( * PDIRECTSCANOUT )
2024-07-31 19:55:52 +02:00
g_pHyprRenderer - > setSurfaceScanoutMode ( PWINDOW - > m_pWLSurface - > resource ( ) , EFFECTIVE_MODE ! = FSMODE_NONE ? PMONITOR - > self . lock ( ) : nullptr ) ;
2023-02-14 18:08:42 +01:00
g_pConfigManager - > ensureVRR ( PMONITOR ) ;
2022-06-30 15:44:26 +02:00
}
2024-04-27 13:43:12 +02:00
PHLWINDOW CCompositor : : getX11Parent ( PHLWINDOW pWindow ) {
2022-06-30 15:44:26 +02:00
if ( ! pWindow - > m_bIsX11 )
return nullptr ;
2024-08-26 17:25:39 +02:00
for ( auto const & w : m_vWindows ) {
2022-06-30 15:44:26 +02:00
if ( ! w - > m_bIsX11 )
continue ;
2024-05-25 22:43:51 +02:00
if ( w - > m_pXWaylandSurface = = pWindow - > m_pXWaylandSurface - > parent )
2024-04-27 13:43:12 +02:00
return w ;
2022-06-30 15:44:26 +02:00
}
2022-09-25 20:07:48 +02:00
2022-06-30 15:44:26 +02:00
return nullptr ;
2022-07-07 11:52:12 +02:00
}
2024-10-20 00:03:29 +02:00
void CCompositor : : scheduleFrameForMonitor ( PHLMONITOR pMonitor , IOutput : : scheduleFrameReason reason ) {
2024-07-21 13:09:54 +02:00
if ( ( m_pAqBackend - > hasSession ( ) & & ! m_pAqBackend - > session - > active ) | | ! m_bSessionActive )
2022-07-13 18:18:23 +02:00
return ;
2022-11-19 14:01:32 +01:00
if ( ! pMonitor - > m_bEnabled )
return ;
2023-02-27 13:32:38 +01:00
if ( pMonitor - > renderingActive )
pMonitor - > pendingFrame = true ;
2024-07-21 13:09:54 +02:00
pMonitor - > output - > scheduleFrame ( reason ) ;
2022-07-26 17:30:30 +02:00
}
2024-10-08 18:50:06 +02:00
PHLWINDOW CCompositor : : getWindowByRegex ( const std : : string & regexp_ ) {
auto regexp = trim ( regexp_ ) ;
2024-04-21 16:19:59 +02:00
if ( regexp . starts_with ( " active " ) )
2024-04-27 13:43:12 +02:00
return m_pLastWindow . lock ( ) ;
2024-10-08 18:50:06 +02:00
else if ( regexp . starts_with ( " floating " ) | | regexp . starts_with ( " tiled " ) ) {
// first floating on the current ws
if ( ! valid ( m_pLastWindow ) )
return nullptr ;
const bool FLOAT = regexp . starts_with ( " floating " ) ;
for ( auto const & w : m_vWindows ) {
if ( ! w - > m_bIsMapped | | w - > m_bIsFloating ! = FLOAT | | w - > m_pWorkspace ! = m_pLastWindow - > m_pWorkspace | | w - > isHidden ( ) )
continue ;
return w ;
}
return nullptr ;
}
2024-04-21 16:19:59 +02:00
2022-07-26 17:30:30 +02:00
eFocusWindowMode mode = MODE_CLASS_REGEX ;
2024-12-16 19:21:44 +01:00
std : : string regexCheck ;
2022-12-16 18:17:31 +01:00
std : : string matchCheck ;
2024-04-21 16:19:59 +02:00
if ( regexp . starts_with ( " class: " ) ) {
2024-12-16 19:21:44 +01:00
regexCheck = regexp . substr ( 6 ) ;
2024-04-21 16:19:59 +02:00
} else if ( regexp . starts_with ( " initialclass: " ) ) {
mode = MODE_INITIAL_CLASS_REGEX ;
2024-12-16 19:21:44 +01:00
regexCheck = regexp . substr ( 13 ) ;
2024-04-21 16:19:59 +02:00
} else if ( regexp . starts_with ( " title: " ) ) {
2022-12-16 18:17:31 +01:00
mode = MODE_TITLE_REGEX ;
2024-12-16 19:21:44 +01:00
regexCheck = regexp . substr ( 6 ) ;
2024-04-21 16:19:59 +02:00
} else if ( regexp . starts_with ( " initialtitle: " ) ) {
mode = MODE_INITIAL_TITLE_REGEX ;
2024-12-16 19:21:44 +01:00
regexCheck = regexp . substr ( 13 ) ;
2023-10-15 20:07:23 +02:00
} else if ( regexp . starts_with ( " address: " ) ) {
2022-12-16 18:17:31 +01:00
mode = MODE_ADDRESS ;
2022-07-26 17:30:30 +02:00
matchCheck = regexp . substr ( 8 ) ;
2023-10-15 20:07:23 +02:00
} else if ( regexp . starts_with ( " pid: " ) ) {
2022-12-16 18:17:31 +01:00
mode = MODE_PID ;
2022-07-26 17:30:30 +02:00
matchCheck = regexp . substr ( 4 ) ;
}
2024-08-26 20:24:30 +02:00
for ( auto const & w : g_pCompositor - > m_vWindows ) {
2024-04-27 13:43:12 +02:00
if ( ! w - > m_bIsMapped | | ( w - > isHidden ( ) & & ! g_pLayoutManager - > getCurrentLayout ( ) - > isWindowReachable ( w ) ) )
2022-07-26 17:30:30 +02:00
continue ;
switch ( mode ) {
case MODE_CLASS_REGEX : {
2024-05-11 00:28:33 +02:00
const auto windowClass = w - > m_szClass ;
2024-12-16 19:21:44 +01:00
if ( ! RE2 : : FullMatch ( windowClass , regexCheck ) )
2024-04-21 16:19:59 +02:00
continue ;
break ;
}
case MODE_INITIAL_CLASS_REGEX : {
const auto initialWindowClass = w - > m_szInitialClass ;
2024-12-16 19:21:44 +01:00
if ( ! RE2 : : FullMatch ( initialWindowClass , regexCheck ) )
2022-07-26 17:30:30 +02:00
continue ;
break ;
}
case MODE_TITLE_REGEX : {
2024-05-11 00:28:33 +02:00
const auto windowTitle = w - > m_szTitle ;
2024-12-16 19:21:44 +01:00
if ( ! RE2 : : FullMatch ( windowTitle , regexCheck ) )
2022-07-26 17:30:30 +02:00
continue ;
break ;
}
2024-04-21 16:19:59 +02:00
case MODE_INITIAL_TITLE_REGEX : {
const auto initialWindowTitle = w - > m_szInitialTitle ;
2024-12-16 19:21:44 +01:00
if ( ! RE2 : : FullMatch ( initialWindowTitle , regexCheck ) )
2024-04-21 16:19:59 +02:00
continue ;
break ;
}
2022-07-26 17:30:30 +02:00
case MODE_ADDRESS : {
2023-09-20 09:26:20 +02:00
std : : string addr = std : : format ( " 0x{:x} " , ( uintptr_t ) w . get ( ) ) ;
2022-07-26 17:30:30 +02:00
if ( matchCheck ! = addr )
continue ;
break ;
}
case MODE_PID : {
2023-09-20 09:26:20 +02:00
std : : string pid = std : : format ( " {} " , w - > getPID ( ) ) ;
2022-07-26 17:30:30 +02:00
if ( matchCheck ! = pid )
continue ;
break ;
}
2022-12-16 18:17:31 +01:00
default : break ;
2022-07-26 17:30:30 +02:00
}
2024-04-27 13:43:12 +02:00
return w ;
2022-07-26 17:30:30 +02:00
}
return nullptr ;
2022-07-31 00:27:32 +02:00
}
2023-01-28 13:26:33 +01:00
void CCompositor : : warpCursorTo ( const Vector2D & pos , bool force ) {
2022-08-01 18:50:16 +02:00
// warpCursorTo should only be used for warps that
2024-05-09 23:23:01 +02:00
// should be disabled with no_warps
2022-08-01 18:50:16 +02:00
2024-05-09 23:23:01 +02:00
static auto PNOWARPS = CConfigValue < Hyprlang : : INT > ( " cursor:no_warps " ) ;
2022-08-01 18:50:16 +02:00
2024-05-27 22:45:14 +02:00
if ( * PNOWARPS & & ! force ) {
const auto PMONITORNEW = getMonitorFromVector ( pos ) ;
2024-10-20 00:03:29 +02:00
if ( PMONITORNEW ! = m_pLastMonitor )
2024-05-27 22:45:14 +02:00
setActiveMonitor ( PMONITORNEW ) ;
2022-08-01 18:50:16 +02:00
return ;
2024-05-27 22:45:14 +02:00
}
2022-08-01 18:50:16 +02:00
2024-05-05 23:18:10 +02:00
g_pPointerManager - > warpTo ( pos ) ;
2022-09-03 11:55:14 +02:00
const auto PMONITORNEW = getMonitorFromVector ( pos ) ;
2024-10-20 00:03:29 +02:00
if ( PMONITORNEW ! = m_pLastMonitor )
2022-11-19 17:41:36 +01:00
setActiveMonitor ( PMONITORNEW ) ;
2022-08-01 18:50:16 +02:00
}
2024-04-27 13:43:12 +02:00
void CCompositor : : closeWindow ( PHLWINDOW pWindow ) {
if ( pWindow & & validMapped ( pWindow ) ) {
2022-08-06 20:57:38 +02:00
g_pXWaylandManager - > sendCloseWindow ( pWindow ) ;
}
2022-08-10 17:46:01 +02:00
}
2024-06-08 10:07:59 +02:00
PHLLS CCompositor : : getLayerSurfaceFromSurface ( SP < CWLSurfaceResource > pSurface ) {
std : : pair < SP < CWLSurfaceResource > , bool > result = { pSurface , false } ;
2023-04-02 14:30:45 +02:00
2024-08-26 17:25:39 +02:00
for ( auto const & ls : m_vLayers ) {
2024-05-09 22:47:21 +02:00
if ( ls - > layerSurface & & ls - > layerSurface - > surface = = pSurface )
return ls ;
2023-04-02 14:42:57 +02:00
2024-05-09 22:47:21 +02:00
if ( ! ls - > layerSurface | | ! ls - > mapped )
continue ;
2023-04-02 14:30:45 +02:00
2024-06-08 10:07:59 +02:00
ls - > layerSurface - > surface - > breadthfirst (
[ ] ( SP < CWLSurfaceResource > surf , const Vector2D & offset , void * data ) {
if ( surf = = ( ( std : : pair < SP < CWLSurfaceResource > , bool > * ) data ) - > first ) {
* ( bool * ) data = true ;
return ;
}
} ,
& result ) ;
2024-05-09 22:47:21 +02:00
if ( result . second )
return ls ;
2022-08-10 17:46:01 +02:00
}
return nullptr ;
2022-08-15 15:59:07 +02:00
}
// returns a delta
Vector2D CCompositor : : parseWindowVectorArgsRelative ( const std : : string & args , const Vector2D & relativeTo ) {
2024-03-31 01:48:39 +01:00
if ( ! args . contains ( ' ' ) & & ! args . contains ( ' \t ' ) )
2022-08-15 15:59:07 +02:00
return relativeTo ;
2023-09-11 00:26:14 +02:00
const auto PMONITOR = m_pLastMonitor ;
bool xIsPercent = false ;
bool yIsPercent = false ;
bool isExact = false ;
2024-03-31 01:48:39 +01:00
CVarList varList ( args , 0 , ' s ' , true ) ;
std : : string x = varList [ 0 ] ;
std : : string y = varList [ 1 ] ;
2022-08-15 15:59:07 +02:00
if ( x = = " exact " ) {
2024-03-31 01:48:39 +01:00
x = varList [ 1 ] ;
y = varList [ 2 ] ;
2023-09-11 00:26:14 +02:00
isExact = true ;
}
2022-08-15 15:59:07 +02:00
2023-09-11 00:26:14 +02:00
if ( x . contains ( ' % ' ) ) {
xIsPercent = true ;
x = x . substr ( 0 , x . length ( ) - 1 ) ;
}
2022-08-15 15:59:07 +02:00
2023-09-11 00:26:14 +02:00
if ( y . contains ( ' % ' ) ) {
yIsPercent = true ;
y = y . substr ( 0 , y . length ( ) - 1 ) ;
2022-08-15 15:59:07 +02:00
}
if ( ! isNumber ( x ) | | ! isNumber ( y ) ) {
Debug : : log ( ERR , " parseWindowVectorArgsRelative: args not numbers " ) ;
return relativeTo ;
}
2023-09-11 00:26:14 +02:00
int X = 0 ;
int Y = 0 ;
if ( isExact ) {
X = xIsPercent ? std : : stof ( x ) * 0.01 * PMONITOR - > vecSize . x : std : : stoi ( x ) ;
Y = yIsPercent ? std : : stof ( y ) * 0.01 * PMONITOR - > vecSize . y : std : : stoi ( y ) ;
} else {
X = xIsPercent ? std : : stof ( x ) * 0.01 * relativeTo . x + relativeTo . x : std : : stoi ( x ) + relativeTo . x ;
Y = yIsPercent ? std : : stof ( y ) * 0.01 * relativeTo . y + relativeTo . y : std : : stoi ( y ) + relativeTo . y ;
}
2022-08-15 15:59:07 +02:00
2023-09-11 00:26:14 +02:00
return Vector2D ( X , Y ) ;
2022-08-22 14:22:21 +02:00
}
2024-08-26 12:25:52 +02:00
PHLWORKSPACE CCompositor : : createNewWorkspace ( const WORKSPACEID & id , const MONITORID & monid , const std : : string & name , bool isEmpty ) {
2024-12-07 18:51:18 +01:00
const auto NAME = name . empty ( ) ? std : : to_string ( id ) : name ;
2022-12-16 18:17:31 +01:00
auto monID = monid ;
2022-10-25 11:30:25 +02:00
// check if bound
2024-10-27 19:45:38 +01:00
if ( const auto PMONITOR = g_pConfigManager - > getBoundMonitorForWS ( NAME ) ; PMONITOR )
2022-10-25 11:30:25 +02:00
monID = PMONITOR - > ID ;
2022-11-27 23:42:22 +01:00
const bool SPECIAL = id > = SPECIAL_WORKSPACE_START & & id < = - 2 ;
2024-10-27 19:45:38 +01:00
const auto PWORKSPACE = m_vWorkspaces . emplace_back ( CWorkspace : : create ( id , getMonitorFromID ( monID ) , NAME , SPECIAL , isEmpty ) ) ;
2022-10-24 19:36:31 +02:00
2023-11-18 18:00:24 +01:00
PWORKSPACE - > m_fAlpha . setValueAndWarp ( 0 ) ;
2022-10-24 19:36:31 +02:00
return PWORKSPACE ;
}
2022-11-19 17:41:36 +01:00
2024-10-20 00:03:29 +02:00
void CCompositor : : setActiveMonitor ( PHLMONITOR pMonitor ) {
if ( m_pLastMonitor = = pMonitor )
2022-11-19 17:41:36 +01:00
return ;
2022-11-23 00:17:10 +01:00
if ( ! pMonitor ) {
2024-05-05 23:18:10 +02:00
m_pLastMonitor . reset ( ) ;
2022-11-23 00:17:10 +01:00
return ;
}
2024-04-02 21:32:39 +02:00
const auto PWORKSPACE = pMonitor - > activeWorkspace ;
2022-11-19 17:41:36 +01:00
2024-02-22 16:12:51 +01:00
g_pEventManager - > postEvent ( SHyprIPCEvent { " focusedmon " , pMonitor - > szName + " , " + ( PWORKSPACE ? PWORKSPACE - > m_szName : " ? " ) } ) ;
2023-02-19 21:54:53 +01:00
EMIT_HOOK_EVENT ( " focusedMon " , pMonitor ) ;
2024-05-05 23:18:10 +02:00
m_pLastMonitor = pMonitor - > self ;
2022-11-19 17:41:36 +01:00
}
2022-11-27 23:42:22 +01:00
2024-08-08 21:01:50 +02:00
bool CCompositor : : isWorkspaceSpecial ( const WORKSPACEID & id ) {
2022-11-27 23:42:22 +01:00
return id > = SPECIAL_WORKSPACE_START & & id < = - 2 ;
}
2024-08-08 21:01:50 +02:00
WORKSPACEID CCompositor : : getNewSpecialID ( ) {
WORKSPACEID highest = SPECIAL_WORKSPACE_START ;
2024-08-26 17:25:39 +02:00
for ( auto const & ws : m_vWorkspaces ) {
2022-11-27 23:42:22 +01:00
if ( ws - > m_bIsSpecialWorkspace & & ws - > m_iID > highest ) {
highest = ws - > m_iID ;
}
}
return highest + 1 ;
}
2023-03-20 16:02:47 +01:00
void CCompositor : : performUserChecks ( ) {
2024-12-06 15:45:02 +01:00
static auto PNOCHECKXDG = CConfigValue < Hyprlang : : INT > ( " misc:disable_xdg_env_checks " ) ;
static auto PNOCHECKQTUTILS = CConfigValue < Hyprlang : : INT > ( " misc:disable_hyprland_qtutils_check " ) ;
2024-09-18 12:22:07 +02:00
if ( ! * PNOCHECKXDG ) {
const auto CURRENT_DESKTOP_ENV = getenv ( " XDG_CURRENT_DESKTOP " ) ;
if ( ! CURRENT_DESKTOP_ENV | | std : : string { CURRENT_DESKTOP_ENV } ! = " Hyprland " ) {
g_pHyprNotificationOverlay - > addNotification (
std : : format ( " Your XDG_CURRENT_DESKTOP environment seems to be managed externally, and the current value is {}. \n This might cause issues unless it's intentional. " ,
CURRENT_DESKTOP_ENV ? CURRENT_DESKTOP_ENV : " unset " ) ,
2024-12-03 19:58:24 +01:00
CHyprColor { } , 15000 , ICON_WARNING ) ;
2024-09-18 12:22:07 +02:00
}
}
2024-11-01 16:52:03 +01:00
2024-12-06 15:45:02 +01:00
if ( ! * PNOCHECKQTUTILS ) {
if ( ! executableExistsInPath ( " hyprland-dialog " ) ) {
g_pHyprNotificationOverlay - > addNotification (
" Your system does not have hyprland-qtutils installed. This is a runtime dependency for some dialogs. Consider installing it. " , CHyprColor { } , 15000 , ICON_WARNING ) ;
}
}
2024-11-01 16:52:03 +01:00
if ( g_pHyprOpenGL - > failedAssetsNo > 0 ) {
g_pHyprNotificationOverlay - > addNotification ( std : : format ( " Hyprland failed to load {} essential asset{}, blame your distro's packager for doing a bad job at packaging! " ,
g_pHyprOpenGL - > failedAssetsNo , g_pHyprOpenGL - > failedAssetsNo > 1 ? " s " : " " ) ,
2024-12-03 19:58:24 +01:00
CHyprColor { 1.0 , 0.1 , 0.1 , 1.0 } , 15000 , ICON_ERROR ) ;
2024-11-01 16:52:03 +01:00
}
2023-03-20 16:02:47 +01:00
}
2023-04-14 16:03:53 +02:00
2024-04-27 13:43:12 +02:00
void CCompositor : : moveWindowToWorkspaceSafe ( PHLWINDOW pWindow , PHLWORKSPACE pWorkspace ) {
2023-04-14 16:03:53 +02:00
if ( ! pWindow | | ! pWorkspace )
return ;
2023-05-29 17:46:06 +02:00
if ( pWindow - > m_bPinned & & pWorkspace - > m_bIsSpecialWorkspace )
return ;
2024-07-31 19:55:52 +02:00
const bool FULLSCREEN = pWindow - > isFullscreen ( ) ;
const auto FULLSCREENMODE = pWindow - > m_sFullscreenState . internal ;
2023-04-14 16:03:53 +02:00
if ( FULLSCREEN )
2024-07-31 19:55:52 +02:00
setWindowFullscreenInternal ( pWindow , FSMODE_NONE ) ;
2023-04-14 16:03:53 +02:00
2024-11-22 17:01:02 +01:00
const PHLWINDOW pFirstWindowOnWorkspace = pWorkspace - > getFirstWindow ( ) ;
const int visibleWindowsOnWorkspace = pWorkspace - > getWindows ( std : : nullopt , true ) ;
2024-10-27 19:45:38 +01:00
const auto PWINDOWMONITOR = pWindow - > m_pMonitor . lock ( ) ;
2024-10-23 01:51:25 +02:00
const auto POSTOMON = pWindow - > m_vRealPosition . goal ( ) - PWINDOWMONITOR - > vecPosition ;
2024-10-27 19:45:38 +01:00
const auto PWORKSPACEMONITOR = pWorkspace - > m_pMonitor . lock ( ) ;
2024-10-23 01:51:25 +02:00
if ( ! pWindow - > m_bIsFloating )
2023-04-14 16:03:53 +02:00
g_pLayoutManager - > getCurrentLayout ( ) - > onWindowRemovedTiling ( pWindow ) ;
2024-10-23 01:51:25 +02:00
pWindow - > moveToWorkspace ( pWorkspace ) ;
2024-10-27 19:45:38 +01:00
pWindow - > m_pMonitor = pWorkspace - > m_pMonitor ;
2023-04-14 16:03:53 +02:00
2024-10-23 01:51:25 +02:00
static auto PGROUPONMOVETOWORKSPACE = CConfigValue < Hyprlang : : INT > ( " group:group_on_movetoworkspace " ) ;
if ( * PGROUPONMOVETOWORKSPACE & & visibleWindowsOnWorkspace = = 1 & & pFirstWindowOnWorkspace & & pFirstWindowOnWorkspace ! = pWindow & &
pFirstWindowOnWorkspace - > m_sGroupData . pNextWindow . lock ( ) & & pWindow - > canBeGroupedInto ( pFirstWindowOnWorkspace ) ) {
pWindow - > m_bIsFloating = pFirstWindowOnWorkspace - > m_bIsFloating ; // match the floating state. Needed to group tiled into floated and vice versa.
if ( ! pWindow - > m_sGroupData . pNextWindow . expired ( ) ) {
PHLWINDOW next = pWindow - > m_sGroupData . pNextWindow . lock ( ) ;
while ( next ! = pWindow ) {
next - > m_bIsFloating = pFirstWindowOnWorkspace - > m_bIsFloating ; // match the floating state of group members
next = next - > m_sGroupData . pNextWindow . lock ( ) ;
}
}
static auto USECURRPOS = CConfigValue < Hyprlang : : INT > ( " group:insert_after_current " ) ;
( * USECURRPOS ? pFirstWindowOnWorkspace : pFirstWindowOnWorkspace - > getGroupTail ( ) ) - > insertWindowToGroup ( pWindow ) ;
pFirstWindowOnWorkspace - > setGroupCurrent ( pWindow ) ;
pWindow - > updateWindowDecos ( ) ;
g_pLayoutManager - > getCurrentLayout ( ) - > recalculateWindow ( pWindow ) ;
if ( ! pWindow - > getDecorationByType ( DECORATION_GROUPBAR ) )
pWindow - > addWindowDeco ( std : : make_unique < CHyprGroupBarDecoration > ( pWindow ) ) ;
} else {
if ( ! pWindow - > m_bIsFloating )
g_pLayoutManager - > getCurrentLayout ( ) - > onWindowCreatedTiling ( pWindow ) ;
2023-04-14 16:03:53 +02:00
2024-10-23 01:51:25 +02:00
if ( pWindow - > m_bIsFloating )
pWindow - > m_vRealPosition = POSTOMON + PWORKSPACEMONITOR - > vecPosition ;
2023-04-14 16:03:53 +02:00
}
2024-03-06 22:33:50 +01:00
pWindow - > updateToplevel ( ) ;
pWindow - > updateDynamicRules ( ) ;
pWindow - > uncacheWindowDecos ( ) ;
2024-10-26 03:22:37 +02:00
pWindow - > updateGroupOutputs ( ) ;
2024-03-06 22:33:50 +01:00
2024-04-27 13:43:12 +02:00
if ( ! pWindow - > m_sGroupData . pNextWindow . expired ( ) ) {
PHLWINDOW next = pWindow - > m_sGroupData . pNextWindow . lock ( ) ;
2023-06-21 20:51:18 +02:00
while ( next ! = pWindow ) {
next - > updateToplevel ( ) ;
2024-04-27 13:43:12 +02:00
next = next - > m_sGroupData . pNextWindow . lock ( ) ;
2023-06-21 20:51:18 +02:00
}
}
2023-04-14 16:03:53 +02:00
if ( FULLSCREEN )
2024-07-31 19:55:52 +02:00
setWindowFullscreenInternal ( pWindow , FULLSCREENMODE ) ;
2023-12-20 23:52:18 +01:00
2024-11-22 17:01:02 +01:00
pWorkspace - > updateWindows ( ) ;
if ( pWindow - > m_pWorkspace )
pWindow - > m_pWorkspace - > updateWindows ( ) ;
2024-08-18 21:02:46 +02:00
g_pCompositor - > updateSuspendedStates ( ) ;
2023-04-14 16:03:53 +02:00
}
2023-07-04 12:05:25 +02:00
2024-04-27 13:43:12 +02:00
PHLWINDOW CCompositor : : getForceFocus ( ) {
2024-08-26 20:24:30 +02:00
for ( auto const & w : m_vWindows ) {
2024-11-22 17:01:02 +01:00
if ( ! w - > m_bIsMapped | | w - > isHidden ( ) | | ! w - > m_pWorkspace | | ! w - > m_pWorkspace - > isVisible ( ) )
2023-07-04 12:05:25 +02:00
continue ;
if ( ! w - > m_bStayFocused )
continue ;
2024-04-27 13:43:12 +02:00
return w ;
2023-07-04 12:05:25 +02:00
}
return nullptr ;
}
2023-07-13 18:05:34 +02:00
2023-08-14 14:22:06 +02:00
void CCompositor : : arrangeMonitors ( ) {
2024-10-20 00:03:29 +02:00
static auto * const PXWLFORCESCALEZERO = ( Hyprlang : : INT * const * ) g_pConfigManager - > getConfigValuePtr ( " xwayland:force_zero_scaling " ) ;
2023-08-15 19:15:37 +02:00
2024-10-20 00:03:29 +02:00
std : : vector < PHLMONITOR > toArrange ;
std : : vector < PHLMONITOR > arranged ;
2023-08-14 14:22:06 +02:00
2024-08-26 20:24:30 +02:00
for ( auto const & m : m_vMonitors )
2024-10-20 00:03:29 +02:00
toArrange . push_back ( m ) ;
2023-08-14 14:22:06 +02:00
2023-09-06 12:51:36 +02:00
Debug : : log ( LOG , " arrangeMonitors: {} to arrange " , toArrange . size ( ) ) ;
2023-08-21 19:52:30 +02:00
2023-08-14 19:27:33 +02:00
for ( auto it = toArrange . begin ( ) ; it ! = toArrange . end ( ) ; ) {
2023-08-14 14:22:06 +02:00
auto m = * it ;
2023-08-16 18:16:36 +02:00
if ( m - > activeMonitorRule . offset ! = Vector2D { - INT32_MAX , - INT32_MAX } ) {
2023-08-14 14:22:06 +02:00
// explicit.
2024-04-25 23:07:50 +02:00
Debug : : log ( LOG , " arrangeMonitors: {} explicit {:j} " , m - > szName , m - > activeMonitorRule . offset ) ;
2023-08-21 19:52:30 +02:00
2023-08-14 14:22:06 +02:00
m - > moveTo ( m - > activeMonitorRule . offset ) ;
arranged . push_back ( m ) ;
it = toArrange . erase ( it ) ;
if ( it = = toArrange . end ( ) )
break ;
2023-08-14 19:27:33 +02:00
continue ;
2023-08-14 14:22:06 +02:00
}
2023-08-14 19:27:33 +02:00
+ + it ;
2023-08-14 14:22:06 +02:00
}
2024-04-23 02:40:03 +02:00
// Variables to store the max and min values of monitors on each axis.
2024-12-13 23:31:30 +01:00
int maxXOffsetRight = 0 ;
int maxXOffsetLeft = 0 ;
int maxYOffsetUp = 0 ;
int maxYOffsetDown = 0 ;
auto recalcMaxOffsets = [ & ] ( ) {
maxXOffsetRight = 0 ;
maxXOffsetLeft = 0 ;
maxYOffsetUp = 0 ;
maxYOffsetDown = 0 ;
// Finds the max and min values of explicitely placed monitors.
for ( auto const & m : arranged ) {
if ( m - > vecPosition . x + m - > vecSize . x > maxXOffsetRight )
maxXOffsetRight = m - > vecPosition . x + m - > vecSize . x ;
if ( m - > vecPosition . x < maxXOffsetLeft )
maxXOffsetLeft = m - > vecPosition . x ;
if ( m - > vecPosition . y + m - > vecSize . y > maxYOffsetDown )
maxYOffsetDown = m - > vecPosition . y + m - > vecSize . y ;
if ( m - > vecPosition . y < maxYOffsetUp )
maxYOffsetUp = m - > vecPosition . y ;
}
} ;
2023-08-14 14:22:06 +02:00
2024-04-23 02:40:03 +02:00
// Iterates through all non-explicitly placed monitors.
2024-08-26 20:24:30 +02:00
for ( auto const & m : toArrange ) {
2024-12-13 23:31:30 +01:00
recalcMaxOffsets ( ) ;
2024-04-23 02:40:03 +02:00
// Moves the monitor to their appropriate position on the x/y axis and
// increments/decrements the corresponding max offset.
2024-04-25 23:07:50 +02:00
Vector2D newPosition = { 0 , 0 } ;
switch ( m - > activeMonitorRule . autoDir ) {
2024-12-13 23:31:30 +01:00
case eAutoDirs : : DIR_AUTO_UP : newPosition . y = maxYOffsetUp - m - > vecSize . y ; break ;
case eAutoDirs : : DIR_AUTO_DOWN : newPosition . y = maxYOffsetDown ; break ;
case eAutoDirs : : DIR_AUTO_LEFT : newPosition . x = maxXOffsetLeft - m - > vecSize . x ; break ;
2024-04-25 23:07:50 +02:00
case eAutoDirs : : DIR_AUTO_RIGHT :
2024-12-13 23:31:30 +01:00
case eAutoDirs : : DIR_AUTO_NONE : newPosition . x = maxXOffsetRight ; break ;
2024-04-25 23:07:50 +02:00
default : UNREACHABLE ( ) ;
2024-04-23 02:40:03 +02:00
}
2024-04-25 23:07:50 +02:00
Debug : : log ( LOG , " arrangeMonitors: {} auto {:j} " , m - > szName , m - > vecPosition ) ;
m - > moveTo ( newPosition ) ;
2024-12-13 23:31:30 +01:00
arranged . emplace_back ( m ) ;
2023-08-14 14:22:06 +02:00
}
2023-08-15 19:15:37 +02:00
2024-04-23 02:40:03 +02:00
// reset maxXOffsetRight (reuse)
2023-08-15 19:15:37 +02:00
// and set xwayland positions aka auto for all
2024-04-23 02:40:03 +02:00
maxXOffsetRight = 0 ;
2024-08-26 20:24:30 +02:00
for ( auto const & m : m_vMonitors ) {
2024-04-25 23:07:50 +02:00
Debug : : log ( LOG , " arrangeMonitors: {} xwayland [{}, {}] " , m - > szName , maxXOffsetRight , 0 ) ;
2024-04-23 02:40:03 +02:00
m - > vecXWaylandPosition = { maxXOffsetRight , 0 } ;
maxXOffsetRight + = ( * PXWLFORCESCALEZERO ? m - > vecTransformedSize . x : m - > vecSize . x ) ;
2023-09-20 17:47:05 +02:00
2024-03-03 19:39:20 +01:00
if ( * PXWLFORCESCALEZERO )
2023-09-20 17:47:05 +02:00
m - > xwaylandScale = m - > scale ;
else
m - > xwaylandScale = 1.f ;
2023-08-15 19:15:37 +02:00
}
2024-10-05 01:44:16 +02:00
PROTO : : xdgOutput - > updateAllOutputs ( ) ;
2023-08-14 14:22:06 +02:00
}
2023-09-24 19:04:38 +02:00
void CCompositor : : enterUnsafeState ( ) {
if ( m_bUnsafeState )
return ;
Debug : : log ( LOG , " Entering unsafe state " ) ;
2023-11-01 19:53:36 +01:00
if ( ! m_pUnsafeOutput - > m_bEnabled )
m_pUnsafeOutput - > onConnect ( false ) ;
2023-09-24 19:04:38 +02:00
2023-11-01 19:53:36 +01:00
m_bUnsafeState = true ;
2024-03-26 03:26:09 +01:00
2024-10-20 00:03:29 +02:00
setActiveMonitor ( m_pUnsafeOutput . lock ( ) ) ;
2023-09-24 19:04:38 +02:00
}
void CCompositor : : leaveUnsafeState ( ) {
if ( ! m_bUnsafeState )
return ;
Debug : : log ( LOG , " Leaving unsafe state " ) ;
m_bUnsafeState = false ;
2024-10-20 00:03:29 +02:00
PHLMONITOR pNewMonitor = nullptr ;
2024-08-26 20:24:30 +02:00
for ( auto const & pMonitor : m_vMonitors ) {
2023-11-01 19:53:36 +01:00
if ( pMonitor - > output ! = m_pUnsafeOutput - > output ) {
2024-10-20 00:03:29 +02:00
pNewMonitor = pMonitor ;
2023-11-01 19:53:36 +01:00
break ;
}
}
RASSERT ( pNewMonitor , " Tried to leave unsafe without a monitor " ) ;
if ( m_pUnsafeOutput - > m_bEnabled )
m_pUnsafeOutput - > onDisconnect ( ) ;
2023-09-24 19:04:38 +02:00
2024-08-26 20:24:30 +02:00
for ( auto const & m : m_vMonitors ) {
2024-10-20 00:03:29 +02:00
scheduleFrameForMonitor ( m ) ;
2023-11-01 19:53:36 +01:00
}
2023-09-24 19:04:38 +02:00
}
2023-10-22 17:58:06 +02:00
2024-06-08 10:07:59 +02:00
void CCompositor : : setPreferredScaleForSurface ( SP < CWLSurfaceResource > pSurface , double scale ) {
2024-04-20 15:14:54 +02:00
PROTO : : fractional - > sendScale ( pSurface , scale ) ;
2024-06-08 10:07:59 +02:00
pSurface - > sendPreferredScale ( std : : ceil ( scale ) ) ;
2024-01-11 13:15:20 +01:00
2024-06-08 10:07:59 +02:00
const auto PSURFACE = CWLSurface : : fromResource ( pSurface ) ;
2024-01-11 13:15:20 +01:00
if ( ! PSURFACE ) {
2024-07-22 18:06:11 +02:00
Debug : : log ( WARN , " Orphaned CWLSurfaceResource {:x} in setPreferredScaleForSurface " , ( uintptr_t ) pSurface . get ( ) ) ;
2024-01-11 13:15:20 +01:00
return ;
}
PSURFACE - > m_fLastScale = scale ;
PSURFACE - > m_iLastScale = static_cast < int32_t > ( std : : ceil ( scale ) ) ;
2023-10-22 17:58:06 +02:00
}
2024-06-08 10:07:59 +02:00
void CCompositor : : setPreferredTransformForSurface ( SP < CWLSurfaceResource > pSurface , wl_output_transform transform ) {
pSurface - > sendPreferredTransform ( transform ) ;
2024-01-11 13:15:20 +01:00
2024-06-08 10:07:59 +02:00
const auto PSURFACE = CWLSurface : : fromResource ( pSurface ) ;
2024-01-11 13:15:20 +01:00
if ( ! PSURFACE ) {
2024-07-22 18:06:11 +02:00
Debug : : log ( WARN , " Orphaned CWLSurfaceResource {:x} in setPreferredTransformForSurface " , ( uintptr_t ) pSurface . get ( ) ) ;
2024-01-11 13:15:20 +01:00
return ;
}
PSURFACE - > m_eLastTransform = transform ;
2023-12-13 18:25:19 +01:00
}
2023-12-23 22:30:49 +01:00
void CCompositor : : updateSuspendedStates ( ) {
2024-08-26 20:24:30 +02:00
for ( auto const & w : g_pCompositor - > m_vWindows ) {
2023-12-23 22:30:49 +01:00
if ( ! w - > m_bIsMapped )
continue ;
2024-11-22 17:01:02 +01:00
w - > setSuspended ( w - > isHidden ( ) | | ! w - > m_pWorkspace | | ! w - > m_pWorkspace - > isVisible ( ) ) ;
2023-12-23 22:30:49 +01:00
}
}
2024-04-27 13:43:12 +02:00
2024-10-26 03:06:13 +02:00
static void checkDefaultCursorWarp ( PHLMONITOR monitor ) {
2024-09-15 00:37:18 +02:00
static auto PCURSORMONITOR = CConfigValue < std : : string > ( " cursor:default_monitor " ) ;
static bool cursorDefaultDone = false ;
static bool firstLaunch = true ;
const auto POS = monitor - > middle ( ) ;
// by default, cursor should be set to first monitor detected
// this is needed as a default if the monitor given in config above doesn't exist
if ( firstLaunch ) {
firstLaunch = false ;
g_pCompositor - > warpCursorTo ( POS , true ) ;
g_pInputManager - > refocus ( ) ;
return ;
}
if ( ! cursorDefaultDone & & * PCURSORMONITOR ! = STRVAL_EMPTY ) {
if ( * PCURSORMONITOR = = monitor - > szName ) {
cursorDefaultDone = true ;
g_pCompositor - > warpCursorTo ( POS , true ) ;
g_pInputManager - > refocus ( ) ;
return ;
}
}
// modechange happend check if cursor is on that monitor and warp it to middle to not place it out of bounds if resolution changed.
2024-10-20 00:03:29 +02:00
if ( g_pCompositor - > getMonitorFromCursor ( ) = = monitor ) {
2024-09-15 00:37:18 +02:00
g_pCompositor - > warpCursorTo ( POS , true ) ;
g_pInputManager - > refocus ( ) ;
}
}
2024-07-21 13:09:54 +02:00
void CCompositor : : onNewMonitor ( SP < Aquamarine : : IOutput > output ) {
// add it to real
2024-08-19 18:44:22 +02:00
auto PNEWMONITOR = g_pCompositor - > m_vRealMonitors . emplace_back ( makeShared < CMonitor > ( output ) ) ;
2024-07-21 13:09:54 +02:00
if ( std : : string ( " HEADLESS-1 " ) = = output - > name ) {
2024-10-20 00:03:29 +02:00
g_pCompositor - > m_pUnsafeOutput = PNEWMONITOR ;
2024-07-21 13:09:54 +02:00
output - > name = " FALLBACK " ; // we are allowed to do this :)
}
Debug : : log ( LOG , " New output with name {} " , output - > name ) ;
PNEWMONITOR - > szName = output - > name ;
PNEWMONITOR - > self = PNEWMONITOR ;
const bool FALLBACK = g_pCompositor - > m_pUnsafeOutput ? output = = g_pCompositor - > m_pUnsafeOutput - > output : false ;
2024-08-08 21:01:50 +02:00
PNEWMONITOR - > ID = FALLBACK ? MONITOR_INVALID : g_pCompositor - > getNextAvailableMonitorID ( output - > name ) ;
2024-07-21 13:09:54 +02:00
PNEWMONITOR - > isUnsafeFallback = FALLBACK ;
EMIT_HOOK_EVENT ( " newMonitor " , PNEWMONITOR ) ;
if ( ! FALLBACK )
PNEWMONITOR - > onConnect ( false ) ;
if ( ! PNEWMONITOR - > m_bEnabled | | FALLBACK )
return ;
// ready to process if we have a real monitor
if ( ( ! g_pHyprRenderer - > m_pMostHzMonitor | | PNEWMONITOR - > refreshRate > g_pHyprRenderer - > m_pMostHzMonitor - > refreshRate ) & & PNEWMONITOR - > m_bEnabled )
2024-10-20 00:03:29 +02:00
g_pHyprRenderer - > m_pMostHzMonitor = PNEWMONITOR ;
2024-07-21 13:09:54 +02:00
g_pCompositor - > m_bReadyToProcess = true ;
g_pConfigManager - > m_bWantsMonitorReload = true ;
2024-10-20 00:03:29 +02:00
g_pCompositor - > scheduleFrameForMonitor ( PNEWMONITOR , IOutput : : AQ_SCHEDULE_NEW_MONITOR ) ;
2024-07-21 13:09:54 +02:00
2024-09-15 00:37:18 +02:00
checkDefaultCursorWarp ( PNEWMONITOR ) ;
2024-08-26 20:24:30 +02:00
for ( auto const & w : g_pCompositor - > m_vWindows ) {
2024-10-27 19:45:38 +01:00
if ( w - > m_pMonitor = = PNEWMONITOR ) {
2024-08-08 21:01:50 +02:00
w - > m_iLastSurfaceMonitorID = MONITOR_INVALID ;
2024-07-21 13:09:54 +02:00
w - > updateSurfaceScaleTransformDetails ( ) ;
}
}
2024-10-20 00:03:29 +02:00
g_pHyprRenderer - > damageMonitor ( PNEWMONITOR ) ;
2024-10-19 17:21:47 +02:00
PNEWMONITOR - > onMonitorFrame ( ) ;
2024-07-21 13:09:54 +02:00
}