X11 - Xrandr дает мне ложные мониторы

Я пытался найти все мониторы и их координаты (ширина w, рост h, x origin/top-left-most xи y origin / top-left-most y) и использовал этот код, он хорошо работает на некоторых системах. Но в других системах я получаю ложные и повторяющиеся записи. Смогу ли я избежать этих дублирующих / ложных записей монитора, если я проверил, является ли монитор зеркальным? Как проверить его зеркало?

Итак, это мой код:

// start - get all monitor resolutions
var screen = XRRGetScreenResources(getXOpenDisplay(), getDefaultRootWindow(getXOpenDisplay()));

var noutputs = screen.noutput;

for (var i=noutputs-1; i>=0; i--) {
    var info = XRRGetOutputInfo(getXOpenDisplay(), screen, screen.outputs[i]);
    if (info.connection == RR_Connected) {
        var ncrtcs = info.ncrtc;
        for (var j=ncrtcs-1; j>=0; j--) {
            var crtc_info = XRRGetCrtcInfo(getXOpenDisplay(), screen, infoCrtcs[j]);
            console.info('screen #' + i + ' mon#' + j + ' details:', crtc_info.x, crtc_info.y, crtc_info.width, crtc_info.height);

            collMonInfos.push({
                x: crtc_info.x,
                y: crtc_info.y,
                w: crtc_info.width,
                h: crtc_info.height
            });

            XRRFreeCrtcInfo(crtc_info);
        }
    }
    XRRFreeOutputInfo(info);
}
XRRFreeScreenResources(screen);
console.info('JSON:', JSON.stringify(collMonInfos));
// end - get all monitor resolutions

И это выводит это в журнал:

"screen #4 mon#0 details:" 0 0 0 0
"screen #3 mon#1 details:" 0 0 1920 1200
"screen #3 mon#0 details:" 1920 469 1366 768
"screen #2 mon#1 details:" 0 0 1920 1200
"screen #2 mon#0 details:" 1920 469 1366 768
"screen #1 mon#1 details:" 0 0 1920 1200
"screen #1 mon#0 details:" 1920 469 1366 768
"screen #0 mon#1 details:" 0 0 1920 1200
"screen #0 mon#0 details:" 1920 469 1366 768

Это в формате JSON:

[{
    "x": 0,
    "y": 0,
    "w": 0,
    "h": 0
}, {
    "x": 0,
    "y": 0,
    "w": 1920,
    "h": 1200
}, {
    "x": 1920,
    "y": 469,
    "w": 1366,
    "h": 768
}, {
    "x": 0,
    "y": 0,
    "w": 1920,
    "h": 1200
}, {
    "x": 1920,
    "y": 469,
    "w": 1366,
    "h": 768
}, {
    "x": 0,
    "y": 0,
    "w": 1920,
    "h": 1200
}, {
    "x": 1920,
    "y": 469,
    "w": 1366,
    "h": 768
}, {
    "x": 0,
    "y": 0,
    "w": 1920,
    "h": 1200
}, {
    "x": 1920,
    "y": 469,
    "w": 1366,
    "h": 768
}]

У меня действительно только 2 монитора, один 1920x1200 и 1366x768. Почему все остальные записи и как проверить, чтобы избежать (а не отфильтровать ретроспективно на основе дубликатов или 0 ч / б)?

1 ответ

Решение

Вы без необходимости перебираете каждый выход, а затем - каждый монитор. Таким образом, вы получаете дубликаты записей. Вам не нужно звонить XRRGetOutputInfo для каждого выхода, так как все необходимые данные (количество мониторов) могут быть найдены в структуре, возвращаемой XRRGetScreenResources, Вот код C, который работает (по крайней мере, для меня):

#include <stdio.h>
#include <stdlib.h>
#include <X11/Xlib.h>
#include <X11/extensions/Xrandr.h>

int main(void) {
    Display *d = XOpenDisplay(getenv("DISPLAY"));
    Window   w = DefaultRootWindow(d);
    XRRScreenResources *xrrr = XRRGetScreenResources(d, w);
    XRRCrtcInfo *xrrci;
    int i;
    int ncrtc = xrrr->ncrtc;
    for (i = 0; i < ncrtc; ++i) {
        xrrci = XRRGetCrtcInfo(d, xrrr, xrrr->crtcs[i]);
        printf("%dx%d+%d+%d\n", xrrci->width, xrrci->height, xrrci->x, xrrci->y);
        XRRFreeCrtcInfo(xrrci);
    }
    XRRFreeScreenResources(xrrr);
    return 0;
}

Принятый ответ у меня не сработал. Я нашел правильный способ сделать это:

#include <stdio.h>
#include <stdlib.h>
#include <X11/Xlib.h>
#include <X11/extensions/Xrandr.h>

int main(void) {
    Display *display = XOpenDisplay(NULL);

    if (NULL == display) {
        perror("No DISPLAY in environment!");
        exit(EXIT_FAILURE);
    }

    Window window = DefaultRootWindow(display);
    XRRScreenResources *screenr = XRRGetScreenResources(display, window);

    // This is the key right here. Use XRRScreenResources::noutput
    int output = screenr->noutput;

    for (int i = 0; i < output; ++i) {
        XRROutputInfo* out_info = XRRGetOutputInfo(display, screenr, screenr->outputs[i]);

        if (NULL != out_info && out_info->connection == RR_Connected) {
            XRRCrtcInfo* crt_info = XRRGetCrtcInfo(display, screenr, out_info->crtc);
            printf("%s\t%dx%d+%d+%d\n", out_info->name, 
                                        crt_info->width, 
                                        crt_info->height,
                                        crt_info->x,
                                        crt_info->y);
            XRRFreeCrtcInfo(crt_info);
        }

        XRRFreeOutputInfo(out_info);
    }

    XRRFreeScreenResources(screenr);
    XCloseDisplay(display);

    return 0;
}

Как я сказал в комментарии к коду, трюк заключается в использовании XRRScreenResources::noutput вместо XRRScreenResources::ncrtc

Другие вопросы по тегам