X11 - рисовать в окне наложения

Я хочу рисовать простые примитивы в определенных пикселях на экране (аналогично этому вопрос). Для этого я рисую поверх всех окон, используя окно Overlay Window Manager. Я вижу форму, которую рисую, и события мыши проходят через нее, но я не вижу, например, движений окна, которые находятся под окном наложения (если я не убью свое приложение). Я новичок в программировании Xlib, прошу прощения за простой вопрос.

#include <assert.h>
#include <stdio.h>
#include <X11/Xlib.h>

#include <X11/extensions/Xcomposite.h>
#include <X11/extensions/Xfixes.h>
#include <X11/extensions/shape.h>

#include <cairo.h>
#include <cairo-xlib.h>

Display *d;
Window overlay;
Window root;
int width, height;

void
allow_input_passthrough (Window w)
{
    XserverRegion region = XFixesCreateRegion (d, NULL, 0);

    XFixesSetWindowShapeRegion (d, w, ShapeBounding, 0, 0, 0);
    XFixesSetWindowShapeRegion (d, w, ShapeInput, 0, 0, region);

    XFixesDestroyRegion (d, region);
}

void
prep_overlay (void)
{
    overlay = XCompositeGetOverlayWindow (d, root);
    allow_input_passthrough (overlay);
}

void draw(cairo_t *cr) {
    int quarter_w = width / 4;
    int quarter_h = height / 4;
    cairo_set_source_rgb(cr, 1.0, 0.0, 0.0);
    cairo_rectangle(cr, quarter_w, quarter_h, quarter_w * 2, quarter_h * 2);
    cairo_fill(cr);
}

int main() {
    d = XOpenDisplay(NULL);

    int s = DefaultScreen(d);
    root = RootWindow(d, s);

    XCompositeRedirectSubwindows (d, root, CompositeRedirectAutomatic);
    XSelectInput (d, root, SubstructureNotifyMask);

    width = DisplayWidth(d, s);
    height = DisplayHeight(d, s);

    prep_overlay();

    cairo_surface_t *surf = cairo_xlib_surface_create(d, overlay,
                                  DefaultVisual(d, s),
                                  width, height);
    cairo_t *cr = cairo_create(surf);

    XSelectInput(d, overlay, ExposureMask);

    draw(cr);

    XEvent ev;
    while (1) {
    XNextEvent(d, &ev);
        if (ev.type == Expose) {
            draw(cr);
        }
    }

    cairo_destroy(cr);
    cairo_surface_destroy(surf);
    XCloseDisplay(d);
    return 0;
}

Как я могу рисовать пиксели в окне наложения и по-прежнему видеть окна ниже?


person user1325516    schedule 14.02.2014    source источник
comment
рассмотрите возможность добавления тега для вашего языка программирования. Многие программисты на C / C ++ могут помочь вам, но если они выполняют фильтрацию по своим «темам», они не увидят ваш вопрос. Удачи.   -  person shellter    schedule 14.02.2014
comment
Вы можете опубликовать целую программу?   -  person n. 1.8e9-where's-my-share m.    schedule 15.02.2014
comment
Я отредактировал исходный пост и включил полную программу. Скомпилировать с: gcc x11drawoverlay.c -o x11drawoverlay -lX11 -lXfixes -lXcomposite -lcairo -I /usr/include/cairo   -  person user1325516    schedule 15.02.2014
comment
Я не получаю никаких результатов от этой программы.   -  person n. 1.8e9-where's-my-share m.    schedule 15.02.2014
comment
Он рисует красный прямоугольник на моем рабочем столе.   -  person user1325516    schedule 16.02.2014
comment
По-прежнему абсолютно ничего не делает на моем рабочем столе. Просто хань   -  person n. 1.8e9-where's-my-share m.    schedule 16.02.2014
comment
Я новичок в программировании X11 / Xlib. Подскажите, пожалуйста, почему зависает?   -  person user1325516    schedule 17.02.2014
comment
Извините, сбой подключения. Он зависает XNextEvent в ожидании событий expose, которые никогда не наступают. Первый draw не действует, прямоугольник не рисуется. Честно говоря, понятия не имею, почему. Я познакомился с программированием X11 несколько лет назад и достаточно хорошо знаком с ядром X11, но новые расширения, такие как Composite, для меня в новинку.   -  person n. 1.8e9-where's-my-share m.    schedule 17.02.2014
comment
Я удалил цикл и просто добавил цикл для рисования: while(1) { overlay = XCompositeGetOverlayWindow (d, root); draw(cr); XCompositeReleaseOverlayWindow (d, root); sleep(50); } Окно наложения по-прежнему блокирует окна внизу. Итак, еще раз, как мне рисовать в оверлейном окне, не блокируя другие окна?   -  person user1325516    schedule 17.02.2014
comment
Это неправильное использование окна наложения. Почему бы вам просто не создать собственное окно переопределения-перенаправления, которое будет ровно настолько большим, насколько вам нужно, и вызвать в нем XRaiseWindow 100 раз в секунду? (Это все равно было бы безумием, но немного менее безумным, чем существующее здесь решение)   -  person Uli Schlachter    schedule 07.01.2019


