You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
266 lines
7.9 KiB
266 lines
7.9 KiB
/* This file is part of the KDE libraries |
|
Copyright (C) 1997 Matthias Kalle Dalheimer (kalle@kde.org) |
|
|
|
This library is free software; you can redistribute it and/or |
|
modify it under the terms of the GNU Library General Public |
|
License as published by the Free Software Foundation; either |
|
version 2 of the License, or (at your option) any later version. |
|
|
|
This library is distributed in the hope that it will be useful, |
|
but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
|
Library General Public License for more details. |
|
|
|
You should have received a copy of the GNU Library General Public License |
|
along with this library; see the file COPYING.LIB. If not, write to |
|
the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, |
|
Boston, MA 02110-1301, USA. |
|
*/ |
|
|
|
#include "kworkspace.h" |
|
#include "config-libkworkspace.h" |
|
|
|
#include <QApplication> |
|
#include <QDataStream> |
|
#include <QFile> |
|
#include <QFileInfo> |
|
#include <QTextStream> |
|
#include <QDateTime> |
|
#include <QtDBus/QtDBus> |
|
#include <stdlib.h> // getenv() |
|
#include <ksmserver_interface.h> |
|
#include <QSocketNotifier> |
|
|
|
#if HAVE_X11 |
|
#include <X11/Xlib.h> |
|
#include <X11/Xutil.h> |
|
#include <X11/Xatom.h> |
|
#include <X11/SM/SMlib.h> |
|
#include <fixx11h.h> |
|
#endif |
|
|
|
#if HAVE_X11 |
|
#define DISPLAY "DISPLAY" |
|
#elif defined(Q_WS_QWS) |
|
#define DISPLAY "QWS_DISPLAY" |
|
#endif |
|
|
|
#include "config-workspace.h" |
|
|
|
#ifdef HAVE_UNISTD_H |
|
#include <unistd.h> |
|
#endif // HAVE_UNISTD_H |
|
|
|
#include <pwd.h> |
|
#include <sys/types.h> |
|
|
|
#include "kworkspace_p.h" |
|
|
|
namespace KWorkSpace |
|
{ |
|
#if HAVE_X11 |
|
static void save_yourself_callback( SmcConn conn_P, SmPointer, int, Bool , int, Bool ) |
|
{ |
|
SmcSaveYourselfDone( conn_P, True ); |
|
} |
|
|
|
static void dummy_callback( SmcConn, SmPointer ) |
|
{ |
|
} |
|
#endif |
|
KRequestShutdownHelper::KRequestShutdownHelper() |
|
{ |
|
#if HAVE_X11 |
|
SmcCallbacks calls; |
|
calls.save_yourself.callback = save_yourself_callback; |
|
calls.die.callback = dummy_callback; |
|
calls.save_complete.callback = dummy_callback; |
|
calls.shutdown_cancelled.callback = dummy_callback; |
|
char* id = NULL; |
|
char err[ 11 ]; |
|
conn = SmcOpenConnection( NULL, NULL, 1, 0, |
|
SmcSaveYourselfProcMask | SmcDieProcMask | SmcSaveCompleteProcMask |
|
| SmcShutdownCancelledProcMask, &calls, NULL, &id, 10, err ); |
|
if( id != NULL ) |
|
free( id ); |
|
if( conn == NULL ) |
|
return; // no SM |
|
// set the required properties, mostly dummy values |
|
SmPropValue propvalue[ 5 ]; |
|
SmProp props[ 5 ]; |
|
propvalue[ 0 ].length = sizeof( unsigned char ); |
|
unsigned char value0 = SmRestartNever; // so that this extra SM connection doesn't interfere |
|
propvalue[ 0 ].value = &value0; |
|
props[ 0 ].name = const_cast< char* >( SmRestartStyleHint ); |
|
props[ 0 ].type = const_cast< char* >( SmCARD8 ); |
|
props[ 0 ].num_vals = 1; |
|
props[ 0 ].vals = &propvalue[ 0 ]; |
|
struct passwd* entry = getpwuid( geteuid() ); |
|
propvalue[ 1 ].length = entry != NULL ? strlen( entry->pw_name ) : 0; |
|
propvalue[ 1 ].value = (SmPointer)( entry != NULL ? entry->pw_name : "" ); |
|
props[ 1 ].name = const_cast< char* >( SmUserID ); |
|
props[ 1 ].type = const_cast< char* >( SmARRAY8 ); |
|
props[ 1 ].num_vals = 1; |
|
props[ 1 ].vals = &propvalue[ 1 ]; |
|
propvalue[ 2 ].length = 0; |
|
propvalue[ 2 ].value = (SmPointer)( "" ); |
|
props[ 2 ].name = const_cast< char* >( SmRestartCommand ); |
|
props[ 2 ].type = const_cast< char* >( SmLISTofARRAY8 ); |
|
props[ 2 ].num_vals = 1; |
|
props[ 2 ].vals = &propvalue[ 2 ]; |
|
propvalue[ 3 ].length = strlen( "requestshutdownhelper" ); |
|
propvalue[ 3 ].value = (SmPointer)"requestshutdownhelper"; |
|
props[ 3 ].name = const_cast< char* >( SmProgram ); |
|
props[ 3 ].type = const_cast< char* >( SmARRAY8 ); |
|
props[ 3 ].num_vals = 1; |
|
props[ 3 ].vals = &propvalue[ 3 ]; |
|
propvalue[ 4 ].length = 0; |
|
propvalue[ 4 ].value = (SmPointer)( "" ); |
|
props[ 4 ].name = const_cast< char* >( SmCloneCommand ); |
|
props[ 4 ].type = const_cast< char* >( SmLISTofARRAY8 ); |
|
props[ 4 ].num_vals = 1; |
|
props[ 4 ].vals = &propvalue[ 4 ]; |
|
SmProp* p[ 5 ] = { &props[ 0 ], &props[ 1 ], &props[ 2 ], &props[ 3 ], &props[ 4 ] }; |
|
SmcSetProperties( conn, 5, p ); |
|
notifier = new QSocketNotifier( IceConnectionNumber( SmcGetIceConnection( conn )), |
|
QSocketNotifier::Read, this ); |
|
connect( notifier, SIGNAL(activated(int)), SLOT(processData())); |
|
#endif |
|
} |
|
|
|
KRequestShutdownHelper::~KRequestShutdownHelper() |
|
{ |
|
#if HAVE_X11 |
|
if( conn != NULL ) |
|
{ |
|
delete notifier; |
|
SmcCloseConnection( conn, 0, NULL ); |
|
} |
|
#endif |
|
} |
|
|
|
void KRequestShutdownHelper::processData() |
|
{ |
|
#if HAVE_X11 |
|
if( conn != NULL ) |
|
IceProcessMessages( SmcGetIceConnection( conn ), 0, 0 ); |
|
#endif |
|
} |
|
|
|
bool KRequestShutdownHelper::requestShutdown( ShutdownConfirm confirm ) |
|
{ |
|
#if HAVE_X11 |
|
if( conn == NULL ) |
|
return false; |
|
SmcRequestSaveYourself( conn, SmSaveBoth, True, SmInteractStyleAny, |
|
confirm == ShutdownConfirmNo, True ); |
|
// flush the request |
|
IceFlush(SmcGetIceConnection(conn)); |
|
#endif |
|
return true; |
|
} |
|
#if HAVE_X11 |
|
static KRequestShutdownHelper* helper = NULL; |
|
|
|
static void cleanup_sm() |
|
{ |
|
delete helper; |
|
} |
|
#endif |
|
|
|
void requestShutDown(ShutdownConfirm confirm, ShutdownType sdtype, ShutdownMode sdmode) |
|
{ |
|
#if HAVE_X11 |
|
/* use ksmserver's dcop interface if necessary */ |
|
if ( confirm == ShutdownConfirmYes || |
|
sdtype != ShutdownTypeDefault || |
|
sdmode != ShutdownModeDefault ) |
|
{ |
|
org::kde::KSMServerInterface ksmserver("org.kde.ksmserver", "/KSMServer", QDBusConnection::sessionBus()); |
|
ksmserver.logout((int)confirm, (int)sdtype, (int)sdmode); |
|
return; |
|
} |
|
|
|
if( helper == NULL ) |
|
{ |
|
helper = new KRequestShutdownHelper(); |
|
qAddPostRoutine(cleanup_sm); |
|
} |
|
helper->requestShutdown( confirm ); |
|
#endif |
|
} |
|
|
|
bool canShutDown( ShutdownConfirm confirm, |
|
ShutdownType sdtype, |
|
ShutdownMode sdmode ) |
|
{ |
|
#if HAVE_X11 |
|
if ( confirm == ShutdownConfirmYes || |
|
sdtype != ShutdownTypeDefault || |
|
sdmode != ShutdownModeDefault ) |
|
{ |
|
org::kde::KSMServerInterface ksmserver("org.kde.ksmserver", "/KSMServer", QDBusConnection::sessionBus()); |
|
QDBusReply<bool> reply = ksmserver.canShutdown(); |
|
if (!reply.isValid()) { |
|
return false; |
|
} |
|
return reply; |
|
} |
|
|
|
return true; |
|
#else |
|
return false; |
|
#endif |
|
} |
|
|
|
bool isShuttingDown() |
|
{ |
|
org::kde::KSMServerInterface ksmserver("org.kde.ksmserver", "/KSMServer", QDBusConnection::sessionBus()); |
|
QDBusReply<bool> reply = ksmserver.isShuttingDown(); |
|
if (!reply.isValid()) { |
|
return false; |
|
} |
|
return reply; |
|
} |
|
|
|
static QTime smModificationTime; |
|
void propagateSessionManager() |
|
{ |
|
#if HAVE_X11 |
|
QByteArray fName = QFile::encodeName(QStandardPaths::writableLocation(QStandardPaths::RuntimeLocation)+"/KSMserver"); |
|
QString display = QString::fromLocal8Bit( ::getenv(DISPLAY) ); |
|
// strip the screen number from the display |
|
display.remove(QRegExp("\\.[0-9]+$")); |
|
int i; |
|
while( (i = display.indexOf(':')) >= 0) |
|
display[i] = '_'; |
|
while( (i = display.indexOf('/')) >= 0) |
|
display[i] = '_'; |
|
|
|
fName += '_'; |
|
fName += display.toLocal8Bit(); |
|
QByteArray smEnv = ::getenv("SESSION_MANAGER"); |
|
bool check = smEnv.isEmpty(); |
|
if ( !check && smModificationTime.isValid() ) { |
|
QFileInfo info( fName ); |
|
QTime current = info.lastModified().time(); |
|
check = current > smModificationTime; |
|
} |
|
if ( check ) { |
|
QFile f( fName ); |
|
if ( !f.open( QIODevice::ReadOnly ) ) |
|
return; |
|
QFileInfo info ( f ); |
|
smModificationTime = QTime( info.lastModified().time() ); |
|
QTextStream t(&f); |
|
t.setCodec( "ISO 8859-1" ); |
|
QString s = t.readLine(); |
|
f.close(); |
|
::setenv( "SESSION_MANAGER", s.toLatin1(), true ); |
|
} |
|
#endif |
|
} |
|
|
|
} // end namespace |
|
|
|
|
|
|