Передача выбора буфера обмена не работает

Я переносил свою систему на X11, и у меня возникла проблема с копированием из буфера обмена (вставка уже работает). Я следил за этим. Через некоторое время я заметил, что его пример тоже не работает. Проблема в том, что когда я пытаюсь вставить его куда-нибудь, вместо XA_STRING запрошенный идентификатор атома равен 434. Я не мог найти, для чего этот атом. Когда я меняю XA_STRING на 434, я получаю другую ошибку. Ниже приведен код.

void SetClipboardText(const std::string &text) {
   XSetSelectionOwner (display, XA_CLIPBOARD, windowhandle, CurrentTime);
   copiedtext=text;
   XFlush(display);
   std::cout<<"COPIED"<<std::endl;
}

...
case SelectionRequest:
    XEvent respond;

    std::cout<<"SELTYPE: "<<event.xselectionrequest.target<<std::endl;

    if(event.xselectionrequest.target==XA_STRING && 
        event.xselectionrequest.selection==XA_CLIPBOARD) {
        std::cout<<"REQUESTED"<<std::endl;

        XChangeProperty (display,
            event.xselectionrequest.requestor,
            event.xselectionrequest.property,
            XA_STRING,
            copiedtext.length(),
            PropModeReplace,
            (unsigned char*) copiedtext.c_str(),
            copiedtext.length()
        );
        respond.xselection.property=event.xselectionrequest.property;
    }
    else {
        respond.xselection.property= None;
    }
    respond.xselection.type= SelectionNotify;
    respond.xselection.display= event.xselectionrequest.display;
    respond.xselection.requestor= event.xselectionrequest.requestor;
    respond.xselection.selection=event.xselectionrequest.selection;
    respond.xselection.target= event.xselectionrequest.target;
    respond.xselection.time = event.xselectionrequest.time;
    XSendEvent (display, event.xselectionrequest.requestor,0,0,&respond);
    XFlush (display);
    break;

Ошибка, которую я получил при замене XA_STRING на 434:

X Error of failed request:  BadValue (integer parameter out of range for operation)
  Major opcode of failed request:  18 (X_ChangeProperty)
  Value in failed request:  0x4
  Serial number of failed request:  33
  Current serial number in output stream:  36

Если уместно, я использую KDE 4.8, а klipper в настоящее время закрыт.

EDIT: пример с веб-сайта.

#include <X11/Xlib.h> 
#include <X11/Xatom.h>
#include <assert.h>   
#include <unistd.h>   
#include <stdio.h>
#include <stdlib.h>

main()
{
Display *dpy = XOpenDisplay(NULL);
assert(dpy);
Window w = XCreateSimpleWindow(dpy, DefaultRootWindow(dpy), 0, 0, 
                 200, 100, 0, 0, 0);
XSelectInput(dpy, w, StructureNotifyMask);
XMapWindow(dpy, w);
XSelectionRequestEvent *req;
XEvent e, respond;
for(;;) {
    XNextEvent(dpy, &e);
    if (e.type == MapNotify) break;
}
XFlush(dpy);
//
Atom a1, a2, a3, type;
XSelectInput(dpy, w, StructureNotifyMask+ExposureMask);
int format, result;
unsigned long len, bytes_left, dummy;
unsigned char *data;
Window Sown;
for (int ii = 0; ii < 50; ii++) {
    XSetSelectionOwner (dpy, XA_PRIMARY, w, CurrentTime);
    XFlush (dpy);
    XNextEvent (dpy, &e);
    if (e.type == SelectionRequest)
    //
    // Somebody wants our data
    //
    {
        req=&(e.xselectionrequest);
        printf ("Selection Request from Mr %i I am %i\n",
            (int)e.xselection.requestor, (int)w);
        printf ("prop:%i tar:%i sel:%i\n", req->property,
            req->target, req->selection);
        if (req->target == XA_STRING)
        {
            XChangeProperty (dpy,
                req->requestor,
                req->property,
                XA_STRING,
                8,
                PropModeReplace,
                (unsigned char*) "It Works",
                8);
            respond.xselection.property=req->property;
        }
        else // Strings only please
        {
            printf ("No String %i\n",
                (int)req->target);
            respond.xselection.property= None;
        }
        respond.xselection.type= SelectionNotify;
        respond.xselection.display= req->display;
        respond.xselection.requestor= req->requestor;
        respond.xselection.selection=req->selection;
        respond.xselection.target= req->target;
        respond.xselection.time = req->time;
        XSendEvent (dpy, req->requestor,0,0,&respond);
        XFlush (dpy);
    }
}
}

Скомпилируйте с помощью

gcc copytest.c -o copytest -std=c99 -lX11

person Cem Kalyoncu    schedule 13.05.2012    source источник
comment
Не могли бы вы опубликовать полный исходный код, который можно скомпилировать?   -  person Pavel Strakhov    schedule 13.05.2012
comment
@Riateche: это может быть проблемой, это довольно большая структура, содержащая сотни файлов. Однако я мог бы создать минимальный рабочий образец или даже отправить приложение, которое я использовал в качестве эталона. Он тоже не работает.   -  person Cem Kalyoncu    schedule 13.05.2012
comment
У меня этот пример работает на Ubuntu 12.04 с Unity. Я не знаю, что может вызвать эти ошибки.   -  person Pavel Strakhov    schedule 13.05.2012
comment
Кажется, ошибка с 434 не связана, я работал над первоначальным приложением. И кажется, что вставка в gedit работает, а вставка в Kwrite или любое приложение kde — нет.   -  person Cem Kalyoncu    schedule 13.05.2012
comment
@Riateche: я нашел, что такое 434, это список целей. Кажется, что KDE требует своего рода целевой список, и мы также должны его реализовать. Я нашел образец здесь: mail-archive.com/cygwin [email protected]/msg00897.html   -  person Cem Kalyoncu    schedule 13.05.2012
comment
Другая ошибка — это формат, он должен быть фиксированным 8.   -  person Cem Kalyoncu    schedule 13.05.2012
comment
Для этого вам лучше использовать Qt или Gtk. Оба имеют обширную поддержку буфера обмена. Если вы настаиваете на кодировании непосредственно в X11 (что, ИМХО, является потерей вашего времени), изучите и следуйте EWMH и ICCCM   -  person Basile Starynkevitch    schedule 20.09.2014
comment
Моя система представляет собой библиотеку, очень похожую на Qt или GTK.   -  person Cem Kalyoncu    schedule 21.09.2014


Ответы (2)


Для работы с приложениями KDE требуется следующее.

else if(event.xselectionrequest.target==XA_TARGETS && 
  event.xselectionrequest.selection==XA_CLIPBOARD) {
    Atom supported[]={XA_STRING};
    XChangeProperty (display,
        event.xselectionrequest.requestor,
        event.xselectionrequest.property,
        XA_TARGETS,
        8,
        PropModeReplace,
        (unsigned char *)(&supported),
        sizeof(supported)
    );
}
person Cem Kalyoncu    schedule 13.05.2012

Вы передали copiedtext.length() для параметра форматирования для XChangeProperty; это не верно.

Вы должны передать 8 для char* и 16 для short* и 32 для int*.

person user4060853    schedule 20.09.2014
comment
Согласно man XChangeProperty 32 соответствует long *, а не int *: Если указан формат 8, данные свойства должны быть массивом char. Если задан формат 16, данные свойства должны быть коротким массивом. Если задан формат 32, данные свойства должны быть длинным массивом. - person uli42; 17.11.2019