2022-05-25 21:43:11 +00:00
|
|
|
/* PCSX2 - PS2 Emulator for PCs
|
|
|
|
* Copyright (C) 2002-2022 PCSX2 Dev Team
|
|
|
|
*
|
|
|
|
* PCSX2 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 Found-
|
|
|
|
* ation, either version 3 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* PCSX2 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 General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along with PCSX2.
|
|
|
|
* If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#if ! __has_feature(objc_arc)
|
|
|
|
#error "Compile this with -fobjc-arc"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "CocoaTools.h"
|
|
|
|
#include "Console.h"
|
2022-04-18 13:35:14 +00:00
|
|
|
#include "General.h"
|
2022-05-25 21:43:11 +00:00
|
|
|
#include "WindowInfo.h"
|
2022-12-05 06:55:17 +00:00
|
|
|
#include <dlfcn.h>
|
|
|
|
#include <mutex>
|
2022-05-25 22:32:10 +00:00
|
|
|
#include <vector>
|
2022-05-25 21:43:11 +00:00
|
|
|
#include <Cocoa/Cocoa.h>
|
|
|
|
#include <QuartzCore/QuartzCore.h>
|
|
|
|
|
2022-05-25 22:32:10 +00:00
|
|
|
// MARK: - Metal Layers
|
|
|
|
|
2022-05-25 21:43:11 +00:00
|
|
|
bool CocoaTools::CreateMetalLayer(WindowInfo* wi)
|
|
|
|
{
|
|
|
|
if (![NSThread isMainThread])
|
|
|
|
{
|
|
|
|
bool ret;
|
|
|
|
dispatch_sync(dispatch_get_main_queue(), [&ret, wi]{ ret = CreateMetalLayer(wi); });
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
CAMetalLayer* layer = [CAMetalLayer layer];
|
|
|
|
if (!layer)
|
|
|
|
{
|
|
|
|
Console.Error("Failed to create Metal layer.");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
NSView* view = (__bridge NSView*)wi->window_handle;
|
|
|
|
[view setWantsLayer:YES];
|
|
|
|
[view setLayer:layer];
|
|
|
|
[layer setContentsScale:[[[view window] screen] backingScaleFactor]];
|
|
|
|
// Store the layer pointer, that way MoltenVK doesn't call [NSView layer] outside the main thread.
|
|
|
|
wi->surface_handle = (__bridge_retained void*)layer;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CocoaTools::DestroyMetalLayer(WindowInfo* wi)
|
|
|
|
{
|
|
|
|
if (![NSThread isMainThread])
|
|
|
|
{
|
|
|
|
dispatch_sync_f(dispatch_get_main_queue(), wi, [](void* ctx){ DestroyMetalLayer(static_cast<WindowInfo*>(ctx)); });
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
NSView* view = (__bridge NSView*)wi->window_handle;
|
|
|
|
CAMetalLayer* layer = (__bridge_transfer CAMetalLayer*)wi->surface_handle;
|
|
|
|
if (!layer)
|
|
|
|
return;
|
|
|
|
wi->surface_handle = nullptr;
|
|
|
|
[view setLayer:nil];
|
|
|
|
[view setWantsLayer:NO];
|
|
|
|
}
|
2022-05-25 22:32:10 +00:00
|
|
|
|
|
|
|
// MARK: - Theme Change Handlers
|
|
|
|
|
|
|
|
@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;
|
|
|
|
|
|
|
|
void CocoaTools::AddThemeChangeHandler(void* ctx, void(handler)(void* ctx))
|
|
|
|
{
|
|
|
|
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];
|
|
|
|
}
|
|
|
|
|
|
|
|
void CocoaTools::RemoveThemeChangeHandler(void* ctx)
|
|
|
|
{
|
|
|
|
assert([NSThread isMainThread]);
|
|
|
|
[s_themeChangeHandler removeCallback:ctx];
|
|
|
|
}
|
2022-04-18 13:35:14 +00:00
|
|
|
|
|
|
|
// MARK: - Sound playback
|
|
|
|
|
|
|
|
bool Common::PlaySoundAsync(const char* path)
|
|
|
|
{
|
|
|
|
NSString* nspath = [[NSString alloc] initWithUTF8String:path];
|
|
|
|
NSSound* sound = [[NSSound alloc] initWithContentsOfFile:nspath byReference:YES];
|
|
|
|
return [sound play];
|
|
|
|
}
|
2022-12-05 06:55:17 +00:00
|
|
|
|
|
|
|
// MARK: - Updater
|
|
|
|
|
|
|
|
std::optional<std::string> CocoaTools::GetNonTranslocatedBundlePath()
|
|
|
|
{
|
|
|
|
// See https://objective-see.com/blog/blog_0x15.html
|
|
|
|
|
|
|
|
NSURL* url = [NSURL fileURLWithPath:[[NSBundle mainBundle] bundlePath]];
|
|
|
|
if (!url)
|
|
|
|
return std::nullopt;
|
|
|
|
|
|
|
|
if (void* handle = dlopen("/System/Library/Frameworks/Security.framework/Security", RTLD_LAZY))
|
|
|
|
{
|
|
|
|
auto IsTranslocatedURL = reinterpret_cast<Boolean(*)(CFURLRef path, bool* isTranslocated, CFErrorRef*__nullable error)>(dlsym(handle, "SecTranslocateIsTranslocatedURL"));
|
|
|
|
auto CreateOriginalPathForURL = reinterpret_cast<CFURLRef __nullable(*)(CFURLRef translocatedPath, CFErrorRef*__nullable error)>(dlsym(handle, "SecTranslocateCreateOriginalPathForURL"));
|
|
|
|
bool is_translocated = false;
|
|
|
|
if (IsTranslocatedURL)
|
|
|
|
IsTranslocatedURL((__bridge CFURLRef)url, &is_translocated, nullptr);
|
|
|
|
if (is_translocated)
|
|
|
|
{
|
|
|
|
if (CFURLRef actual = CreateOriginalPathForURL((__bridge CFURLRef)url, nullptr))
|
|
|
|
url = (__bridge_transfer NSURL*)actual;
|
|
|
|
}
|
|
|
|
dlclose(handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
return std::string([url fileSystemRepresentation]);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<std::string> CocoaTools::MoveToTrash(std::string_view file)
|
|
|
|
{
|
|
|
|
NSURL* url = [NSURL fileURLWithPath:[[NSString alloc] initWithBytes:file.data() length:file.size() encoding:NSUTF8StringEncoding]];
|
|
|
|
NSURL* new_url;
|
|
|
|
if (![[NSFileManager defaultManager] trashItemAtURL:url resultingItemURL:&new_url error:nil])
|
|
|
|
return std::nullopt;
|
|
|
|
return std::string([new_url fileSystemRepresentation]);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CocoaTools::LaunchApplication(std::string_view file)
|
|
|
|
{
|
|
|
|
NSURL* url = [NSURL fileURLWithPath:[[NSString alloc] initWithBytes:file.data() length:file.size() encoding:NSUTF8StringEncoding]];
|
|
|
|
if (@available(macOS 10.15, *))
|
|
|
|
{
|
|
|
|
// replacement api is async which isn't great for us
|
|
|
|
std::mutex done;
|
|
|
|
bool output;
|
|
|
|
done.lock();
|
|
|
|
NSWorkspaceOpenConfiguration* config = [NSWorkspaceOpenConfiguration new];
|
|
|
|
[config setCreatesNewApplicationInstance:YES];
|
|
|
|
[[NSWorkspace sharedWorkspace] openApplicationAtURL:url configuration:config completionHandler:[&](NSRunningApplication*_Nullable app, NSError*_Nullable error) {
|
|
|
|
output = app != nullptr;
|
|
|
|
done.unlock();
|
|
|
|
}];
|
|
|
|
done.lock();
|
|
|
|
done.unlock();
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return [[NSWorkspace sharedWorkspace] launchApplicationAtURL:url options:NSWorkspaceLaunchNewInstance configuration:@{} error:nil];
|
|
|
|
}
|
|
|
|
}
|