Files
noctalia-shell/Services/Noctalia/UpdateService.qml
T
2026-03-31 16:24:11 -04:00

436 lines
13 KiB
QML

pragma Singleton
import QtQuick
import Quickshell
import Quickshell.Io
import qs.Commons
import qs.Services.Noctalia
import qs.Services.UI
Singleton {
id: root
// Version properties
readonly property string baseVersion: "4.7.2"
readonly property bool isDevelopment: false
readonly property string developmentSuffix: "-git"
readonly property string currentVersion: `v${!isDevelopment ? baseVersion : baseVersion + developmentSuffix}`
// Telemetry was introduced in this version - users upgrading from earlier need to see the wizard
readonly property string telemetryIntroVersion: "4.0.2"
// URLs
readonly property string discordUrl: "https://discord.noctalia.dev"
readonly property string feedbackUrl: Quickshell.env("NOCTALIA_CHANGELOG_FEEDBACK_URL") || ""
readonly property string upgradeLogBaseUrl: Quickshell.env("NOCTALIA_UPGRADELOG_URL") || "https://api.noctalia.dev/upgradelog"
// Changelog properties
property bool initialized: false
property bool changelogPending: false
property string changelogFromVersion: ""
property string changelogToVersion: ""
property string previousVersion: ""
property string changelogCurrentVersion: ""
property string releaseContent: ""
property string lastShownVersion: ""
property bool popupScheduled: false
property string fetchError: ""
property string changelogLastSeenVersion: ""
property bool changelogStateLoaded: false
property bool pendingShowRequest: false
property bool pendingTelemetryWizardCheck: false
// Fix for FileView race condition
property bool saveInProgress: false
property bool pendingSave: false
property int saveDebounceTimer: 0
Connections {
target: PanelService
function onPopupMenuWindowRegistered(screen) {
if (popupScheduled) {
if (!viewChangelogTargetScreen || viewChangelogTargetScreen.name === screen.name) {
openWhenReady();
}
}
}
}
signal popupQueued(string fromVersion, string toVersion)
signal telemetryWizardNeeded
function init() {
if (initialized)
return;
initialized = true;
Logger.i("UpdateService", "Version:", root.currentVersion);
// Load changelog state from ShellState
Qt.callLater(() => {
if (typeof ShellState !== 'undefined' && ShellState.isLoaded) {
loadChangelogState();
}
});
}
Connections {
target: typeof ShellState !== 'undefined' ? ShellState : null
function onIsLoadedChanged() {
if (ShellState.isLoaded) {
loadChangelogState();
}
}
}
// Debounce timer to prevent rapid successive saves
Timer {
id: saveDebouncer
interval: 300
repeat: false
onTriggered: executeSave()
}
function handleChangelogRequest() {
const fromVersion = changelogFromVersion || "";
const toVersion = changelogToVersion || "";
if (Settings.shouldOpenSetupWizard) {
// If you'll see the setup wizard then you don't need to see the changelog
markChangelogSeen(toVersion);
return;
}
if (!toVersion)
return;
if (popupScheduled && changelogCurrentVersion === toVersion)
return;
if (!popupScheduled && lastShownVersion === toVersion)
return;
previousVersion = fromVersion;
changelogCurrentVersion = toVersion;
// Fetch the upgrade log from the server
fetchUpgradeLog(fromVersion, toVersion);
popupScheduled = true;
root.popupQueued(previousVersion, changelogCurrentVersion);
clearChangelogRequest();
}
function fetchUpgradeLog(fromVersion, toVersion) {
// Normalize and ensure "v" prefix for consistent URL format
let from = ensureVersionPrefix(fromVersion || changelogLastSeenVersion || "3.0.0");
let to = ensureVersionPrefix(toVersion);
// Strip -git suffix
from = from.replace(root.developmentSuffix, "");
to = to.replace(root.developmentSuffix, "");
// 'from' always needs to be before 'to' (use semantic comparison)
if (compareVersions(from, to) >= 0) {
from = "v3.0.0";
}
const url = `${upgradeLogBaseUrl}/${from}/${to}`;
Logger.i("UpdateService", "Fetching upgrade log:", url);
const request = new XMLHttpRequest();
request.onreadystatechange = function () {
if (request.readyState === XMLHttpRequest.DONE) {
Logger.d("UpdateService", "Request completed with status:", request.status);
Logger.d("UpdateService", "Response text length:", request.responseText ? request.responseText.length : 0);
if (request.status >= 200 && request.status < 300) {
releaseContent = request.responseText || "";
Logger.d("UpdateService", "Successfully fetched upgrade log");
fetchError = "";
openWhenReady();
} else {
Logger.w("UpdateService", "Failed to fetch upgrade log, status:", request.status);
releaseContent = "";
if (request.status === 404) {
// Changelog not available for this version range - skip silently
Logger.w("UpdateService", "Changelog not found, skipping display");
fetchError = "";
popupScheduled = false;
markChangelogSeen(toVersion);
} else {
// Network error or server issue - show error to user
fetchError = I18n.tr("changelog.error.fetch-failed");
openWhenReady();
}
}
}
};
request.open("GET", url);
request.send();
}
function normalizeVersion(version) {
if (!version)
return "";
return version.startsWith("v") ? version.substring(1) : version;
}
function ensureVersionPrefix(version) {
if (!version)
return "";
return version.startsWith("v") ? version : "v" + version;
}
function parseVersionParts(version) {
const clean = normalizeVersion(version);
if (!clean)
return [];
return clean.split(/[^0-9]+/).filter(part => part.length > 0).map(part => parseInt(part));
}
function compareVersions(a, b) {
if (a === b)
return 0;
const partsA = parseVersionParts(a);
const partsB = parseVersionParts(b);
const length = Math.max(partsA.length, partsB.length);
for (var i = 0; i < length; i++) {
const valA = partsA[i] || 0;
const valB = partsB[i] || 0;
if (valA > valB)
return 1;
if (valA < valB)
return -1;
}
return 0;
}
// Check if user is upgrading from a version before telemetry was introduced
function shouldShowTelemetryWizard() {
if (!changelogStateLoaded)
return false;
if (Settings.isFreshInstall)
return false;
if (Settings.shouldOpenSetupWizard)
return false;
// No previous version recorded but settings exist - assume upgrading from old version
// (e.g., user deleted shell-state.json but has existing settings)
if (!changelogLastSeenVersion || changelogLastSeenVersion === "")
return true;
// Check if last seen version is before telemetry introduction
return compareVersions(changelogLastSeenVersion, telemetryIntroVersion) < 0;
}
// Called by shell.qml to check for telemetry wizard after init
// If state isn't loaded yet, sets a pending flag and emits telemetryWizardNeeded later
function checkTelemetryWizardOrChangelog() {
Logger.d("UpdateService", "checkTelemetryWizardOrChangelog called, stateLoaded:", changelogStateLoaded);
if (!changelogStateLoaded) {
// State not loaded yet, set pending flags
Logger.d("UpdateService", "State not loaded yet, setting pending flags");
pendingTelemetryWizardCheck = true;
pendingShowRequest = true;
return;
}
// State is already loaded, check immediately
const needsTelemetryWizard = shouldShowTelemetryWizard();
Logger.d("UpdateService", "shouldShowTelemetryWizard:", needsTelemetryWizard, "lastSeenVersion:", changelogLastSeenVersion);
if (needsTelemetryWizard) {
Logger.i("UpdateService", "Emitting telemetryWizardNeeded signal");
root.telemetryWizardNeeded();
} else {
showLatestChangelog();
}
}
function openWhenReady() {
if (!popupScheduled)
return;
if (!Quickshell.screens || Quickshell.screens.length === 0) {
return;
}
let targetScreen = viewChangelogTargetScreen;
if (targetScreen) {
// Explicit screen requested - validate it
if (!PanelService.canShowPanelsOnScreen(targetScreen)) {
Logger.w("UpdateService", "Changelog cannot be shown on screen without bar:", targetScreen.name);
popupScheduled = false;
viewChangelogTargetScreen = null;
return;
}
} else {
// No explicit screen - find one that can show panels
targetScreen = PanelService.findScreenForPanels();
if (!targetScreen) {
Logger.w("UpdateService", "No screen available to show changelog");
popupScheduled = false;
return;
}
}
const panel = PanelService.getPanel("changelogPanel", targetScreen);
if (!panel) {
// Panel not found yet. Wait for popupMenuWindowRegistered signal.
// This avoids the memory leak (#1306).
Logger.d("UpdateService", "Waiting for changelogPanel on screen:", targetScreen.name);
return;
}
panel.open();
popupScheduled = false;
lastShownVersion = changelogCurrentVersion;
viewChangelogTargetScreen = null;
}
function openDiscord() {
if (!discordUrl)
return;
Quickshell.execDetached(["xdg-open", discordUrl]);
}
function openFeedbackForm() {
if (!feedbackUrl)
return;
Quickshell.execDetached(["xdg-open", feedbackUrl]);
}
function showLatestChangelog() {
if (!currentVersion)
return;
if (!changelogStateLoaded) {
pendingShowRequest = true;
return;
}
// Normalize versions for comparison (strip -git, ensure v prefix)
const lastSeen = ensureVersionPrefix(changelogLastSeenVersion.replace(developmentSuffix, ""));
const target = ensureVersionPrefix(currentVersion.replace(developmentSuffix, ""));
if (lastSeen === target)
return;
if (!Settings.data.general.showChangelogOnStartup) {
// user has opted out of seeing changelogs, mark as seen
markChangelogSeen(target);
return;
}
changelogFromVersion = lastSeen;
changelogToVersion = target;
changelogPending = true;
handleChangelogRequest();
}
// Manual changelog viewing (e.g., from Settings > About > View Changelog)
// Shows all changes since v3.0.0, unlike showLatestChangelog() which uses lastSeenVersion
property var viewChangelogTargetScreen: null
function viewChangelog(screen) {
if (!currentVersion)
return;
const target = ensureVersionPrefix(currentVersion.replace(developmentSuffix, ""));
const fromVersion = "v3.8.2";
previousVersion = fromVersion;
changelogCurrentVersion = target;
viewChangelogTargetScreen = screen || null;
popupScheduled = true;
fetchUpgradeLog(fromVersion, target);
}
function clearChangelogRequest() {
changelogPending = false;
changelogFromVersion = "";
changelogToVersion = "";
}
function markChangelogSeen(version) {
if (!version)
return;
changelogLastSeenVersion = version;
debouncedSaveChangelogState();
}
function loadChangelogState() {
try {
const changelog = ShellState.getChangelogState();
changelogLastSeenVersion = changelog.lastSeenVersion || "";
// Migration is now handled in Settings.qml
Logger.d("UpdateService", "Loaded changelog state from ShellState");
} catch (error) {
Logger.e("UpdateService", "Failed to load changelog state:", error);
}
changelogStateLoaded = true;
// Handle pending telemetry wizard check first
if (pendingTelemetryWizardCheck) {
pendingTelemetryWizardCheck = false;
if (shouldShowTelemetryWizard()) {
root.telemetryWizardNeeded();
} else if (pendingShowRequest) {
pendingShowRequest = false;
Qt.callLater(root.showLatestChangelog);
}
return;
}
if (pendingShowRequest) {
pendingShowRequest = false;
Qt.callLater(root.showLatestChangelog);
}
}
function debouncedSaveChangelogState() {
// Queue a save and restart the debounce timer
pendingSave = true;
saveDebouncer.restart();
}
function executeSave() {
if (!pendingSave)
return;
// Prevent concurrent saves
if (saveInProgress) {
// Retry after a short delay
saveDebouncer.start();
return;
}
pendingSave = false;
saveInProgress = true;
try {
ShellState.setChangelogState({
lastSeenVersion: changelogLastSeenVersion || ""
});
Logger.d("UpdateService", "Saved changelog state to ShellState");
saveInProgress = false;
// Check if another save was queued while we were saving
if (pendingSave) {
Qt.callLater(executeSave);
}
} catch (error) {
Logger.e("UpdateService", "Failed to save changelog state:", error);
saveInProgress = false;
}
}
function saveChangelogState() {
// Immediate save (backward compatibility)
debouncedSaveChangelogState();
}
}