winemac: Use a snapshot of an owned window when a zero-sized owner window is minimized.

Some apps create a zero-sized window as their "main" window and then create
all of the other top-level windows as owned windows with that main window as
the owner.  The user interacts with these owned windows.  When the user
attempts to minimize one of these owned windows, the app instead minimizes the
zero-sized owner window.  When an owner window is minimized, all of its owned
windows are hidden.

The Mac driver faithfully carries out these window operations.  The only
visible windows are hidden and the zero-sized window is minimized.  This
results in an invisible animation of the window down to a slot in the Dock -
a slot which appears mostly empty.  The invisible window thumbnail is badged
with the app icon, but it still looks strange.

On Windows, the Alt-Tab switcher uses the image of the owned window to
represent the zero-sized owner.

This commit attempts to do something similar.  It takes over drawing of the
Dock icon for minimized, zero-sized window.  It grabs a snapshot of one of the
owned windows and draws the app badge onto it.  Since the owned windows are
hidden before the zero-sized owner is minimized and we can't take snapshots of
hidden windows, we use heuristics to guess when it may be useful to grab the
snapshot.  If the user minimizes an owned window from the Cocoa side, we grab
that window's snapshot.  If an owned window is being hidden and no snapshot has
been taken recently, we grab its snapshot on the theory that this may be the
beginning of hiding all of the owned windows before minimizing the owner.

Unfortunately, this doesn't address the invisible animations when minimizing
and unminimizing the zero-sized owner window.

Signed-off-by: Ken Thomases <ken@codeweavers.com>
Signed-off-by: Alexandre Julliard <julliard@winehq.org>
This commit is contained in:
Ken Thomases 2015-10-23 02:48:34 -05:00 committed by Alexandre Julliard
parent 3f280dd20c
commit 496b001ae0
2 changed files with 116 additions and 0 deletions

View File

@ -73,6 +73,8 @@ @interface WineWindow : NSPanel <NSWindowDelegate>
NSPoint dragStartPosition; NSPoint dragStartPosition;
NSPoint dragWindowStartPosition; NSPoint dragWindowStartPosition;
NSTimeInterval lastDockIconSnapshot;
BOOL ignore_windowDeminiaturize; BOOL ignore_windowDeminiaturize;
BOOL ignore_windowResize; BOOL ignore_windowResize;
BOOL fakingClose; BOOL fakingClose;

View File

