Pycairo: как правильно создать экземпляр cairo.XlibSurface()?

#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 .