Merge pull request #456 from analogjupiter/corner-style

Implement "corner style" property for sdpy windows on Windows 11+
This commit is contained in:
Adam D. Ruppe 2024-09-09 20:47:03 -04:00 committed by GitHub
commit 6d30f2fba9
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
3 changed files with 239 additions and 2 deletions

View File

@ -28,12 +28,12 @@
"configurations": [ "configurations": [
{ {
"name": "normal", "name": "normal",
"libs-windows": ["gdi32", "ole32"] "libs-windows": ["dwmapi", "gdi32", "ole32"]
}, },
{ {
"name": "without-opengl", "name": "without-opengl",
"versions": ["without_opengl"], "versions": ["without_opengl"],
"libs-windows": ["gdi32", "ole32"] "libs-windows": ["dwmapi", "gdi32", "ole32"]
}, },
{ {
"name": "cocoa", "name": "cocoa",

View File

@ -408,8 +408,26 @@ struct PresenterConfig {
/// ///
static struct Window { static struct Window {
///
string title = "ARSD Pixmap Presenter"; string title = "ARSD Pixmap Presenter";
///
Size size; Size size;
/++
Window corner style
$(NOTE
At the time of writing, this is only implemented on Windows.
It has no effect elsewhere for now but does no harm either.
Windows: Requires Windows 11 or later.
)
History:
Added September 10, 2024.
+/
CornerStyle corners = CornerStyle.rectangular;
} }
} }
@ -911,6 +929,7 @@ final class PixmapPresenter {
); );
window.windowResized = &this.windowResized; window.windowResized = &this.windowResized;
window.cornerStyle = config.window.corners;
// alloc objects // alloc objects
_poc = new PresenterObjectsContainer( _poc = new PresenterObjectsContainer(

View File

@ -1164,6 +1164,111 @@ version(Windows) {
// for AlphaBlend... a breaking change.... // for AlphaBlend... a breaking change....
version(CRuntime_DigitalMars) { } else version(CRuntime_DigitalMars) { } else
pragma(lib, "msimg32"); pragma(lib, "msimg32");
// core.sys.windows.dwmapi
private {
/++
See_also:
https://learn.microsoft.com/en-us/windows/win32/api/dwmapi/nf-dwmapi-dwmgetwindowattribute
+/
extern extern(Windows) HRESULT DwmGetWindowAttribute(
HWND hwnd,
DWORD dwAttribute,
PVOID pvAttribute,
DWORD cbAttribute
) nothrow @nogc;
/++
See_also:
https://learn.microsoft.com/en-us/windows/win32/api/dwmapi/nf-dwmapi-dwmsetwindowattribute
+/
extern extern(Windows) HRESULT DwmSetWindowAttribute(
HWND hwnd,
DWORD dwAttribute,
LPCVOID pvAttribute,
DWORD cbAttribute
) nothrow @nogc;
/++
See_also:
https://learn.microsoft.com/en-us/windows/win32/api/dwmapi/ne-dwmapi-dwmwindowattribute
+/
enum DWMWINDOWATTRIBUTE {
// incomplete, only declare what we need
/++
Usage:
pvAttribute `DWM_WINDOW_CORNER_PREFERENCE*`
$(NOTE
Requires Windows 11 or later.
)
+/
DWMWA_WINDOW_CORNER_PREFERENCE = 33,
}
/++
See_also:
https://learn.microsoft.com/en-us/windows/win32/api/dwmapi/ne-dwmapi-dwm_window_corner_preference
+/
enum DWM_WINDOW_CORNER_PREFERENCE {
/// System decision
DWMWCP_DEFAULT = 0,
/// Never
DWMWCP_DONOTROUND = 1,
// If "appropriate"
DWMWCP_ROUND = 2,
// If "appropriate", but smaller radius
DWMWCP_ROUNDSMALL = 3
}
bool fromDWM(
DWM_WINDOW_CORNER_PREFERENCE value,
out CornerStyle result,
) @safe pure nothrow @nogc {
switch (value) with (DWM_WINDOW_CORNER_PREFERENCE) {
case DWMWCP_DEFAULT:
result = CornerStyle.automatic;
return true;
case DWMWCP_DONOTROUND:
result = CornerStyle.rectangular;
return true;
case DWMWCP_ROUND:
result = CornerStyle.rounded;
return true;
case DWMWCP_ROUNDSMALL:
result = CornerStyle.roundedSlightly;
return true;
default:
return false;
}
}
bool toDWM(
CornerStyle value,
out DWM_WINDOW_CORNER_PREFERENCE result,
) @safe pure nothrow @nogc {
final switch (value) with (DWM_WINDOW_CORNER_PREFERENCE) {
case CornerStyle.automatic:
result = DWMWCP_DEFAULT;
return true;
case CornerStyle.rectangular:
result = DWMWCP_DONOTROUND;
return true;
case CornerStyle.rounded:
result = DWMWCP_ROUND;
return true;
case CornerStyle.roundedSlightly:
result = DWMWCP_ROUNDSMALL;
return true;
}
}
pragma(lib, "dwmapi");
}
} else version (linux) { } else version (linux) {
//k8: this is hack for rdmd. sorry. //k8: this is hack for rdmd. sorry.
static import core.sys.linux.epoll; static import core.sys.linux.epoll;
@ -1790,6 +1895,31 @@ enum BlockingMode {
onlyIfNotNested = 0x10, onlyIfNotNested = 0x10,
} }
/++
Window corner visuals preference
+/
enum CornerStyle {
/++
Use the default style automatically applied by the system or its window manager/compositor.
+/
automatic,
/++
Prefer rectangular window corners
+/
rectangular,
/++
Prefer rounded window corners
+/
rounded,
/++
Prefer slightly-rounded window corners
+/
roundedSlightly,
}
/++ /++
The flagship window class. The flagship window class.
@ -3835,6 +3965,94 @@ private:
glViewport(0, 0, width, height); glViewport(0, 0, width, height);
} }
} }
// TODO: Implement on non-Windows platforms (where available).
private CornerStyle _fauxCornerStyle = CornerStyle.automatic;
/++
Style of the window's corners
$(WARNING
Currently only implemented on Windows targets.
Has no visual effect elsewhere.
Windows: Requires Windows 11 or later.
)
History:
Added September 09, 2024.
+/
public CornerStyle cornerStyle() @trusted {
version(Windows) {
import std.format;
DWM_WINDOW_CORNER_PREFERENCE dwmCorner;
const apiResult = DwmGetWindowAttribute(
this.hwnd,
DWMWINDOWATTRIBUTE.DWMWA_WINDOW_CORNER_PREFERENCE,
&dwmCorner,
typeof(dwmCorner).sizeof
);
if (apiResult != S_OK) {
// Unsupported?
if (apiResult == E_INVALIDARG) {
// Feature unsupported; Windows version probably too old.
// Requires Windows 11 (build 22000) or later.
return _fauxCornerStyle;
}
const exMsg = format("DwmGetWindowAttribute() failed with error 0x%x.", apiResult);
throw new Exception(exMsg);
}
CornerStyle corner;
if (!dwmCorner.fromDWM(corner)) {
const exMsg = format(
"DwmGetWindowAttribute() reported an unfamiliar corner preference: %s",
dwmCorner,
);
throw new Exception(exMsg);
}
return corner;
} else {
return _fauxCornerStyle;
}
}
/// ditto
public void cornerStyle(const CornerStyle corner) @trusted {
version(Windows) {
import std.format;
DWM_WINDOW_CORNER_PREFERENCE dwmCorner;
if (!corner.toDWM(dwmCorner)) {
assert(false, "This should have been impossible because of a final switch.");
}
const apiResult = DwmSetWindowAttribute(
this.hwnd,
DWMWINDOWATTRIBUTE.DWMWA_WINDOW_CORNER_PREFERENCE,
&dwmCorner,
typeof(dwmCorner).sizeof
);
if (apiResult != S_OK) {
// Unsupported?
if (apiResult == E_INVALIDARG) {
// Feature unsupported; Windows version probably too old.
// Requires Windows 11 (build 22000) or later.
_fauxCornerStyle = corner;
return;
}
const exMsg = format("DwmSetWindowAttribute() failed with error 0x%x.", apiResult);
throw new Exception(exMsg);
}
} else {
_fauxCornerStyle = corner;
}
}
} }
version(OSXCocoa) version(OSXCocoa)