Ответы (2)


Могу я предложить более простое, чистое решение X11, которое не имеет проблемы с мерцанием, с которой я столкнулся, а также упомянул здесь. Он использует функцию override_redirect в Xlib:

#include <assert.h>
#include <stdio.h>
#include <X11/Xlib.h>
#include <X11/X.h>
#include <X11/Xutil.h>

#include <cairo.h>
#include <cairo-xlib.h>

#include <chrono>
#include <thread>

void draw(cairo_t *cr) {
    cairo_set_source_rgba(cr, 1.0, 0.0, 0.0, 0.5);
    cairo_rectangle(cr, 0, 0, 200, 200);
    cairo_fill(cr);
}

int main() {
    Display *d = XOpenDisplay(NULL);
    Window root = DefaultRootWindow(d);
    int default_screen = XDefaultScreen(d);

    // these two lines are really all you need
    XSetWindowAttributes attrs;
    attrs.override_redirect = true;

    XVisualInfo vinfo;
    if (!XMatchVisualInfo(d, DefaultScreen(d), 32, TrueColor, &vinfo)) {
        printf("No visual found supporting 32 bit color, terminating\n");
        exit(EXIT_FAILURE);
    }
    // these next three lines add 32 bit depth, remove if you dont need and change the flags below
    attrs.colormap = XCreateColormap(d, root, vinfo.visual, AllocNone);
    attrs.background_pixel = 0;
    attrs.border_pixel = 0;

    // Window XCreateWindow(
    //     Display *display, Window parent,
    //     int x, int y, unsigned int width, unsigned int height, unsigned int border_width,
    //     int depth, unsigned int class, 
    //     Visual *visual,
    //     unsigned long valuemask, XSetWindowAttributes *attributes
    // );
    Window overlay = XCreateWindow(
        d, root,
        0, 0, 200, 200, 0,
        vinfo.depth, InputOutput, 
        vinfo.visual,
        CWOverrideRedirect | CWColormap | CWBackPixel | CWBorderPixel, &attrs
    );

    XMapWindow(d, overlay);

    cairo_surface_t* surf = cairo_xlib_surface_create(d, overlay,
                                  vinfo.visual,
                                  200, 200);
    cairo_t* cr = cairo_create(surf);

    draw(cr);
    XFlush(d);

    std::this_thread::sleep_for(std::chrono::milliseconds(10000));

    cairo_destroy(cr);
    cairo_surface_destroy(surf);

    XUnmapWindow(d, overlay);
    XCloseDisplay(d);
    return 0;
}

Я пошел дальше и добавил 32-битную глубину, но вы понимаете. Вы можете удалить его, если хотите.

