继续修改触发器(未完成,修改一个触发器可以添加多个变量)

This commit is contained in:
2025-09-22 22:58:51 +08:00
parent 0f869cf410
commit 042bc15288
14 changed files with 434 additions and 121 deletions

23
CHANGELOG.md Normal file
View File

@@ -0,0 +1,23 @@
# 更新日志
## v1.1.0 (2025-09-22)
### 新增功能
- **触发器系统重构**:将触发器与变量的关联关系从一对一改为多对多
- 修改了 `TriggerDefinition` 领域模型,将 `VariableId` 属性改为 `VariableIds` 列表
- 添加了新的数据库实体 `DbTriggerVariable` 来维护多对多关系
- 更新了数据库映射配置和仓储实现
- 修改了 DTO 和服务层以支持新的多对多关系
- 更新了 WPF UI 以支持多变量选择
### 技术实现细节
- 添加了新的 `TriggerVariables` 数据库表来维护触发器与变量的多对多关系
- 更新了 `TriggerRepository` 以处理新的关联表
- 修改了 `TriggerManagementService` 的验证逻辑,确保至少选择一个变量
- 更新了 WPF 视图模型以支持多选变量
- 提供了数据库迁移脚本以更新现有数据库结构
### 兼容性说明
- 此更新涉及数据库结构变更,需要运行迁移脚本
- 现有的触发器数据将被迁移到新的表结构中
- API 接口保持向后兼容,但返回的数据结构已更新

View File

@@ -1,3 +1,4 @@
using System.Collections.Generic;
using DMS.Core.Models.Triggers;
// 引入枚举
@@ -15,9 +16,9 @@ namespace DMS.Application.DTOs
public int Id { get; set; }
/// <summary>
/// 关联的变量 ID
/// 关联的变量 ID 列表
/// </summary>
public int VariableId { get; set; }
public List<int> VariableIds { get; set; } = new List<int>();
/// <summary>
/// 触发器是否处于激活状态

View File

@@ -57,6 +57,9 @@ public class MappingProfile : Profile
CreateMap<EmailTemplate, EmailTemplateDto>().ReverseMap();
CreateMap<EmailLog, EmailLogDto>().ReverseMap();
CreateMap<TriggerDefinition, TriggerDefinitionDto>().ReverseMap();
CreateMap<TriggerDefinition, TriggerDefinitionDto>()
.ForMember(dest => dest.VariableIds, opt => opt.MapFrom(src => src.VariableIds))
.ReverseMap()
.ForMember(dest => dest.VariableIds, opt => opt.MapFrom(src => src.VariableIds));
}
}

View File

