2014-03-03 12:36:47 +00:00
|
|
|
// This file is part of BOINC.
|
|
|
|
// http://boinc.berkeley.edu
|
2016-12-02 12:08:43 +00:00
|
|
|
// Copyright (C) 2016 University of California
|
2014-03-03 12:36:47 +00:00
|
|
|
//
|
|
|
|
// BOINC 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.
|
|
|
|
//
|
|
|
|
// BOINC 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 BOINC. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
// BOINCGUIApp.mm
|
|
|
|
|
|
|
|
#include "MacGUI.pch"
|
|
|
|
#include "BOINCGUIApp.h"
|
2016-12-02 12:08:43 +00:00
|
|
|
#include "BOINCBaseFrame.h"
|
2014-03-03 12:36:47 +00:00
|
|
|
#import <Cocoa/Cocoa.h>
|
|
|
|
|
2014-05-28 13:00:47 +00:00
|
|
|
#if !wxCHECK_VERSION(3,0,1)
|
2014-04-14 11:25:30 +00:00
|
|
|
// This should be fixed after wxCocoa 3.0.0:
|
|
|
|
// http://trac.wxwidgets.org/ticket/16156
|
2014-03-03 12:36:47 +00:00
|
|
|
|
2016-12-02 12:08:43 +00:00
|
|
|
#ifndef NSEventTypeApplicationDefined
|
|
|
|
#define NSEventTypeApplicationDefined NSApplicationDefined
|
|
|
|
#endif
|
|
|
|
|
2014-03-03 12:36:47 +00:00
|
|
|
// Cocoa routines which are part of CBOINCGUIApp
|
|
|
|
// Override standard wxCocoa wxApp::CallOnInit() to allow Manager
|
|
|
|
// to run properly when launched hidden on login via Login Item.
|
|
|
|
bool CBOINCGUIApp::CallOnInit() {
|
|
|
|
NSAutoreleasePool *mypool = [[NSAutoreleasePool alloc] init];
|
|
|
|
|
2016-12-02 12:08:43 +00:00
|
|
|
NSEvent *event = [NSEvent otherEventWithType:NSEventTypeApplicationDefined
|
2014-03-03 12:36:47 +00:00
|
|
|
location:NSMakePoint(0.0, 0.0)
|
|
|
|
modifierFlags:0
|
|
|
|
timestamp:0
|
|
|
|
windowNumber:0
|
|
|
|
context:nil
|
|
|
|
subtype:0 data1:0 data2:0];
|
|
|
|
[NSApp postEvent:event atStart:FALSE];
|
|
|
|
|
|
|
|
bool retVal = wxApp::CallOnInit();
|
|
|
|
|
|
|
|
[mypool release];
|
|
|
|
return retVal;
|
|
|
|
}
|
2014-04-14 11:25:30 +00:00
|
|
|
#endif
|
2014-03-03 12:36:47 +00:00
|
|
|
|
2014-03-14 10:53:12 +00:00
|
|
|
|
|
|
|
// Our application can get into a strange state
|
|
|
|
// if our login item launched it hidden and the
|
|
|
|
// first time the user "opens" it he either
|
|
|
|
// double-clicks on our Finder icon or uses
|
|
|
|
// command-tab. It becomes the frontmost
|
|
|
|
// application (with its menu in the menubar)
|
|
|
|
// but the windows remain hidden, and it does
|
|
|
|
// not receive an activate event, so we must
|
|
|
|
// handle this case by polling.
|
|
|
|
//
|
|
|
|
// We can stop the polling after the windows
|
|
|
|
// have been shown once, since this state only
|
|
|
|
// occurs if the windows have never appeared.
|
|
|
|
//
|
|
|
|
// TODO: Can we avoid polling by using notification
|
|
|
|
// TODO: [NSApplicationDelegate applicationDidUnhide: ] ?
|
|
|
|
//
|
|
|
|
void CBOINCGUIApp::CheckPartialActivation() {
|
|
|
|
// This code is not needed and has bad effects on OS 10.5.
|
2014-03-15 11:56:14 +00:00
|
|
|
// Initializing wasHidden this way avoids the problem
|
|
|
|
// because we are briefly shown at login on OS 10.5.
|
|
|
|
static bool wasHidden = [ NSApp isHidden ];
|
2014-03-14 10:53:12 +00:00
|
|
|
|
2014-03-15 11:56:14 +00:00
|
|
|
if (wasHidden) {
|
|
|
|
if (m_bAboutDialogIsOpen) return;
|
2014-03-14 10:53:12 +00:00
|
|
|
|
2014-03-15 11:56:14 +00:00
|
|
|
if (! [ NSApp isHidden ]) {
|
|
|
|
wasHidden = false;
|
2014-03-14 10:53:12 +00:00
|
|
|
ShowInterface();
|
|
|
|
}
|
2014-03-15 11:56:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-12-02 12:08:43 +00:00
|
|
|
// Returns true if file was modified since system was booted, else false
|
|
|
|
//
|
|
|
|
bool CBOINCGUIApp::WasFileModifiedBeforeSystemBoot(char * filePath) {
|
|
|
|
NSTimeInterval upTime = [[NSProcessInfo processInfo] systemUptime];
|
|
|
|
NSString *path = [NSString stringWithUTF8String:filePath];
|
|
|
|
NSError *error = nil;
|
|
|
|
NSDictionary * attrs = [[NSFileManager defaultManager] attributesOfItemAtPath:path error:&error];
|
|
|
|
if (attrs && !error) { // If file exists, then ...
|
|
|
|
NSDate *fileLastModifiedDate = [attrs fileModificationDate];
|
|
|
|
NSTimeInterval ageOfFile = -[fileLastModifiedDate timeIntervalSinceNow];
|
|
|
|
return (ageOfFile > upTime);
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-15 11:56:14 +00:00
|
|
|
// HideThisApp() is called from CBOINCGUIApp::ShowApplication(bool)
|
|
|
|
// and replaces a call of ShowHideProcess() which is deprecated
|
|
|
|
// under OS 10.9.
|
|
|
|
void CBOINCGUIApp::HideThisApp() {
|
|
|
|
[ NSApp hide:NSApp ];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// Determines if the current process is visible.
|
|
|
|
///
|
|
|
|
/// @return
|
|
|
|
/// true if the current process is visible, otherwise false.
|
|
|
|
///
|
|
|
|
bool CBOINCGUIApp::IsApplicationVisible() {
|
|
|
|
return (! [ NSApp isHidden ]);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
///
|
|
|
|
/// Shows or hides the current process.
|
|
|
|
///
|
|
|
|
/// @param bShow
|
|
|
|
/// true will show the process, false will hide the process.
|
|
|
|
///
|
|
|
|
void CBOINCGUIApp::ShowApplication(bool bShow) {
|
|
|
|
if (bShow) {
|
|
|
|
[ NSApp activateIgnoringOtherApps:YES ];
|
|
|
|
} else {
|
|
|
|
[ NSApp hide:NSApp ];
|
2014-03-14 10:53:12 +00:00
|
|
|
}
|
|
|
|
}
|
2016-12-02 12:08:43 +00:00
|
|
|
|
2017-10-19 13:35:01 +00:00
|
|
|
|
|
|
|
// Returns true if at least a 5 X 5 pixel area of the
|
|
|
|
// window's title bar is entirely on the displays
|
|
|
|
// Note: Arguments are Quickdraw-style coordinates,
|
|
|
|
// but CGDisplayBounds() sets top left corner as (0, 0)
|
|
|
|
Boolean IsWindowOnScreen(int iLeft, int iTop, int iWidth, int iHeight) {
|
|
|
|
CGRect intersectedRect;
|
|
|
|
CGRect titleRect = CGRectMake(iLeft, iTop, iWidth, 22);
|
|
|
|
// Make sure at least a 5X5 piece of title bar is visible
|
|
|
|
titleRect = CGRectInset(titleRect, 5, 5);
|
|
|
|
|
|
|
|
|
|
|
|
NSArray *allScreens = [NSScreen screens];
|
|
|
|
unsigned int i;
|
|
|
|
// The geometries of windows and display arangements are such
|
|
|
|
// that even if the title bar spans multiple windows, a 5X5
|
|
|
|
// section is on-screen only if at least one 5X5 section is
|
|
|
|
// entirely on one or more displays, so this test is sufficient.
|
|
|
|
unsigned int numDisplays = [ allScreens count ];
|
|
|
|
for (i=0; i<numDisplays; i++) {
|
|
|
|
NSScreen *aScreen = (NSScreen *)[ allScreens objectAtIndex:i ];
|
|
|
|
NSRect visibleRect = [aScreen visibleFrame]; // Usable area of screen
|
|
|
|
// Convert to QuickDraw coordinates (Y=0 at top of screen)
|
|
|
|
NSRect fullScreenRect = [aScreen frame];
|
|
|
|
visibleRect.origin.y = fullScreenRect.size.height - visibleRect.origin.y - visibleRect.size.height;
|
|
|
|
|
|
|
|
intersectedRect = CGRectIntersection(visibleRect, titleRect);
|
|
|
|
if (! CGRectIsNull(intersectedRect)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
2017-10-18 14:19:15 +00:00
|
|
|
}
|
|
|
|
|
2016-12-02 12:08:43 +00:00
|
|
|
|
|
|
|
extern bool s_bSkipExitConfirmation;
|
|
|
|
|
|
|
|
// Set s_bSkipExitConfirmation to true if cancelled because of logging out or shutting down
|
|
|
|
//
|
|
|
|
OSErr QuitAppleEventHandler( const AppleEvent *appleEvt, AppleEvent* reply, UInt32 refcon ) {
|
|
|
|
DescType senderType;
|
|
|
|
Size actualSize;
|
|
|
|
pid_t senderPid;
|
|
|
|
OSStatus anErr;
|
|
|
|
|
|
|
|
// Refuse to quit if a modal dialog is open.
|
|
|
|
// Unfortunately, I know of no way to disable the Quit item in our Dock menu
|
|
|
|
if (wxGetApp().IsModalDialogDisplayed()) {
|
|
|
|
NSBeep();
|
|
|
|
return userCanceledErr;
|
|
|
|
}
|
|
|
|
|
|
|
|
anErr = AEGetAttributePtr(appleEvt, keySenderPIDAttr, typeSInt32,
|
|
|
|
&senderType, &senderPid, sizeof(senderPid), &actualSize);
|
|
|
|
if (anErr == noErr) {
|
|
|
|
NSString * bundleID = [[NSRunningApplication runningApplicationWithProcessIdentifier:senderPid] bundleIdentifier];
|
|
|
|
// Consider a Quit command from our Dock menu as coming from this application
|
|
|
|
if (bundleID) {
|
|
|
|
if (([bundleID compare:@"com.apple.dock"] != NSOrderedSame)
|
|
|
|
&& ([bundleID compare:@"edu.berkeley.boinc"] != NSOrderedSame)) {
|
|
|
|
s_bSkipExitConfirmation = true; // Not from our app, our dock icon or our taskbar icon
|
|
|
|
// The following may no longer be needed under wxCocoa-3.0.0
|
|
|
|
wxGetApp().ExitMainLoop(); // Prevents wxMac from issuing events to closed frames
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
wxCommandEvent evt(wxEVT_COMMAND_MENU_SELECTED, wxID_EXIT);
|
|
|
|
wxGetApp().GetFrame()->GetEventHandler()->AddPendingEvent(evt);
|
|
|
|
return noErr;
|
|
|
|
}
|