How do I construct a pack URI to an image that is in a resource file?
I have an assembly called MyAssembly.Resources.dll
, it has a folder called Images, then in there is a resource file called Assets.resx. This resource file contains my image (called MyImage.png). The line of code I have is:
uri = new Uri("pack://application:,,,/MyAssembly.Resources,Culture=neutral,PublicKeyToken=null;component/Images/Assets/MyImage.png");
However when I try to supply this URI to the constructor of a new BitmapImage I get an IOException with the message
Cannot locate resource 'images/assets/myimage.png'.
Note that I have other loose images in the same assembly which I can retrieve fine using a pack URI, those images have their build action set to Resource but they are not embedded in a resx file. Should I be including the name of the resx file in the path?
(I am looking to embed images in resx files so that I can leverage UI culture settings to retrieve the right image (the image contains text)).
As Walt has correctly stated, what you get out of a resx file is a
System.Drawing.Bitmap
. So this needs to be converted to aSystem.Windows.Media.ImageSource
or subtype.I'm not sure if this falls under time wasters for you because it does not employ an URI, but here is how I get images from resx files in another library. I use a simple proxy because the resx designer file only exposes an internal constructor (even if the class is public), then define a ValueConverter that will provide the ImageSource.
AssetsProxy:
Bitmap to ImageSource conversion:
There are two ways to "embed" a resource in an assembly. Windows Forms uses the
Embedded Resource
Build Action. WPF expects resources contained in assemblies to be marked with theResource
Build Action.When you use the Resx editor in Visual Studio to add an image, it marks it as an Embedded Resource. Also, it stores it as type
System.Drawing.Bitmap
. WPF expect aSystem.Windows.Media.ImageSource
type.If you have a dissembler (like ILSpy) you can look at impact of setting different build actions on the files.
Sample ImagesLib project
Here is a screenshot of a project with two images. It's obvious from the names, the
cat_embedded.jpg
is using theEmbedded Resource
Build action and thecat_resource.jpg
is using theResource
Build action.This is what they look like in ILSpy.
See how the cat_resource.jpg file is within the ImageLib.g.resources section? That is where WPF looks for resources. The path to the file is part of the resource name (
images/cat_resource.jpg
). So when you use a path like:you specify the matching path after the word
;component
.The other jpg file is located in a different location in the assembly, and uses periods in the name (
ImageLib.Images.cat_embedded.jpg
).You can try many permutations of that string to try and get the cat_embedded.jpg image, but WPF won't find it.
RESX Editor
Here's another project, that has two images, one marked as a resource and one added by the resx editor.
And here is the disassembled screenshot.
As you can see, the resx image is using the same URI location as the earlier embedded image example. It appears in your case, you are not going to be able to get the images from the resx file using the Pack URI.
Localization
From what you said in your question, what you are trying to accomplish is localization of the images right?
Have you looked at this MSDN article?
WPF Globalization and Localization Overview
I don't think it's possible using the "pack" protocol scheme. This protocol is related to normalized Open Packaging Conventions specs (http://tools.ietf.org/id/draft-shur-pack-uri-scheme-05.txt for pointers). So the pack uri points to the application package's resources (or parts in OPC terms), not to .NET embedded resources.
However, you can define your own scheme, for example "resx" and use it in WPF component uris. New Uri schemes for such usages can be defined using WebRequest.RegisterPrefix.
Here is an example based on a small Wpf application project named "WpfApplication1". This application has a Resource1.resx file defined (and possibly other localized corresponding Resource1 files, like Resource1.fr-FR.resx for french for example). Each of these ResX files define an Image resource named "img" (note this name is not the same as the image file name the resource is based on).
Here is the MainWindow.xaml:
The uri format is this:
and assembly name is optional, so
is also valid and point to resources in the main assembly (my sample uses this)
This is the code that supports it, in App.xaml.cs or somewhere else, you need to register the new scheme:
And the scheme implementation:
I described a component for using resx images in WPF in this blog post: http://wpfglue.wordpress.com/2012/05/31/localization-revisited/ . You will find more posts about using resx resources in WPF under http://wpfglue.wordpress.com/category/localization/
In these posts, I don't use pack uris, but markup extensions.