gpt4 book ai didi

c# - 如何在WPF MVVM Catel应用程序中切换多个 View ?

转载 作者:行者123 更新时间:2023-12-02 14:34:08 28 4
gpt4 key购买 nike

在 MS VS 2015 Professional 中,我使用 Catel 作为 MVVM 框架开发 C# WPF MVVM 应用程序。我的问题是我不知道如何使用按钮实现一个窗口中多个 View 之间的切换。下面我简单介绍一下我的申请。主窗口有三个按钮

<catel:Window x:Class="FlowmeterConfigurator.Views.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:catel="http://catel.codeplex.com"
ResizeMode="CanResize">

<catel:StackGrid x:Name="LayoutRoot">
<catel:StackGrid.RowDefinitions>
<RowDefinition Height="Auto" />
<RowDefinition Height="Auto" />
<RowDefinition Height="Auto"/>
</catel:StackGrid.RowDefinitions>

<ToolBar>
<Button Name="btnConnectDisconnect" Content="Connect/Disconnect"/>
<Button Name="btnFieldSettings" Content="Field Settings"/>
<Button Name="btnCalibration" Content="Flowmeter Calibration"/>
</ToolBar>
</catel:StackGrid>
</catel:Window>

应用程序主窗口有一个 ViewModel。为了简洁起见,我不在这里展示它。除了 MainWindow 之外,我的应用程序中还有三个 View :ConnectDisconnectView、CalibrationView 和 FieldSettingsView。为了简洁起见,我在这里只显示其中之一 (FieldSettingsView),因为所有其他内容都是在 catel:UserControl 的基础上以相同的方式创建的。

<catel:UserControl x:Class="FlowmeterConfigurator.Views.FieldSettingsView"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:catel="http://catel.codeplex.com">

<catel:StackGrid>
<catel:StackGrid.RowDefinitions>
<RowDefinition Height="Auto" />
<RowDefinition Height="Auto" />
</catel:StackGrid.RowDefinitions>
<catel:StackGrid.ColumnDefinitions>
<ColumnDefinition/>
<ColumnDefinition/>
</catel:StackGrid.ColumnDefinitions>

<Label Grid.Row="0" Grid.Column="0" Content="Flowmeter Serial Number"/>
<TextBox Name="SerialNumber" Grid.Row="0" Grid.Column="1"/>
</catel:StackGrid>

</catel:UserControl>

每个 View 都有一个模型。我在这里仅展示这些模型之一,因为所有模型都是以相同的方式创建的。

using Catel.Data;
namespace FlowmeterConfigurator.Models
{
/// <summary>
/// Field Settings Model.
/// </summary>
public class FieldSettingsModel : SavableModelBase<FieldSettingsModel>
{
/// <summary>
/// Returns flowmeter serial number.
/// </summary>
public string SerialNumber
{
get { return GetValue<string>(SerialNumberProperty); }
set { SetValue(SerialNumberProperty, value); }
}

/// <summary>
/// Register SerialNumber property.
/// </summary>
public static readonly PropertyData SerialNumberProperty = RegisterProperty("SerialNumber", typeof(string), null);
}
}

每个 View 都有一个 ViewModel。我在这里只显示这些 ViewModel 之一,因为它们都是以相同的方式创建的。

using Catel;
using Catel.Data;
using Catel.MVVM;
using FlowmeterConfigurator.Models;

namespace FlowmeterConfigurator.ViewModels
{
/// <summary>
/// Field settings ViewModel.
/// </summary>
public class FieldSettingsViewModel : ViewModelBase
{
/// <summary>
/// Creates a FieldSettingsViewModel instance.
/// </summary>
/// <param name="fieldSettingsModel">Field settings Model.</param>
public FieldSettingsViewModel(FieldSettingsModel fieldSettingsModel)
{
Argument.IsNotNull(() => fieldSettingsModel);
FieldSettings = fieldSettingsModel;
}

/// <summary>
/// Returns or sets Field Settings Model.
/// </summary>
[Model]
public FieldSettingsModel FieldSettings
{
get { return GetValue<FieldSettingsModel>(FieldSettingsProperty); }
set { SetValue(FieldSettingsProperty, value); }
}

/// <summary>
/// Here I register FieldSettings property.
/// </summary>
public static readonly PropertyData FieldSettingsProperty = RegisterProperty("FieldSettings", typeof(FieldSettingsModel), null);

/// <summary>
/// Returns or sets flowmeter serial number.
/// </summary>
[ViewModelToModel("FieldSettings")]
public string SerialNumber
{
get { return GetValue<string>(SerialNumberProperty); }
set { SetValue(SerialNumberProperty, value); }
}

/// <summary>
/// Here I register SerialNumber property.
/// </summary>
public static readonly PropertyData SerialNumberProperty = RegisterProperty("SerialNumber", typeof(string), null);
}
}

