#python #cairo #xlib #pycairo
#python #cairo #xlib #pycairo
Вопрос:
Я пытаюсь создавать виджеты с помощью Pycairo для мониторинга систем, таких как Conky. Я пытаюсь рисовать непосредственно на X, используя Cairo Surface с Xlib.
Когда я пытаюсь создать экземпляр класса cairo.XlibSurface(), возвращается ошибка:
TypeError: The XlibSurface type cannot be directly instantiated
Мой простой код:
import cairo
surface = cairo.XlibSurface()
Как правильно создать экземпляр cairo.XlibSurface() ?
Заранее спасибо
Комментарии:
1. Простого способа нет — может быть, вам лучше использовать Cairo через GTK — для чего актуальный пример может быть приведен здесь: pygobject.readthedocs.io/en/latest/guide/cairo_integration.html
2. С gtk легко, но я хочу интегрироваться с xlib и рисовать в окне X, а не gtk. Я не могу найти ни одного примера pycairo с xlib.
Ответ №1:
Не решен вопрос: как использовать Pycairo с Xlib, но есть элегантная альтернатива с использованием Gtk и Pycairo.
Загадка того, как использовать Pycairo с Xlib, продолжается.
Используя окно Gtk с Gdk.WindowTypeHint.DESKTOP
, получаем элегантный вывод.
import gi
gi.require_versions({
'Gdk': '3.0',
'Gtk': '3.0',
'Wnck': '3.0',
'Gst': '1.0',
'AppIndicator3': '0.1',
})
from gi.repository import Gtk, Gdk
import cairo
class Example(Gtk.Window):
def __init__(self):
super(Example, self).__init__()
self.tran_setup()
self.init_ui()
def init_ui(self):
self.connect("draw", self.on_draw)
# self.set_title("Transparent window")
self.resize(300, 250)
self.set_position(Gtk.WindowPosition.NONE)
self.move(0, 40)
self.connect("delete-event", Gtk.main_quit)
# The magic is here
self.set_type_hint(Gdk.WindowTypeHint.DESKTOP)
self.show_all()
def tran_setup(self):
self.set_app_paintable(True)
screen = self.get_screen()
#print(self.get_type_hint())
visual = screen.get_rgba_visual()
if visual != None and screen.is_composited():
self.set_visual(visual)
def on_draw(self, wid, cr):
cr.set_source_rgba(0.2, 0.2, 0.2, 0.4)
cr.set_operator(cairo.OPERATOR_SOURCE)
cr.paint()
cr.set_source_rgb(0.6, 0.6, 0.6)
cr.rectangle(20, 20, 120, 80)
cr.fill()
self.draw(cr)
def draw(self, cr):
cr.set_source_rgb(0, 256, 256)
cr.rectangle(180, 20, 80, 80)
cr.fill()
def main():
app = Example()
Gtk.main()
if __name__ == "__main__":
main()
Вывод:
Ответ №2:
[не полный ответ] У меня достаточно фрагментов, чтобы это могло быть более полезным в качестве ответа, чем в качестве комментария. Извините, что не дошел до сути.
В документации для самого PyCairo XlibSurface упоминается только, что он может быть создан с помощью определенных вызовов GTK — и я не уверен, что это старые привязки только к GTK2 . Я мог бы найти фрагмент, чтобы получить контекст Cairo, используя GTK 3, но, похоже, невозможно, не прибегая к ctypes, вернуться из контекста Cairo к поверхности, которую он использует.
Итак, вам в основном нужно будет вызвать Cairo’s C cairo_xlib_surface_create
, используя ctypes Python, и найти способ создать экземпляр Python cairo.XlibSurface, используя ctypes, из указателя, возвращаемого функцией выше. Это «недостающее звено», которое вам, вероятно, придется найти в устаревшей кодовой базе pygtk на C и реплицировать с помощью ctypes в коде Python.
Что касается параметров, необходимых для cairo_xlib_surface_create
вызова в первую очередь, вы, вероятно, можете получить их с помощью Python XLib — приведенный ниже фрагмент предоставляет вам GID окна и оболочки Python для отображения X и окна:
from Xlib import X, display
pd = display.Display()
win = pd.screen().root.create_window(0, 0, 640, 480, 0, pd.screen().root_depth, X.InputOutput, X.CopyFromParent)
win.map()
pd.sync()
xid = xx.__resource__()
Комментарии:
1. Я ценю ваш ответ, но не решаю мою проблему. Я решил это с помощью окна Gtk с
Gdk.WindowTypeHint.DESKTOP
.