Winform开发框架之肖像显示保存控件的实现

我们在开发一些Winform程序的时候,除了常规的显示普通数据外,有的时候需要显示一些人员肖像或者一些车辆等物体的图片,一般这些内容较小,所以以二进制存储在数据库是一个不错的方案。但由于它们虽然很常用,设计数据库保存的逻辑又会使得整个控件的封装显得麻烦很多。本文介绍的肖像显示保存控件,通过事件的封装处理,让数据的保存不在依赖于数据库存储模块,实现更加通用的特性。

1、肖像显示保存控件的需求

我们在一些程序了里面,可能需要显示一些人员头像,车辆图片,物件图片等,这些图片可以从电脑上选取,也可以拍照,当然最重要的是,方便使用,并能存储到数据库里面,这个就是我们一般的需求了。
1)人员肖像显示效果:



2)车辆图片展示效果



控件的工具条上,提供了编辑图片,保存图片,恢复默认图片,拍照等功能,当然我们希望控件和整个界面一起,实现图片数据的方便加载和保存操作。

2、肖像显示保存控件的设计

前面我们看了一些常用的场景,对我们设计这个通用性的肖像显示保存控件有很好的指导意义,我们只需要把图片显示部分作为一个控件,然后公布一些事件给外部实现,从而实现我们需要的数据加载、数据保存等操作,至于其他的功能,我们可以集成到里面去。
我们设计一个用户自定义控件,上面放一个图片显示框,然后增加一些按钮在上面,并设置好工具栏的图片显示,效果如下所示。



由于我们需要给外部处理数据的绑定(需要从数据库加载)和数据的保存(保存到数据库里面),因此抛出两个委托代理定义。

/// <summary>
/// 保存图片的处理代理定义
/// </summary>
/// <param name="id">记录ID</param>
/// <param name="imageBytes">图片数据</param>
/// <param name="imageType">图片类型</param>
/// <returns></returns>
public delegate bool SavePortraitHandler(string id, byte[] imageBytes, string imageType);

/// <summary>
/// 绑定图片数据的代理定义
/// </summary>
/// <param name="id">记录ID</param>
/// <param name="imageType">图片类型</param>
/// <returns></returns>
public delegate byte[] BindPortraitHandler(string id, string imageType);

然后我们在用户控件里面增加一些属性和事件定义,部分代码如下所示。

/// <summary>
/// 图片数据显示和采集控件
/// </summary>
public partial class PortraitControl : XtraUserControl
{
    #region 事件及属性定义
    /// <summary>
    /// 保存图片操作的事件
    /// </summary>
    public event SavePortraitHandler OnSavePortrait;

    /// <summary>
    /// 绑定图片数据的事件
    /// </summary>
    public event BindPortraitHandler OnBindPortait;

    /// <summary>
    /// 记录ID
    /// </summary>
    public string ID { get; set; }

