
No hard resolution switch is done. Instead a view of the requested size is displayed in the center of the main screen (letter boxing). Running a stress test showed that the performance were equal to a hard switch without the inconvenience of resizing the user's apps. This also follows the guidelines defined by Apple: > Avoid changing the display resolution from that chosen by the user. source: https://developer.apple.com/library/mac/documentation/graphicsimaging/Conceptual/OpenGL-MacProgGuide/opengl_fullscreen/opengl_cgl.html
587 lines
17 KiB
Plaintext
587 lines
17 KiB
Plaintext
////////////////////////////////////////////////////////////
|
|
//
|
|
// SFML - Simple and Fast Multimedia Library
|
|
// Copyright (C) 2007-2014 Marco Antognini (antognini.marco@gmail.com),
|
|
// Laurent Gomila (laurent.gom@gmail.com),
|
|
//
|
|
// This software is provided 'as-is', without any express or implied warranty.
|
|
// In no event will the authors be held liable for any damages arising from the use of this software.
|
|
//
|
|
// Permission is granted to anyone to use this software for any purpose,
|
|
// including commercial applications, and to alter it and redistribute it freely,
|
|
// subject to the following restrictions:
|
|
//
|
|
// 1. The origin of this software must not be misrepresented;
|
|
// you must not claim that you wrote the original software.
|
|
// If you use this software in a product, an acknowledgment
|
|
// in the product documentation would be appreciated but is not required.
|
|
//
|
|
// 2. Altered source versions must be plainly marked as such,
|
|
// and must not be misrepresented as being the original software.
|
|
//
|
|
// 3. This notice may not be removed or altered from any source distribution.
|
|
//
|
|
////////////////////////////////////////////////////////////
|
|
|
|
////////////////////////////////////////////////////////////
|
|
// Headers
|
|
////////////////////////////////////////////////////////////
|
|
#include <SFML/Window/VideoMode.hpp>
|
|
#include <SFML/Window/WindowHandle.hpp>
|
|
#include <SFML/Window/WindowStyle.hpp>
|
|
#include <SFML/Window/OSX/WindowImplCocoa.hpp>
|
|
#include <SFML/System/Err.hpp>
|
|
#include <ApplicationServices/ApplicationServices.h>
|
|
|
|
#import <SFML/Window/OSX/SFApplication.h>
|
|
#import <SFML/Window/OSX/SFOpenGLView.h>
|
|
#import <SFML/Window/OSX/SFWindow.h>
|
|
#import <SFML/Window/OSX/SFWindowController.h>
|
|
#import <OpenGL/OpenGL.h>
|
|
|
|
////////////////////////////////////////////////////////////
|
|
/// SFBlackView is a simple view filled with black, nothing more
|
|
///
|
|
////////////////////////////////////////////////////////////
|
|
@interface SFBlackView : NSView
|
|
@end
|
|
|
|
@implementation SFBlackView
|
|
|
|
////////////////////////////////////////////////////////////
|
|
-(void)drawRect:(NSRect)dirtyRect
|
|
{
|
|
[[NSColor blackColor] setFill];
|
|
NSRectFill(dirtyRect);
|
|
}
|
|
|
|
@end
|
|
|
|
////////////////////////////////////////////////////////////
|
|
/// SFWindowController class: private interface
|
|
///
|
|
////////////////////////////////////////////////////////////
|
|
@interface SFWindowController ()
|
|
|
|
////////////////////////////////////////////////////////////
|
|
/// \brief Retrieves the screen height
|
|
///
|
|
/// \return screen height
|
|
///
|
|
////////////////////////////////////////////////////////////
|
|
-(float)screenHeight;
|
|
|
|
////////////////////////////////////////////////////////////
|
|
/// \brief Retrieves the title bar height
|
|
///
|
|
/// \return title bar height
|
|
///
|
|
////////////////////////////////////////////////////////////
|
|
-(float)titlebarHeight;
|
|
|
|
@end
|
|
|
|
@implementation SFWindowController
|
|
|
|
#pragma mark
|
|
#pragma mark SFWindowController's methods
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(id)initWithWindow:(NSWindow*)window
|
|
{
|
|
if ((self = [super init]))
|
|
{
|
|
m_window = nil;
|
|
m_oglView = nil;
|
|
m_requester = 0;
|
|
m_fullscreen = NO;
|
|
|
|
// Retain the window for our own use.
|
|
m_window = window;
|
|
|
|
if (m_window == nil)
|
|
{
|
|
sf::err() << "No window was given to -[SFWindowController -initWithWindow:]." << std::endl;
|
|
return self;
|
|
}
|
|
|
|
// Create the view.
|
|
m_oglView = [[SFOpenGLView alloc] initWithFrame:[[m_window contentView] frame]];
|
|
|
|
if (m_oglView == nil)
|
|
{
|
|
sf::err() << "Could not create an instance of NSOpenGLView "
|
|
<< "in -[SFWindowController -initWithWindow:]."
|
|
<< std::endl;
|
|
return self;
|
|
}
|
|
|
|
// Set the view to the window as its content view.
|
|
[m_window setContentView:m_oglView];
|
|
}
|
|
|
|
return self;
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(id)initWithMode:(const sf::VideoMode&)mode andStyle:(unsigned long)style
|
|
{
|
|
// If we are not on the main thread we stop here and advice the user.
|
|
if ([NSThread currentThread] != [NSThread mainThread])
|
|
{
|
|
/*
|
|
* See http://lists.apple.com/archives/cocoa-dev/2011/Feb/msg00460.html
|
|
* for more information.
|
|
*/
|
|
sf::err() << "Cannot create a window from a worker thread. (OS X limitation)" << std::endl;
|
|
|
|
return nil;
|
|
}
|
|
|
|
if ((self = [super init]))
|
|
{
|
|
m_window = nil;
|
|
m_oglView = nil;
|
|
m_requester = 0;
|
|
m_fullscreen = style & sf::Style::Fullscreen;
|
|
|
|
if (m_fullscreen)
|
|
[self setupFullscreenViewWithMode:mode];
|
|
else
|
|
[self setupWindowWithMode:mode andStyle:style];
|
|
}
|
|
return self;
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(void)setupFullscreenViewWithMode:(const sf::VideoMode&)mode
|
|
{
|
|
// Create a screen-sized window on the main display
|
|
sf::VideoMode desktop = sf::VideoMode::getDesktopMode();
|
|
NSRect windowRect = NSMakeRect(0, 0, desktop.width, desktop.height);
|
|
m_window = [[SFWindow alloc] initWithContentRect:windowRect
|
|
styleMask:NSBorderlessWindowMask
|
|
backing:NSBackingStoreBuffered
|
|
defer:NO];
|
|
|
|
if (m_window == nil)
|
|
{
|
|
sf::err() << "Could not create an instance of NSWindow "
|
|
<< "in -[SFWindowController setupFullscreenViewWithMode:]."
|
|
<< std::endl;
|
|
return;
|
|
}
|
|
|
|
// Set the window level to be above the menu bar
|
|
[m_window setLevel:NSMainMenuWindowLevel+1];
|
|
|
|
// More window configuration...
|
|
[m_window setOpaque:YES];
|
|
[m_window setHidesOnDeactivate:YES];
|
|
[m_window setAutodisplay:YES];
|
|
[m_window setReleasedWhenClosed:YES];
|
|
|
|
// Register for event
|
|
[m_window setDelegate:self];
|
|
[m_window setAcceptsMouseMovedEvents:YES];
|
|
[m_window setIgnoresMouseEvents:NO];
|
|
|
|
// Create a master view containing our OpenGL view
|
|
NSView* masterView = [[SFBlackView alloc] initWithFrame:windowRect];
|
|
|
|
if (masterView == nil)
|
|
{
|
|
sf::err() << "Could not create an instance of SFBlackView "
|
|
<< "in -[SFWindowController setupFullscreenViewWithMode:]."
|
|
<< std::endl;
|
|
return;
|
|
}
|
|
|
|
// Create our OpenGL view size and the view
|
|
CGFloat x = (desktop.width - mode.width) / 2.0;
|
|
CGFloat y = (desktop.height - mode.height) / 2.0;
|
|
NSRect oglRect = NSMakeRect(x, y, mode.width, mode.height);
|
|
|
|
m_oglView = [[SFOpenGLView alloc] initWithFrame:oglRect];
|
|
|
|
if (m_oglView == nil)
|
|
{
|
|
sf::err() << "Could not create an instance of NSOpenGLView "
|
|
<< "in -[SFWindowController setupFullscreenViewWithMode:]."
|
|
<< std::endl;
|
|
return;
|
|
}
|
|
|
|
// Populate the window and views
|
|
[masterView addSubview:m_oglView];
|
|
[m_window setContentView:masterView];
|
|
|
|
// Finalize setup
|
|
[m_oglView enterFullscreen];
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(void)setupWindowWithMode:(const sf::VideoMode&)mode andStyle:(unsigned long)style
|
|
{
|
|
// We know that style & sf::Style::Fullscreen is false.
|
|
|
|
// Create our window size.
|
|
NSRect rect = NSMakeRect(0, 0, mode.width, mode.height);
|
|
|
|
// Convert the SFML window style to Cocoa window style.
|
|
unsigned int nsStyle = NSBorderlessWindowMask;
|
|
if (style & sf::Style::Titlebar)
|
|
nsStyle |= NSTitledWindowMask | NSMiniaturizableWindowMask;
|
|
if (style & sf::Style::Resize)
|
|
nsStyle |= NSResizableWindowMask;
|
|
if (style & sf::Style::Close)
|
|
nsStyle |= NSClosableWindowMask;
|
|
|
|
// Create the window.
|
|
m_window = [[SFWindow alloc] initWithContentRect:rect
|
|
styleMask:nsStyle
|
|
backing:NSBackingStoreBuffered
|
|
defer:NO]; // Don't defer it!
|
|
/*
|
|
"YES" produces some "invalid drawable".
|
|
See http://www.cocoabuilder.com/archive/cocoa/152482-nsviews-and-nsopenglcontext-invalid-drawable-error.html
|
|
|
|
[...]
|
|
As best as I can figure, this is happening because the NSWindow (and
|
|
hence my view) are not visible on screen yet, and the system doesn't like that.
|
|
[...]
|
|
*/
|
|
|
|
if (m_window == nil)
|
|
{
|
|
sf::err() << "Could not create an instance of NSWindow "
|
|
<< "in -[SFWindowController setupWindowWithMode:andStyle:]."
|
|
<< std::endl;
|
|
|
|
return;
|
|
}
|
|
|
|
// Create the view.
|
|
m_oglView = [[SFOpenGLView alloc] initWithFrame:[[m_window contentView] frame]];
|
|
|
|
if (m_oglView == nil)
|
|
{
|
|
sf::err() << "Could not create an instance of NSOpenGLView "
|
|
<< "in -[SFWindowController setupWindowWithMode:andStyle:]."
|
|
<< std::endl;
|
|
|
|
return;
|
|
}
|
|
|
|
// Set the view to the window as its content view.
|
|
[m_window setContentView:m_oglView];
|
|
|
|
// Register for event.
|
|
[m_window setDelegate:self];
|
|
[m_window setAcceptsMouseMovedEvents:YES];
|
|
[m_window setIgnoresMouseEvents:NO];
|
|
|
|
// And some other things...
|
|
[m_window center];
|
|
[m_window setAutodisplay:YES];
|
|
[m_window setReleasedWhenClosed:YES];
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(void)dealloc
|
|
{
|
|
[self closeWindow];
|
|
[NSMenu setMenuBarVisible:YES];
|
|
|
|
m_window = nil;
|
|
m_oglView = nil;
|
|
}
|
|
|
|
|
|
#pragma mark
|
|
#pragma mark WindowImplDelegateProtocol's methods
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(void)setRequesterTo:(sf::priv::WindowImplCocoa*)requester
|
|
{
|
|
// Forward to the view.
|
|
[m_oglView setRequesterTo:requester];
|
|
m_requester = requester;
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(sf::WindowHandle)getSystemHandle
|
|
{
|
|
return (__bridge sf::WindowHandle)m_window;
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(void)hideMouseCursor
|
|
{
|
|
[NSCursor hide];
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(void)showMouseCursor
|
|
{
|
|
[NSCursor unhide];
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////
|
|
-(NSPoint)position
|
|
{
|
|
// First, get the top left corner of the view in its own base system
|
|
const NSPoint origin = [m_oglView frame].origin;
|
|
const NSSize size = [m_oglView frame].size;
|
|
const NSPoint topLeftCornerOfView = NSMakePoint(origin.x, origin.y + size.height);
|
|
const NSPoint positionInView = [m_oglView convertPointToBase:topLeftCornerOfView];
|
|
|
|
// Then, convert it to window base system
|
|
const NSPoint positionInWindow = [m_oglView convertPoint:positionInView toView:nil];
|
|
// here nil denotes the window containing the view
|
|
|
|
// Next, convert it to the screen base system
|
|
const NSPoint positionInScreen = [[m_oglView window] convertBaseToScreen:positionInWindow];
|
|
|
|
// Finally, flip for SFML window coordinate system
|
|
// Don't forget to discard the title bar !
|
|
const NSPoint positionInSFML = NSMakePoint(positionInScreen.x,
|
|
([self screenHeight] - [self titlebarHeight]) - positionInScreen.y);
|
|
|
|
return positionInSFML;
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////.
|
|
-(void)setWindowPositionToX:(int)x Y:(int)y
|
|
{
|
|
NSPoint point = NSMakePoint(x, y);
|
|
|
|
// Flip for SFML window coordinate system.
|
|
point.y = [self screenHeight] - point.y;
|
|
|
|
// Place the window.
|
|
[m_window setFrameTopLeftPoint:point];
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(NSSize)size
|
|
{
|
|
return [m_oglView frame].size;
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(void)resizeTo:(unsigned int)width by:(unsigned int)height
|
|
{
|
|
// Before resizing, remove resizeable mask to be able to resize
|
|
// beyond the desktop boundaries.
|
|
NSUInteger styleMask = [m_window styleMask];
|
|
|
|
[m_window setStyleMask:styleMask ^ NSResizableWindowMask];
|
|
|
|
// Add titlebar height.
|
|
NSRect frame = NSMakeRect([m_window frame].origin.x,
|
|
[m_window frame].origin.y,
|
|
width,
|
|
height + [self titlebarHeight]);
|
|
|
|
[m_window setFrame:frame display:YES];
|
|
|
|
// And restore the mask
|
|
[m_window setStyleMask:styleMask];
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(void)changeTitle:(NSString*)title
|
|
{
|
|
[m_window setTitle:title];
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(void)hideWindow
|
|
{
|
|
[m_window orderOut:nil];
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(void)showWindow
|
|
{
|
|
[m_window makeKeyAndOrderFront:nil];
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(void)closeWindow
|
|
{
|
|
[m_window close];
|
|
[m_window setDelegate:nil];
|
|
[self setRequesterTo:0];
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(void)enableKeyRepeat
|
|
{
|
|
[m_oglView enableKeyRepeat];
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(void)disableKeyRepeat
|
|
{
|
|
[m_oglView disableKeyRepeat];
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(void)setIconTo:(unsigned int)width
|
|
by:(unsigned int)height
|
|
with:(const sf::Uint8*)pixels
|
|
{
|
|
// Create an empty image representation.
|
|
NSBitmapImageRep* bitmap =
|
|
[[NSBitmapImageRep alloc] initWithBitmapDataPlanes:0 // if 0 : only allocate memory
|
|
pixelsWide:width
|
|
pixelsHigh:height
|
|
bitsPerSample:8 // The number of bits used to specify
|
|
// one pixel in a single component of the data.
|
|
samplesPerPixel:4 // 3 if no alpha, 4 with it
|
|
hasAlpha:YES
|
|
isPlanar:NO // I don't know what it is but it works
|
|
colorSpaceName:NSCalibratedRGBColorSpace
|
|
bytesPerRow:0 // 0 == determine automatically
|
|
bitsPerPixel:0]; // 0 == determine automatically
|
|
|
|
// Load data pixels.
|
|
for (unsigned int y = 0; y < height; ++y)
|
|
{
|
|
for (unsigned int x = 0; x < width; ++x, pixels+=4)
|
|
{
|
|
NSUInteger pixel[4] = { pixels[0], pixels[1], pixels[2], pixels[3] };
|
|
[bitmap setPixel:pixel atX:x y:y];
|
|
}
|
|
}
|
|
|
|
// Create an image from the representation.
|
|
NSImage* icon = [[NSImage alloc] initWithSize:NSMakeSize(width, height)];
|
|
[icon addRepresentation:bitmap];
|
|
|
|
// Set app icon.
|
|
[[SFApplication sharedApplication] setApplicationIconImage:icon];
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(void)processEvent
|
|
{
|
|
// If we are not on the main thread we stop here and advice the user.
|
|
if ([NSThread currentThread] != [NSThread mainThread])
|
|
{
|
|
/*
|
|
* See http://lists.apple.com/archives/cocoa-dev/2011/Feb/msg00460.html
|
|
* for more information.
|
|
*/
|
|
sf::err() << "Cannot fetch event from a worker thread. (OS X restriction)" << std::endl;
|
|
|
|
return;
|
|
}
|
|
|
|
// If we don't have a requester we don't fetch event.
|
|
if (m_requester != 0)
|
|
[SFApplication processEvent];
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(void)applyContext:(NSOpenGLContext*)context
|
|
{
|
|
[m_oglView setOpenGLContext:context];
|
|
[context setView:m_oglView];
|
|
}
|
|
|
|
|
|
#pragma mark
|
|
#pragma mark NSWindowDelegate's methods
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(BOOL)windowShouldClose:(id)sender
|
|
{
|
|
(void)sender;
|
|
|
|
if (m_requester == 0)
|
|
return YES;
|
|
|
|
m_requester->windowClosed();
|
|
return NO;
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(void)windowDidBecomeKey:(NSNotification*)notification
|
|
{
|
|
(void)notification;
|
|
|
|
// Send event.
|
|
if (m_requester == 0)
|
|
return;
|
|
|
|
m_requester->windowGainedFocus();
|
|
|
|
if (m_fullscreen)
|
|
[m_oglView enterFullscreen];
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(void)windowDidResignKey:(NSNotification*)notification
|
|
{
|
|
(void)notification;
|
|
|
|
// Send event.
|
|
if (m_requester == 0)
|
|
return;
|
|
|
|
m_requester->windowLostFocus();
|
|
|
|
if (m_fullscreen)
|
|
[m_oglView exitFullscreen];
|
|
}
|
|
|
|
|
|
#pragma mark
|
|
#pragma mark Other methods
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(float)screenHeight
|
|
{
|
|
NSDictionary* deviceDescription = [[m_window screen] deviceDescription];
|
|
NSNumber* screenNumber = [deviceDescription valueForKey:@"NSScreenNumber"];
|
|
CGDirectDisplayID screenID = (CGDirectDisplayID)[screenNumber intValue];
|
|
CGFloat height = CGDisplayPixelsHigh(screenID);
|
|
return height;
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////
|
|
-(float)titlebarHeight
|
|
{
|
|
return NSHeight([m_window frame]) - NSHeight([[m_window contentView] frame]);
|
|
}
|
|
|
|
@end
|