专栏首页黄腾霄的博客2020-5-14-WPF的RadioButton指定groupname在window关闭后无法check

2020-5-14-WPF的RadioButton指定groupname在window关闭后无法check

今天遇到一个神奇的WPF的RadioButton的group问题,和大家介绍下。


问题描述

英文好的同学可以直接看吕毅 - walterlv同学在github提的issue

最小复现demo,见毅仔的仓库

我将一组RadioButton关联到了同一个GroupName下,并且绑定了同一个ViewModel。

        <Border>
            <RadioButton GroupName="A" IsChecked="{Binding Bar, Source={x:Static local:Foo.Instance}}" Content="Option 1" />
        </Border>
        <Border>
            <RadioButton GroupName="A" Content="Option 2" />
        </Border>
        <Border>
            <RadioButton GroupName="A" Content="Option 3" />
        </Border>
public class Foo : INotifyPropertyChanged
    {
        public static Foo Instance { get; } = new Foo();

        public bool Bar
        {
            get => _bar;
            set
            {
                if (!Equals(_bar, value))
                {
                    _bar = value;
                    PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(Bar)));
                }
            }
        }

        private bool _bar;
        public event PropertyChangedEventHandler PropertyChanged;
    }

然后开启两个窗口,此时显示正常(图片来自吕毅)

如果我们关闭其中一个窗口,另一个窗口就不能再点击至check状态(图片来自吕毅)

原因探究

通过vs断点,我们可以发现ViewModel中被绑定的数值,在点击option1时,先变成true,再变成false。

那么很显然问题是这样的。

  • 初始情况下,我们的两个窗口都处于未点击状态
  • 接着我们尝试点击未关闭的那个窗口的RadioButton,他的状态变成了checked 同时,因为binding,ViewModel的属性也变为true,使关闭的那个window(未被GC)也置为checked
  • 接着就是问题所在了,已经被关闭的window会和还存在的window被识别为同一个GroupName的域。因此 同一个域中的一个RadioButton被点击,会让其他RadioButton被Unchecked。
  • 接着就是通过binding,使得ViewModel属性至为false,其他地方被unchecked

验证

按照猜想,我们已经关闭的窗口的GroupName的scope会和全局的保持为同一个。

因此可以去WPF的源码看看。

我们可以在RadioButton.cs看到其中的代码是依赖于visual root

不过问题在于如果完全按照referenceSouce的实现,这里不会出现问题

因此,这里还需要进一步调查

临时方案

既然知道了是GroupName的scope问题,那么我们可以通过一些方法,针对每一个不同的View实例,提供不同的GroupName字符串。

这类我创建了一个静态类GroupNameProvider,提供了一个附加属性BuildScope

当BuildScope设置为true时,我们可以生成一个guid,并赋值给另一个只读附加属性GroupNameA。

这样我们的RadioButton就可以依赖于这个字符串,实现每个View有不同的GroupName。

代码如下:

public static class GroupNameProvider
    {
        public static readonly DependencyProperty BuildScopeProperty = DependencyProperty.RegisterAttached(
            "BuildScope", typeof(bool), typeof(GroupNameProvider), new PropertyMetadata(PropertyChangedCallback));

        private static void PropertyChangedCallback(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            if (e.NewValue is bool value && value == true)
            {
                SetGroupNameA(d, Guid.NewGuid().ToString("N"));
            }
        }

        public static void SetBuildScope(DependencyObject element, bool value)
        {
            element.SetValue(BuildScopeProperty, value);
        }

        public static bool GetBuildScope(DependencyObject element)
        {
            return (bool) element.GetValue(BuildScopeProperty);
        }

        public static readonly DependencyProperty GroupNameAProperty = DependencyProperty.RegisterAttached(
            "GroupNameA", typeof(string), typeof(GroupNameProvider), new PropertyMetadata(default(string)));

        private static void SetGroupNameA(DependencyObject element, string value)
        {
            element.SetValue(GroupNameAProperty, value);
        }

        public static string GetGroupNameA(DependencyObject element)
        {
            return (string) element.GetValue(GroupNameAProperty);
        }
    }
<Grid x:Name="Panel" local:GroupNameProvider.BuildScope="True">
    <RadioButton IsChecked="{Binding Foo}"
      GroupName="{Binding ElementName=Panel,Path=(local:GroupNameProvider.GroupNameA)}"/>
</Grid>

参考文档:


本文会经常更新,请阅读原文: https://xinyuehtx.github.io/post/WPF%E7%9A%84RadioButton%E6%8C%87%E5%AE%9Agroupname%E5%9C%A8window%E5%85%B3%E9%97%AD%E5%90%8E%E6%97%A0%E6%B3%95check.html ,以避免陈旧错误知识的误导,同时有更好的阅读体验。

本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。欢迎转载、使用、重新发布,但务必保留文章署名黄腾霄(包含链接: https://xinyuehtx.github.io ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请 与我联系

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 2019-7-13-FirstChanceException原理

    异常处理是代码中很重要的一项注意点。但是有时候一些不恰当的异常处理,反而会影响我们在代码运行时的调试。

    黄腾霄
  • Moq基础(一)

    现在我们期望能够写一个单元测试,验证运行DoA方法时,是否向日志写入了Finish A

    黄腾霄
  • 2019-12-15-在windows上使用linux子系统

    目前我们在大部分同学使用的都是Windows系统的电脑,若想能够使用Linux进行一些编程开发,大多数同学还是会使用双系统或者虚拟机。但是在Windows10的...

    黄腾霄
  • 手写tomcat+servlet

    斯文的程序
  • 手写tomcat+servlet

    故久
  • 给注册用户发红包,RabbitMQ实现(分布式事务2)

    DLX,Dead Letter Exchange 的缩写,又死信邮箱、死信交换机。DLX就是一个普通的交换机,和一般的交换机没有任何区别。 当消息在一个队列中变...

    算法之名
  • 使用动态代理只代理接口(非实现类)

    假设现在我们有一个已知的算法,我们需要写任意一个接口打上我们特有的标签,那么这个接口的方法都可以执行这个算法,好比Mybatis的Dao,或者Feign的接口。...

    算法之名
  • java设计模式之创建型模式

    (1)创建型模式,共五种:工厂方法模式、抽象工厂模式、单例模式、建造者模式、原型模式。

    GavinZhou
  • Java经典设计模式之五大创建型模式(附实例和详解)

    (1)创建型模式,共五种:工厂方法模式、抽象工厂模式、单例模式、建造者模式、原型模式。

    Java团长
  • 可能是把 Java 接口讲得最通俗的一篇文章

    读者春夏秋冬在抽象类的那篇文章中留言,“二哥,面试官最喜欢问的一个问题就是,‘兄弟,说说抽象类和接口之间的区别?’,啥时候讲讲接口呗!”

    沉默王二

扫码关注云+社区

领取腾讯云代金券