person Asad-ullah Khan    schedule 04.09.2019
comment
Для меня это блокирует события мыши в Ubuntu 18.04 с GNOME 3. - person David Zhao Akeley; 08.07.2020
comment
Вы имеете в виду, что вы вообще не можете двигать мышью, пока она работает? Или что вы не можете взаимодействовать с объектами, нарисованными на оверлее? Единственное, о чем я могу думать, это то, что мой вызов для сна, возможно, мешает, может быть, попробовать другой вызов для сна? - person Asad-ullah Khan; 08.07.2020
comment
Мышь отлично работает вне окна наложения; что я ощущаю, так это то, что щелчки мыши в области окна не передаются в окно за прозрачным окном. При повторном чтении я не уверен, что это действительно было целью разработки этой программы. - person David Zhao Akeley; 10.07.2020
comment
Ах да, вам, вероятно, нужно будет каким-то образом передать события в окно ниже. Я не знаю, как это сделать. Этот ответ может быть полезен: stackoverflow.com/questions/ 16400937 / - person Asad-ullah Khan; 11.07.2020

sleep(50)! это многовато, это 50 секунд. Я использовал задержку 5 мс, которая работает хорошо.

Кажется, ваша проблема связана со средой выполнения. У вас уже должен быть запущен составной диспетчер отображения. (Не все диспетчеры дисплея работают должным образом, лучше попробовать другие)

Я подтверждаю, что приведенный ниже экран обновился без каких-либо проблем, и я могу с ним взаимодействовать.

Это было запущено:

Ubuntu 15.10
Kernel 4.2.0-18-generic
X.Org X Server 1.17.2
Compiz 0.9.12.2

Вот полный код с простой модификацией задержки:

#include <assert.h>
#include <stdio.h>
#include <time.h>
#include <X11/Xlib.h>

#include <X11/extensions/Xcomposite.h>
#include <X11/extensions/Xfixes.h>
#include <X11/extensions/shape.h>

#include <cairo.h>
#include <cairo-xlib.h>

Display *d;
Window overlay;
Window root;
int width, height;

void
allow_input_passthrough (Window w)
{
    XserverRegion region = XFixesCreateRegion (d, NULL, 0);

    XFixesSetWindowShapeRegion (d, w, ShapeBounding, 0, 0, 0);
    XFixesSetWindowShapeRegion (d, w, ShapeInput, 0, 0, region);

    XFixesDestroyRegion (d, region);
}

void
prep_overlay (void)
{
    overlay = XCompositeGetOverlayWindow (d, root);
    allow_input_passthrough (overlay);
}

void draw(cairo_t *cr) {
    int quarter_w = width / 4;
    int quarter_h = height / 4;
    cairo_set_source_rgb(cr, 1.0, 0.0, 0.0);
    cairo_rectangle(cr, quarter_w, quarter_h, quarter_w * 2, quarter_h * 2);
    cairo_fill(cr);
}

int main() {
    struct timespec ts = {0, 5000000};

    d = XOpenDisplay(NULL);

    int s = DefaultScreen(d);
    root = RootWindow(d, s);

    XCompositeRedirectSubwindows (d, root, CompositeRedirectAutomatic);
    XSelectInput (d, root, SubstructureNotifyMask);

    width = DisplayWidth(d, s);
    height = DisplayHeight(d, s);

    prep_overlay();

    cairo_surface_t *surf = cairo_xlib_surface_create(d, overlay,
                                  DefaultVisual(d, s),
                                  width, height);
    cairo_t *cr = cairo_create(surf);

    XSelectInput(d, overlay, ExposureMask);

    draw(cr);

    XEvent ev;
    while(1) {
      overlay = XCompositeGetOverlayWindow (d, root);
      draw(cr);
      XCompositeReleaseOverlayWindow (d, root);
      nanosleep(&ts, NULL);
    }

    cairo_destroy(cr);
    cairo_surface_destroy(surf);
    XCloseDisplay(d);
    return 0;
}

Наложение X11 в Ubuntu 15.10

Наложение X11 в Ubuntu 15.10, обзор рабочего стола

person user.dz    schedule 16.11.2015
comment
Это так или иначе зависит от среды рабочего стола, работает в Ubuntu (Unity), но не работает в Cinnamon. У меня вопрос: вы знаете, можно ли рисовать без мерцания? Если он будет реализован так, как описано в вашем ответе, он обязательно будет работать, но пользователь сможет увидеть мигающее изображение. - person Daniel; 20.02.2018
comment
@ScienceSE Я опубликовал решение, которое не мерцает ниже, если вам все еще интересно - person Asad-ullah Khan; 04.09.2019