mirror of
https://github.com/noctalia-dev/noctalia-shell.git
synced 2026-05-11 17:08:27 +08:00
436 lines
13 KiB
QML
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.6.7"
|
|
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();
|
|
}
|
|
}
|