kde-workspace/kcontrol/randr/randrdisplay.cpp
Ivailo Monev 6e3e4f8c12 kcontrol: move HAS_RANDR_1_2 configuration definition to config-X11 header
Signed-off-by: Ivailo Monev <xakepa10@gmail.com>
2022-09-22 05:40:33 +03:00

368 lines
8.4 KiB
C++

/*
* Copyright (c) 2008 Harry Bock <hbock@providence.edu>
* Copyright (c) 2007 Gustavo Pichorim Boiko <gustavo.boiko@kdemail.net>
* Copyright (c) 2002,2003 Hamish Rodda <rodda@kde.org>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
#include <config-X11.h>
#include <KLocale>
#include <KConfigGroup>
#include <KDebug>
#include <QApplication>
#include <QDesktopWidget>
#include <QtGui/qx11info_x11.h>
#include "randrdisplay.h"
#ifdef HAS_RANDR_1_2
#include "randrscreen.h"
#endif
#include "legacyrandrscreen.h"
RandRDisplay::RandRDisplay()
: m_valid(true)
{
m_dpy = QX11Info::display();
// Check extension
if(XRRQueryExtension(m_dpy, &m_eventBase, &m_errorBase) == False) {
m_valid = false;
return;
}
int major_version, minor_version;
XRRQueryVersion(m_dpy, &major_version, &minor_version);
m_version = i18n("X Resize and Rotate extension version %1.%2",
major_version,minor_version);
// check if we have the new version of the XRandR extension
RandR::has_1_2 = (major_version > 1 || (major_version == 1 && minor_version >= 2));
RandR::has_1_3 = (major_version > 1 || (major_version == 1 && minor_version >= 3));
if(RandR::has_1_3)
kDebug() << "Using XRANDR extension 1.3 or greater.";
else if(RandR::has_1_2)
kDebug() << "Using XRANDR extension 1.2.";
else kDebug() << "Using legacy XRANDR extension (1.1 or earlier).";
kDebug() << "XRANDR error base: " << m_errorBase;
m_numScreens = ScreenCount(m_dpy);
m_currentScreenIndex = 0;
// set the timestamp to 0
RandR::timestamp = 0;
// This assumption is WRONG with Xinerama
// Q_ASSERT(QApplication::desktop()->numScreens() == ScreenCount(QX11Info::display()));
for (int i = 0; i < m_numScreens; i++) {
#ifdef HAS_RANDR_1_2
if (RandR::has_1_2)
m_screens.append(new RandRScreen(i));
else
#endif
m_legacyScreens.append(new LegacyRandRScreen(i));
}
#if 0
//#ifdef HAS_RANDR_1_2
// check if we have more than one output, if no, revert to the legacy behavior
if (RandR::has_1_2)
{
int count = 0;
foreach(RandRScreen *screen, m_screens)
count += screen->outputs().count();
if (count < 2)
{
RandR::has_1_2 = false;
for (int i = 0; i < m_numScreens; ++i)
{
delete m_screens[i];
m_legacyScreens.append(new LegacyRandRScreen(i));
}
m_screens.clear();
}
}
#endif
setCurrentScreen(DefaultScreen(QX11Info::display()));
}
RandRDisplay::~RandRDisplay()
{
qDeleteAll(m_legacyScreens);
#ifdef HAS_RANDR_1_2
qDeleteAll(m_screens);
#endif
}
bool RandRDisplay::isValid() const
{
return m_valid;
}
const QString& RandRDisplay::errorCode() const
{
return m_errorCode;
}
int RandRDisplay::eventBase() const
{
return m_eventBase;
}
int RandRDisplay::errorBase() const
{
return m_errorBase;
}
const QString& RandRDisplay::version() const
{
return m_version;
}
void RandRDisplay::setCurrentScreen(int index)
{
Q_ASSERT(index < ScreenCount(m_dpy));
m_currentScreenIndex = index;
}
int RandRDisplay::screenIndexOfWidget(QWidget* widget)
{
//int ret = QApplication::desktop()->screenNumber(widget);
//return ret != -1 ? ret : QApplication::desktop()->primaryScreen();
// get info from Qt's X11 info directly; QDesktopWidget seems to use
// Xinerama by default, which doesn't work properly with randr.
// It will return more screens than exist for the display, causing
// a crash in the screen/currentScreen methods.
if(widget)
return widget->x11Info().screen();
return -1;
}
int RandRDisplay::currentScreenIndex() const
{
return m_currentScreenIndex;
}
bool RandRDisplay::needsRefresh() const
{
Time time, config_timestamp;
time = XRRTimes(m_dpy, m_currentScreenIndex, &config_timestamp);
kDebug() << "Cache:" << RandR::timestamp << "Server:" << time << "Config:" << config_timestamp;
return (RandR::timestamp < time);
}
void RandRDisplay::refresh()
{
#ifdef HAS_RANDR_1_2
if (RandR::has_1_2) {
for (int i = 0; i < m_screens.count(); ++i) {
RandRScreen* s = m_screens.at(i);
s->loadSettings();
}
}
else
#endif
{
for (int i = 0; i < m_legacyScreens.size(); ++i) {
LegacyRandRScreen* s = m_legacyScreens.at(i);
s->loadSettings();
}
}
}
bool RandRDisplay::canHandle(const XEvent *e) const
{
if (e->type == m_eventBase + RRScreenChangeNotify)
return true;
#ifdef HAS_RANDR_1_2
else if (e->type == m_eventBase + RRNotify)
return true;
#endif
return false;
}
void RandRDisplay::handleEvent(XEvent *e)
{
if (e->type == m_eventBase + RRScreenChangeNotify) {
#ifdef HAS_RANDR_1_2
if (RandR::has_1_2) {
XRRScreenChangeNotifyEvent *event = (XRRScreenChangeNotifyEvent*)(e);
for (int i=0; i < m_screens.count(); ++i) {
RandRScreen *screen = m_screens.at(i);
if (screen->rootWindow() == event->root)
screen->handleEvent(event);
}
}
else
#endif
{
// handle the event
}
}
#ifdef HAS_RANDR_1_2
else if (e->type == m_eventBase + RRNotify) {
//forward the event to the right screen
XRRNotifyEvent *event = (XRRNotifyEvent*)e;
for (int i=0; i < m_screens.count(); ++i) {
RandRScreen *screen = m_screens.at(i);
if ( screen->rootWindow() == event->window ) {
screen->handleRandREvent(event);
}
}
}
#endif
}
int RandRDisplay::numScreens() const
{
Q_ASSERT(ScreenCount(QX11Info::display()) == m_numScreens);
return m_numScreens;
}
LegacyRandRScreen* RandRDisplay::legacyScreen(int index)
{
return m_legacyScreens.at(index);
}
LegacyRandRScreen* RandRDisplay::currentLegacyScreen()
{
return m_legacyScreens.at(m_currentScreenIndex);
}
#ifdef HAS_RANDR_1_2
RandRScreen* RandRDisplay::screen(int index)
{
return m_screens.at(index);
}
RandRScreen* RandRDisplay::currentScreen()
{
return m_screens.at(m_currentScreenIndex);
}
#endif
bool RandRDisplay::loadDisplay(KConfig& config, bool loadScreens)
{
if (loadScreens)
{
#ifdef HAS_RANDR_1_2
if (RandR::has_1_2)
{
foreach(RandRScreen *s, m_screens)
s->load(config);
}
else
#endif
{
foreach(LegacyRandRScreen* s, m_legacyScreens)
s->load(config);
}
}
return applyOnStartup(config);
}
bool RandRDisplay::applyOnStartup(KConfig& config)
{
return config.group("Display").readEntry("ApplyOnStartup", false);
}
bool RandRDisplay::syncTrayApp(KConfig& config)
{
return config.group("Display").readEntry("SyncTrayApp", false);
}
void RandRDisplay::saveDisplay(KConfig& config, bool syncTrayApp)
{
KConfigGroup group = config.group("Display");
group.writeEntry("SyncTrayApp", syncTrayApp);
#ifdef HAS_RANDR_1_2
if (RandR::has_1_2)
{
foreach(RandRScreen *s, m_screens)
s->save(config);
}
else
#endif
{
foreach(LegacyRandRScreen *s, m_legacyScreens)
s->save(config);
}
}
// to be used during desktop startup, make all screens provide the shell commands
// (using xrandr cli tool), save them here and a script will perform these commands
// early during desktop startup
void RandRDisplay::saveStartup(KConfig& config)
{
KConfigGroup group = config.group("Display");
group.writeEntry("ApplyOnStartup", true);
QStringList commands;
#ifdef HAS_RANDR_1_2
if (RandR::has_1_2)
{
foreach(RandRScreen *s, m_screens)
commands += s->startupCommands();
}
else
#endif
{
foreach(LegacyRandRScreen *s, m_legacyScreens)
commands += s->startupCommands();
}
group.writeEntry( "StartupCommands", commands.join( "\n" ));
}
void RandRDisplay::disableStartup(KConfig& config)
{
KConfigGroup group = config.group("Display");
group.writeEntry("ApplyOnStartup", false);
group.deleteEntry( "StartupCommands" );
}
void RandRDisplay::applyProposed(bool confirm)
{
#ifdef HAS_RANDR_1_2
if (RandR::has_1_2)
foreach(RandRScreen *s, m_screens)
s->applyProposed(confirm);
else
#endif
{
foreach(LegacyRandRScreen *s, m_legacyScreens)
{
if (s->proposedChanged()) {
if (confirm)
s->applyProposedAndConfirm();
else
s->applyProposed();
}
}
}
}
// vim:noet:sts=8:sw=8: