24

我有一个正在添加到 Windows 窗体应用程序的 WPF 控件库。我们希望允许控件可本地化,但是我不确定如何在不复制代码的情况下完全实现这一点。这就是我现在正在做的事情

基本上,在 windows 窗体应用程序中,在主应用程序启动之前,我正在实例化一个 App.xaml,它存在于表单应用程序中(包含指向我的资源的链接,这些资源也存在于表单应用程序中)。这非常适合运行时。

但是,我的用户控件都有Content="{StaticResource SomeVariableName}",最终是空白的。我可以通过在我的控件库中拥有一个 app.xaml 和适当的资源字典来解决这个问题,这些字典与我的 windows 窗体应用程序中的资源字典相匹配。但是,这是重复的代码。

我已经尝试过的事情无济于事:

  • 从我的表单应用程序中实例化位于用户控件库中的 App.xaml。这不起作用,因为我的资源的 URI 正在寻找嵌入式资源,而不是我的本地资源字典(然后我可以简单地将资源文件从控件复制到构建时表单应用程序中的适当位置)。我可以在这里利用DeferableContent吗?不过,据我所知,关于这个属性以及它应该如何使用的在线信息并不多。
  • 我想为 App 和字典使用后期构建,但是,据我所知,App 实例化是对已编译 App.xaml 的静态引用。因此,App.xaml 必须至少存在于表单中
    • 我确实尝试使用移动 resourcedictionary.xaml 的后期构建复制 App.xaml。我认为重复的 app.xaml 是可以的,因为这是驱动力,您可能不想依赖控件中的一个(这会回过头来让您想知道是否应该在控件中使用 App.xaml全部?除非您想允许使用嵌入资源的默认值....)这也失败了,即使它被放置在 URI 应该指向的位置,它也找不到资源。反编译的代码指向Uri resourceLocater = new Uri("/WindowsFormsApplication3;component/app.xaml", UriKind.Relative);

那么,有什么方法可以让它工作并在设计时查看组件默认值并避免重复?或者,在这种情况下复制可以吗?如果我的第二个项目符号的子项似乎没问题(复制的 App.xaml 与构建复制的资源字典),我如何使它不查找组件级别的项,而是查找文件级别的项?

我刚刚注意到的最后一个问题(如有必要,我可以单独发布)。我的 App.xaml 已内置到代码中,因此无论如何我都无法即时创建新的 ResourceDictionaries。有没有办法做到这一点?

最后的选择......可能是最好的选择? -无论如何我都打算使用 Andre van Heerwaarde 的代码,所以我应该只检查文件是否存在并将其作为合并资源即时添加吗?基本上,在我的用户控件中有一个 App.xaml 链接到默认的嵌入式 ResourceDictionary。然后,让代码动态查找适当的本地化资源,这些资源可以是相对文件路径吗?我在这里看到的唯一缺点是无法动态更改默认值......我什至可以在指定的地方(使用某种约定)看到它并且比内置的更喜欢?

哦,我不想要嵌入式资源的原因是最终用户可以在构建部署后添加/修改新的本地化资源。

如果它可以帮助您更好地可视化这一点,我可以添加代码,请告诉我。

更新

我现在在样式方面遇到了进一步的问题,而不仅仅是本地化。

以下是其中一个控件上的一个内部按钮的示例:

<Button Style="{StaticResource GrayButton}"

我尝试/想到的更多事情:

  • 由于库项目中不允许使用 ApplicationDefinitions,因此我无法使用设置的 ResourceDictionary 创建 app.xaml(永远不会使用)。我可以将它嵌入到控件的资源中,但是它总是优先于任何应用程序级别的资源,并且我失去了可定制性。

这是一个实际上听起来像我正在寻找的连接案例,但是它没有为此提供任何真正的解决方案

我能想到的解决方案(超出顶部......不起作用)可能会起作用(并且尚未尝试)对于我认为应该简单的事情来说似乎也需要做很多工作。但是,我可能能够在控件中创建一些可以绑定到的依赖属性,然后允许将使用该控件的项目覆盖这些属性。正如我所说,对于一个非常简单的请求来说,这似乎需要做很多工作:)。这甚至会起作用吗?更重要的是,我是否缺少更好、更简单的解决方案?

4

3 回答 3

15

我曾经遇到过这个问题,我通过删除整个“资源是规范字典中的键索引的对象”来解决它。

