mirror of
https://github.com/wwiinnddyy/LanMountainDesktop.git
synced 2026-06-20 23:54:26 +08:00
* ava12升级 * Enable centralized package versioning Add <Project> and <PropertyGroup> with <ManagePackageVersionsCentrally>true</ManagePackageVersionsCentrally> to Directory.Packages.props to enable centralized package version management across the repository. This allows package versions to be controlled from this single file instead of individual project files. * Migrate codebase to Avalonia 12 APIs Apply Avalonia 12 migration changes: replace SystemDecorations with WindowDecorations and remove ExtendClientAreaChromeHints/ExtendClientAreaTitleBarHeightHint usages; update BindingPlugins removal logic (no-op); switch clipboard usage to ClipboardExtensions.SetTextAsync; update Bitmap.CopyPixels calls to the new signature. Replace TextBox.Watermark with PlaceholderText, convert NumberBox styles to FANumberBox and adjust templates, change Checked/Unchecked handlers to IsCheckedChanged, and adapt FluentIcons usages (SymbolIconSource -> FASymbol/FAFont/FluentIcon equivalents). Fix MainWindow partial classes to inherit Window and correct missing variables/fields/usings. Add migration docs/specs/tasks under .trae and include a small TestFluentIcons project for icon testing. * Migrate to Avalonia 12 and Plugin SDK v5 Upgrade project to the Avalonia 12 baseline and Plugin SDK v5: centralize Avalonia packages, remove legacy WebView.Avalonia usage (use NativeWebView/WebView2 EnvironmentRequested), and update Fluent/Material icon/package usages. Bump multiple package/project versions to 5.0.0 and Avalonia 12.0.1, update plugin template and README/docs to SDK v5, and add PLUGIN_SDK_V5_MIGRATION.md. Also fix runtime/behavior bugs: make DataLocationResolver use a fixed bootstrap launcher data path and avoid recursive ResolveDataRoot; add legacy-state handling and extraction in OobeStateService; and update component settings tests to reflect migrated storage (DB/backup) and reset cache for test reloads. Various csproj, tests, and docs updated to reflect the migration and ensure build/test compatibility. * Update icon glyphs and symbol mappings Replace and refine icon sources across settings pages and controls: many FAFontIconSource glyphs were updated to specific Seagull Fluent Icons codepoints, some FASymbolIconSource usages were replaced with FAFontIconSource, and a number of symbol-to-Symbol enum mappings were adjusted (e.g. "Bell" -> AlertOn, "Shield" -> ShieldLock). Also clarified a comment in SettingsWindow and fixed a trailing newline in StudySettingsPage. Changes standardize icon visuals and bridge FluentIcons glyphs into FluentAvalonia icon sources. * fix.修复合并产生的问题。
247 lines
8.8 KiB
C#
247 lines
8.8 KiB
C#
using System.Text.Json;
|
|
using LanMountainDesktop.Launcher.Models;
|
|
|
|
namespace LanMountainDesktop.Launcher.Services;
|
|
|
|
internal sealed class OobeStateService
|
|
{
|
|
private const int CurrentSchemaVersion = 1;
|
|
|
|
private readonly string _stateDirectory;
|
|
private readonly string _statePath;
|
|
private readonly string _legacyStatePath;
|
|
private readonly string _legacyMarkerPath;
|
|
private readonly LauncherExecutionSnapshot _executionSnapshot;
|
|
|
|
public OobeStateService(
|
|
string appRoot,
|
|
string? stateRootOverride = null,
|
|
LauncherExecutionSnapshot? executionSnapshot = null)
|
|
{
|
|
_ = Path.GetFullPath(appRoot);
|
|
_executionSnapshot = executionSnapshot ?? LauncherExecutionContext.Capture();
|
|
|
|
var stateRoot = string.IsNullOrWhiteSpace(stateRootOverride)
|
|
? ResolveStateRoot(appRoot)
|
|
: Path.GetFullPath(stateRootOverride);
|
|
_stateDirectory = Path.Combine(stateRoot, "Launcher", "state");
|
|
_statePath = Path.Combine(_stateDirectory, "oobe-state.json");
|
|
|
|
var legacyRoot = string.IsNullOrWhiteSpace(stateRootOverride)
|
|
? Path.GetFullPath(appRoot)
|
|
: Path.GetFullPath(stateRootOverride);
|
|
var legacyStateDirectory = Path.Combine(legacyRoot, ".launcher", "state");
|
|
_legacyStatePath = Path.Combine(legacyStateDirectory, "oobe-state.json");
|
|
_legacyMarkerPath = Path.Combine(legacyStateDirectory, "first_run_completed");
|
|
}
|
|
|
|
public OobeLaunchDecision Evaluate(CommandContext context)
|
|
{
|
|
var decision = EvaluateCore(context);
|
|
Logger.Info(
|
|
$"OOBE decision evaluated. LaunchSource='{decision.LaunchSource}'; Status='{decision.Status}'; " +
|
|
$"ShouldShow={decision.ShouldShowOobe}; IsElevated={decision.IsElevated}; " +
|
|
$"StatePath='{decision.StatePath}'; SuppressionReason='{decision.SuppressionReason}'; " +
|
|
$"ResultCode='{decision.ResultCode}'; UserSid='{decision.UserSid ?? string.Empty}'.");
|
|
return decision;
|
|
}
|
|
|
|
public OobeCompletionResult MarkCompleted(CommandContext context)
|
|
{
|
|
try
|
|
{
|
|
Directory.CreateDirectory(_stateDirectory);
|
|
var payload = new OobeStateFile
|
|
{
|
|
SchemaVersion = CurrentSchemaVersion,
|
|
CompletedAtUtc = DateTimeOffset.UtcNow.ToString("O"),
|
|
UserName = _executionSnapshot.UserName,
|
|
UserSid = _executionSnapshot.UserSid,
|
|
LaunchSource = context.LaunchSource
|
|
};
|
|
|
|
var tempPath = Path.Combine(_stateDirectory, $"oobe-state.{Guid.NewGuid():N}.tmp");
|
|
var json = JsonSerializer.Serialize(payload, AppJsonContext.Default.OobeStateFile);
|
|
File.WriteAllText(tempPath, json);
|
|
File.Move(tempPath, _statePath, overwrite: true);
|
|
TryDeleteLegacyMarker();
|
|
|
|
Logger.Info(
|
|
$"OOBE completion persisted. LaunchSource='{context.LaunchSource}'; StatePath='{_statePath}'; " +
|
|
$"UserSid='{_executionSnapshot.UserSid ?? string.Empty}'.");
|
|
|
|
return new OobeCompletionResult
|
|
{
|
|
Success = true,
|
|
ResultCode = "ok"
|
|
};
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Logger.Warn(
|
|
$"Failed to persist OOBE state. LaunchSource='{context.LaunchSource}'; StatePath='{_statePath}'; " +
|
|
$"Error='{ex.Message}'.");
|
|
return new OobeCompletionResult
|
|
{
|
|
Success = false,
|
|
ResultCode = "oobe_state_unavailable",
|
|
ErrorMessage = ex.Message
|
|
};
|
|
}
|
|
}
|
|
|
|
private OobeLaunchDecision EvaluateCore(CommandContext context)
|
|
{
|
|
if (string.Equals(context.LaunchSource, "debug-preview", StringComparison.OrdinalIgnoreCase))
|
|
{
|
|
return BuildSuppressedDecision(context, "debug_preview", "oobe_suppressed_debug_preview");
|
|
}
|
|
|
|
if (context.IsMaintenanceCommand)
|
|
{
|
|
return BuildSuppressedDecision(context, "maintenance", "oobe_suppressed_maintenance");
|
|
}
|
|
|
|
try
|
|
{
|
|
var migratedLegacyMarker = false;
|
|
if (File.Exists(_statePath))
|
|
{
|
|
return EvaluateStateFile(context, _statePath, migratedLegacyState: false);
|
|
}
|
|
|
|
if (File.Exists(_legacyStatePath))
|
|
{
|
|
return EvaluateStateFile(context, _legacyStatePath, migratedLegacyState: false);
|
|
}
|
|
|
|
if (File.Exists(_legacyMarkerPath))
|
|
{
|
|
migratedLegacyMarker = TryMigrateLegacyMarker(context);
|
|
return BuildDecision(context, OobeStateStatus.Completed, shouldShowOobe: false, usedLegacyMarker: true, migratedLegacyMarker: migratedLegacyMarker);
|
|
}
|
|
|
|
if (_executionSnapshot.IsElevated)
|
|
{
|
|
return BuildSuppressedDecision(context, "elevated", "oobe_suppressed_elevated");
|
|
}
|
|
|
|
if (string.Equals(context.LaunchSource, "postinstall", StringComparison.OrdinalIgnoreCase))
|
|
{
|
|
return BuildDecision(context, OobeStateStatus.FirstRun, shouldShowOobe: true);
|
|
}
|
|
|
|
return BuildDecision(context, OobeStateStatus.FirstRun, shouldShowOobe: true);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
return BuildUnavailableDecision(context, ex.Message);
|
|
}
|
|
}
|
|
|
|
private bool TryMigrateLegacyMarker(CommandContext context)
|
|
{
|
|
var result = MarkCompleted(context);
|
|
return result.Success;
|
|
}
|
|
|
|
private OobeLaunchDecision EvaluateStateFile(CommandContext context, string statePath, bool migratedLegacyState)
|
|
{
|
|
using var stream = File.OpenRead(statePath);
|
|
var state = JsonSerializer.Deserialize(stream, AppJsonContext.Default.OobeStateFile);
|
|
if (state is null || state.SchemaVersion <= 0 || string.IsNullOrWhiteSpace(state.CompletedAtUtc))
|
|
{
|
|
return BuildUnavailableDecision(context, "OOBE state file is invalid.");
|
|
}
|
|
|
|
return BuildDecision(context, OobeStateStatus.Completed, shouldShowOobe: false, migratedLegacyMarker: migratedLegacyState);
|
|
}
|
|
|
|
private void TryDeleteLegacyMarker()
|
|
{
|
|
try
|
|
{
|
|
if (File.Exists(_legacyMarkerPath))
|
|
{
|
|
File.Delete(_legacyMarkerPath);
|
|
}
|
|
}
|
|
catch
|
|
{
|
|
}
|
|
}
|
|
|
|
private OobeLaunchDecision BuildDecision(
|
|
CommandContext context,
|
|
OobeStateStatus status,
|
|
bool shouldShowOobe,
|
|
bool usedLegacyMarker = false,
|
|
bool migratedLegacyMarker = false)
|
|
{
|
|
return new OobeLaunchDecision
|
|
{
|
|
Status = status,
|
|
ShouldShowOobe = shouldShowOobe,
|
|
StatePath = _statePath,
|
|
LaunchSource = context.LaunchSource,
|
|
IsElevated = _executionSnapshot.IsElevated,
|
|
UserName = _executionSnapshot.UserName,
|
|
UserSid = _executionSnapshot.UserSid,
|
|
UsedLegacyMarker = usedLegacyMarker,
|
|
MigratedLegacyMarker = migratedLegacyMarker,
|
|
ResultCode = "ok"
|
|
};
|
|
}
|
|
|
|
private OobeLaunchDecision BuildSuppressedDecision(CommandContext context, string reason, string resultCode)
|
|
{
|
|
return new OobeLaunchDecision
|
|
{
|
|
Status = OobeStateStatus.Suppressed,
|
|
ShouldShowOobe = false,
|
|
StatePath = _statePath,
|
|
LaunchSource = context.LaunchSource,
|
|
IsElevated = _executionSnapshot.IsElevated,
|
|
UserName = _executionSnapshot.UserName,
|
|
UserSid = _executionSnapshot.UserSid,
|
|
SuppressionReason = reason,
|
|
ResultCode = resultCode
|
|
};
|
|
}
|
|
|
|
private OobeLaunchDecision BuildUnavailableDecision(CommandContext context, string errorMessage)
|
|
{
|
|
return new OobeLaunchDecision
|
|
{
|
|
Status = OobeStateStatus.Unavailable,
|
|
ShouldShowOobe = false,
|
|
StatePath = _statePath,
|
|
LaunchSource = context.LaunchSource,
|
|
IsElevated = _executionSnapshot.IsElevated,
|
|
UserName = _executionSnapshot.UserName,
|
|
UserSid = _executionSnapshot.UserSid,
|
|
ResultCode = "oobe_state_unavailable",
|
|
ErrorMessage = errorMessage
|
|
};
|
|
}
|
|
|
|
private static string ResolveStateRoot(string appRoot)
|
|
{
|
|
try
|
|
{
|
|
var resolver = new DataLocationResolver(appRoot);
|
|
return resolver.ResolveDataRoot();
|
|
}
|
|
catch
|
|
{
|
|
var appData = Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData);
|
|
if (string.IsNullOrWhiteSpace(appData))
|
|
{
|
|
throw new InvalidOperationException("LocalApplicationData is unavailable.");
|
|
}
|
|
|
|
return Path.Combine(appData, "LanMountainDesktop");
|
|
}
|
|
}
|
|
}
|