30 changed files with 3003 additions and 0 deletions
@ -0,0 +1,9 @@ |
|||||
|
<Application x:Class="SunlightAggregationManager.App" |
||||
|
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" |
||||
|
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" |
||||
|
xmlns:local="clr-namespace:SunlightAggregationManager" |
||||
|
StartupUri="MainWindow.xaml"> |
||||
|
<Application.Resources> |
||||
|
|
||||
|
</Application.Resources> |
||||
|
</Application> |
||||
@ -0,0 +1,17 @@ |
|||||
|
using SunlightAggregationManager.UserClass; |
||||
|
using System.Configuration; |
||||
|
using System.Data; |
||||
|
using System.Windows; |
||||
|
using System.Windows.Controls; |
||||
|
|
||||
|
namespace SunlightAggregationManager |
||||
|
{ |
||||
|
/// <summary>
|
||||
|
/// Interaction logic for App.xaml
|
||||
|
/// </summary>
|
||||
|
public partial class App : Application |
||||
|
{ |
||||
|
|
||||
|
} |
||||
|
|
||||
|
} |
||||
@ -0,0 +1,10 @@ |
|||||
|
using System.Windows; |
||||
|
|
||||
|
[assembly: ThemeInfo( |
||||
|
ResourceDictionaryLocation.None, //where theme specific resource dictionaries are located
|
||||
|
//(used if a resource is not found in the page,
|
||||
|
// or application resource dictionaries)
|
||||
|
ResourceDictionaryLocation.SourceAssembly //where the generic resource dictionary is located
|
||||
|
//(used if a resource is not found in the page,
|
||||
|
// app, or any theme specific resource dictionaries)
|
||||
|
)] |
||||
@ -0,0 +1,48 @@ |
|||||
|
using System; |
||||
|
using System.Collections.Generic; |
||||
|
using System.Linq; |
||||
|
using System.Text; |
||||
|
using System.Threading.Tasks; |
||||
|
using System.Windows; |
||||
|
using System.Windows.Data; |
||||
|
using System.Windows.Input; |
||||
|
using System.Windows.Controls; |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 运行状态变换器
|
||||
|
/// 输入:状态码
|
||||
|
/// 输出:状态文字
|
||||
|
/// </summary>
|
||||
|
namespace SunlightAggregationManager.ConvertMoels |
||||
|
{ |
||||
|
internal class StatenConvert : IValueConverter |
||||
|
{ |
||||
|
|
||||
|
public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture) |
||||
|
{ |
||||
|
if (value == null) |
||||
|
{ |
||||
|
return ""; |
||||
|
} |
||||
|
else |
||||
|
{ |
||||
|
string Staten = ""; |
||||
|
var i = value.ToString(); |
||||
|
|
||||
|
if (i == "10") Staten = ResourceLanguage.ERR;//异常
|
||||
|
if (i == "15") Staten = ResourceLanguage.Offine;//离线
|
||||
|
if (i == "20") Staten = ResourceLanguage.Online;//在线
|
||||
|
if (i == "99") Staten = ResourceLanguage.Disable;//禁用
|
||||
|
|
||||
|
return Staten; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
|
||||
|
public object ConvertBack(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture) |
||||
|
{ |
||||
|
throw new NotImplementedException(); |
||||
|
} |
||||
|
|
||||
|
} |
||||
|
} |
||||
@ -0,0 +1,48 @@ |
|||||
|
using System; |
||||
|
using System.Collections.Generic; |
||||
|
using System.Linq; |
||||
|
using System.Text; |
||||
|
using System.Threading.Tasks; |
||||
|
using System.Windows; |
||||
|
using System.Windows.Data; |
||||
|
using System.Windows.Input; |
||||
|
using System.Windows.Controls; |
||||
|
|
||||
|
namespace SunlightAggregationManager.ConvertMoels |
||||
|
{ |
||||
|
internal class StatenERRConvert : IValueConverter |
||||
|
{ |
||||
|
public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture) |
||||
|
{ |
||||
|
if (value == null) |
||||
|
{ |
||||
|
return "black"; |
||||
|
} |
||||
|
else |
||||
|
{ |
||||
|
if (value.ToString() == "99") |
||||
|
{ |
||||
|
return "red";//零返回红色
|
||||
|
} |
||||
|
else if (value.ToString() == "10") |
||||
|
{ |
||||
|
return "yellow";//零返回黄色
|
||||
|
} |
||||
|
else if (value.ToString() == "20") |
||||
|
{ |
||||
|
return "green";//零返回绿
|
||||
|
} |
||||
|
else |
||||
|
{ |
||||
|
return "black";//非零返回黑色
|
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
|
||||
|
public object ConvertBack(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture) |
||||
|
{ |
||||
|
throw new NotImplementedException(); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
@ -0,0 +1,72 @@ |
|||||
|
<Window x:Class="SunlightAggregationManager.MainWindow" |
||||
|
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" |
||||
|
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:local="clr-namespace:SunlightAggregationManager" |
||||
|
mc:Ignorable="d" |
||||
|
Loaded="Window_Loaded" |
||||
|
Title="SUNLIGHT" Height="720" Width="1280" WindowStyle="None" ResizeMode="NoResize"> |
||||
|
|
||||
|
<Grid> |
||||
|
<Grid x:Name="GridTitle" Height="50" VerticalAlignment="Top" Background="#FF006361" Margin="0,0,0,0" MouseDown="GridTitle_MouseDown"> |
||||
|
<TextBlock Text="Sunlight Aggregation Manager (SERVER)" HorizontalAlignment="Center" VerticalAlignment="Center" FontSize="22" Foreground="White"/> |
||||
|
<Button HorizontalAlignment="Right" Margin="0,0,5,0" VerticalAlignment="Center" Width="50" Click="exit_Click" |
||||
|
Background="#FF006361" BorderBrush="#FF006361" > |
||||
|
<!-- 按钮内容:矢量图标 --> |
||||
|
<Viewbox Width="50" Height="40"> |
||||
|
<Path x:Name="IconPath_exit" Data="M4,2 H14v20H4V2zm2,2v16h6V4H6zm12 8-4-4v3H8v2h6v3l4-4z" Fill="White" /> |
||||
|
</Viewbox> |
||||
|
</Button> |
||||
|
<Button HorizontalAlignment="Right" Margin="0,0,65,0" VerticalAlignment="Center" Width="50" Click="maximize_Click" |
||||
|
Background="#FF006361" BorderBrush="#FF006361"> |
||||
|
<!-- 最大化图标:一个空心的矩形 --> |
||||
|
<Viewbox Width="50" Height="40"> |
||||
|
<Path x:Name="IconPath_maximize" Data="M4,2 H15 V15 H4 Z" Fill="White" /> |
||||
|
</Viewbox> |
||||
|
</Button> |
||||
|
<Button HorizontalAlignment="Right" Margin="0,0,125,0" VerticalAlignment="Center" Width="50" Click="minimize_Click" |
||||
|
Background="#FF006361" BorderBrush="#FF006361"> |
||||
|
<!-- 最小化图标:一条横线 --> |
||||
|
<Viewbox Width="50" Height="40"> |
||||
|
<Path x:Name="IconPath_minimize" Data="M2,2 H18 " Stroke="White" StrokeThickness="2" /> |
||||
|
</Viewbox> |
||||
|
</Button> |
||||
|
</Grid> |
||||
|
<Grid x:Name="GridMenu" Width="200" HorizontalAlignment="Left" Background="#FF00204E" Margin="0,50,0,0" > |
||||
|
<ListView ScrollViewer.HorizontalScrollBarVisibility="Disabled" Foreground="#FF1368BD" Background="{x:Null}" > |
||||
|
<ListViewItem Height="60" MouseLeftButtonUp="ListViewItem_MouseLeftButtonUp_state"> |
||||
|
<TextBlock Text="{x:Static local:ResourceLanguage.state}" VerticalAlignment="Center" Margin="20 10" Foreground="White" HorizontalAlignment="Center" FontSize="24"/> |
||||
|
</ListViewItem> |
||||
|
<ListViewItem Height="60" MouseLeftButtonUp="ListViewItem_MouseLeftButtonUp_serve"> |
||||
|
<TextBlock Text="{x:Static local:ResourceLanguage.serve}" VerticalAlignment="Center" Margin="20 10" Foreground="White" HorizontalAlignment="Center" FontSize="24"/> |
||||
|
</ListViewItem> |
||||
|
<ListViewItem Height="60" MouseLeftButtonUp="ListViewItem_MouseLeftButtonUp_log"> |
||||
|
<TextBlock Text="{x:Static local:ResourceLanguage.log}" VerticalAlignment="Center" Margin="20 10" Foreground="White" HorizontalAlignment="Center" FontSize="24"/> |
||||
|
</ListViewItem> |
||||
|
<ListViewItem Height="60" MouseLeftButtonUp="ListViewItem_MouseLeftButtonUp_machine"> |
||||
|
<TextBlock Text="{x:Static local:ResourceLanguage.machine}" VerticalAlignment="Center" Margin="20 10" Foreground="White" HorizontalAlignment="Center" FontSize="24"/> |
||||
|
</ListViewItem> |
||||
|
<ListViewItem Height="60" MouseLeftButtonUp="ListViewItem_MouseLeftButtonUp_extend"> |
||||
|
<TextBlock Text="{x:Static local:ResourceLanguage.extend}" VerticalAlignment="Center" Margin="20 10" Foreground="White" HorizontalAlignment="Center" FontSize="24"/> |
||||
|
</ListViewItem> |
||||
|
<ListViewItem Height="60" MouseLeftButtonUp="ListViewItem_MouseLeftButtonUp_user"> |
||||
|
<TextBlock Text="{x:Static local:ResourceLanguage.user}" VerticalAlignment="Center" Margin="20 10" Foreground="White" HorizontalAlignment="Center" FontSize="24"/> |
||||
|
</ListViewItem> |
||||
|
<ListViewItem Height="60" MouseLeftButtonUp="ListViewItem_MouseLeftButtonUp_Action"> |
||||
|
<TextBlock Text="{x:Static local:ResourceLanguage.Action}" VerticalAlignment="Center" Margin="20 10" Foreground="White" HorizontalAlignment="Center" FontSize="24"/> |
||||
|
</ListViewItem> |
||||
|
<ListViewItem Height="60" MouseLeftButtonUp="ListViewItem_MouseLeftButtonUp_set"> |
||||
|
<TextBlock Text="{x:Static local:ResourceLanguage.set}" VerticalAlignment="Center" Margin="20 10" Foreground="White" HorizontalAlignment="Center" FontSize="24"/> |
||||
|
</ListViewItem> |
||||
|
<ListViewItem Height="60" MouseLeftButtonUp="ListViewItem_MouseLeftButtonUp_help"> |
||||
|
<TextBlock Text="{x:Static local:ResourceLanguage.help}" VerticalAlignment="Center" Margin="20 10" Foreground="White" HorizontalAlignment="Center" FontSize="24"/> |
||||
|
</ListViewItem> |
||||
|
</ListView> |
||||
|
</Grid> |
||||
|
<Grid x:Name="state" Margin="200,50,0,0"> |
||||
|
<TextBox x:Name="LOG" TextWrapping="Wrap" Margin="5,200,5,5" FontSize="16"/> |
||||
|
</Grid> |
||||
|
<ContentControl x:Name="Picture" Margin="200,50,0,0" Visibility="Hidden"/> |
||||
|
</Grid> |
||||
|
</Window> |
||||
@ -0,0 +1,115 @@ |
|||||
|
using SunlightAggregationManager.UserClass; |
||||
|
using SunlightAggregationManager.ViewModel; |
||||
|
using System.Text; |
||||
|
using System.Windows; |
||||
|
using System.Windows.Controls; |
||||
|
using System.Windows.Data; |
||||
|
using System.Windows.Documents; |
||||
|
using System.Windows.Input; |
||||
|
using System.Windows.Media; |
||||
|
using System.Windows.Media.Imaging; |
||||
|
using System.Windows.Navigation; |
||||
|
using System.Windows.Shapes; |
||||
|
|
||||
|
namespace SunlightAggregationManager |
||||
|
{ |
||||
|
/// <summary>
|
||||
|
/// Interaction logic for MainWindow.xaml
|
||||
|
/// </summary>
|
||||
|
public partial class MainWindow : Window |
||||
|
{ |
||||
|
public MainWindow() |
||||
|
{ |
||||
|
InitializeComponent(); |
||||
|
} |
||||
|
private void Window_Loaded(object sender, RoutedEventArgs e) |
||||
|
{ |
||||
|
GlobalLogManager.RegisterTarget(LOG); |
||||
|
|
||||
|
DataContext = new MainWindowViewModel(); |
||||
|
} |
||||
|
|
||||
|
//标题栏拖动
|
||||
|
private void GridTitle_MouseDown(object sender, MouseButtonEventArgs e) |
||||
|
{ |
||||
|
if (e.ChangedButton == MouseButton.Left) |
||||
|
DragMove(); |
||||
|
} |
||||
|
|
||||
|
//最大
|
||||
|
private void maximize_Click(object sender, RoutedEventArgs e) |
||||
|
{ |
||||
|
if (WindowState == WindowState.Maximized) |
||||
|
{ WindowState = WindowState.Normal; } |
||||
|
else |
||||
|
{ WindowState = WindowState.Maximized; } |
||||
|
} |
||||
|
//最小
|
||||
|
private void minimize_Click(object sender, RoutedEventArgs e) |
||||
|
{ |
||||
|
if (WindowState == WindowState.Minimized) |
||||
|
{ WindowState = WindowState.Normal; } |
||||
|
else |
||||
|
{ WindowState = WindowState.Minimized; } |
||||
|
} |
||||
|
//关闭
|
||||
|
private void exit_Click(object sender, RoutedEventArgs e) |
||||
|
{ |
||||
|
// 关闭整个应用程序
|
||||
|
System.Windows.Application.Current.Shutdown(); |
||||
|
} |
||||
|
|
||||
|
private void ListViewItem_MouseLeftButtonUp_state(object sender, MouseButtonEventArgs e) |
||||
|
{ |
||||
|
state.Visibility = Visibility.Visible; |
||||
|
Picture.Visibility= Visibility.Collapsed; |
||||
|
} |
||||
|
private void ListViewItem_MouseLeftButtonUp_serve(object sender, MouseButtonEventArgs e) |
||||
|
{ |
||||
|
state.Visibility = Visibility.Collapsed; |
||||
|
Picture.Visibility = Visibility.Visible; |
||||
|
} |
||||
|
private void ListViewItem_MouseLeftButtonUp_log(object sender, MouseButtonEventArgs e) |
||||
|
{ |
||||
|
state.Visibility = Visibility.Collapsed; |
||||
|
Picture.Visibility = Visibility.Visible; |
||||
|
Picture.Content = new View.logPage(); |
||||
|
} |
||||
|
private void ListViewItem_MouseLeftButtonUp_machine(object sender, MouseButtonEventArgs e) |
||||
|
{ |
||||
|
state.Visibility = Visibility.Collapsed; |
||||
|
Picture.Visibility = Visibility.Visible; |
||||
|
} |
||||
|
private void ListViewItem_MouseLeftButtonUp_extend(object sender, MouseButtonEventArgs e) |
||||
|
{ |
||||
|
state.Visibility = Visibility.Collapsed; |
||||
|
Picture.Visibility = Visibility.Visible; |
||||
|
} |
||||
|
private void ListViewItem_MouseLeftButtonUp_user(object sender, MouseButtonEventArgs e) |
||||
|
{ |
||||
|
state.Visibility = Visibility.Collapsed; |
||||
|
Picture.Visibility = Visibility.Visible; |
||||
|
Picture.Content = new View.UserPage(); |
||||
|
} |
||||
|
private void ListViewItem_MouseLeftButtonUp_Action(object sender, MouseButtonEventArgs e) |
||||
|
{ |
||||
|
state.Visibility = Visibility.Collapsed; |
||||
|
Picture.Visibility = Visibility.Visible; |
||||
|
} |
||||
|
|
||||
|
private void ListViewItem_MouseLeftButtonUp_set(object sender, MouseButtonEventArgs e) |
||||
|
{ |
||||
|
state.Visibility = Visibility.Collapsed; |
||||
|
Picture.Visibility = Visibility.Visible; |
||||
|
Picture.Content = new View.SettingPage(); |
||||
|
} |
||||
|
private void ListViewItem_MouseLeftButtonUp_help(object sender, MouseButtonEventArgs e) |
||||
|
{ |
||||
|
state.Visibility = Visibility.Collapsed; |
||||
|
Picture.Visibility = Visibility.Visible; |
||||
|
Picture.Content = new View.imgQR(); |
||||
|
} |
||||
|
|
||||
|
|
||||
|
} |
||||
|
} |
||||
@ -0,0 +1,387 @@ |
|||||
|
//------------------------------------------------------------------------------
|
||||
|
// <auto-generated>
|
||||
|
// 此代码由工具生成。
|
||||
|
// 运行时版本:4.0.30319.42000
|
||||
|
//
|
||||
|
// 对此文件的更改可能会导致不正确的行为,并且如果
|
||||
|
// 重新生成代码,这些更改将会丢失。
|
||||
|
// </auto-generated>
|
||||
|
//------------------------------------------------------------------------------
|
||||
|
|
||||
|
namespace SunlightAggregationManager { |
||||
|
using System; |
||||
|
|
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 一个强类型的资源类,用于查找本地化的字符串等。
|
||||
|
/// </summary>
|
||||
|
// 此类是由 StronglyTypedResourceBuilder
|
||||
|
// 类通过类似于 ResGen 或 Visual Studio 的工具自动生成的。
|
||||
|
// 若要添加或移除成员,请编辑 .ResX 文件,然后重新运行 ResGen
|
||||
|
// (以 /str 作为命令选项),或重新生成 VS 项目。
|
||||
|
[global::System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", "18.0.0.0")] |
||||
|
[global::System.Diagnostics.DebuggerNonUserCodeAttribute()] |
||||
|
[global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()] |
||||
|
public class ResourceLanguage { |
||||
|
|
||||
|
private static global::System.Resources.ResourceManager resourceMan; |
||||
|
|
||||
|
private static global::System.Globalization.CultureInfo resourceCulture; |
||||
|
|
||||
|
[global::System.Diagnostics.CodeAnalysis.SuppressMessageAttribute("Microsoft.Performance", "CA1811:AvoidUncalledPrivateCode")] |
||||
|
internal ResourceLanguage() { |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 返回此类使用的缓存的 ResourceManager 实例。
|
||||
|
/// </summary>
|
||||
|
[global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)] |
||||
|
public static global::System.Resources.ResourceManager ResourceManager { |
||||
|
get { |
||||
|
if (object.ReferenceEquals(resourceMan, null)) { |
||||
|
global::System.Resources.ResourceManager temp = new global::System.Resources.ResourceManager("SunlightAggregationManager.ResourceLanguage", typeof(ResourceLanguage).Assembly); |
||||
|
resourceMan = temp; |
||||
|
} |
||||
|
return resourceMan; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 重写当前线程的 CurrentUICulture 属性,对
|
||||
|
/// 使用此强类型资源类的所有资源查找执行重写。
|
||||
|
/// </summary>
|
||||
|
[global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)] |
||||
|
public static global::System.Globalization.CultureInfo Culture { |
||||
|
get { |
||||
|
return resourceCulture; |
||||
|
} |
||||
|
set { |
||||
|
resourceCulture = value; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 行为 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string Action { |
||||
|
get { |
||||
|
return ResourceManager.GetString("Action", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 删除 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string Delete { |
||||
|
get { |
||||
|
return ResourceManager.GetString("Delete", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 部门 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string Department { |
||||
|
get { |
||||
|
return ResourceManager.GetString("Department", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 禁用 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string Disable { |
||||
|
get { |
||||
|
return ResourceManager.GetString("Disable", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 启用 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string Enable { |
||||
|
get { |
||||
|
return ResourceManager.GetString("Enable", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 启用HTTP 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string EnableHTTP { |
||||
|
get { |
||||
|
return ResourceManager.GetString("EnableHTTP", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 启用TCP 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string EnableTCP { |
||||
|
get { |
||||
|
return ResourceManager.GetString("EnableTCP", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 启用TLS1.2 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string EnableTLS { |
||||
|
get { |
||||
|
return ResourceManager.GetString("EnableTLS", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 错误 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string ERR { |
||||
|
get { |
||||
|
return ResourceManager.GetString("ERR", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 扩展 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string extend { |
||||
|
get { |
||||
|
return ResourceManager.GetString("extend", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 组 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string Groups { |
||||
|
get { |
||||
|
return ResourceManager.GetString("Groups", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 帮助 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string help { |
||||
|
get { |
||||
|
return ResourceManager.GetString("help", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 历史 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string history { |
||||
|
get { |
||||
|
return ResourceManager.GetString("history", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 信息 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string information { |
||||
|
get { |
||||
|
return ResourceManager.GetString("information", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 语言 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string Language { |
||||
|
get { |
||||
|
return ResourceManager.GetString("Language", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 最后行为时间 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string LastActionTime { |
||||
|
get { |
||||
|
return ResourceManager.GetString("LastActionTime", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 最后登录地址 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string LastLoginIP { |
||||
|
get { |
||||
|
return ResourceManager.GetString("LastLoginIP", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 最后登录终端 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string LastLoginTerminal { |
||||
|
get { |
||||
|
return ResourceManager.GetString("LastLoginTerminal", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 最后登录时间 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string LastLoginTime { |
||||
|
get { |
||||
|
return ResourceManager.GetString("LastLoginTime", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 最后下线时间 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string LastOfflineTime { |
||||
|
get { |
||||
|
return ResourceManager.GetString("LastOfflineTime", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 日志 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string log { |
||||
|
get { |
||||
|
return ResourceManager.GetString("log", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 设备 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string machine { |
||||
|
get { |
||||
|
return ResourceManager.GetString("machine", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 模式 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string Mode { |
||||
|
get { |
||||
|
return ResourceManager.GetString("Mode", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 姓名 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string Name { |
||||
|
get { |
||||
|
return ResourceManager.GetString("Name", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 离线 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string Offine { |
||||
|
get { |
||||
|
return ResourceManager.GetString("Offine", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 在线 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string Online { |
||||
|
get { |
||||
|
return ResourceManager.GetString("Online", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 密码 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string Password { |
||||
|
get { |
||||
|
return ResourceManager.GetString("Password", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 查询 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string query { |
||||
|
get { |
||||
|
return ResourceManager.GetString("query", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 保存 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string save { |
||||
|
get { |
||||
|
return ResourceManager.GetString("save", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 服务 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string serve { |
||||
|
get { |
||||
|
return ResourceManager.GetString("serve", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 服务器 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string SERVER { |
||||
|
get { |
||||
|
return ResourceManager.GetString("SERVER", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 设置 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string set { |
||||
|
get { |
||||
|
return ResourceManager.GetString("set", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 数据库 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string SQL { |
||||
|
get { |
||||
|
return ResourceManager.GetString("SQL", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 状态 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string state { |
||||
|
get { |
||||
|
return ResourceManager.GetString("state", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 测试 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string Test { |
||||
|
get { |
||||
|
return ResourceManager.GetString("Test", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 查找类似 用户 的本地化字符串。
|
||||
|
/// </summary>
|
||||
|
public static string user { |
||||
|
get { |
||||
|
return ResourceManager.GetString("user", resourceCulture); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
} |
||||
@ -0,0 +1,228 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<root> |
||||
|
<!-- |
||||
|
Microsoft ResX Schema |
||||
|
|
||||
|
Version 2.0 |
||||
|
|
||||
|
The primary goals of this format is to allow a simple XML format |
||||
|
that is mostly human readable. The generation and parsing of the |
||||
|
various data types are done through the TypeConverter classes |
||||
|
associated with the data types. |
||||
|
|
||||
|
Example: |
||||
|
|
||||
|
... ado.net/XML headers & schema ... |
||||
|
<resheader name="resmimetype">text/microsoft-resx</resheader> |
||||
|
<resheader name="version">2.0</resheader> |
||||
|
<resheader name="reader">System.Resources.ResXResourceReader, System.Windows.Forms, ...</resheader> |
||||
|
<resheader name="writer">System.Resources.ResXResourceWriter, System.Windows.Forms, ...</resheader> |
||||
|
<data name="Name1"><value>this is my long string</value><comment>this is a comment</comment></data> |
||||
|
<data name="Color1" type="System.Drawing.Color, System.Drawing">Blue</data> |
||||
|
<data name="Bitmap1" mimetype="application/x-microsoft.net.object.binary.base64"> |
||||
|
<value>[base64 mime encoded serialized .NET Framework object]</value> |
||||
|
</data> |
||||
|
<data name="Icon1" type="System.Drawing.Icon, System.Drawing" mimetype="application/x-microsoft.net.object.bytearray.base64"> |
||||
|
<value>[base64 mime encoded string representing a byte array form of the .NET Framework object]</value> |
||||
|
<comment>This is a comment</comment> |
||||
|
</data> |
||||
|
|
||||
|
There are any number of "resheader" rows that contain simple |
||||
|
name/value pairs. |
||||
|
|
||||
|
Each data row contains a name, and value. The row also contains a |
||||
|
type or mimetype. Type corresponds to a .NET class that support |
||||
|
text/value conversion through the TypeConverter architecture. |
||||
|
Classes that don't support this are serialized and stored with the |
||||
|
mimetype set. |
||||
|
|
||||
|
The mimetype is used for serialized objects, and tells the |
||||
|
ResXResourceReader how to depersist the object. This is currently not |
||||
|
extensible. For a given mimetype the value must be set accordingly: |
||||
|
|
||||
|
Note - application/x-microsoft.net.object.binary.base64 is the format |
||||
|
that the ResXResourceWriter will generate, however the reader can |
||||
|
read any of the formats listed below. |
||||
|
|
||||
|
mimetype: application/x-microsoft.net.object.binary.base64 |
||||
|
value : The object must be serialized with |
||||
|
: System.Runtime.Serialization.Formatters.Binary.BinaryFormatter |
||||
|
: and then encoded with base64 encoding. |
||||
|
|
||||
|
mimetype: application/x-microsoft.net.object.soap.base64 |
||||
|
value : The object must be serialized with |
||||
|
: System.Runtime.Serialization.Formatters.Soap.SoapFormatter |
||||
|
: and then encoded with base64 encoding. |
||||
|
|
||||
|
mimetype: application/x-microsoft.net.object.bytearray.base64 |
||||
|
value : The object must be serialized into a byte array |
||||
|
: using a System.ComponentModel.TypeConverter |
||||
|
: and then encoded with base64 encoding. |
||||
|
--> |
||||
|
<xsd:schema id="root" xmlns="" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:msdata="urn:schemas-microsoft-com:xml-msdata"> |
||||
|
<xsd:import namespace="http://www.w3.org/XML/1998/namespace" /> |
||||
|
<xsd:element name="root" msdata:IsDataSet="true"> |
||||
|
<xsd:complexType> |
||||
|
<xsd:choice maxOccurs="unbounded"> |
||||
|
<xsd:element name="metadata"> |
||||
|
<xsd:complexType> |
||||
|
<xsd:sequence> |
||||
|
<xsd:element name="value" type="xsd:string" minOccurs="0" /> |
||||
|
</xsd:sequence> |
||||
|
<xsd:attribute name="name" use="required" type="xsd:string" /> |
||||
|
<xsd:attribute name="type" type="xsd:string" /> |
||||
|
<xsd:attribute name="mimetype" type="xsd:string" /> |
||||
|
<xsd:attribute ref="xml:space" /> |
||||
|
</xsd:complexType> |
||||
|
</xsd:element> |
||||
|
<xsd:element name="assembly"> |
||||
|
<xsd:complexType> |
||||
|
<xsd:attribute name="alias" type="xsd:string" /> |
||||
|
<xsd:attribute name="name" type="xsd:string" /> |
||||
|
</xsd:complexType> |
||||
|
</xsd:element> |
||||
|
<xsd:element name="data"> |
||||
|
<xsd:complexType> |
||||
|
<xsd:sequence> |
||||
|
<xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" /> |
||||
|
<xsd:element name="comment" type="xsd:string" minOccurs="0" msdata:Ordinal="2" /> |
||||
|
</xsd:sequence> |
||||
|
<xsd:attribute name="name" type="xsd:string" use="required" msdata:Ordinal="1" /> |
||||
|
<xsd:attribute name="type" type="xsd:string" msdata:Ordinal="3" /> |
||||
|
<xsd:attribute name="mimetype" type="xsd:string" msdata:Ordinal="4" /> |
||||
|
<xsd:attribute ref="xml:space" /> |
||||
|
</xsd:complexType> |
||||
|
</xsd:element> |
||||
|
<xsd:element name="resheader"> |
||||
|
<xsd:complexType> |
||||
|
<xsd:sequence> |
||||
|
<xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" /> |
||||
|
</xsd:sequence> |
||||
|
<xsd:attribute name="name" type="xsd:string" use="required" /> |
||||
|
</xsd:complexType> |
||||
|
</xsd:element> |
||||
|
</xsd:choice> |
||||
|
</xsd:complexType> |
||||
|
</xsd:element> |
||||
|
</xsd:schema> |
||||
|
<resheader name="resmimetype"> |
||||
|
<value>text/microsoft-resx</value> |
||||
|
</resheader> |
||||
|
<resheader name="version"> |
||||
|
<value>2.0</value> |
||||
|
</resheader> |
||||
|
<resheader name="reader"> |
||||
|
<value>System.Resources.ResXResourceReader, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value> |
||||
|
</resheader> |
||||
|
<resheader name="writer"> |
||||
|
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value> |
||||
|
</resheader> |
||||
|
<data name="Action" xml:space="preserve"> |
||||
|
<value>行为</value> |
||||
|
</data> |
||||
|
<data name="Delete" xml:space="preserve"> |
||||
|
<value>删除</value> |
||||
|
</data> |
||||
|
<data name="Department" xml:space="preserve"> |
||||
|
<value>部门</value> |
||||
|
</data> |
||||
|
<data name="Disable" xml:space="preserve"> |
||||
|
<value>禁用</value> |
||||
|
</data> |
||||
|
<data name="Enable" xml:space="preserve"> |
||||
|
<value>启用</value> |
||||
|
</data> |
||||
|
<data name="EnableHTTP" xml:space="preserve"> |
||||
|
<value>启用HTTP</value> |
||||
|
</data> |
||||
|
<data name="EnableTCP" xml:space="preserve"> |
||||
|
<value>启用TCP</value> |
||||
|
</data> |
||||
|
<data name="EnableTLS" xml:space="preserve"> |
||||
|
<value>启用TLS1.2</value> |
||||
|
</data> |
||||
|
<data name="ERR" xml:space="preserve"> |
||||
|
<value>错误</value> |
||||
|
</data> |
||||
|
<data name="extend" xml:space="preserve"> |
||||
|
<value>扩展</value> |
||||
|
</data> |
||||
|
<data name="Groups" xml:space="preserve"> |
||||
|
<value>组</value> |
||||
|
</data> |
||||
|
<data name="help" xml:space="preserve"> |
||||
|
<value>帮助</value> |
||||
|
</data> |
||||
|
<data name="history" xml:space="preserve"> |
||||
|
<value>历史</value> |
||||
|
</data> |
||||
|
<data name="information" xml:space="preserve"> |
||||
|
<value>信息</value> |
||||
|
</data> |
||||
|
<data name="Language" xml:space="preserve"> |
||||
|
<value>语言</value> |
||||
|
</data> |
||||
|
<data name="LastActionTime" xml:space="preserve"> |
||||
|
<value>最后行为时间</value> |
||||
|
</data> |
||||
|
<data name="LastLoginIP" xml:space="preserve"> |
||||
|
<value>最后登录地址</value> |
||||
|
</data> |
||||
|
<data name="LastLoginTerminal" xml:space="preserve"> |
||||
|
<value>最后登录终端</value> |
||||
|
</data> |
||||
|
<data name="LastLoginTime" xml:space="preserve"> |
||||
|
<value>最后登录时间</value> |
||||
|
</data> |
||||
|
<data name="LastOfflineTime" xml:space="preserve"> |
||||
|
<value>最后下线时间</value> |
||||
|
</data> |
||||
|
<data name="log" xml:space="preserve"> |
||||
|
<value>日志</value> |
||||
|
</data> |
||||
|
<data name="machine" xml:space="preserve"> |
||||
|
<value>设备</value> |
||||
|
</data> |
||||
|
<data name="Mode" xml:space="preserve"> |
||||
|
<value>模式</value> |
||||
|
</data> |
||||
|
<data name="Name" xml:space="preserve"> |
||||
|
<value>姓名</value> |
||||
|
</data> |
||||
|
<data name="Offine" xml:space="preserve"> |
||||
|
<value>离线</value> |
||||
|
</data> |
||||
|
<data name="Online" xml:space="preserve"> |
||||
|
<value>在线</value> |
||||
|
</data> |
||||
|
<data name="Password" xml:space="preserve"> |
||||
|
<value>密码</value> |
||||
|
</data> |
||||
|
<data name="query" xml:space="preserve"> |
||||
|
<value>查询</value> |
||||
|
</data> |
||||
|
<data name="save" xml:space="preserve"> |
||||
|
<value>保存</value> |
||||
|
</data> |
||||
|
<data name="serve" xml:space="preserve"> |
||||
|
<value>服务</value> |
||||
|
</data> |
||||
|
<data name="SERVER" xml:space="preserve"> |
||||
|
<value>服务器</value> |
||||
|
</data> |
||||
|
<data name="set" xml:space="preserve"> |
||||
|
<value>设置</value> |
||||
|
</data> |
||||
|
<data name="SQL" xml:space="preserve"> |
||||
|
<value>数据库</value> |
||||
|
</data> |
||||
|
<data name="state" xml:space="preserve"> |
||||
|
<value>状态</value> |
||||
|
</data> |
||||
|
<data name="Test" xml:space="preserve"> |
||||
|
<value>测试</value> |
||||
|
</data> |
||||
|
<data name="user" xml:space="preserve"> |
||||
|
<value>用户</value> |
||||
|
</data> |
||||
|
</root> |
||||
@ -0,0 +1,39 @@ |
|||||
|
<Project Sdk="Microsoft.NET.Sdk"> |
||||
|
|
||||
|
<PropertyGroup> |
||||
|
<OutputType>WinExe</OutputType> |
||||
|
<TargetFramework>net10.0-windows</TargetFramework> |
||||
|
<Nullable>enable</Nullable> |
||||
|
<ImplicitUsings>enable</ImplicitUsings> |
||||
|
<UseWPF>true</UseWPF> |
||||
|
</PropertyGroup> |
||||
|
|
||||
|
<ItemGroup> |
||||
|
<PackageReference Include="CommunityToolkit.Mvvm" Version="8.4.2" /> |
||||
|
<PackageReference Include="Microsoft.Data.SqlClient" Version="7.0.0" /> |
||||
|
<PackageReference Include="Microsoft.Data.Sqlite" Version="10.0.5" /> |
||||
|
<PackageReference Include="Net.Codecrete.QrCodeGenerator" Version="2.1.0" /> |
||||
|
<PackageReference Include="System.Data.SqlClient" Version="4.9.1" /> |
||||
|
<PackageReference Include="TouchSocket" Version="4.2.3" /> |
||||
|
<PackageReference Include="vocaluxe.dependencies.netfwtypelib" Version="1.0.1" /> |
||||
|
</ItemGroup> |
||||
|
|
||||
|
<ItemGroup> |
||||
|
<Compile Update="ResourceLanguage.Designer.cs"> |
||||
|
<DesignTime>True</DesignTime> |
||||
|
<AutoGen>True</AutoGen> |
||||
|
<DependentUpon>ResourceLanguage.resx</DependentUpon> |
||||
|
</Compile> |
||||
|
<Compile Update="View\imgQR.xaml.cs"> |
||||
|
<SubType>Code</SubType> |
||||
|
</Compile> |
||||
|
</ItemGroup> |
||||
|
|
||||
|
<ItemGroup> |
||||
|
<EmbeddedResource Update="ResourceLanguage.resx"> |
||||
|
<Generator>PublicResXFileCodeGenerator</Generator> |
||||
|
<LastGenOutput>ResourceLanguage.Designer.cs</LastGenOutput> |
||||
|
</EmbeddedResource> |
||||
|
</ItemGroup> |
||||
|
|
||||
|
</Project> |
||||
@ -0,0 +1,3 @@ |
|||||
|
<Solution> |
||||
|
<Project Path="SunlightAggregationManager.csproj" /> |
||||
|
</Solution> |
||||
@ -0,0 +1,209 @@ |
|||||
|
using System; |
||||
|
using System.Collections.Generic; |
||||
|
using System.Text.Json; |
||||
|
using System.Text; |
||||
|
using TouchSocket.Core; |
||||
|
using TouchSocket.Sockets; |
||||
|
using SunlightAggregationManager.ViewModel; |
||||
|
using System.Security.Cryptography.Pkcs; |
||||
|
|
||||
|
namespace SunlightAggregationManager.UserClass |
||||
|
{ |
||||
|
public class AsyncTcpServer |
||||
|
{ |
||||
|
public class Person |
||||
|
{ |
||||
|
public required string User { get; set; } |
||||
|
public required string Password { get; set; } |
||||
|
public required string IP { get; set; } |
||||
|
public required string Terminal { get; set; } |
||||
|
public required string Dat { get; set; } |
||||
|
} |
||||
|
public static TcpService service = new TcpService(); |
||||
|
public static async Task TcpMain(int port1 ,int port2) |
||||
|
{ |
||||
|
NetFwManger.AllowPort(port1, "TCP");//开放7790端口
|
||||
|
NetFwManger.AllowPort(port2, "TCP");//开放7790端口
|
||||
|
//TcpService service = new TcpService();
|
||||
|
service.Connecting = (client, e) => { return EasyTask.CompletedTask; };//有客户端正在连接
|
||||
|
service.Connected = (client, e) => {return EasyTask.CompletedTask;};//有客户端成功连接
|
||||
|
service.Closing = (client, e) => { return EasyTask.CompletedTask; };//有客户端正在断开连接,只有当主动断开时才有效。
|
||||
|
service.Closed = (client, e) => { |
||||
|
//离线记录
|
||||
|
Dictionary<string, object> myDictOff = new Dictionary<string, object>(); |
||||
|
myDictOff.Add("State", 15); |
||||
|
myDictOff.Add("LastActionTime", DateTime.Now); |
||||
|
myDictOff.Add("LastOfflineTime", DateTime.Now); |
||||
|
myDictOff.Add("Note", "Offline"); |
||||
|
MainWindowViewModel.Updata_Memory(MainWindowViewModel._userData, "LinkID='" + client.Id + "'", myDictOff); |
||||
|
Console.WriteLine("[Offline]IP[" + client.IP + "]ID["+ client.Id + "]"); |
||||
|
return EasyTask.CompletedTask; |
||||
|
};//有客户端断开连接
|
||||
|
service.Received = (client, e) =>{ |
||||
|
string DAT = e.Memory.Span.ToString(Encoding.UTF8); |
||||
|
try |
||||
|
{ |
||||
|
Person? deserializedPerson = JsonSerializer.Deserialize<Person>(DAT); |
||||
|
|
||||
|
var _user = MainWindowViewModel.Selet_Memory(MainWindowViewModel._userData, "State", "User='" + deserializedPerson?.User + |
||||
|
"' and Password ='" + deserializedPerson?.Password + "'"); |
||||
|
if (_user != null) |
||||
|
{ |
||||
|
string _dat = _user?.ToString() ?? "0"; |
||||
|
if (_dat != "99") |
||||
|
{ |
||||
|
if (_dat == "20") |
||||
|
{//行为
|
||||
|
Dictionary<string, object> myDict = new Dictionary<string, object>(); |
||||
|
myDict.Add("LinkID", client.Id); |
||||
|
myDict.Add("LastLoginIP", deserializedPerson?.IP ?? client.IP); |
||||
|
myDict.Add("LastActionTime", DateTime.Now); |
||||
|
myDict.Add("Note", deserializedPerson?.Dat ?? "unknown"); |
||||
|
MainWindowViewModel.Updata_Memory(MainWindowViewModel._userData, "User='" + deserializedPerson?.User + "'", myDict); |
||||
|
|
||||
|
Console.WriteLine("[Log on]User[" + deserializedPerson?.User + "]Note[" + (deserializedPerson?.Dat ?? "unknown") + "]"); |
||||
|
} |
||||
|
else |
||||
|
{//首次登录记录
|
||||
|
Dictionary<string, object> myDict = new Dictionary<string, object>(); |
||||
|
myDict.Add("State", 20); |
||||
|
myDict.Add("LinkID", client.Id); |
||||
|
myDict.Add("LastLoginIP", deserializedPerson?.IP ?? client.IP); |
||||
|
myDict.Add("LastLoginTime", DateTime.Now); |
||||
|
myDict.Add("LastLoginTerminal", deserializedPerson?.Terminal ?? "unknown device"); |
||||
|
myDict.Add("LastActionTime", DateTime.Now); |
||||
|
myDict.Add("Note", deserializedPerson?.Dat ?? "unknown"); |
||||
|
MainWindowViewModel.Updata_Memory(MainWindowViewModel._userData, "User='" + deserializedPerson?.User + "'", myDict); |
||||
|
|
||||
|
Console.WriteLine("[Log on]User[" + deserializedPerson?.User + "]IP[" + deserializedPerson?.IP ?? client.IP + |
||||
|
"]ID[" + client.Id + "]Terminal[" + (deserializedPerson?.Terminal ?? "unknown device") + "]"); |
||||
|
} |
||||
|
} |
||||
|
else |
||||
|
{//停用账号关闭连接
|
||||
|
client.SendAsync("Account Deactivated"); |
||||
|
foreach (var item in AsyncTcpServer.service.Clients) |
||||
|
{ |
||||
|
if (item.Id == client.Id) |
||||
|
{ |
||||
|
item.CloseAsync(); |
||||
|
|
||||
|
//在断开连接后,释放对象。
|
||||
|
item.Dispose(); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
else |
||||
|
{ //无效账号关闭连接
|
||||
|
client.SendAsync("Invalid Account"); |
||||
|
foreach (var item in AsyncTcpServer.service.Clients) |
||||
|
{ |
||||
|
if (item.Id == client.Id) |
||||
|
{ |
||||
|
item.CloseAsync(); |
||||
|
item.Dispose(); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
} |
||||
|
catch (Exception ex) |
||||
|
{//错误
|
||||
|
Console.WriteLine("Tcp:" + ex.ToString()); |
||||
|
client.SendAsync("Target instruction parsing error manager will close connection"); |
||||
|
foreach (var item in AsyncTcpServer.service.Clients) |
||||
|
{ |
||||
|
if (item.Id == client.Id) |
||||
|
{ |
||||
|
item.CloseAsync(); |
||||
|
item.Dispose(); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
//string DAT = e.ByteBlock.Span.ToString(Encoding.UTF8).Substring(5);
|
||||
|
//string SYSDAT = "";// = e.ByteBlock.Span.ToString(Encoding.ASCII).Substring(5);
|
||||
|
// string SYSKEY = "";
|
||||
|
// if (DAT.Length >= 16) SYSKEY = DAT.Substring(0, 16);
|
||||
|
// if (DAT.Length > 16) SYSDAT = DAT.Substring(16);
|
||||
|
|
||||
|
|
||||
|
return EasyTask.CompletedTask; |
||||
|
}; |
||||
|
|
||||
|
await service.SetupAsync(new TouchSocketConfig()//载入配置
|
||||
|
// .SetMaxBufferSize(1024 * 1024 * 100)
|
||||
|
//.SetMinBufferSize(1024 * 1024)
|
||||
|
.SetListenIPHosts(new IPHost[] { new IPHost(port1), new IPHost(port2) })//同时监听两个地址
|
||||
|
.ConfigureContainer(a =>//容器的配置顺序应该在最前面
|
||||
|
{ |
||||
|
}) |
||||
|
.ConfigurePlugins(a => |
||||
|
{ |
||||
|
}) |
||||
|
); |
||||
|
await service.StartAsync();//启动
|
||||
|
|
||||
|
Console.WriteLine("TCP Port:"+ port1.ToString()); |
||||
|
Console.WriteLine("TCP Port:"+ port2.ToString()); |
||||
|
Console.WriteLine("TCP SERVER:START"); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
class MyTcpService : TcpService<MyTcpSessionClient> |
||||
|
{ |
||||
|
protected override MyTcpSessionClient NewClient() |
||||
|
{ |
||||
|
return new MyTcpSessionClient(); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
class MyTcpSessionClient : TcpSessionClient |
||||
|
{ |
||||
|
internal void SetDataHandlingAdapter(SingleStreamDataHandlingAdapter adapter) |
||||
|
{ |
||||
|
base.SetAdapter(adapter); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 此插件实现,按照不同端口,使用不同适配器。
|
||||
|
/// <list type="bullet">
|
||||
|
/// <item>7789端口:使用"**"结尾的数据</item>
|
||||
|
/// <item>7790端口:使用"##"结尾的数据</item>
|
||||
|
/// </list>
|
||||
|
/// </summary>
|
||||
|
internal class DifferentProtocolPlugin : PluginBase, ITcpConnectingPlugin, ITcpReceivedPlugin |
||||
|
{ |
||||
|
public async Task OnTcpConnecting(ITcpSession client, ConnectingEventArgs e) |
||||
|
{ |
||||
|
if (client is MyTcpSessionClient sessionClient) |
||||
|
{ |
||||
|
if (sessionClient.ServicePort == 7789) |
||||
|
{ |
||||
|
sessionClient.SetDataHandlingAdapter(new TerminatorPackageAdapter("**")); |
||||
|
} |
||||
|
else |
||||
|
{ |
||||
|
sessionClient.SetDataHandlingAdapter(new TerminatorPackageAdapter("##")); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
await e.InvokeNext(); |
||||
|
} |
||||
|
|
||||
|
public async Task OnTcpReceived(ITcpSession client, ReceivedDataEventArgs e) |
||||
|
{ |
||||
|
//如果是自定义适配器,此处解析时,可以判断e.RequestInfo的类型
|
||||
|
|
||||
|
if (client is ITcpSessionClient sessionClient) |
||||
|
{ |
||||
|
sessionClient.Logger.Info($"{sessionClient.GetIPPort()}收到数据,服务器端口:{sessionClient.ServicePort},数据:{e.Memory.Span.ToString(Encoding.UTF8)}"); |
||||
|
} |
||||
|
|
||||
|
await e.InvokeNext(); |
||||
|
} |
||||
|
|
||||
|
} |
||||
|
} |
||||
@ -0,0 +1,79 @@ |
|||||
|
using CommunityToolkit.Mvvm.Input; |
||||
|
using Microsoft.Data.SqlClient; |
||||
|
using System; |
||||
|
using System.Collections.Generic; |
||||
|
using System.Text; |
||||
|
using System.Windows.Controls; |
||||
|
using System.Windows.Input; |
||||
|
using TouchSocket.Sockets; |
||||
|
|
||||
|
namespace SunlightAggregationManager.UserClass |
||||
|
{ |
||||
|
public class DataBase |
||||
|
{ |
||||
|
private string DB_NAME = null!; |
||||
|
private string DB_TYPE = null!; |
||||
|
private string DB_ID = null!; |
||||
|
private string DB_IP = null!; |
||||
|
private string DB_PASSWORD = null!; |
||||
|
|
||||
|
|
||||
|
public void Config(string ip, string name, string type, string id, string password) |
||||
|
{ |
||||
|
DB_IP = ip; |
||||
|
DB_ID = id; |
||||
|
DB_NAME = name; |
||||
|
DB_TYPE = type; |
||||
|
DB_PASSWORD = password; |
||||
|
|
||||
|
_= ConfigAsync(); |
||||
|
|
||||
|
} |
||||
|
public async Task ConfigAsync() |
||||
|
{ |
||||
|
await Task.Delay(3000); |
||||
|
//进入后台线程操作
|
||||
|
await Task.Run(() => |
||||
|
{ |
||||
|
try |
||||
|
{ |
||||
|
// 构建连接字符串
|
||||
|
var builder = new SqlConnectionStringBuilder |
||||
|
{ |
||||
|
DataSource = DB_IP, |
||||
|
InitialCatalog = DB_NAME, |
||||
|
IntegratedSecurity = DB_TYPE == "Windows Authentication", |
||||
|
TrustServerCertificate = true, |
||||
|
ConnectTimeout = 5 |
||||
|
}; |
||||
|
|
||||
|
if (!builder.IntegratedSecurity) |
||||
|
{ |
||||
|
builder.UserID = DB_ID; |
||||
|
builder.Password = DB_PASSWORD; |
||||
|
} |
||||
|
|
||||
|
//连接数据库
|
||||
|
using var conn_SC = new SqlConnection(builder.ConnectionString); |
||||
|
conn_SC.Open(); |
||||
|
using (var cmd = new SqlCommand("SELECT 1", conn_SC)) |
||||
|
{ |
||||
|
cmd.ExecuteReader(); |
||||
|
} |
||||
|
} |
||||
|
catch (Exception ex) |
||||
|
{ |
||||
|
Console.WriteLine("DataBase Link timeout\n"+ex.ToString()); |
||||
|
// 失败处理
|
||||
|
return; // 直接返回,不执行后续代码
|
||||
|
} |
||||
|
Console.WriteLine("DataBase Link succeed"); |
||||
|
}); |
||||
|
} |
||||
|
|
||||
|
public void Database(string dat) |
||||
|
{ |
||||
|
|
||||
|
} |
||||
|
} |
||||
|
} |
||||
@ -0,0 +1,83 @@ |
|||||
|
using System; |
||||
|
using System.Collections.Generic; |
||||
|
using System.IO; |
||||
|
using System.Runtime.InteropServices; |
||||
|
using System.Text; |
||||
|
using System.Windows.Input; |
||||
|
|
||||
|
namespace SunlightAggregationManager.UserClass |
||||
|
{ |
||||
|
internal class IniFile |
||||
|
{ |
||||
|
public class IniFiles |
||||
|
{ |
||||
|
public string path; |
||||
|
[DllImport("kernel32")] //返回0表示失败,非0为成功
|
||||
|
private static extern long WritePrivateProfileString(string section, string key, string val, string filePath); |
||||
|
[DllImport("kernel32")] //返回取得字符串缓冲区的长度
|
||||
|
private static extern int GetPrivateProfileString(string section, string key, string def, StringBuilder retVal, int size, string filePath); |
||||
|
/// <summary>
|
||||
|
/// 保存ini文件的路径
|
||||
|
/// 调用示例:var ini = IniFiles("C:\file.ini");
|
||||
|
/// </summary>
|
||||
|
/// <param name="INIPath"></param>
|
||||
|
public IniFiles(string iniPath) |
||||
|
{ |
||||
|
this.path = iniPath; |
||||
|
} |
||||
|
/// <summary>
|
||||
|
/// 写Ini文件
|
||||
|
/// 调用示例:ini.IniWritevalue("Server","name","localhost");
|
||||
|
/// </summary>
|
||||
|
/// <param name="Section">[缓冲区]</param>
|
||||
|
/// <param name="Key">键</param>
|
||||
|
/// <param name="value">值</param>
|
||||
|
public void IniWritevalue(string Section, string Key, string? value) |
||||
|
{ |
||||
|
// 确保路径有效
|
||||
|
if (!File.Exists(path)) |
||||
|
{ |
||||
|
// 如果文件不存在,创建一个空的
|
||||
|
using (FileStream fs = File.Create(path)) { } |
||||
|
} |
||||
|
|
||||
|
// 处理 null 值,避免 API 调用出现问题
|
||||
|
string safeValue = value ?? string.Empty; |
||||
|
|
||||
|
long result = WritePrivateProfileString(Section, Key, safeValue, path); |
||||
|
|
||||
|
if (result==0) |
||||
|
{ |
||||
|
// 可以选择抛出异常或记录日志
|
||||
|
// throw new Win32Exception(Marshal.GetLastWin32Error());
|
||||
|
} |
||||
|
} |
||||
|
/// <summary>
|
||||
|
/// 读Ini文件
|
||||
|
/// 调用示例:ini.IniWritevalue("Server","name");
|
||||
|
/// </summary>
|
||||
|
/// <param name="Section">[缓冲区]</param>
|
||||
|
/// <param name="Key">键</param>
|
||||
|
/// <returns>值</returns>
|
||||
|
public string IniReadvalue(string Section, string Key) |
||||
|
{ |
||||
|
try |
||||
|
{ |
||||
|
if (!File.Exists(path)) |
||||
|
{ |
||||
|
return string.Empty; |
||||
|
} |
||||
|
|
||||
|
var sb = new StringBuilder(4096); |
||||
|
GetPrivateProfileString(Section, Key, string.Empty, sb, sb.Capacity, path); |
||||
|
return sb.ToString(); |
||||
|
} |
||||
|
catch (Exception) |
||||
|
{//错误返回0
|
||||
|
return "0"; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
} |
||||
|
} |
||||
|
} |
||||
@ -0,0 +1,75 @@ |
|||||
|
using NetFwTypeLib; |
||||
|
using System; |
||||
|
using System.Collections.Generic; |
||||
|
using System.Runtime.InteropServices; |
||||
|
using System.Text; |
||||
|
|
||||
|
namespace SunlightAggregationManager.UserClass |
||||
|
{ |
||||
|
public class NetFwManger |
||||
|
{ |
||||
|
private static string GetName(int port, string protocol) |
||||
|
{ |
||||
|
return "port-" + protocol + "-" + port; |
||||
|
} |
||||
|
|
||||
|
public static void AllowPort(int port, string protocol) |
||||
|
{ |
||||
|
|
||||
|
DelPort(port, protocol); |
||||
|
//创建一个INetFwRule对象
|
||||
|
Type type = Type.GetTypeFromProgID("HNetCfg.FwRule") ?? throw new |
||||
|
InvalidOperationException("Failed to retrieve HNetCfg.FwRule type. Ensure the COM component is registered."); |
||||
|
INetFwRule? rule = (INetFwRule?)Activator.CreateInstance(type); |
||||
|
|
||||
|
//设置规则的属性
|
||||
|
rule?.Action = NET_FW_ACTION_.NET_FW_ACTION_ALLOW; //允许连接
|
||||
|
rule?.Direction = NET_FW_RULE_DIRECTION_.NET_FW_RULE_DIR_IN; //入站规则
|
||||
|
rule?.Enabled = true; //启用规则
|
||||
|
rule?.InterfaceTypes = "All"; //适用于所有网络接口
|
||||
|
rule?.Name = GetName(port, protocol); //规则名称
|
||||
|
if (protocol.ToLower() == "tcp") |
||||
|
{ |
||||
|
rule?.Protocol = (int)NET_FW_IP_PROTOCOL_.NET_FW_IP_PROTOCOL_TCP; //TCP协议
|
||||
|
} |
||||
|
else |
||||
|
{ |
||||
|
rule?.Protocol = (int)NET_FW_IP_PROTOCOL_.NET_FW_IP_PROTOCOL_UDP; //UDP协议
|
||||
|
} |
||||
|
|
||||
|
rule?.LocalPorts = "" + port; //本地端口号
|
||||
|
|
||||
|
//获取FirewallPolicy对象
|
||||
|
Type policyType = Type.GetTypeFromProgID("HNetCfg.FwPolicy2") ?? throw new |
||||
|
InvalidOperationException("Failed to retrieve HNetCfg.FwRule type. Ensure the COM component is registered."); |
||||
|
INetFwPolicy2? policy = (INetFwPolicy2?)Activator.CreateInstance(policyType); |
||||
|
|
||||
|
//将规则添加到防火墙策略中
|
||||
|
policy?.Rules.Add(rule); |
||||
|
} |
||||
|
|
||||
|
public static void DelPort(int port, string protocol) |
||||
|
{ |
||||
|
//获取FirewallPolicy对象
|
||||
|
Type policyType = Type.GetTypeFromProgID("HNetCfg.FwPolicy2") ?? throw new |
||||
|
InvalidOperationException("Failed to retrieve HNetCfg.FwRule type. Ensure the COM component is registered."); |
||||
|
INetFwPolicy2? policy = (INetFwPolicy2?)Activator.CreateInstance(policyType); |
||||
|
|
||||
|
//获取现有的规则集合
|
||||
|
INetFwRules? rules = policy?.Rules; |
||||
|
if (rules != null) |
||||
|
{ |
||||
|
//查找名称的规则并删除它
|
||||
|
foreach (INetFwRule rule in rules) |
||||
|
{ |
||||
|
if (rule.Name == GetName(port, protocol)) |
||||
|
{ |
||||
|
rules.Remove(rule.Name); |
||||
|
Console.WriteLine(@"Firewall rule deleted successfully."); |
||||
|
break; |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
} |
||||
@ -0,0 +1,510 @@ |
|||||
|
using Microsoft.Data.Sqlite; |
||||
|
using System; |
||||
|
using System.Collections.Generic; |
||||
|
using System.Data; |
||||
|
using System.Data.Common; |
||||
|
using System.Drawing; |
||||
|
using System.IO; |
||||
|
using System.Linq; |
||||
|
using System.Text; |
||||
|
using System.Threading.Tasks; |
||||
|
using System.Windows.Media.Animation; |
||||
|
using System.Windows.Shapes; |
||||
|
|
||||
|
|
||||
|
namespace SunlightAggregationManager.UserClass |
||||
|
{ |
||||
|
|
||||
|
// 辅助锁类 (代码中引用了 ClsLock,这里提供一个简单的实现或占位)
|
||||
|
// 请根据实际情况替换为你项目中的 ClsLock,或使用 ReaderWriterLockSlim
|
||||
|
public class ClsLock : IDisposable |
||||
|
{ |
||||
|
private readonly ReaderWriterLockSlim _lock = new ReaderWriterLockSlim(); |
||||
|
|
||||
|
public IDisposable Read() |
||||
|
{ |
||||
|
_lock.EnterReadLock(); |
||||
|
return new DisposableAction(() => _lock.ExitReadLock()); |
||||
|
} |
||||
|
|
||||
|
public void Dispose() => _lock?.Dispose(); |
||||
|
|
||||
|
private class DisposableAction : IDisposable |
||||
|
{ |
||||
|
private readonly Action _action; |
||||
|
public DisposableAction(Action action) => _action = action; |
||||
|
public void Dispose() => _action?.Invoke(); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
public class SqliteHelper |
||||
|
{ |
||||
|
#region 字段
|
||||
|
private SqliteTransaction? dbTrans = null!; |
||||
|
private static readonly Dictionary<string, ClsLock> RWL = new Dictionary<string, ClsLock>(); |
||||
|
private readonly string mDataFile; |
||||
|
private readonly string mPassWord = ""; |
||||
|
private readonly string lockName = ""; |
||||
|
private SqliteConnection mConn = null!; |
||||
|
#endregion
|
||||
|
|
||||
|
#region 构造函数
|
||||
|
public SqliteHelper(string dataFile) |
||||
|
{ |
||||
|
this.mDataFile = dataFile ?? throw new ArgumentNullException(nameof(dataFile)); |
||||
|
this.mDataFile = dataFile; // 注意:原代码有两次赋值,保留逻辑
|
||||
|
if (!RWL.ContainsKey(dataFile)) |
||||
|
{ |
||||
|
lockName = dataFile; |
||||
|
RWL.Add(dataFile, new ClsLock()); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
public SqliteHelper(string dataFile, string PassWord) |
||||
|
{ |
||||
|
this.mDataFile = dataFile ?? throw new ArgumentNullException(nameof(dataFile)); |
||||
|
this.mPassWord = PassWord ?? throw new ArgumentNullException(nameof(PassWord)); |
||||
|
this.mDataFile = dataFile; |
||||
|
if (!RWL.ContainsKey(dataFile)) |
||||
|
{ |
||||
|
lockName = dataFile; |
||||
|
RWL.Add(dataFile, new ClsLock()); |
||||
|
} |
||||
|
} |
||||
|
#endregion
|
||||
|
|
||||
|
#region 打开/关闭 数据库
|
||||
|
// 注意:Microsoft.Data.Sqlite.Core 不支持连接字符串构建器,需手动拼接
|
||||
|
// 加密支持通常需要 Microsoft.Data.Sqlite.Core + SQLitePCLRaw.bundle_green (且加密功能可能受限)
|
||||
|
|
||||
|
public void Open() |
||||
|
{ |
||||
|
// 构建连接字符串
|
||||
|
var connectionString = $"Data Source={mDataFile}"; |
||||
|
if (!string.IsNullOrWhiteSpace(mPassWord)) |
||||
|
{ |
||||
|
connectionString += $";Password={mPassWord}"; // 注意:原生 Microsoft.Data.Sqlite 不支持加密
|
||||
|
// 如果需要加密,通常需要使用 SQLitePCLRaw 的特定提供程序或第三方库
|
||||
|
} |
||||
|
|
||||
|
mConn = new SqliteConnection(connectionString); |
||||
|
|
||||
|
// 如果文件不存在,Open() 会自动创建数据库文件
|
||||
|
// 但表结构需要你自己执行 CREATE TABLE
|
||||
|
mConn.Open(); |
||||
|
|
||||
|
Console.WriteLine("The database was opened successfully"); |
||||
|
} |
||||
|
|
||||
|
public void Close() |
||||
|
{ |
||||
|
if (this.mConn != null) |
||||
|
{ |
||||
|
try |
||||
|
{ |
||||
|
this.mConn.Close(); |
||||
|
// 注意:不要在这里移除 RWL,因为这是静态的,可能影响其他实例
|
||||
|
// if (RWL.ContainsKey(LockName)) { RWL.Remove(LockName); }
|
||||
|
} |
||||
|
catch |
||||
|
{ |
||||
|
Console.WriteLine("Shutdown failed"); |
||||
|
} |
||||
|
} |
||||
|
Console.WriteLine("The database was shut down successfully"); |
||||
|
} |
||||
|
#endregion
|
||||
|
|
||||
|
#region 事务
|
||||
|
public void BeginTrain() |
||||
|
{ |
||||
|
EnsureConnection(); |
||||
|
dbTrans = mConn.BeginTransaction(); |
||||
|
} |
||||
|
|
||||
|
public void DBCommit() |
||||
|
{ |
||||
|
try |
||||
|
{ |
||||
|
dbTrans?.Commit(); |
||||
|
dbTrans?.Dispose(); |
||||
|
dbTrans = null; // 提交后置空
|
||||
|
} |
||||
|
catch (Exception) |
||||
|
{ |
||||
|
dbTrans?.Rollback(); |
||||
|
dbTrans?.Dispose(); |
||||
|
dbTrans = null; |
||||
|
} |
||||
|
} |
||||
|
#endregion
|
||||
|
|
||||
|
#region 工具
|
||||
|
// OpenConnection 逻辑已合并到 Open() 方法中,因为 SqliteConnection 构造函数只接受字符串
|
||||
|
|
||||
|
public SqliteConnection Connection |
||||
|
{ |
||||
|
get { return mConn; } |
||||
|
private set { mConn = value ?? throw new ArgumentNullException(); } |
||||
|
} |
||||
|
|
||||
|
protected void EnsureConnection() |
||||
|
{ |
||||
|
if (this.mConn == null) |
||||
|
{ |
||||
|
throw new Exception("SQLiteManager.Connection=null"); |
||||
|
} |
||||
|
if (mConn.State != ConnectionState.Open) |
||||
|
{ |
||||
|
mConn.Open(); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
public string GetDataFile() => this.mDataFile; |
||||
|
|
||||
|
public bool TableExists(string table) |
||||
|
{ |
||||
|
if (string.IsNullOrEmpty(table)) return false; |
||||
|
EnsureConnection(); |
||||
|
|
||||
|
// 注意:参数化查询在表名/列名上无效,这里使用字符串拼接(需确保table变量安全)
|
||||
|
// 或者使用 PRAGMA table_info(table_name)
|
||||
|
var sql = $"SELECT count(*) FROM sqlite_master WHERE type='table' AND name='{table}'"; |
||||
|
|
||||
|
using (var cmd = new SqliteCommand(sql, Connection)) |
||||
|
{ |
||||
|
var result = cmd.ExecuteScalar(); |
||||
|
return Convert.ToInt32(result) > 0; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
public bool Vacuum() |
||||
|
{ |
||||
|
try |
||||
|
{ |
||||
|
using (var Command = new SqliteCommand("VACUUM", Connection)) |
||||
|
{ |
||||
|
Command.ExecuteNonQuery(); |
||||
|
} |
||||
|
return true; |
||||
|
} |
||||
|
catch (Exception) // SqliteException
|
||||
|
{ |
||||
|
return false; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 将 IDataReader 转换为 DataSet
|
||||
|
/// </summary>
|
||||
|
/// <param name="reader">已经执行了查询的 DataReader</param>
|
||||
|
/// <returns>包含数据的 DataSet</returns>
|
||||
|
public DataSet ReaderToDataSet(IDataReader reader) |
||||
|
{ |
||||
|
DataSet ds = new DataSet(); |
||||
|
|
||||
|
do |
||||
|
{ |
||||
|
DataTable schemaTable = reader.GetSchemaTable()!; |
||||
|
DataTable dataTable = new DataTable(); |
||||
|
|
||||
|
if (schemaTable != null) |
||||
|
{ |
||||
|
// 创建列
|
||||
|
foreach (DataRow schemaRow in schemaTable.Rows) |
||||
|
{ |
||||
|
string colName = schemaRow["ColumnName"].ToString()!; |
||||
|
Type dataType = (Type)schemaRow["DataType"]; |
||||
|
|
||||
|
// 防止列名重复
|
||||
|
string uniqueColName = colName; |
||||
|
int i = 1; |
||||
|
while (dataTable.Columns.Contains(uniqueColName)) |
||||
|
{ |
||||
|
uniqueColName = colName + i++; |
||||
|
} |
||||
|
|
||||
|
DataColumn column = new DataColumn(uniqueColName, dataType); |
||||
|
dataTable.Columns.Add(column); |
||||
|
} |
||||
|
|
||||
|
// 读取行
|
||||
|
while (reader.Read()) |
||||
|
{ |
||||
|
object[] values = new object[dataTable.Columns.Count]; |
||||
|
for (int i = 0; i < dataTable.Columns.Count; i++) |
||||
|
{ |
||||
|
values[i] = reader.IsDBNull(i) ? DBNull.Value : reader.GetValue(i); |
||||
|
} |
||||
|
dataTable.Rows.Add(values); |
||||
|
} |
||||
|
} |
||||
|
else |
||||
|
{ |
||||
|
// 如果没有 Schema (例如 "SELECT 1" 这种标量查询)
|
||||
|
// 需要手动创建列
|
||||
|
for (int i = 0; i < reader.FieldCount; i++) |
||||
|
{ |
||||
|
dataTable.Columns.Add("Column" + i); |
||||
|
} |
||||
|
while (reader.Read()) |
||||
|
{ |
||||
|
object[] values = new object[reader.FieldCount]; |
||||
|
reader.GetValues(values); |
||||
|
dataTable.Rows.Add(values); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
ds.Tables.Add(dataTable); |
||||
|
} while (reader.NextResult()); // 处理多个结果集
|
||||
|
|
||||
|
return ds; |
||||
|
} |
||||
|
|
||||
|
#endregion
|
||||
|
|
||||
|
|
||||
|
#region 执行SQL (Reader & Scalar)
|
||||
|
|
||||
|
public SqliteDataReader? ExecuteReader(string sql, SqliteParameter[]? paramArr) |
||||
|
{ |
||||
|
if (string.IsNullOrEmpty(sql)) throw new ArgumentNullException(nameof(sql)); |
||||
|
EnsureConnection(); |
||||
|
|
||||
|
// 注意:using 语句会立即释放 Reader,导致无法读取
|
||||
|
// 这里的 RWL 锁在 using 外部处理,或者调用方自己处理锁
|
||||
|
// 这里简化逻辑,直接返回 Reader,调用方需自行管理连接状态
|
||||
|
var cmd = new SqliteCommand(sql, Connection); |
||||
|
|
||||
|
if (paramArr != null) |
||||
|
{ |
||||
|
cmd.Parameters.AddRange(paramArr); |
||||
|
} |
||||
|
|
||||
|
// CommandBehavior.CloseConnection 确保 Reader 关闭时连接也关闭
|
||||
|
// 但这里我们管理连接,所以不加
|
||||
|
try |
||||
|
{ |
||||
|
// 注意:不要在这里 Dispose Command,否则 Reader 会失效
|
||||
|
// 返回 Reader,由调用方在读取完毕后关闭
|
||||
|
return cmd.ExecuteReader(); |
||||
|
} |
||||
|
catch(Exception ex) |
||||
|
{ |
||||
|
Console.WriteLine("[SQLITE_ExecuteReader:err]" + ex.ToString()); |
||||
|
cmd?.Dispose(); |
||||
|
return null; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
public DataSet? ExecuteDataSet(string sql, SqliteParameter[]? paramArr) |
||||
|
{ |
||||
|
var dat = ExecuteReader(sql, paramArr); |
||||
|
if (dat == null) |
||||
|
{ |
||||
|
return null; |
||||
|
} |
||||
|
return ReaderToDataSet(dat); |
||||
|
} |
||||
|
|
||||
|
// ExecuteScalar
|
||||
|
public object? ExecuteScalar(string sql, SqliteParameter[]? paramArr) |
||||
|
{ |
||||
|
if (string.IsNullOrEmpty(sql)) throw new ArgumentNullException(nameof(sql)); |
||||
|
EnsureConnection(); |
||||
|
|
||||
|
using (RWL[lockName].Read()) |
||||
|
{ |
||||
|
using (var cmd = new SqliteCommand(sql, Connection)) |
||||
|
{ |
||||
|
if (paramArr != null) |
||||
|
{ |
||||
|
cmd.Parameters.AddRange(paramArr); |
||||
|
} |
||||
|
try |
||||
|
{ |
||||
|
return cmd.ExecuteScalar(); |
||||
|
} |
||||
|
catch (Exception ex) |
||||
|
{ |
||||
|
Console.WriteLine("[SQLITE_ExecuteScalar:err]" + ex.ToString()); |
||||
|
return null; |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
// QueryOne (返回字典或对象)
|
||||
|
public Dictionary<string, object?>? QueryOne(string table, string conditionCol, object conditionVal) |
||||
|
{ |
||||
|
if (string.IsNullOrEmpty(table)) return null; |
||||
|
EnsureConnection(); |
||||
|
|
||||
|
string sql = $"SELECT * FROM [{table}]"; // 使用 [] 防止关键字冲突
|
||||
|
var parameters = new List<SqliteParameter>(); |
||||
|
|
||||
|
if (!string.IsNullOrEmpty(conditionCol)) |
||||
|
{ |
||||
|
sql += $" WHERE [{conditionCol}] = @{conditionCol}"; |
||||
|
parameters.Add(new SqliteParameter($"@{conditionCol}", conditionVal)); |
||||
|
} |
||||
|
|
||||
|
using (var cmd = new SqliteCommand(sql, Connection)) |
||||
|
{ |
||||
|
cmd.Parameters.AddRange(parameters.ToArray()); |
||||
|
using (var reader = cmd.ExecuteReader()) |
||||
|
{ |
||||
|
if (reader.Read()) |
||||
|
{ |
||||
|
var dict = new Dictionary<string, object?>(); |
||||
|
for (int i = 0; i < reader.FieldCount; i++) |
||||
|
{ |
||||
|
var value = reader.GetValue(i); |
||||
|
if (value == DBNull.Value) value = null; |
||||
|
dict[reader.GetName(i)] = value; |
||||
|
} |
||||
|
return dict; |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
return null; |
||||
|
} |
||||
|
#endregion
|
||||
|
|
||||
|
#region 增 删 改
|
||||
|
// 注意:参数前缀统一为 @
|
||||
|
|
||||
|
public int InsertData(string table, Dictionary<string, object> entity) |
||||
|
{ |
||||
|
if (string.IsNullOrEmpty(table)) throw new ArgumentNullException(nameof(table)); |
||||
|
EnsureConnection(); |
||||
|
|
||||
|
string sql = BuildInsert(table, entity); |
||||
|
var parameters = BuildParamArray(entity); |
||||
|
return ExecuteNonQuery(sql, parameters); |
||||
|
} |
||||
|
|
||||
|
public int Update(string table, Dictionary<string, object> entity, string where, SqliteParameter[]? whereParams) |
||||
|
{ |
||||
|
if (string.IsNullOrEmpty(table)) throw new ArgumentNullException(nameof(table)); |
||||
|
EnsureConnection(); |
||||
|
|
||||
|
string sql = BuildUpdate(table, entity); |
||||
|
var parameters = BuildParamArray(entity); |
||||
|
|
||||
|
if (!string.IsNullOrEmpty(where)) |
||||
|
{ |
||||
|
sql += " WHERE " + where; |
||||
|
if (whereParams != null) |
||||
|
{ |
||||
|
var combined = new SqliteParameter[parameters.Length + whereParams.Length]; |
||||
|
parameters.CopyTo(combined, 0); |
||||
|
whereParams.CopyTo(combined, parameters.Length); |
||||
|
parameters = combined; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
return ExecuteNonQuery(sql, parameters); |
||||
|
} |
||||
|
|
||||
|
public int Delete(string table, string where, SqliteParameter[]? whereParams) |
||||
|
{ |
||||
|
if (string.IsNullOrEmpty(table)) return 0; |
||||
|
EnsureConnection(); |
||||
|
|
||||
|
string sql = $"DELETE FROM [{table}]"; |
||||
|
if (!string.IsNullOrEmpty(where)) |
||||
|
{ |
||||
|
sql += " WHERE " + where; |
||||
|
} |
||||
|
|
||||
|
return ExecuteNonQuery(sql, whereParams); |
||||
|
} |
||||
|
|
||||
|
public int ExecuteNonQuery(string sql, SqliteParameter[]? paramArr) |
||||
|
{ |
||||
|
if (string.IsNullOrEmpty(sql)) throw new ArgumentNullException(nameof(sql)); |
||||
|
EnsureConnection(); |
||||
|
|
||||
|
using (RWL[lockName].Read()) // 注意:写操作通常建议用 Write 锁
|
||||
|
{ |
||||
|
using (var cmd = new SqliteCommand(sql, Connection)) |
||||
|
{ |
||||
|
if (paramArr != null) |
||||
|
{ |
||||
|
cmd.Parameters.AddRange(paramArr); |
||||
|
} |
||||
|
try |
||||
|
{ |
||||
|
return cmd.ExecuteNonQuery(); |
||||
|
} |
||||
|
catch (Exception ex) |
||||
|
{ |
||||
|
Console.WriteLine("[SQLITE_ExecuteScalar:err]" + ex.ToString()); |
||||
|
return 0; |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
// 构建 SQL 和 参数的方法保持不变,仅泛型类型变更
|
||||
|
private SqliteParameter[] BuildParamArray(Dictionary<string, object> entity) |
||||
|
{ |
||||
|
var list = new List<SqliteParameter>(); |
||||
|
foreach (string key in entity.Keys) |
||||
|
{ |
||||
|
// 注意:参数名必须包含 @ 前缀
|
||||
|
list.Add(new SqliteParameter($"@{key}", entity[key])); |
||||
|
} |
||||
|
return list.ToArray(); |
||||
|
} |
||||
|
|
||||
|
private string BuildInsert(string table, Dictionary<string, object> entity) |
||||
|
{ |
||||
|
var buf = new StringBuilder(); |
||||
|
buf.Append("INSERT INTO ").Append(table).Append(" ("); |
||||
|
var cols = new List<string>(); |
||||
|
var vals = new List<string>(); |
||||
|
foreach (string key in entity.Keys) |
||||
|
{ |
||||
|
cols.Add(key); |
||||
|
vals.Add($"@{key}"); |
||||
|
} |
||||
|
buf.Append(string.Join(",", cols)).Append(") VALUES ("); |
||||
|
buf.Append(string.Join(",", vals)).Append(")"); |
||||
|
return buf.ToString(); |
||||
|
} |
||||
|
|
||||
|
private string BuildUpdate(string table, Dictionary<string, object> entity) |
||||
|
{ |
||||
|
var buf = new StringBuilder(); |
||||
|
buf.Append("UPDATE ").Append(table).Append(" SET "); |
||||
|
var sets = new List<string>(); |
||||
|
foreach (string key in entity.Keys) |
||||
|
{ |
||||
|
sets.Add($"{key} = @{key}"); |
||||
|
} |
||||
|
buf.Append(string.Join(",", sets)); |
||||
|
return buf.ToString(); |
||||
|
} |
||||
|
|
||||
|
// 辅助方法:DataRow 转 Dictionary (由于没有 DataTable,通常直接从 Reader 转)
|
||||
|
// 这里保留原方法签名,但实际使用中可能需要调整
|
||||
|
public Dictionary<string, object?> DataTableToDictionary(DataTable dataTable) |
||||
|
{ |
||||
|
var result = new Dictionary<string, object?>(); |
||||
|
if (dataTable.Rows.Count > 0) |
||||
|
{ |
||||
|
var row = dataTable.Rows[0]; |
||||
|
foreach (DataColumn column in dataTable.Columns) |
||||
|
{ |
||||
|
var value = row[column]; |
||||
|
if (value == DBNull.Value) value = null; |
||||
|
result[column.ColumnName] = value; |
||||
|
} |
||||
|
} |
||||
|
return result; |
||||
|
} |
||||
|
#endregion
|
||||
|
} |
||||
|
} |
||||
@ -0,0 +1,62 @@ |
|||||
|
using System; |
||||
|
using System.IO; |
||||
|
using System.Text; |
||||
|
using System.Windows; // WPF 的核心命名空间
|
||||
|
using System.Windows.Controls; // WPF TextBox 所在的命名空间
|
||||
|
using System.Windows.Threading; // 用于 Dispatcher
|
||||
|
|
||||
|
namespace SunlightAggregationManager.UserClass |
||||
|
{ |
||||
|
public class ControlWriter : TextWriter |
||||
|
{ |
||||
|
private readonly TextBox _textbox; |
||||
|
|
||||
|
public ControlWriter(TextBox textbox) |
||||
|
{ |
||||
|
_textbox = textbox; |
||||
|
} |
||||
|
|
||||
|
public override void Write(char value) => AppendText(DateTime.Now + ": " + value.ToString()); |
||||
|
public override void Write(string? value) => AppendText(DateTime.Now + ": " + value); |
||||
|
public override void WriteLine(string? value) => AppendText(DateTime.Now + ": "+value + Environment.NewLine); |
||||
|
|
||||
|
private void AppendText(string text) |
||||
|
{ |
||||
|
if (!_textbox.Dispatcher.CheckAccess()) |
||||
|
{ |
||||
|
_textbox.Dispatcher.Invoke(DispatcherPriority.Background, new Action(() => AppendText(text))); |
||||
|
return; |
||||
|
} |
||||
|
_textbox.AppendText(text); |
||||
|
_textbox.ScrollToEnd(); |
||||
|
} |
||||
|
|
||||
|
public override Encoding Encoding => Encoding.UTF8; |
||||
|
} |
||||
|
|
||||
|
//全局日志管理器
|
||||
|
public static class GlobalLogManager |
||||
|
{ |
||||
|
private static ControlWriter? _currentWriter; |
||||
|
|
||||
|
// 【核心功能】注册目标 TextBox
|
||||
|
// 子页面加载时调用这个方法
|
||||
|
public static void RegisterTarget(TextBox logBox) |
||||
|
{ |
||||
|
// 创建新的 Writer
|
||||
|
_currentWriter = new ControlWriter(logBox); |
||||
|
|
||||
|
// 【关键】将 Console 的输出重定向到这个 Writer
|
||||
|
// 这样 Console.WriteLine("...") 就会自动写入到 logBox 中
|
||||
|
Console.SetOut(_currentWriter); |
||||
|
|
||||
|
Console.WriteLine("Start"); |
||||
|
} |
||||
|
|
||||
|
//提供一个直接写入的方法(如果不使用 Console.WriteLine)
|
||||
|
public static void Write(string message) |
||||
|
{ |
||||
|
_currentWriter?.WriteLine(message); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
@ -0,0 +1,12 @@ |
|||||
|
<Window x:Class="SunlightAggregationManager.UserWindow.UserSet" |
||||
|
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" |
||||
|
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:local="clr-namespace:SunlightAggregationManager.UserWindow" |
||||
|
mc:Ignorable="d" |
||||
|
Title="UserSet" Height="450" Width="800" WindowStyle="None" > |
||||
|
<Grid> |
||||
|
|
||||
|
</Grid> |
||||
|
</Window> |
||||
@ -0,0 +1,25 @@ |
|||||
|
using System; |
||||
|
using System.Collections.Generic; |
||||
|
using System.Text; |
||||
|
using System.Windows; |
||||
|
using System.Windows.Controls; |
||||
|
using System.Windows.Data; |
||||
|
using System.Windows.Documents; |
||||
|
using System.Windows.Input; |
||||
|
using System.Windows.Media; |
||||
|
using System.Windows.Media.Imaging; |
||||
|
using System.Windows.Shapes; |
||||
|
|
||||
|
namespace SunlightAggregationManager.UserWindow |
||||
|
{ |
||||
|
/// <summary>
|
||||
|
/// UserSet.xaml 的交互逻辑
|
||||
|
/// </summary>
|
||||
|
public partial class UserSet : Window |
||||
|
{ |
||||
|
public UserSet() |
||||
|
{ |
||||
|
InitializeComponent(); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
@ -0,0 +1,38 @@ |
|||||
|
<UserControl x:Class="SunlightAggregationManager.View.SettingPage" |
||||
|
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" |
||||
|
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" |
||||
|
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" |
||||
|
xmlns:d="http://schemas.microsoft.com/expression/blend/2008" |
||||
|
xmlns:local="clr-namespace:SunlightAggregationManager.View" |
||||
|
xmlns:lang="clr-namespace:SunlightAggregationManager" |
||||
|
mc:Ignorable="d" |
||||
|
d:DesignHeight="720" d:DesignWidth="1080"> |
||||
|
<Grid> |
||||
|
|
||||
|
<ComboBox HorizontalAlignment="Left" Height="40" Margin="150,10,0,0" x:Name="comboBoxLanguage" VerticalAlignment="Top" Width="400" FontSize="24"/> |
||||
|
<TextBlock HorizontalAlignment="Left" Height="40" Margin="10,10,0,0" TextWrapping="Wrap" Text="{x:Static lang:ResourceLanguage.Language}" VerticalAlignment="Top" Width="120" FontSize="25" Padding="0,0,0,0"/> |
||||
|
<TextBlock HorizontalAlignment="Left" Height="40" Margin="10,60,0,0" TextWrapping="Wrap" Text="{x:Static lang:ResourceLanguage.SERVER}" VerticalAlignment="Top" Width="120" FontSize="25"/> |
||||
|
<TextBlock HorizontalAlignment="Left" Height="40" Margin="10,110,0,0" TextWrapping="Wrap" Text="{x:Static lang:ResourceLanguage.Mode}" VerticalAlignment="Top" Width="120" FontSize="25"/> |
||||
|
<TextBlock HorizontalAlignment="Left" Height="40" Margin="10,160,0,0" TextWrapping="Wrap" Text="{x:Static lang:ResourceLanguage.SQL}" VerticalAlignment="Top" Width="120" FontSize="25"/> |
||||
|
<TextBlock HorizontalAlignment="Left" Height="40" Margin="10,210,0,0" TextWrapping="Wrap" Text="{x:Static lang:ResourceLanguage.user}" VerticalAlignment="Top" Width="120" FontSize="25"/> |
||||
|
<TextBlock HorizontalAlignment="Left" Height="40" Margin="10,260,0,0" TextWrapping="Wrap" Text="{x:Static lang:ResourceLanguage.Password}" VerticalAlignment="Top" Width="120" FontSize="25"/> |
||||
|
<TextBox x:Name="TEXT_SQLIP" HorizontalAlignment="Left" Height="40" Margin="150,60,0,0" TextWrapping="NoWrap" VerticalAlignment="Top" Width="400" FontSize="24"/> |
||||
|
<ComboBox x:Name="TEXT_SQMOD" Height="40" Margin="150,110,0,0" VerticalAlignment="Top" Width="400" FontSize="24" HorizontalAlignment="Left"/> |
||||
|
<TextBox x:Name="TEXT_SQLNAME" HorizontalAlignment="Left" Height="40" Margin="150,160,0,0" TextWrapping="NoWrap" VerticalAlignment="Top" Width="400" FontSize="24"/> |
||||
|
<TextBox x:Name="TEXT_SQLUSER" HorizontalAlignment="Left" Height="40" Margin="150,210,0,0" TextWrapping="NoWrap" VerticalAlignment="Top" Width="400" FontSize="24"/> |
||||
|
<TextBox x:Name="TEXT_SQLPASWORD" HorizontalAlignment="Left" Height="40" Margin="150,260,0,0" TextWrapping="NoWrap" VerticalAlignment="Top" Width="400" FontSize="24"/> |
||||
|
<Button Content="{x:Static lang:ResourceLanguage.Test}" HorizontalAlignment="Left" Height="35" Margin="370,335,0,0" VerticalAlignment="Top" Width="80" Click="test_Click"/> |
||||
|
<TextBlock HorizontalAlignment="Left" Height="40" Margin="10,335,0,0" x:Name="textlog" TextWrapping="Wrap" Text="------------" VerticalAlignment="Top" Width="140" FontSize="20"/> |
||||
|
|
||||
|
|
||||
|
<CheckBox x:Name="EnableTCP" Content="{x:Static lang:ResourceLanguage.EnableTCP}" HorizontalAlignment="Left" Margin="600,10,0,0" VerticalAlignment="Top" Height="40" Width="300" FontSize="24" |
||||
|
Checked="EnableTCP_Checked" Unchecked="EnableTCP_Unchecked"/> |
||||
|
<CheckBox x:Name="EnableHTTP" Content="{x:Static lang:ResourceLanguage.EnableHTTP}" HorizontalAlignment="Left" Margin="600,60,0,0" VerticalAlignment="Top" Height="40" Width="300" FontSize="24" |
||||
|
Checked="EnableHTTP_Checked" Unchecked="EnableHTTP_Unchecked"/> |
||||
|
<CheckBox x:Name="EnableTLS" Content="{x:Static lang:ResourceLanguage.EnableTLS}" HorizontalAlignment="Left" Margin="600,110,0,0" VerticalAlignment="Top" Height="40" Width="300" FontSize="24" |
||||
|
Checked="EnableTLS_Checked" Unchecked="EnableTLS_Unchecked"/> |
||||
|
|
||||
|
<Button Content="{x:Static lang:ResourceLanguage.save}" HorizontalAlignment="Right" Height="35" Margin="470,335,0,0" VerticalAlignment="Bottom" Width="80" Click="save_Click"/> |
||||
|
|
||||
|
</Grid> |
||||
|
</UserControl> |
||||
@ -0,0 +1,157 @@ |
|||||
|
using System; |
||||
|
using System.Collections.Generic; |
||||
|
//using System.Data.SqlClient;
|
||||
|
using Microsoft.Data.SqlClient; |
||||
|
using System.Text; |
||||
|
using System.Windows; |
||||
|
using System.Windows.Controls; |
||||
|
using System.Windows.Data; |
||||
|
using System.Windows.Documents; |
||||
|
using System.Windows.Input; |
||||
|
using System.Windows.Media; |
||||
|
using System.Windows.Media.Imaging; |
||||
|
using System.Windows.Navigation; |
||||
|
using System.Windows.Shapes; |
||||
|
|
||||
|
namespace SunlightAggregationManager.View |
||||
|
{ |
||||
|
/// <summary>
|
||||
|
/// SettingPage.xaml 的交互逻辑
|
||||
|
/// </summary>
|
||||
|
public partial class SettingPage : UserControl |
||||
|
{ |
||||
|
//调用配置文件
|
||||
|
private readonly UserClass.IniFile.IniFiles Configini = new UserClass.IniFile.IniFiles(Convert.ToString(System.AppDomain.CurrentDomain.BaseDirectory) + "Configini.ini"); |
||||
|
|
||||
|
public SettingPage() |
||||
|
{ |
||||
|
InitializeComponent(); |
||||
|
|
||||
|
string[] Language = { "en-US", "zh-CN", "zh-TW" }; |
||||
|
string[] SQMOD = { "Windows Authentication", "SQL SERVER" }; |
||||
|
comboBoxLanguage.ItemsSource = Language; |
||||
|
TEXT_SQMOD.ItemsSource = SQMOD; |
||||
|
try |
||||
|
{ |
||||
|
comboBoxLanguage.Text = Configini.IniReadvalue("SYS", "Language"); |
||||
|
TEXT_SQLIP.Text = Configini.IniReadvalue("SQL_SERVER", "SQL1"); //读配置文件
|
||||
|
TEXT_SQLNAME.Text = Configini.IniReadvalue("SQL_SERVER", "SQL2"); |
||||
|
TEXT_SQMOD.Text = Configini.IniReadvalue("SQL_SERVER", "SQL3"); |
||||
|
TEXT_SQLUSER.Text = Configini.IniReadvalue("SQL_SERVER", "SQL4"); |
||||
|
TEXT_SQLPASWORD.Text = Configini.IniReadvalue("SQL_SERVER", "SQL5"); |
||||
|
} |
||||
|
catch (Exception) { } |
||||
|
} |
||||
|
|
||||
|
private async void test_Click(object sender, RoutedEventArgs e) |
||||
|
{ |
||||
|
// UI 更新:显示正在测试
|
||||
|
textlog.Text = "TEST"; |
||||
|
// 建议使用静态缓存的画笔,避免频繁创建对象导致内存抖动
|
||||
|
textlog.Foreground = Brushes.Blue; |
||||
|
try |
||||
|
{ |
||||
|
// 构建连接字符串
|
||||
|
var builder = new SqlConnectionStringBuilder |
||||
|
{ |
||||
|
DataSource = TEXT_SQLIP.Text, |
||||
|
InitialCatalog = TEXT_SQLNAME.Text, |
||||
|
IntegratedSecurity = TEXT_SQMOD.Text == "Windows Authentication", |
||||
|
TrustServerCertificate = true |
||||
|
}; |
||||
|
|
||||
|
if (!builder.IntegratedSecurity) |
||||
|
{ |
||||
|
builder.UserID = TEXT_SQLUSER.Text; |
||||
|
builder.Password = TEXT_SQLPASWORD.Text; |
||||
|
} |
||||
|
|
||||
|
// 异步连接数据库
|
||||
|
await using var conn_SC = new SqlConnection(builder.ConnectionString); |
||||
|
|
||||
|
// 使用 OpenAsync 代替 Open,避免卡死界面
|
||||
|
await conn_SC.OpenAsync(); |
||||
|
using (var cmd = new SqlCommand("SELECT 1", conn_SC)) |
||||
|
{ |
||||
|
cmd.ExecuteReader(); |
||||
|
} |
||||
|
} |
||||
|
catch (Exception) |
||||
|
{ |
||||
|
// 失败处理
|
||||
|
textlog.Text = "Link timeout"; |
||||
|
textlog.Foreground = Brushes.Red; |
||||
|
return; // 直接返回,不执行后续代码
|
||||
|
} |
||||
|
|
||||
|
// 成功处理
|
||||
|
textlog.Text = "Link succeed"; |
||||
|
textlog.Foreground = Brushes.Green; |
||||
|
|
||||
|
Console.WriteLine("IP = "+TEXT_SQLIP.Text); |
||||
|
Console.WriteLine("DataBase = " + TEXT_SQLNAME.Text); |
||||
|
Console.WriteLine("Link succeed"); |
||||
|
|
||||
|
// 弹出对话框本身会阻塞 UI 线程
|
||||
|
MessageBoxResult SqlShow = System.Windows.MessageBox.Show( |
||||
|
"Whether the connection is successfully saved", |
||||
|
"SQL", |
||||
|
MessageBoxButton.YesNo, |
||||
|
MessageBoxImage.Information); |
||||
|
|
||||
|
if (SqlShow == MessageBoxResult.Yes) |
||||
|
{ |
||||
|
Configini.IniWritevalue("SQL_SERVER", "SQL1", TEXT_SQLIP.Text); |
||||
|
Configini.IniWritevalue("SQL_SERVER", "SQL2", TEXT_SQLNAME.Text); |
||||
|
Configini.IniWritevalue("SQL_SERVER", "SQL3", TEXT_SQMOD.SelectedValue?.ToString()); |
||||
|
Configini.IniWritevalue("SQL_SERVER", "SQL4", TEXT_SQLUSER.Text); |
||||
|
Configini.IniWritevalue("SQL_SERVER", "SQL5", TEXT_SQLPASWORD.Text); |
||||
|
|
||||
|
Console.WriteLine("Save database settings"); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
private void save_Click(object sender, RoutedEventArgs e) |
||||
|
{ |
||||
|
Configini.IniWritevalue("SYS", "Language", comboBoxLanguage.SelectedValue?.ToString()); |
||||
|
Configini.IniWritevalue("SQL_SERVER", "SQL1", TEXT_SQLIP.Text); |
||||
|
Configini.IniWritevalue("SQL_SERVER", "SQL2", TEXT_SQLNAME.Text); |
||||
|
Configini.IniWritevalue("SQL_SERVER", "SQL3", TEXT_SQMOD.SelectedValue?.ToString()); |
||||
|
Configini.IniWritevalue("SQL_SERVER", "SQL4", TEXT_SQLUSER.Text); |
||||
|
Configini.IniWritevalue("SQL_SERVER", "SQL5", TEXT_SQLPASWORD.Text); |
||||
|
|
||||
|
Console.WriteLine("Save global settings"); |
||||
|
} |
||||
|
|
||||
|
//是否启用tcp
|
||||
|
private void EnableTCP_Checked(object sender, RoutedEventArgs e) |
||||
|
{ |
||||
|
Configini.IniWritevalue("NETWORK", "TCP", "1"); |
||||
|
} |
||||
|
|
||||
|
private void EnableTCP_Unchecked(object sender, RoutedEventArgs e) |
||||
|
{ |
||||
|
Configini.IniWritevalue("NETWORK", "TCP", "0"); |
||||
|
} |
||||
|
|
||||
|
private void EnableHTTP_Checked(object sender, RoutedEventArgs e) |
||||
|
{ |
||||
|
Configini.IniWritevalue("NETWORK", "HTTP", "1"); |
||||
|
} |
||||
|
|
||||
|
private void EnableHTTP_Unchecked(object sender, RoutedEventArgs e) |
||||
|
{ |
||||
|
Configini.IniWritevalue("NETWORK", "HTTP", "0"); |
||||
|
} |
||||
|
|
||||
|
private void EnableTLS_Checked(object sender, RoutedEventArgs e) |
||||
|
{ |
||||
|
Configini.IniWritevalue("NETWORK", "TLS", "1"); |
||||
|
} |
||||
|
|
||||
|
private void EnableTLS_Unchecked(object sender, RoutedEventArgs e) |
||||
|
{ |
||||
|
Configini.IniWritevalue("NETWORK", "TLS", "0"); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
@ -0,0 +1,13 @@ |
|||||
|
<UserControl x:Class="SunlightAggregationManager.View.StatePage" |
||||
|
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" |
||||
|
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" |
||||
|
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" |
||||
|
xmlns:d="http://schemas.microsoft.com/expression/blend/2008" |
||||
|
xmlns:local="clr-namespace:SunlightAggregationManager.View" |
||||
|
mc:Ignorable="d" |
||||
|
d:DesignHeight="720" d:DesignWidth="1080"> |
||||
|
<Grid> |
||||
|
<TextBox x:Name="LOG" TextWrapping="Wrap" Margin="5,200,5,5" FontSize="16"/> |
||||
|
|
||||
|
</Grid> |
||||
|
</UserControl> |
||||
@ -0,0 +1,33 @@ |
|||||
|
using Net.Codecrete.QrCodeGenerator; |
||||
|
using SunlightAggregationManager.UserClass; |
||||
|
using System; |
||||
|
using System.Collections.Generic; |
||||
|
using System.Drawing; |
||||
|
using System.IO; |
||||
|
using System.Text; |
||||
|
using System.Windows; |
||||
|
using System.Windows.Controls; |
||||
|
using System.Windows.Data; |
||||
|
using System.Windows.Documents; |
||||
|
using System.Windows.Input; |
||||
|
using System.Windows.Media; |
||||
|
using System.Windows.Media.Imaging; |
||||
|
using System.Windows.Navigation; |
||||
|
using System.Windows.Shapes; |
||||
|
using static System.Net.WebRequestMethods; |
||||
|
|
||||
|
namespace SunlightAggregationManager.View |
||||
|
{ |
||||
|
/// <summary>
|
||||
|
/// StatePage.xaml 的交互逻辑
|
||||
|
/// </summary>
|
||||
|
public partial class StatePage : UserControl |
||||
|
{ |
||||
|
public StatePage() |
||||
|
{ |
||||
|
InitializeComponent(); |
||||
|
} |
||||
|
|
||||
|
} |
||||
|
|
||||
|
} |
||||
@ -0,0 +1,114 @@ |
|||||
|
<UserControl x:Class="SunlightAggregationManager.View.UserPage" |
||||
|
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" |
||||
|
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" |
||||
|
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" |
||||
|
xmlns:d="http://schemas.microsoft.com/expression/blend/2008" |
||||
|
xmlns:local="clr-namespace:SunlightAggregationManager.View" |
||||
|
xmlns:ConvertMoels="clr-namespace:SunlightAggregationManager.ConvertMoels" |
||||
|
xmlns:lang="clr-namespace:SunlightAggregationManager" |
||||
|
xmlns:viewmodel="clr-namespace:SunlightAggregationManager.ViewModel" |
||||
|
d:DataContext="{d:DesignInstance Type=viewmodel:MainWindowViewModel}" |
||||
|
mc:Ignorable="d" |
||||
|
d:DesignHeight="720" d:DesignWidth="1080"> |
||||
|
<!--设定变换器--> |
||||
|
<UserControl.Resources> |
||||
|
<ConvertMoels:StatenConvert x:Key="StatenConvert"/> |
||||
|
<ConvertMoels:StatenERRConvert x:Key="StatenERRConvert"/> |
||||
|
</UserControl.Resources> |
||||
|
<Grid> |
||||
|
<DataGrid x:Name="DataGridmachinesdata" |
||||
|
ItemsSource="{Binding UserData.DefaultView}" |
||||
|
SelectionChanged="DataGridmachinesdata_SelectionChanged" |
||||
|
SelectionMode="Single" |
||||
|
AlternationCount="2" |
||||
|
IsReadOnly="True" |
||||
|
FontSize="26" |
||||
|
Margin="5,5,5,5" |
||||
|
AutoGenerateColumns="False" |
||||
|
MinColumnWidth="30" |
||||
|
HorizontalGridLinesBrush="#FFC9C9C9" |
||||
|
VerticalGridLinesBrush="#FFC9C9C9" |
||||
|
GridLinesVisibility="All" |
||||
|
BorderBrush="#CCCCCC" |
||||
|
BorderThickness="1,1,1,1" |
||||
|
ColumnHeaderHeight="40" |
||||
|
HorizontalContentAlignment="Right" |
||||
|
CanUserReorderColumns="False" |
||||
|
VerticalContentAlignment="Center"> |
||||
|
<DataGrid.RowStyle > |
||||
|
<Style TargetType="{x:Type DataGridRow}"> |
||||
|
<Style.Triggers> |
||||
|
<Trigger Property="ItemsControl.AlternationIndex" Value="0"> |
||||
|
<Setter Property="Background" Value="#FFFFFFFF" /> |
||||
|
<Setter Property="Height" Value="40" /> |
||||
|
</Trigger> |
||||
|
<Trigger Property="ItemsControl.AlternationIndex" Value="1"> |
||||
|
<Setter Property="Background" Value="#FFF0F0F0" /> |
||||
|
<Setter Property="Height" Value="40" /> |
||||
|
</Trigger> |
||||
|
<Trigger Property="IsMouseOver" Value="False"> |
||||
|
</Trigger> |
||||
|
</Style.Triggers> |
||||
|
</Style> |
||||
|
</DataGrid.RowStyle> |
||||
|
<DataGrid.CellStyle> |
||||
|
<Style TargetType="DataGridCell"> |
||||
|
<Setter Property="BorderThickness" Value="0"/> |
||||
|
<Setter Property="MinWidth" Value="20"/> |
||||
|
<Style.Triggers> |
||||
|
<Trigger Property="IsSelected" Value="True"> |
||||
|
<Setter Property="Background" Value="#FFC0C0C0"/> |
||||
|
<Setter Property="BorderBrush" Value="#FFC0C0C0"/> |
||||
|
<Setter Property="Foreground" Value="#000000"/> |
||||
|
</Trigger> |
||||
|
</Style.Triggers> |
||||
|
</Style> |
||||
|
</DataGrid.CellStyle> |
||||
|
<DataGrid.ContextMenu> |
||||
|
<ContextMenu> |
||||
|
<MenuItem Header="{x:Static lang:ResourceLanguage.information}" Click="MenuItem_Information"/> |
||||
|
<MenuItem Header="{x:Static lang:ResourceLanguage.Offine}" Click="MenuItem_Offine"/> |
||||
|
<MenuItem Header="{x:Static lang:ResourceLanguage.Enable}" Click="MenuItem_Enable"/> |
||||
|
<MenuItem Header="{x:Static lang:ResourceLanguage.Disable}" Click="MenuItem_Disable"/> |
||||
|
</ContextMenu> |
||||
|
</DataGrid.ContextMenu> |
||||
|
<DataGrid.Columns> |
||||
|
<!--列信息绑定--> |
||||
|
<DataGridTextColumn Header="{x:Static lang:ResourceLanguage.user}" Width="200" FontSize="26" |
||||
|
Binding="{Binding User}" MaxWidth="200" MinWidth="200" CanUserReorder="False"/> |
||||
|
|
||||
|
<DataGridTextColumn Header="{x:Static lang:ResourceLanguage.Name}" Width="200" FontSize="26" |
||||
|
Binding="{Binding Name}" MaxWidth="200" MinWidth="200" CanUserReorder="False"/> |
||||
|
<DataGridTextColumn Header="{x:Static lang:ResourceLanguage.state}" Width="120" FontSize="26" |
||||
|
MaxWidth="120" MinWidth="120" CanUserReorder="False"> |
||||
|
<!--事件名称:工作状态数字转文字显示,转换器StatenConvert--> |
||||
|
<DataGridTextColumn.ElementStyle> |
||||
|
<Style TargetType="{x:Type TextBlock}"> |
||||
|
<Setter Property="Text" Value="{Binding Path=State,Converter={StaticResource StatenConvert}}"> |
||||
|
</Setter> |
||||
|
<Setter Property="Foreground" Value="{Binding Path=State,Converter={StaticResource StatenERRConvert}}"> |
||||
|
</Setter> |
||||
|
</Style> |
||||
|
</DataGridTextColumn.ElementStyle> |
||||
|
</DataGridTextColumn> |
||||
|
<DataGridTextColumn Header="{x:Static lang:ResourceLanguage.Department}" Width="150" FontSize="26" |
||||
|
Binding="{Binding Department}" MinWidth="150" CanUserReorder="False"/> |
||||
|
<DataGridTextColumn Header="{x:Static lang:ResourceLanguage.LastLoginIP}" Width="250" FontSize="26" |
||||
|
Binding="{Binding LastLoginIP}" MinWidth="250" CanUserReorder="False"/> |
||||
|
<DataGridTextColumn Header="{x:Static lang:ResourceLanguage.LastLoginTerminal}" Width="300" FontSize="26" |
||||
|
Binding="{Binding LastLoginTerminal}" MinWidth="300" CanUserReorder="False"/> |
||||
|
<DataGridTextColumn Header="{x:Static lang:ResourceLanguage.LastLoginTime}" Width="300" FontSize="26" |
||||
|
Binding="{Binding LastLoginTime}" MinWidth="300" CanUserReorder="False"/> |
||||
|
<DataGridTextColumn Header="{x:Static lang:ResourceLanguage.LastActionTime}" Width="300" FontSize="26" |
||||
|
Binding="{Binding LastActionTime}" MinWidth="300" CanUserReorder="False"/> |
||||
|
<DataGridTextColumn Header="{x:Static lang:ResourceLanguage.LastOfflineTime}" Width="300" FontSize="26" |
||||
|
Binding="{Binding LastOfflineTime}" MinWidth="300" CanUserReorder="False"/> |
||||
|
<DataGridTextColumn Header="{x:Static lang:ResourceLanguage.Groups}" Width="200" FontSize="26" |
||||
|
Binding="{Binding Groups}" MinWidth="200" CanUserReorder="False"/> |
||||
|
<DataGridTextColumn Header="{x:Static lang:ResourceLanguage.information}" Width="*" FontSize="26" |
||||
|
Binding="{Binding Note}" MinWidth="200" CanUserReorder="False"/> |
||||
|
|
||||
|
</DataGrid.Columns> |
||||
|
</DataGrid> |
||||
|
</Grid> |
||||
|
</UserControl> |
||||
@ -0,0 +1,90 @@ |
|||||
|
using SunlightAggregationManager.UserClass; |
||||
|
using SunlightAggregationManager.ViewModel; |
||||
|
using System; |
||||
|
using System.Collections.Generic; |
||||
|
using System.Data; |
||||
|
using System.Text; |
||||
|
using System.Windows; |
||||
|
using System.Windows.Controls; |
||||
|
using System.Windows.Data; |
||||
|
using System.Windows.Documents; |
||||
|
using System.Windows.Input; |
||||
|
using System.Windows.Media; |
||||
|
using System.Windows.Media.Imaging; |
||||
|
using System.Windows.Navigation; |
||||
|
using System.Windows.Shapes; |
||||
|
using System.Xml.Linq; |
||||
|
using TouchSocket.Core; |
||||
|
using TouchSocket.Sockets; |
||||
|
|
||||
|
namespace SunlightAggregationManager.View |
||||
|
{ |
||||
|
/// <summary>
|
||||
|
/// UserPage.xaml 的交互逻辑
|
||||
|
/// </summary>
|
||||
|
public partial class UserPage : UserControl |
||||
|
{ |
||||
|
public UserPage() |
||||
|
{ |
||||
|
InitializeComponent(); |
||||
|
} |
||||
|
|
||||
|
private string user_id = ""; |
||||
|
|
||||
|
private void DataGridmachinesdata_SelectionChanged(object sender, SelectionChangedEventArgs e) |
||||
|
{ |
||||
|
int rownum = DataGridmachinesdata.SelectedIndex;//获取鼠标选中行并定义变量
|
||||
|
if (rownum != -1)//判断鼠标定位是否有效
|
||||
|
{ |
||||
|
var row_ = DataGridmachinesdata.Columns[0].GetCellContent(DataGridmachinesdata.Items[rownum]); |
||||
|
if (row_ is TextBlock textBlock) |
||||
|
{ |
||||
|
user_id = textBlock.Text;//定位第0列,
|
||||
|
} |
||||
|
|
||||
|
|
||||
|
|
||||
|
} |
||||
|
|
||||
|
} |
||||
|
|
||||
|
private void MenuItem_Information(object sender, RoutedEventArgs e)//信息
|
||||
|
{ |
||||
|
|
||||
|
} |
||||
|
|
||||
|
private async void MenuItem_Offine(object sender, RoutedEventArgs e)//下线
|
||||
|
{ |
||||
|
var _state = MainWindowViewModel.Selet_Memory(MainWindowViewModel._userData, "State", "User='" + user_id + "'"); |
||||
|
var _user = MainWindowViewModel.Selet_Memory(MainWindowViewModel._userData, "LinkID", "User='" + user_id +"'"); |
||||
|
|
||||
|
if ((_user != null)&&(_state?.ToString()=="20")) |
||||
|
{ |
||||
|
foreach (var item in AsyncTcpServer.service.Clients) |
||||
|
{ |
||||
|
if (item.Id == _user.ToString()) |
||||
|
{ |
||||
|
await item.CloseAsync(); |
||||
|
|
||||
|
//在断开连接后,释放对象。
|
||||
|
item.Dispose(); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
private void MenuItem_Enable(object sender, RoutedEventArgs e)//启用账号
|
||||
|
{ |
||||
|
MainWindowViewModel.Updata_Memory(MainWindowViewModel._userData, "User='" + user_id + "'", "0", "State"); |
||||
|
} |
||||
|
|
||||
|
private void MenuItem_Disable(object sender, RoutedEventArgs e)//停用账号
|
||||
|
{ |
||||
|
MainWindowViewModel.Updata_Memory(MainWindowViewModel._userData, "User='" + user_id + "'", "99", "State"); |
||||
|
} |
||||
|
|
||||
|
|
||||
|
|
||||
|
|
||||
|
} |
||||
|
} |
||||
@ -0,0 +1,17 @@ |
|||||
|
<UserControl x:Class="SunlightAggregationManager.View.imgQR" |
||||
|
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" |
||||
|
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" |
||||
|
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" |
||||
|
xmlns:d="http://schemas.microsoft.com/expression/blend/2008" |
||||
|
xmlns:local="clr-namespace:SunlightAggregationManager.View" |
||||
|
mc:Ignorable="d" |
||||
|
Loaded="imgQR_Loaded" |
||||
|
d:DesignHeight="720" d:DesignWidth="1080"> |
||||
|
<Grid> |
||||
|
<TextBlock Text="APP" Margin="300,100,20,10" Foreground="Black" HorizontalAlignment="Left" FontSize="48" VerticalAlignment="Top"/> |
||||
|
<TextBlock Text="LINK" VerticalAlignment="Top" Margin="0,100,300,10" Foreground="Black" HorizontalAlignment="Right" FontSize="48"/> |
||||
|
<Image x:Name="imgQR_app" MinWidth="300" MinHeight="300" MaxWidth="300" MaxHeight="300" Width="300" Height="300" Stretch="Uniform" Margin="200,0,0,0" HorizontalAlignment="Left"/> |
||||
|
<Image x:Name="imgQR_link" MinWidth="300" MinHeight="300" MaxWidth="300" MaxHeight="300" Width="300" Height="300" Stretch="Uniform" HorizontalAlignment="Right" Margin="0,0,200,0"/> |
||||
|
|
||||
|
</Grid> |
||||
|
</UserControl> |
||||
@ -0,0 +1,105 @@ |
|||||
|
using Net.Codecrete.QrCodeGenerator; |
||||
|
using System; |
||||
|
using System.Collections.Generic; |
||||
|
using System.Drawing; |
||||
|
using System.IO; |
||||
|
using System.Text; |
||||
|
using System.Windows; |
||||
|
using System.Windows.Controls; |
||||
|
using System.Windows.Data; |
||||
|
using System.Windows.Documents; |
||||
|
using System.Windows.Input; |
||||
|
using System.Windows.Media; |
||||
|
using System.Windows.Media.Imaging; |
||||
|
using System.Windows.Media.TextFormatting; |
||||
|
using System.Windows.Navigation; |
||||
|
using System.Windows.Shapes; |
||||
|
using static System.Net.WebRequestMethods; |
||||
|
|
||||
|
namespace SunlightAggregationManager.View |
||||
|
{ |
||||
|
/// <summary>
|
||||
|
/// StatePage.xaml 的交互逻辑
|
||||
|
/// </summary>
|
||||
|
public partial class imgQR : UserControl |
||||
|
{ |
||||
|
//调用配置文件
|
||||
|
private readonly UserClass.IniFile.IniFiles Configini = new UserClass.IniFile.IniFiles(Convert.ToString(System.AppDomain.CurrentDomain.BaseDirectory) + "Configini.ini"); |
||||
|
|
||||
|
|
||||
|
public imgQR() |
||||
|
{ |
||||
|
InitializeComponent(); |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 生成二维码
|
||||
|
/// 调用示例: imgQR_CREATE("TEST", imgQR);;
|
||||
|
/// </summary>
|
||||
|
/// <param name="data">[字符串]</param>
|
||||
|
/// <param name="Image">Image控件</param>
|
||||
|
void imgQR_CREATE(string DAT, System.Windows.Controls.Image image) |
||||
|
{ |
||||
|
// 生成二维码对象
|
||||
|
QrCode qr = QrCode.EncodeText(DAT, QrCode.Ecc.Medium); |
||||
|
|
||||
|
// 手动绘制 Bitmap (不使用扩展方法)
|
||||
|
int scale = 10; // 每个点的像素大小
|
||||
|
int border = 4; // 边距
|
||||
|
int size = (qr.Size + border * 2) * scale; |
||||
|
// 创建位图
|
||||
|
using (Bitmap bmp = new Bitmap(size, size)) |
||||
|
{ |
||||
|
using (Graphics gfx = Graphics.FromImage(bmp)) |
||||
|
{ |
||||
|
// 填充背景(白色)
|
||||
|
gfx.Clear(System.Drawing.Color.White); |
||||
|
|
||||
|
// 绘制二维码点阵(黑色)
|
||||
|
using (System.Drawing.Brush brush = new SolidBrush(System.Drawing.Color.Black)) |
||||
|
{ |
||||
|
for (int y = 0; y < qr.Size; y++) |
||||
|
{ |
||||
|
for (int x = 0; x < qr.Size; x++) |
||||
|
{ |
||||
|
if (qr.GetModule(x, y)) |
||||
|
{ |
||||
|
System.Drawing.Rectangle rect = new System.Drawing.Rectangle( |
||||
|
(x + border) * scale, |
||||
|
(y + border) * scale, |
||||
|
scale, scale); |
||||
|
gfx.FillRectangle(brush, rect); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
// 转换为 WPF 的 BitmapImage 并显示
|
||||
|
using (MemoryStream ms = new MemoryStream()) |
||||
|
{ |
||||
|
bmp.Save(ms, System.Drawing.Imaging.ImageFormat.Png); |
||||
|
ms.Position = 0; |
||||
|
|
||||
|
BitmapImage bitmapImage = new BitmapImage(); |
||||
|
bitmapImage.BeginInit(); |
||||
|
bitmapImage.StreamSource = ms; |
||||
|
bitmapImage.CacheOption = BitmapCacheOption.OnLoad; |
||||
|
bitmapImage.EndInit(); |
||||
|
|
||||
|
// 赋值给 Image 控件 (假设控件名为 imgQR)
|
||||
|
image.Source = bitmapImage; |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
private void imgQR_Loaded(object sender, RoutedEventArgs e) |
||||
|
{ |
||||
|
string text_APP = "TEST"; // 要生成二维码的内容
|
||||
|
string text_LINK = Configini.IniReadvalue("SYS", "LINK"); // 要生成二维码的内容
|
||||
|
|
||||
|
imgQR_CREATE(text_APP, imgQR_app); |
||||
|
imgQR_CREATE(text_LINK, imgQR_link); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
@ -0,0 +1,31 @@ |
|||||
|
<UserControl x:Class="SunlightAggregationManager.View.logPage" |
||||
|
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" |
||||
|
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" |
||||
|
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" |
||||
|
xmlns:d="http://schemas.microsoft.com/expression/blend/2008" |
||||
|
xmlns:local="clr-namespace:SunlightAggregationManager.View" |
||||
|
mc:Ignorable="d" |
||||
|
Loaded="logPage_Loaded" |
||||
|
d:DesignHeight="720" d:DesignWidth="1080"> |
||||
|
<Grid> |
||||
|
<Grid.ColumnDefinitions> |
||||
|
<ColumnDefinition Width="600"/> |
||||
|
<ColumnDefinition Width="5"/> |
||||
|
<ColumnDefinition/> |
||||
|
</Grid.ColumnDefinitions> |
||||
|
<ComboBox Height="45" x:Name="comboBox_log" VerticalAlignment="Top" FontSize="20" Background="#FF90BEFF" |
||||
|
DropDownClosed="comboBox_log_DropDownClosed" BorderBrush="#FF00204E" Padding="1,1,1,1"/> |
||||
|
<DataGrid Grid.Column="0" x:Name="gridLog" Margin="0,50,0,0" HeadersVisibility ="Column" |
||||
|
IsReadOnly="True" AutoGenerateColumns="False" MouseDoubleClick="DataGrid_gridLogClick" > |
||||
|
<DataGrid.Columns> |
||||
|
<!--列信息绑定--> |
||||
|
<DataGridTextColumn Binding="{Binding Name}" Header="Name" Width="195" MaxWidth="195" MinWidth="195" CanUserReorder="False"/> |
||||
|
<DataGridTextColumn Binding="{Binding Length}" Header="Volume" Width="50" MaxWidth="150" MinWidth="100" CanUserReorder="False"/> |
||||
|
<DataGridTextColumn Binding="{Binding CreationTimeUtc}" Header="Creation time" Width="150" MaxWidth="150" MinWidth="150" CanUserReorder="False"/> |
||||
|
<DataGridTextColumn Binding="{Binding LastWriteTimeUtc}" Header="Last modified time" Width="150" MaxWidth="150" MinWidth="150" CanUserReorder="False"/> |
||||
|
</DataGrid.Columns> |
||||
|
</DataGrid> |
||||
|
<GridSplitter Grid.Column="1" Width="5" HorizontalAlignment="Stretch" Background="#FF00204E"/> |
||||
|
<RichTextBox Grid.Column="2" x:Name="Logtext" IsReadOnly="True" Background="White" SelectionBrush="White"/> |
||||
|
</Grid> |
||||
|
</UserControl> |
||||
@ -0,0 +1,143 @@ |
|||||
|
using System; |
||||
|
using System.Collections.Generic; |
||||
|
using System.Data; |
||||
|
using System.IO; |
||||
|
using System.Text; |
||||
|
using System.Windows; |
||||
|
using System.Windows.Controls; |
||||
|
using System.Windows.Data; |
||||
|
using System.Windows.Documents; |
||||
|
using System.Windows.Input; |
||||
|
using System.Windows.Media; |
||||
|
using System.Windows.Media.Imaging; |
||||
|
using System.Windows.Navigation; |
||||
|
using System.Windows.Shapes; |
||||
|
|
||||
|
namespace SunlightAggregationManager.View |
||||
|
{ |
||||
|
/// <summary>
|
||||
|
/// logPage.xaml 的交互逻辑
|
||||
|
/// </summary>
|
||||
|
public partial class logPage : UserControl |
||||
|
{ |
||||
|
public logPage() |
||||
|
{ |
||||
|
InitializeComponent(); |
||||
|
} |
||||
|
|
||||
|
private readonly string Log_time = DateTime.Now.ToString("yyyy-MM-dd"); |
||||
|
DataTable logdataTable = new DataTable(); //建立缓存
|
||||
|
long TEXT_L; |
||||
|
|
||||
|
private void logPage_Loaded(object sender, RoutedEventArgs e)//页面打开事件
|
||||
|
{ |
||||
|
comboBox_log.ItemsSource = new string[3] { "Log", "ERR", "Exchange" }; |
||||
|
comboBox_log.Text = "Log"; |
||||
|
|
||||
|
logdataTable.Columns.Add("Name", typeof(string)); |
||||
|
logdataTable.Columns.Add("Length", typeof(int)); |
||||
|
logdataTable.Columns.Add("CreationTimeUtc", typeof(string)); |
||||
|
logdataTable.Columns.Add("LastWriteTimeUtc", typeof(string)); |
||||
|
|
||||
|
DirectoryInfo loginfo = new DirectoryInfo(System.Environment.CurrentDirectory + "\\Log"); //new文件夹
|
||||
|
logdataTable.Clear(); |
||||
|
foreach (var item in loginfo.GetFiles()) |
||||
|
{ |
||||
|
DataRow FileRow = logdataTable.NewRow(); |
||||
|
FileRow["Name"] = item.Name; |
||||
|
FileRow["Length"] = item.Length / 1024; |
||||
|
FileRow["CreationTimeUtc"] = item.CreationTimeUtc; |
||||
|
FileRow["LastWriteTimeUtc"] = item.LastWriteTimeUtc; |
||||
|
logdataTable.Rows.Add(FileRow); |
||||
|
} |
||||
|
logdataTable.DefaultView.Sort = "Name DESC"; |
||||
|
gridLog.ItemsSource = logdataTable.DefaultView.ToTable().DefaultView; |
||||
|
} |
||||
|
|
||||
|
private void LOG_file()//建立列表
|
||||
|
{ |
||||
|
DirectoryInfo loginfo = new DirectoryInfo(System.Environment.CurrentDirectory + "\\" + comboBox_log.Text); //new文件夹
|
||||
|
logdataTable.Clear(); |
||||
|
foreach (var item in loginfo.GetFiles()) |
||||
|
{ |
||||
|
DataRow FileRow = logdataTable.NewRow(); |
||||
|
FileRow["Name"] = item.Name; |
||||
|
FileRow["Length"] = item.Length / 1024; |
||||
|
FileRow["CreationTimeUtc"] = item.CreationTimeUtc; |
||||
|
FileRow["LastWriteTimeUtc"] = item.LastWriteTimeUtc; |
||||
|
logdataTable.Rows.Add(FileRow); |
||||
|
} |
||||
|
logdataTable.DefaultView.Sort = "Name DESC"; |
||||
|
gridLog.ItemsSource = logdataTable.DefaultView.ToTable().DefaultView; |
||||
|
} |
||||
|
|
||||
|
private void DataGrid_gridLogClick(object sender, MouseButtonEventArgs e)//数据表双击事件
|
||||
|
{ |
||||
|
int rownum = gridLog.SelectedIndex;//获取鼠标选中行并定义变量
|
||||
|
if (rownum != -1)//判断鼠标定位是否有效
|
||||
|
{ |
||||
|
/*定位选中行及指定列单元格文本信息*/ |
||||
|
//获取单元格内容
|
||||
|
var cellContent = gridLog.Columns[0].GetCellContent(gridLog.Items[rownum]); |
||||
|
//安全转换为 TextBlock 并获取文本
|
||||
|
var textValue = (cellContent as TextBlock)?.Text; |
||||
|
if (!string.IsNullOrEmpty(textValue)) |
||||
|
{ |
||||
|
LOGDATA_file(textValue);//
|
||||
|
} |
||||
|
|
||||
|
var Content_L = gridLog.Columns[1].GetCellContent(gridLog.Items[rownum]); |
||||
|
var Value_L = (Content_L as TextBlock)?.Text.ToString(); |
||||
|
if (!string.IsNullOrEmpty(Value_L)) |
||||
|
{ |
||||
|
TEXT_L = long.Parse(Value_L) * 1024 + 1024; |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/// <summary>
|
||||
|
/// 读取文件显示到前端
|
||||
|
/// 调用示例: LOGDATA_file(file path);
|
||||
|
/// </summary>
|
||||
|
/// <param name="path">文件路径</param>
|
||||
|
private async void LOGDATA_file(string dat) //读取文件显示到前端
|
||||
|
{ |
||||
|
Logtext.SelectAll(); |
||||
|
Logtext.Cut(); |
||||
|
Logtext.Document = new FlowDocument(); |
||||
|
string filePath = System.Environment.CurrentDirectory + "\\" + comboBox_log.Text + "\\"; |
||||
|
try |
||||
|
{ |
||||
|
Logtext.AppendText("Loading..."); |
||||
|
string log_ = ""; |
||||
|
string log_DAT = await Task<string>.Run(() => |
||||
|
{ |
||||
|
// 使用StreamReader读取文件
|
||||
|
using (StreamReader reader = new StreamReader(filePath + dat)) |
||||
|
{ |
||||
|
// 读取文件直到文件的末尾
|
||||
|
while (!reader.EndOfStream) |
||||
|
{ |
||||
|
// 添加文件的每一行到RichTextBox
|
||||
|
log_ = log_ + reader.ReadLine() + "\n"; |
||||
|
} |
||||
|
} |
||||
|
return log_; |
||||
|
}); |
||||
|
Logtext.SelectAll(); |
||||
|
Logtext.Cut(); |
||||
|
Logtext.AppendText(log_DAT); |
||||
|
} |
||||
|
catch (Exception ex) |
||||
|
{ |
||||
|
// 处理可能发生的任何异常
|
||||
|
MessageBox.Show("Error reading file: " + ex.Message); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
private void comboBox_log_DropDownClosed(object sender, EventArgs e) |
||||
|
{ |
||||
|
LOG_file(); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
@ -0,0 +1,231 @@ |
|||||
|
using CommunityToolkit.Mvvm; |
||||
|
using CommunityToolkit.Mvvm.ComponentModel; |
||||
|
using CommunityToolkit.Mvvm.Input; |
||||
|
using SunlightAggregationManager.UserClass; |
||||
|
using System; |
||||
|
using System.Collections.Generic; |
||||
|
using System.Data; |
||||
|
using System.Net; |
||||
|
using System.Reflection.PortableExecutable; |
||||
|
using System.Security.Cryptography; |
||||
|
using System.Text; |
||||
|
using System.Windows.Controls; |
||||
|
using System.Windows.Input; |
||||
|
using System.Xml.Linq; |
||||
|
|
||||
|
namespace SunlightAggregationManager.ViewModel |
||||
|
{ |
||||
|
public partial class MainWindowViewModel : ObservableObject |
||||
|
{ |
||||
|
//调用配置文件
|
||||
|
private static UserClass.IniFile.IniFiles Configini = new UserClass.IniFile.IniFiles(Convert.ToString(System.AppDomain.CurrentDomain.BaseDirectory) + "Configini.ini"); |
||||
|
public static string SQLIP=""; //读配置文件
|
||||
|
public static string SQLNAME=""; |
||||
|
public static string SQMOD= ""; |
||||
|
public static string SQLUSER = ""; |
||||
|
public static string SQLPASWORD = ""; |
||||
|
public static string MachineName = "SERVER"; |
||||
|
public static string TCP_E = "0"; |
||||
|
public static string HTTP_E = "0"; |
||||
|
public static string TLS_E = "0"; |
||||
|
public static UserClass.SqliteHelper SQLiteHelpers = null!; //定义数据库
|
||||
|
private readonly string DBAddress = Environment.CurrentDirectory + "\\DataBase\\SunlightAggregationManager.db"; //数据库路径
|
||||
|
|
||||
|
public static DataTable ActionLog = new DataTable(); |
||||
|
|
||||
|
[ObservableProperty] |
||||
|
public static DataTable _machines = new DataTable(); //设备表缓存
|
||||
|
[ObservableProperty] |
||||
|
public static DataTable _userData = new DataTable(); |
||||
|
|
||||
|
|
||||
|
public MainWindowViewModel() |
||||
|
{ |
||||
|
SQLIP = Configini.IniReadvalue("SQL_SERVER", "SQL1"); //读配置文件
|
||||
|
SQLNAME = Configini.IniReadvalue("SQL_SERVER", "SQL2"); |
||||
|
SQMOD = Configini.IniReadvalue("SQL_SERVER", "SQL3"); |
||||
|
SQLUSER = Configini.IniReadvalue("SQL_SERVER", "SQL4"); |
||||
|
SQLPASWORD = Configini.IniReadvalue("SQL_SERVER", "SQL5"); |
||||
|
|
||||
|
MachineName = Configini.IniReadvalue("SYS", "Name"); |
||||
|
|
||||
|
TCP_E = Configini.IniReadvalue("NETWORK", "TCP"); |
||||
|
HTTP_E = Configini.IniReadvalue("NETWORK", "HTTP"); |
||||
|
TLS_E = Configini.IniReadvalue("NETWORK", "TLS"); |
||||
|
|
||||
|
//本地数据库(sqlite)
|
||||
|
try |
||||
|
{ |
||||
|
SQLiteHelpers = new UserClass.SqliteHelper(DBAddress); //数据库连接路径
|
||||
|
SQLiteHelpers.Open(); //打开数据库
|
||||
|
UserData = SQLiteHelpers.ExecuteDataSet("select * from USER order by UserID asc", null)?.Tables[0] ?? new DataTable(); |
||||
|
Machines = SQLiteHelpers.ExecuteDataSet("select * from Machines order by ID asc", null)?.Tables[0] ?? new DataTable(); |
||||
|
ActionLog = (SQLiteHelpers.ExecuteDataSet("select * from ActionLog", null)?.Tables[0] ?? new DataTable()).Clone(); |
||||
|
// SQLiteHelpers.Close();
|
||||
|
|
||||
|
if (UserData.Columns.Contains("UserID")) |
||||
|
{ |
||||
|
UserData.Columns.Remove("UserID"); |
||||
|
} |
||||
|
DataRow[] dataRows = UserData.Select("State<90"); |
||||
|
foreach (DataRow row in dataRows) |
||||
|
{ |
||||
|
row["State"] = 0; |
||||
|
row["LinkID"] = 0; |
||||
|
} |
||||
|
|
||||
|
|
||||
|
UserData.RowChanged += UserData_Updata;//注册userdata表更新事件
|
||||
|
|
||||
|
} |
||||
|
catch (Exception ex) |
||||
|
{ |
||||
|
Console.WriteLine(ex.ToString()); |
||||
|
} |
||||
|
|
||||
|
//运行数据库
|
||||
|
DataBase dataBase = new DataBase(); |
||||
|
dataBase.Config(SQLIP, SQLNAME, SQMOD, SQLUSER, SQLPASWORD); |
||||
|
|
||||
|
if (TCP_E == "1")//启动tcp(内网直连)
|
||||
|
{ |
||||
|
int P1, P2; |
||||
|
try |
||||
|
{ |
||||
|
P1 =int.Parse( Configini.IniReadvalue("NETWORK", "TCP_PORT1")); |
||||
|
} |
||||
|
catch (Exception) { P1 = 7789; } |
||||
|
try |
||||
|
{ |
||||
|
P2 = int.Parse(Configini.IniReadvalue("NETWORK", "TCP_PORT2")); |
||||
|
} |
||||
|
catch (Exception) { P2 = 7790; } |
||||
|
|
||||
|
using var _ = AsyncTcpServer.TcpMain(P1,P2); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
|
||||
|
private void UserData_Updata(object sender, DataRowChangeEventArgs e) |
||||
|
{ |
||||
|
Dictionary<string, object> row_entity = new Dictionary<string, object>(); |
||||
|
Dictionary<string, object> userlog = new Dictionary<string, object>(); |
||||
|
|
||||
|
if (e.Action == DataRowAction.Change) |
||||
|
{ |
||||
|
//修改的行回传
|
||||
|
foreach (DataColumn col in UserData.Columns) |
||||
|
{ |
||||
|
row_entity.Add(col.ColumnName, e.Row.Field<object>(col.ColumnName) ?? DBNull.Value);//传入字段和值(值允许为空)
|
||||
|
|
||||
|
if (ActionLog.Columns.Contains(col.ColumnName))//添加到行为记录表
|
||||
|
{ |
||||
|
userlog.Add(col.ColumnName, e.Row.Field<object>(col.ColumnName) ?? DBNull.Value);//传入字段和值(值允许为空)
|
||||
|
} |
||||
|
} |
||||
|
SQLiteHelpers.Update("USER", row_entity,"User='"+ e.Row.Field<object>("User")+"'", null); |
||||
|
|
||||
|
userlog.Add("Action", "Command"); |
||||
|
userlog.Add("Time", DateTime.Now.ToString("yyyy-MM-dd dd:HH:mm:ss:fff")); |
||||
|
SQLiteHelpers.InsertData("ActionLog", userlog); |
||||
|
|
||||
|
} |
||||
|
if (e.Action == DataRowAction.Add) |
||||
|
{ |
||||
|
//添加的行回传
|
||||
|
foreach (DataColumn col in UserData.Columns) |
||||
|
{ |
||||
|
row_entity.Add(col.ColumnName, e.Row.Field<object>(col.ColumnName) ?? DBNull.Value);//传入字段和值(值允许为空)
|
||||
|
} |
||||
|
SQLiteHelpers.InsertData("USER", row_entity); |
||||
|
|
||||
|
userlog.Add("Time", DateTime.Now.ToString("yyyy-MM-dd dd:HH:mm:ss:fff")); |
||||
|
userlog.Add("Action", "New Account"); |
||||
|
userlog.Add("Name", MachineName); |
||||
|
userlog.Add("Terminal", "Terminal"); |
||||
|
userlog.Add("Note", "User = "+ e.Row.Field<object>("User")); |
||||
|
SQLiteHelpers.InsertData("ActionLog", userlog); |
||||
|
} |
||||
|
if (e.Action == DataRowAction.Delete) |
||||
|
{ |
||||
|
//删除的行回传
|
||||
|
SQLiteHelpers.Delete("USER", "User='" + e.Row.Field<object>("User") + "'", null); |
||||
|
|
||||
|
userlog.Add("Time", DateTime.Now.ToString("yyyy-MM-dd dd:HH:mm:ss:fff")); |
||||
|
userlog.Add("Action", "Delete Account"); |
||||
|
userlog.Add("Name", MachineName); |
||||
|
userlog.Add("Terminal", "Terminal"); |
||||
|
userlog.Add("Note", "User = " + e.Row.Field<object>("User")); |
||||
|
SQLiteHelpers.InsertData("ActionLog", userlog); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
public static object? Selet_Memory(DataTable DB, string name, string? key)//查询
|
||||
|
{ |
||||
|
try |
||||
|
{ |
||||
|
lock (DB) |
||||
|
{ |
||||
|
DataRow drEmployee = DB.Select(key).First(); |
||||
|
object? index = drEmployee.Field<object>(name); |
||||
|
return index; |
||||
|
} |
||||
|
} |
||||
|
catch (Exception ex) |
||||
|
{ |
||||
|
Console.WriteLine("SDTD:" + ex.ToString()); |
||||
|
return null; |
||||
|
} |
||||
|
} |
||||
|
public static void Updata_Memory(DataTable DB, string key, string? Value, string name)//更新数据
|
||||
|
{ |
||||
|
try |
||||
|
{ |
||||
|
lock (DB) |
||||
|
{ |
||||
|
var dr = DB.Select(key); |
||||
|
if (dr.Length > 0) |
||||
|
{ |
||||
|
DataRow drEmployee = dr.First(); |
||||
|
drEmployee.BeginEdit(); |
||||
|
drEmployee[name] = Value; |
||||
|
drEmployee.EndEdit(); |
||||
|
drEmployee.AcceptChanges(); |
||||
|
// drEmployee.ClearErrors();
|
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
catch (Exception ex) |
||||
|
{ |
||||
|
Console.WriteLine("SDTD:" + ex.ToString()); |
||||
|
} |
||||
|
} |
||||
|
public static void Updata_Memory(DataTable DB, string key, Dictionary<string, object> Value)//更新数据
|
||||
|
{ |
||||
|
try |
||||
|
{ |
||||
|
lock (DB) |
||||
|
{ |
||||
|
var dr = DB.Select(key); |
||||
|
if (dr.Length > 0) |
||||
|
{ |
||||
|
DataRow drEmployee = dr.First(); |
||||
|
drEmployee.BeginEdit(); |
||||
|
|
||||
|
foreach (KeyValuePair<string, object> kvp in Value) |
||||
|
{ |
||||
|
drEmployee[kvp.Key] = kvp.Value; |
||||
|
} |
||||
|
drEmployee.EndEdit(); |
||||
|
drEmployee.AcceptChanges(); |
||||
|
// drEmployee.ClearErrors();
|
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
catch (Exception ex) |
||||
|
{ |
||||
|
Console.WriteLine("SDTD:" + ex.ToString()); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
} |
||||
Loading…
Reference in new issue