Microsoft PRISM:在WPF菜单控件中使用多个区域

时间:2016-09-19 15:50:58

标签: c# wpf menu prism

我们正在PRISM和大量模块的帮助下实施模块化软件。该框架包含MainWindow Shell,其中包括工具栏和菜单的定义。当模块例如是模块时,每个模块应该能够在运行时添加自己的工具栏和菜单条目。初始化。为此,使用区域经理。对于工具栏,它运行良好,但对于菜单,几乎没有问题出现:

box-shadow

.....

区域<Grid> <Grid.RowDefinitions> <RowDefinition Height="Auto" /> <RowDefinition Height="Auto" /> <RowDefinition /> </Grid.RowDefinitions> <!-- <Menu Grid.Row="0" IsMainMenu="True"> <MenuItem Header="Test" /> </Menu> --> <!-- Main Menu --> <StackPanel Grid.Row="0" Orientation="Horizontal"> <ItemsControl Name="MainMenuRegion" prism:RegionManager.RegionName="{x:Static common:RegionNames.MainMenu}"> <ItemsControl.ItemsPanel> <ItemsPanelTemplate> <WrapPanel Orientation="Horizontal" /> </ItemsPanelTemplate> </ItemsControl.ItemsPanel> </ItemsControl> <ItemsControl Name="AppMenuRegion" prism:RegionManager.RegionName="{x:Static common:RegionNames.AppMenu}"> <ItemsControl.ItemsPanel> <ItemsPanelTemplate> <WrapPanel Orientation="Horizontal" /> </ItemsPanelTemplate> </ItemsControl.ItemsPanel> </ItemsControl> <ItemsControl Name="HelpMenuRegion" prism:RegionManager.RegionName="{x:Static common:RegionNames.HelpMenu}"> <ItemsControl.ItemsPanel> <ItemsPanelTemplate> <WrapPanel Orientation="Horizontal" /> </ItemsPanelTemplate> </ItemsControl.ItemsPanel> </ItemsControl> </StackPanel> 用于常见条目,例如File,Edit,Veiw。区域RegionNames.MainMenu用于添加自己的菜单的几个模块。区域RegionNames.AppMenu用于常用条目窗口和帮助。使用区域的动机是获取菜单条目的订单。每个模块都以这种方式为菜单创建自己的视图(用户控件)和视图模型:

RegionNames.HelpMenu

此外,对于我们使用数据模板的菜单项:

<Menu ItemsSource="{Binding MainMenuItems}" />

但这种做法并不奏效。尝试在两个不同的区域中注册相同的视图时,如下所示:

<Style TargetType="{x:Type MenuItem}">
    <Setter Property="Command" Value="{Binding Command}" />
    <Setter Property="CommandParameter" Value="{Binding CommandParameter}" />
    <Setter Property="Header" Value="{Binding Text, UpdateSourceTrigger=PropertyChanged}" />
    <Setter Property="IsChecked" Value="{Binding IsChecked, UpdateSourceTrigger=PropertyChanged}" />
    <Setter Property="IsEnabled" Value="{Binding IsEnabled, UpdateSourceTrigger=PropertyChanged}" />
    <Setter Property="Visibility" Value="{Binding IsVisible, UpdateSourceTrigger=PropertyChanged, Converter={StaticResource VisibilityConverter}}" />
    <Setter Property="ItemsSource" Value="{Binding Children, UpdateSourceTrigger=PropertyChanged}" />
    <Setter Property="Icon" Value="{DynamicResource itemIcon}" />
    <Style.Triggers>
        <DataTrigger Binding="{Binding IsSeparator, UpdateSourceTrigger=PropertyChanged}" Value="true">
            <Setter Property="Template">
                <Setter.Value>
                    <ControlTemplate>
                        <Separator />
                    </ControlTemplate>
                </Setter.Value>
            </Setter>
        </DataTrigger>
        <DataTrigger Binding="{Binding Icon}" Value="{x:Null}">
            <Setter Property="Icon" Value="{x:Null}" />
        </DataTrigger>
    </Style.Triggers>
</Style>

只有最后一个区域(MainMenu区域)才能获得菜单条目。在两个不同视图的情况下,相同的行为。似乎最后一个覆盖了第一个。

我搜索了合适的解决方案,但不幸的是到目前为止我还没有找到一个这个用例。希望我能以这种方式找到解决这个问题的解决方案或其他方法。提前谢谢。

2 个答案:

答案 0 :(得分:0)

IRegionViewRegistry regionViewRegistry = ServiceLocator.Current.GetInstance<IRegionViewRegistry>();
regionViewRegistry.RegisterViewWithRegion(RegionNames.HelpMenu, typeof(MainMenu)); // registering same view with diferent region 
regionViewRegistry.RegisterViewWithRegion(RegionNames.MainMenu, typeof(MainMenu));

如果我理解正确,那么你定义了3个区域(在你的情况下你可以添加自定义用户控件的占位符是菜单项。)

1)RegionNames.HelpMenu

2)RegionNames。 AppMenu

3)RegionNames.MainMenu

现在您需要创建3个自定义用户控件并将其注册到该区域。

 public partial class MainMenuSelection : UserControl 
    {
       public MainMenuSelection () 
       { 
         InitializeComponent();
       } 
    }

同样适用于应用菜单和帮助菜单

然后你就可以使用

ServiceLocator.Current.GetInstance<IRegionViewRegistry>();
regionViewRegistry.RegisterViewWithRegion(RegionNames.HelpMenu,typeof(HelpMenuSelection));
regionViewRegistry.RegisterViewWithRegion(RegionNames.MainMenu, typeof(MainMenuSelection)); // You are registering view (mainmenuselection) with unique region placeholder(RegionNames.MainMenu)

答案 1 :(得分:0)

感谢您的回答。但我这样做就是你所描述的

    IRegionViewRegistry regionViewRegistry = ServiceLocator.Current.GetInstance<IRegionViewRegistry>();
    regionViewRegistry.RegisterViewWithRegion(RegionNames.MainMenu, typeof(MainMenu));
    regionViewRegistry.RegisterViewWithRegion(RegionNames.HelpMenu, typeof(HelpMenu));

和菜单项的UserControl

<UserControl x:Class="XXX.GUI.View.MainMenu"
         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"
         d:DesignHeight="300"
         d:DesignWidth="300"
         mc:Ignorable="d">

     <MenuItem ItemsSource="{Binding MainMenuItems}" /> 
</UserControl>

在我的ViewModel中,创建了MainMenuItems的可观察集合。 当我这样做时,MenuItem不是顶级而不是第二级,并且无法将帮助菜单注入帮助菜单区域。