2022-04-20 00:34:17 -04:00
|
|
|
// SPDX-License-Identifier: GPL-3.0-only
|
|
|
|
/*
|
2023-08-04 19:41:47 +02:00
|
|
|
* Prism Launcher - Minecraft Launcher
|
2022-04-20 00:34:17 -04:00
|
|
|
* Copyright (C) 2022 Kenneth Chew <kenneth.c0@protonmail.com>
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, version 3.
|
|
|
|
*
|
|
|
|
* This program 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 this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2022-04-25 19:33:17 -04:00
|
|
|
#include "MacSparkleUpdater.h"
|
2022-04-20 00:34:17 -04:00
|
|
|
|
|
|
|
#include "Application.h"
|
|
|
|
|
|
|
|
#include <Cocoa/Cocoa.h>
|
|
|
|
#include <Sparkle/Sparkle.h>
|
|
|
|
|
|
|
|
@interface UpdaterObserver : NSObject
|
|
|
|
|
|
|
|
@property(nonatomic, readonly) SPUUpdater* updater;
|
|
|
|
|
|
|
|
/// A callback to run when the state of `canCheckForUpdates` for the `updater` changes.
|
2023-08-02 18:35:35 +02:00
|
|
|
@property(nonatomic, copy) void (^callback)(bool);
|
2022-04-20 00:34:17 -04:00
|
|
|
|
|
|
|
- (id)initWithUpdater:(SPUUpdater*)updater;
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation UpdaterObserver
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
- (id)initWithUpdater:(SPUUpdater*)updater {
|
2022-04-20 00:34:17 -04:00
|
|
|
self = [super init];
|
|
|
|
_updater = updater;
|
|
|
|
[self addObserver:self forKeyPath:@"updater.canCheckForUpdates" options:NSKeyValueObservingOptionNew context:nil];
|
|
|
|
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
- (void)observeValueForKeyPath:(NSString*)keyPath
|
2022-04-20 00:34:17 -04:00
|
|
|
ofObject:(id)object
|
2023-08-02 18:35:35 +02:00
|
|
|
change:(NSDictionary<NSKeyValueChangeKey, id>*)change
|
|
|
|
context:(void*)context {
|
|
|
|
if ([keyPath isEqualToString:@"updater.canCheckForUpdates"]) {
|
2022-04-20 00:34:17 -04:00
|
|
|
bool canCheck = [change[NSKeyValueChangeNewKey] boolValue];
|
|
|
|
self.callback(canCheck);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
@interface UpdaterDelegate : NSObject <SPUUpdaterDelegate>
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
@property(nonatomic, copy) NSSet<NSString*>* allowedChannels;
|
2022-04-20 00:34:17 -04:00
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation UpdaterDelegate
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
- (NSSet<NSString*>*)allowedChannelsForUpdater:(SPUUpdater*)updater {
|
2022-04-20 00:34:17 -04:00
|
|
|
return _allowedChannels;
|
|
|
|
}
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
class MacSparkleUpdater::Private {
|
|
|
|
public:
|
|
|
|
SPUStandardUpdaterController* updaterController;
|
|
|
|
UpdaterObserver* updaterObserver;
|
|
|
|
UpdaterDelegate* updaterDelegate;
|
|
|
|
NSAutoreleasePool* autoReleasePool;
|
2022-04-20 00:34:17 -04:00
|
|
|
};
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
MacSparkleUpdater::MacSparkleUpdater() {
|
2022-04-25 19:33:17 -04:00
|
|
|
priv = new MacSparkleUpdater::Private();
|
2022-04-20 00:34:17 -04:00
|
|
|
|
|
|
|
// Enable Cocoa's memory management.
|
|
|
|
NSApplicationLoad();
|
|
|
|
priv->autoReleasePool = [[NSAutoreleasePool alloc] init];
|
|
|
|
|
|
|
|
// Delegate is used for setting/getting allowed update channels.
|
|
|
|
priv->updaterDelegate = [[UpdaterDelegate alloc] init];
|
|
|
|
|
|
|
|
// Controller is the interface for actually doing the updates.
|
|
|
|
priv->updaterController = [[SPUStandardUpdaterController alloc] initWithStartingUpdater:true
|
2023-08-02 18:35:35 +02:00
|
|
|
updaterDelegate:priv->updaterDelegate
|
|
|
|
userDriverDelegate:nil];
|
2022-04-20 00:34:17 -04:00
|
|
|
|
|
|
|
priv->updaterObserver = [[UpdaterObserver alloc] initWithUpdater:priv->updaterController.updater];
|
|
|
|
// Use KVO to run a callback that emits a Qt signal when `canCheckForUpdates` changes, so the UI can respond accordingly.
|
|
|
|
priv->updaterObserver.callback = ^(bool canCheck) {
|
2023-08-02 18:35:35 +02:00
|
|
|
emit canCheckForUpdatesChanged(canCheck);
|
2022-04-20 00:34:17 -04:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
MacSparkleUpdater::~MacSparkleUpdater() {
|
2022-04-20 00:34:17 -04:00
|
|
|
[priv->updaterObserver removeObserver:priv->updaterObserver forKeyPath:@"updater.canCheckForUpdates"];
|
|
|
|
|
|
|
|
[priv->updaterController release];
|
|
|
|
[priv->updaterObserver release];
|
|
|
|
[priv->updaterDelegate release];
|
|
|
|
[priv->autoReleasePool release];
|
|
|
|
delete priv;
|
|
|
|
}
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
void MacSparkleUpdater::checkForUpdates() {
|
2022-04-20 00:34:17 -04:00
|
|
|
[priv->updaterController checkForUpdates:nil];
|
|
|
|
}
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
bool MacSparkleUpdater::getAutomaticallyChecksForUpdates() {
|
2022-04-20 00:34:17 -04:00
|
|
|
return priv->updaterController.updater.automaticallyChecksForUpdates;
|
|
|
|
}
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
double MacSparkleUpdater::getUpdateCheckInterval() {
|
2022-04-20 00:34:17 -04:00
|
|
|
return priv->updaterController.updater.updateCheckInterval;
|
|
|
|
}
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
QSet<QString> MacSparkleUpdater::getAllowedChannels() {
|
2022-04-20 00:34:17 -04:00
|
|
|
// Convert NSSet<NSString> -> QSet<QString>
|
|
|
|
__block QSet<QString> channels;
|
2023-08-02 18:35:35 +02:00
|
|
|
[priv->updaterDelegate.allowedChannels enumerateObjectsUsingBlock:^(NSString* channel, BOOL* stop) {
|
|
|
|
channels.insert(QString::fromNSString(channel));
|
2022-04-20 00:34:17 -04:00
|
|
|
}];
|
|
|
|
return channels;
|
|
|
|
}
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
bool MacSparkleUpdater::getBetaAllowed() {
|
2022-04-25 19:33:17 -04:00
|
|
|
return getAllowedChannels().contains("beta");
|
|
|
|
}
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
void MacSparkleUpdater::setAutomaticallyChecksForUpdates(bool check) {
|
|
|
|
priv->updaterController.updater.automaticallyChecksForUpdates = check ? YES : NO; // make clang-tidy happy
|
2022-04-20 00:34:17 -04:00
|
|
|
}
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
void MacSparkleUpdater::setUpdateCheckInterval(double seconds) {
|
2022-04-20 00:34:17 -04:00
|
|
|
priv->updaterController.updater.updateCheckInterval = seconds;
|
|
|
|
}
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
void MacSparkleUpdater::clearAllowedChannels() {
|
2022-04-20 00:34:17 -04:00
|
|
|
priv->updaterDelegate.allowedChannels = [NSSet set];
|
|
|
|
}
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
void MacSparkleUpdater::setAllowedChannel(const QString& channel) {
|
|
|
|
if (channel.isEmpty()) {
|
2022-04-20 00:34:17 -04:00
|
|
|
clearAllowedChannels();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
NSSet<NSString*>* nsChannels = [NSSet setWithObject:channel.toNSString()];
|
2022-04-20 00:34:17 -04:00
|
|
|
priv->updaterDelegate.allowedChannels = nsChannels;
|
|
|
|
}
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
void MacSparkleUpdater::setAllowedChannels(const QSet<QString>& channels) {
|
|
|
|
if (channels.isEmpty()) {
|
2022-04-20 00:34:17 -04:00
|
|
|
clearAllowedChannels();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString channelsConfig = "";
|
|
|
|
// Convert QSet<QString> -> NSSet<NSString>
|
2023-08-02 18:35:35 +02:00
|
|
|
NSMutableSet<NSString*>* nsChannels = [NSMutableSet setWithCapacity:channels.count()];
|
|
|
|
foreach (const QString channel, channels) {
|
2022-04-20 00:34:17 -04:00
|
|
|
[nsChannels addObject:channel.toNSString()];
|
|
|
|
channelsConfig += channel + " ";
|
|
|
|
}
|
|
|
|
|
|
|
|
priv->updaterDelegate.allowedChannels = nsChannels;
|
|
|
|
}
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
void MacSparkleUpdater::setBetaAllowed(bool allowed) {
|
|
|
|
if (allowed) {
|
2022-04-25 19:33:17 -04:00
|
|
|
setAllowedChannel("beta");
|
2023-08-02 18:35:35 +02:00
|
|
|
} else {
|
2022-04-25 19:33:17 -04:00
|
|
|
clearAllowedChannels();
|
|
|
|
}
|
|
|
|
}
|