refactor: cleaup and implemented mvvm

This commit is contained in:
tux
2025-07-05 06:18:00 +05:30
parent 896b2a9f40
commit 7ffdfea18f
13 changed files with 249 additions and 210 deletions

View File

@@ -2,14 +2,16 @@
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:vm="using:highminded.models"
mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="450"
x:Class="highminded.ui.controls.ChatUserControl">
x:Class="highminded.ui.controls.ChatUserControl"
x:DataType="vm:ChatViewModel">
<Grid RowDefinitions="*,Auto" Margin="15" RowSpacing="10">
<HtmlPanel Grid.Row="0" Name="ResultBlock" BaseStylesheet="* { font: Inter; color: white; }"
IsContextMenuEnabled="False" IsSelectionEnabled="False" />
<TextBox Grid.Row="1" Name="PromptBox" CornerRadius="5" TextWrapping="Wrap"
KeyDown="PromptBox_OnKeyDown" />
<HtmlPanel Grid.Row="0" BaseStylesheet="* { font: Inter; color: white; }"
IsContextMenuEnabled="False" IsSelectionEnabled="False" Text="{Binding Content}" />
<TextBox Grid.Row="1" CornerRadius="5" TextWrapping="Wrap"
KeyDown="PromptBox_OnKeyDown" Text="{Binding Prompt}" />
</Grid>
</UserControl>

View File

@@ -7,6 +7,7 @@ using System.Linq;
using System.Text;
using System.Threading.Tasks;
using Avalonia.Input;
using highminded.models;
using highminded.utils;
using OpenAI.Chat;
using Markdig;
@@ -16,14 +17,18 @@ namespace highminded.ui.controls;
public partial class ChatUserControl : UserControl
{
private readonly MarkdownPipeline _pipeline = null!;
private AudioCapture _audioCapture = null!;
private readonly ChatViewModel _model;
private readonly MarkdownPipeline _pipeline;
private readonly AudioCapture _audioCapture;
public ChatUserControl()
{
InitializeComponent();
_pipeline = new MarkdownPipelineBuilder().UseAdvancedExtensions().UseColorCode().Build();
DataContext = InMemoryDb.Obj.ChatViewModel;
_audioCapture = new AudioCapture();
_model = InMemoryDb.Obj.ChatViewModel;
_pipeline = new MarkdownPipelineBuilder().UseAdvancedExtensions().UseColorCode().Build();
}
public void StartRecord()
@@ -33,42 +38,15 @@ public partial class ChatUserControl : UserControl
var dirPath = Path.Combine(Environment.CurrentDirectory, "audio");
var filePath = Path.Combine(dirPath, fileName);
Directory.CreateDirectory(dirPath);
InMemoryDb.Obj.MainViewModel.IsRecording = true;
_audioCapture.StartRecording(filePath);
}
public void StopRecord()
{
if (_audioCapture != null)
{
_audioCapture.StopRecording();
SendAudio();
}
}
public async void SendAudio()
{
try
{
var dirPath = Path.Combine(Environment.CurrentDirectory, "audio");
var latestAudio = new DirectoryInfo(dirPath).GetFiles().OrderByDescending(f => f.LastWriteTime).First();
var filePath = Path.Combine(dirPath, latestAudio.Name);
var audioFileBytes = await File.ReadAllBytesAsync(filePath);
var audioBytes = BinaryData.FromBytes(audioFileBytes);
List<ChatMessage> messages =
[
new UserChatMessage(
ChatMessageContentPart.CreateTextPart(InMemoryDb.Obj.SettingsManager.Settings.AudioPrompt),
ChatMessageContentPart.CreateInputAudioPart(audioBytes, ChatInputAudioFormat.Wav)
)
];
await ProcessChatStreamAsync(messages);
}
catch (Exception err)
{
ResultBlock.Text = err.Message;
}
_audioCapture.StopRecording();
InMemoryDb.Obj.MainViewModel.IsRecording = false;
SendAudio();
}
public async void SendScreenshot()
@@ -99,7 +77,33 @@ public partial class ChatUserControl : UserControl
}
catch (Exception err)
{
ResultBlock.Text = err.Message;
_model.Content = err.Message;
}
}
private async void SendAudio()
{
try
{
var dirPath = Path.Combine(Environment.CurrentDirectory, "audio");
var latestAudio = new DirectoryInfo(dirPath).GetFiles().OrderByDescending(f => f.LastWriteTime).First();
var filePath = Path.Combine(dirPath, latestAudio.Name);
var audioFileBytes = await File.ReadAllBytesAsync(filePath);
var audioBytes = BinaryData.FromBytes(audioFileBytes);
List<ChatMessage> messages =
[
new UserChatMessage(
ChatMessageContentPart.CreateTextPart(InMemoryDb.Obj.SettingsManager.Settings.AudioPrompt),
ChatMessageContentPart.CreateInputAudioPart(audioBytes, ChatInputAudioFormat.Wav)
)
];
await ProcessChatStreamAsync(messages);
}
catch (Exception err)
{
_model.Content = err.Message;
}
}
@@ -109,15 +113,14 @@ public partial class ChatUserControl : UserControl
{
if (e.Key != Key.Enter) return;
var prompt = PromptBox.Text;
if (prompt is null) return;
PromptBox.Clear();
var prompt = _model.Prompt;
if (prompt == string.Empty) return;
_model.Prompt = string.Empty;
await ProcessChatStreamAsync(prompt);
}
catch (Exception err)
{
ResultBlock.Text = err.Message;
_model.Content = err.Message;
}
}
@@ -140,7 +143,7 @@ public partial class ChatUserControl : UserControl
responseBuilder.Append(token);
var html = Markdig.Markdown.ToHtml(responseBuilder.ToString(), _pipeline);
ResultBlock.Text = html;
_model.Content = html;
}
}
}

