b3ee61e41a5afc6ac3e052985fd2bc7c8dd73c94
[lilass.git] / qt_frontend.py
1 # DSL - easy Display Setup for Laptops
2 # Copyright (C) 2012-2015 Ralf Jung <post@ralfj.de>
3 #
4 # This program is free software; you can redistribute it and/or modify
5 # it under the terms of the GNU General Public License as published by
6 # the Free Software Foundation; either version 2 of the License, or
7 # (at your option) any later version.
8 #
9 # This program is distributed in the hope that it will be useful,
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12 # GNU General Public License for more details.
13 #
14 # You should have received a copy of the GNU General Public License
15 # along with this program; if not, write to the Free Software
16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17 import sys, os
18 from screen import RelativeScreenPosition, ScreenSetup
19
20 try:
21     # Be fine with PyQt4 not being installed
22     from PyQt5 import QtCore, QtWidgets, uic
23
24     class PositionSelection(QtWidgets.QDialog):
25         def __init__(self, situation):
26             # set up main window
27             super(PositionSelection, self).__init__()
28             self._situation = situation
29             uifile = os.path.join(os.path.dirname(__file__), 'qt_dialogue.ui')
30             uic.loadUi(uifile, self)
31             
32             # fill relative position box
33             for pos in RelativeScreenPosition:
34                 self.relPos.addItem(pos.text, pos)
35             
36             # keep resolutions in sync when in mirror mode
37             def syncIfMirror(source, target):
38                 def _slot(idx):
39                     if self.isMirror:
40                         target.setCurrentIndex(idx)
41                 source.currentIndexChanged.connect(_slot)
42             syncIfMirror(self.intRes, self.extRes)
43             syncIfMirror(self.extRes, self.intRes)
44
45             # connect the update function
46             self.intEnabled.toggled.connect(self.updateEnabledControls)
47             self.extEnabled.toggled.connect(self.updateEnabledControls)
48             self.relPos.currentIndexChanged.connect(self.updateEnabledControls)
49
50             # if situation has a lastSetup, use its values as initial state
51             if situation.lastSetup:
52                 last = situation.lastSetup
53                 self.intEnabled.setChecked(last.intResolution is not None)
54                 self.extEnabled.setChecked(last.extResolution is not None)
55                 if last.relPosition:
56                     self.relPos.setCurrentIndex(last.relPosition.value-1)
57
58             # make sure we are in a correct state
59             self.updateEnabledControls()
60
61         def getRelativeScreenPosition(self):
62             idx = self.relPos.currentIndex()
63             return self.relPos.itemData(idx)
64         
65         def fillResolutionBox(self, box, resolutions):
66             # if the count did not change, update in-place (this avoids flicker)
67             if box.count() == len(resolutions):
68                 for idx, res in enumerate(resolutions):
69                     box.setItemText(idx, str(res))
70                     box.setItemData(idx, res)
71             else:
72                 # first clear it
73                 while box.count() > 0:
74                     box.removeItem(0)
75                 # then fill it
76                 for res in resolutions:
77                     box.addItem(str(res), res)
78         
79         def updateEnabledControls(self):
80             intEnabled = self.intEnabled.isChecked()
81             extEnabled = self.extEnabled.isChecked()
82             bothEnabled = intEnabled and extEnabled
83             self.isMirror = bothEnabled and self.getRelativeScreenPosition() == RelativeScreenPosition.MIRROR # only if both are enabled, we can really mirror
84             # configure screen controls
85             self.intRes.setEnabled(intEnabled)
86             self.intPrimary.setEnabled(intEnabled and not self.isMirror)
87             self.extRes.setEnabled(extEnabled)
88             self.extPrimary.setEnabled(extEnabled and not self.isMirror)
89             if not intEnabled and extEnabled:
90                 self.extPrimary.setChecked(True)
91             elif not extEnabled and intEnabled:
92                 self.intPrimary.setChecked(True)
93             # which resolutions do we offer?
94             if self.isMirror:
95                 commonRes = self._situation.commonResolutions()
96                 self.fillResolutionBox(self.intRes, commonRes)
97                 self.fillResolutionBox(self.extRes, commonRes)
98                 self.intRes.setCurrentIndex(self.extRes.currentIndex())
99             else:
100                 self.fillResolutionBox(self.intRes, self._situation.internalResolutions())
101                 self.fillResolutionBox(self.extRes, self._situation.externalResolutions())
102             # configure position control
103             self.posGroup.setEnabled(bothEnabled)
104             self.posLabel1.setEnabled(bothEnabled)
105             self.posLabel2.setEnabled(bothEnabled)
106             self.relPos.setEnabled(bothEnabled)
107             # avoid having no screen
108             self.buttonBox.button(QtWidgets.QDialogButtonBox.Ok).setEnabled(intEnabled or extEnabled)
109         
110         def run(self):
111             self.exec_()
112             if not self.result(): return None
113             intRes = self.intRes.itemData(self.intRes.currentIndex()) if self.intEnabled.isChecked() else None
114             extRes = self.extRes.itemData(self.extRes.currentIndex()) if self.extEnabled.isChecked() else None
115             return ScreenSetup(intRes, extRes, self.getRelativeScreenPosition(), self.extPrimary.isChecked())
116 except ImportError:
117     pass
118
119 # Qt frontend
120 class QtFrontend:
121     def __init__(self):
122         from PyQt5 import QtWidgets
123         self.app = QtWidgets.QApplication(sys.argv)
124         print("Qt loaded")
125     
126     def error(self, message):
127         from PyQt5 import QtWidgets
128         QtWidgets.QMessageBox.critical(None, 'Fatal error', message)
129     
130     def setup(self, situation):
131         return PositionSelection(situation).run()
132     
133     @staticmethod
134     def isAvailable():
135         try:
136             import PyQt5
137             return True
138         except ImportError:
139             return False
140