openlp/openlp/core/display/screens.py

210 lines
8.2 KiB
Python
Raw Normal View History

# -*- coding: utf-8 -*-
2012-12-29 13:35:16 +00:00
# vim: autoindent shiftwidth=4 expandtab textwidth=120 tabstop=4 softtabstop=4
###############################################################################
# OpenLP - Open Source Lyrics Projection #
# --------------------------------------------------------------------------- #
2016-12-31 11:01:36 +00:00
# Copyright (c) 2008-2017 OpenLP Developers #
# --------------------------------------------------------------------------- #
# 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; version 2 of the License. #
# #
# 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., 59 #
# Temple Place, Suite 330, Boston, MA 02111-1307 USA #
###############################################################################
2010-06-10 21:30:50 +00:00
"""
The :mod:`screen` module provides management functionality for a machines'
2011-02-26 21:43:41 +00:00
displays.
2010-06-10 21:30:50 +00:00
"""
import json
import logging
2015-11-07 00:49:40 +00:00
from PyQt5 import QtCore
2011-03-01 17:16:59 +00:00
2017-10-07 07:05:07 +00:00
from openlp.core.common.registry import Registry
from openlp.core.common.settings import Settings
from openlp.core.common.i18n import translate
2011-03-01 17:16:59 +00:00
2010-02-27 15:31:23 +00:00
log = logging.getLogger(__name__)
2013-01-11 19:36:28 +00:00
class Screen(object):
"""
A Python representation of a screen
"""
def __init__(self, number=None, geometry=None, is_primary=False, is_display=False):
"""
Set up the screen object
:param int number: The Qt number of this screen
:param QRect geometry: The geometry of this screen as a QRect object
:param bool is_primary: Whether or not this screen is the primary screen
:param bool is_display: Whether or not this screen should be used to display lyrics
"""
self.number = number
self.geometry = geometry
self.display_geometry = geometry
self.is_primary = is_primary
self.is_display = is_display
def __str__(self):
"""
Return a string for displaying this screen
:return str: A nicely formatted string
"""
name = '{screen} {number:d}'.format(screen=translate('OpenLP.ScreenList', 'Screen'), number=self.number + 1)
if self.is_primary:
name = '{name} ({primary})'.format(name=name, primary=translate('OpenLP.ScreenList', 'primary'))
return name
2010-01-22 18:59:36 +00:00
class ScreenList(object):
"""
Wrapper to handle the parameters of the display screen.
To get access to the screen list call ``ScreenList()``.
"""
2013-08-31 18:17:38 +00:00
log.info('Screen loaded')
__instance__ = None
def __new__(cls):
2013-02-01 19:58:18 +00:00
"""
Re-implement __new__ to create a true singleton.
"""
if not cls.__instance__:
cls.__instance__ = object.__new__(cls)
return cls.__instance__
def __iter__(self):
"""
Convert this object into an iterable, so that we can iterate over it instead of the inner list
"""
for screen in self.screens:
yield screen
@classmethod
def create(cls, desktop):
2011-03-01 17:16:59 +00:00
"""
2011-03-02 17:56:06 +00:00
Initialise the screen list.
:param desktop: A QDesktopWidget object.
2011-03-01 17:16:59 +00:00
"""
screen_list = cls()
screen_list.desktop = desktop
screen_list.screens = []
screen_list.screen_count_changed()
screen_list.load_screen_settings()
screen_list.desktop.resized.connect(screen_list.on_screen_resolution_changed)
screen_list.desktop.screenCountChanged.connect(screen_list.on_screen_count_changed)
screen_list.desktop.primaryScreenChanged.connect(screen_list.on_primary_screen_changed)
return screen_list
def on_screen_resolution_changed(self, number):
2010-06-10 21:30:50 +00:00
"""
2011-03-01 17:16:59 +00:00
Called when the resolution of a screen has changed.
2011-02-26 21:43:41 +00:00
2011-03-01 17:16:59 +00:00
``number``
The number of the screen, which size has changed.
"""
log.info('screen_resolution_changed {number:d}'.format(number=number))
for screen in self.screens:
if number == screen.number:
screen.geometry = self.desktop.screenGeometry(number)
screen.is_primary = self.desktop.primaryScreen() == number
2013-08-31 18:17:38 +00:00
Registry().execute('config_screen_changed')
2011-03-01 17:16:59 +00:00
break
2011-02-26 21:43:41 +00:00
def on_screen_count_changed(self, changed_screen=None):
2010-06-10 21:30:50 +00:00
"""
2011-03-01 17:16:59 +00:00
Called when a screen has been added or removed.
``changed_screen``
2011-03-01 17:16:59 +00:00
The screen's number which has been (un)plugged.
"""
2011-06-01 06:31:22 +00:00
# Do not log at start up.
screen_count = self.desktop.screenCount()
if changed_screen is not None:
log.info('screen_count_changed {count:d}'.format(count=screen_count))
2011-03-01 17:16:59 +00:00
# Remove unplugged screens.
del self.screens[screen_count - 1:]
2011-03-01 17:16:59 +00:00
# Add new screens.
2013-08-31 18:17:38 +00:00
for number in range(self.desktop.screenCount()):
if not self.has_screen(number):
self.screens.append(Screen(number, self.desktop.getGeometry(number),
self.desktop.primaryScreen() == number))
2011-06-01 06:31:22 +00:00
# We do not want to send this message at start up.
if changed_screen is not None:
2011-03-01 17:16:59 +00:00
# Reload setting tabs to apply possible changes.
2013-08-31 18:17:38 +00:00
Registry().execute('config_screen_changed')
def on_primary_screen_changed(self):
"""
The primary screen has changed, let's sort it out and then notify everyone
"""
for screen in self.screens:
screen.is_primary = self.desktop.primaryScreen() == screen.number
Registry().execute('config_screen_changed')
def get_display_screen_list(self):
"""
Returns a list with the screens. This should only be used to display available screens to the user::
2014-05-02 06:42:17 +00:00
['Screen 1 (primary)', 'Screen 2']
"""
2011-03-04 14:44:21 +00:00
screen_list = []
for screen in self.screens:
screen_name = '{name} {number:d}'.format(name=translate('OpenLP.ScreenList', 'Screen'),
number=screen.number + 1)
if screen.is_primary:
screen_name = '{name} ({primary})'.format(name=screen_name,
primary=translate('OpenLP.ScreenList', 'primary'))
screen_list.append(screen_name)
return screen_list
def has_screen(self, number):
2010-06-10 21:30:50 +00:00
"""
2011-02-26 21:43:41 +00:00
Confirms a screen is known.
2014-03-17 19:05:55 +00:00
:param number: The screen number (int).
2010-06-10 21:30:50 +00:00
"""
for screen in self.screens:
if screen.number == number:
return True
return False
def get_number_for_window(self, window):
"""
2011-12-23 21:52:54 +00:00
Return the screen number that the centre of the passed window is in.
2014-03-17 19:05:55 +00:00
:param window: A QWidget we are finding the location of.
"""
for screen in self.screens:
if screen.geometry == window.geometry() or screen.display_geometry == window.geometry():
return screen
return None
def load_screen_settings(self):
"""
Loads the screen size and the monitor number from the settings.
"""
2013-02-10 16:05:52 +00:00
# Add the screen settings to the settings dict. This has to be done here due to cyclic dependency.
2013-01-20 11:46:19 +00:00
# Do not do this anywhere else.
screen_settings = {
'core/monitors': '{}'
2013-01-20 11:46:19 +00:00
}
Settings.extend_default_settings(screen_settings)
monitors = json.loads(Settings().value('core/monitors'))
for screen in self.screens:
monitor = monitors.get(screen.number)
if monitor:
screen.display_geometry = QtCore.QRect(monitor['x'], monitor['y'], monitor['width'], monitor['height'])
screen.is_display = True