diff options
| author | tpearson <tpearson@283d02a7-25f6-0310-bc7c-ecb5cbfe19da> | 2011-08-08 05:02:16 +0000 | 
|---|---|---|
| committer | tpearson <tpearson@283d02a7-25f6-0310-bc7c-ecb5cbfe19da> | 2011-08-08 05:02:16 +0000 | 
| commit | 486c5bfc26cbada0d8e05371c01211de663a209d (patch) | |
| tree | 9b3a01eb1cc04f699e239bb06a76286f2acaae58 /kcontrol | |
| parent | 35cf72fe7f736a8c2768f3c8035f0d318c55da0a (diff) | |
| download | tdebase-486c5bfc.tar.gz tdebase-486c5bfc.zip | |
Added the foundation for the new displayconfig module that will allow full XRandR 1.2 functionality from a Trinity control center GUI
git-svn-id: svn://anonsvn.kde.org/home/kde/branches/trinity/kdebase@1245566 283d02a7-25f6-0310-bc7c-ecb5cbfe19da
Diffstat (limited to 'kcontrol')
| -rw-r--r-- | kcontrol/CMakeLists.txt | 1 | ||||
| -rw-r--r-- | kcontrol/displayconfig/CMakeLists.txt | 39 | ||||
| -rw-r--r-- | kcontrol/displayconfig/Makefile.am | 17 | ||||
| -rw-r--r-- | kcontrol/displayconfig/NOTES | 7 | ||||
| -rw-r--r-- | kcontrol/displayconfig/displayconfig.cpp | 1359 | ||||
| -rw-r--r-- | kcontrol/displayconfig/displayconfig.desktop | 23 | ||||
| -rw-r--r-- | kcontrol/displayconfig/displayconfig.h | 144 | ||||
| -rw-r--r-- | kcontrol/displayconfig/displayconfigbase.ui | 299 | ||||
| -rw-r--r-- | kcontrol/displayconfig/monitorworkspace.cpp | 151 | ||||
| -rw-r--r-- | kcontrol/displayconfig/monitorworkspace.h | 96 | 
10 files changed, 2136 insertions, 0 deletions
| diff --git a/kcontrol/CMakeLists.txt b/kcontrol/CMakeLists.txt index 0bd7f3706..dab1a20da 100644 --- a/kcontrol/CMakeLists.txt +++ b/kcontrol/CMakeLists.txt @@ -65,6 +65,7 @@ if( BUILD_KCONTROL )    add_subdirectory( kthememanager )    add_subdirectory( kfontinst )    add_subdirectory( access ) +  add_subdirectory( displayconfig )    tde_conditional_add_subdirectory( WITH_SAMBA samba )    tde_conditional_add_subdirectory( WITH_XRANDR randr ) diff --git a/kcontrol/displayconfig/CMakeLists.txt b/kcontrol/displayconfig/CMakeLists.txt new file mode 100644 index 000000000..a96b60197 --- /dev/null +++ b/kcontrol/displayconfig/CMakeLists.txt @@ -0,0 +1,39 @@ +################################################# +# +#  (C) 2010-2011 Serghei Amelian +#  serghei (DOT) amelian (AT) gmail.com +# +#  Improvements and feedback are welcome +# +#  This file is released under GPL >= 2 +# +################################################# + +include_directories( +  ${CMAKE_CURRENT_BINARY_DIR} +  ${CMAKE_BINARY_DIR} +  ${CMAKE_CURRENT_SOURCE_DIR} +  ${TDE_INCLUDE_DIR} +  ${TQT_INCLUDE_DIRS} +) + +link_directories( +  ${TQT_LIBRARY_DIRS} +) + + +##### other data ################################ + +install( FILES displayconfig.desktop DESTINATION ${XDG_APPS_INSTALL_DIR} ) + + +##### kcm_displayconfig (module) #################### + +set_source_files_properties( displayconfig.cpp PROPERTIES COMPILE_FLAGS -DKDE_CONFDIR=\\"${TDE_CONFIG_DIR}\\" ) + +tde_add_kpart( kcm_displayconfig AUTOMOC +  SOURCES +    monitorworkspace.cpp displayconfig.cpp displayconfigbase.ui displayconfig.skel +  LINK kio-shared krandr-shared kutils-shared +  DESTINATION ${PLUGIN_INSTALL_DIR} +) diff --git a/kcontrol/displayconfig/Makefile.am b/kcontrol/displayconfig/Makefile.am new file mode 100644 index 000000000..7fcfe17c1 --- /dev/null +++ b/kcontrol/displayconfig/Makefile.am @@ -0,0 +1,17 @@ +AM_CPPFLAGS  = $(all_includes) +kde_module_LTLIBRARIES = kcm_displayconfig.la + +kcm_displayconfig_la_SOURCES = displayconfig.cpp displayconfigbase.ui displayconfig.skel + +kcm_displayconfig_la_LDFLAGS  = $(all_libraries) -lkrandr -module -avoid-version -no-undefined + +kcm_displayconfig_la_LIBADD = -lkdeui $(LIB_KIO) $(LIB_XRANDR) + +METASOURCES = AUTO + +noinst_HEADERS = displayconfig.h + +messages: rc.cpp +	$(XGETTEXT) *.cpp -o $(podir)/kcmdisplayconfig.pot + +xdg_apps_DATA = displayconfig.desktop diff --git a/kcontrol/displayconfig/NOTES b/kcontrol/displayconfig/NOTES new file mode 100644 index 000000000..4328797cc --- /dev/null +++ b/kcontrol/displayconfig/NOTES @@ -0,0 +1,7 @@ +**************** THIS IS NOT YET COMPLETE **************** +AS SUCH IT SHOULD REMAIN DISABLED UNTIL IT IS COMPLETE +********************************************************** + +NOTES: +1.) This loads the monitor settings quite well and even lets the user play with them, but it cannot save. or even apply the settings to hardware yet. +2.) Profile support should probably be added.  Imagine having multiple display profiles--a portable device could plug into a docking station and be set up in seconds!  The same idea applies for presentations, etc.
\ No newline at end of file diff --git a/kcontrol/displayconfig/displayconfig.cpp b/kcontrol/displayconfig/displayconfig.cpp new file mode 100644 index 000000000..85b5afde0 --- /dev/null +++ b/kcontrol/displayconfig/displayconfig.cpp @@ -0,0 +1,1359 @@ +/** + * displayconfig.cpp + * + * Copyright (c) 2009-2010 Timothy Pearson <kb9vqf@pearsoncomputing.net> + * + *  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 <tqcheckbox.h> +#include <tqlabel.h> +#include <tqlayout.h> +#include <tqlineedit.h> +#include <tqpushbutton.h> +#include <tqtabwidget.h> + +#include <dcopclient.h> + +#include <kaboutdata.h> +#include <kapplication.h> +#include <kconfig.h> +#include <kcombobox.h> +#include <kdebug.h> +#include <kdialog.h> +#include <kglobal.h> +#include <klistview.h> +#include <klocale.h> +#include <kmessagebox.h> +#include <kpopupmenu.h> +#include <kinputdialog.h> +#include <kurlrequester.h> +#include <kcmoduleloader.h> +#include <kgenericfactory.h> + +#include <unistd.h> +#include <ksimpleconfig.h> +#include <string> +#include <stdio.h> +#include <tqstring.h> + +#include <math.h> +#define PI 3.1415926535897932384626433832795028841971693993751058209749445923078164062862089986280348253421170679 + +#include "displayconfig.h" + +using namespace std; + +/**** DLL Interface ****/ +typedef KGenericFactory<KDisplayConfig, TQWidget> KDisplayCFactory; +K_EXPORT_COMPONENT_FACTORY( kcm_displayconfig, KDisplayCFactory("kcmdisplayconfig") ) + +KSimpleConfig *systemconfig; + + +// This routine is courtsey of an answer on "Stack Overflow" +// It takes an LSB-first int and makes it an MSB-first int (or vice versa) +unsigned int reverse_bits(register unsigned int x) +{ +    x = (((x & 0xaaaaaaaa) >> 1) | ((x & 0x55555555) << 1)); +    x = (((x & 0xcccccccc) >> 2) | ((x & 0x33333333) << 2)); +    x = (((x & 0xf0f0f0f0) >> 4) | ((x & 0x0f0f0f0f) << 4)); +    x = (((x & 0xff00ff00) >> 8) | ((x & 0x00ff00ff) << 8)); +    return((x >> 16) | (x << 16)); +} + +TQString capitalizeString(TQString in) { +	return in.left(1).upper() + in.right(in.length()-1); +} + +TQPoint moveTQRectOutsideTQRect(TQRect base, TQRect movable, int fallback_level = 0) { +	TQPoint final_result; + +	double base_center_x = base.x() + (base.width()/2); +	double base_center_y = base.y() + (base.height()/2); +	double movable_center_x = movable.x() + (movable.width()/2); +	double movable_center_y = movable.y() + (movable.height()/2); + +	double max_x_movement = (base.width()/2) + (movable.width()/2); +	double max_y_movement = (base.height()/2) + (movable.height()/2); + +	double x_diff = abs(base_center_x-movable_center_x); +	double y_diff = abs(base_center_y-movable_center_y); + +	int invert_movement; + +	// Calculate the angles of the four corners of the base rectangle +	double angle_1 = atan2((base.height()/2),	(base.width()/2)); +	double angle_2 = atan2((base.height()/2),	(base.width()/2)*(-1)); +	double angle_3 = atan2((base.height()/2)*(-1),	(base.width()/2)*(-1)); +	double angle_4 = atan2((base.height()/2)*(-1),	(base.width()/2)); + +	// Calculate the angle that the movable rectangle center is on +	double movable_angle = atan2(base_center_y-movable_center_y, movable_center_x-base_center_x); + +	// Fix up coordinates +	if (angle_1 < 0) angle_1 = angle_1 + (2*PI); +	if (angle_2 < 0) angle_2 = angle_2 + (2*PI); +	if (angle_3 < 0) angle_3 = angle_3 + (2*PI); +	if (angle_4 < 0) angle_4 = angle_4 + (2*PI); +	if (movable_angle < 0) movable_angle = movable_angle + (2*PI); + +	// Now calculate quadrant +	int quadrant; +	if ((movable_angle < angle_2) && (movable_angle >= angle_1)) { +		quadrant = 2; +	} +	else if ((movable_angle < angle_3) && (movable_angle >= angle_2)) { +		quadrant = 3; +	} +	else if ((movable_angle < angle_4) && (movable_angle >= angle_3)) { +		quadrant = 4; +	} +	else { +		quadrant = 1; +	} + +	if (fallback_level == 0) { +		if ((quadrant == 2) || (quadrant == 4)) { +			// Move it in the Y direction +			if (movable_center_y < base_center_y) +				invert_movement = -1; +			else +				invert_movement = 1; +			final_result = TQPoint(0, (max_y_movement-y_diff)*invert_movement); +		} +		else { +			// Move it in the X direction +			if (movable_center_x < base_center_x) +				invert_movement = -1; +			else +				invert_movement = 1; +			final_result = TQPoint((max_x_movement-x_diff)*invert_movement, 0); +		} +	} +	if (fallback_level == 1) { +		if ((quadrant == 1) || (quadrant == 3)) { +			// Move it in the Y direction +			if (movable_center_y < base_center_y) +				invert_movement = -1; +			else +				invert_movement = 1; +			final_result = TQPoint(0, (max_y_movement-y_diff)*invert_movement); +		} +		else { +			// Move it in the X direction +			if (movable_center_x < base_center_x) +				invert_movement = -1; +			else +				invert_movement = 1; +			final_result = TQPoint((max_x_movement-x_diff)*invert_movement, 0); +		} +	} +	if (fallback_level == 2) { +		// Ooh, nasty, I need to move the rectangle the other (suboptimal) direction +		if ((quadrant == 2) || (quadrant == 4)) { +			// Move it in the Y direction +			if (movable_center_y >= base_center_y) +				invert_movement = -1; +			else +				invert_movement = 1; +			final_result = TQPoint(0, (max_y_movement+y_diff)*invert_movement); +		} +		else { +			// Move it in the X direction +			if (movable_center_x >= base_center_x) +				invert_movement = -1; +			else +				invert_movement = 1; +			final_result = TQPoint((max_x_movement+x_diff)*invert_movement, 0); +		} +	} +	if (fallback_level == 3) { +		// Ooh, nasty, I need to move the rectangle the other (suboptimal) direction +		if ((quadrant == 1) || (quadrant == 3)) { +			// Move it in the Y direction +			if (movable_center_y >= base_center_y) +				invert_movement = -1; +			else +				invert_movement = 1; +			final_result = TQPoint(0, (max_y_movement+y_diff)*invert_movement); +		} +		else { +			// Move it in the X direction +			if (movable_center_x >= base_center_x) +				invert_movement = -1; +			else +				invert_movement = 1; +			final_result = TQPoint((max_x_movement+x_diff)*invert_movement, 0); +		} +	} + +	// Check for intersection +	TQRect test_rect = movable; +	test_rect.moveBy(final_result.x(), final_result.y()); +	if (test_rect.intersects(base)) { +		if (final_result.x() > 0) +			final_result.setX(final_result.x()+1); +		if (final_result.x() < 0) +			final_result.setX(final_result.x()-1); +		if (final_result.y() > 0) +			final_result.setY(final_result.y()+1); +		if (final_result.y() < 0) +			final_result.setY(final_result.y()-1); +	} + +	return final_result; +} + +TQPoint moveTQRectSoThatItTouchesTQRect(TQRect base, TQRect movable, int fallback_level = 0) { +	TQPoint final_result; + +	double base_center_x = base.x() + (base.width()/2); +	double base_center_y = base.y() + (base.height()/2); +	double movable_center_x = movable.x() + (movable.width()/2); +	double movable_center_y = movable.y() + (movable.height()/2); + +	double max_x_movement = (base.width()/2) + (movable.width()/2); +	double max_y_movement = (base.height()/2) + (movable.height()/2); + +	double x_diff = abs(base_center_x-movable_center_x); +	double y_diff = abs(base_center_y-movable_center_y); + +	int invert_movement; + +	// Calculate the angles of the four corners of the base rectangle +	double angle_1 = atan2((base.height()/2),	(base.width()/2)); +	double angle_2 = atan2((base.height()/2),	(base.width()/2)*(-1)); +	double angle_3 = atan2((base.height()/2)*(-1),	(base.width()/2)*(-1)); +	double angle_4 = atan2((base.height()/2)*(-1),	(base.width()/2)); + +	// Calculate the angle that the movable rectangle center is on +	double movable_angle = atan2(base_center_y-movable_center_y, movable_center_x-base_center_x); + +	// Fix up coordinates +	if (angle_1 < 0) angle_1 = angle_1 + (2*PI); +	if (angle_2 < 0) angle_2 = angle_2 + (2*PI); +	if (angle_3 < 0) angle_3 = angle_3 + (2*PI); +	if (angle_4 < 0) angle_4 = angle_4 + (2*PI); +	if (movable_angle < 0) movable_angle = movable_angle + (2*PI); + +	// Now calculate quadrant +	int quadrant; +	if ((movable_angle < angle_2) && (movable_angle >= angle_1)) { +		quadrant = 2; +	} +	else if ((movable_angle < angle_3) && (movable_angle >= angle_2)) { +		quadrant = 3; +	} +	else if ((movable_angle < angle_4) && (movable_angle >= angle_3)) { +		quadrant = 4; +	} +	else { +		quadrant = 1; +	} + +	if (fallback_level == 0) { +		if ((quadrant == 2) || (quadrant == 4)) { +			// Move it in the Y direction +			if (movable_center_y < base_center_y) +				invert_movement = -1; +			else +				invert_movement = 1; +			final_result = TQPoint(0, (max_y_movement-y_diff)*invert_movement); +		} +		else { +			// Move it in the X direction +			if (movable_center_x < base_center_x) +				invert_movement = -1; +			else +				invert_movement = 1; +			final_result = TQPoint((max_x_movement-x_diff)*invert_movement, 0); +		} +	} + +	// Check for intersection +	TQRect test_rect = movable; +	test_rect.moveBy(final_result.x(), final_result.y()); +	if (test_rect.intersects(base)) { +		if (final_result.x() > 0) +			final_result.setX(final_result.x()-1); +		if (final_result.x() < 0) +			final_result.setX(final_result.x()+1); +		if (final_result.y() > 0) +			final_result.setY(final_result.y()-1); +		if (final_result.y() < 0) +			final_result.setY(final_result.y()+1); +	} + +	return final_result; +} + +TQPoint moveTQRectOutsideMonitorRegion(TQRect rect, MonitorRegion region) { +	// This is a fun little class (not!) +	// It needs to move the TQRect so that it does not overlap any rectangles in the region +	long rect_center_x = rect.x() + (rect.width()/2); +	long rect_center_y = rect.y() + (rect.height()/2); + +	// First, see if the rectangle actually overlaps the region +	if (!region.contains(rect)) +		return TQPoint(0,0); + +	// Then, break the region into the series of source rectangles +	TQMemArray<TQRect> rectangles = region.rects(); + +	// Next, find which rectangle is closest to the center of the TQRect +	int closest = 0; +	long distance = 16384*16384; +	int fallback_mode; +	long test_distance; +	long test_center_x; +	long test_center_y; +	for ( int i = 0; i < rectangles.size(); i++ ) { +		test_center_x = rectangles[i].x() + (rectangles[i].width()/2); +		test_center_y = rectangles[i].y() + (rectangles[i].height()/2); +		test_distance = pow(test_center_x-rect_center_x,2) + pow(test_center_y-rect_center_y,2); +		if (test_distance < distance) { +			// Make sure this is an outer rectangle; i,e. there is empty space where +			// we would want to move the TQRect... +			// To do that we will move the TQRect in all four possible directions, +			// and see if any put the TQRect in an empty location +			// If they do, then this current rectangle is considered usable +			// and it is added to the distance checking routine. +			TQPoint test_moveby = moveTQRectOutsideTQRect(rectangles[i], rect, 0); +			TQRect test_rect = rect; +			test_rect.moveBy(test_moveby.x(), test_moveby.y()); +			if (!region.contains(test_rect)) { +				closest = i; +				distance = test_distance; +				fallback_mode = 0; +			} +			else { +				test_moveby = moveTQRectOutsideTQRect(rectangles[i], rect, 1); +				test_rect = rect; +				test_rect.moveBy(test_moveby.x(), test_moveby.y()); +				if (!region.contains(test_rect)) { +					closest = i; +					distance = test_distance; +					fallback_mode = 1; +				} +				else { +					test_moveby = moveTQRectOutsideTQRect(rectangles[i], rect, 2); +					test_rect = rect; +					test_rect.moveBy(test_moveby.x(), test_moveby.y()); +					if (!region.contains(test_rect)) { +						closest = i; +						distance = test_distance; +						fallback_mode = 2; +					} +					else { +						test_moveby = moveTQRectOutsideTQRect(rectangles[i], rect, 3); +						test_rect = rect; +						test_rect.moveBy(test_moveby.x(), test_moveby.y()); +						if (!region.contains(test_rect)) { +							closest = i; +							distance = test_distance; +							fallback_mode = 3; +						} +					} +				} +			} +		} +	} + +	// Finally, calculate the required translation to move the TQRect outside the MonitorRegion +	// so that it touches the closest line found above +	return moveTQRectOutsideTQRect(rectangles[closest], rect, fallback_mode); +} + +TQPoint compressTQRectTouchingMonitorRegion(TQRect rect, MonitorRegion region, TQSize workspace) { +	// This is another fun little class (not!) +	// It needs to move the TQRect so that it touches the closest outside line of the MonitorRegion +	bool should_move; +	long rect_center_x = rect.x() + (rect.width()/2); +	long rect_center_y = rect.y() + (rect.height()/2); + +	// First, break the region into the series of source rectangles +	TQMemArray<TQRect> rectangles = region.rects(); + +	// Next, find which rectangle is closest to the center of the TQRect +	should_move = false; +	int closest = 0; +	long distance = 16384*16384; +	int fallback_mode; +	long test_distance; +	long test_center_x; +	long test_center_y; +	for ( int i = 0; i < rectangles.size(); i++ ) { +		test_center_x = rectangles[i].x() + (rectangles[i].width()/2); +		test_center_y = rectangles[i].y() + (rectangles[i].height()/2); +		test_distance = pow(test_center_x-rect_center_x,2) + pow(test_center_y-rect_center_y,2); +		if ( (abs(test_center_x-(workspace.width()/2))<2) && (abs(test_center_y-(workspace.height()/2))<2) ) { +			test_distance=0;	// Give the primary monitor "gravity" so it can attract all other monitors to itself +		} +		if (test_distance < distance) { +			// Make sure this is an outer rectangle; i,e. there is empty space where +			// we would want to move the TQRect... +			// To do that we will move the TQRect in all four possible directions, +			// and see if any put the TQRect in an empty location +			// If they do, then this current rectangle is considered usable +			// and it is added to the distance checking routine. +			TQPoint test_moveby = moveTQRectSoThatItTouchesTQRect(rectangles[i], rect, 0); +			TQRect test_rect = rect; +			test_rect.moveBy(test_moveby.x(), test_moveby.y()); +			if (!region.contains(test_rect)) { +				closest = i; +				distance = test_distance; +				fallback_mode = 0; +				should_move = true; +			} +		} +	} + +	// Finally, calculate the required translation to move the TQRect outside the MonitorRegion +	// so that it touches the closest line found above +	if (should_move) +		return moveTQRectSoThatItTouchesTQRect(rectangles[closest], rect, fallback_mode); +	else +		return TQPoint(0, 0); +} + +void KDisplayConfig::updateDraggableMonitorInformation (int monitor_id) { +	updateDraggableMonitorInformationInternal(monitor_id, true); +} + +void KDisplayConfig::updateDraggableMonitorInformationInternal (int monitor_id, bool recurse) { +	int i; +	int j; +	DraggableMonitor *primary_monitor; +	DraggableMonitor *moved_monitor; +	SingleScreenData *screendata; +	TQObjectList monitors; + +	// Find the moved draggable monitor object +	monitors = base->monitorPhyArrange->childrenListObject(); +	if ( monitors.count() ) { +		for ( i = 0; i < int(monitors.count()); ++i ) { +			if (::tqqt_cast<DraggableMonitor*>(TQT_TQWIDGET(monitors.at( i )))) { +				DraggableMonitor *monitor = static_cast<DraggableMonitor*>(TQT_TQWIDGET(monitors.at( i ))); +				if (monitor->screen_id == monitor_id) { +					moved_monitor = monitor; +					screendata = m_screenInfoArray.at(moved_monitor->screen_id); +				} +			} +		} +	} + +	if (screendata->is_extended) { +		moved_monitor->show(); +	} +	else { +		moved_monitor->hide(); +	} + +	// Handle resizing +	moved_monitor->setFixedSize(screendata->current_x_pixel_count*base->monitorPhyArrange->resize_factor, screendata->current_y_pixel_count*base->monitorPhyArrange->resize_factor); + +	// Find the primary monitor +	for (i=0;i<numberOfScreens;i++) { +		screendata = m_screenInfoArray.at(i); +		if (screendata->is_primary) +			j=i; +	} +	monitors = base->monitorPhyArrange->childrenListObject(); +	if ( monitors.count() ) { +		for ( i = 0; i < int(monitors.count()); ++i ) { +			if (::tqqt_cast<DraggableMonitor*>(TQT_TQWIDGET(monitors.at( i )))) { +				DraggableMonitor *monitor = static_cast<DraggableMonitor*>(TQT_TQWIDGET(monitors.at( i ))); +				if (monitor->screen_id == j) +					primary_monitor = monitor; +			} +		} +	} + +	if (moved_monitor != primary_monitor) { +		// Run layout rules +		applyMonitorLayoutRules(moved_monitor); + +		int toffset_x = moved_monitor->x() - ((base->monitorPhyArrange->width()/2)-(primary_monitor->width()/2)); +		int toffset_y = moved_monitor->y() - ((base->monitorPhyArrange->height()/2)-(primary_monitor->height()/2)); +	 +		int offset_x = toffset_x / base->monitorPhyArrange->resize_factor; +		int offset_y = toffset_y / base->monitorPhyArrange->resize_factor; +	 +		screendata = m_screenInfoArray.at(monitor_id); +		screendata->absolute_x_position = offset_x; +		screendata->absolute_y_position = offset_y; +	} +	else { +		// Reset the position of the primary monitor +		moveMonitor(primary_monitor, 0, 0); +	} + +	layoutDragDropDisplay(); + +// 	// FIXME Yes, this should work.  For some reason it creates big problems instead +// 	// Run layout rules on all monitors +// 	if (recurse == true) { +// 		applyMonitorLayoutRules(); +// 	} +} + +bool KDisplayConfig::applyMonitorLayoutRules() { +	int i; +	for (i=0;i<numberOfScreens;i++) { +		updateDraggableMonitorInformationInternal(i, false); +	} + +	return false; +} + +bool KDisplayConfig::applyMonitorLayoutRules(DraggableMonitor* monitor_to_move) { +	int i; +	int j; +	bool monitor_was_moved; +	SingleScreenData *screendata; +	TQObjectList monitors; + +	// Ensure that the monitors: +	// 1) Do NOT overlap +	// 2) Touch on at least one side +	monitor_was_moved = false; + +	// Handle 1) + +	// First, create a region from the monitor rectangles +	// The moved monitor cannot exist within that region +	MonitorRegion other_monitors; +	monitors = base->monitorPhyArrange->childrenListObject(); +	if ( monitors.count() ) { +		for ( i = 0; i < int(monitors.count()); ++i ) { +			if (::tqqt_cast<DraggableMonitor*>(TQT_TQWIDGET(monitors.at( i )))) { +				DraggableMonitor *monitor = static_cast<DraggableMonitor*>(TQT_TQWIDGET(monitors.at( i ))); +				if (monitor != monitor_to_move) { +					other_monitors = other_monitors.unite(MonitorRegion(monitor->geometry())); +				} +			} +		} +	} + +	// Now get the required move X/Y direction +	TQPoint req_move = moveTQRectOutsideMonitorRegion(monitor_to_move->geometry(), other_monitors); + +	// And move the monitor +	if (!monitor_to_move->isHidden()) +		monitor_to_move->move(monitor_to_move->x()+req_move.x(), monitor_to_move->y()+req_move.y()); +	else { +		req_move.setX(0); +		req_move.setY(0); +		monitor_to_move->move(base->monitorPhyArrange->width(), base->monitorPhyArrange->height()); +	} + +	if ((req_move.x() != 0) || (req_move.y() != 0)) +		monitor_was_moved = true; + +	// Handle 2) +	// Now we need to shrink the monitors so that no gaps appear between then +	// All shrinking must take place towards the nearest extant monitor edge + +	// First, determine which rectangles touch the primary monitor, or touch rectangles that then +	// in turn touch the primary monitor +	// FIXME + +	// Only run this routine if we don't touch the primary monitor, or touch any rectangles that +	// actually do touch the primary monitor (possible through other rectangles, etc.)... +	// This would be for efficiency +	// FIXME +// 	if () { +		TQPoint req_move2(-1,-1); +		while ((req_move2.x() != 0) || (req_move2.y() != 0)) { +			// First, create a region from the monitor rectangles +			MonitorRegion other_monitors2; +			monitors = base->monitorPhyArrange->childrenListObject(); +			if ( monitors.count() ) { +				for ( i = 0; i < int(monitors.count()); ++i ) { +					if (::tqqt_cast<DraggableMonitor*>(TQT_TQWIDGET(monitors.at( i )))) { +						DraggableMonitor *monitor = static_cast<DraggableMonitor*>(TQT_TQWIDGET(monitors.at( i ))); +						if (monitor != monitor_to_move) { +							other_monitors2 = other_monitors2.unite(MonitorRegion(monitor->geometry())); +						} +					} +				} +			} +	 +			// Now get the required move X/Y direction +			req_move2 = compressTQRectTouchingMonitorRegion(monitor_to_move->geometry(), other_monitors, base->monitorPhyArrange->size()); +		 +			// And move the monitor +			if (!monitor_to_move->isHidden()) +				monitor_to_move->move(monitor_to_move->x()+req_move2.x(), monitor_to_move->y()+req_move2.y()); +			else { +				req_move2.setX(0); +				req_move2.setY(0); +				monitor_to_move->move(base->monitorPhyArrange->width(), base->monitorPhyArrange->height()); +			} +	 +			if ((req_move2.x() != 0) || (req_move2.y() != 0)) +				monitor_was_moved = true; +		} +// 	} + +	return monitor_was_moved; +} + +void KDisplayConfig::moveMonitor(DraggableMonitor* monitor, int realx, int realy) { +	int i; +	int j; +	DraggableMonitor *primary_monitor; +	SingleScreenData *screendata; + +	// Find the primary monitor +	for (i=0;i<numberOfScreens;i++) { +		screendata = m_screenInfoArray.at(i); +		if (screendata->is_primary) +			j=i; +	} +	TQObjectList monitors = base->monitorPhyArrange->childrenListObject(); +	if ( monitors.count() ) { +		for ( i = 0; i < int(monitors.count()); ++i ) { +			if (::tqqt_cast<DraggableMonitor*>(TQT_TQWIDGET(monitors.at( i )))) { +				DraggableMonitor *monitor = static_cast<DraggableMonitor*>(TQT_TQWIDGET(monitors.at( i ))); +				if (monitor->screen_id == j) +					primary_monitor = monitor; +			} +		} +	} + +	int tx = realx * base->monitorPhyArrange->resize_factor; +	int ty = realy * base->monitorPhyArrange->resize_factor; + +	if (!monitor->isHidden()) +		monitor->move((base->monitorPhyArrange->width()/2)-(primary_monitor->width()/2)+tx,(base->monitorPhyArrange->height()/2)-(primary_monitor->height()/2)+ty); +	else +		monitor->move(base->monitorPhyArrange->width(), base->monitorPhyArrange->height()); +} + +int KDisplayConfig::realResolutionSliderValue() { +	return base->resolutionSlider->maxValue() - base->resolutionSlider->value(); +} + +void KDisplayConfig::setRealResolutionSliderValue(int index) { +	base->resolutionSlider->setValue(base->resolutionSlider->maxValue() - index); +} + +/**** KDisplayConfig ****/ + +KDisplayConfig::KDisplayConfig(TQWidget *parent, const char *name, const TQStringList &) +  : KCModule(KDisplayCFactory::instance(), parent, name), m_randrsimple(0) +{ + +	m_randrsimple = new KRandrSimpleAPI(); +	 +	TQVBoxLayout *layout = new TQVBoxLayout(this, KDialog::marginHint(), KDialog::spacingHint()); +	systemconfig = new KSimpleConfig( TQString::tqfromLatin1( KDE_CONFDIR "/kdisplay/kdisplayconfigrc" )); +	 +	KAboutData *about = +		new KAboutData(I18N_NOOP("kcmdisplayconfig"), I18N_NOOP("TDE Display Profile Control Module"), +			0, 0, KAboutData::License_GPL, +			I18N_NOOP("(c) 2011 Timothy Pearson")); + +	about->addAuthor("Timothy Pearson", 0, "kb9vqf@pearsoncomputing.net"); +	setAboutData( about ); +	 +	base = new DisplayConfigBase(this); +	layout->add(base); +	 +	setRootOnlyMsg(i18n("<b>The global display configuration is a system wide setting, and requires administrator access</b><br>To alter the system's global display configuration, click on the \"Administrator Mode\" button below.")); +	setUseRootOnlyMsg(true); + +	connect(base->systemEnableSupport, TQT_SIGNAL(clicked()), TQT_SLOT(changed())); +	connect(base->systemEnableSupport, TQT_SIGNAL(clicked()), TQT_SLOT(processLockoutControls())); +	connect(base->monitorDisplaySelectDD, TQT_SIGNAL(activated(int)), TQT_SLOT(changed())); +	connect(base->resolutionSlider, TQT_SIGNAL(valueChanged(int)), TQT_SLOT(resolutionSliderChanged(int))); +	connect(base->monitorDisplaySelectDD, TQT_SIGNAL(activated(int)), TQT_SLOT(selectScreen(int))); +	connect(base->monitorPhyArrange, TQT_SIGNAL(workspaceRelayoutNeeded()), this, TQT_SLOT(layoutDragDropDisplay())); + +	connect(base->isPrimaryMonitorCB, TQT_SIGNAL(clicked()), TQT_SLOT(changed())); +	connect(base->isPrimaryMonitorCB, TQT_SIGNAL(clicked()), TQT_SLOT(ensurePrimaryMonitorIsAvailable())); +	connect(base->isExtendedMonitorCB, TQT_SIGNAL(clicked()), TQT_SLOT(changed())); +	connect(base->isExtendedMonitorCB, TQT_SIGNAL(clicked()), TQT_SLOT(updateExtendedMonitorInformation())); + +	connect(base->systemEnableSupport, TQT_SIGNAL(toggled(bool)), base->monitorDisplaySelectDD, TQT_SLOT(setEnabled(bool))); + +	load(); + +	addTab( "iccconfig", i18n( "Color Profiles" ) );	// [FIXME[ No way to save settings here yet + +	processLockoutControls(); +} + +KDisplayConfig::~KDisplayConfig() +{ +	delete systemconfig; +	if (m_randrsimple) { +		delete m_randrsimple; +		m_randrsimple = 0; +	} +} + +void KDisplayConfig::updateExtendedMonitorInformation () { +	SingleScreenData *screendata; + +	screendata = m_screenInfoArray.at(base->monitorDisplaySelectDD->currentItem()); +	screendata->is_extended = base->isExtendedMonitorCB->isChecked(); + +	updateDisplayedInformation(); +} + +void KDisplayConfig::deleteProfile () { + +} + +void KDisplayConfig::renameProfile () { + +} + +void KDisplayConfig::addProfile () { + +} + +void KDisplayConfig::load() +{ +	load( false ); +} + +void KDisplayConfig::selectProfile (int slotNumber) { + +} + +void KDisplayConfig::selectScreen (int slotNumber) { +	base->monitorDisplaySelectDD->setCurrentItem(slotNumber); +	updateDisplayedInformation(); +} + +void KDisplayConfig::updateArray (void) { +	// Discover display information +	int i; +	int j; + +	XRROutputInfo *output_info; + +	// Clear existing info +	SingleScreenData *screendata; +	for ( screendata = m_screenInfoArray.first(); screendata; screendata = m_screenInfoArray.next() ) { +		m_screenInfoArray.remove(screendata); +		delete screendata; +	} +	 +	numberOfScreens = 0; +	if (m_randrsimple->isValid() == true) { +		randr_display = XOpenDisplay(NULL); +		randr_screen_info = m_randrsimple->read_screen_info(randr_display); +		for (i = 0; i < randr_screen_info->n_output; i++) { +			output_info = randr_screen_info->outputs[i]->info; + +			// Create new data object +			screendata = new SingleScreenData; +			m_screenInfoArray.append(screendata); +			screendata->screenFriendlyName = TQString(i18n("%1. %2 output on %3")).arg(i+1).arg(capitalizeString(output_info->name)).arg(":0");	// [FIXME] How can I get the name of the Xorg graphics driver currently in use? +			screendata->generic_screen_detected = false; + +			// Attempt to use KMS to find screen EDID and name +			TQString edid = getEDIDMonitorName(0, output_info->name);	// [FIXME] Don't hardwire to card 0! +			if (!edid.isNull()) { +				screendata->screenFriendlyName = TQString(i18n("%1. %2 on %3 on card %4")).arg(i+1).arg(edid).arg(capitalizeString(output_info->name)).arg("0");	// [FIXME] How can I get the name of the Xorg graphics driver currently in use? +			} + +			// Get resolutions +			RandRScreen *cur_screen = m_randrsimple->screen(i); +			if (cur_screen) { +				screendata->screen_connected = true; +				for (int j = 0; j < cur_screen->numSizes(); j++) { +					screendata->resolutions.append(i18n("%1 x %2").tqarg(cur_screen->pixelSize(j).width()).tqarg(cur_screen->pixelSize(j).height())); +				} +				screendata->current_resolution_index = cur_screen->proposedSize(); +	 +				// Get refresh rates +				TQStringList rr = cur_screen->refreshRates(screendata->current_resolution_index); +				for (TQStringList::Iterator it = rr.begin(); it != rr.end(); ++it) { +					screendata->refresh_rates.append(*it); +				} +				screendata->current_refresh_rate_index = cur_screen->proposedRefreshRate(); +	 +				// Get color depths +				// [FIXME] +				screendata->color_depths.append(i18n("Default")); +				screendata->current_color_depth_index = 0; +	 +				// Get orientation flags +				// RandRScreen::Rotate0 +				// RandRScreen::Rotate90 +				// RandRScreen::Rotate180 +				// RandRScreen::Rotate270 +				// RandRScreen::ReflectX +				// RandRScreen::ReflectY +	 +				screendata->rotations.append(i18n("Normal")); +				screendata->rotations.append(i18n("Rotate 90 degrees")); +				screendata->rotations.append(i18n("Rotate 180 degrees")); +				screendata->rotations.append(i18n("Rotate 270 degrees")); +				screendata->current_orientation_mask = cur_screen->proposedRotation(); +				switch (screendata->current_orientation_mask & RandRScreen::RotateMask) { +					case RandRScreen::Rotate0: +						screendata->current_rotation_index = 0; +						break; +					case RandRScreen::Rotate90: +						screendata->current_rotation_index = 1; +						break; +					case RandRScreen::Rotate180: +						screendata->current_rotation_index = 2; +						break; +					case RandRScreen::Rotate270: +						screendata->current_rotation_index = 3; +						break; +					default: +						// Shouldn't hit this one +						Q_ASSERT(screendata->current_orientation_mask & RandRScreen::RotateMask); +						break; +				} +				screendata->has_x_flip = (screendata->current_orientation_mask & RandRScreen::ReflectX); +				screendata->has_y_flip = (screendata->current_orientation_mask & RandRScreen::ReflectY); +				screendata->supports_transformations = (cur_screen->rotations() != RandRScreen::Rotate0); +	 +				// Determine if this display is primary and/or extended +				// [FIXME] +				screendata->is_primary = false; +				screendata->is_extended = false; +	 +				// Get this screen's absolute position +				// [FIXME] +				screendata->absolute_x_position = 0; +				screendata->absolute_y_position = 0; +	 +				// Get this screen's current resolution +				screendata->current_x_pixel_count = cur_screen->pixelSize(screendata->current_resolution_index).width(); +				screendata->current_y_pixel_count = cur_screen->pixelSize(screendata->current_resolution_index).height(); +			} +			else { +				// Fill in generic data for this disconnected output +				screendata->screenFriendlyName = screendata->screenFriendlyName + TQString(" (") + i18n("disconnected") + TQString(")"); +				screendata->screen_connected = false; + +				screendata->resolutions = i18n("Default"); +				screendata->refresh_rates = i18n("Default"); +				screendata->color_depths = i18n("Default"); +				screendata->rotations = i18n("N/A"); +				 +				screendata->current_resolution_index = 0; +				screendata->current_refresh_rate_index = 0; +				screendata->current_color_depth_index = 0; +				 +				screendata->current_rotation_index = 0; +				screendata->current_orientation_mask = 0; +				screendata->has_x_flip = false; +				screendata->has_y_flip = false; +				screendata->supports_transformations = false; +				 +				screendata->is_primary = false; +				screendata->is_extended = false; +				screendata->absolute_x_position = 0; +				screendata->absolute_y_position = 0; +				screendata->current_x_pixel_count = 640; +				screendata->current_y_pixel_count = 480; +			} + +			// Check for more screens... +			numberOfScreens++; +		} +	} +	else { +		screendata = new SingleScreenData; +		m_screenInfoArray.append(screendata); + +		// Fill in a bunch of generic data +		screendata->screenFriendlyName = i18n("Default output on generic video card"); +		screendata->generic_screen_detected = true; +		screendata->screen_connected = true; +		 +		screendata->resolutions = i18n("Default"); +		screendata->refresh_rates = i18n("Default"); +		screendata->color_depths = i18n("Default"); +		screendata->rotations = i18n("N/A"); +		 +		screendata->current_resolution_index = 0; +		screendata->current_refresh_rate_index = 0; +		screendata->current_color_depth_index = 0; +		 +		screendata->current_rotation_index = 0; +		screendata->current_orientation_mask = 0; +		screendata->has_x_flip = false; +		screendata->has_y_flip = false; +		screendata->supports_transformations = false; +		 +		screendata->is_primary = true; +		screendata->is_extended = true; +		screendata->absolute_x_position = 0; +		screendata->absolute_y_position = 0; +		screendata->current_x_pixel_count = 640; +		screendata->current_y_pixel_count = 480; + +		numberOfScreens++; +	} + +	// [FIXME] +	// Set this on the real primary monitor only! +	screendata = m_screenInfoArray.at(0); +	screendata->is_primary = true; +} + +TQString KDisplayConfig::getEDIDMonitorName(int card, TQString displayname) { +	TQString edid; +	TQByteArray binaryedid = getEDID(card, displayname);  +	if (binaryedid.isNull()) +		return TQString(); + +	// Get the manufacturer ID +	unsigned char letter_1 = ((binaryedid[8]>>2) & 0x1F) + 0x40; +	unsigned char letter_2 = (((binaryedid[8] & 0x03) << 3) | ((binaryedid[9]>>5) & 0x07)) + 0x40; +	unsigned char letter_3 = (binaryedid[9] & 0x1F) + 0x40; +	TQChar qletter_1 = TQChar(letter_1); +	TQChar qletter_2 = TQChar(letter_2); +	TQChar qletter_3 = TQChar(letter_3); +	TQString manufacturer_id = TQString("%1%2%3").arg(qletter_1).arg(qletter_2).arg(qletter_3); + +	// Get the model ID +	unsigned int raw_model_id = (((binaryedid[10] << 8) | binaryedid[11]) << 16) & 0xFFFF0000; +	// Reverse the bit order +	unsigned int model_id = reverse_bits(raw_model_id); + +	// Try to get the model name +	bool has_friendly_name = false; +	unsigned char descriptor_block[18]; +	int i; +	for (i=72;i<90;i++) { +		descriptor_block[i-72] = binaryedid[i] & 0xFF; +	} +	if ((descriptor_block[0] != 0) || (descriptor_block[1] != 0) || (descriptor_block[3] != 0xFC)) { +		for (i=90;i<108;i++) { +			descriptor_block[i-90] = binaryedid[i] & 0xFF; +		} +		if ((descriptor_block[0] != 0) || (descriptor_block[1] != 0) || (descriptor_block[3] != 0xFC)) { +			for (i=108;i<126;i++) { +				descriptor_block[i-108] = binaryedid[i] & 0xFF; +			} +		} +	} + +	TQString monitor_name; +	if ((descriptor_block[0] == 0) && (descriptor_block[1] == 0) && (descriptor_block[3] == 0xFC)) { +		char* pos = strchr((char *)(descriptor_block+5), '\n'); +		if (pos) { +			*pos = 0; +			has_friendly_name = true; +			monitor_name = TQString((char *)(descriptor_block+5)); +		} +		else { +			has_friendly_name = false; +		} +	} + +	// [FIXME] +	// Look up manudacturer names if possible! + +	if (has_friendly_name) +		edid = TQString("%1 %2").arg(manufacturer_id).arg(monitor_name); +	else +		edid = TQString("%1 0x%2").arg(manufacturer_id).arg(model_id, 0, 16); + +	return edid; +} + +TQByteArray KDisplayConfig::getEDID(int card, TQString displayname) { +	TQFile file(TQString("/sys/class/drm/card%1-%2/edid").arg(card).arg(displayname)); +	if (!file.open (IO_ReadOnly)) +		return TQByteArray(); +	TQByteArray binaryedid = file.readAll(); +	file.close(); +	return binaryedid; +} + +void KDisplayConfig::updateDisplayedInformation () { +	// Insert data into the GUI +	int i; +	SingleScreenData *screendata; + +	ensureMonitorDataConsistency(); + +	screendata = m_screenInfoArray.at(base->monitorDisplaySelectDD->currentItem()); + +	if (screendata->screen_connected) { +		base->resolutionSlider->setEnabled(true); +		base->refreshRateDD->setEnabled(true); +		base->rotationSelectDD->setEnabled(true); +		base->orientationHFlip->setEnabled(true); +		base->orientationVFlip->setEnabled(true); +		base->isPrimaryMonitorCB->setEnabled(true); +		base->isExtendedMonitorCB->setEnabled(true); +	} + +	// Update the resolutions for the selected screen +	base->resolutionSlider->blockSignals(true); +	base->resolutionSlider->setMaxValue(screendata->refresh_rates.count()); +	setRealResolutionSliderValue(screendata->current_resolution_index); +	resolutionSliderTextUpdate(realResolutionSliderValue()); +	base->resolutionSlider->blockSignals(false); + +	// Now the refresh rates for the selected screen +	base->refreshRateDD->blockSignals(true); +	base->refreshRateDD->clear(); +	for (i=0;i<screendata->refresh_rates.count();i++) { +		base->refreshRateDD->insertItem(screendata->refresh_rates[i], i); +	} +	base->refreshRateDD->setCurrentItem(screendata->current_refresh_rate_index); +	base->refreshRateDD->blockSignals(false); + +	// Now the rotations and transformations for the selected screen +	base->rotationSelectDD->blockSignals(true); +	base->orientationHFlip->blockSignals(true); +	base->orientationVFlip->blockSignals(true); +	base->rotationSelectDD->clear(); +	if (screendata->supports_transformations) { +		for (i=0;i<screendata->rotations.count();i++) { +			base->rotationSelectDD->insertItem(screendata->rotations[i], i); +		} +		base->rotationSelectDD->setCurrentItem(screendata->current_rotation_index); +		base->orientationHFlip->show(); +		base->orientationVFlip->show(); +		base->orientationHFlip->setChecked(screendata->has_x_flip); +		base->orientationVFlip->setChecked(screendata->has_y_flip); +	} +	else { +		base->rotationSelectDD->insertItem("Normal", 0); +		base->rotationSelectDD->setCurrentItem(0); +		base->orientationHFlip->hide(); +		base->orientationVFlip->hide(); + +	} +	base->rotationSelectDD->blockSignals(false); +	base->orientationHFlip->blockSignals(false); +	base->orientationVFlip->blockSignals(false); + +	base->isPrimaryMonitorCB->blockSignals(true); +	base->isExtendedMonitorCB->blockSignals(true); +	if (screendata->generic_screen_detected) { +		base->isPrimaryMonitorCB->setEnabled(false); +		base->isPrimaryMonitorCB->setChecked(true); +		base->isExtendedMonitorCB->setEnabled(false); +		base->isExtendedMonitorCB->setChecked(true); +	} +	else { +		base->isPrimaryMonitorCB->setEnabled(true); +		base->isPrimaryMonitorCB->setChecked(screendata->is_primary); +		if (screendata->is_primary) { +			base->isExtendedMonitorCB->setEnabled(false); +			base->isExtendedMonitorCB->setChecked(true); +		} +		else { +			base->isExtendedMonitorCB->setEnabled(true); +			base->isExtendedMonitorCB->setChecked(screendata->is_extended); +		} +	} +	base->isPrimaryMonitorCB->blockSignals(false); +	base->isExtendedMonitorCB->blockSignals(false); + +	if (!screendata->screen_connected) { +		base->resolutionSlider->setEnabled(false); +		base->refreshRateDD->setEnabled(false); +		base->rotationSelectDD->setEnabled(false); +		base->orientationHFlip->setEnabled(false); +		base->orientationVFlip->setEnabled(false); +		base->isPrimaryMonitorCB->setEnabled(false); +		base->isExtendedMonitorCB->setEnabled(false); +	} +} + +void KDisplayConfig::refreshDisplayedInformation () { +	// Insert data into the GUI +	int i; +	SingleScreenData *screendata; + +	// First, the screens +	int currentScreenIndex = base->monitorDisplaySelectDD->currentItem(); +	base->monitorDisplaySelectDD->clear(); +	for (i=0;i<numberOfScreens;i++) { +		screendata = m_screenInfoArray.at(i); +		base->monitorDisplaySelectDD->insertItem(screendata->screenFriendlyName, i); +	} +	base->monitorDisplaySelectDD->setCurrentItem(currentScreenIndex); + +	updateDisplayedInformation(); + +	updateDragDropDisplay(); +} + +void KDisplayConfig::updateDragDropDisplay() { +	// Insert data into the GUI +	int i; +	int largest_x_pixels; +	int largest_y_pixels; +	TQObjectList monitors; +	SingleScreenData *screendata; + +	// Clear any screens from the workspace +	monitors = base->monitorPhyArrange->childrenListObject(); +	if ( monitors.count() ) { +		for ( i = 0; i < int(monitors.count()); ++i ) { +			if (::tqqt_cast<DraggableMonitor*>(TQT_TQWIDGET(monitors.at( i )))) { +				TQWidget *monitor = TQT_TQWIDGET(monitors.at( i )); +				if ( !monitor->close(TRUE) ) { +					Q_ASSERT("zombie monitor will not go away!"); +				} +			} +		} +	} + +	int currentScreenIndex = base->monitorDisplaySelectDD->currentItem(); + +	// Add the screens to the workspace +	// Set the scaling small to start with +	base->monitorPhyArrange->resize_factor = 0.0625;	// This always needs to divide by a multiple of 2 +	for (i=0;i<numberOfScreens;i++) { +		screendata = m_screenInfoArray.at(i); +		DraggableMonitor *m = new DraggableMonitor( base->monitorPhyArrange, 0, WStyle_Customize | WDestructiveClose | WStyle_NoBorder | WX11BypassWM ); +		connect(m, TQT_SIGNAL(workspaceRelayoutNeeded()), this, TQT_SLOT(layoutDragDropDisplay())); +		connect(m, TQT_SIGNAL(monitorSelected(int)), this, TQT_SLOT(selectScreen(int))); +		connect(m, TQT_SIGNAL(monitorDragComplete(int)), this, TQT_SLOT(updateDraggableMonitorInformation(int))); +		m->screen_id = i; +		m->setFixedSize(screendata->current_x_pixel_count*base->monitorPhyArrange->resize_factor, screendata->current_y_pixel_count*base->monitorPhyArrange->resize_factor); +		m->setText(TQString("%1").arg(i+1)); +		m->show(); +		updateDraggableMonitorInformation(i);	// Make sure the new monitors don't overlap +	} + +	layoutDragDropDisplay(); +} + +void KDisplayConfig::layoutDragDropDisplay() { +	int i; +	int largest_x_pixels; +	int largest_y_pixels; +	TQObjectList monitors; +	SingleScreenData *screendata; + +	// Ensure data is consistent +	ensureMonitorDataConsistency(); + +	// Arrange the screens +	// First, center the primary monitor +	monitors = base->monitorPhyArrange->childrenListObject(); +	if ( monitors.count() ) { +		for ( i = 0; i < int(monitors.count()); ++i ) { +			if (::tqqt_cast<DraggableMonitor*>(TQT_TQWIDGET(monitors.at( i )))) { +				DraggableMonitor *monitor = static_cast<DraggableMonitor*>(TQT_TQWIDGET(monitors.at( i ))); +				screendata = m_screenInfoArray.at(monitor->screen_id); +				moveMonitor(monitor, screendata->absolute_x_position, screendata->absolute_y_position); +			} +		} +	} +} + +void KDisplayConfig::ensureMonitorDataConsistency() { +	int i; +	SingleScreenData *screendata; + +	for (i=0;i<numberOfScreens;i++) { +		screendata = m_screenInfoArray.at(i); +		if (!screendata->screen_connected) { +			screendata->is_primary = false; +			screendata->is_extended = false; +		} +	} + +	bool has_primary_monitor = false; +	for (i=0;i<numberOfScreens;i++) { +		screendata = m_screenInfoArray.at(i); +		if (screendata->is_primary) +			has_primary_monitor = true; +	} +	if (!has_primary_monitor) { +		for (i=0;i<numberOfScreens;i++) { +			screendata = m_screenInfoArray.at(i); +			if (!has_primary_monitor) { +				if (screendata->is_extended) { +					screendata->is_primary = true; +					screendata->is_extended = true; +					has_primary_monitor = true; +				} +			} +		} +	} +	if (!has_primary_monitor) { +		for (i=0;i<numberOfScreens;i++) { +			screendata = m_screenInfoArray.at(i); +			if (!has_primary_monitor) { +				if (screendata->screen_connected) { +					screendata->is_primary = true; +					screendata->is_extended = true; +					has_primary_monitor = true; +				} +			} +		} +	} + +	for (i=0;i<numberOfScreens;i++) { +		screendata = m_screenInfoArray.at(i); +		if (screendata->is_primary) { +			screendata->absolute_x_position = 0; +			screendata->absolute_y_position = 0; +			screendata->is_extended = true; +		} +	} + +	for (i=0;i<numberOfScreens;i++) { +		screendata = m_screenInfoArray.at(i); +		TQString resolutionstring = screendata->resolutions[screendata->current_resolution_index]; +		int separator_pos = resolutionstring.find(" x "); +		TQString x_res_string = resolutionstring.left(separator_pos); +		TQString y_res_string = resolutionstring.right(resolutionstring.length()-separator_pos-3); +		screendata->current_x_pixel_count = x_res_string.toInt(); +		screendata->current_y_pixel_count = y_res_string.toInt(); +	} +} + +void KDisplayConfig::resolutionSliderTextUpdate(int index) { +	SingleScreenData *screendata; +	screendata = m_screenInfoArray.at(base->monitorDisplaySelectDD->currentItem()); + +	base->resolutionLabel->setText(screendata->resolutions[realResolutionSliderValue()] + TQString(" ") + i18n("pixels")); +} + +void KDisplayConfig::resolutionSliderChanged(int index) { +	SingleScreenData *screendata; +	screendata = m_screenInfoArray.at(base->monitorDisplaySelectDD->currentItem()); + +	screendata->current_resolution_index = realResolutionSliderValue(); +	updateDisplayedInformation(); +	updateDraggableMonitorInformation(base->monitorDisplaySelectDD->currentItem()); +} + +TQString KDisplayConfig::extractFileName(TQString displayName, TQString profileName) { + +} + +void KDisplayConfig::ensurePrimaryMonitorIsAvailable() { +	// Ensure that only one monitor, and not less than one monitor, is marked as primary +	int i; +	SingleScreenData *screendata; + +	// First, the screens +	int currentScreenIndex = base->monitorDisplaySelectDD->currentItem(); +	for (i=0;i<numberOfScreens;i++) { +		screendata = m_screenInfoArray.at(i); +		if (i != currentScreenIndex) +			screendata->is_primary = false; +	} +	screendata = m_screenInfoArray.at(currentScreenIndex); +	screendata->is_primary = true; +	refreshDisplayedInformation(); +} + +int KDisplayConfig::findProfileIndex(TQString profileName) { + +} + +int KDisplayConfig::findScreenIndex(TQString screenName) { + +} + +void KDisplayConfig::processLockoutControls() { +	if (getuid() != 0 || !systemconfig->checkConfigFilesWritable( true )) { +		base->globalTab->setEnabled(false); +		base->resolutionTab->setEnabled(false); +	} +	else { +		base->globalTab->setEnabled(true); +		if (base->systemEnableSupport->isChecked()) { +			base->resolutionTab->setEnabled(true); +		} +		else { +			base->resolutionTab->setEnabled(false); +		} +	} +} + +void KDisplayConfig::addTab( const TQString name, const TQString label ) +{ +	// [FIXME] This is incomplete...Apply may not work... +	TQWidget *page = new TQWidget( base->mainTabContainerWidget, name.latin1() ); +	TQVBoxLayout *top = new TQVBoxLayout( page, KDialog::marginHint() ); +	 +	KCModule *kcm = KCModuleLoader::loadModule( name, page ); +	 +	if ( kcm ) +	{ +		top->addWidget( kcm ); +		base->mainTabContainerWidget->addTab( page, label ); +		 +		connect( kcm, TQT_SIGNAL( changed(bool) ), this, TQT_SLOT( changed() ) ); +		//m_modules.insert(kcm, false); +	} +	else { +		delete page; +	} +} + +void KDisplayConfig::load(bool useDefaults ) +{ +	// Update the toggle buttons with the current configuration +	int i; +	int j; +	 +	updateArray(); +	 +	systemconfig->setGroup(NULL); +	base->systemEnableSupport->setChecked(systemconfig->readBoolEntry("EnableDisplayControl", false)); +	 +	refreshDisplayedInformation(); +	 +	emit changed(useDefaults); +} + +void KDisplayConfig::save() +{ +	int i; +	int j; +	KRandrSimpleAPI *m_randrsimple = new KRandrSimpleAPI(); + +	// Write system configuration +	systemconfig->setGroup(NULL); +	systemconfig->writeEntry("EnableDisplayControl", base->systemEnableSupport->isChecked()); + + +	systemconfig->sync(); + +	emit changed(false); +} + +void KDisplayConfig::defaults() +{ +	load( true ); +} + +TQString KDisplayConfig::quickHelp() const +{ +	return i18n("<h1>Monitor & Display Configuration</h1> This module allows you to configure monitors attached to your" +	" computer via TDE."); +} + +#include "displayconfig.moc" diff --git a/kcontrol/displayconfig/displayconfig.desktop b/kcontrol/displayconfig/displayconfig.desktop new file mode 100644 index 000000000..9fc6a06c4 --- /dev/null +++ b/kcontrol/displayconfig/displayconfig.desktop @@ -0,0 +1,23 @@ +[Desktop Entry] +Exec=kcmshell displayconfig +Icon=background +Type=Application +DocPath=kcontrol/displayconfig/index.html + +X-KDE-Library=displayconfig +X-KDE-ParentApp=kcontrol +X-KDE-RootOnly=true +X-KDE-SubstituteUID=true + +Categories=Qt;KDE;X-KDE-settings-peripherals; +Comment=Configure display +Comment[en_US]=Configure display +DocPath=kcontrol/displayconfig.html +GenericName= +GenericName[en_US]= +Keywords=monitor,resolution,display +MimeType= +Name=Monitor and Display +Name[en_US]=Monitor and Display + +NoDisplay=true
\ No newline at end of file diff --git a/kcontrol/displayconfig/displayconfig.h b/kcontrol/displayconfig/displayconfig.h new file mode 100644 index 000000000..d0d7a318c --- /dev/null +++ b/kcontrol/displayconfig/displayconfig.h @@ -0,0 +1,144 @@ +/** + * displayconfig.h + * + * Copyright (c) 2009-2010 Timothy Pearson <kb9vqf@pearsoncomputing.net> + * + *  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. + */ + +#ifndef _KCM_DisplayCONFIG_H +#define _KCM_DisplayCONFIG_H + +#ifdef HAVE_CONFIG_H +#include <config.h> +#endif + +#include <tqptrlist.h> +#include <tqslider.h> +#include <tqworkspace.h> +#include <tqobjectlist.h> +#include <tqwidgetlist.h> + +#include <dcopobject.h> + +#include <libkrandr/libkrandr.h> + +#include "monitorworkspace.h" +#include "displayconfigbase.h" + +class KConfig; +class KPopupMenu; +class KListViewItem; + +struct SingleScreenData { +	TQString screenFriendlyName; +	bool generic_screen_detected; +	bool screen_connected; + +	TQStringList resolutions; +	TQStringList refresh_rates; +	TQStringList color_depths; +	TQStringList rotations; + +	int current_resolution_index; +	int current_refresh_rate_index; +	int current_color_depth_index; + +	int current_rotation_index; +	int current_orientation_mask; +	bool has_x_flip; +	bool has_y_flip; +	bool supports_transformations; + +	bool is_primary; +	bool is_extended; +	int absolute_x_position; +	int absolute_y_position; +	int current_x_pixel_count; +	int current_y_pixel_count; +}; + +class KDisplayConfig : public KCModule, public DCOPObject +{ +  K_DCOP +  Q_OBJECT + + +public: +	//KDisplayConfig(TQWidget *parent = 0L, const char *name = 0L); +	KDisplayConfig(TQWidget *parent, const char *name, const TQStringList &); +	virtual ~KDisplayConfig(); +	 +	DisplayConfigBase *base; +	 +	void load(); +	void load( bool useDefaults); +	void save(); +	void defaults(); +	 +	int buttons(); +	TQString quickHelp() const; + +k_dcop: + +private: + +	KConfig *config; +	bool _ok; +	Display *randr_display; +	ScreenInfo *randr_screen_info; +	int numberOfProfiles; +	int numberOfScreens; +	TQStringList cfgScreenInfo; +	TQStringList cfgProfiles; +	void refreshDisplayedInformation (); +	void updateDisplayedInformation (); +	TQString extractFileName(TQString displayName, TQString profileName); +	TQString *displayFileArray; +	int findProfileIndex(TQString profileName); +	int findScreenIndex(TQString screenName); +	TQString m_defaultProfile; +	KRandrSimpleAPI *m_randrsimple; +	TQPtrList<SingleScreenData> m_screenInfoArray; +	int realResolutionSliderValue(); +	void setRealResolutionSliderValue(int index); +	void addTab( const TQString name, const TQString label ); +	void moveMonitor(DraggableMonitor* monitor, int realx, int realy); +	bool applyMonitorLayoutRules(void); +	bool applyMonitorLayoutRules(DraggableMonitor* primary_monitor); +	void updateDraggableMonitorInformationInternal (int, bool); +	TQByteArray getEDID(int, TQString); +	TQString getEDIDMonitorName(int, TQString); + +private slots: +	void selectProfile (int slotNumber); +	void selectScreen (int slotNumber); +	void resolutionSliderChanged(int index); +	void resolutionSliderTextUpdate(int index); +	void updateArray (void); +	void addProfile (void); +	void renameProfile (void); +	void deleteProfile (void); +	void ensurePrimaryMonitorIsAvailable (void); +	void updateDragDropDisplay (void); +	void layoutDragDropDisplay (void); +	void ensureMonitorDataConsistency (void); +	void updateDraggableMonitorInformation (int); +	void updateExtendedMonitorInformation (void); +	void processLockoutControls (void); +}; + +#endif + diff --git a/kcontrol/displayconfig/displayconfigbase.ui b/kcontrol/displayconfig/displayconfigbase.ui new file mode 100644 index 000000000..d19438128 --- /dev/null +++ b/kcontrol/displayconfig/displayconfigbase.ui @@ -0,0 +1,299 @@ +<!DOCTYPE UI><UI version="3.0" stdsetdef="1"> +<class>DisplayConfigBase</class> +<widget class="TQWidget"> +    <property name="name"> +	    <cstring>DisplayConfigBase</cstring> +    </property> +    <property name="geometry"> +        <rect> +            <x>0</x> +            <y>0</y> +            <width>519</width> +            <height>356</height> +        </rect> +    </property> +    <grid> +        <property name="name"> +            <cstring>unnamed</cstring> +        </property> +        <widget class="TQTabWidget" row="0" column="0"> +            <property name="name"> +                <cstring>mainTabContainerWidget</cstring> +            </property> +            <property name="enabled"> +                <bool>true</bool> +            </property> +	    <widget class="TQWidget"> +		    <property name="name"> +			    <cstring>globalTab</cstring> +		    </property> +		    <attribute name="title"> +			    <string>Global Settings</string> +		    </attribute> +		    <grid> +			    <property name="name"> +				    <cstring>unnamed</cstring> +			    </property> +			    <widget class="TQGroupBox" row="0" column="0"> +				    <property name="name"> +					    <cstring>groupSystemSettings</cstring> +				    </property> +				    <property name="title"> +					    <string>Global</string> +				    </property> +				    <grid> +					    <property name="name"> +						    <cstring>unnamed</cstring> +					    </property> +					    <widget class="TQCheckBox" row="0" column="0" colspan="4"> +						    <property name="name"> +							    <cstring>systemEnableSupport</cstring> +						    </property> +						    <property name="text"> +							    <string>&Enable global display control</string> +						    </property> +					    </widget> +				    </grid> +			    </widget> +		    </grid> +	    </widget> +            <widget class="TQWidget"> +                <property name="name"> +                    <cstring>resolutionTab</cstring> +                </property> +                <attribute name="title"> +                    <string>Resolution and Layout</string> +                </attribute> +                <grid> +                    <property name="name"> +                        <cstring>unnamed</cstring> +                    </property> +		    <widget class="TQGroupBox" row="1" column="0"> +			    <property name="name"> +				    <cstring>groupMonitorSettings</cstring> +			    </property> +			    <property name="title"> +				    <string>Monitors</string> +			    </property> +			    <grid> +				    <property name="name"> +					    <cstring>unnamed</cstring> +				    </property> +				    <widget class="TQLabel" row="0" column="0" colspan="4"> +					    <property name="name"> +						    <cstring>textLabel4_3</cstring> +					    </property> +					    <property name="text"> +						    <string>Drag the monitor icons to match the physical arrangement of your monitors.</string> +					    </property> +				    </widget> +				    <widget class="MonitorWorkspace" row="1" column="0" colspan="4"> +					    <property name="name"> +						    <cstring>monitorPhyArrange</cstring> +					    </property> +					    <property name="sizePolicy"> +						    <sizepolicy> +							    <hsizetype>7</hsizetype> +							    <vsizetype>1</vsizetype> +							    <horstretch>1</horstretch> +							    <verstretch>0</verstretch> +						    </sizepolicy> +					    </property> +					    <property name="maximumSize"> +						    <size> +							    <width>32767</width> +							    <height>200</height> +						    </size> +					    </property> +				    </widget> +				    <widget class="TQLabel" row="2" column="0" colspan="4"> +					    <property name="name"> +						    <cstring>textLabel4_4</cstring> +					    </property> +					    <property name="text"> +						    <string>Display:</string> +					    </property> +				    </widget> +				    <widget class="KComboBox" row="3" column="0" colspan="4"> +					    <property name="name"> +						    <cstring>monitorDisplaySelectDD</cstring> +					    </property> +				    </widget> +				    <widget class="TQGroupBox" row="4" column="0" colspan="2"> +					    <property name="name"> +						    <cstring>groupResolution</cstring> +					    </property> +					    <property name="title"> +						    <string>&Screen resolution</string> +					    </property> +					    <grid> +						    <property name="name"> +							    <cstring>unnamed</cstring> +						    </property> +						    <widget class="TQLabel" row="0" column="0"> +							    <property name="name"> +								    <cstring>textLabel4_5</cstring> +							    </property> +							    <property name="text"> +								    <string>Less</string> +							    </property> +						    </widget> +						    <widget class="TQSlider" row="0" column="1"> +							    <property name="name"> +								    <cstring>resolutionSlider</cstring> +							    </property> +							    <property name="maxValue"> +								    <number>0</number> +							    </property> +							    <property name="orientation"> +								    <enum>Horizontal</enum> +							    </property> +							    <property name="tickmarks"> +								    <enum>Below</enum> +							    </property> +							    <property name="tickInterval"> +								    <number>1</number> +							    </property> +						    </widget> +						    <widget class="TQLabel" row="0" column="2"> +							    <property name="name"> +								    <cstring>textLabel4_6</cstring> +							    </property> +							    <property name="text"> +								    <string>More</string> +							    </property> +						    </widget> +						    <widget class="TQLabel" row="1" column="1"> +							    <property name="name"> +								    <cstring>resolutionLabel</cstring> +							    </property> +							    <property name="text"> +								    <string>unset</string> +							    </property> +							    <property name="alignment"> +								    <set>AlignCenter</set> +							    </property> +						    </widget> +					    </grid> +		    		    </widget> +				    <widget class="TQGroupBox" row="4" column="2" colspan="2"> +					    <property name="name"> +						    <cstring>groupColorDepth</cstring> +					    </property> +					    <property name="title"> +						    <string>&Refresh rate</string> +					    </property> +					    <grid> +						    <property name="name"> +							    <cstring>unnamed</cstring> +						    </property> +						    <widget class="KComboBox" row="0" column="0"> +							    <property name="name"> +								    <cstring>refreshRateDD</cstring> +							    </property> +						    </widget> +						    <widget class="TQWidget" row="1" column="0"> +							    <property name="name"> +								    <cstring>colorBarWidget</cstring> +							    </property> +						    </widget> +					    </grid> +		    		    </widget> +				    <widget class="TQGroupBox" row="5" column="0" colspan="4"> +					    <property name="name"> +						    <cstring>groupOrientation</cstring> +					    </property> +					    <property name="title"> +						    <string>Screen Transformations</string> +					    </property> +					    <grid> +						    <property name="name"> +							    <cstring>unnamed</cstring> +						    </property> +						    <widget class="KComboBox" row="0" column="0" colspan="1"> +							    <property name="name"> +								    <cstring>rotationSelectDD</cstring> +							    </property> +						    </widget> +						    <widget class="TQCheckBox" row="0" column="1" colspan="1"> +							    <property name="name"> +								    <cstring>orientationHFlip</cstring> +							    </property> +							    <property name="text"> +								    <string>&Mirror screen horizontally</string> +							    </property> +						    </widget> +						    <widget class="TQCheckBox" row="0" column="2" colspan="1"> +							    <property name="name"> +								    <cstring>orientationVFlip</cstring> +							    </property> +							    <property name="text"> +								    <string>Mirror screen &vertically</string> +							    </property> +						    </widget> +					    </grid> +				    </widget> +				    <widget class="TQCheckBox" row="6" column="0" colspan="4"> +					    <property name="name"> +						    <cstring>isPrimaryMonitorCB</cstring> +					    </property> +					    <property name="text"> +						    <string>&Use this device as the primary monitor.</string> +					    </property> +				    </widget> +				    <widget class="TQCheckBox" row="7" column="0" colspan="4"> +					    <property name="name"> +						    <cstring>isExtendedMonitorCB</cstring> +					    </property> +					    <property name="text"> +						    <string>&Extend my Trinity desktop onto this monitor.</string> +					    </property> +				    </widget> +				    <widget class="KPushButton" row="8" column="3" colspan="4"> +					    <property name="name"> +						    <cstring>identifyMonitors</cstring> +					    </property> +					    <property name="text"> +						    <string>&Identify</string> +					    </property> +				    </widget> +			    </grid> +		    </widget> +                    <spacer row="4" column="0"> +                        <property name="name" stdset="0"> +                            <cstring>Spacer4</cstring> +                        </property> +                        <property name="orientation"> +                            <enum>Vertical</enum> +                        </property> +                        <property name="sizeType"> +                            <enum>Expanding</enum> +                        </property> +                        <property name="tqsizeHint"> +                            <size> +                                <width>20</width> +                                <height>20</height> +                            </size> +                        </property> +                    </spacer> +                </grid> +            </widget> +        </widget> +    </grid> +</widget> +<includes> +	<include location="local" impldecl="in implementation">DisplayConfigBase.ui.h</include> +</includes> +<Q_SLOTS> +    <slot>enableSupport_toggled(bool)</slot> +</Q_SLOTS> +<includes> +    <include location="local" impldecl="in implementation">kdialog.h</include> +    <include location="local" impldecl="in implementation">kcombobox.h</include> +    <include location="local" impldecl="in implementation">kpushbutton.h</include> +    <include location="local" impldecl="in implementation">tqworkspace.h</include> +    <include location="global" impldecl="in implementation">monitorworkspace.h</include> +</includes> +<layoutdefaults spacing="3" margin="6"/> +<layoutfunctions spacing="KDialog::spacingHint" margin="KDialog::marginHint"/> +</UI> diff --git a/kcontrol/displayconfig/monitorworkspace.cpp b/kcontrol/displayconfig/monitorworkspace.cpp new file mode 100644 index 000000000..b8907b71d --- /dev/null +++ b/kcontrol/displayconfig/monitorworkspace.cpp @@ -0,0 +1,151 @@ +/** + * monitorworkspace.cpp + * + * Copyright (c) 2011 Timothy Pearson <kb9vqf@pearsoncomputing.net> + * + *  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 <tqcheckbox.h> +#include <tqlabel.h> +#include <tqlayout.h> +#include <tqlineedit.h> +#include <tqpushbutton.h> + +#include <unistd.h> +#include <string> +#include <stdio.h> +#include <tqstring.h> + +#include "monitorworkspace.h" + +using namespace std; + +/**** Monitor Region ****/ + +MonitorRegion::MonitorRegion() +{ +} + +MonitorRegion::MonitorRegion(TQRect rect) +{ +	rectangles.resize(1); +	rectangles[0] = rect; +} + +MonitorRegion::MonitorRegion(TQMemArray<TQRect> newrects) +{ +	rectangles = newrects; +} + +MonitorRegion::~MonitorRegion() +{ +} + +TQMemArray<TQRect> MonitorRegion::rects() +{ +	return rectangles; +} + +MonitorRegion MonitorRegion::unite(MonitorRegion rect) +{ +	int i; +	int j; +	TQMemArray<TQRect> newrectangles = rectangles.copy();	// This MUST be a copy, otherwise Bad Things will happen VERY quickly! +	newrectangles.resize(rectangles.count() + rect.rects().count()); +	j=0; +	for (i=rectangles.count();i<newrectangles.count();i++) { +		newrectangles[i] = rect.rects()[j]; +		j++; +	} +	MonitorRegion newregion(newrectangles); +	return newregion; +} + +bool MonitorRegion::contains(TQRect rect) +{ +	int i; +	for (i=0;i<rectangles.count();i++) { +		if (rectangles[i].intersects(rect)) +			return true; +	} +	return false; +} + +/**** Draggable Monitor Widget ****/ +DraggableMonitor::DraggableMonitor( TQWidget* parent, const char* name, int wflags ) +	: TQLabel( parent, name, wflags ) +{ +	setAlignment(AlignHCenter | AlignVCenter); +	setFrameShape(Box); +	setFrameShadow(Plain); +	setLineWidth(4); +	setMidLineWidth(4); +} + +DraggableMonitor::~DraggableMonitor() +{ + +} + +void DraggableMonitor::mousePressEvent(TQMouseEvent *event) +{ +	lastMousePosition = event->pos(); +	emit(monitorSelected(screen_id)); +} + +void DraggableMonitor::mouseMoveEvent(TQMouseEvent *event) +{ +	TQPoint mousePos = event->pos(); +	TQPoint mouseMove = TQPoint(mousePos.x() - lastMousePosition.x(), mousePos.y() - lastMousePosition.y()); + +	int moveToX = x()+mouseMove.x(); +	int moveToY = y()+mouseMove.y(); + +	int maxX = parentWidget()->width() - width() - 1; +	int maxY = parentWidget()->height() - height() - 1; + +	if (moveToX < 1) moveToX = 1; +	if (moveToY < 1) moveToY = 1; +	if (moveToX > maxX) moveToX = maxX; +	if (moveToY > maxY) moveToY = maxY; + +	move(moveToX, moveToY); +} + +void DraggableMonitor::mouseReleaseEvent(TQMouseEvent *event) +{ +	emit(monitorDragComplete(screen_id)); +} + +/**** Draggable Monitor Container ****/ +MonitorWorkspace::MonitorWorkspace( TQWidget* parent, const char* name ) +	: TQWorkspace( parent, name ) +{ + +} + +MonitorWorkspace::~MonitorWorkspace() +{ + +} + +void MonitorWorkspace::resizeEvent( TQResizeEvent* re ) +{ +	TQWorkspace::resizeEvent(re); +	emit(workspaceRelayoutNeeded()); +} + +#include "monitorworkspace.moc" diff --git a/kcontrol/displayconfig/monitorworkspace.h b/kcontrol/displayconfig/monitorworkspace.h new file mode 100644 index 000000000..3bbac4968 --- /dev/null +++ b/kcontrol/displayconfig/monitorworkspace.h @@ -0,0 +1,96 @@ +/** + * monitorworkspace.h + * + * Copyright (c) 2011 Timothy Pearson <kb9vqf@pearsoncomputing.net> + * + *  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. + */ + +#ifndef MONITORWORKSPACE_H +#define MONITORWORKSPACE_H + +#ifdef HAVE_CONFIG_H +#include <config.h> +#endif + +#include <tqlabel.h> +#include <tqptrlist.h> +#include <tqslider.h> +#include <tqworkspace.h> +#include <tqobjectlist.h> +#include <tqwidgetlist.h> + +class MonitorRegion: public TQt +{ +public: +	MonitorRegion(); +	MonitorRegion(TQRect rect); +	MonitorRegion(TQMemArray<TQRect> newrects); +	~MonitorRegion(); + +	TQMemArray<TQRect> rects(); +	MonitorRegion unite(MonitorRegion region); +	bool contains(TQRect); + +private: +	TQMemArray<TQRect> rectangles; +}; + +class DraggableMonitor: public TQLabel +{ +	Q_OBJECT +	TQ_OBJECT +public: +	DraggableMonitor( TQWidget* parent, const char* name, int wflags ); +	~DraggableMonitor(); + +protected: +// 	void closeEvent( TQCloseEvent * ); +	void mousePressEvent(TQMouseEvent *event); +	void mouseReleaseEvent(TQMouseEvent *event); +	void mouseMoveEvent(TQMouseEvent *event); + +signals: +	void workspaceRelayoutNeeded(); +	void monitorDragComplete(int); +	void monitorSelected(int); + +public: +	int screen_id; + +private: +	TQPoint lastMousePosition; +}; + +class MonitorWorkspace : public TQWorkspace +{ +	Q_OBJECT +	TQ_OBJECT +public: +	MonitorWorkspace( TQWidget* parent, const char* name ); +	~MonitorWorkspace(); + +protected: +	virtual void resizeEvent( TQResizeEvent* re ); + +signals: +	void workspaceRelayoutNeeded(); + +public: +	float resize_factor; +}; + +#endif // MONITORWORKSPACE_H + | 
