AX Commander 비교본 런처 기능 대량 이식
변경 목적: Agent Compare 아래 비교본의 개발 문서와 런처 소스를 기준으로 현재 AX Commander에 빠져 있던 신규 런처 기능을 동일한 흐름으로 옮겨, 비교본 수준의 기능 폭을 현재 제품에 반영했습니다. 핵심 수정사항: 비교본의 신규 런처 핸들러 다수를 src/AxCopilot/Handlers로 이식하고 App.xaml.cs 등록 흐름에 연결했습니다. 빠른 링크, 파일 태그, 알림 센터, 포모도로, 파일 브라우저, 핫키 관리, OCR, 세션/스케줄/매크로, Git/정규식/네트워크/압축/해시/UUID/JWT/QR 등 AX Commander 기능을 추가했습니다. 핵심 수정사항: 신규 기능이 실제 동작하도록 AppSettings 확장, SchedulerService/FileTagService/NotificationCenterService/IconCacheService/UrlTemplateEngine/PomodoroService 추가, 배치 이름변경/세션/스케줄/매크로 편집 창 추가, NotificationService와 Symbols 보강, QR/OCR용 csproj 의존성과 Windows 타겟 프레임워크를 반영했습니다. 문서 반영: README.md와 docs/DEVELOPMENT.md에 비교본 기반 런처 기능 이식 이력과 검증 결과를 업데이트했습니다. 검증 결과: dotnet build src/AxCopilot/AxCopilot.csproj -c Release -v minimal -p:OutputPath=bin\\verify\\ -p:IntermediateOutputPath=obj\\verify\\ 실행 기준 경고 0개, 오류 0개를 확인했습니다.
This commit is contained in:
386
src/AxCopilot/Views/SessionEditorWindow.xaml.cs
Normal file
386
src/AxCopilot/Views/SessionEditorWindow.xaml.cs
Normal file
@@ -0,0 +1,386 @@
|
||||
using System.IO;
|
||||
using System.Windows;
|
||||
using System.Windows.Controls;
|
||||
using System.Windows.Input;
|
||||
using System.Windows.Media;
|
||||
using AxCopilot.Models;
|
||||
using AxCopilot.Services;
|
||||
|
||||
namespace AxCopilot.Views;
|
||||
|
||||
/// <summary>
|
||||
/// L5-4: 앱 세션 편집기.
|
||||
/// 세션 이름, 앱 목록(경로 + 라벨 + 스냅 위치)을 편집하여 저장합니다.
|
||||
/// </summary>
|
||||
public partial class SessionEditorWindow : Window
|
||||
{
|
||||
private readonly SettingsService _settings;
|
||||
private readonly AppSession? _original; // 편집 모드 원본 (새 세션이면 null)
|
||||
private readonly List<AppRowUi> _rows = new();
|
||||
|
||||
// 스냅 팝업 대상 행
|
||||
private AppRowUi? _snapTargetRow;
|
||||
|
||||
// 사용 가능한 스냅 위치 목록 (키 → 표시명)
|
||||
private static readonly (string Key, string Label)[] SnapOptions =
|
||||
[
|
||||
("full", "전체화면"),
|
||||
("left", "왼쪽 절반"),
|
||||
("right", "오른쪽 절반"),
|
||||
("tl", "좌상단 1/4"),
|
||||
("tr", "우상단 1/4"),
|
||||
("bl", "좌하단 1/4"),
|
||||
("br", "우하단 1/4"),
|
||||
("center", "중앙 80%"),
|
||||
("third-l", "좌측 1/3"),
|
||||
("third-c", "중앙 1/3"),
|
||||
("third-r", "우측 1/3"),
|
||||
("two3-l", "좌측 2/3"),
|
||||
("two3-r", "우측 2/3"),
|
||||
("none", "스냅 없음"),
|
||||
];
|
||||
|
||||
// ──────────────────────────────────────────────────────────────────────
|
||||
/// <summary>
|
||||
/// 세션 편집기를 엽니다.
|
||||
/// </summary>
|
||||
/// <param name="session">편집할 기존 세션. null이면 새로 만들기 모드.</param>
|
||||
/// <param name="settings">설정 서비스.</param>
|
||||
public SessionEditorWindow(AppSession? session, SettingsService settings)
|
||||
{
|
||||
InitializeComponent();
|
||||
_settings = settings;
|
||||
_original = session;
|
||||
|
||||
BuildSnapPopup();
|
||||
LoadSession(session);
|
||||
}
|
||||
|
||||
/// <summary>새 세션 모드일 때 기본 이름을 설정합니다.</summary>
|
||||
public string InitialName
|
||||
{
|
||||
set { if (_original == null) NameBox.Text = value; }
|
||||
}
|
||||
|
||||
// ─── 초기화 ───────────────────────────────────────────────────────────
|
||||
private void LoadSession(AppSession? session)
|
||||
{
|
||||
if (session == null)
|
||||
{
|
||||
NameBox.Text = "새 세션";
|
||||
DescBox.Text = "";
|
||||
}
|
||||
else
|
||||
{
|
||||
NameBox.Text = session.Name;
|
||||
DescBox.Text = session.Description;
|
||||
foreach (var app in session.Apps)
|
||||
AddRow(app.Path, app.Label, app.SnapPosition, app.Arguments, app.DelayMs);
|
||||
}
|
||||
|
||||
RefreshEmptyState();
|
||||
}
|
||||
|
||||
private void BuildSnapPopup()
|
||||
{
|
||||
var panel = new StackPanel { Margin = new Thickness(0) };
|
||||
|
||||
foreach (var (key, label) in SnapOptions)
|
||||
{
|
||||
var keyCapture = key;
|
||||
var border = new Border
|
||||
{
|
||||
CornerRadius = new CornerRadius(4),
|
||||
Padding = new Thickness(10, 5, 10, 5),
|
||||
Cursor = Cursors.Hand
|
||||
};
|
||||
|
||||
border.MouseEnter += (_, _) =>
|
||||
border.Background = new SolidColorBrush(Color.FromArgb(0x28, 0xFF, 0xFF, 0xFF));
|
||||
border.MouseLeave += (_, _) =>
|
||||
border.Background = Brushes.Transparent;
|
||||
|
||||
var stack = new StackPanel { Orientation = Orientation.Horizontal };
|
||||
stack.Children.Add(new TextBlock
|
||||
{
|
||||
Text = keyCapture,
|
||||
FontFamily = new FontFamily("Cascadia Code, Consolas"),
|
||||
FontSize = 11,
|
||||
Foreground = TryFindResource("AccentColor") as Brush ?? Brushes.DodgerBlue,
|
||||
VerticalAlignment = VerticalAlignment.Center,
|
||||
MinWidth = 68,
|
||||
});
|
||||
stack.Children.Add(new TextBlock
|
||||
{
|
||||
Text = label,
|
||||
FontSize = 11,
|
||||
Foreground = TryFindResource("SecondaryText") as Brush ?? Brushes.Gray,
|
||||
VerticalAlignment = VerticalAlignment.Center,
|
||||
});
|
||||
border.Child = stack;
|
||||
|
||||
border.MouseLeftButtonUp += (_, _) =>
|
||||
{
|
||||
if (_snapTargetRow != null)
|
||||
{
|
||||
_snapTargetRow.SnapPosition = keyCapture;
|
||||
_snapTargetRow.UpdateSnapLabel();
|
||||
}
|
||||
SnapPickerPopup.IsOpen = false;
|
||||
};
|
||||
|
||||
panel.Children.Add(border);
|
||||
}
|
||||
|
||||
SnapOptionsList.Content = panel;
|
||||
}
|
||||
|
||||
// ─── 앱 행 추가 ───────────────────────────────────────────────────────
|
||||
private void AddRow(string path = "", string label = "", string snap = "full",
|
||||
string args = "", int delayMs = 0)
|
||||
{
|
||||
var row = new AppRowUi(path, label, snap, args, delayMs);
|
||||
_rows.Add(row);
|
||||
|
||||
var rowGrid = BuildRowGrid(row);
|
||||
AppListPanel.Children.Add(rowGrid);
|
||||
RefreshEmptyState();
|
||||
}
|
||||
|
||||
private Grid BuildRowGrid(AppRowUi row)
|
||||
{
|
||||
var grid = new Grid { Margin = new Thickness(14, 2, 4, 2) };
|
||||
grid.ColumnDefinitions.Add(new ColumnDefinition { Width = new GridLength(1, GridUnitType.Star) });
|
||||
grid.ColumnDefinitions.Add(new ColumnDefinition { Width = new GridLength(100) });
|
||||
grid.ColumnDefinitions.Add(new ColumnDefinition { Width = new GridLength(108) });
|
||||
grid.ColumnDefinitions.Add(new ColumnDefinition { Width = new GridLength(30) });
|
||||
|
||||
// 경로 TextBox
|
||||
var pathBox = new TextBox
|
||||
{
|
||||
Text = row.Path,
|
||||
FontSize = 11,
|
||||
Foreground = TryFindResource("PrimaryText") as Brush ?? Brushes.White,
|
||||
Background = TryFindResource("LauncherBackground") as Brush ?? Brushes.Black,
|
||||
BorderBrush = TryFindResource("BorderColor") as Brush ?? Brushes.Gray,
|
||||
BorderThickness = new Thickness(1),
|
||||
Padding = new Thickness(6, 4, 6, 4),
|
||||
VerticalContentAlignment = VerticalAlignment.Center,
|
||||
ToolTip = "앱 실행 파일 경로",
|
||||
Margin = new Thickness(0, 0, 4, 0),
|
||||
};
|
||||
pathBox.TextChanged += (_, _) => row.Path = pathBox.Text;
|
||||
Grid.SetColumn(pathBox, 0);
|
||||
|
||||
// 라벨 TextBox
|
||||
var labelBox = new TextBox
|
||||
{
|
||||
Text = row.Label,
|
||||
FontSize = 11,
|
||||
Foreground = TryFindResource("SecondaryText") as Brush ?? Brushes.Gray,
|
||||
Background = TryFindResource("LauncherBackground") as Brush ?? Brushes.Black,
|
||||
BorderBrush = TryFindResource("BorderColor") as Brush ?? Brushes.Gray,
|
||||
BorderThickness = new Thickness(1),
|
||||
Padding = new Thickness(6, 4, 6, 4),
|
||||
VerticalContentAlignment = VerticalAlignment.Center,
|
||||
ToolTip = "표시 이름 (선택)",
|
||||
Margin = new Thickness(0, 0, 4, 0),
|
||||
};
|
||||
labelBox.TextChanged += (_, _) => row.Label = labelBox.Text;
|
||||
Grid.SetColumn(labelBox, 1);
|
||||
|
||||
// 스냅 선택 Border (클릭 시 팝업)
|
||||
var snapBtn = new Border
|
||||
{
|
||||
CornerRadius = new CornerRadius(4),
|
||||
BorderBrush = TryFindResource("BorderColor") as Brush ?? Brushes.Gray,
|
||||
BorderThickness = new Thickness(1),
|
||||
Padding = new Thickness(6, 4, 6, 4),
|
||||
Cursor = Cursors.Hand,
|
||||
Margin = new Thickness(0, 0, 4, 0),
|
||||
ToolTip = "스냅 위치 선택",
|
||||
};
|
||||
snapBtn.MouseEnter += (_, _) =>
|
||||
snapBtn.Background = new SolidColorBrush(Color.FromArgb(0x18, 0xFF, 0xFF, 0xFF));
|
||||
snapBtn.MouseLeave += (_, _) =>
|
||||
snapBtn.Background = Brushes.Transparent;
|
||||
|
||||
var snapLabel = new TextBlock
|
||||
{
|
||||
FontFamily = new FontFamily("Cascadia Code, Consolas"),
|
||||
FontSize = 10,
|
||||
Foreground = TryFindResource("AccentColor") as Brush ?? Brushes.DodgerBlue,
|
||||
VerticalAlignment = VerticalAlignment.Center,
|
||||
TextTrimming = TextTrimming.CharacterEllipsis,
|
||||
};
|
||||
snapBtn.Child = snapLabel;
|
||||
|
||||
// AppRowUi가 라벨 TextBlock을 참조할 수 있도록 저장
|
||||
row.SnapLabelRef = snapLabel;
|
||||
row.SnapButtonRef = snapBtn;
|
||||
row.UpdateSnapLabel();
|
||||
|
||||
snapBtn.MouseLeftButtonUp += (sender, e) =>
|
||||
{
|
||||
_snapTargetRow = row;
|
||||
SnapPickerPopup.PlacementTarget = (FrameworkElement)sender;
|
||||
SnapPickerPopup.IsOpen = true;
|
||||
e.Handled = true;
|
||||
};
|
||||
Grid.SetColumn(snapBtn, 2);
|
||||
|
||||
// 삭제 버튼
|
||||
var delBtn = new Border
|
||||
{
|
||||
Width = 24,
|
||||
Height = 24,
|
||||
CornerRadius = new CornerRadius(4),
|
||||
Cursor = Cursors.Hand,
|
||||
ToolTip = "이 앱 제거",
|
||||
VerticalAlignment = VerticalAlignment.Center,
|
||||
HorizontalAlignment = HorizontalAlignment.Center,
|
||||
};
|
||||
delBtn.MouseEnter += (_, _) =>
|
||||
delBtn.Background = new SolidColorBrush(Color.FromArgb(0x30, 0xEF, 0x53, 0x50));
|
||||
delBtn.MouseLeave += (_, _) =>
|
||||
delBtn.Background = Brushes.Transparent;
|
||||
|
||||
var delIcon = new TextBlock
|
||||
{
|
||||
Text = "\uE74D",
|
||||
FontFamily = new FontFamily("Segoe MDL2 Assets"),
|
||||
FontSize = 12,
|
||||
Foreground = new SolidColorBrush(Color.FromRgb(0xEF, 0x53, 0x50)),
|
||||
HorizontalAlignment = HorizontalAlignment.Center,
|
||||
VerticalAlignment = VerticalAlignment.Center,
|
||||
};
|
||||
delBtn.Child = delIcon;
|
||||
|
||||
delBtn.MouseLeftButtonUp += (_, _) =>
|
||||
{
|
||||
_rows.Remove(row);
|
||||
AppListPanel.Children.Remove(grid);
|
||||
RefreshEmptyState();
|
||||
};
|
||||
Grid.SetColumn(delBtn, 3);
|
||||
|
||||
grid.Children.Add(pathBox);
|
||||
grid.Children.Add(labelBox);
|
||||
grid.Children.Add(snapBtn);
|
||||
grid.Children.Add(delBtn);
|
||||
|
||||
return grid;
|
||||
}
|
||||
|
||||
private void RefreshEmptyState()
|
||||
{
|
||||
EmptyState.Visibility = _rows.Count == 0 ? Visibility.Visible : Visibility.Collapsed;
|
||||
}
|
||||
|
||||
// ─── 이벤트 핸들러 ────────────────────────────────────────────────────
|
||||
private void TitleBar_MouseDown(object sender, MouseButtonEventArgs e)
|
||||
{
|
||||
if (e.LeftButton == MouseButtonState.Pressed) DragMove();
|
||||
}
|
||||
|
||||
private void BtnClose_Click(object sender, MouseButtonEventArgs e) => Close();
|
||||
private void BtnCancel_Click(object sender, MouseButtonEventArgs e) => Close();
|
||||
|
||||
private void BtnAddApp_Click(object sender, MouseButtonEventArgs e)
|
||||
{
|
||||
using var dlg = new System.Windows.Forms.OpenFileDialog
|
||||
{
|
||||
Title = "앱 실행 파일 선택",
|
||||
Filter = "실행 파일 (*.exe)|*.exe|모든 파일 (*.*)|*.*",
|
||||
};
|
||||
if (dlg.ShowDialog() != System.Windows.Forms.DialogResult.OK) return;
|
||||
|
||||
var label = Path.GetFileNameWithoutExtension(dlg.FileName);
|
||||
AddRow(dlg.FileName, label, "full");
|
||||
}
|
||||
|
||||
private void BtnSave_Click(object sender, MouseButtonEventArgs e)
|
||||
{
|
||||
var name = NameBox.Text.Trim();
|
||||
if (string.IsNullOrWhiteSpace(name))
|
||||
{
|
||||
NotificationService.Notify("세션 편집기", "세션 이름을 입력하세요.");
|
||||
NameBox.Focus();
|
||||
return;
|
||||
}
|
||||
|
||||
// 빈 경로 행 필터링
|
||||
var validApps = _rows
|
||||
.Where(r => !string.IsNullOrWhiteSpace(r.Path))
|
||||
.Select(r => new SessionApp
|
||||
{
|
||||
Path = r.Path.Trim(),
|
||||
Arguments = r.Arguments.Trim(),
|
||||
Label = r.Label.Trim(),
|
||||
SnapPosition = r.SnapPosition,
|
||||
DelayMs = r.DelayMs,
|
||||
}).ToList();
|
||||
|
||||
var session = new AppSession
|
||||
{
|
||||
Name = name,
|
||||
Description = DescBox.Text.Trim(),
|
||||
Apps = validApps,
|
||||
CreatedAt = _original?.CreatedAt ?? DateTime.Now,
|
||||
};
|
||||
|
||||
// 기존 세션 교체 또는 신규 추가
|
||||
if (_original != null)
|
||||
{
|
||||
var idx = _settings.Settings.AppSessions.IndexOf(_original);
|
||||
if (idx >= 0)
|
||||
_settings.Settings.AppSessions[idx] = session;
|
||||
else
|
||||
_settings.Settings.AppSessions.Add(session);
|
||||
}
|
||||
else
|
||||
{
|
||||
// 동일 이름 세션이 있으면 교체
|
||||
var existing = _settings.Settings.AppSessions
|
||||
.FirstOrDefault(s => s.Name.Equals(name, StringComparison.OrdinalIgnoreCase));
|
||||
if (existing != null)
|
||||
_settings.Settings.AppSessions.Remove(existing);
|
||||
_settings.Settings.AppSessions.Add(session);
|
||||
}
|
||||
|
||||
_settings.Save();
|
||||
NotificationService.Notify("AX Copilot",
|
||||
$"세션 '{name}' 저장됨 ({validApps.Count}개 앱)");
|
||||
LogService.Info($"세션 저장: {name} ({validApps.Count}개 앱)");
|
||||
Close();
|
||||
}
|
||||
}
|
||||
|
||||
// ─── 앱 행 UI 모델 ────────────────────────────────────────────────────────────
|
||||
internal class AppRowUi
|
||||
{
|
||||
public string Path { get; set; }
|
||||
public string Label { get; set; }
|
||||
public string SnapPosition { get; set; }
|
||||
public string Arguments { get; set; }
|
||||
public int DelayMs { get; set; }
|
||||
|
||||
// UI 참조 (라벨 갱신용)
|
||||
internal System.Windows.Controls.TextBlock? SnapLabelRef { get; set; }
|
||||
internal System.Windows.Controls.Border? SnapButtonRef { get; set; }
|
||||
|
||||
public AppRowUi(string path, string label, string snap, string args, int delayMs)
|
||||
{
|
||||
Path = path;
|
||||
Label = label;
|
||||
SnapPosition = snap;
|
||||
Arguments = args;
|
||||
DelayMs = delayMs;
|
||||
}
|
||||
|
||||
public void UpdateSnapLabel()
|
||||
{
|
||||
if (SnapLabelRef != null)
|
||||
SnapLabelRef.Text = SnapPosition;
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user