View File

@@ -2,39 +2,42 @@
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:vm="using:highminded.models"
mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="450"
x:Class="highminded.ui.controls.SettingsUserControl">
x:Class="highminded.ui.controls.SettingsUserControl"
x:DataType="vm:SettingsViewModel">
<Grid Margin="15" RowSpacing="10" ColumnSpacing="10" RowDefinitions="Auto,Auto,Auto,Auto,Auto"
ColumnDefinitions="*,*">
<Grid Margin="15" RowSpacing="10" ColumnSpacing="10" RowDefinitions="Auto,Auto,Auto,Auto,Auto" ColumnDefinitions="*,*">
<StackPanel Grid.Row="0" Grid.Column="0">
<TextBlock Text="Model" />
<TextBox Name="ModelTextBox" />
<TextBox Text="{Binding Model}" />
</StackPanel>
<StackPanel Grid.Row="0" Grid.Column="1">
<TextBlock Text="API URL" />
<TextBox Name="ApiUrlTextBox" />
<TextBox Text="{Binding ApiUrl}" />
</StackPanel>
<StackPanel Grid.Row="1" Grid.Column="0" Grid.ColumnSpan="2">
<TextBlock Text="API Key" />
<TextBox Name="ApiKeyTextBox" PasswordChar="*" />
<TextBox PasswordChar="*" Text="{Binding ApiKey}" />
</StackPanel>
<StackPanel Grid.Row="2" Grid.Column="0" Grid.ColumnSpan="2">
<TextBlock Text="Screenshot Prompt" />
<TextBox Name="ScreenshotPromptTextBox" TextWrapping="Wrap" MinLines="3" MaxLines="3"/>
</StackPanel>
<StackPanel Grid.Row="2" Grid.Column="0" Grid.ColumnSpan="2">
<TextBlock Text="Screenshot Prompt" />
<TextBox TextWrapping="Wrap" MinLines="3" MaxLines="3" Text="{Binding ScreenshotPrompt}" />
</StackPanel>
<StackPanel Grid.Row="3" Grid.Column="0" Grid.ColumnSpan="2">
<TextBlock Text="Audio Prompt" />
<TextBox TextWrapping="Wrap" MinLines="3" MaxLines="3" Text="{Binding AudioPrompt}" />
</StackPanel>
<StackPanel Grid.Row="3" Grid.Column="0" Grid.ColumnSpan="2">
<TextBlock Text="Audio Prompt" />
<TextBox Name="AudioPromptTextbox" TextWrapping="Wrap" MinLines="3" MaxLines="3"/>
</StackPanel>
<StackPanel Grid.Row="4" Grid.Column="0" Grid.ColumnSpan="2">
<Button Name="SaveSettingsBtn" Content="Save" Click="SaveSettingsBtn_OnClick"/>
</StackPanel>
<Button Content="Save" Click="SaveSettingsBtn_OnClick" />
</StackPanel>
</Grid>
</UserControl>

View File

@@ -9,20 +9,11 @@ public partial class SettingsUserControl : UserControl
public SettingsUserControl()
{
InitializeComponent();
ModelTextBox.Text = InMemoryDb.Obj.SettingsManager.Settings.Model;
ApiUrlTextBox.Text = InMemoryDb.Obj.SettingsManager.Settings.ApiURL;
ApiKeyTextBox.Text = InMemoryDb.Obj.SettingsManager.Settings.ApiKey;
ScreenshotPromptTextBox.Text = InMemoryDb.Obj.SettingsManager.Settings.ScreenshotPrompt;
AudioPromptTextbox.Text = InMemoryDb.Obj.SettingsManager.Settings.AudioPrompt;
DataContext = InMemoryDb.Obj.SettingsViewModel;
}
private void SaveSettingsBtn_OnClick(object? sender, RoutedEventArgs e)
{
InMemoryDb.Obj.SaveSettings(new AppSettings()
{
ApiKey = ApiKeyTextBox.Text, ApiURL = ApiUrlTextBox.Text, Model = ModelTextBox.Text,
ScreenshotPrompt = ScreenshotPromptTextBox.Text, AudioPrompt = AudioPromptTextbox.Text
});
InMemoryDb.Obj.SaveSettings();
}
}