This article explains how to modernize MFC MDI project with custom UWP Controls through XAML Islands. With custom UWP controls, it allows us to define control layout easily through XAML pages. Not only putting standard UWP controls into the custom control, we can also integrate other custom controls as well, such as latest WinUI controls. We will use a new Xaml Application project to bring the custom UWP controls into our MFC project. This article has mentioned how to use the new XAML application in Win32 C++ application, we will give more detailed steps for MFC project.
Overall, in our MFC project, we will have two parts to demonstrate this method:
Create MFC App in Visual Studio 2019, will name it MFCAPP
Use below configuration to create the MFCAPP project
click Finish Build and Run it, here is its default UI
In Solution Explorer, right-click the MFCAPP project node, click Retarget Project, select the 10.0.18362.0 or a later SDK release, and then click OK.
Install the Microsoft.Windows.CppWinRT NuGet package:
After install the nuget package, check the MFC project properties, you will notice its C++ version is ISO C++17, which is required by C++/WinRT:
Build this MFCApp, we can see winrt projected files are generated in the “Generated Files” folder:
Install the Microsoft.Toolkit.Win32.UI.SDK NuGet package:
Install the Microsoft.VCRTForwarders.140 nuget package as well. Running Custom UWP Control in this project will require VC libs.
In Solution Explorer, right-click the solution node and select Add -> New Project.
Add a Blank App (C++/WinRT) project to your solution.
Note: Please make sure this is C++/WinRT template. If cannot find it, you'll want to download and install the latest version of the C++/WinRT Visual Studio Extension (VSIX) from the Visual Studio Marketplace.
Give it a name MyApp, and create it, Make sure the target version and minimum version are both set to Windows 10, version 1903 or later.
Right click the MyApp and open its properties, make sure its C++/WinRT configuration is as below:
Change its output type from .EXE to Dynamic Library
Save a dummy.exe into the MyApp folder. It doesn’t need to be a real exe, just input “dummy exe file” in notepad, and save it as dummy.exe.
Make sure the Content property of dummy.exe is True.
Now edit Package.appxmanifest, change the Executable attribute to “dummy.exe”
Right click the MyApp project, select Unload Project
Right click the MyApp (Unloaded) project, select Edit MyApp.vcxproj
Add below properties to the MyApp.vcxproj project file:
<PropertyGroup Label="Globals">
<WindowsAppContainer>true</WindowsAppContainer>
<AppxGeneratePriEnabled>true</AppxGeneratePriEnabled>
<ProjectPriIndexName>App</ProjectPriIndexName>
<AppxPackage>true</AppxPackage>
</PropertyGroup>
For example:
Right click the MyApp (Unloaded) project, select Reload Project.
Right click Mainpage.xml, select Remove. And then click Delete
Copy App.Xaml, App.cpp, App.h, App.idl contents to overwrite current ones:
App.Xaml
<Toolkit:XamlApplication
x:Class="MyApp.App"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="using:MyApp"
xmlns:Toolkit="using:Microsoft.Toolkit.Win32.UI.XamlHost"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
RequestedTheme="Light"
mc:Ignorable="d">
</Toolkit:XamlApplication>
App.cpp
#include "pch.h"
#include "App.h"
using namespace winrt;
using namespace Windows::UI::Xaml;
namespace winrt::MyApp::implementation
{
App::App()
{
Initialize();
AddRef();
m_inner.as<::IUnknown>()->Release();
}
App::~App()
{
Close();
}
}
App.h
//
// Declaration of the App class.
//
#pragma once
#include "App.g.h"
#include "App.base.h"
namespace winrt::MyApp::implementation
{
class App : public AppT2<App>
{
public:
App();
~App();
};
}
namespace winrt::MyApp::factory_implementation
{
class App : public AppT<App, implementation::App>
{
};
}
App.idl
namespace MyApp
{
[default_interface]
runtimeclass App: Microsoft.Toolkit.Win32.UI.XamlHost.XamlApplication
{
App();
}
}
app.base.h
#pragma once
namespace winrt::MyApp::implementation
{
template <typename D, typename... I>
struct App_baseWithProvider : public App_base<D, ::winrt::Windows::UI::Xaml::Markup::IXamlMetadataProvider>
{
using IXamlType = ::winrt::Windows::UI::Xaml::Markup::IXamlType;
IXamlType GetXamlType(::winrt::Windows::UI::Xaml::Interop::TypeName const& type)
{
return AppProvider()->GetXamlType(type);
}
IXamlType GetXamlType(::winrt::hstring const& fullName)
{
return AppProvider()->GetXamlType(fullName);
}
::winrt::com_array<::winrt::Windows::UI::Xaml::Markup::XmlnsDefinition> GetXmlnsDefinitions()
{
return AppProvider()->GetXmlnsDefinitions();
}
private:
bool _contentLoaded{ false };
std::shared_ptr<XamlMetaDataProvider> _appProvider;
std::shared_ptr<XamlMetaDataProvider> AppProvider()
{
if (!_appProvider)
{
_appProvider = std::make_shared<XamlMetaDataProvider>();
}
return _appProvider;
}
};
template <typename D, typename... I>
using AppT2 = App_baseWithProvider<D, I...>;
}
Install Microsoft.Toolkit.Win32.UI.XamlApplication Nuget package.
If you build MyApp now, it should create MyApp.dll without any error.
This step is necessary for our next steps because we need to include winrt header files in different projects properly, and MFCApp also needs to reference MyApp resource files.
<?xml version="1.0" encoding="utf-8"?>
<Project xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
<PropertyGroup>
<IntDir>$(SolutionDir)\obj\$(Platform)\$(Configuration)\$(MSBuildProjectName)\</IntDir>
<OutDir>$(SolutionDir)\bin\$(Platform)\$(Configuration)\$(MSBuildProjectName)\</OutDir>
<GeneratedFilesDir>$(IntDir)Generated Files\</GeneratedFilesDir>
</PropertyGroup>
</Project>
Click Views -> Other Windows -> Property Manager
Right click MFCApp, select Add Existing Property Sheet, add the new solution.props file
Repeat the step 4. for MyApp. We can close the Property Manager window now.
Right click the project node MFCApp, select Properties. Set
Output Directory: $(OutDir)
Intermidia Directory: $(IntDir)
Repeat the step 6 for MyApp.
Right click the Solution node, and choose Project Dependencies, make sure MFCApp depends on MyApp:
Rebuild the whole solution, it should work without errors.
<?xml version="1.0" encoding="utf-8" standalone="yes"?>
<assembly
xmlns="urn:schemas-microsoft-com:asm.v1"
xmlns:asmv3="urn:schemas-microsoft-com:asm.v3"
manifestVersion="1.0">
<asmv3:file name="MyApp.dll">
<activatableClass
name="MyApp.App"
threadingModel="both"
xmlns="urn:schemas-microsoft-com:winrt.v1" />
<activatableClass
name="MyApp.XamlMetadataProvider"
threadingModel="both"
xmlns="urn:schemas-microsoft-com:winrt.v1" />
<activatableClass
name="MyApp.TreeViewUserControl"
threadingModel="both"
xmlns="urn:schemas-microsoft-com:winrt.v1" />
</asmv3:file>
</assembly>
Now the project structure is like as below:
Right click the Win32 Project MFCApp, select Unload Project
Right click the MFCApp (Unloaded) project, select Edit MFCApp.vcxproj
Remove the three lines from the MFCApp.vcxproj project file:
<ItemGroup>
<Manifest Include="app.manifest" />
</ItemGroup>
Add below properties to the MFCApp.vcxproj project file before the “<Import Project=”“$(VCTargetsPath).Cpp.targets” />" line:
<ItemGroup>
<Manifest Include="app.manifest" />
<AppxManifest Include="$(SolutionDir)\bin\$(Platform)\$(Configuration)\$(AppProjectName)\AppxManifest.xml" />
</ItemGroup>
<PropertyGroup>
<AppProjectName>MyApp</AppProjectName>
</PropertyGroup>
<PropertyGroup>
<AppIncludeDirectories>$(SolutionDir)\obj\$(Platform)\$(Configuration)\$(AppProjectName)\;$(SolutionDir)\obj\$(Platform)\$(Configuration)\$(AppProjectName)\Generated Files\;</AppIncludeDirectories>
</PropertyGroup>
<ItemGroup>
<NativeReferenceFile Include="$(SolutionDir)\bin\$(Platform)\$(Configuration)\$(AppProjectName)\*.xbf">
<DeploymentContent>true</DeploymentContent>
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
</NativeReferenceFile>
<NativeReferenceFile Include="$(SolutionDir)\bin\$(Platform)\$(Configuration)\$(AppProjectName)\*.dll">
<DeploymentContent>true</DeploymentContent>
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
</NativeReferenceFile>
<NativeReferenceFile Include="$(SolutionDir)\bin\$(Platform)\$(Configuration)\$(AppProjectName)\resources.pri">
<DeploymentContent>true</DeploymentContent>
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
</NativeReferenceFile>
</ItemGroup>
<------Right Here--------->
<Import Project="$(VCTargetsPath)\Microsoft.Cpp.targets" />
Right click the MFCApp (Unloaded) project, select Reload Project.
Right click MFCApp, select Properties, setup $(AppIncludeDirectories) as a part of include file path, this macro has been defined in the above project file:
Set “Per Monitor DPI Aware” for DPI Awareness otherwise you may be not able to start this MFCApp when it is “High DPI Aware” and hit configuration error in Manifest:
Open pch.h, add below code to include necessary winrt header files:
#pragma push_macro("GetCurrentTime")
#pragma push_macro("TRY")
#undef GetCurrentTime
#undef TRY
#include <winrt/Windows.Foundation.Collections.h>
#include <winrt/Windows.system.h>
#include <winrt/windows.ui.xaml.hosting.h>
#include <windows.ui.xaml.hosting.desktopwindowxamlsource.h>
#include <winrt/windows.ui.xaml.controls.h>
#include <winrt/Windows.ui.xaml.media.h>
#include <winrt/Windows.UI.Core.h>
#include <winrt/myapp.h>
#pragma pop_macro("TRY")
#pragma pop_macro("GetCurrentTime")
Regarding the reason of using “GetCurrentTime” and “TRY” macros, please refer to: https://docs.microsoft.com/en-us/windows/uwp/cpp-and-winrt-apis/faq
Using winrt namespaces in MFCAPPView.h and MFCAPP.h
using namespace winrt;
using namespace Windows::UI;
using namespace Windows::UI::Composition;
using namespace Windows::UI::Xaml::Hosting;
using namespace Windows::Foundation::Numerics;
using namespace Windows::UI::Xaml::Controls;
Declare hostApp in MFCApp.h
public:
winrt::MyApp::App hostApp{ nullptr };
hostApp = winrt::MyApp::App{};
<---Right Here--->
// create main MDI Frame window
CMainFrame* pMainFrame = new CMainFrame;
private:
winrt::Windows::UI::Xaml::Hosting::DesktopWindowXamlSource _desktopWindowXamlSource{ nullptr };
winrt::MyApp::TreeViewUserControl _treeViewUserControl{ nullptr };
if (_desktopWindowXamlSource == nullptr)
{
_desktopWindowXamlSource = DesktopWindowXamlSource{};
auto interop = _desktopWindowXamlSource.as<IDesktopWindowXamlSourceNative>();
check_hresult(interop->AttachToWindow(GetSafeHwnd()));
HWND xamlHostHwnd = NULL;
check_hresult(interop->get_WindowHandle(&xamlHostHwnd));
_treeViewUserControl = winrt::MyApp::TreeViewUserControl();
_desktopWindowXamlSource.Content(_treeViewUserControl);
RECT windowRect;
GetWindowRect(&windowRect);
::SetWindowPos(xamlHostHwnd, NULL, 0, 0, windowRect.right - windowRect.left, windowRect.bottom - windowRect.top, SWP_SHOWWINDOW);
}
Clean up resources when the view is disconstructed in MFCAppView.cpp
CMFCAppView::~CMFCAppView()
{
if (_desktopWindowXamlSource != nullptr)
{
_desktopWindowXamlSource.Close();
_desktopWindowXamlSource = nullptr;
}
}
Add AdjustLayout function to make XAML content layout properly in MFCAppView.cpp :
```C++
void CMFCAppView::AdjustLayout()
{
if (_desktopWindowXamlSource != nullptr)
{
auto interop = _desktopWindowXamlSource.as<IDesktopWindowXamlSourceNative>();
HWND xamlHostHwnd = NULL;
check_hresult(interop->get_WindowHandle(&xamlHostHwnd));
RECT windowRect;
GetWindowRect(&windowRect);
::SetWindowPos(xamlHostHwnd, NULL, 0, 0, windowRect.right - windowRect.left, windowRect.bottom - windowRect.top, SWP_SHOWWINDOW);
}
}
```
Don't forget to declare it in MFCAppView.h:
```C++
public:
void AdjustLayout();
```
Right click the MFCApp project, select Class Wizard
Add a handler to handle WM_SIZE so that when view size changes we can handle it:
Modify the OnSize method handler:
```C++
void CMFCAppView::OnSize(UINT nType, int cx, int cy)
{
CView::OnSize(nType, cx, cy);
AdjustLayout();
}
```
[!NOTE] It is possible some version of WinUI nuget package doesn’t create Microsoft.UI.Xaml.Controls class registering info into AppxManifest.xml, which is required by MFCApp later. This version used above works well. If you found MFCApp failed to run with “Class is not registered” error, please try this version.
Modify App.Xaml, TreeViewUserControl.Xaml, pch.h and TreeViewUserContro.cpp as below:
For detailed reasons on modifying these files, can refer to this article
Add the Windows UI (WinUI) Theme Resources to App.Xmal
<Toolkit:XamlApplication
x:Class="MyApp.App"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="using:MyApp"
xmlns:Toolkit="using:Microsoft.Toolkit.Win32.UI.XamlHost"
xmlns:MSMarkup="using:Microsoft.UI.Xaml.Markup"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
RequestedTheme="Light"
mc:Ignorable="d">
<Toolkit:XamlApplication.Resources>
<XamlControlsResources xmlns="using:Microsoft.UI.Xaml.Controls"/>
</Toolkit:XamlApplication.Resources>
</Toolkit:XamlApplication>
Add WinUI reference and WinUI TreeView control in TreeViewUserControl.Xaml
<UserControl
x:Class="MyApp.TreeViewUserControl"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="using:MyApp"
xmlns:muxc="using:Microsoft.UI.Xaml.Controls"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
mc:Ignorable="d">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" VerticalAlignment="Center">
<Button x:Name="Button" Click="ClickHandler">Click Me</Button>
<muxc:TreeView x:Name="WinUITreeView">
<muxc:TreeView.RootNodes>
<muxc:TreeViewNode Content="Flavors"
IsExpanded="True">
<muxc:TreeViewNode.Children>
<muxc:TreeViewNode Content="Vanilla"/>
<muxc:TreeViewNode Content="Strawberry"/>
<muxc:TreeViewNode Content="Chocolate"/>
</muxc:TreeViewNode.Children>
</muxc:TreeViewNode>
</muxc:TreeView.RootNodes>
</muxc:TreeView>
</StackPanel>
</UserControl>
Include WinUI winrt header files in pch.h
#include "winrt/Microsoft.UI.Xaml.Controls.h"
#include "winrt/Microsoft.UI.Xaml.XamlTypeInfo.h"
TreeViewUserControl.cpp
void TreeViewUserControl::ClickHandler(IInspectable const&, RoutedEventArgs const&)
{
Button().Content(box_value(L"Clicked"));
winrt::Microsoft::UI::Xaml::Controls::TreeViewNode tn = winrt::Microsoft::UI::Xaml::Controls::TreeViewNode{};
tn.Content(winrt::box_value(L"Clicked"));
WinUITreeView().RootNodes().First().Current().Children().Append(tn);
}
This article gives detailed steps on how to leverage XamplApplication to host custom XAML control in document view of traditional MFC Mulitple Document Interface project, and the important thing is you can use WinUI to modernize the MFC application now. The whole smaple solution can be found from this repo: https://github.com/freistli/ModernizeApp/tree/master/MFC/MFCAppWinUI
You must be a registered user to add a comment. If you've already registered, sign in. Otherwise, register and sign in.