    /// <summary>
    /// 图片是否被修改过
    /// </summary>
    public bool ImageIsDirty { get; set; }

然后我们需要实现的就是,公布两个公共方法,分别是数据的绑定,绑定到界面控件的操作函数,我们主要注意的是,这里调用了事件OnBindPortait,以实现数据库的解耦,代码如下所示。

/// <summary>
/// 绑定图片的操作,触发绑定事件处理
/// </summary>
/// <param name="id">记录ID</param>
public void BindPicture(string id)
{
    try
    {
        this.ID = id; //设置控件的ID值

        #region 更新图片显示操作

        if (OnBindPortait == null)
        {
            MessageDxUtil.ShowTips("控件未指定OnBindPortait事件处理");
            return;
        }

        byte[] imageBytes = OnBindPortait(id, ImageType);
        if (imageBytes != null)
        {
            this.picPortrait.Image = ImageHelper.ImageFromBytes(imageBytes);
        }
        else
        {
            ResetDefaultImage(ImageType);
        }

        #endregion
    }
    catch (Exception ex)
    {
        MessageDxUtil.ShowError(ex.Message);
        LogTextHelper.Error(ex);
    }
}

数据库保存的实现,和上面思路差不多,也是实现事件的处理即可,通过调用事件OnSavePortrait,实现数据库的解耦。

/// <summary>
/// 保存图片到服务器
/// </summary>
public bool SavePicture(string id)
{
    this.ID = id;//设置控件的ID值

    if (string.IsNullOrEmpty(id))
    {
        MessageDxUtil.ShowTips("记录ID未指定,无法保存,请先保存数据!");
        return false;
    }
    if (OnSavePortrait == null)
    {
        MessageDxUtil.ShowTips("控件未指定OnSavePortrait处理事件!");
        return false;
    }

    if (picPortrait.Image != null)
    {
        try
        {
            byte[] imageBytes = ImageHelper.ImageToBytes(this.picPortrait.Image);

            bool sucess = false;
            if (OnSavePortrait != null)
            {
                sucess = OnSavePortrait(this.ID, imageBytes, ImageType);
            }
            return sucess;

        }
        catch (Exception ex)
        {
            MessageDxUtil.ShowError(ex.Message);
            LogTextHelper.Error(ex);
        }
    }
    return false;
}

3、肖像显示保存控件的使用

完成以上控件的设计和处理后,编译后,模块将增加一个用户控件,这样我们就可以在需要的界面模块里面,把这个控件拖动到设计界面里面去了,设计效果和下图类似。



这个时候,肖像显示保存控件就作为一个整体进行使用了。
由于我们设计控件的时候,我们把它和外部数据库存储和加载进行了隔离,因此这里需要进行整合,通过事件进行整合处理。调用代码如下所示。

public partial class FrmEditDriver : BaseEditForm
{
    public FrmEditDriver()
    {
        InitializeComponent();

        this.portraitControl1.ImageType = "个人肖像";
        this.portraitControl1.OnBindPortait += new BindPortraitHandler(portraitControl1_OnBindPortait);
        this.portraitControl1.OnSavePortrait += new SavePortraitHandler(portraitControl1_OnSavePortrait);
    }

然后实现其中自动增加的事件函数即可,主要就是调用业务类实现数据的存储和加载处理逻辑,代码如下所示。

bool portraitControl1_OnSavePortrait(string id, byte[] imageBytes, string imageType)
{
    return BLLFactory<Driver>.Instance.UpdateImageBytes(id, imageBytes, imageType);
}

byte[] portraitControl1_OnBindPortait(string id, string imageType)
{
    return BLLFactory<Driver>.Instance.GetImageBytes(id, imageType);
}

在数据的显示函数里面,我们主动调用控件的函数实现界面数据的绑定显示。

/// <summary>
/// 数据显示的函数
/// </summary>
public override void DisplayData()
{
    InitDictItem();//数据字典加载(公用)

    if (!string.IsNullOrEmpty(ID))
    {
        #region 显示信息
        DriverInfo info = BLLFactory<Driver>.Instance.FindByID(ID);
        if (info != null)
        {
            tempInfo = info;//重新给临时对象赋值,使之指向存在的记录对象

            txtHandNo.Text = info.HandNo;
            txtName.Text = info.Name;
            txtMobile.Text = info.Mobile;
            txtDept.Text = info.Dept;
            txtStartDriveDate.SetDateTime(info.StartDriveDate);
            txtSex.Text = info.Sex;
            txtBirthday.SetDateTime(info.Birthday);
            txtNote.Text = info.Note;
        }
        #endregion
    }
    else
    {
    }

    //绑定图片
    this.portraitControl1.BindPicture(tempInfo.ID);
}

而在保存按钮实现数据保存的时候,我们也可以调用控件自身的保存操作函数,实现数据的存储。

/// <summary>
/// 新增状态下的数据保存
/// </summary>
/// <returns></returns>
public override bool SaveAddNew()
{
    DriverInfo info = tempInfo;//必须使用存在的局部变量,因为部分信息可能被附件使用
    SetInfo(info);

    try
    {
        #region 新增数据

        bool succeed = BLLFactory<Driver>.Instance.Insert(info);
        if (succeed)
        {
            //可添加其他关联操作
            this.portraitControl1.SavePicture(tempInfo.ID);

            return true;
        }
        #endregion
    }
    catch (Exception ex)
    {
        LogTextHelper.Error(ex);
        MessageDxUtil.ShowError(ex.Message);
    }
    return false;
}

通过和数据库操作实现解耦,我们可以对这个控件进行更方便的重用,而代码也很简单,这样也就实现了我们统一化、简单化、可复用性的目标了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,236评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,867评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,715评论 0 340
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,899评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,895评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,733评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,085评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,722评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,025评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,696评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,816评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,447评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,057评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,009评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,254评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,204评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,561评论 2 343

推荐阅读更多精彩内容