在我的应用程序加载后,必须立即显示 ConnectDisconnectView。然后用户可以使用主窗口工具栏上的按钮随意切换 View 。 View 之间的切换必须按以下方式进行:如果(例如)当前显示的 View 是“ConnectDisconnectView”并且用户按下“Field Settings”按钮,则“ConnectDisconnectView” View 必须从主窗口中消失,并且“FieldSettingsView” View 必须出现并且必须显示在主窗口中。等等。也就是说,当按下主窗口工具栏中的适当按钮(例如“流量计校准”)时,必须在主窗口中显示适当的 View (CalibrationView),并且不得显示其他 View 。如何在我的应用程序中实现此功能?我们将非常感谢您的帮助。

附注当然,正如您所看到的,为了简洁和清晰,此处减少了 View 的数量和内容。在现实世界中,我的应用程序中的 View 数量约为 20 - 25 个,它们必须包含复杂的图形和表格信息。

最佳答案

首先我向您展示 xaml 代码:

<catel:Window.Resources>
<catel:ViewModelToViewConverter x:Key="ViewModelToViewConverter" />
</catel:Window.Resources>

<catel:StackGrid x:Name="LayoutRoot">
<ContentControl Content="{Binding CurrentPage, Converter={StaticResource ViewModelToViewConverter}}" />

<ToolBar>
<Button Name="btnConnectDisconnect" Command={Binding Connect} Content="Connect/Disconnect"/>
<Button Name="btnFieldSettings" Command={Binding Field} Content="Field Settings"/>
<Button Name="btnCalibration" Command={Binding Calibration} Content="Flowmeter Calibration"/>
</ToolBar>
</catel:StackGrid>

然后在 C# 代码中你需要这样:

using Catel.Data;
using Catel.MVVM;
using System.Threading.Tasks;

public class MainWindowViewModel : ViewModelBase
{
public MainWindowViewModel()
{
this.Connect = new Command(HandleConnectCommand);
this.Field = new Command(HandleFieldCommand);
this.Calibration = new Command(HandleCalibrationCommand);

this.CurrentPage = new ConnectViewModel();
}

/// <summary>
/// Gets or sets the CurrentPage value.
/// </summary>
public IViewModel CurrentPage
{
get { return GetValue<IViewModel>(CurrentPageProperty); }
set { SetValue(CurrentPageProperty, value); }
}

/// <summary>
/// Register the CurrentPage property so it is known in the class.
/// </summary>
public static readonly PropertyData CurrentPageProperty = RegisterProperty("CurrentPage", typeof(IViewModel), null);

public Command Connect { get; private set; }

public Command Field { get; private set; }

public Command Calibration { get; private set; }

protected override async Task InitializeAsync()
{
await base.InitializeAsync();
// TODO: subscribe to events here
}

protected override async Task CloseAsync()
{
// TODO: unsubscribe from events here
await base.CloseAsync();
}

private void HandleCalibrationCommand()
{
this.CurrentPage = new CalibrationViewModel();
}

private void HandleFieldCommand()
{
this.CurrentPage = new FieldViewModel();
}

private void HandleConnectCommand()
{
this.CurrentPage = new ConnectViewModel();
}
}

当您启动应用程序时,CurrentPage 将使用数据上下文 ConnectViewModel() 加载。然后使用按钮命令,您可以更改另一个 View 模型的日期上下文。

关于c# - 如何在WPF MVVM Catel应用程序中切换多个 View ?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34942428/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com