Skip to content
New issue

Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? # to your account

Fix top border & misc tweaks #167

Merged
merged 2 commits into from
Mar 4, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
10 changes: 3 additions & 7 deletions src/core/contexts/win32windowcontext.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -920,13 +920,8 @@ namespace QWK {

const DynamicApis &apis = DynamicApis::instance();
const auto &extendMargins = [this, &apis, hwnd]() {
// For some unknown reason, the window background is totally black when the host object
// is a QWidget. And extending the window frame into the client area seems to fix it
// magically.
// We don't need the following *HACK* for QtQuick windows.
if (!m_host->isWidgetType()) {
return;
}
// For some unknown reason, the window background is totally black and extending
// the window frame into the client area seems to fix it magically.
// After many times of trying, we found that the Acrylic/Mica/Mica Alt background
// only appears on the native Win32 window's background, so naturally we want to
// extend the window frame into the whole client area to be able to let the special
Expand Down Expand Up @@ -955,6 +950,7 @@ namespace QWK {

const auto &effectBugWorkaround = [this, hwnd]() {
// We don't need the following *HACK* for QWidget windows.
// Completely based on actual experiments, root reason is totally unknown.
if (m_host->isWidgetType()) {
return;
}
Expand Down
27 changes: 22 additions & 5 deletions src/core/shared/windows10borderhandler_p.h
Original file line number Diff line number Diff line change
Expand Up @@ -31,10 +31,12 @@ namespace QWK {
}

inline void setupNecessaryAttributes() {
// https://github.com/microsoft/terminal/blob/71a6f26e6ece656084e87de1a528c4a8072eeabd/src/cascadia/WindowsTerminal/NonClientIslandWindow.cpp#L940
// Must extend top frame to client area
static QVariant defaultMargins = QVariant::fromValue(QMargins(0, 1, 0, 0));
ctx->setWindowAttribute(QStringLiteral("extra-margins"), defaultMargins);
if (!isWin11OrGreater()) {
// https://github.com/microsoft/terminal/blob/71a6f26e6ece656084e87de1a528c4a8072eeabd/src/cascadia/WindowsTerminal/NonClientIslandWindow.cpp#L940
// Must extend top frame to client area
static QVariant defaultMargins = QVariant::fromValue(QMargins(0, 1, 0, 0));
ctx->setWindowAttribute(QStringLiteral("extra-margins"), defaultMargins);
}

// Enable dark mode by default, otherwise the system borders are white
ctx->setWindowAttribute(QStringLiteral("dark-mode"), true);
Expand All @@ -45,7 +47,17 @@ namespace QWK {
(Qt::WindowMinimized | Qt::WindowMaximized | Qt::WindowFullScreen));
}

inline void drawBorder() {
inline void drawBorderEmulated(QPainter *painter, const QRect &rect) {
QRegion region(rect);
void *args[] = {
painter,
const_cast<QRect *>(&rect),
&region,
};
ctx->virtual_hook(AbstractWindowContext::DrawWindows10BorderHook_Emulated, args);
}

inline void drawBorderNative() {
ctx->virtual_hook(AbstractWindowContext::DrawWindows10BorderHook_Native, nullptr);
}

Expand All @@ -54,6 +66,11 @@ namespace QWK {
}

inline void updateExtraMargins(bool windowActive) {
if (isWin11OrGreater()) {
return;
}

// ### FIXME: transparent seam
if (windowActive) {
// Restore margins when the window is active
static QVariant defaultMargins = QVariant::fromValue(QMargins(0, 1, 0, 0));
Expand Down
13 changes: 3 additions & 10 deletions src/quick/quickwindowagent_win.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -97,21 +97,14 @@ namespace QWK {
BorderItem::~BorderItem() = default;

void BorderItem::updateGeometry() {
setHeight(borderThickness());
setHeight(borderThickness() / window()->devicePixelRatio());
setVisible(isNormalWindow());
}

void BorderItem::paint(QPainter *painter) {
Q_UNUSED(painter)
if (shouldEnableEmulatedPainter()) {
QRect rect(QPoint(0, 0), size().toSize());
QRegion region(rect);
void *args[] = {
painter,
&rect,
&region,
};
ctx->virtual_hook(AbstractWindowContext::DrawWindows10BorderHook_Emulated, args);
drawBorderEmulated(painter, QRect({0, 0}, size().toSize()));
} else {
needPaint = true;
}
Expand Down Expand Up @@ -170,7 +163,7 @@ namespace QWK {
void BorderItem::_q_afterSynchronizing() {
if (needPaint) {
needPaint = false;
drawBorder();
drawBorderNative();
}
}

Expand Down
2 changes: 1 addition & 1 deletion src/widgets/widgetwindowagent.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -61,7 +61,7 @@ namespace QWK {
w->setAttribute(Qt::WA_DontCreateNativeAncestors);
// Make sure the native window handle is actually created before we apply
// various hooks.
w->setAttribute(Qt::WA_NativeWindow); // ### FIXME: Remove
//w->setAttribute(Qt::WA_NativeWindow); // ### FIXME: Check

d->setup(w, new WidgetItemDelegate());
d->hostWidget = w;
Expand Down
4 changes: 2 additions & 2 deletions src/widgets/widgetwindowagent_win.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -83,7 +83,7 @@ namespace QWK {
// Due to the timer or user action, Qt will repaint some regions spontaneously,
// even if there is no WM_PAINT message, we must wait for it to finish painting
// and then update the top border area.
drawBorder();
drawBorderNative();
}

inline void forwardEventToWindowAndDraw(QWindow *window, QEvent *event) {
Expand All @@ -97,7 +97,7 @@ namespace QWK {

// Upon receiving the WM_PAINT message, Qt will repaint the entire view, and we
// must wait for it to finish painting before drawing this top border area.
drawBorder();
drawBorderNative();
}

protected:
Expand Down