Search code examples
pythonwindowpyqt4stayontop

Creating a transparent overlay with qt


I've been learning python recently and now I wanted to (try to) create my first real application, a subtitle player for Linux. So far I've been using the Greenfish subtitle player, which is aimed at Windows users and not properly working in Linux.

I wanted to create the application in qt, since I discovered that transparent windows are not possible in tkinter, but if anybody knows a better framework please suggest!

Now before starting I've been researching the web for several hours to discover how to get my application to show over a full screened flash video and it seems like this is not possible. However the aforementioned GF subtitle player manages to do so in Windows, but not in Linux(maybe it's also because it's running through wine).

So my question is it possible to create a transparent application that remains over a fullscreened flash video and if so, could you point me in the right direction?

Thanks in advance.

edit: here some example code I've been trying. The window produced by this piece of code does not stay above a fullscreened video

import sys
from PyQt4 import QtGui, QtCore

class mymainwindow(QtGui.QMainWindow):
    def __init__(self):
        QtGui.QMainWindow.__init__(self, None, QtCore.Qt.WindowStaysOnTopHint)

app = QtGui.QApplication(sys.argv)
mywindow = mymainwindow()
mywindow.show()
app.exec_()
mywindow.show()

Solution

  • The example code below will create a centred, frameless window that should stay on top of all other windows on Linux (you can click on the window to close it).

    PyQt5/6:

    import sys
    from PyQt5.QtWidgets import QApplication, QMainWindow, QStyle
    from PyQt5.QtCore import Qt, QSize
    # from PyQt6.QtWidgets import QApplication, QMainWindow, QStyle
    # from PyQt6.QtCore import Qt, QSize
    
    class Window(QMainWindow):
        def __init__(self):
            super().__init__()
            self.setWindowFlags(
                Qt.WindowType.WindowStaysOnTopHint |
                Qt.WindowType.FramelessWindowHint |
                Qt.WindowType.X11BypassWindowManagerHint
                )
            self.setGeometry(QStyle.alignedRect(
                Qt.LayoutDirection.LeftToRight,
                Qt.AlignmentFlag.AlignCenter,
                QSize(220, 32),
                QApplication.instance().primaryScreen().availableGeometry(),
                ))
    
        def mousePressEvent(self, event):
            QApplication.instance().quit()
    
    if __name__ == '__main__':
    
        app = QApplication(sys.argv)
        window = Window()
        window.show()
        app.exec()
    

    PyQt4:

    import sys
    from PyQt4.QtGui import QApplication, QMainWindow, QStyle
    from PyQt4.QtCore import Qt, QSize
    
    class Window(QtGui.QMainWindow):
        def __init__(self):
            super(Window, self).__init__()
            self.setWindowFlags(
                Qt.WindowStaysOnTopHint |
                Qt.FramelessWindowHint |
                Qt.X11BypassWindowManagerHint
                )
            self.setGeometry(QStyle.alignedRect(
                Qt.LeftToRight, Qt.AlignCenter,
                QSize(220, 32),
                qApp.desktop().availableGeometry()))
    
        def mousePressEvent(self, event):
            qApp.quit()
    
    if __name__ == '__main__':
    
        app = QApplication(sys.argv)
        window = Window()
        window.show()
        app.exec_()