mirror of
https://github.com/gnustep/libs-back.git
synced 2025-02-24 12:21:34 +00:00
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/back/trunk@26226 72102866-910b-0410-8b05-ffd578937521
479 lines
12 KiB
Objective-C
479 lines
12 KiB
Objective-C
/* WIN32Server - Implements window handling for MSWindows
|
|
|
|
Copyright (C) 2005 Free Software Foundation, Inc.
|
|
|
|
Written by: Tom MacSween <macsweent@sympatico.ca>
|
|
Date August 2005
|
|
|
|
This file is part of the GNU Objective C User Interface Library.
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Lesser General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 3 of the License, or (at your option) any later version.
|
|
|
|
This library is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
License along with this library; see the file COPYING.LIB.
|
|
If not, see <http://www.gnu.org/licenses/> or write to the
|
|
Free Software Foundation, 51 Franklin Street, Fifth Floor,
|
|
Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#include <AppKit/NSEvent.h>
|
|
#include <AppKit/NSWindow.h>
|
|
#include "win32/WIN32Server.h"
|
|
#include "win32/WIN32Geometry.h"
|
|
|
|
@implementation WIN32Server (w32_movesize)
|
|
|
|
- (LRESULT) decodeWM_MOVEParams:(HWND)hwnd : (WPARAM)wParam : (LPARAM)lParam
|
|
{
|
|
NSPoint eventLocation;
|
|
NSRect rect;
|
|
RECT r;
|
|
NSEvent *ev = nil;
|
|
|
|
GetWindowRect(hwnd, &r);
|
|
rect = MSScreenRectToGS(r, [EVENT_WINDOW(hwnd) styleMask], self);
|
|
eventLocation = rect.origin;
|
|
|
|
ev = [NSEvent otherEventWithType: NSAppKitDefined
|
|
location: eventLocation
|
|
modifierFlags: 0
|
|
timestamp: 0
|
|
windowNumber: (int)hwnd
|
|
context: GSCurrentContext()
|
|
subtype: GSAppKitWindowMoved
|
|
data1: rect.origin.x
|
|
data2: rect.origin.y];
|
|
|
|
|
|
if (hwnd == (HWND)flags.menuRef)
|
|
{
|
|
//need native code here?
|
|
if (flags.HOLD_MENU_FOR_MOVE == FALSE)
|
|
{
|
|
[EVENT_WINDOW(hwnd) sendEvent: ev];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (flags.HOLD_TRANSIENT_FOR_MOVE == FALSE)
|
|
[EVENT_WINDOW(hwnd) sendEvent: ev];
|
|
}
|
|
|
|
flags.HOLD_MENU_FOR_MOVE=FALSE;
|
|
flags.HOLD_MINI_FOR_MOVE=FALSE;
|
|
flags.HOLD_TRANSIENT_FOR_MOVE=FALSE;
|
|
|
|
return 0;
|
|
}
|
|
|
|
- (LRESULT) decodeWM_SIZEParams:(HWND)hwnd : (WPARAM)wParam : (LPARAM)lParam
|
|
{
|
|
NSPoint eventLocation;
|
|
NSRect rect;
|
|
RECT r;
|
|
NSEvent *ev =nil;
|
|
|
|
GetWindowRect(hwnd, &r);
|
|
rect = MSScreenRectToGS(r, [EVENT_WINDOW(hwnd) styleMask], self);
|
|
eventLocation = rect.origin;
|
|
|
|
switch ((int)wParam)
|
|
{
|
|
case SIZE_MAXHIDE:
|
|
{
|
|
// stubbed for future development
|
|
}
|
|
break;
|
|
|
|
case SIZE_MAXIMIZED:
|
|
{
|
|
// stubbed for future development
|
|
}
|
|
break;
|
|
|
|
case SIZE_MAXSHOW:
|
|
{
|
|
// stubbed for future development
|
|
}
|
|
break;
|
|
|
|
case SIZE_MINIMIZED:
|
|
{
|
|
if (flags.HOLD_MINI_FOR_SIZE == TRUE) //// this is fix for [5, 25 bug]
|
|
break;
|
|
|
|
/* FIXME ... RFM 06-Mr-2008....
|
|
* I don't understand the following code sending
|
|
* a resize event to the gui when a window is minimised.
|
|
* Minimising a GNUstep window does NOT change its size,
|
|
* rather it causes it to be hidden and replace by its
|
|
* miniwindow counterpart.
|
|
* Probably the correct action therefore is to ensure that the
|
|
* gui knows the window is miniaturised and do nothing else.
|
|
* If we send a resize event telling the gui that its window is
|
|
* now tiny, it will most likely mess up most drawing in the
|
|
* window.
|
|
*/
|
|
if (1)
|
|
{
|
|
[EVENT_WINDOW(hwnd) miniaturize: self];
|
|
break;
|
|
}
|
|
/* Original unused code follows: */
|
|
|
|
// make event
|
|
ev = [NSEvent otherEventWithType: NSAppKitDefined
|
|
location: eventLocation
|
|
modifierFlags: 0
|
|
timestamp: 0
|
|
windowNumber: (int)hwnd
|
|
context: GSCurrentContext()
|
|
subtype: GSAppKitWindowResized
|
|
data1: rect.size.width
|
|
data2: rect.size.height];
|
|
|
|
if (hwnd == (HWND)flags.menuRef)
|
|
{
|
|
if (flags.HOLD_MENU_FOR_SIZE == FALSE)
|
|
{
|
|
[[NSApp mainMenu] setMenuChangedMessagesEnabled: YES];
|
|
[EVENT_WINDOW(hwnd) sendEvent: ev];
|
|
[self resizeBackingStoreFor: hwnd];
|
|
[EVENT_WINDOW(hwnd) miniaturize: self];
|
|
[[NSApp mainMenu] setMenuChangedMessagesEnabled: NO];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (flags.HOLD_TRANSIENT_FOR_SIZE == FALSE)
|
|
{
|
|
[EVENT_WINDOW(hwnd) sendEvent:ev];
|
|
[self resizeBackingStoreFor: hwnd];
|
|
if ([self usesNativeTaskbar])
|
|
{
|
|
[EVENT_WINDOW(hwnd) miniaturize: self];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
|
|
case SIZE_RESTORED:
|
|
{
|
|
// make event
|
|
ev = [NSEvent otherEventWithType: NSAppKitDefined
|
|
location: eventLocation
|
|
modifierFlags: 0
|
|
timestamp: 0
|
|
windowNumber: (int)hwnd
|
|
context: GSCurrentContext()
|
|
subtype: GSAppKitWindowResized
|
|
data1: rect.size.width
|
|
data2: rect.size.height];
|
|
|
|
if (hwnd == (HWND)flags.menuRef)
|
|
{
|
|
if (flags.HOLD_MENU_FOR_SIZE == FALSE)
|
|
{
|
|
[EVENT_WINDOW(hwnd) _setVisible: YES];
|
|
[EVENT_WINDOW(hwnd) sendEvent: ev];
|
|
[self resizeBackingStoreFor: hwnd];
|
|
//[EVENT_WINDOW(hwnd) deminiaturize:self];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (flags.HOLD_TRANSIENT_FOR_SIZE == FALSE)
|
|
{
|
|
[EVENT_WINDOW(hwnd) sendEvent: ev];
|
|
[self resizeBackingStoreFor: hwnd];
|
|
// fixes part one of bug [5, 25] see notes
|
|
if ([self usesNativeTaskbar])
|
|
[EVENT_WINDOW(hwnd) deminiaturize:self];
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
ev = nil;
|
|
flags.HOLD_MENU_FOR_SIZE = FALSE;
|
|
flags.HOLD_MINI_FOR_SIZE = FALSE;
|
|
flags.HOLD_TRANSIENT_FOR_SIZE = FALSE;
|
|
|
|
return 0;
|
|
}
|
|
|
|
- (void) decodeWM_NCCALCSIZEParams: (WPARAM)wParam : (LPARAM)lParam : (HWND)hwnd
|
|
{
|
|
// stub for future dev
|
|
|
|
/*NCCALCSIZE_PARAMS * newRects;
|
|
|
|
NSPoint eventLocation;
|
|
NSRect rect;
|
|
RECT drect;
|
|
NSEvent *ev =nil;
|
|
|
|
if (wParam == TRUE)
|
|
{
|
|
// get first rect from NCCALCSIZE_PARAMS Structure
|
|
newRects=(NCCALCSIZE_PARAMS *)lParam;
|
|
// get rect 1 from array
|
|
drect=newRects->rgrc[1];
|
|
|
|
//create a size event and send it to the window
|
|
rect = MSScreenRectToGS(drect, [EVENT_WINDOW(hwnd) styleMask], self);
|
|
eventLocation = rect.origin;
|
|
|
|
// make event
|
|
ev = [NSEvent otherEventWithType: NSAppKitDefined
|
|
location: eventLocation
|
|
modifierFlags: 0
|
|
timestamp: 0
|
|
windowNumber: (int)hwnd
|
|
context: GSCurrentContext()
|
|
subtype: GSAppKitWindowResized
|
|
data1: rect.size.width
|
|
data2: rect.size.height];
|
|
|
|
[EVENT_WINDOW(hwnd) sendEvent:ev];*/
|
|
//[[EVENT_WINDOW(hwnd) contentView] display];
|
|
//[self resizeBackingStoreFor:hwnd];
|
|
|
|
|
|
/* ev = [NSEvent otherEventWithType: NSAppKitDefined
|
|
location: eventLocation
|
|
modifierFlags: 0
|
|
timestamp: 0
|
|
windowNumber: (int)hwnd
|
|
context: GSCurrentContext()
|
|
subtype: GSAppKitWindowMoved
|
|
data1: rect.origin.x
|
|
data2: rect.origin.y];
|
|
|
|
[EVENT_WINDOW(hwnd) sendEvent:ev];
|
|
|
|
//printf(" Rect 1 =\n%s", [[self MSRectDetails:drect] cString]);
|
|
|
|
}
|
|
|
|
//printf("wParam is %s\n", wParam ? "TRUE" : "FALSE");*/
|
|
}
|
|
|
|
- (void) decodeWM_WINDOWPOSCHANGEDParams: (WPARAM)wParam : (LPARAM)lParam : (HWND)hwnd
|
|
{
|
|
WINDOWPOS *inf = (WINDOWPOS*)lParam;
|
|
|
|
if ((inf->flags & SWP_NOZORDER) == 0)
|
|
{
|
|
HWND hi;
|
|
HWND lo;
|
|
int hl;
|
|
int ll;
|
|
|
|
/* For debugging, log current window stack.
|
|
*/
|
|
if (GSDebugSet(@"WTrace") == YES)
|
|
{
|
|
NSString *s = @"Window list:\n";
|
|
|
|
hi = GetDesktopWindow();
|
|
hi = GetWindow(hi, GW_CHILD);
|
|
if (hi > 0)
|
|
{
|
|
hi = GetWindow(hi, GW_HWNDLAST);
|
|
}
|
|
while (hi > 0)
|
|
{
|
|
TCHAR buf[32];
|
|
|
|
hi = GetNextWindow(hi, GW_HWNDPREV);
|
|
|
|
if (hi > 0
|
|
&& GetClassName(hi, buf, 32) == 18
|
|
&& strncmp(buf, "GNUstepWindowClass", 18) == 0)
|
|
{
|
|
if (GetWindowLong(hi, OFF_ORDERED) == 1)
|
|
{
|
|
hl = GetWindowLong(hi, OFF_LEVEL);
|
|
s = [s stringByAppendingFormat: @"%d (%d)\n", hi, hl];
|
|
}
|
|
}
|
|
}
|
|
NSLog(@"window pos changed: %@", s);
|
|
}
|
|
|
|
/* This window has changed its z-order, so we take the opportunity
|
|
* to check that all the GNUstep windows are in the correct order
|
|
* of their levels.
|
|
* We do this as a simple bubble sort ...swapping over a pait of
|
|
* windows if we find any pair in the list which are in the wrong
|
|
* order. This sort has the virtue of being simple and of
|
|
* maintaining window order within a level. ... but may be too slow.
|
|
*/
|
|
hi = GetDesktopWindow();
|
|
hi = GetWindow(hi, GW_CHILD);
|
|
while (hi > 0)
|
|
{
|
|
TCHAR buf[32];
|
|
|
|
/* Find a GNUstep window which is ordered in and above desktop
|
|
*/
|
|
while (hi > 0)
|
|
{
|
|
if (GetClassName(hi, buf, 32) == 18
|
|
&& strncmp(buf, "GNUstepWindowClass", 18) == 0
|
|
&& GetWindowLong(hi, OFF_ORDERED) == 1
|
|
&& (hl = GetWindowLong(hi, OFF_LEVEL))
|
|
> NSDesktopWindowLevel)
|
|
{
|
|
break;
|
|
}
|
|
hi = GetNextWindow(hi, GW_HWNDNEXT);
|
|
}
|
|
|
|
if (hi > 0)
|
|
{
|
|
NSDebugLLog(@"WTrace", @"sort hi %d (%d)", hi, hl);
|
|
/* Find the next (lower in z-order) GNUstep window which
|
|
* is ordered in and above desktop
|
|
*/
|
|
lo = GetNextWindow(hi, GW_HWNDNEXT);
|
|
while (lo > 0)
|
|
{
|
|
if (GetClassName(lo, buf, 32) == 18
|
|
&& strncmp(buf, "GNUstepWindowClass", 18) == 0
|
|
&& GetWindowLong(lo, OFF_ORDERED) == 1
|
|
&& (ll = GetWindowLong(lo, OFF_LEVEL))
|
|
> NSDesktopWindowLevel)
|
|
{
|
|
break;
|
|
}
|
|
lo = GetNextWindow(lo, GW_HWNDNEXT);
|
|
}
|
|
|
|
if (lo > 0)
|
|
{
|
|
NSDebugLLog(@"WTrace", @"sort lo %d (%d)", lo, ll);
|
|
/* Check to see if the higher of the two windows should
|
|
* actually be lower.
|
|
*/
|
|
if (hl < ll)
|
|
{
|
|
HWND higher;
|
|
|
|
/* Insert the low window before the high one.
|
|
* ie after the window preceding the high one.
|
|
*/
|
|
higher = GetNextWindow(hi, GW_HWNDPREV);
|
|
if (higher == 0)
|
|
{
|
|
higher = HWND_TOP;
|
|
}
|
|
NSDebugLLog(@"WTrace", @"swap %d (%d) with %d (%d)", hi, hl, lo, ll);
|
|
SetWindowPos(lo, higher, 0, 0, 0, 0,
|
|
SWP_NOSIZE | SWP_NOMOVE | SWP_NOACTIVATE);
|
|
|
|
/* Done this iteration of the sort ... the next
|
|
* iteration takes place when we get notified
|
|
* that the swap we have just donew is complete.
|
|
*/
|
|
break;
|
|
}
|
|
}
|
|
hi = lo;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
- (void) decodeWM_WINDOWPOSCHANGINGParams: (WPARAM)wParam : (LPARAM)lParam : (HWND)hwnd
|
|
{
|
|
WINDOWPOS *inf = (WINDOWPOS*)lParam;
|
|
|
|
if ((inf->flags & SWP_NOZORDER) == 0)
|
|
{
|
|
WIN_INTERN *win;
|
|
|
|
/* desktop level windows should stay at the bottom of the
|
|
* window list, so we can simply override any re-ordering
|
|
* to ensure that they are at the bottom unless another
|
|
* desktop level window is inserted below them.
|
|
*/
|
|
win = (WIN_INTERN *)GetWindowLong(hwnd, GWL_USERDATA);
|
|
if (win->level <= NSDesktopWindowLevel)
|
|
{
|
|
inf->hwndInsertAfter = HWND_BOTTOM;
|
|
}
|
|
}
|
|
}
|
|
|
|
- (LRESULT) decodeWM_GETMINMAXINFOParams: (WPARAM)wParam : (LPARAM)lParam : (HWND)hwnd
|
|
{
|
|
// reused from original author (added debug code)
|
|
WIN_INTERN *win = (WIN_INTERN *)GetWindowLong(hwnd, GWL_USERDATA);
|
|
MINMAXINFO *mm;
|
|
|
|
if (win != NULL)
|
|
{
|
|
mm = (MINMAXINFO*)lParam;
|
|
mm->ptMinTrackSize = win->minmax.ptMinTrackSize;
|
|
mm->ptMaxTrackSize = win->minmax.ptMaxTrackSize;
|
|
return 0;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
- (LRESULT) decodeWM_EXITSIZEMOVEParams: (WPARAM)wParam : (LPARAM)lParam : (HWND)hwnd
|
|
{
|
|
// may have a small bug here note it for follow up
|
|
/*
|
|
decodeWM_MOVE and decodeWM_SIZE will send event if they have one.
|
|
no posting is needed.
|
|
*/
|
|
[self resizeBackingStoreFor: hwnd];
|
|
[self decodeWM_MOVEParams:hwnd :wParam :lParam];
|
|
[self decodeWM_SIZEParams:hwnd :wParam :lParam];
|
|
|
|
//Make sure DefWindowProc gets called
|
|
return 0;
|
|
}
|
|
|
|
- (void) decodeWM_SIZINGParams:(HWND)hwnd : (WPARAM)wParam : (LPARAM)lParam
|
|
{
|
|
// stub for future dev
|
|
|
|
flags.HOLD_PAINT_FOR_SIZING=TRUE;
|
|
|
|
//[EVENT_WINDOW(hwnd) displayIfNeeded];
|
|
//[self decodeWM_SIZEParams:(HWND)hwnd : (WPARAM)wParam : (LPARAM)lParam];
|
|
//printf("SIZING called\n");
|
|
|
|
//return TRUE;
|
|
}
|
|
|
|
- (LRESULT) decodeWM_MOVINGParams:(HWND)hwnd : (WPARAM)wParam : (LPARAM)lParam
|
|
{
|
|
// stub for future dev
|
|
[self decodeWM_MOVEParams:(HWND)hwnd : (WPARAM)wParam : (LPARAM)lParam];
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
@end
|
|
|
|
|
|
|