我的意思是,在一个项目中定义资源并通过它的“关键”在另一个项目中引用它的简单事实应该让任何理智的人起鸡皮疙瘩。我想要强有力的参考。

我对这个问题的解决方案是创建一个自定义工具,将我的资源 xaml 文件转换为静态类,每个资源都有一个属性:

所以 MyResources.xaml:

<ResourceDictionary>
  <SolidColorBrush x:Key="LightBrush" ... />
  <SolidColorBrush x:Key="DarkBrush" ... />
</ResourceDictionary>

成为 MyResources.xaml.cs

public static class MyResources {

  static MyResources() {
    // load the xaml file and assign values to static properties
  }

  public static SolidColorBrush LightBrush { get; set; }
  public static SolidColorBrush DarkBrush { get; set; }

}

要引用资源,您可以使用 thex:Static而不是StaticResource

<Border 
   Fill="{x:Static MyResources.LightBrush}"
   BorderBrush="{x:Static MyResources.DarkBrush}"
   ... />

现在您获得了强大的引用、自动完成和资源的编译时检查。

于 2012-04-26T13:16:57.810 回答
0

我在处理样式主题和可用的静态资源时也遇到了问题。所以,我创建了一个独立的库,除了要使用的主题之外什么都没有,就像你之前链接问题的 MERGED 资源一样。

然后,在 Windows 表单 (.xaml) 中,我只是引用了那个库,比如

<Window x:Class="MyAppNamespace.MyView"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" ... />

  <Window.Resources>
    <ResourceDictionary>
      <ResourceDictionary.MergedDictionaries>
        <!-- Common base theme -->
        <ResourceDictionary   Source="pack://application:,,,/MyLibrary;component/Themes/MyMainThemeWrapper.xaml" />
        </ResourceDictionary.MergedDictionaries>
      </ResourceDictionary>
  </Window.Resources>

  <Rest of XAML for the WPF window>

</Window>

“组件”似乎是指给定“MyLibrary”项目的根。在实际项目中,我创建了一个名为“Themes”的子文件夹,因此源包括... ;component/Themes/...

“MyMainThemeWrapper.xaml”非常类似于嵌套的合并资源字典,它可以完美地查看其他库中的所有内容。

于 2012-04-20T15:58:59.920 回答
0

这是我对您的问题的部分解决方案。我没有尝试处理松散的资源,但我在 WinForms 和 WPF 之间共享资源方面取得了一些成功。

  • 创建一个类库以将您的资源包含在 .ResX 文件中(例如 Resources.resx、Resources.fr.resx 等)
  • 在 WPF 用户控件库中创建 WPF 控件
  • 创建您的 WinForms 主机
  • 使用Infralution.Localization.Wpf标记扩展和文化管理器从 WPF 引用资源库中的资源,例如

    <TextBlock Text="{Resx ResxName=ResourceLib.Resources, Key=Test}"/>
    
  • 将您的 WPF 用户控件的内容作为控件模板放入一个或多个资源字典中,例如

    <ControlTemplate x:Key="TestTemplate">
        <Grid>
            <Grid.RowDefinitions>
                <RowDefinition/>
                <RowDefinition/>
            </Grid.RowDefinitions>
    
            <TextBlock Text="{Resx ResxName=ResourceLib.Resources, Key=Test}"/>
        </Grid>
    </ControlTemplate>
    
  • 在用户控件中使用资源模板

    <UserControl x:Class="WpfControls.UserControl1"
                 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" mc:Ignorable="d" 
                 d:DesignHeight="300" d:DesignWidth="300" >
        <UserControl.Resources>
            <ResourceDictionary>
                <ResourceDictionary.MergedDictionaries>
                    <ResourceDictionary Source="ResourceDictionary.xaml"/>
                </ResourceDictionary.MergedDictionaries>
            </ResourceDictionary>
        </UserControl.Resources>
        <ContentControl Template="{StaticResource TestTemplate}" />
    </UserControl>
    
  • 添加几行代码以使事情正常进行

    public partial class UserControl1 : UserControl
    {
        // we require a reference to the resource library to ensure it's loaded into memory
        private Class1 _class1 = new Class1();
    
        public UserControl1()
        {
            // Use the CultureManager to switch to the current culture
            CultureManager.UICulture = Thread.CurrentThread.CurrentCulture;
    
            InitializeComponent();
        }
    }
    

这是一个名为 WindowsFormsHost.7z的简单演示应用程序

于 2012-04-25T20:11:49.080 回答