@@ -109,6 +109,10 @@ namespace DMS.Application.Services.Triggers.Impl
/// </summary>
private void ValidateTriggerDto(TriggerDefinitionDto dto)
{
// 检查是否至少关联了一个变量
if (dto.VariableIds == null || !dto.VariableIds.Any())
throw new ArgumentException("触发器必须至少关联一个变量。");
// 添加必要的验证逻辑
switch (dto.Condition)
{

View File

@@ -1,4 +1,5 @@
using System;
using System.Collections.Generic;
namespace DMS.Core.Models.Triggers
{
@@ -37,9 +38,9 @@ namespace DMS.Core.Models.Triggers
public int Id { get; set; }
/// <summary>
/// 关联的变量 ID
/// 关联的变量列表
/// </summary>
public int VariableId { get; set; }
public List<int> VariableIds { get; set; } = new List<int>();
/// <summary>
/// 触发器是否处于激活状态

View File

@@ -1,4 +1,5 @@
using System;
using System.Collections.Generic;
using DMS.Core.Models.Triggers;
using SqlSugar;
using SqlSugar.DbConvert;
@@ -17,11 +18,6 @@ public class DbTriggerDefinition
[SugarColumn(IsPrimaryKey = true, IsIdentity = true)]
public int Id { get; set; }
/// <summary>
/// 关联的变量 ID。
/// </summary>
public int VariableId { get; set; }
/// <summary>
/// 触发器是否处于激活状态。
/// </summary>
@@ -96,4 +92,10 @@ public class DbTriggerDefinition
/// 最后更新时间。
/// </summary>
public DateTime UpdatedAt { get; set; } = DateTime.UtcNow;
/// <summary>
/// 关联的变量 ID 列表(通过中间表关联)。
/// </summary>
[SugarColumn(IsIgnore = true)]
public List<int> VariableIds { get; set; } = new List<int>();
}

View File

@@ -0,0 +1,22 @@
using SqlSugar;
namespace DMS.Infrastructure.Entities;
/// <summary>
/// 数据库实体:表示触发器与变量的多对多关联关系。
/// </summary>
public class DbTriggerVariable
{
[SugarColumn(IsPrimaryKey = true, IsIdentity = true)]
public int Id { get; set; }
/// <summary>
/// 外键,指向 TriggerDefinitions 表的 Id。
/// </summary>
public int TriggerDefinitionId { get; set; }
/// <summary>
/// 外键,指向 Variables 表的 Id。
/// </summary>
public int VariableId { get; set; }
}

View File

@@ -0,0 +1,23 @@
-- 创建触发器与变量关联表
CREATE TABLE TriggerVariables (
Id INTEGER PRIMARY KEY AUTOINCREMENT,
TriggerDefinitionId INTEGER NOT NULL,
VariableId INTEGER NOT NULL,
FOREIGN KEY (TriggerDefinitionId) REFERENCES TriggerDefinitions(Id) ON DELETE CASCADE,
FOREIGN KEY (VariableId) REFERENCES Variables(Id) ON DELETE CASCADE
);
-- 创建索引以提高查询性能
CREATE INDEX IX_TriggerVariables_TriggerDefinitionId ON TriggerVariables (TriggerDefinitionId);
CREATE INDEX IX_TriggerVariables_VariableId ON TriggerVariables (VariableId);
-- 迁移现有数据将TriggerDefinitions表中的VariableId迁移到新的关联表中
INSERT INTO TriggerVariables (TriggerDefinitionId, VariableId)
SELECT Id, VariableId
FROM TriggerDefinitions
WHERE VariableId IS NOT NULL AND VariableId > 0;
-- 删除TriggerDefinitions表中的VariableId列如果数据库支持
-- 注意SQLite不支持直接删除列所以这里只是注释说明
-- 在支持的数据库中,可以使用以下语句:
-- ALTER TABLE TriggerDefinitions DROP COLUMN VariableId;

View File

@@ -50,10 +50,12 @@ public class MappingProfile : Profile
opt => opt.MapFrom(src => src.SuppressionDurationTicks.HasValue ?
TimeSpan.FromTicks(src.SuppressionDurationTicks.Value) :
(TimeSpan?)null))
.ForMember(dest => dest.VariableIds, opt => opt.MapFrom(src => src.VariableIds))
.ReverseMap()
.ForMember(dest => dest.SuppressionDurationTicks,
opt => opt.MapFrom(src => src.SuppressionDuration.HasValue ?
src.SuppressionDuration.Value.Ticks :
(long?)null));
(long?)null))
.ForMember(dest => dest.VariableIds, opt => opt.MapFrom(src => src.VariableIds));
}
}

View File

