There are a lot of exciting enhancements for Windows Forms developers in .NET 7. We are committed to support and innovate in the Windows Forms runtime, so let’s look at what is new in .NET 7.
Accessibility improvements and fixes
We strive to improve the experience for users of assistive technology. This release adds further improvements to accessibility, including but not limited to the following:
- Many announcement-related issues observed in screen readers (e.g., Narrator) have been addressed, ensuring the information about controls is correct and complete. For example,
ListView
now correctly announces when a group is expanded or collapsed. - More controls now provide UI Automation support:
TreeView
DateTimePicker
ToolStripContainer
andToolStripPanel
FlowLayoutPanel
,TableLayoutPanel
andSplitContainer
PrintPreviewControl
WebBrowser
- Multiple memory leaks resulting from running Windows Forms applications under assistive tools (e.g., Narrator) have been fixed. Specifically, some controls and their corresponding accessible objects would remain in memory after a form that hosted those controls was closed.
- Assistive tools now accurately draw focus indicators and report correct bounding rectangles for nested forms and some elements of composite controls, such as
DataGridView
,ListView
,TabControl
and others. - ExpandCollapse Control Pattern support in
ListView
,TreeView
andPropertyGrid
controls has been corrected to activate only for expandable items. - Various color contrast ratio corrections in
Button
,DataGridView
andPropertyGrid
controls. - Visibility improvements for
ToolStripTextBox
andToolStripButton
in high-contrast themes.
High DPI and scaling improvements
Throughout the .NET 7 release we continued our work on improving the high DPI support in multi-monitor scenarios (that is, PerMonitorV2
). Most notably, the Windows Forms runtime now can:
- Correctly scale nested controls (e.g., a button which resides in panel, which itself is placed on a tabpage). This class of layout issues was caused by the different order of Windows messages – in
PerMonitorV2
mode, a parent control receive theWM_DPICHANGED
message after all its children have received the message and scaled themselves. - Scale
Form.MaximumSize
andForm.MinimumSize
properties based on the current monitor DPI settings for applications that runApplicationHighDpiMode
set toPerMonitorV2
.
The progress is slow because any changes in the layout engine carry a very high risk. To help reduce the risk we have (re-)introduced “feature switches” back into Windows Forms runtime. This will allow us to add new features and functionality with an escape plan, if those don’t work out for our developers.
In .NET 7 we added the first switch System.Windows.Forms.ScaleTopLevelFormMinMaxSizeForDpi
that controls whether the form’s MaximumSize
and MinimumSize
properties are scaled whenever WM_DPICHANGED
message is received.
In .NET 7 the new scaling behaviour is disabled by default, but in .NET 8 it will be on by default. The switch can be configured via runtimeconfig.json like so:
{
"runtimeOptions": {
"tfm": "net7.0",
"frameworks": [
...
],
"configProperties": {
"System.Windows.Forms.ScaleTopLevelFormMinMaxSizeForDpi": true,
}
}
}
Databinding improvements
We want to help Windows Forms developers to modernize their existing apps and to adopt new cloud-based technologies. One way to modernize is the ability to move business logic of a typical line-of-business (LOB) application away from the Windows Forms code-behind pattern towards more modern patterns like Model-View-ViewModel (MVVM), which can be easily reused and tested. We also wanted to provide an incentive to pivot away from the 15-year-old Typed DataSets. They have very limited support in the Windows Forms out-of-process designer because Visual Studio no longer supports Data Source Provider Service for the .NET runtime. The feedback to the proposed new Binding strategy was very positive and indicated that we’re on the right track with that.
Windows Forms already had a powerful binding engine, but to achieve the above goals we introduced support for a more modern form of data binding with data contexts and command patterns, similar to data binding provided by WPF.
The new API surface is as follows:
public class Control {
+ [BindableAttribute(true)]
+ public virtual object DataContext { get; set; }
+ [BrowsableAttribute(true)]
+ public event EventHandler DataContextChanged;
+ protected virtual void OnDataContextChanged(EventArgs e);
+ protected virtual void OnParentDataContextChanged(EventArgs e);
}
+ [RequiresPreviewFeaturesAttribute]
+ public abstract class BindableComponent : Component, IBindableComponent, IComponent, IDisposable {
+ protected BindableComponent();
+ public BindingContext? BindingContext { get; set; }
+ public ControlBindingsCollection DataBindings { get; }
+ public event EventHandler BindingContextChanged;
+ protected virtual void OnBindingContextChanged(EventArgs e);
+ }
public abstract class ButtonBase : Control {
+ [BindableAttribute(true)]
+ [RequiresPreviewFeaturesAttribute]
+ public ICommand? Command { get; set; }
+ [BindableAttribute(true)]
+ public object? CommandParameter { [RequiresPreviewFeaturesAttribute] get; [RequiresPreviewFeaturesAttribute] set; }
+ [RequiresPreviewFeaturesAttribute]
+ public event EventHandler? CommandCanExecuteChanged;
+ [RequiresPreviewFeaturesAttribute]
+ public event EventHandler? CommandChanged;
+ [RequiresPreviewFeaturesAttribute]
+ public event EventHandler? CommandParameterChanged;
+ [RequiresPreviewFeaturesAttribute]
+ protected virtual void OnCommandCanExecuteChanged(EventArgs e);
+ [RequiresPreviewFeaturesAttribute]
+ protected virtual void OnCommandChanged(EventArgs e);
+ [RequiresPreviewFeaturesAttribute]
+ protected virtual void OnCommandParameterChanged(EventArgs e);
+ [RequiresPreviewFeaturesAttribute]
+ protected virtual void OnRequestCommandExecute(EventArgs e);
}
+ public abstract class ToolStripItem : BindableComponent, IComponent, IDisposable, IDropTarget {
+ [BindableAttribute(true)]
+ [RequiresPreviewFeaturesAttribute]
+ public ICommand Command { get; set; }
+ [BindableAttribute(true)]
+ public object CommandParameter { [RequiresPreviewFeaturesAttribute] get; [RequiresPreviewFeaturesAttribute] set; }
+ [RequiresPreviewFeaturesAttribute]
+ public event EventHandler CommandCanExecuteChanged;
+ [RequiresPreviewFeaturesAttribute]
+ public event EventHandler CommandChanged;
+ [RequiresPreviewFeaturesAttribute]
+ public event EventHandler CommandParameterChanged;
+ [RequiresPreviewFeaturesAttribute]
+ protected virtual void OnCommandCanExecuteChanged(EventArgs e);
+ [RequiresPreviewFeaturesAttribute]
+ protected virtual void OnCommandChanged(EventArgs e);
+ [RequiresPreviewFeaturesAttribute]
+ protected virtual void OnCommandParameterChanged(EventArgs e);
+ [RequiresPreviewFeaturesAttribute]
+ protected virtual void OnRequestCommandExecute(EventArgs e);
}
The new data binding features allows you to fully embrace the MVVM pattern and the use of object-relational mappers from ViewModels in Windows Forms much easier than before. This, in turn, makes it possible to reduce code in the code-behind files, and opens new testing possibilities. More importantly, it enables code share between Windows Forms and other .NET GUI frameworks such as WPF, UWP/WinUI and MAUI. (To preempt any questions, there are no plans to introduce support for XAML into Windows Forms.)
It’s important to add that these new binding features are only available under “preview” status in .NET 7, so that we can react to developer feedback and tweak the binding during the .NET 8 development cycle. We expect that the new binding functionality will become generally available in .NET 8.
To enable the new binding, you need to add <EnablePreviewFeatures>true</EnablePreviewFeatures>
to your project file. This is supported for both C# and Visual Basic from the start.
ComWrappers and native AOT
In .NET 7 as part of a wider experiment we replaced the selected Windows Forms built-in COM interops with ComWrappers.
– What’s the difference? – you may ask. Mostly it’s that the built-in COM gets completely disabled in trimming/native AOT scenarios. That was one of the reasons why trimming was disabled for Windows Forms in .NET 5.
– Why does this matter for Windows Forms which disables trimming during dotnet publish
step? – you may ask again. Even if trimming is disabled it still works, and the migration work allows for further experiments. As an example, it greatly simplifies the ability to run code in native AOT without additional customizations. In addition, if you provide a global ComWrappers implementation for the missing accessibility COM objects, you could run a Windows Forms application under native AOT.
:warning: This work is still highly experimental, and some scenarios are rough and require manual work. Mostly this comes from the native AOT limitations and trimming quirks. Please keep in mind that you need to test your application after you publish.
If you feel adventurous and wish to experiment or just like to know more about ComWrappers and native AOT for Windows Forms, see @kant2002‘s post on Medium “Again WinForms and NativeAOT” and his GitHub repository.
Other notable changes
In no particular order:
- dotnet/winforms#6576 enhanced drag-and-drop handling to match the Windows drag-and-drop functionality with richer display effects with an icon and a text label.
- dotnet/winforms#6244 brought the file and folder dialogs options to parity with the options available in Windows dialogs, such as don’t add to recent, ok button needs interaction, force show hidden, and more.
- dotnet/winforms#7403 made
TreeView
control respect theDoubleBuffer
property. - dotnet/winforms#6234 extended
ErrorProvider
withHasErrors
property. - dotnet/winforms#6335 fixed the
Form
‘s snap layout in Windows 11.
And we have some very cool new UI integration tests as a direct result of some of these changes.
Breaking Changes
While we intend to maintain backward compatibility with .NET Framework and previous .NET releases as much as possible, it is not always prudent. Follow the link to see the list of breaking changes.
Community contributions
The community continues to play an important part in the life of Windows Forms runtime and the SDK. In .NET 7 timeframe we have merged over 700 pull requests (excluding the automated pull requests) in dotnet/winforms repository with almost 60% of those pull requests authored by the community. All these contributions – big and small – are really appreciated.
We’d like to call out a few community contributors:
- @gpetrou has been tirelessly helping with nullable reference types annotations. This work isn’t over, but as you can see from the API diff the progress is substantial.
- @kant2002 has been driving the ComWrappers and Windows Forms native AOT stories.
- @willibrandon has helped modernizing the drag and drop and the dialogs functionality, as well as writing very elaborate tests for the drag and drop scenarios.
- @kirsan31 has taken interest in perf-related issues and removed redundant finalizers, as well as helped stabilizing flaky tests.
- @jbhensley spent some time fixing various control rendering issues (some of which date back to .NET Framework).
Looking ahead
We are committed to providing the complete support for PerMonitorV2
mode. We will continue to work on the scaling-related issues (e.g., dotnet/winforms#7973). We’re also working on a new layout and anchor calculation mechanisms, which are not only expected to address the anchoring issues but should also deliver performance improvements. You can learn more about the new achor layout at dotnet/winforms#7956. In addition, we’ve been considering how we can improve the designer-related developer experience in this area.
We are also set to continue our modernization journey and to closer align our functionality with Windows. We have started replacing our manually created interop code with the source-generated interops produced by CsWin32 project. This not only reduces the maintenance burden for the team, but also opens up new possibilities (e.g., support Windows 11 dark mode). We also expect that it may help advancing the trimming and the native AOT stories.
Furthermore, we will continue our work to migrate built-in COM to ComWrapper and bypass the .NET COM interop layer, where possible. And, hopefully, next time around, we will have a significantly better story for self-contained, single-file Windows Forms applications.
Reporting bugs and suggesting features
If you have any comments, suggestions or faced some issues, please let us know! Submit Visual Studio and Designer related issues via Visual Studio Feedback (look for a button in the top right corner in Visual Studio), and Windows Forms runtime related issues at dotnet/winforms repository.
We also consider API proposals that further enrich Windows Forms SDK and make it easier to build Windows applications. And if you champion a proposal – there is a high chance that you will see it in the Windows Forms SDK.
You can also become a contributor to the Windows Forms code base! Our repository has items marked “help wanted” and API approved for implementation, we would really appreciate your help implementing them!
Happy coding!
WHY THERE IS NO UPDATES IN THE UI, TO. MATCH WINDOWS 11 STYLES? I was waiting for the team to add default windows 11 inspired ui chnages when building apps with windows forms. Please team must do this
If Microsoft wants developers to move away from old DataSets there needs to be coded samples that show how to work with EF Core to work with a DataGridView control. There are many developers out there that struggle with simple DataAdapters and have found they are reluctant to move to EF Core. If one does a search for data binding for windows forms they point to old ways and anything new is insufficient.
Good point. And we have started providing new guidance especially around this.
But I agree, additional guidance in this area is helpful, and our planning is to add more of those in the upcoming months, starting with a dedicated post around the new .NET 7 Data Binding Features.
I would love some guidance on using ClickOnce to migrate from .NET Framework to .NET core.
We’ve migrated our app a number of times but have always had to rollback as we can’t figure out how to manage the transition of 1000’s of Click Once deployed instances from Framework to Core?
Hi Fergal, In regards to your comment about wanting some guidance on ClickOnce. Please open on issue at https://github.com/dotnet/deployment-tools/issue describing your situation and what guidance seeking. TIA
Hi Igor, thanks for mentioning my PR on winforms about adding HasError!
Unfortunately, the link that you mention about my PR of adding HasErrors property is incorrect: it should be https://github.com/dotnet/winforms/pull/6234 instead of https://github.com/dotnet/winforms/pull/6244
Could you correct the link? Thanks in advance!
Fixed. Thank you.
Thank you very much for the information. One thing bothering us is the designer support with .NET. This is stopping us from migrating to .NET from Framework as we still use a lot the DataSets. It’s a kind of catch 22. We would like to migrate and then modernize step by step our forms to go away from the DataSets, but we cannot really migrate if we don’t modify it. We have got many hundred forms in our software. I think we need to test what you mean by “very little support” for DataSets in the designer, but I am afraid this could be a K.O. for us.
Basically, what is missing is a service called “Data Source Provider Service”. This is a service which doesn’t belong to WinForms, but the Designer relies on that service (as does WPF and WebForms to some extent, if I remember correctly. So WPF is also affected by that).
For .NET Framework projects, the Data Source Provider Service is running in the context of Visual Studio, which is also .NET Framework. Therefore, it doesn’t have any problems to solve the Typed DataSet types, and that is necessary to support interactive data binding and data mapping/schema updates. For .NET, that doesn’t work. It cannot resolve the .NET types running in a .NET Framework process. The WinForms .NET Designer, however, supports that by providing two processes, one of which is targeting .NET Framework and runs in the context of VS, and the other targeting .NET in a dedicated Server process. This Server process is where type resolution, actual instantiating of controls and the rendering of the Form happens, and both processes are communicating with each other constantly.
Minimum support for Typed DataSets to that end means, that the new .NET Designer can deal with Typed DataSet-Data Source definitions which are already in the project. It takes them simply as they are and let them run in the context of the Server process – which is .NET – for example when you want to change the binding based on a typed dataset for a specific control on a Form or a UserControl. But the tools to maintain those DataSets would also need all the other (non-WinForms) editors to be running not in the context of Visual Studio, but in a dedicated Server process – and that is currently out of scope for .NET and for WinForms, since they are not part of it.
So, while there is support to use Typed DataSets which came over from a Framework project, and even remap binding targets via the DesignBindingPicker in the Property Browser, actively creating new Typed DataSets or editing existing ones does not work that way. But, as I said, bringing over Typed DataSets DataSource definitions from a Framework project should work just fine.
Thanks for the work done. I ask a usual question, is it now possible to develop Winforms user interface without resizing the DPI?
Could you please clarify what you mean by “without resizing the DPI”?
Being able to design Windows Forms interfaces when running Visual Studio at greater than 100% DPI scaling without having to worry about correct scaling at 100%. Right now Visual Studio warns you when opening a Windows Forms project and your DPI scaling is greater than 100%.
Thanks for updating Winform. It is still the easiest way to get quick windows applications up and running. Great update!