根据角色决定是否显示或者启用控件

让我们一起来想一下这么一个场景:我们的程序中有很多控件,他们需要根据当前用户的角色不同去决定是否要显示,或者是否启用?你会怎么做这个事情呢?

很多朋友说,这还不简单吗?我在窗体加载的时候去检查就可以了。这个思路是行得通的,但在实现上有几个问题

1. 在什么地方存储这些角色设置呢

2. 每个控件都要去验证,会不会出现大量重复的代码

我的解决方案是参照Tooltip那种组件的做法来实现。想想看,假如我们要为大量控件设置tooltip,那么我们会用该组件,而如果一个窗体上有一个Tooltip组件之后,那么其他的控件的属性中就会多出来一个特殊的属性,如下图所示

image

也就是说,Tooltip给原有控件进行了一个扩展。这的确很神奇。如果我们在这里设置了内容,我们来看看在后台代码中发生了什么事情

image

这就是它的工作原理,我们来总结一下

1. Tooltip是一个组件,它"神奇"地扩展了现有控件的属性

2. 实际上该属性的设置是通过一个SetTooltip的方法来做的。【而不能用标准的属性指派的方式去做】

了解了以上原理之后,我们回到自己的需求:我们需要让每个控件能随着用户的角色不同而决定是否显示,或者启用。这难道与Tooltip的机制不是很相似么?好吧,我们开始干吧

using System.Collections.Generic;
using System.ComponentModel;
using System.Windows.Forms;

using System.Threading;

namespace Xizhang.Security.Windows

{
    /// <summary>
    /// 该类型扩展了控件的安全设置,可以根据角色来决定是否显示或者启用
    /// 作者:陈希章
    /// </summary>
    [ProvideProperty("SecureVisible",typeof(Control))]//这里公开两个属性,其实也不能说是属性,因为该类型里面并不需要该属性。总之它跟上面的Tooltip很类似。
    [ProvideProperty("SecureEnable",typeof(Control))]
    public partial class SecurityContext : Component,IExtenderProvider //必须实现IExtenderProvider接口
    {
        Dictionary<Control, string> VisiblerolesList = new Dictionary<Control, string>();
        Dictionary<Control, string> EnablerolesList = new Dictionary<Control, string>();

        public SecurityContext()
        {
            InitializeComponent();
        }

        public SecurityContext(IContainer container)
        {
            container.Add(this);
            InitializeComponent();
        }

        #region IExtenderProvider 成员

        public bool CanExtend(object extendee)//这是接口唯一要实现的方法,决定对那些控件进行扩展,这将决定在哪些控件的属性窗格中看到它
        {
            if (extendee is Control && !(extendee is Form))//我们这里指定必须是控件,但不能是窗体
            {
                return true;
            }
            return false;
        }

        public void SetSecureVisible(Control ctrl, string role) //注意这个和下面那个方法。既然我们公开了属性名叫SecureVisible,那么我们就必须实现两个相应的方法,一个是Set,一个是Get
        {
            if (!VisiblerolesList.ContainsKey(ctrl))
            {
                if (!string.IsNullOrEmpty(role))
                {
                    ctrl.Visible = Thread.CurrentPrincipal.IsInRole(role);
                }
                VisiblerolesList.Add(ctrl, role);
            }
            else
                VisiblerolesList[ctrl] = role;
        }

        public string GetSecureVisible(Control ctrl)
        {
            if (VisiblerolesList.ContainsKey(ctrl))
                return VisiblerolesList[ctrl];
            else
                return string.Empty;
        }

        public void SetSecureEnable(Control ctrl, string role)
        {
            if (!EnablerolesList.ContainsKey(ctrl))
            {
                if (!string.IsNullOrEmpty(role))
                {
                    ctrl.Enabled = Thread.CurrentPrincipal.IsInRole(role);
                }
                EnablerolesList.Add(ctrl, role);
            }
            else
                EnablerolesList[ctrl] = role;
        }

        public string GetSecureEnable(Control ctrl)
        {
            if (EnablerolesList.ContainsKey(ctrl))
                return EnablerolesList[ctrl];
            else
                return string.Empty;
        }
        #endregion
    }
}

2. 在窗体上使用它,我们看到是和Tooltip类似的效果

image

3. 激动人心的时刻到了,我们现在在登录的时候指派角色

MyIdentity identity = new MyIdentity("chenxizhang");
MyPrinciple principle = new MyPrinciple(identity, new string[] { "Admin" });
Thread.CurrentPrincipal = principle;

注意,这里我用的是自定义标识和主体。关于该主题,你可以参考http://www.cnblogs.com/chenxizhang/archive/2008/08/17/1269658.html

注意观察,我们在SecureEnable中设置为Admin,表示只有Admin角色才能使用该按钮。我们在SecureVisible中设置为空,表示任何人都可以看到该按钮。然后,因为我们在登录的时候指派角色为Admin,所以该用户是既可以看到,又可以使用该按钮的。如下图所示

image

换言之,如果用户不具备Admin角色,那么按钮就会变成灰色

MyIdentity identity = new MyIdentity("chenxizhang");
MyPrinciple principle = new MyPrinciple(identity, new string[] { "PowerUser" });
Thread.CurrentPrincipal = principle;

image
posted @ 2008-08-17 10:43  陈希章  阅读(458)  评论(0编辑  收藏  举报