2022-12-04 11:03:45 +00:00
|
|
|
// SPDX-FileCopyrightText: 2019-2022 Connor McLaughlin <stenzek@gmail.com>
|
|
|
|
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
|
|
|
|
|
2022-09-21 12:44:52 +00:00
|
|
|
#include "platform_misc.h"
|
2023-08-23 12:06:48 +00:00
|
|
|
#include "window_info.h"
|
|
|
|
#include "cocoa_tools.h"
|
|
|
|
|
2022-09-21 12:44:52 +00:00
|
|
|
#include "common/log.h"
|
2023-09-20 13:49:14 +00:00
|
|
|
#include "common/small_string.h"
|
2023-08-23 12:06:48 +00:00
|
|
|
|
2022-09-21 12:44:52 +00:00
|
|
|
#include <IOKit/pwr_mgt/IOPMLib.h>
|
2023-02-05 05:00:51 +00:00
|
|
|
#include <Cocoa/Cocoa.h>
|
|
|
|
#include <QuartzCore/QuartzCore.h>
|
2022-09-21 12:44:52 +00:00
|
|
|
#include <cinttypes>
|
2023-02-05 05:00:51 +00:00
|
|
|
#include <vector>
|
2022-09-21 12:44:52 +00:00
|
|
|
|
2023-08-23 12:06:48 +00:00
|
|
|
Log_SetChannel(PlatformMisc);
|
|
|
|
|
|
|
|
#if __has_feature(objc_arc)
|
|
|
|
#error ARC should not be enabled.
|
|
|
|
#endif
|
2022-09-21 12:44:52 +00:00
|
|
|
|
|
|
|
static IOPMAssertionID s_prevent_idle_assertion = kIOPMNullAssertionID;
|
|
|
|
|
|
|
|
static bool SetScreensaverInhibitMacOS(bool inhibit)
|
|
|
|
{
|
|
|
|
if (inhibit)
|
|
|
|
{
|
|
|
|
const CFStringRef reason = CFSTR("System Running");
|
|
|
|
if (IOPMAssertionCreateWithName(kIOPMAssertionTypePreventUserIdleDisplaySleep, kIOPMAssertionLevelOn, reason,
|
|
|
|
&s_prevent_idle_assertion) != kIOReturnSuccess)
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("IOPMAssertionCreateWithName() failed");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
IOPMAssertionRelease(s_prevent_idle_assertion);
|
|
|
|
s_prevent_idle_assertion = kIOPMNullAssertionID;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool s_screensaver_suspended;
|
|
|
|
|
2023-08-23 12:06:48 +00:00
|
|
|
void PlatformMisc::SuspendScreensaver()
|
2022-09-21 12:44:52 +00:00
|
|
|
{
|
2022-11-18 08:14:39 +00:00
|
|
|
if (s_screensaver_suspended)
|
2022-09-21 12:44:52 +00:00
|
|
|
|
|
|
|
if (!SetScreensaverInhibitMacOS(true))
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Failed to suspend screensaver.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
s_screensaver_suspended = true;
|
|
|
|
}
|
|
|
|
|
2023-08-23 12:06:48 +00:00
|
|
|
void PlatformMisc::ResumeScreensaver()
|
2022-09-21 12:44:52 +00:00
|
|
|
{
|
|
|
|
if (!s_screensaver_suspended)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!SetScreensaverInhibitMacOS(false))
|
|
|
|
Log_ErrorPrint("Failed to resume screensaver.");
|
|
|
|
|
|
|
|
s_screensaver_suspended = false;
|
|
|
|
}
|
|
|
|
|
2023-08-23 12:06:48 +00:00
|
|
|
bool PlatformMisc::PlaySoundAsync(const char* path)
|
2022-09-21 12:44:52 +00:00
|
|
|
{
|
|
|
|
NSString* nspath = [[NSString alloc] initWithUTF8String:path];
|
|
|
|
NSSound* sound = [[NSSound alloc] initWithContentsOfFile:nspath byReference:YES];
|
|
|
|
const bool result = [sound play];
|
|
|
|
[sound release];
|
|
|
|
[nspath release];
|
|
|
|
return result;
|
|
|
|
}
|
2023-02-05 05:00:51 +00:00
|
|
|
|
2023-08-23 12:06:48 +00:00
|
|
|
NSString* CocoaTools::StringViewToNSString(const std::string_view& str)
|
|
|
|
{
|
|
|
|
if (str.empty())
|
|
|
|
return nil;
|
|
|
|
|
|
|
|
return [[[NSString alloc] initWithBytes:str.data()
|
|
|
|
length:static_cast<NSUInteger>(str.length())
|
|
|
|
encoding:NSUTF8StringEncoding] autorelease];
|
|
|
|
}
|
|
|
|
|
2023-02-05 05:00:51 +00:00
|
|
|
// From https://github.com/PCSX2/pcsx2/blob/1b673d9dd0829a48f5f0b6604c1de2108e981399/common/CocoaTools.mm
|
|
|
|
|
|
|
|
@interface PCSX2KVOHelper : NSObject
|
|
|
|
|
|
|
|
- (void)addCallback:(void*)ctx run:(void(*)(void*))callback;
|
|
|
|
- (void)removeCallback:(void*)ctx;
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation PCSX2KVOHelper
|
|
|
|
{
|
|
|
|
std::vector<std::pair<void*, void(*)(void*)>> _callbacks;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)addCallback:(void*)ctx run:(void(*)(void*))callback
|
|
|
|
{
|
|
|
|
_callbacks.push_back(std::make_pair(ctx, callback));
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)removeCallback:(void*)ctx
|
|
|
|
{
|
|
|
|
auto new_end = std::remove_if(_callbacks.begin(), _callbacks.end(), [ctx](const auto& entry){
|
|
|
|
return ctx == entry.first;
|
|
|
|
});
|
|
|
|
_callbacks.erase(new_end, _callbacks.end());
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context
|
|
|
|
{
|
|
|
|
for (const auto& callback : _callbacks)
|
|
|
|
callback.second(callback.first);
|
|
|
|
}
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
static PCSX2KVOHelper* s_themeChangeHandler;
|
|
|
|
|
2023-08-23 12:06:48 +00:00
|
|
|
void CocoaTools::AddThemeChangeHandler(void* ctx, void(handler)(void* ctx))
|
2023-02-05 05:00:51 +00:00
|
|
|
{
|
|
|
|
assert([NSThread isMainThread]);
|
|
|
|
if (!s_themeChangeHandler)
|
|
|
|
{
|
|
|
|
s_themeChangeHandler = [[PCSX2KVOHelper alloc] init];
|
|
|
|
NSApplication* app = [NSApplication sharedApplication];
|
|
|
|
[app addObserver:s_themeChangeHandler
|
|
|
|
forKeyPath:@"effectiveAppearance"
|
|
|
|
options:0
|
|
|
|
context:nil];
|
|
|
|
}
|
|
|
|
[s_themeChangeHandler addCallback:ctx run:handler];
|
|
|
|
}
|
|
|
|
|
2023-08-23 12:06:48 +00:00
|
|
|
void CocoaTools::RemoveThemeChangeHandler(void* ctx)
|
2023-02-05 05:00:51 +00:00
|
|
|
{
|
|
|
|
assert([NSThread isMainThread]);
|
|
|
|
[s_themeChangeHandler removeCallback:ctx];
|
|
|
|
}
|
2023-08-24 08:02:55 +00:00
|
|
|
|
|
|
|
bool CocoaTools::CreateMetalLayer(WindowInfo *wi)
|
|
|
|
{
|
|
|
|
// Punt off to main thread if we're not calling from it already.
|
|
|
|
if (![NSThread isMainThread])
|
|
|
|
{
|
|
|
|
bool ret;
|
|
|
|
dispatch_sync(dispatch_get_main_queue(), [&ret, wi]() {
|
|
|
|
ret = CreateMetalLayer(wi);
|
|
|
|
});
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
CAMetalLayer* layer = [CAMetalLayer layer];
|
|
|
|
if (layer == nil)
|
|
|
|
{
|
|
|
|
Log_ErrorPrint("Failed to create CAMetalLayer");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
NSView* view = (__bridge NSView*)wi->window_handle;
|
|
|
|
[view setWantsLayer:TRUE];
|
|
|
|
[view setLayer:layer];
|
|
|
|
[layer setContentsScale:[[[view window] screen] backingScaleFactor]];
|
|
|
|
|
|
|
|
wi->surface_handle = (__bridge void*)layer;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CocoaTools::DestroyMetalLayer(WindowInfo *wi)
|
|
|
|
{
|
|
|
|
if (!wi->surface_handle)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Punt off to main thread if we're not calling from it already.
|
|
|
|
if (![NSThread isMainThread])
|
|
|
|
{
|
|
|
|
dispatch_sync(dispatch_get_main_queue(), [wi]() { DestroyMetalLayer(wi); });
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
NSView* view = (__bridge NSView*)wi->window_handle;
|
|
|
|
CAMetalLayer* layer = (__bridge CAMetalLayer*)wi->surface_handle;
|
|
|
|
[view setLayer:nil];
|
|
|
|
[view setWantsLayer:NO];
|
|
|
|
[layer release];
|
|
|
|
}
|