@@ -34,6 +34,16 @@ namespace DMS.Infrastructure.Repositories
public async Task<IEnumerable<TriggerDefinition>> GetAllAsync()
{
var dbList = await base.GetAllAsync();
// 加载关联的变量ID
foreach (var dbTrigger in dbList)
{
var variableIds = await _dbContext.GetInstance()
.Queryable<DbTriggerVariable>()
.Where(tv => tv.TriggerDefinitionId == dbTrigger.Id)
.Select(tv => tv.VariableId)
.ToListAsync();
dbTrigger.VariableIds = variableIds;
}
return _mapper.Map<List<TriggerDefinition>>(dbList);
}
@@ -43,6 +53,16 @@ namespace DMS.Infrastructure.Repositories
public async Task<TriggerDefinition?> GetByIdAsync(int id)
{
var dbTrigger = await base.GetByIdAsync(id);
if (dbTrigger != null)
{
// 加载关联的变量ID
var variableIds = await _dbContext.GetInstance()
.Queryable<DbTriggerVariable>()
.Where(tv => tv.TriggerDefinitionId == dbTrigger.Id)
.Select(tv => tv.VariableId)
.ToListAsync();
dbTrigger.VariableIds = variableIds;
}
return _mapper.Map<TriggerDefinition>(dbTrigger);
}
@@ -51,7 +71,21 @@ namespace DMS.Infrastructure.Repositories
/// </summary>
public async Task<TriggerDefinition> AddAsync(TriggerDefinition trigger)
{
var dbTrigger = await base.AddAsync(_mapper.Map<DbTriggerDefinition>(trigger));
var dbTrigger = _mapper.Map<DbTriggerDefinition>(trigger);
dbTrigger = await base.AddAsync(dbTrigger);
// 保存关联的变量ID
if (trigger.VariableIds != null && trigger.VariableIds.Any())
{
var triggerVariables = trigger.VariableIds.Select(variableId => new DbTriggerVariable
{
TriggerDefinitionId = dbTrigger.Id,
VariableId = variableId
}).ToList();
await _dbContext.GetInstance().Insertable(triggerVariables).ExecuteCommandAsync();
}
return _mapper.Map(dbTrigger, trigger);
}
@@ -60,8 +94,33 @@ namespace DMS.Infrastructure.Repositories
/// </summary>
public async Task<TriggerDefinition?> UpdateAsync(TriggerDefinition trigger)
{
var rowsAffected = await base.UpdateAsync(_mapper.Map<DbTriggerDefinition>(trigger));
return rowsAffected > 0 ? trigger : null;
var dbTrigger = _mapper.Map<DbTriggerDefinition>(trigger);
var rowsAffected = await base.UpdateAsync(dbTrigger);
if (rowsAffected > 0)
{
// 删除旧的关联关系
await _dbContext.GetInstance()
.Deleteable<DbTriggerVariable>()
.Where(tv => tv.TriggerDefinitionId == dbTrigger.Id)
.ExecuteCommandAsync();
// 插入新的关联关系
if (trigger.VariableIds != null && trigger.VariableIds.Any())
{
var triggerVariables = trigger.VariableIds.Select(variableId => new DbTriggerVariable
{
TriggerDefinitionId = dbTrigger.Id,
VariableId = variableId
}).ToList();
await _dbContext.GetInstance().Insertable(triggerVariables).ExecuteCommandAsync();
}
return trigger;
}
return null;
}
/// <summary>
@@ -71,6 +130,14 @@ namespace DMS.Infrastructure.Repositories
{
var stopwatch = new Stopwatch();
stopwatch.Start();
// 先删除关联的变量关系
await _dbContext.GetInstance()
.Deleteable<DbTriggerVariable>()
.Where(tv => tv.TriggerDefinitionId == id)
.ExecuteCommandAsync();
// 再删除触发器本身
var rowsAffected = await _dbContext.GetInstance().Deleteable<DbTriggerDefinition>()
.In(id)
.ExecuteCommandAsync();
@@ -86,9 +153,34 @@ namespace DMS.Infrastructure.Repositories
{
var stopwatch = new Stopwatch();
stopwatch.Start();
var dbList = await _dbContext.GetInstance().Queryable<DbTriggerDefinition>()
.Where(t => t.VariableId == variableId)
// 先查询关联表获取触发器ID
var triggerIds = await _dbContext.GetInstance()
.Queryable<DbTriggerVariable>()
.Where(tv => tv.VariableId == variableId)
.Select(tv => tv.TriggerDefinitionId)
.ToListAsync();
// 再查询触发器定义
var dbList = new List<DbTriggerDefinition>();
if (triggerIds.Any())
{
dbList = await _dbContext.GetInstance().Queryable<DbTriggerDefinition>()
.In(it => it.Id, triggerIds)
.ToListAsync();
// 加载每个触发器的变量ID列表
foreach (var dbTrigger in dbList)
{
var variableIds = await _dbContext.GetInstance()
.Queryable<DbTriggerVariable>()
.Where(tv => tv.TriggerDefinitionId == dbTrigger.Id)
.Select(tv => tv.VariableId)
.ToListAsync();
dbTrigger.VariableIds = variableIds;
}
}
stopwatch.Stop();
_logger.LogInformation($"GetByVariableId {typeof(DbTriggerDefinition).Name},VariableId={variableId},耗时:{stopwatch.ElapsedMilliseconds}ms");
return _mapper.Map<List<TriggerDefinition>>(dbList);

View File

@@ -1,3 +1,4 @@
using System.Collections.ObjectModel;
using System.Text.Json;
using System.Windows;
using CommunityToolkit.Mvvm.ComponentModel;
@@ -5,8 +6,10 @@ using CommunityToolkit.Mvvm.Input;
using DMS.Application.DTOs;
using DMS.Application.Interfaces;
using DMS.Application.Interfaces.Database;
using DMS.Core.Interfaces;
using DMS.Core.Models.Triggers;
using DMS.WPF.Interfaces;
using DMS.WPF.ViewModels.Items;
namespace DMS.WPF.ViewModels.Dialogs
{
@@ -17,14 +20,21 @@ namespace DMS.WPF.ViewModels.Dialogs
{
private readonly IVariableAppService _variableAppService; // To populate variable selection dropdown
private readonly IDialogService _dialogService;
private readonly IDataStorageService _dataStorageService;
private readonly INotificationService _notificationService;
[ObservableProperty]
private string _searchText = "";
[ObservableProperty]
private TriggerDefinitionDto _trigger = new();
[ObservableProperty]
private List<VariableDto> _availableVariables = new();
[ObservableProperty]
private ObservableCollection<VariableItemViewModel> _selectedVariables = new();
// Properties for easier binding in XAML for SendEmail action config
[ObservableProperty]
private string _emailRecipients = "";
@@ -38,13 +48,29 @@ namespace DMS.WPF.ViewModels.Dialogs
public TriggerDialogViewModel(
IVariableAppService variableAppService,
IDialogService dialogService,
IDataStorageService dataStorageService,
INotificationService notificationService)
{
_variableAppService = variableAppService ?? throw new ArgumentNullException(nameof(variableAppService));
_dialogService = dialogService ?? throw new ArgumentNullException(nameof(dialogService));
_dataStorageService = dataStorageService;
_notificationService = notificationService ?? throw new ArgumentNullException(nameof(notificationService));
}
partial void OnSearchTextChanged(string searchText)
{
SelectedVariables.Clear();
foreach (var variableKv in _dataStorageService.Variables)
{
if (variableKv.Value.Name.Contains(SearchText))
{
SelectedVariables.Add(variableKv.Value);
}
}
}
/// <summary>
/// 初始化视图模型(传入待编辑的触发器)
/// </summary>
@@ -60,6 +86,19 @@ namespace DMS.WPF.ViewModels.Dialogs
// Load available variables for selection dropdown
await LoadVariablesAsync();
// Load selected variables
if (Trigger.VariableIds != null && Trigger.VariableIds.Any())
{
foreach (var variableId in Trigger.VariableIds)
{
var variable = AvailableVariables.FirstOrDefault(v => v.Id == variableId);
if (variable != null)
{
// SelectedVariables.Add(variable);
}
}
}
// Parse action configuration if it's SendEmail
if (Trigger.Action == ActionType.SendEmail && !string.IsNullOrEmpty(Trigger.ActionConfigurationJson))
{
@@ -109,9 +148,9 @@ namespace DMS.WPF.ViewModels.Dialogs
private async Task SaveAsync()
{
// Basic validation
if (Trigger.VariableId == default(int))
if (SelectedVariables == null || !SelectedVariables.Any())
{
_notificationService.ShowWarn("请选择关联的变量");
_notificationService.ShowWarn("请至少选择一个关联的变量");
return;
}
@@ -121,6 +160,9 @@ namespace DMS.WPF.ViewModels.Dialogs
return;
}
// 设置选中的变量ID
Trigger.VariableIds = SelectedVariables.Select(v => v.Id).ToList();
// Validate condition-specific fields
switch (Trigger.Condition)
{

View File

@@ -107,7 +107,7 @@ namespace DMS.WPF.ViewModels.Triggers
var triggerToEdit = new TriggerDefinitionDto
{
Id = SelectedTrigger.Id,
VariableId = SelectedTrigger.VariableId,
VariableIds = new List<int>(SelectedTrigger.VariableIds),
IsActive = SelectedTrigger.IsActive,
Condition = SelectedTrigger.Condition,
Threshold = SelectedTrigger.Threshold,

View File

@@ -5,13 +5,17 @@
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:ui="http://schemas.inkore.net/lib/ui/wpf/modern"
xmlns:enums="clr-namespace:DMS.Core.Models.Triggers;assembly=DMS.Core"
xmlns:vc="clr-namespace:DMS.WPF.ValueConverts"
xmlns:vmd="clr-namespace:DMS.WPF.ViewModels.Dialogs"
xmlns:hc="https://handyorg.github.io/handycontrol"
xmlns:ex="clr-namespace:DMS.Extensions"
xmlns:converters="clr-namespace:DMS.WPF.Converters"
Title="{Binding Title}"
mc:Ignorable="d"
d:DesignHeight="600" d:DesignWidth="600"
MinWidth="500" MinHeight="500"
d:DesignHeight="600"
d:DesignWidth="600"
d:DataContext="{d:DesignInstance vmd:TriggerDialogViewModel}"
MinWidth="500"
MinHeight="500"
PrimaryButtonText="{Binding PrimaryButText}"
CloseButtonText="取消"
PrimaryButtonCommand="{Binding SaveCommand}"
@@ -19,155 +23,239 @@
DefaultButton="Primary">
<ui:ContentDialog.Resources>
<converters:EnumToVisibilityConverter x:Key="LocalEnumToVisibilityConverter"/>
<ex:EnumBindingSource x:Key="ConditionTypeEnum" EnumType="{x:Type enums:ConditionType}" />
<ex:EnumBindingSource x:Key="ActionTypeEnum" EnumType="{x:Type enums:ActionType}" />
<converters:EnumToVisibilityConverter x:Key="LocalEnumToVisibilityConverter" />
<ex:EnumBindingSource x:Key="ConditionTypeEnum"
EnumType="{x:Type enums:ConditionType}" />
<ex:EnumBindingSource x:Key="ActionTypeEnum"
EnumType="{x:Type enums:ActionType}" />
</ui:ContentDialog.Resources>
<ScrollViewer VerticalScrollBarVisibility="Auto">
<StackPanel Margin="10">
<!-- Basic Info Section -->
<GroupBox Header="基本信息" Padding="5">
<GroupBox Header="基本信息"
Padding="5">
<StackPanel>
<DockPanel Margin="0,0,0,5">
<Label Content="关联变量:" Width="100" VerticalAlignment="Center"/>
<ComboBox ItemsSource="{Binding AvailableVariables}"
DisplayMemberPath="Name"
SelectedValuePath="Id"
SelectedValue="{Binding Trigger.VariableId}"
Width="200" HorizontalAlignment="Left"/>
</DockPanel>
<StackPanel Orientation="Horizontal">
<StackPanel>
<hc:TextBox hc:InfoElement.Title="搜索变量:"
Text="{Binding SearchText, UpdateSourceTrigger=PropertyChanged }">
</hc:TextBox>
<DataGrid
AutoGenerateColumns="False"
CanUserAddRows="False"
ItemsSource="{Binding SelectedVariables}">
<DataGrid.Columns>
<DataGridTextColumn Binding="{Binding Name}" Header="变量名"></DataGridTextColumn>
</DataGrid.Columns>
</DataGrid>
</StackPanel>
</StackPanel>
<DockPanel Margin="0,0,0,5">
<Label Content="描述:" Width="100" VerticalAlignment="Center"/>
<Label Content="描述:"
Width="100"
VerticalAlignment="Center" />
<TextBox Text="{Binding Trigger.Description, UpdateSourceTrigger=PropertyChanged}"
Width="300" HorizontalAlignment="Left"/>
Width="300"
HorizontalAlignment="Left" />
</DockPanel>
<CheckBox Content="激活" IsChecked="{Binding Trigger.IsActive}" Margin="0,0,0,5"/>
<CheckBox Content="激活"
IsChecked="{Binding Trigger.IsActive}"
Margin="0,0,0,5" />
</StackPanel>
</GroupBox>
<!-- Condition Section -->
<GroupBox Header="触发条件" Padding="5" Margin="0,10,0,0">
<GroupBox Header="触发条件"
Padding="5"
Margin="0,10,0,0">
<StackPanel>
<DockPanel Margin="0,0,0,5">
<Label Content="条件类型:" Width="100" VerticalAlignment="Center"/>
<Label Content="条件类型:"
Width="100"
VerticalAlignment="Center" />
<ComboBox ItemsSource="{Binding Source={StaticResource ConditionTypeEnum}}"
SelectedItem="{Binding Trigger.Condition}"
Width="200" HorizontalAlignment="Left"/>
Width="200"
HorizontalAlignment="Left" />
</DockPanel>
<!-- Conditional Fields based on Condition Type -->
<StackPanel Visibility="{Binding Trigger.Condition, Converter={StaticResource LocalEnumToVisibilityConverter}, ConverterParameter=GreaterThan}">
<StackPanel
Visibility="{Binding Trigger.Condition, Converter={StaticResource LocalEnumToVisibilityConverter}, ConverterParameter=GreaterThan}">
<DockPanel Margin="0,0,0,5">
<Label Content="阈值:" Width="100" VerticalAlignment="Center"/>
<Label Content="阈值:"
Width="100"
VerticalAlignment="Center" />
<TextBox Text="{Binding Trigger.Threshold, UpdateSourceTrigger=PropertyChanged}"
Width="100" HorizontalAlignment="Left"/>
Width="100"
HorizontalAlignment="Left" />
</DockPanel>
</StackPanel>
<StackPanel Visibility="{Binding Trigger.Condition, Converter={StaticResource LocalEnumToVisibilityConverter}, ConverterParameter=LessThan}">
<StackPanel
Visibility="{Binding Trigger.Condition, Converter={StaticResource LocalEnumToVisibilityConverter}, ConverterParameter=LessThan}">
<DockPanel Margin="0,0,0,5">
<Label Content="阈值:" Width="100" VerticalAlignment="Center"/>
<Label Content="阈值:"
Width="100"
VerticalAlignment="Center" />
<TextBox Text="{Binding Trigger.Threshold, UpdateSourceTrigger=PropertyChanged}"
Width="100" HorizontalAlignment="Left"/>
Width="100"
HorizontalAlignment="Left" />
</DockPanel>
</StackPanel>
<StackPanel Visibility="{Binding Trigger.Condition, Converter={StaticResource LocalEnumToVisibilityConverter}, ConverterParameter=EqualTo}">
<StackPanel
Visibility="{Binding Trigger.Condition, Converter={StaticResource LocalEnumToVisibilityConverter}, ConverterParameter=EqualTo}">
<DockPanel Margin="0,0,0,5">
<Label Content="阈值:" Width="100" VerticalAlignment="Center"/>
<Label Content="阈值:"
Width="100"
VerticalAlignment="Center" />
<TextBox Text="{Binding Trigger.Threshold, UpdateSourceTrigger=PropertyChanged}"
Width="100" HorizontalAlignment="Left"/>
Width="100"
HorizontalAlignment="Left" />
</DockPanel>
</StackPanel>
<StackPanel Visibility="{Binding Trigger.Condition, Converter={StaticResource LocalEnumToVisibilityConverter}, ConverterParameter=NotEqualTo}">
<StackPanel
Visibility="{Binding Trigger.Condition, Converter={StaticResource LocalEnumToVisibilityConverter}, ConverterParameter=NotEqualTo}">
<DockPanel Margin="0,0,0,5">
<Label Content="阈值:" Width="100" VerticalAlignment="Center"/>
<Label Content="阈值:"
Width="100"
VerticalAlignment="Center" />
<TextBox Text="{Binding Trigger.Threshold, UpdateSourceTrigger=PropertyChanged}"
Width="100" HorizontalAlignment="Left"/>
Width="100"
HorizontalAlignment="Left" />
</DockPanel>
</StackPanel>
<StackPanel Visibility="{Binding Trigger.Condition, Converter={StaticResource LocalEnumToVisibilityConverter}, ConverterParameter=InRange}">
<StackPanel
Visibility="{Binding Trigger.Condition, Converter={StaticResource LocalEnumToVisibilityConverter}, ConverterParameter=InRange}">
<DockPanel Margin="0,0,0,5">
<Label Content="下限:" Width="100" VerticalAlignment="Center"/>
<Label Content="下限:"
Width="100"
VerticalAlignment="Center" />
<TextBox Text="{Binding Trigger.LowerBound, UpdateSourceTrigger=PropertyChanged}"
Width="100" HorizontalAlignment="Left"/>
Width="100"
HorizontalAlignment="Left" />
</DockPanel>
<DockPanel Margin="0,0,0,5">
<Label Content="上限:" Width="100" VerticalAlignment="Center"/>
<Label Content="上限:"
Width="100"
VerticalAlignment="Center" />
<TextBox Text="{Binding Trigger.UpperBound, UpdateSourceTrigger=PropertyChanged}"
Width="100" HorizontalAlignment="Left"/>
Width="100"
HorizontalAlignment="Left" />
</DockPanel>
</StackPanel>
<StackPanel Visibility="{Binding Trigger.Condition, Converter={StaticResource LocalEnumToVisibilityConverter}, ConverterParameter=OutOfRange}">
<StackPanel
Visibility="{Binding Trigger.Condition, Converter={StaticResource LocalEnumToVisibilityConverter}, ConverterParameter=OutOfRange}">
<DockPanel Margin="0,0,0,5">
<Label Content="下限:" Width="100" VerticalAlignment="Center"/>
<Label Content="下限:"
Width="100"
VerticalAlignment="Center" />
<TextBox Text="{Binding Trigger.LowerBound, UpdateSourceTrigger=PropertyChanged}"
Width="100" HorizontalAlignment="Left"/>
Width="100"
HorizontalAlignment="Left" />
</DockPanel>
<DockPanel Margin="0,0,0,5">
<Label Content="上限:" Width="100" VerticalAlignment="Center"/>
<Label Content="上限:"
Width="100"
VerticalAlignment="Center" />
<TextBox Text="{Binding Trigger.UpperBound, UpdateSourceTrigger=PropertyChanged}"
Width="100" HorizontalAlignment="Left"/>
Width="100"
HorizontalAlignment="Left" />
</DockPanel>
</StackPanel>
</StackPanel>
</GroupBox>
<!-- Action Section -->
<GroupBox Header="触发动作" Padding="5" Margin="0,10,0,0">
<GroupBox Header="触发动作"
Padding="5"
Margin="0,10,0,0">
<StackPanel>
<DockPanel Margin="0,0,0,5">
<Label Content="动作类型:" Width="100" VerticalAlignment="Center"/>
<Label Content="动作类型:"
Width="100"
VerticalAlignment="Center" />
<ComboBox ItemsSource="{Binding Source={StaticResource ActionTypeEnum}}"
SelectedItem="{Binding Trigger.Action}"
Width="200" HorizontalAlignment="Left"/>
Width="200"
HorizontalAlignment="Left" />
</DockPanel>
<!-- Conditional Fields based on Action Type -->
<StackPanel Visibility="{Binding Trigger.Action, Converter={StaticResource LocalEnumToVisibilityConverter}, ConverterParameter=SendEmail}">
<StackPanel
Visibility="{Binding Trigger.Action, Converter={StaticResource LocalEnumToVisibilityConverter}, ConverterParameter=SendEmail}">
<DockPanel Margin="0,0,0,5">
<Label Content="收件人 (分号分隔):" Width="150" VerticalAlignment="Top"/>
<TextBox Text="{Binding EmailRecipients, UpdateSourceTrigger=PropertyChanged, ValidatesOnDataErrors=True}"
AcceptsReturn="True" TextWrapping="Wrap"
Width="350" Height="60" HorizontalAlignment="Left"/>
<Label Content="收件人 (分号分隔):"
Width="150"
VerticalAlignment="Top" />
<TextBox
Text="{Binding EmailRecipients, UpdateSourceTrigger=PropertyChanged, ValidatesOnDataErrors=True}"
AcceptsReturn="True"
TextWrapping="Wrap"
Width="350"
Height="60"
HorizontalAlignment="Left" />
</DockPanel>
<DockPanel Margin="0,0,0,5">
<Label Content="邮件主题模板:" Width="150" VerticalAlignment="Center"/>
<TextBox Text="{Binding EmailSubjectTemplate, UpdateSourceTrigger=PropertyChanged, ValidatesOnDataErrors=True}"
Width="350" HorizontalAlignment="Left"/>
<Label Content="邮件主题模板:"
Width="150"
VerticalAlignment="Center" />
<TextBox
Text="{Binding EmailSubjectTemplate, UpdateSourceTrigger=PropertyChanged, ValidatesOnDataErrors=True}"
Width="350"
HorizontalAlignment="Left" />
</DockPanel>
<DockPanel Margin="0,0,0,5">
<Label Content="邮件内容模板:" Width="150" VerticalAlignment="Top"/>
<TextBox Text="{Binding EmailBodyTemplate, UpdateSourceTrigger=PropertyChanged, ValidatesOnDataErrors=True}"
AcceptsReturn="True" TextWrapping="Wrap"
Width="350" Height="100" HorizontalAlignment="Left"/>
<Label Content="邮件内容模板:"
Width="150"
VerticalAlignment="Top" />
<TextBox
Text="{Binding EmailBodyTemplate, UpdateSourceTrigger=PropertyChanged, ValidatesOnDataErrors=True}"
AcceptsReturn="True"
TextWrapping="Wrap"
Width="350"
Height="100"
HorizontalAlignment="Left" />
</DockPanel>
</StackPanel>
<!-- Add placeholders for other action types if needed -->
<StackPanel Visibility="{Binding Trigger.Action, Converter={StaticResource LocalEnumToVisibilityConverter}, ConverterParameter=ActivateAlarm}">
<TextBlock Text="配置激活报警动作..." Margin="5"/>
<StackPanel
Visibility="{Binding Trigger.Action, Converter={StaticResource LocalEnumToVisibilityConverter}, ConverterParameter=ActivateAlarm}">
<TextBlock Text="配置激活报警动作..."
Margin="5" />
</StackPanel>
<StackPanel Visibility="{Binding Trigger.Action, Converter={StaticResource LocalEnumToVisibilityConverter}, ConverterParameter=WriteToLog}">
<TextBlock Text="配置写入日志动作..." Margin="5"/>
<StackPanel
Visibility="{Binding Trigger.Action, Converter={StaticResource LocalEnumToVisibilityConverter}, ConverterParameter=WriteToLog}">
<TextBlock Text="配置写入日志动作..."
Margin="5" />
</StackPanel>
</StackPanel>
</GroupBox>
<!-- Suppression Section -->
<GroupBox Header="抑制设置" Padding="5" Margin="0,10,0,0">
<GroupBox Header="抑制设置"
Padding="5"
Margin="0,10,0,0">
<StackPanel>
<DockPanel Margin="0,0,0,5">
<Label Content="抑制持续时间 (秒):" Width="150" VerticalAlignment="Center"/>
<TextBox Text="{Binding Trigger.SuppressionDuration, Converter={StaticResource NullableTimeSpanToSecondsConverter}, UpdateSourceTrigger=PropertyChanged}"
Width="100" HorizontalAlignment="Left"/>
<Label Content="抑制持续时间 (秒):"
Width="150"
VerticalAlignment="Center" />
<TextBox
Text="{Binding Trigger.SuppressionDuration, Converter={StaticResource NullableTimeSpanToSecondsConverter}, UpdateSourceTrigger=PropertyChanged}"
Width="100"
HorizontalAlignment="Left" />
</DockPanel>
</StackPanel>
</GroupBox>

View File

@@ -43,3 +43,13 @@ opcUaService.Disconnect();
### Testing
Unit tests for the OPC UA service are included in the `DMS.Infrastructure.UnitTests` project. Run them using your preferred test runner.
## Trigger System
The trigger system has been updated to support associating triggers with multiple variables instead of just one. This allows for more flexible trigger configurations where a single trigger can monitor multiple variables.
### Key Changes
- Modified `TriggerDefinition` to use a list of variable IDs instead of a single variable ID
- Added a new `TriggerVariables` table to maintain the many-to-many relationship between triggers and variables
- Updated the UI to support selecting multiple variables when creating or editing triggers
- Updated all related services and repositories to handle the new many-to-many relationship