Ok let's see:

* DND now should work reasonable (still not finished DND on icons)
* Added menus (icon specific, desktop)
* Icon renaming works too
* Some code to change icons fonts (unfinished)
* Desktop main instance will be now first created and then call rest of
  init data. This will allow init data to re-use desktop instance and peek
  it's own data (area() for example). Previously, calling such functions would
  crash app since constructor didn't finished with creation of desktop instance.
This commit is contained in:
Sanel Zukan 2007-07-02 10:28:18 +00:00
parent 4521b56042
commit adc3a8efcd
6 changed files with 339 additions and 94 deletions

View File

@ -17,10 +17,13 @@
#include <FL/Fl.h>
#include <FL/fl_draw.h>
#include <FL/Fl_Shared_Image.h>
#include <FL/Fl_Menu_Button.h>
#include <FL/x.h>
#include <FL/fl_ask.h>
#include <edelib/Debug.h>
#include <edelib/IconTheme.h>
#include <edelib/Nls.h>
#define USE_SHAPE 1
@ -38,6 +41,36 @@
// label offset from icon y()+h(), so selection box can be drawn nicely
#define LABEL_OFFSET 2
void rename_cb(Fl_Widget*, void* d);
Fl_Menu_Item icon_menu[] = {
{_(" &Open "), 0, 0},
{_(" &Rename "), 0, rename_cb, 0},
{_(" &Delete "), 0, 0, 0, FL_MENU_DIVIDER},
{_(" &Properties "), 0, 0},
{0}
};
Fl_Menu_Item icon_trash_menu[] = {
{_(" &Open "), 0, 0},
{_(" &Properties "), 0, 0, 0, FL_MENU_DIVIDER},
{_(" &Empty "), 0, 0},
{0}
};
void rename_cb(Fl_Widget*, void* d) {
DesktopIcon* di = (DesktopIcon*)d;
EASSERT(di != NULL);
const char* new_name = fl_input(_("New name"), di->label());
if(!new_name)
return;
if(new_name[0] == '\0')
return;
di->rename(new_name);
}
DesktopIcon::DesktopIcon(GlobalIconSettings* gs, IconSettings* is, int bg) :
Fl_Widget(is->x, is->y, ICONSIZE, ICONSIZE) {
@ -60,11 +93,24 @@ DesktopIcon::DesktopIcon(GlobalIconSettings* gs, IconSettings* is, int bg) :
settings->icon = is->icon;
settings->type = is->type;
settings->key_name= is->key_name;
settings->full_path = is->full_path;
// x,y are not needed since x(), y() are filled with it
// setting fonts is TODO :P
#if 0
Fl::set_font((Fl_Font)20, "-windows-*-medium-r-*-*-14-*-*-*-*-*-*-*");
labelfont((Fl_Font)20);
#endif
label(settings->name.c_str());
imenu = new Fl_Menu_Button(0, 0, 0, 0);
if(settings->type == ICON_TRASH)
imenu->menu(icon_trash_menu);
else
imenu->menu(icon_menu);
if(!settings->icon.empty()) {
const char* nn = settings->icon.c_str();
@ -87,24 +133,7 @@ DesktopIcon::DesktopIcon(GlobalIconSettings* gs, IconSettings* is, int bg) :
EDEBUG(ESTRLOC ": Got empty icon name ?!?\n");
}
// make sure icons are visible on desktop
int dx = Desktop::instance()->x();
int dy = Desktop::instance()->y();
int dw = Desktop::instance()->w();
int dh = Desktop::instance()->h();
int ix = x();
int iy = y();
if(x() < dx)
ix = dx;
if(y() < dy)
iy = dy;
if(x() + w() > dw)
ix = (dx + dw) - w();
if(y() + h() > dh)
iy = (dy + dh) - h();
position(ix, iy);
fix_position(x(), y());
//Use desktop color as color for icon background
color(bg);
@ -120,6 +149,8 @@ DesktopIcon::~DesktopIcon() {
delete settings;
if(micon)
delete micon;
delete imenu;
}
void DesktopIcon::update_label_size(void) {
@ -130,14 +161,34 @@ void DesktopIcon::update_label_size(void) {
* make sure current font size/type is set (internaly to fltk)
* so fl_measure() can correctly calculate label width and height
*/
int old = fl_font();
int old_sz = fl_size();
fl_font(labelfont(), labelsize());
fl_measure(label(), lwidth, lheight, align());
fl_font(old, old_sz);
lwidth += 12;
lheight += 5;
}
void DesktopIcon::fix_position(int X, int Y) {
int dx, dy, dw, dh;
Desktop::instance()->area(dx, dy, dw, dh);
if(X < dx)
X = dx;
if(Y < dy)
Y = dy;
if(X + w() > dw)
X = (dx + dw) - w();
if(Y + h() > dh)
Y = (dy + dh) - h();
position(X, Y);
}
void DesktopIcon::drag(int x, int y, bool apply) {
if(!micon) {
micon = new MovableIcon(this);
@ -173,9 +224,9 @@ void DesktopIcon::drag(int x, int y, bool apply) {
ix = (w()/2) - (image()->w()/2);
iy = (h()/2) - (image()->h()/2);
}
position(micon->x() - ix, micon->y() - iy);
fix_position(micon->x() - ix, micon->y() - iy);
#else
position(micon->x(), micon->y());
fix_position(micon->x(), micon->y());
#endif
delete micon;
micon = NULL;
@ -198,11 +249,31 @@ int DesktopIcon::drag_icon_y(void) {
return micon->y();
}
void DesktopIcon::rename(const char* str) {
if(!str)
return;
if(settings->name == str)
return;
settings->name = str;
label(settings->name.c_str());
update_label_size();
redraw();
}
void DesktopIcon::fast_redraw(void) {
EASSERT(parent() != NULL && "Impossible !");
// LABEL_OFFSET + 2 include selection box line height too
parent()->damage(FL_DAMAGE_ALL, x(), y(), w(), h() + lheight + LABEL_OFFSET + 2);
int wsz = w();
int xpos = x();
if(lwidth > w()) {
wsz = lwidth + 4;
xpos = x() - 4;
}
// LABEL_OFFSET + 2 include selection box line height too; same for xpos
parent()->damage(FL_DAMAGE_ALL, xpos, y(), wsz, h() + lheight + LABEL_OFFSET + 2);
}
void DesktopIcon::draw(void) {
@ -218,6 +289,8 @@ void DesktopIcon::draw(void) {
iy += y();
im->draw(ix, iy);
EDEBUG(ESTRLOC ": DesktopIcon icon redraw\n");
}
if(globals->label_draw && (damage() & (FL_DAMAGE_ALL | EDAMAGE_CHILD_LABEL))) {
@ -231,6 +304,12 @@ void DesktopIcon::draw(void) {
fl_rectf(X, Y, lwidth, lheight);
}
int old_font = fl_font();
int old_font_sz = fl_size();
// draw with icon's font
fl_font(labelfont(), labelsize());
// pseudo-shadow
fl_color(FL_BLACK);
fl_draw(label(), X+1, Y+1, lwidth, lheight, align(), 0, 0);
@ -238,6 +317,9 @@ void DesktopIcon::draw(void) {
fl_color(globals->label_foreground);
fl_draw(label(), X, Y, lwidth, lheight, align(), 0, 0);
// restore old font
fl_font(old_font, old_font_sz);
if(is_focused()) {
/*
* draw focused box on our way so later
@ -262,6 +344,8 @@ void DesktopIcon::draw(void) {
// revert to old color whatever that be
fl_color(old);
EDEBUG(ESTRLOC ": DesktopIcon label redraw\n");
}
}
@ -279,16 +363,34 @@ int DesktopIcon::handle(int event) {
case FL_MOVE:
return 1;
case FL_PUSH:
if(Fl::event_button() == 3) {
// Fl_Menu_Item::popup() by default does not call callbacks
const Fl_Menu_Item* m = imenu->menu()->popup(Fl::event_x(), Fl::event_y());
if(m && m->callback())
m->do_callback(0, this);
}
return 1;
case FL_RELEASE:
if(Fl::event_clicks() > 0)
EDEBUG(ESTRLOC ": EXECUTE: %s\n", settings->cmd.c_str());
return 1;
case FL_DND_ENTER:
case FL_DND_DRAG:
case FL_DND_LEAVE:
return 1;
case FL_DND_RELEASE:
EDEBUG(ESTRLOC ": FL_DND_RELEASE on icon\n");
return 1;
case FL_PASTE:
EDEBUG(ESTRLOC ": FL_PASTE on icon with %s\n", Fl::event_text());
return 1;
default:
break;
}
return 1;
return 0;
}
MovableIcon::MovableIcon(DesktopIcon* ic) : Fl_Window(ic->x(), ic->y(), ic->w(), ic->h()), icon(ic), mask(0) {

View File

@ -25,6 +25,8 @@ class GlobalIconSettings;
class IconSettings;
class MovableIcon;
class Fl_Menu_Button;
class DesktopIcon : public Fl_Widget {
private:
IconSettings* settings;
@ -36,7 +38,10 @@ class DesktopIcon : public Fl_Widget {
MovableIcon* micon;
Fl_Menu_Button* imenu;
void update_label_size(void);
void fix_position(int X, int Y);
public:
DesktopIcon(GlobalIconSettings* gisett, IconSettings* isett, int bg);
@ -71,7 +76,7 @@ class DesktopIcon : public Fl_Widget {
Fl_Image* icon_image(void) { return image(); }
const IconSettings* get_settings(void) const { return settings; }
void rename(const char* str);
};
class MovableIcon : public Fl_Window {

View File

@ -354,5 +354,5 @@ char* get_basename(const char* path) {
if(p)
return (p + 1);
return (char*) path;
return (char*)path;
}

View File

@ -530,5 +530,5 @@ int Wallpaper::handle(int event) {
return parent()->handle(event);
}
return Fl_Box::handle(event);
return 0;
}

View File

@ -23,16 +23,23 @@
#include <edelib/MimeType.h>
#include <edelib/StrUtil.h>
#include <edelib/IconTheme.h>
#include <edelib/Nls.h>
#include <FL/Fl.h>
#include <FL/Fl_Shared_Image.h>
#include <FL/x.h>
#include <FL/fl_draw.h>
#include <FL/Fl_Box.h>
#include <FL/Fl_Shared_Image.h>
#include <FL/Fl_Menu_Button.h>
#include <FL/fl_ask.h>
#if 0
#include <X11/extensions/Xdamage.h>
#endif
#include <signal.h>
#include <stdlib.h> // rand, srand
#include <time.h> // time
#include <ctype.h> // isprint
#define CONFIG_NAME "eiconman.conf"
@ -47,7 +54,34 @@
* Which widgets Fl::belowmouse() should skip. This should be updated
* when new non-icon child is added to Desktop window.
*/
#define NOT_SELECTABLE(widget) ((widget == this) || (widget == wallpaper))
#define NOT_SELECTABLE(widget) ((widget == this) || (widget == wallpaper) || (widget == dmenu))
Fl_Menu_Item desktop_menu[] = {
{_(" &New desktop item "), 0, 0, 0, FL_SUBMENU},
{_(" &Application... "), 0, 0},
{_(" &Directory... "), 0, 0},
{_(" &URL... "), 0, 0},
{_(" &Link... "), 0, 0},
{0},
{_(" &Arrange "), 0, 0, 0, FL_SUBMENU | FL_MENU_DIVIDER},
{_(" &By name "), 0, 0},
{_(" &Vertical line up "), 0, 0},
{_(" &Horizontal line up "), 0, 0},
{0},
{_(" &Copy "), 0, 0},
{_(" &Paste "), 0, 0, 0, FL_MENU_DIVIDER},
{_(" &Icons settings... "), 0, 0},
{_(" &Background... "), 0, 0},
{0}
};
#if 0
XserverRegion xregion;
XserverRegion xpart;
Damage xdamage;
int xevent_base, xerror_base;
#endif
Desktop* Desktop::pinstance = NULL;
bool running = false;
@ -77,6 +111,22 @@ void restart_signal(int signum) {
}
int desktop_xmessage_handler(int event) {
#if 0
if(fl_xevent->type == xevent_base + XDamageNotify) {
XDamageNotifyEvent* xdev = (XDamageNotifyEvent*)fl_xevent;
EDEBUG(ESTRLOC ": Damaged region %i %i %i %i on 0x%lx\n", xdev->area.x, xdev->area.y,
xdev->area.width, xdev->area.height, xdev->drawable);
//XDamageSubtract(fl_display, xdev->damage, None, None);
XFixesSetRegion(fl_display, xpart, &xdev->area, 1);
XFixesUnionRegion(fl_display, xregion, xregion, xpart);
}
XDamageSubtract(fl_display, xdamage, xregion, None);
XFixesSetRegion(fl_display, xregion, 0, 0);
#endif
if(fl_xevent->type == PropertyNotify) {
if(fl_xevent->xproperty.atom == _XA_NET_CURRENT_DESKTOP) {
Desktop::instance()->notify_desktop_changed();
@ -135,12 +185,15 @@ void Desktop::init_internals(void) {
* then icons, so they can be drawn at top of them.
*/
begin();
dmenu = new Fl_Menu_Button(0, 0, 0, 0);
dmenu->menu(desktop_menu);
wallpaper = new Wallpaper(0, 0, w(), h());
//wallpaper->set("/home/sanel/wallpapers/katebig.jpg");
wallpaper->set("/home/sanel/wallpapers/katebig.jpg");
//wallpaper->set_tiled("/home/sanel/wallpapers/katesmall.jpg");
//wallpaper->set_tiled("/home/sanelz/walls/katesmall.jpg");
//wallpaper->set_tiled("/home/sanelz/walls/kate.jpg");
wallpaper->set("/home/sanelz/walls/katebig.jpg");
//wallpaper->set("/home/sanelz/walls/katebig.jpg");
//wallpaper->hide();
//wallpaper->set("/home/sanelz/walls/katesmall.jpg");
//wallpaper->set("/home/sanelz/walls/nin/1024x768-04.jpg");
@ -151,6 +204,7 @@ void Desktop::init_internals(void) {
set_bg_color(dsett->color, false);
read_config();
running = true;
}
@ -183,6 +237,13 @@ void Desktop::show(void) {
if(!shown()) {
Fl_X::make_xid(this);
net_make_me_desktop(this);
#if 0
XDamageQueryExtension(fl_display, &xevent_base, &xerror_base);
xdamage = XDamageCreate(fl_display, fl_xid(this), XDamageReportBoundingBox);
xregion = XFixesCreateRegionFromWindow(fl_display, fl_xid(this), 0);
xpart = XFixesCreateRegionFromWindow(fl_display, fl_xid(this), 0);
#endif
}
}
@ -432,7 +493,7 @@ void Desktop::unfocus_all(void) {
}
}
void Desktop::select(DesktopIcon* ic) {
void Desktop::select(DesktopIcon* ic, bool do_redraw) {
EASSERT(ic != NULL);
if(in_selection(ic))
@ -442,7 +503,9 @@ void Desktop::select(DesktopIcon* ic) {
if(!ic->is_focused()) {
ic->do_focus();
ic->fast_redraw();
if(do_redraw)
ic->fast_redraw();
}
}
@ -489,8 +552,8 @@ void Desktop::move_selection(int x, int y, bool apply) {
ic->drag(prev_x + tmp_x, prev_y + tmp_y, apply);
// very slow if not checked
if(apply == true)
ic->fast_redraw();
//if(apply == true)
// ic->fast_redraw();
}
selection_x = x;
@ -504,7 +567,6 @@ void Desktop::move_selection(int x, int y, bool apply) {
redraw();
}
#if 0
/*
* Tries to figure out icon below mouse. It is alternative to
* Fl::belowmouse() since with this we hunt only icons, not other
@ -523,7 +585,6 @@ DesktopIcon* Desktop::below_mouse(int px, int py) {
return NULL;
}
#endif
void Desktop::select_in_area(void) {
if(!selbox->show)
@ -568,12 +629,10 @@ void Desktop::select_in_area(void) {
if(ic->is_focused()) {
ic->do_unfocus();
// updated from Desktop::draw()
ic->damage(EDAMAGE_CHILD_LABEL);
/*
* need to redraw whole screen since icon label is
* outside icon's drawable rectangle
*/
redraw();
ic->fast_redraw();
//ic->damage(EDAMAGE_CHILD_LABEL);
//ic->redraw();
//ic->damage(EDAMAGE_CHILD_LABEL);
}
}
}
@ -615,50 +674,113 @@ void Desktop::notify_desktop_changed(void) {
void Desktop::drop_source(const char* src, int src_len, int x, int y) {
if(!src)
return;
/*
* must copy clipboard content since with given size since
* it could be null chars in the middle of it
*/
char* src_copy = new char[src_len+1];
int i;
for(i = 0; i < src_len; i+=2) {
if(isascii(src[i]) && src[i] != '\0')
src_copy[i] = src[i];
if(src_len < 1)
return;
char* src_copy = new char[src_len + 1];
int real_len = 0;
// mozilla sends junk in form: ascii<0>ascii<0>..., don't know why
for(int i = 0, j = 0; i < src_len; i++) {
if(src[i] != 0) {
src_copy[j++] = src[i];
real_len++;
}
}
src_copy[i] = '\0';
src_copy[real_len] = '\0';
EDEBUG(ESTRLOC ": DND, received %s\n", src_copy);
if(real_len < 1) {
delete [] src_copy;
return;
}
delete [] src_copy;
EDEBUG(ESTRLOC ": DND on Desktop, got: %s\n", src_copy);
return;
// valid url's (if got) ends with \r\n, clean that
char* pp = strstr(src_copy, "\r\n");
if(pp)
*pp = '\0';
char* sptr = 0;
if((real_len > 7) && (strncmp(src_copy, "file://", 7) == 0))
sptr = src_copy + 7;
else
sptr = src_copy;
if(!edelib::file_exists(sptr) && !edelib::dir_exists(sptr)) {
fl_message("Droping file content is not implemented yet. Soon, soon... :)");
delete [] src_copy;
return;
}
IconSettings is;
is.x = x;
is.y = y;
// absolute path is (for now) seen as non-url
if(src[0] == '/')
is.cmd_is_url = false;
else
is.cmd_is_url = true;
bool is_read = false;
is.name = get_basename(src);
is.cmd = "(none)";
is.type = ICON_NORMAL;
if(edelib::str_ends(src_copy, ".desktop")) {
edelib::DesktopFile dconf;
edelib::MimeType mt;
if(!mt.set(src)) {
EWARNING(ESTRLOC ": MimeType for %s failed, not dropping icon\n", src);
return;
edelib::String path = sptr;
edelib::DesktopFile dfile;
if(dfile.load(path.c_str())) {
char buff[256];
if(dfile.type() == edelib::DESK_FILE_TYPE_LINK) {
dfile.url(buff, 256);
is.cmd_is_url = true;
}
else {
dfile.exec(buff, 256);
is.cmd_is_url = false;
}
is.cmd = buff;
is.type = ICON_NORMAL;
dfile.name(buff, 256);
is.name = buff;
dfile.icon(buff, 256);
is.icon = buff;
is_read = true;
}
}
if(!is_read) {
// absolute path is (for now) seen as non-url
if(sptr[0] == '/')
is.cmd_is_url = false;
else
is.cmd_is_url = true;
is.cmd = "(none)";
is.type = ICON_NORMAL;
edelib::MimeType mt;
if(!mt.set(sptr)) {
EWARNING(ESTRLOC ": MimeType for %s (%s) failed, not dropping icon\n", sptr, src_copy);
delete [] src_copy;
return;
}
is.name = get_basename(sptr);
is.icon = mt.icon_name();
}
is.icon = mt.icon_name();
DesktopIcon* dic = new DesktopIcon(&gisett, &is, color());
add_icon(dic);
delete [] src_copy;
redraw();
}
void Desktop::draw(void) {
if(!damage())
return;
if(damage() & (FL_DAMAGE_ALL | FL_DAMAGE_EXPOSE)) {
/*
* If any overlay was previously visible during full
@ -668,14 +790,13 @@ void Desktop::draw(void) {
clear_xoverlay();
DESKTOP_WINDOW::draw();
EDEBUG("REDRAW ALL\n");
EDEBUG(ESTRLOC ": REDRAW ALL\n");
}
if(damage() & (FL_DAMAGE_ALL | EDAMAGE_OVERLAY)) {
if(selbox->show) {
if(damage() & EDAMAGE_OVERLAY) {
if(selbox->show)
draw_xoverlay(selbox->x, selbox->y, selbox->w, selbox->h);
EDEBUG("DRAW OVERLAY\n");
} else
else
clear_xoverlay();
/*
@ -683,10 +804,15 @@ void Desktop::draw(void) {
* just update their label since it is indicator of selection
*/
for(int i = 0; i < children(); i++) {
if(child(i)->damage() == EDAMAGE_CHILD_LABEL)
if(child(i)->damage() == EDAMAGE_CHILD_LABEL) {
update_child(*child(i));
child(i)->clear_damage();
EDEBUG(ESTRLOC ": ICON REDRAW \n");
}
}
}
clear_damage();
}
int Desktop::handle(int event) {
@ -719,7 +845,10 @@ int Desktop::handle(int event) {
if(Fl::event_button() == 1) {
selbox->x = Fl::event_x();
selbox->y = Fl::event_y();
} else if(Fl::event_button() == 3) {
dmenu->menu()->popup(Fl::event_x(), Fl::event_y());
}
return 1;
}
@ -805,8 +934,6 @@ int Desktop::handle(int event) {
/*
* Now pickup those who are in is_focused() state.
* Here is not used select() since it will fill selectionbuff with
* redrawing whole window each time. This is not what we want.
*
* Possible flickers due overlay will be later removed when is
* called move_selection(), which will in turn redraw icons again
@ -817,7 +944,7 @@ int Desktop::handle(int event) {
for(unsigned int i = 0; i < icons.size(); i++) {
if(icons[i]->is_focused())
select(icons[i]);
select(icons[i], false);
}
return 1;
@ -837,21 +964,27 @@ int Desktop::handle(int event) {
moving = false;
return 1;
case FL_DND_ENTER:
case FL_DND_DRAG:
case FL_DND_LEAVE:
EDEBUG("FL_DND_ENTER|FL_DND_DRAG|FL_DND_LEAVE\n");
return 1;
case FL_DND_RELEASE:
EDEBUG(ESTRLOC ": DND on desktop\n");
//Fl::paste(*this);
return 1;
case FL_PASTE:
EDEBUG("=======> %s\n", Fl::event_text());
EDEBUG("=======> %s\n", Fl::event_text());
//drop_source(Fl::event_text(), Fl::event_length(), Fl::event_x_root(), Fl::event_y_root());
case FL_DND_RELEASE: {
DesktopIcon* di = below_mouse(Fl::event_x(), Fl::event_y());
if(di)
return di->handle(event);
return 1;
}
case FL_PASTE: {
DesktopIcon* di = below_mouse(Fl::event_x(), Fl::event_y());
if(di)
return di->handle(event);
drop_source(Fl::event_text(), Fl::event_length(), Fl::event_x(), Fl::event_y());
return 1;
}
case FL_ENTER:
case FL_LEAVE:
@ -895,6 +1028,7 @@ int main() {
Fl::wait();
Desktop::shutdown();
edelib::IconTheme::shutdown();
return 0;

View File

@ -63,6 +63,7 @@ struct IconSettings {
edelib::String icon;
edelib::String icon2; // for stateable icons, like trash (empty/full)
edelib::String key_name; // name used as key when storing positions
edelib::String full_path; // for tracking changes
};
// Selection overlay
@ -75,9 +76,11 @@ class Wallpaper;
class DesktopIcon;
class NotifyBox;
class Fl_Menu_Button;
typedef edelib::vector<DesktopIcon*> DesktopIconList;
#define DESKTOP_WINDOW Fl_Double_Window
#define DESKTOP_WINDOW Fl_Window
class Desktop : public DESKTOP_WINDOW {
private:
@ -91,8 +94,9 @@ class Desktop : public DESKTOP_WINDOW {
GlobalIconSettings gisett;
DesktopSettings* dsett;
Wallpaper* wallpaper;
NotifyBox* notify;
Fl_Menu_Button* dmenu;
Wallpaper* wallpaper;
NotifyBox* notify;
DesktopIconList icons;
DesktopIconList selectionbuff;
@ -106,7 +110,7 @@ class Desktop : public DESKTOP_WINDOW {
void unfocus_all(void);
void select(DesktopIcon* ic);
void select(DesktopIcon* ic, bool do_redraw = true);
void select_only(DesktopIcon* ic);
bool in_selection(const DesktopIcon* ic);
void move_selection(int x, int y, bool apply);
@ -115,7 +119,7 @@ class Desktop : public DESKTOP_WINDOW {
void drop_source(const char* src, int src_len, int x, int y);
//DesktopIcon* below_mouse(int px, int py);
DesktopIcon* below_mouse(int px, int py);
public:
Desktop();
@ -134,13 +138,13 @@ class Desktop : public DESKTOP_WINDOW {
void save_config(void);
void update_workarea(void);
void area(int& X, int& Y, int& W, int& H) { X = x(); Y = y(); W = w(); H = h(); }
void set_bg_color(int c, bool do_redraw = true);
void notify_box(const char* msg, bool copy = false);
void notify_box_color(Fl_Color col);
void notify_desktop_changed(void);
Fl_Window* desktop_window(void) { return this; }
};
#endif