November 8th, 2022

What’s new in Windows Forms in .NET 7.0

Igor Velikorossov
Software Engineer

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 and ToolStripPanel
    • FlowLayoutPanel, TableLayoutPanel and SplitContainer
    • 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 and PropertyGrid controls has been corrected to activate only for expandable items.
  • Various color contrast ratio corrections in Button, DataGridView and PropertyGrid controls.
  • Visibility improvements for ToolStripTextBox and ToolStripButton in high-contrast themes.

Accessibility improvements

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 the WM_DPICHANGED message after all its children have received the message and scaled themselves.
  • Scale Form.MaximumSize and Form.MinimumSize properties based on the current monitor DPI settings for applications that run ApplicationHighDpiMode set to PerMonitorV2.

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.

Databinding, command binding

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:

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!

Author

Igor Velikorossov
Software Engineer

Engineer on Windows Forms team

13 comments

Discussion is closed. Login to edit/delete existing comments.

Newest
Newest
Popular
Oldest
  • Faheem Ahmad

    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

  • Karen Payne

    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.

    • Klaus LoeffelmannMicrosoft employee

      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.

  • Fergal Moran

    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?

    • Michael SimonsMicrosoft employee

      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

  • Raul

    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.

    • Klaus LoeffelmannMicrosoft employee

      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.

  • Davide

    Thanks for the work done. I ask a usual question, is it now possible to develop Winforms user interface without resizing the DPI?

    • Igor VelikorossovMicrosoft employee Author

      Could you please clarify what you mean by “without resizing the DPI”?

      • Greg Karch

        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%.

  • Allen Anderson

    Thanks for updating Winform. It is still the easiest way to get quick windows applications up and running. Great update!

Feedback