Files
community/Ink Canvas/MainWindow_cs/MW_SettingsToLoad.cs
T

834 lines
34 KiB
C#
Raw Normal View History

2026-04-23 22:09:27 +08:00
using H.NotifyIcon;
2025-08-31 11:43:52 +08:00
using Ink_Canvas.Helpers;
using Newtonsoft.Json;
2025-12-20 17:52:56 +08:00
using Newtonsoft.Json.Linq;
2025-08-31 11:43:52 +08:00
using OSVersionExtension;
using System;
2025-12-20 17:52:56 +08:00
using System.Collections.Generic;
2025-05-25 09:29:48 +08:00
using System.Reflection;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Ink;
using System.Windows.Interop;
using System.Windows.Media;
using System.Windows.Media.Imaging;
2025-10-02 18:48:11 +08:00
using System.Windows.Threading;
2025-05-25 09:29:48 +08:00
using File = System.IO.File;
using OperatingSystem = OSVersionExtension.OperatingSystem;
2025-10-03 17:08:46 +08:00
using WinForms = System.Windows.Forms;
2025-05-25 09:29:48 +08:00
2025-08-03 16:46:33 +08:00
namespace Ink_Canvas
{
public partial class MainWindow : Window
{
/// <summary>
/// 从配置文件加载用户设置并将其应用到主窗口和相关控件的状态(包括启动、外观、画布、手势、PPT、自动化等各项配置)。
/// </summary>
/// <param name="isStartup">指示当前为应用启动阶段;为 true 时按启动流程应用启动相关设置(例如触发启动专用动作和启动时的行为)。</param>
2026-02-23 12:04:54 +08:00
/// <summary>
/// 从当前配置文件重新加载设置并应用到界面(热重载),不触发启动逻辑与自动更新检查。
/// 用于配置文件切换后立即生效。
/// </summary>
public void ReloadSettingsFromFile()
{
LoadSettings(false, skipAutoUpdateCheck: true);
}
/// <param name="skipAutoUpdateCheck">指示是否跳过自动更新检查;为 true 时不会在加载设置后执行自动更新检测。</param>
2026-02-07 11:54:40 +08:00
private void LoadSettings(bool isStartup = false, bool skipAutoUpdateCheck = false)
2025-08-03 16:46:33 +08:00
{
try
{
if (File.Exists(App.RootPath + settingsFileName))
{
try
{
2025-05-25 09:29:48 +08:00
string text = File.ReadAllText(App.RootPath + settingsFileName);
Settings = JsonConvert.DeserializeObject<Settings>(text);
2025-10-03 17:08:46 +08:00
2025-12-20 17:52:56 +08:00
if (Settings != null)
{
CleanupObsoleteSettings(text);
}
2025-10-01 00:01:35 +08:00
// 验证设置是否成功加载
if (Settings == null)
{
LogHelper.WriteLogToFile("配置文件解析失败,尝试从备份恢复", LogHelper.LogType.Warning);
if (AutoBackupManager.TryRestoreFromBackup())
{
// 重新尝试加载
text = File.ReadAllText(App.RootPath + settingsFileName);
Settings = JsonConvert.DeserializeObject<Settings>(text);
if (Settings != null)
{
2025-12-20 17:52:56 +08:00
// 清理过期配置项
CleanupObsoleteSettings(text);
2025-10-01 00:01:35 +08:00
}
}
2025-10-03 17:08:46 +08:00
2025-10-01 00:01:35 +08:00
// 如果仍然失败,使用默认设置
if (Settings == null)
{
LogHelper.WriteLogToFile("从备份恢复失败,使用默认设置", LogHelper.LogType.Warning);
BtnResetToSuggestion_Click(null, null);
}
}
}
catch (Exception ex)
{
LogHelper.WriteLogToFile($"配置文件加载失败: {ex.Message}", LogHelper.LogType.Error);
2025-10-03 17:08:46 +08:00
2025-10-01 00:01:35 +08:00
// 尝试从备份恢复
LogHelper.WriteLogToFile("尝试从备份恢复配置文件", LogHelper.LogType.Warning);
if (AutoBackupManager.TryRestoreFromBackup())
{
try
{
string text = File.ReadAllText(App.RootPath + settingsFileName);
Settings = JsonConvert.DeserializeObject<Settings>(text);
if (Settings != null)
{
2025-12-20 17:52:56 +08:00
// 清理过期配置项
CleanupObsoleteSettings(text);
2025-10-01 00:01:35 +08:00
}
}
catch (Exception restoreEx)
{
LogHelper.WriteLogToFile($"从备份恢复后重新加载失败: {restoreEx.Message}", LogHelper.LogType.Error);
2025-10-01 11:29:59 +08:00
BtnResetToSuggestion_Click(null, null);
2025-10-01 00:01:35 +08:00
}
}
2025-10-03 17:08:46 +08:00
2025-10-01 00:01:35 +08:00
// 如果仍然失败,使用默认设置
if (Settings == null)
{
LogHelper.WriteLogToFile("从备份恢复失败,使用默认设置", LogHelper.LogType.Warning);
BtnResetToSuggestion_Click(null, null);
}
2025-05-25 09:29:48 +08:00
}
2025-08-03 16:46:33 +08:00
}
else
{
2025-10-01 00:01:35 +08:00
LogHelper.WriteLogToFile("配置文件不存在,尝试从备份恢复", LogHelper.LogType.Warning);
if (AutoBackupManager.TryRestoreFromBackup())
{
try
{
string text = File.ReadAllText(App.RootPath + settingsFileName);
Settings = JsonConvert.DeserializeObject<Settings>(text);
if (Settings != null)
{
2025-12-20 17:52:56 +08:00
// 清理过期配置项
CleanupObsoleteSettings(text);
2025-10-01 00:01:35 +08:00
}
}
catch (Exception restoreEx)
{
LogHelper.WriteLogToFile($"从备份恢复后加载失败: {restoreEx.Message}", LogHelper.LogType.Error);
2025-10-01 11:29:59 +08:00
BtnResetToSuggestion_Click(null, null);
2025-10-01 00:01:35 +08:00
}
}
2025-10-01 11:34:37 +08:00
else
{
// 备份恢复失败(备份目录不存在等),使用默认设置
LogHelper.WriteLogToFile("备份恢复失败,使用默认设置", LogHelper.LogType.Warning);
BtnResetToSuggestion_Click(null, null);
}
2025-10-03 17:08:46 +08:00
2025-10-01 00:01:35 +08:00
// 如果仍然失败,使用默认设置
if (Settings == null)
{
LogHelper.WriteLogToFile("从备份恢复失败,使用默认设置", LogHelper.LogType.Warning);
BtnResetToSuggestion_Click(null, null);
}
2025-05-25 09:29:48 +08:00
}
}
2025-08-03 16:46:33 +08:00
catch (Exception ex)
{
2025-05-25 09:29:48 +08:00
LogHelper.WriteLogToFile(ex.ToString(), LogHelper.LogType.Error);
}
2026-03-03 15:58:26 +08:00
try
{
if (Settings?.Appearance != null)
{
var preferredLanguage = Settings.Appearance.Language ?? string.Empty;
if (!string.IsNullOrWhiteSpace(preferredLanguage))
{
LocalizationHelper.TrySetCulture(preferredLanguage);
}
}
}
catch (Exception ex)
{
LogHelper.WriteLogToFile($"从配置应用界面语言失败: {ex.Message}", LogHelper.LogType.Error);
}
2026-02-16 20:22:41 +08:00
try
{
ProcessProtectionManager.ApplyFromSettings();
}
catch
{
}
2025-05-25 09:29:48 +08:00
// Startup
2025-08-03 16:46:33 +08:00
if (isStartup)
{
2025-05-25 09:29:48 +08:00
CursorIcon_Click(null, null);
}
2026-02-06 23:00:57 +08:00
try
{
2026-02-06 23:54:33 +08:00
if (Settings?.Startup != null)
2026-02-06 23:00:57 +08:00
{
}
}
catch
{
}
2025-08-03 16:46:33 +08:00
if (Settings.Startup != null)
{
if (isStartup)
{
if (Settings.Automation.AutoDelSavedFiles)
{
2026-04-23 22:09:27 +08:00
DelAutoSavedFiles.DeleteFilesOlder(Settings.Automation.AutoSavedStrokesLocation,
Settings.Automation.AutoDelSavedFilesDaysThreshold);
2025-05-25 09:29:48 +08:00
}
}
2025-08-03 16:46:33 +08:00
if (Settings.Startup.IsEnableNibMode)
{
2026-04-07 00:33:19 +08:00
ToggleSwitchEnableNibMode.IsOn = true;
BoardToggleSwitchEnableNibMode.IsOn = true;
2025-05-25 09:29:48 +08:00
BoundsWidth = Settings.Advanced.NibModeBoundsWidth;
2025-08-03 16:46:33 +08:00
}
else
{
2026-04-07 00:33:19 +08:00
ToggleSwitchEnableNibMode.IsOn = false;
BoardToggleSwitchEnableNibMode.IsOn = false;
2025-05-25 09:29:48 +08:00
BoundsWidth = Settings.Advanced.FingerModeBoundsWidth;
}
2026-04-07 00:33:19 +08:00
// 设置自动更新相关选项
2026-02-07 11:54:40 +08:00
if (Settings.Startup.IsAutoUpdate && !skipAutoUpdateCheck)
2025-08-03 16:46:33 +08:00
{
if (isStartup)
{
LogHelper.WriteLogToFile("AutoUpdate | Running auto-update check at startup");
AutoUpdate();
}
else
{
LogHelper.WriteLogToFile("AutoUpdate | Running auto-update check after settings change");
AutoUpdate();
2025-06-29 11:56:38 +08:00
}
2025-05-25 09:29:48 +08:00
}
2025-08-03 16:46:33 +08:00
}
else
{
2025-05-25 09:29:48 +08:00
Settings.Startup = new Startup();
Settings.Startup.IsEnableNibMode = false;
ToggleSwitchEnableNibMode.IsOn = false;
BoardToggleSwitchEnableNibMode.IsOn = false;
BoundsWidth = Settings.Advanced.FingerModeBoundsWidth;
2025-05-25 09:29:48 +08:00
}
if (Settings.Startup != null)
{
if (Settings.Startup.CrashAction == 0)
{
App.CrashAction = App.CrashActionType.SilentRestart;
}
else
{
App.CrashAction = App.CrashActionType.NoAction;
}
}
// Appearance - UI initialization (settings loading moved to AppearancePage)
2025-08-03 16:46:33 +08:00
if (Settings.Appearance != null)
{
if (!Settings.Appearance.IsEnableDisPlayNibModeToggler)
{
2025-05-25 09:29:48 +08:00
NibModeSimpleStackPanel.Visibility = Visibility.Collapsed;
BoardNibModeSimpleStackPanel.Visibility = Visibility.Collapsed;
2025-08-03 16:46:33 +08:00
}
2025-05-25 09:29:48 +08:00
if (Settings.Appearance.ViewboxFloatingBarScaleTransformValue != 0)
2025-05-25 09:29:48 +08:00
{
double val = Settings.Appearance.ViewboxFloatingBarScaleTransformValue;
ViewboxFloatingBarScaleTransform.ScaleX =
(val > 0.5 && val < 1.25) ? val : val <= 0.5 ? 0.5 : val >= 1.25 ? 1.25 : 1;
ViewboxFloatingBarScaleTransform.ScaleY =
(val > 0.5 && val < 1.25) ? val : val <= 0.5 ? 0.5 : val >= 1.25 ? 1.25 : 1;
}
2025-08-03 16:46:33 +08:00
switch (Settings.Appearance.UnFoldButtonImageType)
{
2025-05-25 09:29:48 +08:00
case 0:
RightUnFoldBtnImgChevron.Source =
new BitmapImage(new Uri("pack://application:,,,/Resources/new-icons/unfold-chevron.png"));
RightUnFoldBtnImgChevron.Width = 14;
RightUnFoldBtnImgChevron.Height = 14;
RightUnFoldBtnImgChevron.RenderTransform = new RotateTransform(180);
LeftUnFoldBtnImgChevron.Source =
new BitmapImage(new Uri("pack://application:,,,/Resources/new-icons/unfold-chevron.png"));
LeftUnFoldBtnImgChevron.Width = 14;
LeftUnFoldBtnImgChevron.Height = 14;
LeftUnFoldBtnImgChevron.RenderTransform = null;
break;
case 1:
RightUnFoldBtnImgChevron.Source =
new BitmapImage(new Uri("pack://application:,,,/Resources/new-icons/pen-white.png"));
RightUnFoldBtnImgChevron.Width = 18;
RightUnFoldBtnImgChevron.Height = 18;
RightUnFoldBtnImgChevron.RenderTransform = null;
LeftUnFoldBtnImgChevron.Source =
new BitmapImage(new Uri("pack://application:,,,/Resources/new-icons/pen-white.png"));
LeftUnFoldBtnImgChevron.Width = 18;
LeftUnFoldBtnImgChevron.Height = 18;
LeftUnFoldBtnImgChevron.RenderTransform = null;
break;
}
ViewboxFloatingBar.Opacity = Settings.Appearance.ViewboxFloatingBarOpacityValue;
2025-09-07 13:30:46 +08:00
if (Settings.Appearance.EnableViewboxBlackBoardScaleTransform)
2025-05-25 09:29:48 +08:00
{
ViewboxBlackboardCenterSideScaleTransform.ScaleX = 0.8;
ViewboxBlackboardCenterSideScaleTransform.ScaleY = 0.8;
2025-08-03 16:46:33 +08:00
}
else
{
2025-05-25 09:29:48 +08:00
ViewboxBlackboardCenterSideScaleTransform.ScaleX = 1;
ViewboxBlackboardCenterSideScaleTransform.ScaleY = 1;
}
2025-08-03 16:46:33 +08:00
if (Settings.Appearance.IsTransparentButtonBackground)
{
2025-05-25 09:29:48 +08:00
BtnExit.Background = new SolidColorBrush(StringToColor("#7F909090"));
2025-08-03 16:46:33 +08:00
}
else
{
2025-05-25 09:29:48 +08:00
BtnExit.Background = BtnSwitchTheme.Content.ToString() == "深色"
? new SolidColorBrush(StringToColor("#FFCCCCCC"))
: new SolidColorBrush(StringToColor("#FF555555"));
2025-05-25 09:29:48 +08:00
}
if (Settings.Appearance.FloatingBarImg >= 12 + Settings.Appearance.CustomFloatingBarImgs.Count)
2025-07-15 20:30:10 +08:00
{
Settings.Appearance.FloatingBarImg = 0;
2025-05-25 09:29:48 +08:00
}
2025-08-03 16:46:33 +08:00
2025-07-15 20:30:10 +08:00
UpdateFloatingBarIcon();
2025-08-11 19:08:05 +08:00
UpdateFloatingBarButtonsVisibility();
2025-10-01 09:48:44 +08:00
UpdateFloatingBarIcons();
2025-08-11 19:08:05 +08:00
var _taskbar = Application.Current.Resources["TaskbarTrayIcon"];
if (_taskbar is FrameworkElement fe)
fe.Visibility = Settings.Appearance.EnableTrayIcon ? Visibility.Visible : Visibility.Collapsed;
2025-05-25 09:29:48 +08:00
SystemEvents_UserPreferenceChanged(null, null);
2025-08-03 16:46:33 +08:00
}
else
{
2025-05-25 09:29:48 +08:00
Settings.Appearance = new Appearance();
}
// PowerPointSettings
2025-08-03 16:46:33 +08:00
if (Settings.PowerPointSettings != null)
{
if (Settings.PowerPointSettings.PowerPointSupport)
{
2025-07-29 01:15:32 +08:00
// PPT监控将在Window_Loaded中启动
2025-08-03 16:46:33 +08:00
}
2025-12-20 22:56:13 +08:00
2025-05-25 09:29:48 +08:00
UpdatePPTBtnSlidersStatus();
UpdatePPTBtnPreview();
2025-08-03 16:46:33 +08:00
}
else
{
2025-05-25 09:29:48 +08:00
Settings.PowerPointSettings = new PowerPointSettings();
}
// Gesture
2025-08-03 16:46:33 +08:00
if (Settings.Gesture != null)
{
if (Settings.Gesture.AutoSwitchTwoFingerGesture)
{
if (Topmost)
{
2025-05-25 09:29:48 +08:00
Settings.Gesture.IsEnableTwoFingerTranslate = false;
2025-08-03 16:46:33 +08:00
}
else
{
2025-05-25 09:29:48 +08:00
Settings.Gesture.IsEnableTwoFingerTranslate = true;
}
}
2025-08-03 16:46:33 +08:00
}
else
{
2025-05-25 09:29:48 +08:00
Settings.Gesture = new Gesture();
}
// Canvas
2025-08-03 16:46:33 +08:00
if (Settings.Canvas != null)
{
2025-05-25 09:29:48 +08:00
drawingAttributes.Height = Settings.Canvas.InkWidth;
drawingAttributes.Width = Settings.Canvas.InkWidth;
InkWidthSlider.Value = Settings.Canvas.InkWidth * 2;
HighlighterWidthSlider.Value = Settings.Canvas.HighlighterWidth;
2026-02-14 12:49:39 +08:00
int alpha = (int)Settings.Canvas.InkAlpha;
if (alpha < 0) alpha = 0; if (alpha > 255) alpha = 255;
var inkColor = drawingAttributes.Color;
drawingAttributes.Color = Color.FromArgb((byte)alpha, inkColor.R, inkColor.G, inkColor.B);
inkCanvas.DefaultDrawingAttributes.Color = drawingAttributes.Color;
if (InkAlphaSlider != null) InkAlphaSlider.Value = alpha;
if (BoardInkAlphaSlider != null) BoardInkAlphaSlider.Value = alpha;
2025-05-25 09:29:48 +08:00
2025-08-03 16:46:33 +08:00
if (Settings.Canvas.UsingWhiteboard)
{
2025-05-25 09:29:48 +08:00
GridBackgroundCover.Background = new SolidColorBrush(Color.FromRgb(234, 235, 237));
WaterMarkTime.Foreground = new SolidColorBrush(Color.FromRgb(22, 41, 36));
WaterMarkDate.Foreground = new SolidColorBrush(Color.FromRgb(22, 41, 36));
BlackBoardWaterMark.Foreground = new SolidColorBrush(Color.FromRgb(22, 41, 36));
isUselightThemeColor = false;
2025-08-03 16:46:33 +08:00
}
else
{
2025-05-25 09:29:48 +08:00
GridBackgroundCover.Background = new SolidColorBrush(Color.FromRgb(22, 41, 36));
WaterMarkTime.Foreground = new SolidColorBrush(Color.FromRgb(234, 235, 237));
WaterMarkDate.Foreground = new SolidColorBrush(Color.FromRgb(234, 235, 237));
BlackBoardWaterMark.Foreground = new SolidColorBrush(Color.FromRgb(234, 235, 237));
isUselightThemeColor = true;
}
2025-08-03 16:46:33 +08:00
if (Settings.Canvas.IsShowCursor)
{
2025-05-25 09:29:48 +08:00
inkCanvas.ForceCursor = true;
2025-08-03 16:46:33 +08:00
}
else
{
2025-05-25 09:29:48 +08:00
inkCanvas.ForceCursor = false;
}
2025-08-03 16:46:33 +08:00
2025-06-17 22:05:22 +08:00
// 初始化屏蔽压感开关状态
2025-10-02 17:04:12 +08:00
inkCanvas.DefaultDrawingAttributes.IgnorePressure = Settings.Canvas.DisablePressure;
2025-06-17 22:05:22 +08:00
2026-04-05 09:38:16 +08:00
2026-03-28 17:04:50 +08:00
if (Settings.Canvas.EnableVelocityBrushTip)
{
Settings.Canvas.InkStyle = 3;
Settings.Canvas.EnableVelocityBrushTip = false;
}
if (Settings.Canvas.InkStyle < 0 || Settings.Canvas.InkStyle > 3)
Settings.Canvas.InkStyle = 0;
2026-03-28 17:11:30 +08:00
int penStyleUi = PenStyleUiIndexFromInkStyle(Settings.Canvas.InkStyle);
ComboBoxPenStyle.SelectedIndex = penStyleUi;
BoardComboBoxPenStyle.SelectedIndex = penStyleUi;
2025-05-25 09:29:48 +08:00
ComboBoxEraserSizeFloatingBar.SelectedIndex = Settings.Canvas.EraserSize;
BoardComboBoxEraserSize.SelectedIndex = Settings.Canvas.EraserSize;
2025-08-03 16:46:33 +08:00
switch (Settings.Canvas.EraserShapeType)
{
case 0:
{
double k = 1;
switch (Settings.Canvas.EraserSize)
{
case 0:
k = 0.5;
break;
case 1:
k = 0.8;
break;
case 3:
k = 1.25;
break;
case 4:
k = 1.5;
break;
}
inkCanvas.EraserShape = new EllipseStylusShape(k * 90, k * 90);
inkCanvas.EditingMode = InkCanvasEditingMode.None;
break;
2025-05-25 09:29:48 +08:00
}
2025-08-03 16:46:33 +08:00
case 1:
{
double k = 1;
switch (Settings.Canvas.EraserSize)
{
case 0:
k = 0.7;
break;
case 1:
k = 0.9;
break;
case 3:
k = 1.2;
break;
case 4:
k = 1.5;
break;
}
inkCanvas.EraserShape = new RectangleStylusShape(k * 90 * 0.6, k * 90);
inkCanvas.EditingMode = InkCanvasEditingMode.None;
break;
2025-05-25 09:29:48 +08:00
}
}
CheckEraserTypeTab();
2025-07-20 15:21:59 +08:00
// 初始化贝塞尔曲线平滑设置
if (Settings.Canvas.UseAdvancedBezierSmoothing)
{
// 如果启用高级贝塞尔平滑,则禁用原来的FitToCurve
drawingAttributes.FitToCurve = false;
}
else if (Settings.Canvas.FitToCurve)
{
// 如果启用原来的FitToCurve,则禁用高级贝塞尔平滑
2025-05-25 09:29:48 +08:00
drawingAttributes.FitToCurve = true;
2025-07-20 15:21:59 +08:00
}
else
{
// 两者都禁用
2025-05-25 09:29:48 +08:00
drawingAttributes.FitToCurve = false;
}
2025-07-26 19:03:07 +08:00
2025-06-17 22:37:37 +08:00
// 初始化直线自动拉直相关设置
2025-06-19 14:30:24 +08:00
// 直线拉直灵敏度也在这里初始化,即使它存储在InkToShape中
2025-06-29 14:15:20 +08:00
// 初始化高精度直线拉直设置
2025-08-03 16:46:33 +08:00
2025-06-17 22:37:37 +08:00
// 初始化直线端点吸附相关设置
2025-08-03 16:46:33 +08:00
}
else
{
2025-05-25 09:29:48 +08:00
Settings.Canvas = new Canvas();
}
// Advanced - UI initialization (settings loading moved to AdvancedPage)
2025-08-03 16:46:33 +08:00
if (Settings.Advanced != null)
{
if (Settings.Advanced.IsEnableFullScreenHelper)
{
2025-05-25 09:29:48 +08:00
FullScreenHelper.MarkFullscreenWindowTaskbarList(new WindowInteropHelper(this).Handle, true);
}
2025-06-06 11:15:01 +08:00
if (Settings.Advanced.IsEnableAvoidFullScreenHelper)
{
AvoidFullScreenHelper.StartAvoidFullScreen(this);
2025-10-02 18:48:11 +08:00
Dispatcher.BeginInvoke(new Action(() =>
{
if (isLoaded)
{
MoveWindow(new WindowInteropHelper(this).Handle, 0, 0,
WinForms.Screen.PrimaryScreen.Bounds.Width, WinForms.Screen.PrimaryScreen.Bounds.Height, true);
}
}), DispatcherPriority.ApplicationIdle);
2025-06-06 11:15:01 +08:00
}
2025-08-03 16:46:33 +08:00
if (Settings.Advanced.IsEnableEdgeGestureUtil)
{
2025-05-25 09:29:48 +08:00
if (OSVersion.GetOperatingSystem() >= OperatingSystem.Windows10)
EdgeGestureUtil.DisableEdgeGestures(new WindowInteropHelper(this).Handle, true);
}
2025-08-03 16:46:33 +08:00
}
else
{
2025-05-25 09:29:48 +08:00
Settings.Advanced = new Advanced();
}
// InkToShape
2025-08-03 16:46:33 +08:00
if (Settings.InkToShape != null)
{
FloatingBarToggleSwitchEnableInkToShape.IsOn = Settings.InkToShape.IsInkToShapeEnabled;
BoardToggleSwitchEnableInkToShape.IsOn = Settings.InkToShape.IsInkToShapeEnabled;
2025-08-03 16:46:33 +08:00
}
else
{
2025-05-25 09:29:48 +08:00
Settings.InkToShape = new InkToShape();
}
// RandSettings - UI initialization (settings loading moved to RandomDrawPage)
2025-08-03 16:46:33 +08:00
if (Settings.RandSettings != null)
{
2026-04-13 13:33:08 +08:00
BoardRandomDrawToolBtn.Visibility = Settings.RandSettings.ShowRandomAndSingleDraw ? Visibility.Visible : Visibility.Collapsed;
BoardSingleDrawToolBtn.Visibility = Settings.RandSettings.ShowRandomAndSingleDraw ? Visibility.Visible : Visibility.Collapsed;
2025-08-03 16:46:33 +08:00
}
else
{
2025-05-25 09:29:48 +08:00
Settings.RandSettings = new RandSettings();
}
2025-09-06 21:26:46 +08:00
// ModeSettings
2026-04-23 23:53:52 +08:00
if (Settings.ModeSettings == null)
2025-09-06 21:26:46 +08:00
{
2026-04-23 23:53:52 +08:00
Settings.ModeSettings = new ModeSettings();
2025-09-06 21:26:46 +08:00
}
2026-04-23 23:53:52 +08:00
if (isStartup && Settings.ModeSettings.IsPPTOnlyMode)
2025-09-06 21:26:46 +08:00
{
2026-04-23 23:53:52 +08:00
Hide();
LogHelper.WriteLogToFile("启动时检测到仅PPT模式,主窗口已隐藏", LogHelper.LogType.Event);
2025-09-06 21:26:46 +08:00
}
2025-05-25 09:29:48 +08:00
// Automation
2025-08-03 16:46:33 +08:00
if (Settings.Automation != null)
{
2025-05-25 09:29:48 +08:00
StartOrStoptimerCheckAutoFold();
2025-05-25 09:29:48 +08:00
if (Settings.Automation.IsAutoKillEasiNote || Settings.Automation.IsAutoKillPptService ||
Settings.Automation.IsAutoKillHiteAnnotation || Settings.Automation.IsAutoKillInkCanvas
|| Settings.Automation.IsAutoKillICA || Settings.Automation.IsAutoKillIDT ||
Settings.Automation.IsAutoKillVComYouJiao
2025-08-03 16:46:33 +08:00
|| Settings.Automation.IsAutoKillSeewoLauncher2DesktopAnnotation)
{
2025-05-25 09:29:48 +08:00
timerKillProcess.Start();
2025-08-03 16:46:33 +08:00
}
else
{
2025-05-25 09:29:48 +08:00
timerKillProcess.Stop();
}
2025-08-03 16:46:33 +08:00
}
else
{
2025-05-25 09:29:48 +08:00
Settings.Automation = new Automation();
}
// auto align
2025-08-03 16:46:33 +08:00
if (BtnPPTSlideShowEnd.Visibility == Visibility.Visible)
{
2025-05-25 09:29:48 +08:00
ViewboxFloatingBarMarginAnimation(60);
2025-08-03 16:46:33 +08:00
}
else
{
2025-05-25 09:29:48 +08:00
ViewboxFloatingBarMarginAnimation(100, true);
}
2025-08-23 23:13:39 +08:00
// 加载墨迹渐隐设置
LoadInkFadeSettings();
2026-02-14 00:23:05 +08:00
// 加载画笔自动恢复设置
LoadBrushAutoRestoreSettings();
}
/// <summary>
/// 将画笔自动恢复相关的设置应用到界面控件并在启用时初始化自动恢复定时器。
/// </summary>
/// <remarks>
/// 会将 Settings.Canvas 中的 BrushAutoRestore 配置同步到对应的切换开关、时间文本框、颜色下拉框、宽度和透明度滑块;当颜色缺失时会使用默认值 `#FFFF0000`,当宽度无效时使用默认值 `5`。若功能被启用,会初始化并启动定时器以执行自动恢复任务。方法执行过程中会记录加载结果或错误信息到日志。
/// </remarks>
2026-02-14 00:23:05 +08:00
private void LoadBrushAutoRestoreSettings()
{
try
{
2026-03-03 16:04:20 +08:00
2026-02-14 00:23:05 +08:00
// 如果功能已启用,初始化并启动定时器
if (Settings.Canvas.EnableBrushAutoRestore)
{
InitBrushAutoRestoreTimer();
ScheduleBrushAutoRestore();
}
LogHelper.WriteLogToFile("画笔自动恢复设置已加载", LogHelper.LogType.Event);
}
catch (Exception ex)
{
LogHelper.WriteLogToFile($"加载画笔自动恢复设置时出错: {ex.Message}", LogHelper.LogType.Error);
}
2025-08-23 23:13:39 +08:00
}
/// <summary>
/// 加载墨迹渐隐设置
/// </summary>
private void LoadInkFadeSettings()
{
try
{
// 同步批注子面板中的开关状态
if (ToggleSwitchInkFadeInPanel != null)
{
ToggleSwitchInkFadeInPanel.IsOn = Settings.Canvas.EnableInkFade;
}
2025-08-24 09:17:58 +08:00
// 同步普通画笔面板中的开关状态
if (ToggleSwitchInkFadeInPanel2 != null)
{
ToggleSwitchInkFadeInPanel2.IsOn = Settings.Canvas.EnableInkFade;
}
2025-08-23 23:13:39 +08:00
// 同步墨迹渐隐管理器的状态
if (_inkFadeManager != null)
{
_inkFadeManager.IsEnabled = Settings.Canvas.EnableInkFade;
_inkFadeManager.UpdateFadeTime(Settings.Canvas.InkFadeTime);
}
// 根据设置更新墨迹渐隐控制开关的可见性
UpdateInkFadeControlVisibility();
2025-08-23 23:13:39 +08:00
LogHelper.WriteLogToFile("墨迹渐隐设置已加载", LogHelper.LogType.Event);
}
catch (Exception ex)
{
LogHelper.WriteLogToFile($"加载墨迹渐隐设置时出错: {ex.Message}", LogHelper.LogType.Error);
}
2025-05-25 09:29:48 +08:00
}
2025-12-20 17:52:56 +08:00
/// <summary>
/// 清理配置文件中的过期设置
/// </summary>
2025-12-20 17:52:56 +08:00
/// <param name="userConfigJson">用户配置的JSON字符串</param>
/// <remarks>
/// 清理过期设置时:
/// 1. 创建默认配置对象
/// 2. 将默认配置和用户配置都序列化为JObject
/// 3. 递归比较并删除用户配置中多余的键
/// 4. 如果有清理操作,重新反序列化并保存
/// 5. 记录清理结果到日志
/// </remarks>
2025-12-20 17:52:56 +08:00
private void CleanupObsoleteSettings(string userConfigJson)
{
try
{
// 创建默认配置对象
Settings defaultSettings = new Settings();
// 将默认配置和用户配置都序列化为JObject
2026-04-05 14:18:33 +08:00
JObject defaultConfigObj = JObject.FromObject(defaultSettings); EnsureDefaultConfigSchemaIncludesIgnoredNullKeys(defaultConfigObj);
2025-12-20 17:52:56 +08:00
JObject userConfigObj = JObject.Parse(userConfigJson);
// 记录是否有清理操作
bool hasChanges = false;
// 递归比较并删除用户配置中多余的键
RemoveObsoleteProperties(userConfigObj, defaultConfigObj, ref hasChanges);
// 如果有清理操作,重新反序列化并保存
if (hasChanges)
{
string cleanedJson = userConfigObj.ToString(Formatting.Indented);
Settings = JsonConvert.DeserializeObject<Settings>(cleanedJson);
SaveSettingsToFile();
LogHelper.WriteLogToFile("已清理过期配置项", LogHelper.LogType.Event);
}
}
catch (Exception ex)
{
LogHelper.WriteLogToFile($"清理过期配置时出错: {ex.Message}", LogHelper.LogType.Error);
}
}
/// <summary>
/// 递归删除用户配置中多余的属性
/// </summary>
2025-12-20 17:52:56 +08:00
/// <param name="userObj">用户配置的JObject</param>
/// <param name="defaultObj">默认配置的JObject</param>
/// <param name="hasChanges">是否有变更的引用标志</param>
/// <remarks>
/// 递归删除多余属性时:
/// 1. 检查用户配置和默认配置是否为空
/// 2. 获取需要删除的键列表
/// 3. 遍历用户配置的所有属性
/// 4. 如果默认配置中不存在该属性,标记为删除
/// 5. 如果两个属性都是对象类型,递归比较
/// 6. 处理数组中的对象(如自定义图标列表等)
/// 7. 删除标记的键
/// 8. 设置变更标志
/// </remarks>
2026-04-05 09:25:53 +08:00
private static void EnsureDefaultConfigSchemaIncludesIgnoredNullKeys(JObject defaultConfigObj)
{
if (defaultConfigObj == null) return;
if (defaultConfigObj["appearance"] is JObject appearance && !appearance.ContainsKey("hitokotoCategories"))
appearance["hitokotoCategories"] = JValue.CreateNull();
}
2025-12-20 17:52:56 +08:00
private void RemoveObsoleteProperties(JObject userObj, JObject defaultObj, ref bool hasChanges)
{
if (userObj == null || defaultObj == null)
return;
// 获取需要删除的键列表(避免在遍历时修改集合)
List<string> keysToRemove = new List<string>();
foreach (var property in userObj.Properties())
{
string propertyName = property.Name;
// 如果默认配置中不存在该属性,标记为删除
if (!defaultObj.ContainsKey(propertyName))
{
keysToRemove.Add(propertyName);
continue;
}
// 如果两个属性都是对象类型,递归比较
JToken userValue = property.Value;
JToken defaultValue = defaultObj[propertyName];
if (userValue != null && defaultValue != null)
{
if (userValue.Type == JTokenType.Object && defaultValue.Type == JTokenType.Object)
{
RemoveObsoleteProperties(userValue as JObject, defaultValue as JObject, ref hasChanges);
}
// 处理数组中的对象(如自定义图标列表等)
else if (userValue.Type == JTokenType.Array && defaultValue.Type == JTokenType.Array)
{
JArray userArray = userValue as JArray;
JArray defaultArray = defaultValue as JArray;
if (userArray != null && defaultArray != null && userArray.Count > 0 && defaultArray.Count > 0)
{
// 如果数组元素是对象,比较第一个元素的属性结构
if (userArray[0].Type == JTokenType.Object && defaultArray[0].Type == JTokenType.Object)
{
for (int i = 0; i < userArray.Count; i++)
{
if (userArray[i] is JObject userItemObj && defaultArray[0] is JObject defaultItemObj)
{
RemoveObsoleteProperties(userItemObj, defaultItemObj, ref hasChanges);
}
}
}
}
}
}
}
// 删除标记的键
foreach (string key in keysToRemove)
{
userObj.Remove(key);
hasChanges = true;
}
}
2025-05-25 09:29:48 +08:00
}
2026-04-19 14:42:11 +08:00
}