C#-用Winform制作一个简单的密码管理工具

182 阅读5分钟

为什么要做?

首先是为了练习一下c#。 想必大家都有过记不起某个平台的账号密码的经历,那种感受着实令人抓狂。那这么多账号密码根本记不住!我之前用python写过一个超级简单(连账号信息都写在代码里那种)的控制台程序用来给我提示密码,但是我想添加一个账号时直接被麻烦到吐。 所以我才想用Winform做一个简单的小工具来帮助我记忆。(仅供我自己使用(所以界面会比较丑。。),但是我会把代码贴出来所以有点c#基础的其实都可以自己做一个)

设想

我的需求非常简单

  1. 在我需要的时候,输入一个平台能把对应的账号密码显示出来
  2. 能够添加账号信息

还有一点就是在查询账号之前需要输入一个口令来验证身份,这个口令只有我自己知道(我把它存在了app.config文件中,后续如果有需要可以扩展出更改口令的功能),所以即使别人用我的电脑运行起来这个程序,他不知道口令也是没用的。

账号信息怎么存?

我曾经想要用SQL Server存,毕竟c#与它如此亲近,但是我要存的东西本质上只是一些字符串,感觉有点大材小用所以没有选择SQL Server。最近学了JS了解了一些json的知识发现json文件是个很好的选择于是我便决用它了。但是我还不会用c#处理json数据呀。于是我又去网上找方法,然后我就发现了newtonsoft.json(json.NET) ,它时一款.NET中开源的json序列化与反序列化工具。有了它,就可以解决我的问题了。

code

首先把入口界面搭建起来,一个超级简单的窗体,只放了三个控件。

入口

为了方便操作给它加一个退出的快捷键ESC。非常简单只需要在KeyDown事件中写下如下代码:

private void Form1_KeyDown(object sender, KeyEventArgs e)
{
    if(e.KeyCode==Keys.Escape)
    {
        this.Close();
    }
}

然后是确定这个按钮的功能:当输入正确的口令后,点击确定可以进入到下一个界面。

口令放在哪?

我将口令这个数据放在了App.config文件中,在解决方案资源管理器中右键添加新建项就可以添加它了。然后向其中加入如下代码:

  <appSettings>
    <add key="CMD" value="123123"/>
  </appSettings>

然后给刚才的入口窗体添加一个字段CMD并用ConfigurationManager将刚才的配置读取出来赋值给它:

public readonly string CMD=ConfigurationManager.AppSettings["CMD"].ToString() ;

上面的确定按钮的功能是口令正确是将下个界面显示出来,所以我们先把下个界面创建出来:

有了这个界面,就可以去写第一个界面中确定按钮的点击事件了:

private void button1_Click(object sender, EventArgs e)
{
    ;
    string entered_cmd = textBox1.Text; 
    if(entered_cmd.Equals(CMD))
    {
        (new GetPwd()).Show();
        this.Hide();
    }
    else
    {
        MessageBox.Show("错误!","警告",MessageBoxButtons.OK, MessageBoxIcon.Warning);
    }
}

这样第一个界面的代码就写完了(没错就是这么简单)。

账号信息查询界面

也就是上面的第二个窗体。它加载的初始状态是这样:

用于显示账号和密码的两个文本框我设置成了不可见(当输入的平台是存在的并点击确定就可以看到了)和只读。像这样:

一个小问题

和上一个窗体一样,我也给这个窗体绑定了enter(相当于点击确定按钮)和esc快捷键。但是当我关闭这个窗体时,已经无法再进行任何操作了可是这个程序进程却没有结束。怎么解决这个问题呢?前段时间学习了委托于是我想到了一个比学校老师曾经教过的更好的方法(当时还没学习委托,老师教的是把第一个窗体对象传递给第二个窗体):给第二个窗体定义一个委托字段 public Action close_main;然后在显示该窗体是将前一个窗体的close()方法传递过来即可。 前面的代码修改一下:

(new GetPwd() {close_main=this.Close}).Show();

然后在第二个窗体的FormClosed事件中执行这个方法就可以了。

private void GetPwd_FormClosed(object sender, FormClosedEventArgs e)
{
    close_main();
}

查询

所有的账号信息我都放在一个json文件中:PWD.json,有关对其的各种逻辑性操作我抽成了一个类Manager:

class Manager
{
    public static string GetjsonString()
    {
        // 获取整个json字符串
        if(!File.Exists(Path.GetFullPath(@"..//..") + ".../PWD.JSON"))
        {
            File.Create(Path.GetFullPath(@"..//..") + ".../PWD.JSON");
        }
        StreamReader jsonFile = File.OpenText(Path.GetFullPath(@"..//..") + ".../PWD.JSON");
        string res = jsonFile.ReadToEnd();
        jsonFile.Close();
        return res;
    }
    public static List<Acount> GetAcountsList()
    {
        //将json字符串序列化为一个集合对象
        return JsonConvert.DeserializeObject<List<Acount>>(GetjsonString());
    }
    public static Acount SearchAcount(string platform)
    {
        //搜索平台为platform的账号
        List<Acount> acounts = GetAcountsList();
        if(acounts==null||acounts.Count==0)
        {
            return null;
        }
        var res = from a in acounts
              where a.platForm == platform
              select a;
        return res.Count()>=1?res.First():null;
    }
    public static bool CheckExistence(Acount acount)
    {
        //检查acount 这个账号是否存在
        //检查该账号是否存在
        List<Acount> acounts = GetAcountsList();
        var res = from a in acounts
                  where a.platForm==acount.platForm&&a.acount == acount.acount && a.password == acount.password
                  select a;
        return res.Count() == 1;
    }
    public static void AddAcount(Acount a)
    {
        //添加一个账号信息
        if(a.platForm==""||a.acount==""||a.password=="")
        {
            MessageBox.Show("请输入完整信息!", "警告", MessageBoxButtons.OK, MessageBoxIcon.Warning);
            return ;
        }
        List<Acount> acounts = GetAcountsList();
        acounts.Add(a);
        string res = JsonConvert.SerializeObject(acounts);
        StreamWriter jsonwriter = new StreamWriter(Path.GetFullPath(@"..//..") + ".../PWD.JSON");
        jsonwriter.WriteLine(res);
        jsonwriter.Close();
    }
}

有了这些方法,其他地方就非常简单了。查询界面的确定按钮点击事件:

private void button1_Click(object sender, EventArgs e)
{
    Acount acount = Manager.SearchAcount(txt_platform.Text.ToLower());
    if(acount!=null)
    {
        txt_uId.Text = acount.acount;
        txt_pwd.Text = acount.password;
        txt_uId.Visible = true;
        txt_pwd.Visible = true;
        txt_uId.ReadOnly = true;
        txt_pwd.ReadOnly = true;
    }
    else
    {
        MessageBox.Show("无此账号!", "警告", MessageBoxButtons.OK, MessageBoxIcon.Warning);
    }
}

添加

第三个界面用来添加,通过第二个界面的Add按钮呼出。 长这样:

我的设想是想要添加账号信息需要再次验证一下身份:输入一个已有的正确账号信息来通过验证,验证通过后仍然使用这个界面来添加(三个文本框依次用来输入平台、账号、密码)。这个界面只有一个按钮,验证时它的text属性为"Check",添加时为"Add",窗体的标题也有相应的变化。该按钮的点击事件如下:

private void button1_Click(object sender, EventArgs e)
{
    if(btn_check.Text=="Check")
    {
        if (Manager.CheckExistence(new Acount(txt_check_platform.Text.ToLower(), txt_check_uid.Text, txt_check_pwd.Text)))
        {
            txt_check_platform.Text = "";
            txt_check_uid.Text = "";
            txt_check_pwd.Text = "";
            this.Text = "添加密码";
            btn_check.Text = "Add";
            txt_check_platform.Focus();
        }
        else
        {
            MessageBox.Show("验证失败!!", "验证反馈", MessageBoxButtons.OK, MessageBoxIcon.Warning);
        }
    }
    else
    {
        Manager.AddAcount(new Acount(txt_check_platform.Text.ToLower(), txt_check_uid.Text, txt_check_pwd.Text));
        this.close();
    }            
}

添加时:

另外,第三个界面也做了和前一个界面相同的快捷键和关闭处理

最后

这个程序非常非常简单,当然后续还是可以扩展许多其他功能的。虽然制作过程很简单但是我还是有很重要的收获,就是newtonsoft.json这个工具。当然还有一点时减轻了我对无数密码的记忆负担。

以记录自己的学习历程