@ -1265,6 +1265,10 @@ - (void) doOrderOut
if ([self isMiniaturized]) if ([self isMiniaturized])
pendingMinimize = TRUE; pendingMinimize = TRUE;
WineWindow* parent = (WineWindow*)self.parentWindow;
if ([parent isKindOfClass:[WineWindow class]])
[parent grabDockIconSnapshotFromWindow:self force:NO];
[self becameIneligibleParentOrChild]; [self becameIneligibleParentOrChild];
if ([self isMiniaturized]) if ([self isMiniaturized])
{ {
@ -1569,6 +1573,109 @@ - (void) endWindowDragging
} }
} }
- (BOOL) isEmptyShaped
{
return (self.shapeData.length == sizeof(CGRectZero) && !memcmp(self.shapeData.bytes, &CGRectZero, sizeof(CGRectZero)));
}
- (BOOL) canProvideSnapshot
{
return (self.windowNumber > 0 && ![self isEmptyShaped]);
}
- (void) grabDockIconSnapshotFromWindow:(WineWindow*)window force:(BOOL)force
{
if (![self isEmptyShaped])
return;
NSTimeInterval now = [[NSProcessInfo processInfo] systemUptime];
if (!force && now < lastDockIconSnapshot + 1)
return;
if (window)
{
if (![window canProvideSnapshot])
return;
}
else
{
CGFloat bestArea;
for (WineWindow* childWindow in self.childWindows)
{
if (![childWindow isKindOfClass:[WineWindow class]] || ![childWindow canProvideSnapshot])
continue;
NSSize size = childWindow.frame.size;
CGFloat area = size.width * size.height;
if (!window || area > bestArea)
{
window = childWindow;
bestArea = area;
}
}
if (!window)
return;
}
const void* windowID = (const void*)(CGWindowID)window.windowNumber;
CFArrayRef windowIDs = CFArrayCreate(NULL, &windowID, 1, NULL);
CGImageRef windowImage = CGWindowListCreateImageFromArray(CGRectNull, windowIDs, kCGWindowImageBoundsIgnoreFraming);
CFRelease(windowIDs);
if (!windowImage)
return;
NSImage* appImage = [NSApp applicationIconImage];
if (!appImage)
appImage = [NSImage imageNamed:NSImageNameApplicationIcon];
NSImage* dockIcon = [[[NSImage alloc] initWithSize:NSMakeSize(256, 256)] autorelease];
[dockIcon lockFocus];
CGContextRef cgcontext = [[NSGraphicsContext currentContext] graphicsPort];
CGRect rect = CGRectMake(8, 8, 240, 240);
size_t width = CGImageGetWidth(windowImage);
size_t height = CGImageGetHeight(windowImage);
if (width > height)
{
rect.size.height *= height / (double)width;
rect.origin.y += (CGRectGetWidth(rect) - CGRectGetHeight(rect)) / 2;
}
else if (width != height)
{
rect.size.width *= width / (double)height;
rect.origin.x += (CGRectGetHeight(rect) - CGRectGetWidth(rect)) / 2;
}
CGContextDrawImage(cgcontext, rect, windowImage);
[appImage drawInRect:NSMakeRect(156, 4, 96, 96)];
[dockIcon unlockFocus];
CGImageRelease(windowImage);
NSImageView* imageView = (NSImageView*)self.dockTile.contentView;
if (![imageView isKindOfClass:[NSImageView class]])
{
imageView = [[[NSImageView alloc] initWithFrame:NSMakeRect(0, 0, 256, 256)] autorelease];
imageView.imageScaling = NSImageScaleProportionallyUpOrDown;
self.dockTile.contentView = imageView;
}
imageView.image = dockIcon;
[self.dockTile display];
lastDockIconSnapshot = now;
}
- (void) checkEmptyShaped
{
if (self.dockTile.contentView && ![self isEmptyShaped])
{
self.dockTile.contentView = nil;
lastDockIconSnapshot = 0;
}
}
/* /*
* ---------- NSWindow method overrides ---------- * ---------- NSWindow method overrides ----------
@ -1766,6 +1873,10 @@ - (void) miniaturize:(id)sender
macdrv_event* event = macdrv_create_event(WINDOW_MINIMIZE_REQUESTED, self); macdrv_event* event = macdrv_create_event(WINDOW_MINIMIZE_REQUESTED, self);
[queue postEvent:event]; [queue postEvent:event];
macdrv_release_event(event); macdrv_release_event(event);
WineWindow* parent = (WineWindow*)self.parentWindow;
if ([parent isKindOfClass:[WineWindow class]])
[parent grabDockIconSnapshotFromWindow:self force:YES];
} }
- (void) toggleFullScreen:(id)sender - (void) toggleFullScreen:(id)sender
@ -2123,6 +2234,7 @@ - (void) windowWillExitFullScreen:(NSNotification*)notification
- (void)windowWillMiniaturize:(NSNotification *)notification - (void)windowWillMiniaturize:(NSNotification *)notification
{ {
[self becameIneligibleParentOrChild]; [self becameIneligibleParentOrChild];
[self grabDockIconSnapshotFromWindow:nil force:NO];
} }
- (NSSize) windowWillResize:(NSWindow*)sender toSize:(NSSize)frameSize - (NSSize) windowWillResize:(NSWindow*)sender toSize:(NSSize)frameSize
@ -2584,6 +2696,7 @@ void macdrv_set_window_shape(macdrv_window w, const CGRect *rects, int count)
{ {
window.shape = nil; window.shape = nil;
window.shapeData = nil; window.shapeData = nil;
[window checkEmptyShaped];
} }
else else
{ {
@ -2598,6 +2711,7 @@ void macdrv_set_window_shape(macdrv_window w, const CGRect *rects, int count)
[path appendBezierPathWithRect:NSRectFromCGRect(rects[i])]; [path appendBezierPathWithRect:NSRectFromCGRect(rects[i])];
window.shape = path; window.shape = path;
window.shapeData = [NSData dataWithBytes:rects length:length]; window.shapeData = [NSData dataWithBytes:rects length:length];
[window checkEmptyShaped];
} }
} }
}); });