为了避免丢失和损坏,编译器允许我们把外部文件编译进程序主体、成为程序主体不可分割的一部分,这就是传统意义上的程序资源,即二进制资源;
WPF 的四个等级资源:
- 数据库里的数据 (仓库)
- 资源文件 (行旅箱)
- 对象资源 (背包)
- 变量中的数据 (手中)
1. 对象级的定义和查找
<Window.Resources>
<ResourceDictionary>
...
</ResourceDictionary>
</Window.Resources>
视情况可以简写不写。
<Window
...
xmlns:sys="clr-namespace:System;assembly=mscorlib">
<Window.Resources>
<sys:String x:Key="str">
Hello, Wpf.
</sys:String>
<sys:Double x:Key="dbl">3.1415926</sys:Double>
</Window.Resources>
<StackPanel>
<TextBlock Text="{StaticResource ResourceKey=str}"/>
</StackPanel>
// C# 里面找资源,会一层一层往上找,直到 Application.Resources
string text = (string)this.FindResource("str"); //找不到就抛出异常
string text = (string)this.TryFindResource("str"); //找不到就返回 null
// 如果已经明确资源放在哪,可以这样写:
string text = (string)this.Resource["str"]; //找不到就返回 null
2. 像 css 那样引用独立文件
- 新建独立文件 /Themes/Style.xaml
<ResourceDictionary
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="clr-namespace:WpfApp1">
...
</ResourceDictionary>
- 通过 ResourceDictionary 的 Source 属性指定要引用的独立文件
<Application.Resources>
<ResourceDictionary>
<ResourceDictionary.MergedDictionaries>
<ResourceDictionary Source="/Themes/Style.xaml" />
</ResourceDictionary.MergedDictionaries>
</ResourceDictionary>
</Application.Resources>
3. 动 or 静
StaticResource 静态资源指在程序载入内存时对资源的一次性使用,之后不再去访问这个资源;
DynamicResource 动态资源是指在程序运行过程中仍然会去访问资源;
就像主题皮肤,如果不会变的,就用 StaticResource,如果程序运行过程中还要切换不同皮肤,就使用 DynamicResource。
<Window.Resources>
<TextBlock x:Key="tb1" Text="I am a textBlock"/>
</Window.Resources>
<StackPanel>
<Button x:Name="btn1" Content="{StaticResource tb1}" />
<Button x:Name="btn2" Content="{DynamicResource tb1}" />
</StackPanel>
...
void function(object sender, RoutedEventArgs e)
{
this.Resources["tb1"] = new TextBlock() { Text = "I have changed" };
}
执行 function 方法后,会看到 btn1 上的文字依旧是 "I am a textBlock", 而 btn2 按钮上的文字已变成 "I have changed"
4. 二进制资源
4.1 字符串
4.1.1 注册
- 应用程序 Properties 名称空间中的 Resources.resx 资源文件,跟 Settings.settings 类似采用键值对的方式;
- 为了让 XAML 编译器能够访问这个类,一定要把 Resources.resx 的访问级别由 Internal 改为 Public;
4.1.2 使用
- 定义举例:Properties.Resources.resx 下定义名称为 password 值为 passwd 的键值对;
- 调用举例:
<Window
...
xmlns:prop="clr-namespace:WpfApp1.Properties">
<StackPanel>
<TextBlock Text="{x:Static prop:Resources.username}"/>
<TextBlock x:Name="tb1"/>
</StackPanel>
</Window>
this.tb1.Text = Properties.Resources.password;
4.2 媒体资源
直接将文件引入项目中;
- 如果是想把文件编译进目标成为二进制资源,必须在窗口属性中把文件的 生成操作 属性值为 Resource,复制到输出目录 属性设为 不复制;
- 如果想输出到生成的方案中,就把生成操作设置为 内容,复制到输出目录属性 设为 如果较新则复制,这种方式可以在生成方案之后客户端继续更换资源(如更换图片)。
4.3 使用 Pack URI 路径访问二进制资源
路径模板:
- pack://application:,,,[/程序集名称;][可选版本号;][文件夹名称/]文件名称
- pack://application:,,, 可以省略
其中,程序集名称 可选版本号常使用缺省值。
<!--下面三个指定 Source 的方式效果一样-->
<Image x:Name="img1" Source="Images/2.jpg" Stretch="Fill" />
<Image x:Name="img2" Source="/Images/2.jpg" Stretch="Fill" />
<Image x:Name="img3" Source="pack://application:,,,/Images/2.png" Stretch="Fill" />
等同于下列的后台代码:
// {pack://application:,,,/WpfApp1;component/Images/2.png}
this.img1.Source = new BitmapImage(new Uri("Images/2.png", UriKind.Relative));
// {pack://application:,,,/Images/2.png}
this.img2.Source = new BitmapImage(new Uri("/Images/2.png", UriKind.Relative));
this.img3.Source = new BitmapImage(new Uri("pack://application:,,,/Images/2.png", UriKind.Absolute));
其中 pack://application:,,, 开头表示绝对路径,需使用 UriKind.Absolute;
缩略写法代表相对路径,UriKind 必须为 Relative,且代表根目录的 / 可以省略;
使用相对路径时,可以借助 DOS 的语法进行导航,如 ./ 代表上一级 ../ 代表父级