/* BEGIN_COMMON_COPYRIGHT_HEADER * (c)LGPL2+ * * LXQt - a lightweight, Qt based, desktop toolset * http://razor-qt.org * * Copyright: 2010-2011 Razor team * Authors: * Petr Vanek <petr@scribus.info> * Copyright (c) 2016 Luís Pereira <luis.artur.pereira@gmail.com> * Copyright (c) 2012 The Chromium Authors. All rights reserved. * * This program or library is free software; you can redistribute it * and/or modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 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 * Lesser General Public License for more details. * You should have received a copy of the GNU Lesser General * Public License along with this library; if not, write to the * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301 USA * * END_COMMON_COPYRIGHT_HEADER */ #include <QProcess> #include "lxqtscreensaver.h" #include "lxqttranslator.h" #include <memory> #include <XdgIcon> #include <QMessageBox> #include <QAction> #include <QPointer> #include <QProcess> #include <QCoreApplication> // for Q_DECLARE_TR_FUNCTIONS #include <QX11Info> #include <X11/extensions/scrnsaver.h> // Avoid polluting everything with X11/Xlib.h: typedef struct _XDisplay Display; typedef unsigned long XAtom; typedef unsigned long XID; extern "C" { int XFree(void*); } template <class T, class R, R (*F)(T*)> struct XObjectDeleter { inline void operator()(void* ptr) const { F(static_cast<T*>(ptr)); } }; template <class T, class D = XObjectDeleter<void, int, XFree>> using XScopedPtr = std::unique_ptr<T, D>; namespace LXQt { static bool GetProperty(XID window, const std::string& property_name, long max_length, Atom* type, int* format, unsigned long* num_items, unsigned char** property); static bool GetIntArrayProperty(XID window, const std::string& property_name, std::vector<int>* value); static bool GetProperty(XID window, const std::string& property_name, long max_length, Atom* type, int* format, unsigned long* num_items, unsigned char** property) { Atom property_atom = XInternAtom(QX11Info::display(), property_name.c_str(), false); unsigned long remaining_bytes = 0; return XGetWindowProperty(QX11Info::display(), window, property_atom, 0, // offset into property data to read max_length, // max length to get False, // deleted AnyPropertyType, type, format, num_items, &remaining_bytes, property); } static bool GetIntArrayProperty(XID window, const std::string& property_name, std::vector<int>* value) { Atom type = None; int format = 0; // size in bits of each item in 'property' unsigned long num_items = 0; unsigned char* properties = NULL; int result = GetProperty(window, property_name, (~0L), // (all of them) &type, &format, &num_items, &properties); XScopedPtr<unsigned char> scoped_properties(properties); if (result != Success) return false; if (format != 32) return false; long* int_properties = reinterpret_cast<long*>(properties); value->clear(); for (unsigned long i = 0; i < num_items; ++i) { value->push_back(static_cast<int>(int_properties[i])); } return true; } class ScreenSaverPrivate { Q_DECLARE_TR_FUNCTIONS(LXQt::ScreenSaver); Q_DECLARE_PUBLIC(ScreenSaver) ScreenSaver* const q_ptr; public: ScreenSaverPrivate(ScreenSaver *q) : q_ptr(q) {}; void _l_xdgProcess_finished(int, QProcess::ExitStatus); bool isScreenSaverLocked(); QPointer<QProcess> m_xdgProcess; }; void ScreenSaverPrivate::_l_xdgProcess_finished(int err, QProcess::ExitStatus status) { // http://portland.freedesktop.org/xdg-utils-1.1.0-rc1/scripts/xdg-screensaver Q_Q(ScreenSaver); if (err == 0) emit q->activated(); else { QMessageBox *box = new QMessageBox; box->setAttribute(Qt::WA_DeleteOnClose); box->setIcon(QMessageBox::Warning); if (err == 1) { box->setWindowTitle(tr("Screen Saver Error")); box->setText(tr("An error occurred starting screensaver. " "Syntax error in xdg-screensaver arguments.")); } else if (err == 3) { box->setWindowTitle(tr("Screen Saver Activation Error")); box->setText(tr("An error occurred starting screensaver. " "Ensure you have xscreensaver installed and running.")); } else if (err == 4) { box->setWindowTitle(tr("Screen Saver Activation Error")); box->setText(tr("An error occurred starting screensaver. " "Action 'activate' failed. " "Ensure you have xscreensaver installed and running.")); } else { box->setWindowTitle(tr("Screen Saver Activation Error")); box->setText(tr("An error occurred starting screensaver. " "Unknown error - undocumented return value from xdg-screensaver: %1.") .arg(err)); } box->exec(); } emit q->done(); } bool ScreenSaverPrivate::isScreenSaverLocked() { XScreenSaverInfo *info = 0; Display *display = QX11Info::display(); XID window = DefaultRootWindow(display); info = XScreenSaverAllocInfo(); XScreenSaverQueryInfo(QX11Info::display(), window, info); const int state = info->state; XFree(info); if (state == ScreenSaverOn) return true; // Ironically, xscreensaver does not conform to the XScreenSaver protocol, so // info.state == ScreenSaverOff or info.state == ScreenSaverDisabled does not // necessarily mean that a screensaver is not active, so add a special check // for xscreensaver. XAtom lock_atom = XInternAtom(display, "LOCK", false); std::vector<int> atom_properties; if (GetIntArrayProperty(window, "_SCREENSAVER_STATUS", &atom_properties) && atom_properties.size() > 0) { if (atom_properties[0] == static_cast<int>(lock_atom)) return true; } return false; } ScreenSaver::ScreenSaver(QObject * parent) : QObject(parent), d_ptr(new ScreenSaverPrivate(this)) { Q_D(ScreenSaver); d->m_xdgProcess = new QProcess(this); connect(d->m_xdgProcess, SIGNAL(finished(int,QProcess::ExitStatus)), this, SLOT(_l_xdgProcess_finished(int,QProcess::ExitStatus))); } ScreenSaver::~ScreenSaver() { delete d_ptr; } QList<QAction*> ScreenSaver::availableActions() { QList<QAction*> ret; QAction * act; act = new QAction(XdgIcon::fromTheme("system-lock-screen", "lock"), tr("Lock Screen"), this); connect(act, SIGNAL(triggered()), this, SLOT(lockScreen())); ret.append(act); return ret; } void ScreenSaver::lockScreen() { Q_D(ScreenSaver); if (!d->isScreenSaverLocked()) d->m_xdgProcess->start("xdg-screensaver", QStringList() << "lock"); } } // namespace LXQt #include "moc_lxqtscreensaver.cpp"