mirror of
https://github.com/edeproject/ede.git
synced 2023-08-10 21:13:03 +03:00
New settings manager based on XSettingsManager. Also it have
ability to serialize data at the shutdown and load it at the startup.
This commit is contained in:
parent
15b342009e
commit
667ce4722f
@ -213,8 +213,7 @@ EvokeService::~EvokeService() {
|
||||
if(logfile)
|
||||
delete logfile;
|
||||
|
||||
if(xsm)
|
||||
delete xsm;
|
||||
stop_xsettings_manager();
|
||||
|
||||
if(lockfile) {
|
||||
edelib::file_remove(lockfile);
|
||||
@ -464,7 +463,7 @@ void EvokeService::init_autostart(bool safe) {
|
||||
void EvokeService::init_xsettings_manager(void) {
|
||||
xsm = new Xsm;
|
||||
|
||||
if(xsm->is_running()) {
|
||||
if(Xsm::manager_running(fl_display, fl_screen)) {
|
||||
int ret = edelib::ask(_("XSETTINGS manager already running on this screen. Would you like to replace it?"));
|
||||
if(ret < 1) {
|
||||
stop_xsettings_manager();
|
||||
@ -474,23 +473,22 @@ void EvokeService::init_xsettings_manager(void) {
|
||||
}
|
||||
|
||||
do_it:
|
||||
if(!xsm->init()) {
|
||||
if(!xsm->init(fl_display, fl_screen)) {
|
||||
edelib::alert(_("Unable to load XSETTINGS manager properly"));
|
||||
stop_xsettings_manager();
|
||||
}
|
||||
|
||||
if(!xsm) return;
|
||||
|
||||
/* testing code
|
||||
* FIXME: move this to outside client
|
||||
*/
|
||||
xsm->set_int("Net/DoubleClickTime", 234);
|
||||
xsm->set_color("Net/Background/Normal", 34, 45, 23, 0);
|
||||
xsm->set_string("Net/UserName", "John Foo");
|
||||
if(xsm->load_serialized("ede-settings.xml"))
|
||||
xsm->notify();
|
||||
}
|
||||
|
||||
void EvokeService::stop_xsettings_manager(void) {
|
||||
if(!xsm)
|
||||
return;
|
||||
|
||||
xsm->save_serialized("ede-settings.xml");
|
||||
delete xsm;
|
||||
xsm = NULL;
|
||||
}
|
||||
@ -807,7 +805,7 @@ bool EvokeService::find_and_unregister_process(pid_t pid, EvokeProcess& pc) {
|
||||
int EvokeService::handle(const XEvent* xev) {
|
||||
EVOKE_LOG("Got event %i\n", xev->type);
|
||||
|
||||
if(xsm && xsm->should_quit(xev)) {
|
||||
if(xsm && xsm->should_terminate(xev)) {
|
||||
EVOKE_LOG("XSETTINGS manager shutdown\n");
|
||||
stop_xsettings_manager();
|
||||
// return 1;
|
||||
|
@ -27,5 +27,6 @@ LinkAgainst evoke : -lao -lvorbis -lvorbisfile ;
|
||||
|
||||
EdeProgram evoke : $(SOURCE) ;
|
||||
FltkProgramBare test/evoke_test : test/evoke_test.cpp ;
|
||||
FltkProgramBare test/stress_test : test/stress_test.cpp ;
|
||||
#TranslationStrings locale : $(SOURCE) ;
|
||||
EdeManual doc/evoke.txt ;
|
||||
|
501
evoke/Xsm.cpp
501
evoke/Xsm.cpp
@ -11,395 +11,162 @@
|
||||
*/
|
||||
|
||||
#include "Xsm.h"
|
||||
|
||||
#include <edelib/Debug.h>
|
||||
#include <edelib/List.h>
|
||||
#include <edelib/TiXml.h>
|
||||
#include <edelib/File.h>
|
||||
#include <string.h>
|
||||
#include <limits.h>
|
||||
|
||||
#include <FL/x.h>
|
||||
#include <X11/Xmd.h> // CARD16, CARD32
|
||||
|
||||
#include <stdio.h> // snprintf
|
||||
#include <stdlib.h> // free
|
||||
#include <string.h> // strdup, strcmp
|
||||
|
||||
#define XSETTINGS_PAD(n, p) ((n + p - 1) & (~(p - 1)))
|
||||
|
||||
typedef edelib::list<XsettingsSetting*> XsettingsList;
|
||||
typedef edelib::list<XsettingsSetting*>::iterator XsettingsListIter;
|
||||
|
||||
struct XsmData {
|
||||
Window window;
|
||||
Atom manager_atom;
|
||||
Atom selection_atom;
|
||||
Atom xsettings_atom;
|
||||
unsigned long serial;
|
||||
|
||||
XsettingsList list;
|
||||
};
|
||||
|
||||
struct XsettingsBuffer {
|
||||
unsigned char* content;
|
||||
unsigned char* pos;
|
||||
int len;
|
||||
};
|
||||
|
||||
struct XsettingsColor {
|
||||
unsigned char red, green, blue, alpha;
|
||||
};
|
||||
|
||||
struct XsettingsSetting {
|
||||
char* name;
|
||||
XsettingsType type;
|
||||
|
||||
union {
|
||||
int v_int;
|
||||
char* v_string;
|
||||
XsettingsColor v_color;
|
||||
} data;
|
||||
|
||||
unsigned long last_change_serial;
|
||||
};
|
||||
|
||||
struct TimeStampInfo {
|
||||
Window window;
|
||||
Atom timestamp_prop_atom;
|
||||
};
|
||||
|
||||
// Bool (X type) is used since this function is going to XIfEvent()
|
||||
Bool timestamp_predicate(Display* dpy, XEvent* xev, XPointer arg) {
|
||||
TimeStampInfo* info = (TimeStampInfo*)arg;
|
||||
|
||||
if(xev->type == PropertyNotify &&
|
||||
xev->xproperty.window == info->window &&
|
||||
xev->xproperty.atom == info->timestamp_prop_atom) {
|
||||
int ignore_xerrors(Display* display, XErrorEvent* xev) {
|
||||
return True;
|
||||
}
|
||||
|
||||
return False;
|
||||
}
|
||||
|
||||
Time get_server_time(Display* dpy, Window win) {
|
||||
unsigned char c = 'a';
|
||||
TimeStampInfo info;
|
||||
XEvent xev;
|
||||
|
||||
info.timestamp_prop_atom = XInternAtom(dpy, "_TIMESTAMP_PROP", False);
|
||||
info.window = win;
|
||||
|
||||
XChangeProperty(dpy, win, info.timestamp_prop_atom, info.timestamp_prop_atom,
|
||||
8, PropModeReplace, &c, 1);
|
||||
|
||||
XIfEvent(dpy, &xev, timestamp_predicate, (XPointer)&info);
|
||||
|
||||
return xev.xproperty.time;
|
||||
}
|
||||
|
||||
char settings_byte_order(void) {
|
||||
CARD32 myint = 0x01020304;
|
||||
return (*(char*)&myint == 1) ? MSBFirst : LSBFirst;
|
||||
}
|
||||
|
||||
bool settings_equal(const XsettingsSetting* s1, const XsettingsSetting* s2) {
|
||||
EASSERT(s1 != NULL && s2 != NULL);
|
||||
|
||||
if(s1->type != s2->type)
|
||||
return false;
|
||||
if(strcmp(s1->name, s2->name) != 0)
|
||||
return false;
|
||||
|
||||
switch(s1->type) {
|
||||
case XS_TYPE_INT:
|
||||
return s1->data.v_int == s2->data.v_int;
|
||||
case XS_TYPE_COLOR:
|
||||
return (s1->data.v_color.red == s2->data.v_color.red) &&
|
||||
(s1->data.v_color.green == s2->data.v_color.green) &&
|
||||
(s1->data.v_color.blue == s2->data.v_color.blue) &&
|
||||
(s1->data.v_color.alpha == s2->data.v_color.alpha);
|
||||
case XS_TYPE_STRING:
|
||||
return (strcmp(s1->data.v_string, s2->data.v_string) == 0);
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
int setting_len(const XsettingsSetting* setting) {
|
||||
int len = 8; // type + pad + name-len + last-change-serial
|
||||
len += XSETTINGS_PAD(strlen(setting->name), 4);
|
||||
|
||||
switch(setting->type) {
|
||||
case XS_TYPE_INT:
|
||||
len += 4;
|
||||
break;
|
||||
case XS_TYPE_COLOR:
|
||||
len += 8;
|
||||
break;
|
||||
case XS_TYPE_STRING:
|
||||
len += 4 + XSETTINGS_PAD(strlen(setting->data.v_string), 4);
|
||||
break;
|
||||
}
|
||||
|
||||
return len;
|
||||
}
|
||||
|
||||
void setting_store(const XsettingsSetting* setting, XsettingsBuffer* buffer) {
|
||||
int len, str_len;
|
||||
|
||||
*(buffer->pos++) = setting->type;
|
||||
*(buffer->pos++) = 0;
|
||||
|
||||
str_len = strlen(setting->name);
|
||||
*(CARD16*)(buffer->pos) = str_len;
|
||||
buffer->pos += 2;
|
||||
|
||||
len = XSETTINGS_PAD(str_len, 4);
|
||||
memcpy(buffer->pos, setting->name, str_len);
|
||||
len -= str_len;
|
||||
buffer->pos += str_len;
|
||||
|
||||
for(; len > 0; len--)
|
||||
*(buffer->pos++) = 0;
|
||||
|
||||
*(CARD32*)(buffer->pos) = setting->last_change_serial;
|
||||
buffer->pos += 4;
|
||||
|
||||
switch(setting->type) {
|
||||
case XS_TYPE_INT:
|
||||
*(CARD32*)(buffer->pos) = setting->data.v_int;
|
||||
buffer->pos += 4;
|
||||
break;
|
||||
case XS_TYPE_COLOR:
|
||||
*(CARD16*)(buffer->pos) = setting->data.v_color.red;
|
||||
*(CARD16*)(buffer->pos + 2) = setting->data.v_color.green;
|
||||
*(CARD16*)(buffer->pos + 4) = setting->data.v_color.blue;
|
||||
*(CARD16*)(buffer->pos + 6) = setting->data.v_color.alpha;
|
||||
buffer->pos += 8;
|
||||
break;
|
||||
case XS_TYPE_STRING:
|
||||
str_len = strlen(setting->data.v_string);
|
||||
*(CARD32*)(buffer->pos) = str_len;
|
||||
buffer->pos += 4;
|
||||
|
||||
len = XSETTINGS_PAD(str_len, 4);
|
||||
memcpy(buffer->pos, setting->data.v_string, str_len);
|
||||
len -= str_len;
|
||||
buffer->pos += str_len;
|
||||
|
||||
for(; len > 0; len--)
|
||||
*(buffer->pos++) = 0;
|
||||
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
Xsm::Xsm() : data(NULL) { }
|
||||
Xsm::Xsm() { }
|
||||
|
||||
Xsm::~Xsm() {
|
||||
if(data) {
|
||||
XDestroyWindow(fl_display, data->window);
|
||||
XsettingsList& lst = data->list;
|
||||
|
||||
if(!lst.empty()) {
|
||||
XsettingsListIter it = lst.begin(), it_end = lst.end();
|
||||
for(; it != it_end; ++it)
|
||||
delete_setting(*it);
|
||||
|
||||
lst.clear();
|
||||
EDEBUG("Xsm::~Xsm()\n");
|
||||
}
|
||||
|
||||
delete data;
|
||||
}
|
||||
|
||||
puts("Xsm::~Xsm()");
|
||||
}
|
||||
|
||||
bool Xsm::is_running(void) {
|
||||
char buff[256];
|
||||
|
||||
snprintf(buff, sizeof(buff)-1, "_XSETTINGS_S%d", fl_screen);
|
||||
Atom selection = XInternAtom(fl_display, buff, False);
|
||||
|
||||
if(XGetSelectionOwner(fl_display, selection))
|
||||
return true;
|
||||
bool Xsm::load_serialized(const char* file) {
|
||||
TiXmlDocument doc(file);
|
||||
if(!doc.LoadFile())
|
||||
return false;
|
||||
|
||||
const char* name = NULL, *type = NULL;
|
||||
const char* v_string = NULL;
|
||||
int v_int = 0;
|
||||
int v_red = 0, v_green = 0, v_blue = 0, v_alpha = 0;
|
||||
int cmp = 0;
|
||||
|
||||
TiXmlNode* elem = doc.FirstChild("ede-settings");
|
||||
if(!elem)
|
||||
return false;
|
||||
|
||||
for(elem = elem->FirstChildElement(); elem; elem = elem->NextSibling()) {
|
||||
if(strcmp(elem->Value(), "setting") != 0) {
|
||||
EWARNING(ESTRLOC ": Got unknown child in 'ede-setting' %s\n", elem->Value());
|
||||
continue;
|
||||
}
|
||||
|
||||
bool Xsm::init(void) {
|
||||
char buff[256];
|
||||
name = elem->ToElement()->Attribute("name");
|
||||
if(!name) {
|
||||
EWARNING(ESTRLOC ": Missing name key\n");
|
||||
continue;
|
||||
}
|
||||
|
||||
data = new XsmData;
|
||||
type = elem->ToElement()->Attribute("type");
|
||||
if(!type) {
|
||||
EWARNING(ESTRLOC ": Missing type key\n");
|
||||
continue;
|
||||
}
|
||||
|
||||
snprintf(buff, sizeof(buff)-1, "_XSETTINGS_S%d", fl_screen);
|
||||
data->selection_atom = XInternAtom(fl_display, buff, False);
|
||||
data->xsettings_atom = XInternAtom(fl_display, "_XSETTINGS_SETTINGS", False);
|
||||
data->manager_atom = XInternAtom(fl_display, "MANAGER", False);
|
||||
if(strcmp(type, "int") == 0)
|
||||
cmp = 1;
|
||||
else if(strcmp(type, "string") == 0)
|
||||
cmp = 2;
|
||||
else if(strcmp(type, "color") == 0)
|
||||
cmp = 3;
|
||||
else {
|
||||
EWARNING(ESTRLOC ": Unknown type %s\n", type);
|
||||
continue;
|
||||
}
|
||||
|
||||
data->serial = 0;
|
||||
switch(cmp) {
|
||||
case 1:
|
||||
if(elem->ToElement()->QueryIntAttribute("value", &v_int) == TIXML_SUCCESS)
|
||||
set(name, v_int);
|
||||
else
|
||||
EWARNING(ESTRLOC ": Unable to query integer value\n");
|
||||
break;
|
||||
case 2:
|
||||
v_string = elem->ToElement()->Attribute("value");
|
||||
if(v_string)
|
||||
set(name, v_string);
|
||||
break;
|
||||
case 3:
|
||||
if((elem->ToElement()->QueryIntAttribute("red", &v_red) == TIXML_SUCCESS) &&
|
||||
(elem->ToElement()->QueryIntAttribute("green", &v_green) == TIXML_SUCCESS) &&
|
||||
(elem->ToElement()->QueryIntAttribute("blue", &v_blue) == TIXML_SUCCESS) &&
|
||||
(elem->ToElement()->QueryIntAttribute("alpha", &v_alpha) == TIXML_SUCCESS)) {
|
||||
set(name, v_red, v_green, v_blue, v_alpha);
|
||||
}
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
data->window = XCreateSimpleWindow(fl_display, RootWindow(fl_display, fl_screen),
|
||||
0, 0, 10, 10, 0, WhitePixel(fl_display, fl_screen), WhitePixel(fl_display, fl_screen));
|
||||
|
||||
XSelectInput(fl_display, data->window, PropertyChangeMask);
|
||||
Time timestamp = get_server_time(fl_display, data->window);
|
||||
|
||||
XSetSelectionOwner(fl_display, data->selection_atom, data->window, timestamp);
|
||||
|
||||
// check if we got ownership
|
||||
if(XGetSelectionOwner(fl_display, data->selection_atom) == data->window) {
|
||||
XClientMessageEvent xev;
|
||||
|
||||
xev.type = ClientMessage;
|
||||
xev.window = RootWindow(fl_display, fl_screen);
|
||||
xev.message_type = data->manager_atom;
|
||||
xev.format = 32;
|
||||
xev.data.l[0] = timestamp;
|
||||
xev.data.l[1] = data->selection_atom;
|
||||
xev.data.l[2] = data->window;
|
||||
xev.data.l[3] = 0; // manager specific data
|
||||
xev.data.l[4] = 0; // manager specific data
|
||||
|
||||
XSendEvent(fl_display, RootWindow(fl_display, fl_screen), False, StructureNotifyMask, (XEvent*)&xev);
|
||||
return true;
|
||||
}
|
||||
|
||||
bool Xsm::save_serialized(const char* file) {
|
||||
// FIXME: a lot of this code could be in edelib
|
||||
Atom type;
|
||||
int format;
|
||||
unsigned long n_items, bytes_after;
|
||||
unsigned char* data;
|
||||
int result;
|
||||
edelib::XSettingsList* settings = NULL, *iter = NULL;
|
||||
|
||||
int (*old_handler)(Display*, XErrorEvent*);
|
||||
|
||||
// possible ?
|
||||
if(!manager_data->manager_win)
|
||||
return false;
|
||||
|
||||
old_handler = XSetErrorHandler(ignore_xerrors);
|
||||
result = XGetWindowProperty(manager_data->display, manager_data->manager_win, manager_data->xsettings_atom,
|
||||
0, LONG_MAX, False, manager_data->xsettings_atom,
|
||||
&type, &format, &n_items, &bytes_after, (unsigned char**)&data);
|
||||
|
||||
XSetErrorHandler(old_handler);
|
||||
if(result == Success && type != None) {
|
||||
if(type != manager_data->xsettings_atom)
|
||||
EWARNING(ESTRLOC ": Invalid type for XSETTINGS property\n");
|
||||
else if(format != 8)
|
||||
EWARNING(ESTRLOC ": Invalid format for XSETTINGS property\n");
|
||||
else
|
||||
settings = edelib::xsettings_decode(data, n_items, NULL);
|
||||
XFree(data);
|
||||
}
|
||||
|
||||
if(!settings)
|
||||
return false;
|
||||
|
||||
edelib::File setting_file;
|
||||
if(!setting_file.open(file, edelib::FIO_WRITE)) {
|
||||
EWARNING(ESTRLOC ": Unable to write to %s\n", file);
|
||||
edelib::xsettings_list_free(settings);
|
||||
return false;
|
||||
}
|
||||
|
||||
bool Xsm::should_quit(const XEvent* xev) {
|
||||
EASSERT(data != NULL);
|
||||
setting_file.printf("<? xml version=\"1.0\" encoding=\"UTF-8\" ?>\n");
|
||||
setting_file.printf("<ede-settings>\n");
|
||||
|
||||
if(xev->xany.window == data->window &&
|
||||
xev->xany.type == SelectionClear &&
|
||||
xev->xselectionclear.selection == data->selection_atom) {
|
||||
iter = settings;
|
||||
while(iter) {
|
||||
setting_file.printf("\t<setting name=\"%s\" ", iter->setting->name);
|
||||
switch(iter->setting->type) {
|
||||
case edelib::XSETTINGS_TYPE_INT:
|
||||
setting_file.printf("type=\"int\" value=\"%i\" />\n", iter->setting->data.v_int);
|
||||
break;
|
||||
case edelib::XSETTINGS_TYPE_STRING:
|
||||
setting_file.printf("type=\"string\" value=\"%s\" />\n", iter->setting->data.v_string);
|
||||
break;
|
||||
case edelib::XSETTINGS_TYPE_COLOR:
|
||||
setting_file.printf("type=\"color\" red=\"%i\" green=\"%i\" blue=\"%i\" alpha=\"%i\" />\n",
|
||||
iter->setting->data.v_color.red,
|
||||
iter->setting->data.v_color.green,
|
||||
iter->setting->data.v_color.blue,
|
||||
iter->setting->data.v_color.alpha);
|
||||
break;
|
||||
}
|
||||
|
||||
iter = iter->next;
|
||||
}
|
||||
setting_file.printf("</ede-settings>\n");
|
||||
|
||||
setting_file.close();
|
||||
edelib::xsettings_list_free(settings);
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
void Xsm::set_setting(const XsettingsSetting* setting) {
|
||||
EASSERT(data != NULL);
|
||||
|
||||
XsettingsList& lst = data->list;
|
||||
XsettingsListIter it = lst.begin(), it_end = lst.end();
|
||||
|
||||
/*
|
||||
* Check if entry already exists. If setting with the same
|
||||
* name already exists, but with different values, that setting
|
||||
* will be completely replaced with new one.
|
||||
*/
|
||||
for(; it != it_end; ++it) {
|
||||
if(strcmp((*it)->name, setting->name) == 0) {
|
||||
if(settings_equal((*it), setting)) {
|
||||
return;
|
||||
} else {
|
||||
delete_setting(*it);
|
||||
lst.erase(it);
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
XsettingsSetting* new_setting = new XsettingsSetting;
|
||||
new_setting->name = strdup(setting->name);
|
||||
//new_setting->last_change_serial = setting->last_change_serial;
|
||||
new_setting->type = setting->type;
|
||||
|
||||
switch(new_setting->type) {
|
||||
case XS_TYPE_INT:
|
||||
new_setting->data.v_int = setting->data.v_int;
|
||||
break;
|
||||
case XS_TYPE_COLOR:
|
||||
new_setting->data.v_color.red = setting->data.v_color.red;
|
||||
new_setting->data.v_color.green = setting->data.v_color.green;
|
||||
new_setting->data.v_color.blue = setting->data.v_color.blue;
|
||||
new_setting->data.v_color.alpha = setting->data.v_color.alpha;
|
||||
break;
|
||||
case XS_TYPE_STRING:
|
||||
new_setting->data.v_string = strdup(setting->data.v_string);
|
||||
break;
|
||||
}
|
||||
|
||||
new_setting->last_change_serial = data->serial;
|
||||
lst.push_back(new_setting);
|
||||
}
|
||||
|
||||
void Xsm::delete_setting(XsettingsSetting* setting) {
|
||||
if(!setting)
|
||||
return;
|
||||
|
||||
if(setting->type == XS_TYPE_STRING)
|
||||
free(setting->data.v_string);
|
||||
free(setting->name);
|
||||
delete setting;
|
||||
setting = NULL;
|
||||
}
|
||||
|
||||
|
||||
void Xsm::set_int(const char* name, int val) {
|
||||
XsettingsSetting setting;
|
||||
|
||||
setting.name = (char*)name;
|
||||
setting.type = XS_TYPE_INT;
|
||||
setting.data.v_int = val;
|
||||
|
||||
set_setting(&setting);
|
||||
}
|
||||
|
||||
void Xsm::set_color(const char* name, unsigned char red, unsigned char green, unsigned char blue, unsigned char alpha) {
|
||||
XsettingsSetting setting;
|
||||
|
||||
setting.name = (char*)name;
|
||||
setting.type = XS_TYPE_COLOR;
|
||||
setting.data.v_color.red = red;
|
||||
setting.data.v_color.green = green;
|
||||
setting.data.v_color.blue = blue;
|
||||
setting.data.v_color.alpha = alpha;
|
||||
|
||||
set_setting(&setting);
|
||||
}
|
||||
|
||||
void Xsm::set_string(const char* name, const char* str) {
|
||||
XsettingsSetting setting;
|
||||
|
||||
setting.name = (char*)name;
|
||||
setting.type = XS_TYPE_STRING;
|
||||
setting.data.v_string = (char*)str;
|
||||
|
||||
set_setting(&setting);
|
||||
}
|
||||
|
||||
void Xsm::notify(void) {
|
||||
EASSERT(data != NULL);
|
||||
|
||||
int n_settings = 0;
|
||||
XsettingsBuffer buff;
|
||||
|
||||
buff.len = 12; // byte-order + pad + SERIAL + N_SETTINGS
|
||||
|
||||
XsettingsList& lst = data->list;
|
||||
XsettingsListIter it = lst.begin(), it_end = lst.end();
|
||||
|
||||
for(; it != it_end; ++it) {
|
||||
buff.len += setting_len(*it);
|
||||
n_settings++;
|
||||
}
|
||||
|
||||
buff.content = new unsigned char[buff.len];
|
||||
buff.pos = buff.content;
|
||||
|
||||
*buff.pos = settings_byte_order();
|
||||
|
||||
buff.pos += 4;
|
||||
*(CARD32*)buff.pos = data->serial++;
|
||||
buff.pos += 4;
|
||||
*(CARD32*)buff.pos = n_settings;
|
||||
buff.pos += 4;
|
||||
|
||||
for(it = lst.begin(); it != it_end; ++it)
|
||||
setting_store(*it, &buff);
|
||||
|
||||
XChangeProperty(fl_display, data->window, data->xsettings_atom, data->xsettings_atom,
|
||||
8, PropModeReplace, buff.content, buff.len);
|
||||
|
||||
delete [] buff.content;
|
||||
}
|
||||
|
47
evoke/Xsm.h
47
evoke/Xsm.h
@ -13,54 +13,17 @@
|
||||
#ifndef __XSM_H__
|
||||
#define __XSM_H__
|
||||
|
||||
#include <X11/Xlib.h> // XEvent
|
||||
#include <edelib/XSettingsManager.h>
|
||||
|
||||
/*
|
||||
* This is manager class for XSETTINGS protocol. XSETTINGS provides a mechanism
|
||||
* for applications to share simple configuration settings like background
|
||||
* colors no matter what toolkit is used. For now only gtk fully supports it
|
||||
* and support for ede apps is going to be added.
|
||||
*
|
||||
* The protocol (0.5 version) is described at:
|
||||
* http://standards.freedesktop.org/xsettings-spec/xsettings-spec-0.5.html
|
||||
*
|
||||
* This code is greatly based on xsettings referent implementation I found on
|
||||
* freedesktop.org cvs, since specs are very unclear about the details.
|
||||
* Author of referent implementation is Owen Taylor.
|
||||
* XSETTINGS manager with serialization capability.
|
||||
*/
|
||||
|
||||
struct XsmData;
|
||||
struct XsettingsSetting;
|
||||
struct XsettingsBuffer;
|
||||
|
||||
/*
|
||||
* Enum order must be exact on client side too, since via these values
|
||||
* client will try to decode settings.
|
||||
*/
|
||||
enum XsettingsType {
|
||||
XS_TYPE_INT = 0,
|
||||
XS_TYPE_STRING,
|
||||
XS_TYPE_COLOR
|
||||
};
|
||||
|
||||
class Xsm {
|
||||
private:
|
||||
XsmData* data;
|
||||
void set_setting(const XsettingsSetting* setting);
|
||||
void delete_setting(XsettingsSetting* setting);
|
||||
|
||||
class Xsm : public edelib::XSettingsManager {
|
||||
public:
|
||||
Xsm();
|
||||
~Xsm();
|
||||
bool is_running(void);
|
||||
bool init(void);
|
||||
bool should_quit(const XEvent* xev);
|
||||
|
||||
void set_int(const char* name, int val);
|
||||
void set_color(const char* name,
|
||||
unsigned char red, unsigned char green, unsigned char blue, unsigned char alpha);
|
||||
void set_string(const char* name, const char* str);
|
||||
void notify(void);
|
||||
bool load_serialized(const char* file);
|
||||
bool save_serialized(const char* file);
|
||||
};
|
||||
|
||||
#endif
|
||||
|
19
evoke/ede-settings.xml
Normal file
19
evoke/ede-settings.xml
Normal file
@ -0,0 +1,19 @@
|
||||
<? xml version="1.0" encoding="UTF-8" ?>
|
||||
<ede-settings>
|
||||
<setting name="Fltk/Background" type="color" red="186" green="182" blue="77" alpha="0" />
|
||||
<setting name="Fltk/Background2" type="color" red="34" green="42" blue="161" alpha="0" />
|
||||
<setting name="Gtk/CanChangeAccels" type="int" value="0" />
|
||||
<setting name="Gtk/CursorThemeName" type="string" value="default" />
|
||||
<setting name="Gtk/CursorThemeSize" type="int" value="16" />
|
||||
<setting name="Gtk/FontName" type="string" value="Sans 10" />
|
||||
<setting name="Gtk/KeyThemeName" type="string" value="Default" />
|
||||
<setting name="Gtk/ToolbarStyle" type="string" value="icons" />
|
||||
<setting name="Net/Background/Normal" type="color" red="186" green="45" blue="23" alpha="0" />
|
||||
<setting name="Net/CursorBlink" type="int" value="1" />
|
||||
<setting name="Net/CursorBlinkTime" type="int" value="500" />
|
||||
<setting name="Net/DndDragThreshold" type="int" value="8" />
|
||||
<setting name="Net/DoubleClickTime" type="int" value="300" />
|
||||
<setting name="Net/IconThemeName" type="string" value="Rodent" />
|
||||
<setting name="Net/ThemeName" type="string" value="Raleigh" />
|
||||
<setting name="Net/UserName" type="string" value="John Foo" />
|
||||
</ede-settings>
|
Loading…
Reference in New Issue
Block a user