handle some transparency and centering in x11 notificationarea icon

This commit is contained in:
Adam D. Ruppe 2017-12-04 11:19:30 -05:00
parent 0be86f7406
commit fc454a6c2a
1 changed files with 120 additions and 22 deletions

View File

@ -2657,6 +2657,7 @@ class NotificationAreaIcon : CapableOfHandlingNativeEvent {
version(X11) { version(X11) {
void recreateAfterDisconnect() { void recreateAfterDisconnect() {
stateDiscarded = false; stateDiscarded = false;
clippixmap = None;
throw new Exception("NOT IMPLEMENTED"); throw new Exception("NOT IMPLEMENTED");
} }
@ -2748,6 +2749,8 @@ class NotificationAreaIcon : CapableOfHandlingNativeEvent {
auto gc = DefaultGC(display, DefaultScreen(display)); auto gc = DefaultGC(display, DefaultScreen(display));
XClearWindow(display, nativeHandle); XClearWindow(display, nativeHandle);
XSetClipMask(display, gc, clippixmap);
XSetForeground(display, gc, XSetForeground(display, gc,
cast(uint) 0 << 16 | cast(uint) 0 << 16 |
cast(uint) 0 << 8 | cast(uint) 0 << 8 |
@ -2762,10 +2765,18 @@ class NotificationAreaIcon : CapableOfHandlingNativeEvent {
XFillArc(display, nativeHandle, XFillArc(display, nativeHandle,
gc, width / 4, height / 4, width * 2 / 4, height * 2 / 4, 0 * 64, 360 * 64); gc, width / 4, height / 4, width * 2 / 4, height * 2 / 4, 0 * 64, 360 * 64);
} else { } else {
int dx = 0;
int dy = 0;
if(width > img.width)
dx = (width - img.width) / 2;
if(height > img.height)
dy = (height - img.height) / 2;
XSetClipOrigin(display, gc, dx, dy);
if (img.usingXshm) if (img.usingXshm)
XShmPutImage(display, cast(Drawable)nativeHandle, gc, img.handle, 0, 0, 0, 0, img.width, img.height, false); XShmPutImage(display, cast(Drawable)nativeHandle, gc, img.handle, 0, 0, dx, dy, img.width, img.height, false);
else else
XPutImage(display, cast(Drawable)nativeHandle, gc, img.handle, 0, 0, 0, 0, img.width, img.height); XPutImage(display, cast(Drawable)nativeHandle, gc, img.handle, 0, 0, dx, dy, img.width, img.height);
} }
flushGui(); flushGui();
} }
@ -2800,13 +2811,42 @@ class NotificationAreaIcon : CapableOfHandlingNativeEvent {
} }
private void createXWin () { private void createXWin () {
if(getTrayOwner() == None) auto trayOwner = getTrayOwner();
if(trayOwner == None)
throw new Exception("No notification area found"); throw new Exception("No notification area found");
// create window // create window
auto display = XDisplayConnection.get; auto display = XDisplayConnection.get;
auto nativeWindow = XCreateWindow(display, RootWindow(display, DefaultScreen(display)), 0, 0, 16, 16, 0, 24, InputOutput, cast(Visual*) CopyFromParent, 0, null);
Visual* v = cast(Visual*) CopyFromParent;
/+
auto visualProp = getX11PropertyData(trayOwner, GetAtom!("_NET_SYSTEM_TRAY_VISUAL", true)(display));
if(visualProp !is null) {
c_ulong[] info = cast(c_ulong[]) visualProp;
if(info.length == 1) {
auto vid = info[0];
int returned;
XVisualInfo t;
t.visualid = vid;
auto got = XGetVisualInfo(display, VisualIDMask, &t, &returned);
if(got !is null) {
if(returned == 1) {
v = got.visual;
import std.stdio;
writeln("using special visual ", *got);
}
XFree(got);
}
}
}
+/
auto nativeWindow = XCreateWindow(display, RootWindow(display, DefaultScreen(display)), 0, 0, 16, 16, 0, 24, InputOutput, v, 0, null);
assert(nativeWindow); assert(nativeWindow);
XSetWindowBackgroundPixmap(display, nativeWindow, 1 /* ParentRelative */);
nativeHandle = nativeWindow; nativeHandle = nativeWindow;
///+ ///+
@ -2814,7 +2854,7 @@ class NotificationAreaIcon : CapableOfHandlingNativeEvent {
info[0] = 0; info[0] = 0;
info[1] = 1; info[1] = 1;
string title = "simpledisplay.d program"; string title = this.name is null ? "simpledisplay.d program" : this.name;
auto XA_UTF8 = XInternAtom(display, "UTF8_STRING".ptr, false); auto XA_UTF8 = XInternAtom(display, "UTF8_STRING".ptr, false);
auto XA_NETWM_NAME = XInternAtom(display, "_NET_WM_NAME".ptr, false); auto XA_NETWM_NAME = XInternAtom(display, "_NET_WM_NAME".ptr, false);
XChangeProperty(display, nativeWindow, XA_NETWM_NAME, XA_UTF8, 8, PropModeReplace, title.ptr, cast(uint)title.length); XChangeProperty(display, nativeWindow, XA_NETWM_NAME, XA_UTF8, 8, PropModeReplace, title.ptr, cast(uint)title.length);
@ -2880,6 +2920,7 @@ class NotificationAreaIcon : CapableOfHandlingNativeEvent {
} }
void updateNetWmIcon() { void updateNetWmIcon() {
if(img is null) return;
auto display = XDisplayConnection.get; auto display = XDisplayConnection.get;
// FIXME: ensure this is correct // FIXME: ensure this is correct
arch_ulong[] buffer; arch_ulong[] buffer;
@ -2913,6 +2954,7 @@ class NotificationAreaIcon : CapableOfHandlingNativeEvent {
private Timer timer; private Timer timer;
private Window nativeHandle; private Window nativeHandle;
private Pixmap clippixmap = None;
private int width = 16; private int width = 16;
private int height = 16; private int height = 16;
private bool active = false; private bool active = false;
@ -3019,7 +3061,10 @@ class NotificationAreaIcon : CapableOfHandlingNativeEvent {
// but on X, we need an Image, so its canonical ctor is there. They should // but on X, we need an Image, so its canonical ctor is there. They should
// forward to each other though. // forward to each other though.
version(X11) { version(X11) {
this(name, icon is null ? null : Image.fromMemoryImage(icon), onClick); this.name = name;
this.onClick = onClick;
createXWin();
this.icon = icon;
} else version(Windows) { } else version(Windows) {
this.onClick = onClick; this.onClick = onClick;
this.win32Icon = new WindowsIcon(icon); this.win32Icon = new WindowsIcon(icon);
@ -3067,8 +3112,9 @@ class NotificationAreaIcon : CapableOfHandlingNativeEvent {
this(string name, Image icon, void delegate(MouseButton button) onClick) { this(string name, Image icon, void delegate(MouseButton button) onClick) {
version(X11) { version(X11) {
this.onClick = onClick; this.onClick = onClick;
this.img = icon; this.name = name;
createXWin(); createXWin();
this.icon = icon;
} else version(Windows) { } else version(Windows) {
this(name, icon is null ? null : icon.toTrueColorImage(), onClick); this(name, icon is null ? null : icon.toTrueColorImage(), onClick);
} else version(OSXCocoa) { } else version(OSXCocoa) {
@ -3082,15 +3128,15 @@ class NotificationAreaIcon : CapableOfHandlingNativeEvent {
+/ +/
this(string name, MemoryImage icon, void delegate(int x, int y, MouseButton button, ModifierState mods) onClickEx) { this(string name, MemoryImage icon, void delegate(int x, int y, MouseButton button, ModifierState mods) onClickEx) {
this.onClickEx = onClickEx; this.onClickEx = onClickEx;
if (icon !is null) this.img = Image.fromMemoryImage(icon);
createXWin(); createXWin();
if (icon !is null) this.icon = icon;
} }
/// ditto /// ditto
this(string name, Image icon, void delegate(int x, int y, MouseButton button, ModifierState mods) onClickEx) { this(string name, Image icon, void delegate(int x, int y, MouseButton button, ModifierState mods) onClickEx) {
this.onClickEx = onClickEx; this.onClickEx = onClickEx;
this.img = icon;
createXWin(); createXWin();
this.icon = icon;
} }
} }
@ -3111,7 +3157,13 @@ class NotificationAreaIcon : CapableOfHandlingNativeEvent {
} }
string name_;
@property void name(string n) { @property void name(string n) {
name_ = n;
}
@property string name() {
return name_;
} }
/// ///
@ -3120,6 +3172,8 @@ class NotificationAreaIcon : CapableOfHandlingNativeEvent {
if (!active) return; if (!active) return;
if (i !is null) { if (i !is null) {
this.img = Image.fromMemoryImage(i); this.img = Image.fromMemoryImage(i);
this.clippixmap = transparencyMaskFromMemoryImage(i, nativeHandle);
//import std.stdio; writeln("using pixmap ", clippixmap);
updateNetWmIcon(); updateNetWmIcon();
redraw(); redraw();
} else { } else {
@ -3336,10 +3390,28 @@ class NotificationAreaIcon : CapableOfHandlingNativeEvent {
} }
~this() { ~this() {
version(X11)
if(clippixmap != None)
XFreePixmap(XDisplayConnection.get, clippixmap);
close(); close();
} }
} }
version(X11)
/// call XFreePixmap on the return value
Pixmap transparencyMaskFromMemoryImage(MemoryImage i, Window window) {
char[] data = new char[](i.width * i.height / 8 + 2);
data[] = 0;
int bitOffset = 0;
foreach(c; i.getAsTrueColorImage().imageData.colors) { // FIXME inefficient unnecessary conversion in palette cases
ubyte v = c.a > 128 ? 1 : 0;
data[bitOffset / 8] |= v << (bitOffset%8);
bitOffset++;
}
auto handle = XCreateBitmapFromData(XDisplayConnection.get, cast(Drawable) window, data.ptr, i.width, i.height);
return handle;
}
// basic functions to make timers // basic functions to make timers
@ -3753,6 +3825,15 @@ version(X11) {
return a; return a;
} }
/// Platform specific for X11 - gets atom names as a string
string getAtomName(Atom atom, Display* display) {
auto got = XGetAtomName(display, atom);
scope(exit) XFree(got);
import core.stdc.string;
string s = got[0 .. strlen(got)].idup;
return s;
}
/// Asserts ownership of PRIMARY and copies the text into a buffer that clients can request later /// Asserts ownership of PRIMARY and copies the text into a buffer that clients can request later
void setPrimarySelection(SimpleWindow window, string text) { void setPrimarySelection(SimpleWindow window, string text) {
setX11Selection!"PRIMARY"(window, text); setX11Selection!"PRIMARY"(window, text);
@ -9276,6 +9357,32 @@ int XSetSelectionOwner(Display *display, Atom selection, Window owner, Time time
Window XGetSelectionOwner(Display *display, Atom selection); Window XGetSelectionOwner(Display *display, Atom selection);
struct XVisualInfo {
Visual* visual;
VisualID visualid;
int screen;
uint depth;
int c_class;
c_ulong red_mask;
c_ulong green_mask;
c_ulong blue_mask;
int colormap_size;
int bits_per_rgb;
}
enum VisualNoMask= 0x0;
enum VisualIDMask= 0x1;
enum VisualScreenMask=0x2;
enum VisualDepthMask= 0x4;
enum VisualClassMask= 0x8;
enum VisualRedMaskMask=0x10;
enum VisualGreenMaskMask=0x20;
enum VisualBlueMaskMask=0x40;
enum VisualColormapSizeMask=0x80;
enum VisualBitsPerRGBMask=0x100;
enum VisualAllMask= 0x1FF;
XVisualInfo* XGetVisualInfo(Display*, c_long, XVisualInfo*, int*);
@ -9836,6 +9943,10 @@ Atom XInternAtom(
Bool /* only_if_exists */ Bool /* only_if_exists */
); );
Status XInternAtoms(Display*, char**, int, Bool);
char* XGetAtomName(Display*, Atom);
Status XGetAtomNames(Display*, Atom*, int count, char**);
alias int Status; alias int Status;
@ -10526,19 +10637,6 @@ static if (!SdpyIsUsingIVGLBinds) {
void glXWaitX(); void glXWaitX();
} }
struct XVisualInfo {
Visual *visual;
VisualID visualid;
int screen;
int depth;
int c_class; /* C++ */
arch_ulong red_mask;
arch_ulong green_mask;
arch_ulong blue_mask;
int colormap_size;
int bits_per_rgb;
}
} }
} }