mirror of https://github.com/AxioDL/boo.git
337 lines
10 KiB
C++
337 lines
10 KiB
C++
#ifndef APPLICATION_UNIX_CPP
|
|
#error This file may only be included from CApplicationUnix.cpp
|
|
#endif
|
|
|
|
#include "boo/IApplication.hpp"
|
|
|
|
#define explicit explicit_c
|
|
#include <xcb/xcb.h>
|
|
#include <xcb/xcb_event.h>
|
|
#include <xkbcommon/xkbcommon-x11.h>
|
|
#include <xcb/xkb.h>
|
|
#include <xcb/xinput.h>
|
|
#include <xcb/glx.h>
|
|
#undef explicit
|
|
|
|
#include <X11/Xlib.h>
|
|
#include <GL/glx.h>
|
|
#include <GL/glxext.h>
|
|
|
|
#include <dbus/dbus.h>
|
|
DBusConnection* registerDBus(const char* appName, bool& isFirst);
|
|
|
|
#include <sys/param.h>
|
|
|
|
namespace boo
|
|
{
|
|
|
|
PFNGLXGETVIDEOSYNCSGIPROC FglXGetVideoSyncSGI = nullptr;
|
|
PFNGLXWAITVIDEOSYNCSGIPROC FglXWaitVideoSyncSGI = nullptr;
|
|
|
|
int XCB_GLX_EVENT_BASE = 0;
|
|
int XINPUT_OPCODE = 0;
|
|
|
|
static xcb_window_t getWindowOfEvent(xcb_generic_event_t* event, bool& windowEvent)
|
|
{
|
|
switch (XCB_EVENT_RESPONSE_TYPE(event))
|
|
{
|
|
case XCB_CLIENT_MESSAGE:
|
|
{
|
|
xcb_client_message_event_t* ev = (xcb_client_message_event_t*)event;
|
|
windowEvent = true;
|
|
return ev->window;
|
|
}
|
|
case XCB_EXPOSE:
|
|
{
|
|
xcb_expose_event_t* ev = (xcb_expose_event_t*)event;
|
|
windowEvent = true;
|
|
return ev->window;
|
|
}
|
|
case XCB_CONFIGURE_NOTIFY:
|
|
{
|
|
xcb_configure_notify_event_t* ev = (xcb_configure_notify_event_t*)event;
|
|
windowEvent = true;
|
|
return ev->window;
|
|
}
|
|
case XCB_KEY_PRESS:
|
|
{
|
|
xcb_key_press_event_t* ev = (xcb_key_press_event_t*)event;
|
|
windowEvent = true;
|
|
return ev->event;
|
|
}
|
|
case XCB_KEY_RELEASE:
|
|
{
|
|
xcb_key_release_event_t* ev = (xcb_key_release_event_t*)event;
|
|
windowEvent = true;
|
|
return ev->event;
|
|
}
|
|
case XCB_BUTTON_PRESS:
|
|
{
|
|
xcb_button_press_event_t* ev = (xcb_button_press_event_t*)event;
|
|
windowEvent = true;
|
|
return ev->event;
|
|
}
|
|
case XCB_BUTTON_RELEASE:
|
|
{
|
|
xcb_button_release_event_t* ev = (xcb_button_release_event_t*)event;
|
|
windowEvent = true;
|
|
return ev->event;
|
|
}
|
|
case XCB_MOTION_NOTIFY:
|
|
{
|
|
xcb_motion_notify_event_t* ev = (xcb_motion_notify_event_t*)event;
|
|
windowEvent = true;
|
|
return ev->event;
|
|
}
|
|
case XCB_GE_GENERIC:
|
|
{
|
|
xcb_ge_event_t* gev = (xcb_ge_event_t*)event;
|
|
if (gev->pad0 == XINPUT_OPCODE)
|
|
{
|
|
switch (gev->event_type)
|
|
{
|
|
case XCB_INPUT_MOTION:
|
|
{
|
|
xcb_input_motion_event_t* ev = (xcb_input_motion_event_t*)event;
|
|
windowEvent = true;
|
|
return ev->event;
|
|
}
|
|
case XCB_INPUT_TOUCH_BEGIN:
|
|
{
|
|
xcb_input_touch_begin_event_t* ev = (xcb_input_touch_begin_event_t*)event;
|
|
windowEvent = true;
|
|
return ev->event;
|
|
}
|
|
case XCB_INPUT_TOUCH_UPDATE:
|
|
{
|
|
xcb_input_touch_update_event_t* ev = (xcb_input_touch_update_event_t*)event;
|
|
windowEvent = true;
|
|
return ev->event;
|
|
}
|
|
case XCB_INPUT_TOUCH_END:
|
|
{
|
|
xcb_input_touch_end_event_t* ev = (xcb_input_touch_end_event_t*)event;
|
|
windowEvent = true;
|
|
return ev->event;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
windowEvent = false;
|
|
return 0;
|
|
}
|
|
|
|
IWindow* _CWindowXCBNew(const std::string& title, xcb_connection_t* conn);
|
|
|
|
class ApplicationXCB final : public IApplication
|
|
{
|
|
IApplicationCallback& m_callback;
|
|
const std::string m_uniqueName;
|
|
const std::string m_friendlyName;
|
|
const std::string m_pname;
|
|
const std::vector<std::string> m_args;
|
|
|
|
/* DBus single-instance */
|
|
bool m_singleInstance;
|
|
DBusConnection* m_dbus = NULL;
|
|
|
|
/* All windows */
|
|
std::unordered_map<xcb_window_t, IWindow*> m_windows;
|
|
|
|
xcb_connection_t* m_xcbConn = NULL;
|
|
int m_xcbFd, m_dbusFd, m_maxFd;
|
|
|
|
void _deletedWindow(IWindow* window)
|
|
{
|
|
m_windows.erase((xcb_window_t)window->getPlatformHandle());
|
|
}
|
|
|
|
public:
|
|
ApplicationXCB(IApplicationCallback& callback,
|
|
const std::string& uniqueName,
|
|
const std::string& friendlyName,
|
|
const std::string& pname,
|
|
const std::vector<std::string>& args,
|
|
bool singleInstance)
|
|
: m_callback(callback),
|
|
m_uniqueName(uniqueName),
|
|
m_friendlyName(friendlyName),
|
|
m_pname(pname),
|
|
m_args(args),
|
|
m_singleInstance(singleInstance)
|
|
{
|
|
/* DBus single instance registration */
|
|
bool isFirst;
|
|
m_dbus = registerDBus(uniqueName.c_str(), isFirst);
|
|
if (m_singleInstance)
|
|
{
|
|
if (!isFirst)
|
|
{
|
|
/* This is a duplicate instance, send signal and return */
|
|
if (args.size())
|
|
{
|
|
/* create a signal & check for errors */
|
|
DBusMessage*
|
|
msg = dbus_message_new_signal("/boo/signal/FileHandler",
|
|
"boo.signal.FileHandling",
|
|
"Open");
|
|
|
|
/* append arguments onto signal */
|
|
DBusMessageIter argsIter;
|
|
dbus_message_iter_init_append(msg, &argsIter);
|
|
for (const std::string& arg : args)
|
|
{
|
|
const char* sigvalue = arg.c_str();
|
|
dbus_message_iter_append_basic(&argsIter, DBUS_TYPE_STRING, &sigvalue);
|
|
}
|
|
|
|
/* send the message and flush the connection */
|
|
dbus_uint32_t serial;
|
|
dbus_connection_send(m_dbus, msg, &serial);
|
|
dbus_connection_flush(m_dbus);
|
|
dbus_message_unref(msg);
|
|
}
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
/* This is the first instance, register for signal */
|
|
// add a rule for which messages we want to see
|
|
DBusError err = {};
|
|
dbus_bus_add_match(m_dbus, "type='signal',interface='boo.signal.FileHandling'", &err);
|
|
dbus_connection_flush(m_dbus);
|
|
}
|
|
}
|
|
|
|
/* Open X connection */
|
|
m_xcbConn = xcb_connect(NULL, NULL);
|
|
|
|
/* GLX extension data */
|
|
const xcb_query_extension_reply_t* glxExtData =
|
|
xcb_get_extension_data(m_xcbConn, &xcb_glx_id);
|
|
XCB_GLX_EVENT_BASE = glxExtData->first_event;
|
|
FglXGetVideoSyncSGI = PFNGLXGETVIDEOSYNCSGIPROC(glXGetProcAddress((GLubyte*)"glXGetVideoSyncSGI"));
|
|
FglXWaitVideoSyncSGI = PFNGLXWAITVIDEOSYNCSGIPROC(glXGetProcAddress((GLubyte*)"glXWaitVideoSyncSGI"));
|
|
|
|
/* The xkb extension requests that the X server does not
|
|
* send repeated keydown events when a key is held */
|
|
xkb_x11_setup_xkb_extension(m_xcbConn,
|
|
XKB_X11_MIN_MAJOR_XKB_VERSION,
|
|
XKB_X11_MIN_MINOR_XKB_VERSION,
|
|
XKB_X11_SETUP_XKB_EXTENSION_NO_FLAGS,
|
|
NULL, NULL, NULL, NULL);
|
|
xcb_xkb_per_client_flags(m_xcbConn, XCB_XKB_ID_USE_CORE_KBD,
|
|
XCB_XKB_PER_CLIENT_FLAG_DETECTABLE_AUTO_REPEAT,
|
|
XCB_XKB_PER_CLIENT_FLAG_DETECTABLE_AUTO_REPEAT, 0, 0, 0);
|
|
|
|
/* Xinput major opcode */
|
|
const xcb_query_extension_reply_t* xiReply =
|
|
xcb_get_extension_data(m_xcbConn, &xcb_input_id);
|
|
XINPUT_OPCODE = xiReply->major_opcode;
|
|
|
|
/* Get file descriptors of xcb and dbus interfaces */
|
|
m_xcbFd = xcb_get_file_descriptor(m_xcbConn);
|
|
dbus_connection_get_unix_fd(m_dbus, &m_dbusFd);
|
|
m_maxFd = MAX(m_xcbFd, m_dbusFd);
|
|
|
|
xcb_flush(m_xcbConn);
|
|
}
|
|
|
|
~ApplicationXCB()
|
|
{
|
|
xcb_disconnect(m_xcbConn);
|
|
}
|
|
|
|
EPlatformType getPlatformType() const
|
|
{
|
|
return PLAT_XCB;
|
|
}
|
|
|
|
void pump()
|
|
{
|
|
if (!m_xcbConn)
|
|
return;
|
|
|
|
xcb_generic_event_t* event;
|
|
fd_set fds;
|
|
FD_ZERO(&fds);
|
|
FD_SET(m_xcbFd, &fds);
|
|
FD_SET(m_dbusFd, &fds);
|
|
select(m_maxFd+1, &fds, NULL, NULL, NULL);
|
|
|
|
if (FD_ISSET(m_xcbFd, &fds))
|
|
{
|
|
event = xcb_poll_for_event(m_xcbConn);
|
|
if (event)
|
|
{
|
|
bool windowEvent;
|
|
xcb_window_t evWindow = getWindowOfEvent(event, windowEvent);
|
|
//fprintf(stderr, "EVENT %d\n", XCB_EVENT_RESPONSE_TYPE(event));
|
|
if (windowEvent)
|
|
{
|
|
auto window = m_windows.find(evWindow);
|
|
if (window != m_windows.end())
|
|
window->second->_incomingEvent(event);
|
|
}
|
|
free(event);
|
|
}
|
|
}
|
|
|
|
if (FD_ISSET(m_dbusFd, &fds))
|
|
{
|
|
DBusMessage* msg;
|
|
dbus_connection_read_write(m_dbus, 0);
|
|
while ((msg = dbus_connection_pop_message(m_dbus)))
|
|
{
|
|
/* check if the message is a signal from the correct interface and with the correct name */
|
|
if (dbus_message_is_signal(msg, "boo.signal.FileHandling", "Open"))
|
|
{
|
|
/* read the parameters */
|
|
std::vector<std::string> paths;
|
|
DBusMessageIter iter;
|
|
dbus_message_iter_init(msg, &iter);
|
|
while (dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_INVALID)
|
|
{
|
|
const char* argVal;
|
|
dbus_message_iter_get_basic(&iter, &argVal);
|
|
paths.push_back(argVal);
|
|
dbus_message_iter_next(&iter);
|
|
}
|
|
m_callback.appFilesOpen(this, paths);
|
|
}
|
|
dbus_message_unref(msg);
|
|
}
|
|
}
|
|
}
|
|
|
|
const std::string& getUniqueName() const
|
|
{
|
|
return m_uniqueName;
|
|
}
|
|
|
|
const std::string& getFriendlyName() const
|
|
{
|
|
return m_friendlyName;
|
|
}
|
|
|
|
const std::string& getProcessName() const
|
|
{
|
|
return m_pname;
|
|
}
|
|
|
|
const std::vector<std::string>& getArgs() const
|
|
{
|
|
return m_args;
|
|
}
|
|
|
|
IWindow* newWindow(const std::string& title)
|
|
{
|
|
IWindow* newWindow = _CWindowXCBNew(title, m_xcbConn);
|
|
m_windows[(xcb_window_t)newWindow->getPlatformHandle()] = newWindow;
|
|
return newWindow;
|
|
}
|
|
};
|
|
|
|
}
|